TimeAccounting
1.0.3
OTRS GmbH
http://otrs.org/
GNU GENERAL PUBLIC LICENSE Version 2, June 1991
A new release with the improvements of Torsten Thau. Thank you Torsten.
Fixed the problem if the agent has ro-permissions and try to see old time accounting overviews.
Make the TimeAccounting module fit for OTRS 2.2.
first version build with OPMS
2006-05-23 insert a ProjectReporting page
2006-05-16 fixed action new bug in setting
2006-04-24 insert the delete dtl
2006-04-18 fixed vacation bug
2006-04-11 fixed Show all bug
2006-04-11 fixed typo
2006-04-10 include new functions
2005-08-10 fixed initial bugs
2005-08-08 first package
A TimeAccounting Modul.
Ein Zeiterfassungsmodul.
2.1.x
2.2.x
# add groups
$Self->{GroupObject}->GroupAdd(
Name => 'time_accounting',
Comment => '',
ValidID => 1,
UserID => 1,
);
2007-06-05 16:22:39
opms.otrs.com
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iQXBwbGljYXRpb24iPgogICAgPENWUz4kSWQ6IFRpbWVBY2NvdW50aW5nLnhtbCx2IDEuNyAyMDA3LzA1LzExIDE0OjM3OjA1IHRyIEV4cCAkPC9DVlM+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGcm9udGVuZDo6TW9kdWxlIyMjQWdlbnRUaW1lQWNjb3VudGluZyIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkZyb250ZW5kTW9kdWxlUmVnaXN0cmF0aW9uIGZvciBUaW1lQWNjb3VudGluZ01vZHVsZSAgdG8gZGVmaW5lIHRoZSByZWdpc3RyYXRpb24gc2V0dGluZ3MgaW4gdGhlIHByb2dyYW0gYW5kIHRvIHNldCByaWdodHMgYW5kIHBvc2l0aW9ucyBvZiBpY29ucyBpbiB0aGUgbmF2aWdhdGlvbiBiYXIuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkZyb250ZW5kTW9kdWxSZWdpc3RyYXRpb24gZvxyIGRhcyBaZWl0ZXJmYXNzdW5nc01vZHVsLiAgSGllcgpr9m5uZW4gU2llIGRpZSBBbmdhYmVuIGb8ciBkaWUgUmVnaXN0cmllcnVuZyBkZXMgTW9kdWxzIGltIFByb2dyYW1tIHTkdGlnZW4uIEZlcm5lcgpr9m5uZW4gZGllIFJlY2h0ZSB1bmQgZGllIFBvc2l0aW9uZW4gZGVyIEljb25zIGluIGRlciBOYXZpZ2F0aW9uc2xlaXN0ZSBnZXNldHp0IHdlcmRlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEZyb250ZW5kTW9kdWxlUmVnPgogICAgICAgICAgICAgICAgPEdyb3VwPnRpbWVfYWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgICAgICAgICA8R3JvdXBSbz50aW1lX2FjY291bnRpbmc8L0dyb3VwUm8+CiAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+VGltZUFjY291bnRpbmc8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgPE5hdkJhck5hbWU+VGltZUFjY291bnRpbmc8L05hdkJhck5hbWU+CiAgICAgICAgICAgICAgICA8VGl0bGU+VGltZUFjY291bnRpbmc8L1RpdGxlPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+VGltZUFjY291bnRpbmc8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPlRpbWVBY2NvdW50aW5nPC9OYW1lPgogICAgICAgICAgICAgICAgICAgIDxJbWFnZT50aW1lX2FjY291bnRpbmcucG5nPC9JbWFnZT4KICAgICAgICAgICAgICAgICAgICA8TGluaz5BY3Rpb249QWdlbnRUaW1lQWNjb3VudGluZyZhbXA7U3ViYWN0aW9uPUVkaXQ8L0xpbms+CiAgICAgICAgICAgICAgICAgICAgPE5hdkJhcj5UaW1lQWNjb3VudGluZzwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPk1lbnU8L1R5cGU+CiAgICAgICAgICAgICAgICAgICAgPFByaW8+NjAwMDwvUHJpbz4KICAgICAgICAgICAgICAgICAgICA8QWNjZXNzS2V5PjwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz5JdGVtQXJlYTwvQmxvY2s+CiAgICAgICAgICAgICAgICA8L05hdkJhcj4KICAgICAgICAgICAgICAgIDxOYXZCYXI+CiAgICAgICAgICAgICAgICA8R3JvdXBSbz50aW1lX2FjY291bnRpbmc8L0dyb3VwUm8+CiAgICAgICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPk92ZXJ2aWV3PC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgICAgICA8TmFtZT5PdmVydmlldzwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+b3ZlcnZpZXcucG5nPC9JbWFnZT4KICAgICAgICAgICAgICAgICAgICA8TGluaz5BY3Rpb249QWdlbnRUaW1lQWNjb3VudGluZyZhbXA7U3ViYWN0aW9uPU92ZXJ2aWV3PC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPlRpbWVBY2NvdW50aW5nPC9OYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEFjY2Vzc0tleT5vPC9BY2Nlc3NLZXk+CiAgICAgICAgICAgICAgICAgICAgPFByaW8+MTAwPC9QcmlvPgogICAgICAgICAgICAgICAgPC9OYXZCYXI+CiAgICAgICAgICAgICAgICA8TmF2QmFyPgogICAgICAgICAgICAgICAgPEdyb3VwUm8+dGltZV9hY2NvdW50aW5nPC9Hcm91cFJvPgogICAgICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5UaW1lUmVjb3JkLUVkaXQ8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPkVkaXQ8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPm5ldy5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudFRpbWVBY2NvdW50aW5nJmFtcDtTdWJhY3Rpb249RWRpdDwvTGluaz4KICAgICAgICAgICAgICAgICAgICA8VHlwZT48L1R5cGU+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPjwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPE5hdkJhcj5UaW1lQWNjb3VudGluZzwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+bjwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjIwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgIDxHcm91cD50aW1lX2FjY291bnRpbmc8L0dyb3VwPgogICAgICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5SZXBvcnRpbmc8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPlJlcG9ydGluZzwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+cmVwb3J0aW5nLnBuZzwvSW1hZ2U+CiAgICAgICAgICAgICAgICAgICAgPExpbms+QWN0aW9uPUFnZW50VGltZUFjY291bnRpbmcmYW1wO1N1YmFjdGlvbj1SZXBvcnRpbmc8L0xpbms+CiAgICAgICAgICAgICAgICAgICAgPFR5cGU+PC9UeXBlPgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz48L0Jsb2NrPgogICAgICAgICAgICAgICAgICAgIDxOYXZCYXI+VGltZUFjY291bnRpbmc8L05hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8QWNjZXNzS2V5PjwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjUwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8R3JvdXA+dGltZV9hY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+U2V0dGluZzwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+U2V0dGluZzwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+bW9kdWxlX3NldHRpbmcucG5nPC9JbWFnZT4KICAgICAgICAgICAgICAgICAgICA8TGluaz5BY3Rpb249QWdlbnRUaW1lQWNjb3VudGluZyZhbXA7U3ViYWN0aW9uPVNldHRpbmc8L0xpbms+CiAgICAgICAgICAgICAgICAgICAgPFR5cGU+PC9UeXBlPgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz48L0Jsb2NrPgogICAgICAgICAgICAgICAgICAgIDxOYXZCYXI+VGltZUFjY291bnRpbmc8L05hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8QWNjZXNzS2V5PjwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjkwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICA8L0Zyb250ZW5kTW9kdWxlUmVnPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0UHJvamVjdE5hbWUiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZhdWx0IG5hbWUgZm9yIG5ldyBwcm9qZWN0cy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+SW5pdGlhbGVyIFByb2plY3RuYW1lIGJlaW0gZXJzdGVsbGVuIGVpbmVzIG5ldWVuIFByb2pla3QgaW4gZGVyIFplaXRlcmZhc3N1bmcuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0UHJvamVjdFN0YXR1cyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgc3RhdHVzIG9mIGFuIG5ldyBwcm9qZWN0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5BbmZhbmdzc3RhdHVzIGVpbmVzIG5ldWVuIFByb2pla3RzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPE9wdGlvbiBTZWxlY3RlZElEPSIxIj4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMCI+aW52YWxpZDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMSI+dmFsaWQ8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0QWN0aW9uTmFtZSIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgbmFtZSBmb3IgbmV3IGFjdGlvbnMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkluaXRpYWxlciBOYW1lIGVpbmVyIG5ldWVuIFTkdGlna2VpdCBpbiBkZXIgWmVpdGVyZmFzc3VuZy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+PC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRBY3Rpb25TdGF0dXMiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZhdWx0IHN0YXR1cyBvZiBhbiBuZXcgYWN0aW9uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5BbmZhbmdzc3RhdHVzIGVpbmVyIG5ldWVuIFTkdGlna2VpdC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxPcHRpb24gU2VsZWN0ZWRJRD0iMSI+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjAiPmludmFsaWQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjEiPnZhbGlkPC9JdGVtPgogICAgICAgICAgICA8L09wdGlvbj4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdFVzZXJXZWVrbHlIb3VycyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgc2V0dGluZyBmb3IgdGhlIHN0YW5kYXJkIHdlZWtseSBob3Vycy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+U3RhbmRhcmQgV29jaGVuYXJiZWl0c3plaXQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjQwPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRVc2VyTGVhdmVEYXlzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzZXR0aW5nIGZvciBsZWF2ZSBkYXlzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5TdGFuZGFyZCBVcmxhdWJzdGFnZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+MjM8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdFVzZXJPdmVydGltZSIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgc2V0dGluZyBmb3Igb3ZlcnRpbWUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkV2ZW50dWVsbCB2b3JoYW5kZW5lciDcYmVyc3R1bmRlbvxiZXJ0cmFnLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZyBSZWdleD0iIj4wPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRVc2VyRGF0ZVN0YXJ0IiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzZXR0aW5nIGZvciBkYXRlIHN0YXJ0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5TdGFydGRhdHVtIGb8ciBkaWUgRWluZ2FiZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjIwMDgtMDEtMDE8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdFVzZXJEYXRlRW5kIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzZXR0aW5nIGZvciBkYXRlIGVuZC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RW5kZGF0dW0gZvxyIGRpZSBFaW5nYWJlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+MjAwOC0xMi0zMTwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0VXNlclN0YXR1cyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgc3RhdHVzIG9mIGFuIG5ldyB1c2VyLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5BbmZhbmdzc3RhdHVzIGVpbmVzIG5ldWVuIEJlbnV0emVycy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxPcHRpb24gU2VsZWN0ZWRJRD0iMSI+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjAiPmludmFsaWQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjEiPnZhbGlkPC9JdGVtPgogICAgICAgICAgICA8L09wdGlvbj4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6TWF4SW50ZXJ2YWxPZkluY29tcGxldGVEYXlzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+V29ya2luZyBkYXlzIGFmdGVyIGF0IHRoZSBsYXRlc3Qgd29ya2luZyB1bml0cyBoYXZlIHRvIGluc2VydC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+TGVndCBmZXN0LCBuYWNoIHdpZXZpZWxlbiBXZXJrdGFnZW4gc3DkdGVzdGVucyBkaWUgQXJiZWl0c3N0dW5kZW4gZWluZ2V0cmFnZW4gd2VyZGVuIG38c3Nlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+NDwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpNYXhJbnRlcnZhbE9mSW5jb21wbGV0ZURheXNCZWZvcmVXYXJuaW5nIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+U2hvdyB3YXJuaW5nLCBpZiB0byBtdWNoIGluY29tcGxldGUgd29ya2luZyBkYXlzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5aZWlndCBlaW5lIFdhcm51bmcsIHdlbm4genV2aWVsZSBBcmJlaXRzdGFnZSBuaWNodCBlaW5nZXRyYWdlbiB3dXJkZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjI8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGcm9udGVuZDo6Tm90aWZ5TW9kdWxlIyMjODg4LVRpbWVBY2NvdW50aW5nIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBub3RpZmljYXRpb24gbW9kdWxlIGdpdmVzIGEgd2FybmluZyBpZiB0byBtdWNoIGluY29tcGxldGUgd29ya2luZyBkYXlzIGF2YWlsYWJsZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+TW9kdWwsIGRhcyBkZW4gQWdlbnQgaW0gTm90aWZpY2F0aW9uLUJlcmVpY2ggZGVzIEFnZW50LUludGVyZmFjZXMgZGFy/GJlciBpbmZvcm1pZXJ0LCB3ZW5uIHNjaG9uIHp1IGxhbmdlIGtlaW5lIFN0dW5kZW4gbWVociBlaW5nZXRyYWdlbiB3dXJkZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ6Ok1vZHVsZU5vdGlmeTwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJNb2R1bGUiPktlcm5lbDo6T3V0cHV0OjpIVE1MOjpOb3RpZmljYXRpb25UaW1lQWNjb3VudGluZzwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpNYXhBbGxvd2VkSW5zZXJ0RGF5cyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkZvciBob3cgbWFueSBkYXlzIGFnbyB5b3UgY2FuIGluc2VydCB3b3JraW5nIHVuaXRzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5MZWd0IGZlc3QsIGJpcyB3YW5uIG1hbiBpbiDkbHRlcmUgWmVpdGVpbnRy5GdlIGJlYXJiZWl0ZW4ga2FubiAoei4gQi4gMTAgVGFnZSB6dXL8Y2tsaWVnZW5kKS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+MTA8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6UmVkdWNlVGltZSIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRvIHVzZSwgaWYgc29tZSBhY3Rpb25zIGFjb3VudCByZWR1Y2VkIHdvcmtpbmcgaG91cnMoS2V5ID0mZ3Q7IHRyYXZlbGluZzsgQ29udGVudCA9Jmd0OyA1MCkuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkhpZXIga2FubiBtYW4gZWluZ2ViZW4sIG9iIGb8ciBlaW5lIGJlc3RpbW10ZSBU5HRpZ2tlaXQsIGRpZSB6dSB2ZXJyZWNobmVuZGVuIFN0dW5kZW4gZ2Vr/HJ6dCB3ZXJkZW4uIFouIEIgd2VubiBSZWlzZXplaXRlbiBudXIgenVyIEjkbGZ0ZSB2ZXJn/HRldCB3ZXJkZW4gKEtleSA9Jmd0OyBqb3VybmV5OyBDb250ZW50ID0mZ3Q7IDUwKS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJ0cmF2ZWxpbmciPjUwPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OklucHV0SG91cnNXaXRob3V0U3RhcnRFbmRUaW1lIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+U2V0dGluZywgaWYgd29ya2luZyBob3VycyBjYW4gaW5zZXJ0IHdpdGhvdXQgc3RhcnQtIGFuZCBlbmR0aW1lLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5MZWd0IGZlc3QsIG9iIG1hbiBBcmJlaXRzc3R1bmRlbiBhdWNoIG9obmUgQW5mYW5ncy0gdW5kIEVuZHplaXQgZWluZ2ViZW4ga2Fubi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxPcHRpb24gU2VsZWN0ZWRJRD0iMSI+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjAiPk5vPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIxIj5ZZXM8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlByZUFwcGxpY2F0aW9uTW9kdWxlIyMjQWdlbnRUaW1lQWNjb3VudGluZyIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgbW9kdWxlIHlvdSBjYW4gZm9yY2UgaW5zZXJ0cyBpbiB0aGUgVGltZUFjY291bnRpbmcuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPk1pdCBIaWxmZSBkaWVzZXMgTW9kdWxzIGv2bm5lbiBFaW50cuRnZSBpbiBkaWUgWmVpdGVyZmFzc3VuZyBlcnp3dW5nZW4gd2VyZGVuLCBpbiBkZW0gYmVpbSBFaW5sb2dnZW4gZGFzIFBvcnRhbCBnZWJsb2NrdCB3aXJkIHVuZCBudXIgZGFzIFplaXRlcmZhc3N1bmdzZmVuc3RlciBlcnNjaGVpbnQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPktlcm5lbDo6TW9kdWxlczo6QWdlbnRUaW1lQWNjb3VudGluZzwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KPC9vdHJzX2NvbmZpZz4K
#--
# Kernel/System/TimeAccounting.pm - all time accounting functions
# Copyright (C) 2003-2007 OTRS GmbH, http://otrs.com/
# --
# $Id: TimeAccounting.pm,v 1.10 2007/06/05 14:19:49 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl.txt.
# --

package Kernel::System::TimeAccounting;

use strict;
use vars qw(@ISA $VERSION);
$VERSION = '$Revision: 1.10 $';
$VERSION =~ s/^\$.*:\W(.*)\W.+?$/$1/;

use Date::Pcalc qw(Today Days_in_Month Day_of_Week);
#use Kernel::System::CalendarEvent;

=head1 NAME#

Kernel::System::TimeAccounting - timeaccounting lib

=head1 SYNOPSIS

All timeaccounting functions

=head1 PUBLIC INTERFACE#

=over 4

=item new()

create a object

    use Kernel::Config;
    use Kernel::System::Log;
    use Kernel::System::DB;
    use Kernel::System::TimeAccounting;

    my $ConfigObject = Kernel::Config->new();
    my $LogObject    = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        LogObject => $LogObject,
    );
    my $TimeAccountingObject = Kernel::System::TimeAccounting->new(
        ConfigObject => $ConfigObject,
        LogObject => $LogObject,
        DBObject => $DBObject,
        UserID => 123,
    );

=cut

sub new {
    my $Type = shift;
    my %Param = @_;

    # allocate new hash for object
    my $Self = {};
    bless ($Self, $Type);

    # check needed objects
    foreach (qw(DBObject ConfigObject LogObject UserID TimeObject)) {
        $Self->{$_} = $Param{$_} || die "Got no $_!";
    }

    return $Self;
}

=item UserCurrentPeriodGet()

returns a hash with the user of the current period data

    my %UserData = $TimeAccountingObject->UserCurrentPeriodGet(
        Year  => '2005',
        Month => '12';
        Day   => '24'
    );

=cut

sub UserCurrentPeriodGet {
    my $Self    = shift;
    my %Param   = @_;
    my %Data    = ();
    foreach (qw(Year Month Day)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "UserCurrentPeriodGet: Need $_!");
            return;
        }
    }

    my $Date = sprintf("%04d-%02d-%02d", $Param{Year}, $Param{Month}, $Param{Day}) . " 00:00:00";

    # db select
    $Self->{DBObject}->Prepare (
        SQL => "SELECT user_id, preference_period, date_start, date_end, weekly_hours, leave_days, overtime, status FROM time_accounting_user_period ".
            "WHERE date_start <= '" . $Date . "' AND date_end  >='" . $Date . "'",
    );

    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {

        $Data{$Row[0]}{UserID}           =  $Row[0];
        $Data{$Row[0]}{Period}           =  $Row[1];
        $Data{$Row[0]}{DateStart}        =  substr($Row[2], 0, 10);
        $Data{$Row[0]}{DateEnd}          =  substr($Row[3], 0, 10);
        $Data{$Row[0]}{WeeklyHours}      =  $Row[4];
        $Data{$Row[0]}{LeaveDays}        =  $Row[5];
        $Data{$Row[0]}{Overtime}         =  $Row[6];
        $Data{$Row[0]}{UserStatus}       =  $Row[7];
    }

    return %Data;
}

=item UserReporting()

returns a hash with informations about leavedays, overtimes,
workinghours etc. of all users

    my %Data = $TimeAccountingObject->UserReporting(
        Year  => '2005',
        Month => '12',
        Day   => '12',      # Optional
    );

=cut

sub UserReporting {
    my $Self    = shift;
    my %Param   = @_;
    my %Data    = ();
    foreach (qw(Year Month Day)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
        if (!$Param{$_} && $_ ne 'Day') {
            $Self->{LogObject}->Log(Priority => 'error', Message => "UserReporting: Need $_!");
            return;
        }
    }
    if (!$Param{Day}) {
        $Param{Day} = Days_in_Month($Param{Year},  $Param{Month});
    }

    my %UserCurrentPeriod = $Self->UserCurrentPeriodGet(%Param);
    my $YearStart       = 0;
    my $MonthStart      = 0;
    my $DayStart        = 0;
    my $MonthStartPoint = 0;
    my $DayStartPoint   = 0;
    my $MonthEndPoint   = 0;
    my $DayEndPoint     = 0;
    my $YearEnd         = $Param{Year};
    my $MonthEnd        = $Param{Month};
    my $DayEnd          = $Param{Day};

    foreach my $UserID (keys %UserCurrentPeriod){
        if ($UserCurrentPeriod{$UserID}{DateStart} =~ /^(\d+)-(\d+)-(\d+)/) {
            $YearStart  = $1;
            $MonthStart = $2;
            $DayStart   = $3;
        }
        $Data{$UserID}{LeaveDay}         = 0;
        $Data{$UserID}{Diseased}         = 0;
        $Data{$UserID}{Overtime}         = 0;
        $Data{$UserID}{TargetState}      = 0;
        $Data{$UserID}{LeaveDayTotal}    = 0;
        $Data{$UserID}{DiseasedTotal}    = 0;
        $Data{$UserID}{OvertimeTotal}    = 0;
        $Data{$UserID}{TargetStateTotal} = 0;

        my $Counter = 0;
        for (my $Year = $YearStart; $Year <= $YearEnd; $Year++) {
            if ($Year == $YearStart) {
                $MonthStartPoint = $MonthStart;
            }
            else {
                $MonthStartPoint = 1;
            }
            if ($Year == $YearEnd) {
                $MonthEndPoint = $MonthEnd;
            }
            else {
                $MonthEndPoint = 12;
            }
            for (my $Month = $MonthStartPoint; $Month <= $MonthEndPoint; $Month++) {
                if ($Year == $YearStart && $Month == $MonthStart ) {
                    $DayStartPoint = $DayStart;
                }
                else {
                    $DayStartPoint = 1;
                }
                if ($Year == $YearEnd && $Month == $MonthEnd ) {
                    $DayEndPoint = $DayEnd;
                }
                else {
                    $DayEndPoint = Days_in_Month($Year, $Month);
                }
                for (my $Day = $DayStartPoint; $Day <= $DayEndPoint; $Day++) {
                    my %WorkingUnit = $Self->WorkingUnitsGet(
                        Year   => $Year,
                        Month  => $Month,
                        Day    => $Day,
                        UserID => $UserID,
                    );
                    my $WorkingHours = 0;
                    my $LeaveDay     = 0;
                    my $Diseased     = 0;
                    my $Overtime     = 0;
                    my $TargetState  = 0;
                    foreach (keys %WorkingUnit) {
                        $WorkingHours += $WorkingUnit{$_}{Period};
                        if ($WorkingUnit{$_}{ProjectID} == -1) {
                            if ($WorkingUnit{$_}{ActionID} == -2) {
                                $Data{$UserID}{LeaveDayTotal}++;
                                $LeaveDay = 1;
                            }
                            elsif ($WorkingUnit{$_}{ActionID} == -1) {
                                $Data{$UserID}{DiseasedTotal}++;
                                $Diseased = 1;
                            }
                            elsif ($WorkingUnit{$_}{ActionID} == -3) {
                                $Data{$UserID}{OvertimeTotal}++;
                                $Overtime = 1;
                            }
                        }
                    }

                    $Data{$UserID}{WorkingHoursTotal} += $WorkingHours;
                    my $VacationCheck = $Self->VacationCheck(
                        Year  => $Year,
                        Month => $Month,
                        Day   => $Day,
                    );
                    my $Weekday = Day_of_Week($Year, $Month, $Day);
                    if ($Weekday != 6 && $Weekday != 7
                        && !$VacationCheck
                        && !$Diseased && !$LeaveDay
                    ) {
                        $Data{$UserID}{TargetStateTotal} += $UserCurrentPeriod{$UserID}{WeeklyHours}/5;
                        $TargetState = $UserCurrentPeriod{$UserID}{WeeklyHours}/5;
                    }

                    if ($Month == $MonthEnd && $Year == $YearEnd) {
                        $Data{$UserID}{TargetState}  += $TargetState;
                        $Data{$UserID}{WorkingHours} += $WorkingHours;
                        $Data{$UserID}{LeaveDay}     += $LeaveDay;
                        $Data{$UserID}{Diseased}     += $Diseased;
                    }
                }
            }
        }
        $Data{$UserID}{Overtime}          = $Data{$UserID}{WorkingHours}           - $Data{$UserID}{TargetState};
        $Data{$UserID}{OvertimeTotal}     = $UserCurrentPeriod{$UserID}{Overtime}  + $Data{$UserID}{WorkingHoursTotal} - $Data{$UserID}{TargetStateTotal};
        $Data{$UserID}{OvertimeUntil}     = $Data{$UserID}{OvertimeTotal}          - $Data{$UserID}{Overtime};
        $Data{$UserID}{LeaveDayRemaining} = $UserCurrentPeriod{$UserID}{LeaveDays} - $Data{$UserID}{LeaveDayTotal};
    }
    return %Data;
}

=item ProjectSettingsGet()

returns a hash with the project data

    my %ProjectData = $TimeAccountingObject->ProjectSettingsGet();

=cut

sub ProjectSettingsGet {
    my $Self    = shift;
    my %Data    = ();

    # db select
    $Self->{DBObject}->Prepare (
        SQL => "SELECT id, project, description, status FROM time_accounting_project",
    );

    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $Data{Project}           {$Row[0]} =  $Row[1];
        $Data{ProjectDescription}{$Row[0]} =  $Row[2];
        $Data{ProjectStatus}     {$Row[0]} =  $Row[3];
    }
    return %Data;
}

=item ProjectSettingsInsert()

insert new project data in the db

    $TimeAccountingObject->ProjectSettingsInsert(
        Project       => 'internal', # optional
        ProjectStatus => 1 || 0,     # optional
    );

=cut

sub ProjectSettingsInsert {
    my $Self    = shift;
    my %Param   = @_;

    if (!$Param{Project}) {
        $Param{Project}       = $Self->{ConfigObject}->Get('TimeAccounting::DefaultProjectName') || '';
    }
    if (!$Param{ProjectStatus}) {
        $Param{ProjectStatus} = $Self->{ConfigObject}->Get('TimeAccounting::DefaultProjectStatus') || '0';
    }

    if (!$Param{ProjectDescription}) {
        $Param{ProjectDescription} = '';
    }

    # db quote
    foreach (keys %Param) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }

    # build sql

    my $SQL = "INSERT INTO time_accounting_project (project, description, status) " .
        "VALUES " .
        "('" . $Param{Project} . "', '" . $Param{ProjectDescription} . "', '" . $Param{ProjectStatus} . "')";

    # db insert
    if (!$Self->{DBObject}->Do(SQL => $SQL)) {
        return;
    }
    return 1;
}

=item ProjectSettingsUpdate()

update project data in the db

    $TimeAccountingObject->ProjectSettingsUpdate(
        1    => {                        # equal ProjectID
            Project       => 'internal',
            ProjectStatus => 1 || 0,
        },
        2    => {                        # equal ProjectID
            Project       => 'projectname',
            ProjectStatus => 1 || 0,
        },
        3    => ......
    );

=cut

sub ProjectSettingsUpdate {
    my $Self    = shift;
    my %Param   = @_;

    foreach my $ProjectID (sort keys  %Param) {
        if ($Param{$ProjectID}{Project}) {
            # db quote
            foreach (keys %{$Param{$ProjectID}}) {
                $Param{$ProjectID}{$_} = $Self->{DBObject}->Quote($Param{$ProjectID}{$_}) || '';
            }
            # build sql
            my $SQL = "UPDATE time_accounting_project " .
                "SET project = '" . $Param{$ProjectID}{Project}.
                    "', status = '" . $Param{$ProjectID}{ProjectStatus} .
                    "', description = '" . $Param{$ProjectID}{ProjectDescription} . "' " .
                "WHERE id = '" . $ProjectID . "'";
            # db insert
            if (!$Self->{DBObject}->Do(SQL => $SQL)) {
                return;
            }
        }
    }
    return 1;
}

=item ActionSettingsGet()

returns a hash with the action settings

    my %ActionData = $TimeAccountingObject->ActionSettingsGet();

=cut

sub ActionSettingsGet {
    my $Self    = shift;
    my %Data    = ();

    # db select
    $Self->{DBObject}->Prepare (
        SQL => "SELECT id, action, status FROM time_accounting_action",
    );

    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $Data{$Row[0]}{Action}       =  $Row[1];
        $Data{$Row[0]}{ActionStatus} =  $Row[2];
    }
    return %Data;
}

=item ActionSettingsInsert()

insert new action data in the db

    $TimeAccountingObject->ActionSettingsInsert(
        Action       => 'meeting',   # optional
        ActionStatus => 1 || 0,      # optional
    );

=cut

sub ActionSettingsInsert {
    my $Self    = shift;
    my %Param   = @_;

    # db quote
    foreach (keys %Param) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }

    if (!$Param{Action}) {
        $Param{Action}       = $Self->{ConfigObject}->Get('TimeAccounting::DefaultActionName') || '';
    }
    if (!$Param{ActionStatus}) {
        $Param{ActionStatus} = $Self->{ConfigObject}->Get('TimeAccounting::DefaultActionStatus') || '0';
    }

    # build sql
    my $SQL = "INSERT INTO time_accounting_action (action, status)" .
        " VALUES" .
        " ('" . $Param{Action}. "' , '" . $Param{ActionStatus} . "')";
    # db insert
    if (!$Self->{DBObject}->Do(SQL => $SQL)) {
        return;
    }
    return 1;
}

=item ActionSettingsUpdate()

update action data in the db

    $TimeAccountingObject->ActionSettingsUpdate(
        1    => {                      # equal ActionID
            Action       => 'meeting',
            ActionStatus => 1 || 0,
        },
        2    => {                      # equal ActionID
            Action       => 'journey',
            ActionStatus => 1 || 0,
        },
        3    => ......
    );

=cut

sub ActionSettingsUpdate {
    my $Self    = shift;
    my %Param   = @_;
    my $SQL     = '';

    foreach my $ActionID (sort keys  %Param) {
        if ($Param{$ActionID}{Action}) {
            # db quote
            foreach (keys %{$Param{$ActionID}}) {
                $Param{$ActionID}{$_} = $Self->{DBObject}->Quote($Param{$ActionID}{$_}) || '';
            }
            # build sql
            $SQL = "UPDATE time_accounting_action " .
                "SET action = '" . $Param{$ActionID}{Action}. "', status = '" . $Param{$ActionID}{ActionStatus} . "' " .
                "WHERE id = '" . $ActionID . "'";
            # db insert
            if (!$Self->{DBObject}->Do(SQL => $SQL)) {
                return;
            }
        }
    }
    return 1;
}

=item UserGet()

returns a hash with the user data

    my %UserData = $TimeAccountingObject->UserGet();

=cut

sub UserGet {
    my $Self    = shift;
    my %Data    = ();
    # db select
    $Self->{DBObject}->Prepare (
        SQL => "SELECT user_id, description, show_overtime, create_project FROM time_accounting_user",
    );

    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $Data{$Row[0]}{UserID}        =  $Row[0];
        $Data{$Row[0]}{Description}   =  $Row[1];
        $Data{$Row[0]}{ShowOvertime}  =  $Row[2];
        $Data{$Row[0]}{CreateProject} =  $Row[3];
    }
    return %Data;
}

=item UserSettingsGet()

returns a hash with the user period data

    my %UserData = $TimeAccountingObject->UserSettingsGet();

=cut

sub UserSettingsGet {
    my $Self    = shift;
    my %Data    = ();
    # db select
    $Self->{DBObject}->Prepare (
        SQL => "SELECT user_id, preference_period, date_start, date_end, weekly_hours, leave_days, overtime, status FROM time_accounting_user_period",
    );

    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $Data{$Row[0]}{$Row[1]}{UserID}           =  $Row[0];
        $Data{$Row[0]}{$Row[1]}{Period}           =  $Row[1];
        $Data{$Row[0]}{$Row[1]}{DateStart}        =  substr($Row[2], 0, 10);
        $Data{$Row[0]}{$Row[1]}{DateEnd}          =  substr($Row[3], 0, 10);
        $Data{$Row[0]}{$Row[1]}{WeeklyHours}      =  $Row[4];
        $Data{$Row[0]}{$Row[1]}{LeaveDays}        =  $Row[5];
        $Data{$Row[0]}{$Row[1]}{Overtime}         =  $Row[6];
        $Data{$Row[0]}{$Row[1]}{UserStatus}       =  $Row[7];
    }
    return %Data;
}

=item UserSettingsInsert()

insert new user data in the db

    $TimeAccountingObject->UserSettingsInsert(
        UserID       => '2',
        Period       => '2',
    );

=cut

sub UserSettingsInsert {
    my $Self    = shift;
    my %Param   = @_;
    my $ID      = '';
    my $SQL     = '';

    $Param{WeeklyHours} = $Self->{ConfigObject}->Get('TimeAccounting::DefaultUserWeeklyHours') || '40';
    $Param{LeaveDays}   = $Self->{ConfigObject}->Get('TimeAccounting::DefaultUserLeaveDays')   || '25';
    $Param{UserStatus}  = $Self->{ConfigObject}->Get('TimeAccounting::DefaultUserStatus')      || '1';
    $Param{Overtime}    = $Self->{ConfigObject}->Get('TimeAccounting::DefaultUserOvertime')    || '0';
    $Param{DateEnd}     = $Self->{ConfigObject}->Get('TimeAccounting::DefaultUserDateEnd')     || '2007-12-31';
    $Param{DateStart}   = $Self->{ConfigObject}->Get('TimeAccounting::DefaultUserDateStart')   || '2007-01-01';
    $Param{Description}   = $Self->{ConfigObject}->Get('TimeAccounting::DefaultUserDescription') || '';

    # db quote
    foreach (keys %Param) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_});
        if( !defined($Param{$_})) {
            $Param{$_} = '';
        }
    }

    # build sql
    $SQL = "INSERT INTO time_accounting_user_period (user_id, preference_period, date_start, date_end," .
        " weekly_hours, leave_days, overtime, status)" .
        " VALUES" .
        " ('$Param{UserID}', '$Param{Period}', '$Param{DateStart} 00:00:00', '$Param{DateEnd} 00:00:00'," .
        " '$Param{WeeklyHours}', '$Param{LeaveDays}', '$Param{Overtime}', '$Param{UserStatus}')";
    # db insert
    if (!$Self->{DBObject}->Do(SQL => $SQL)) {
        return;
    }

    # Split the following code in a seperate function!

    #check if the user still exists
    my $UserID;
    # build sql
    $Self->{DBObject}->Prepare (
        SQL => "SELECT user_id FROM time_accounting_user WHERE user_id = '" . $Param{UserID}. "'",
    );

    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $UserID           =  $Row[0];
    }
    if (!defined($UserID)) {
        $SQL = "INSERT INTO time_accounting_user (user_id, description)" .
            " VALUES" .
            " ('$Param{UserID}', '$Param{Description}')";
        # db insert
        if (!$Self->{DBObject}->Do(SQL => $SQL)) {
            return;
        }
    }
    return 1;
}

=item UserSettingsUpdate()

update user data in the db

    $TimeAccountingObject->UserSettingsUpdate(
        UserID => 1,
        Description => 'Some Text',
        CreateProject => 1 || 0,
        ShowOvertime  => 1 || 0,
        1    => {
            UserID => 1,
            Description => 'Some Text',
            CreateProject => 1 || 0,
            ShowOvertime  => 1 || 0,
            1 => {
                UserID       => 1,
                Period       => '1',
                DateStart    => '2005-12-12',
                DateEnd      => '2005-12-31',
                WeeklyHours  => '38',
                LeaveDays    => '25',
                Overtime     => '38',
                UserStatus   => 1 || 0,
            },
        },
        2    => {
            1 => {
                UserID       => 2,
                Period       => '1',
                DateStart    => '2005-12-12',
                DateEnd      => '2005-12-31',
                WeeklyHours  => '38',
                LeaveDays    => '25',
                Overtime     => '38',
                UserStatus   => 1 || 0,
            },
        },
        3    => ......
    );

=cut

sub UserSettingsUpdate {
    my $Self    = shift;
    my %Param   = @_;
    foreach my $UserID (sort keys  %Param) {
        foreach (qw(UserID Description)) {
            $Param{$UserID}{$_} = $Self->{DBObject}->Quote($Param{$UserID}{$_}) || '';
        }

        #set default for ShowOverTime...
        if( !defined($Param{$UserID}{ShowOvertime})) {
            $Param{$UserID}{ShowOvertime} = 0;
        }

        #set default for CreateProject...
        if( !defined($Param{$UserID}{CreateProject})) {
            $Param{$UserID}{CreateProject} = 0;
        }

        # build sql
        my $SQL = "UPDATE time_accounting_user " .
            "SET description = '"   . $Param{$UserID}{Description} .
            "', show_overtime = '"  . $Param{$UserID}{ShowOvertime} .
            "', create_project = '" . $Param{$UserID}{CreateProject} . "' " .
            "WHERE user_id = '" . $Param{$UserID}{UserID}. "'";
        # db insert
        if (!$Self->{DBObject}->Do(SQL => $SQL)) {
            return;
        }

        foreach (qw(UserID Description ShowOvertime CreateProject)) {
            delete ($Param{$UserID}{$_});
        }

        foreach my $Period (keys %{$Param{$UserID}}) {
            # db quote
            foreach (keys %{$Param{$UserID}{$Period}}) {
                $Param{$UserID}{$Period}{$_} = $Self->{DBObject}->Quote($Param{$UserID}{$Period}{$_}) || '';
            }
            # build sql
            my $SQL = "UPDATE time_accounting_user_period " .
                "SET leave_days = '"  . $Param{$UserID}{$Period}{LeaveDays} .
                "', date_start = '"   . $Param{$UserID}{$Period}{DateStart} .
                "', date_end = '"     . $Param{$UserID}{$Period}{DateEnd} .
                "', overtime = '"     . $Param{$UserID}{$Period}{Overtime} .
                "', weekly_hours = '" . $Param{$UserID}{$Period}{WeeklyHours} .
                "', status = '"       . $Param{$UserID}{$Period}{UserStatus}. "' " .
                "WHERE user_id = '" . $Param{$UserID}{$Period}{UserID} .
                "' AND preference_period = '" . $Period . "'";
            # db insert
            if (!$Self->{DBObject}->Do(SQL => $SQL)) {
                return;
            }
        }
    }
    return 1;
}

=item WorkingUnitsCompletnessCheck()

returns a hash with the incomplete working days and
the information if the incomplete working day are in the allowed
range.

    my %WorkingUnitsCheck = $TimeAccountingObject->WorkingUnitsCompletnessCheck(
        UserID => '2',    # Optional
    );

=cut

sub WorkingUnitsCompletnessCheck {
    my $Self                = shift;
    my %Param               = @_;
    my %Data                = ();
    my $WorkingUnitID       = 0;
    my %CompleteWorkingDays = ();
    my ($Sec, $Min, $Hour, $Day, $Month, $Year) = $Self->{TimeObject}->SystemTime2Date(
        SystemTime => $Self->{TimeObject}->SystemTime(),
    );

    $Param{UserID} = $Self->{DBObject}->Quote($Param{UserID}) || '';

    if (!$Param{UserID}) {
        $Param{UserID} = $Self->{UserID};
    }

    $Self->{DBObject}->Prepare (
        SQL => "SELECT DISTINCT time_start FROM time_accounting_table WHERE user_id = '$Param{UserID}'",
    );

    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        if ($Row[0] =~ /^(\d+)-(\d+)-(\d+)(.+?)/) {
            $CompleteWorkingDays{$1}{$2}{$3} = 1;
        }
    }
    my %UserCurrentPeriod = $Self->UserCurrentPeriodGet(
        Year   => $Year,
        Month  => $Month,
        Day    => $Day,
    );
    my $WorkingDays     = 0;
    my $YearStart       = 0;
    my $MonthStart      = 0;
    my $DayStart        = 0;
    my $MonthStartPoint = 0;
    my $DayStartPoint   = 0;
    my $MonthEndPoint   = 0;
    my $DayEndPoint     = 0;
    my $YearEnd         = $Year;
    my $MonthEnd        = $Month;
    my $DayEnd          = $Day;

    if ($UserCurrentPeriod{$Param{UserID}}{DateStart} =~ /^(\d+)-(\d+)-(\d+)/) {
        $YearStart  = $1;
        $MonthStart = $2;
        $DayStart   = $3;
    }

    for (my $Year = $YearStart; $Year <= $YearEnd; $Year++) {
        if ($Year == $YearStart) {
            $MonthStartPoint = $MonthStart;
        }
        else {
            $MonthStartPoint = 1;
        }
        if ($Year == $YearEnd) {
            $MonthEndPoint = $MonthEnd;
        }
        else {
            $MonthEndPoint = 12;
        }
        for (my $Month = $MonthStartPoint; $Month <= $MonthEndPoint; $Month++) {
            if ($Year == $YearStart && $Month == $MonthStart ) {
                $DayStartPoint = $DayStart;
            }
            else {
                $DayStartPoint = 1;
            }
            if ($Year == $YearEnd && $Month == $MonthEnd ) {
                $DayEndPoint = $DayEnd;
            }
            else {
                $DayEndPoint = Days_in_Month($Year, $Month);
            }
            my $MonthString = sprintf("%02d",$Month);
            for (my $Day = $DayStartPoint; $Day <= $DayEndPoint; $Day++) {
                my $VacationCheck = $Self->VacationCheck(
                    Year  => $Year,
                    Month => $Month,
                    Day   => $Day,
                );

                my $DayString = sprintf("%02d",$Day);
                my $Weekday = Day_of_Week($Year, $Month, $Day);
                if ($Weekday != 6 && $Weekday != 7 && !$VacationCheck) {
                    $WorkingDays++;
                }
                if ($Weekday != 6 && $Weekday != 7
                    && !$VacationCheck
                    && !$CompleteWorkingDays{$Year}{$MonthString}{$DayString}
                ) {
                    $Data{Incomplete}{$Year}{$MonthString}{$DayString} = $WorkingDays;
                }
            }
        }
    }
    my $MaxIntervallOfIncompleteDays              = $Self->{ConfigObject}->Get('TimeAccounting::MaxIntervalOfIncompleteDays') || '5';
    my $MaxIntervallOfIncompleteDaysBeforeWarning = $Self->{ConfigObject}->Get('TimeAccounting::MaxIntervalOfIncompleteDaysBeforeWarning') || '3';

    foreach my $Year (keys %{$Data{Incomplete}}) {
        foreach my $Month (keys %{$Data{Incomplete}{$Year}}) {
            foreach my $Day (keys %{$Data{Incomplete}{$Year}{$Month}}) {
                if ($Data{Incomplete}{$Year}{$Month}{$Day} < $WorkingDays - $MaxIntervallOfIncompleteDays) {
                    $Data{EnforceInsert} = 1;
                }
                elsif ($Data{Incomplete}{$Year}{$Month}{$Day} < $WorkingDays - $MaxIntervallOfIncompleteDaysBeforeWarning) {
                    $Data{Warning} = 1;
                }
            }
        }
    }
    return %Data;
}

=item WorkingUnitsGet()

returns a hash with the working units data

    my %WorkingUnitsData = $TimeAccountingObject->WorkingUnitsGet(
        Year   => '2005',
        Month  => '7',
        Day    => '13',
        UserID => '2',    # Optional
    );

=cut

sub WorkingUnitsGet {
    my $Self          = shift;
    my %Param         = @_;
    my %Data          = ();
    my $WorkingUnitID = 0;
    foreach (keys %Param) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    if (!$Param{UserID}) {
        $Param{UserID} = $Self->{UserID};
    }

    my $Date = sprintf("%04d-%02d-%02d", $Param{Year}, $Param{Month}, $Param{Day});
    $Self->{DBObject}->Prepare (
        SQL => "SELECT user_id, project_id, action_id, remark, time_start, time_end," .
            " period FROM time_accounting_table WHERE time_start LIKE '$Date%'" .
            " AND user_id = '$Param{UserID}' ORDER by id",
    );

    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        if ($Row[4] =~ /^(.+?)\s(\d+:\d+):(\d+)/) {
            $WorkingUnitID++;
            $Data{$WorkingUnitID}{UserID}     =  $Row[0];
            $Data{$WorkingUnitID}{ProjectID}  =  $Row[1];
            $Data{$WorkingUnitID}{ActionID}   =  $Row[2];
            $Data{$WorkingUnitID}{Remark}     =  $Row[3];
            $Data{$WorkingUnitID}{StartTime}  =  $2;
            if( defined($Row[6])) {
                $Data{$WorkingUnitID}{Period}     =  sprintf ("%.2f", $Row[6]);
            } else {
                $Data{$WorkingUnitID}{Period}     =  0;
            }

            $Data{$WorkingUnitID}{Date}       =  $1;
            if ($Row[5] =~ /^(.+?)\s(\d+:\d+):(\d+)/) {
                $Data{$WorkingUnitID}{EndTime}    =  $2;
            }
        }
    }
    return %Data;
}

=item WorkingUnitsInsert()

insert working units in the db

    $TimeAccountingObject->WorkingUnitsInsert(
        Year  => '2005',
        Month => '07',
        Day   => '02',
        1    => {
            ProjectID => 1,
            ActionID  => 23,
            Remark    => 4,
            StartTime => '7:30',
            EndTime   => '11:00',
            Period    => '8.5',
        },
        2 ......
    );

=cut

sub WorkingUnitsInsert {
    my $Self    = shift;
    my %Param   = @_;
    my $SQL     = '';

    foreach (qw(Year Month Day)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "WorkingUnitsInsert: Need $_!");
            return;
        }
    }
    my $Date = sprintf("%04d-%02d-%02d", $Param{Year}, $Param{Month}, $Param{Day});

    if (!$Self->WorkingUnitsDelete(
            Year  => $Param{Year},
            Month => $Param{Month},
            Day   => $Param{Day},
        )
    ) {
        $Self->{LogObject}->Log(Priority => 'error', Message => 'Can\'t delete Working Units!');
        return;
    }

    delete $Param{Year};
    delete $Param{Month};
    delete $Param{Day};

    #insert new working units
    foreach my $WorkingUnitID (sort keys  %Param) {
        # db quote
        foreach (keys %{$Param{$WorkingUnitID}}) {
            $Param{$WorkingUnitID}{$_} = $Self->{DBObject}->Quote($Param{$WorkingUnitID}{$_}) || '';
        }
        my $StartTime = $Date." ".$Param{$WorkingUnitID}{StartTime};
        my $EndTime   = $Date." ".$Param{$WorkingUnitID}{EndTime};

        if( (!defined($Param{$WorkingUnitID}{Period})) || ($Param{$WorkingUnitID}{Period} eq '')) {
            $Param{$WorkingUnitID}{Period} = 'NULL';
        }

        # build sql
        $SQL = "INSERT INTO time_accounting_table (user_id, project_id, action_id, remark," .
            " time_start, time_end, period, created )" .
            " VALUES" .
            " ('$Self->{UserID}', '$Param{$WorkingUnitID}{ProjectID}', '$Param{$WorkingUnitID}{ActionID}'," .
            " '$Param{$WorkingUnitID}{Remark}', '$StartTime', '$EndTime', $Param{$WorkingUnitID}{Period}," .
            " current_timestamp)";
        # db insert
        if (!$Self->{DBObject}->Do(SQL => $SQL)) {
            return;
        }
    }
    return 1;
}

=item WorkingUnitsDelete()

delets working units in the db

    $TimeAccountingObject->WorkingUnitsDelete(
        Year  => '2005',
        Month => '7',
        Day   => '13',
    );

=cut

sub WorkingUnitsDelete {
    my $Self    = shift;
    my %Param   = @_;
    my $SQL     = '';

    foreach (qw(Year Month Day)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "WorkingUnitsInsert: Need $_!");
            return;
        }
    }
    my $Date = sprintf("%04d-%02d-%02d", $Param{Year}, $Param{Month}, $Param{Day});

    # delete old working units
    $SQL = "DELETE FROM time_accounting_table " .
        "WHERE time_start LIKE '$Date%' AND user_id = '$Self->{UserID}'" ;
    if (!$Self->{DBObject}->Do(SQL => $SQL)) {
        return ;
    }
    return 1;
}

# FRAMEWORK-2.1: this function should be included in the timeobject
=item VacationCheck()

check if the selected day is a vacation (it doesn't matter if you
insert 01 or 1 for month or day in the function or in the SysConfig)

    $TimeAccountingObject->VacationCheck(
        Year  => '2005',
        Month => '7', || 07
        Day   => '13',
    );

=cut

sub VacationCheck {
    my $Self         = shift;
    my %Param        = @_;
    my $VacationName = '';

    # check required params
    foreach (qw(Year Month Day)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "VacationCheck: Need $_!"
            );
            return;
        }
    }
    $Param{Month} = sprintf("%02d", $Param{Month});
    $Param{Day}   = sprintf("%02d", $Param{Day});

    my $TimeVacationDays        = $Self->{ConfigObject}->Get('TimeVacationDays');
    my $TimeVacationDaysOneTime = $Self->{ConfigObject}->Get('TimeVacationDaysOneTime');
    if (defined($TimeVacationDays->{$Param{Month}}->{$Param{Day}})) {
        return $TimeVacationDays->{$Param{Month}}->{$Param{Day}};
    }
    elsif (defined($TimeVacationDaysOneTime->{$Param{Year}}->{$Param{Month}}->{$Param{Day}})) {
        return $TimeVacationDaysOneTime->{$Param{Year}}->{$Param{Month}}->{$Param{Day}};
    }
    elsif (defined($TimeVacationDays->{int($Param{Month})}->{int($Param{Day})})) {
        return $TimeVacationDays->{int($Param{Month})}->{int($Param{Day})};
    }
    elsif (defined($TimeVacationDaysOneTime->{$Param{Year}}->{int($Param{Month})}->{int($Param{Day})})) {
        return $TimeVacationDaysOneTime->{$Param{Year}}->{int($Param{Month})}->{int($Param{Day})};
    }
    return;
}

=item ProjectActionReporting()

returns a hash with the hours dependent project and action data

    my %ProjectData = $TimeAccountingObject->ProjectActionReporting(
        Year  => 2005,
        Month => 7,
        UserID => 123,
    );

=cut

sub ProjectActionReporting {
    my $Self     = shift;
    my %Param    = @_;
    my %Data     = ();
    my $IDSelect = '';
    foreach (qw(Year Month)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "ProjectActionReporting: Need $_!");
            return;
        }
    }
    if ($Param{UserID}) {
        $Param{UserID} = $Self->{DBObject}->Quote($Param{UserID}) || '';
        $IDSelect = " AND user_id = '$Param{UserID}'";
    }
    # hours per month
    my $DateString = $Param{Year} . "-" . sprintf("%02d", $Param{Month});

    my $SQL_Query_TimeStart = "time_start <= '$DateString-31 23:59:59'$IDSelect";

    #tto: yes, I know there are some non-leap-years, ...
    #...but com'on these few entries in the error log won't bother too much... :-)
    if( ($Param{Month} == 2) || ($Param{Month} == 4) || ($Param{Month} == 6) || ($Param{Month} == 9) || ($Param{Month} == 11)){
        $SQL_Query_TimeStart = "time_start <= '$DateString-30 23:59:59'$IDSelect";
    }

    # Total hours
    $Self->{DBObject}->Prepare (
        SQL => "SELECT project_id, action_id, period FROM time_accounting_table WHERE $SQL_Query_TimeStart",
    );
    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $Data{Total}{$Row[0]}{$Row[1]}{Hours} += $Row[2];
    }

    $Self->{DBObject}->Prepare (
        SQL => "SELECT project_id, action_id, period FROM time_accounting_table" .
            " WHERE time_start >= '$DateString-01 00:00:00' AND $SQL_Query_TimeStart",
    );

    # fetch Data
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $Data{PerMonth}{$Row[0]}{$Row[1]}{Hours} += $Row[2];
    }
    return %Data;

}

1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (http://otrs.org/).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see http://www.gnu.org/licenses/gpl.txt.

=head1 VERSION

$Revision: 1.10 $ $Date: 2007/06/05 14:19:49 $

=cut

IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9BZ2VudFRpbWVBY2NvdW50aW5nLnBtIC0gdGhlIGRlIGxhbmd1YWdlIGZvciBBZ2VudFRpbWVBY2NvdW50aW5nCiMgQ29weXJpZ2h0IChDKSAyMDAzLTIwMDcgT1RSUyBHbWJILCBodHRwOi8vb3Rycy5jb20vCiMgLS0KIyAkSWQ6IGRlX0FnZW50VGltZUFjY291bnRpbmcucG0sdiAxLjQgMjAwNy8wNi8wNSAxNDoxMzo0NCB0ciBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6ZGVfQWdlbnRUaW1lQWNjb3VudGluZzsKCnVzZSBzdHJpY3Q7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICRTZWxmLT57VHJhbnNsYXRpb259ID0geyAleyRTZWxmLT57VHJhbnNsYXRpb259fSwKICAgICMgVGVtcGxhdGU6IEFnZW50VGltZUFjY291bnRpbmcKICAgICAgICAnU2V0dGluZycgICAgICAgID0+ICdLb25maWd1cmF0aW9uJywKICAgICAgICAnUHJvamVjdFNldHRpbmcnID0+ICdQcm9qZWt0a29uZmlndXJhdGlvbicsCiAgICAgICAgI25ldyB0cmFuc2xhdGlvbnMuLi4KICAgICAgICAnRGF0ZScgICAgICAgID0+ICdEYXR1bScsCiAgICAgICAgJ3VudGlsJyA9PiAnYmlzJywKICAgICAgICAnV29ya2luZ0hvdXJzJyAgICA9PiAnQXJiZWl0c3N0dW5kZW4nLAogICAgICAgICdXb3JraW5nIEhvdXJzJyAgICA9PiAnQXJiZWl0c3N0dW5kZW4nLAogICAgICAgICdXZWVrbHkgSG91cnMnID0+ICdXb2NoZW5zdHVuZGVuJywKICAgICAgICAndGhpcyBtb250aCcgICAgPT4gJ2RpZXNlciBNb25hdCcsCiAgICAgICAgJ092ZXJ0aW1lIHRvdGFsJyAgICA9PiAnU3VtbWUg3GJlcnN0dW5kZW4nLAogICAgICAgICdMZWF2ZSBEYXlzJyAgICA9PiAnVGFnZSBhYndlc2VuZCcsCiAgICAgICAgJ0xlYXZlRGF5IHRoaXMgbW9udGgnICAgID0+ICdVcmxhdWJzdGFnZSAoZGllc2VyIE1vbmF0KScsCiAgICAgICAgJ0xlYXZlRGF5VG90YWwnICAgID0+ICdVcmxhdWJzdGFnZSAoU3VtbWUpJywKICAgICAgICAnTGVhdmVEYXkgcmVtYWluaW5nJyAgICA9PiAnVXJsYXVic3RhZ2UgKHZlcmJsZWliZW5kKScsCiAgICAgICAgJ0Rpc2Vhc2VkIHRoaXMgbW9udGgnICAgID0+ICdFcmtyYW5rdCAoZGllc2VyIE1vbmF0KScsCiAgICAgICAgJ0Rpc2Vhc2VkVG90YWwnICAgID0+ICdFcmtyYW5rdCAoU3VtbWUpJywKICAgICAgICAnVGltZUFjY291bnRpbmcnID0+ICdaZWl0ZXJmYXNzdW5nJywKICAgICAgICAnVGltZSBBY2NvdW50aW5nIE92ZXJ2aWV3IHBlciBNb250aCcgPT4gJ01vbmF0c/xiZXJzaWNodCBaZWl0ZXJmYXNzdW5nJywKICAgICAgICAnRWRpdCBUaW1lIEFjY291bnRpbmcnID0+ICdaZWl0ZXJmYXNzdW5nIGJlYXJiZWl0ZW4nLAogICAgICAgICdFZGl0IFRpbWUgQWNjb3VudGluZyBQcmVmZXJlbmNlJyA9PiAnWmVpdGVyZmFzc3VuZ3NlaW5zdGVsbHVuZ2VuIGJlYXJiZWl0ZW4nLAogICAgICAgICdVc2VyIFJlcG9ydGluZycgPT4gJ051dHplcmJlcmljaHRlJywKICAgICAgICAnUHJvamVjdCBSZXBvcnRpbmcnID0+ICdQcm9qZWt0YmVyaWNodGUnLAogICAgICAgICdUaW1lIFByZXBvcnRpbmcnID0+ICdaZWl0YmVyaWNodGUnLAogICAgICAgICdMZWF2ZURheSBSZW1haW5pbmcnID0+ICdWZXJibGVpYmVuZGUgVXJsYXVic3RhZ2UnLAogICAgICAgICdwZXIgTW9udGgnID0+ICdwcm8gTW9uYXQnLAogICAgICAgICdIb3VycycgPT4gJ1N0dW5kZW4nLAogICAgICAgICdUaW1lIHNlbGVjdGlvbicgPT4gJ0F1c3dhaGwgRGF0dW0nLAogICAgICAgICdJbmNvbXBsZXRlIERheXMnID0+ICduaWNodCBhdXNnZWb8bGx0ZSBUYWdlJywKICAgICAgICAnUHJvamVjdCcgPT4gJ1Byb2pla3QnLAogICAgICAgICdQcm9qZWN0cycgPT4gJ1Byb2pla3RlJywKICAgICAgICAnVG90YWwnID0+ICdTdW1tZScsCiAgICAgICAgJ1JlcG9ydGluZycgPT4gJ0JlcmljaHRzd2VzZW4nLAogICAgICAgICdBY3Rpb24gU2V0dGluZycgPT4gJ0FrdGlvbnNlaW5zdGVsbHVuZ2VuJywKICAgICAgICAnVXNlciBTZXR0aW5nJyA9PiAnTnV0emVyZWluc3RlbGx1bmdlbicsCiAgICAgICAgJ1Nob3cgT3ZlcnRpbWUnID0+ICfcYmVyc3R1bmRlbiBhbnplaWdlbicsCiAgICAgICAgJ0NyZWF0ZSBQcm9qZWN0JyA9PiAnUHJvamVrdCBlcnN0ZWxsZW4nLAogICAgICAgICdOZXcgdXNlciBzZXR0aW5nJyA9PiAnTmV1ZSBOdXR6ZXJlaW5zdGVsbHVuZycsCiAgICAgICAgJ1JlbWFyaycgPT4gJ0FubWVya3VuZycsCiAgICAgICAgJ1N0YXJ0dGltZScgPT4gJ0JlZ2lubicsCiAgICAgICAgJ0VuZHRpbWUnID0+ICdFbmRlJywKICAgICAgICAnU3RhcnRkYXRlJyA9PiAnRGF0dW0gQmVnaW5uJywKICAgICAgICAnRW5kZGF0ZScgPT4gJ0RhdHVtIEVuZGUnLAogICAgICAgICdQZXJpb2QnID0+ICdEYXVlcicsCiAgICAgICAgJ0xlYXZlIERheScgPT4gJ1VybGF1YnN0YWcnLAogICAgICAgICdMZWF2ZURheScgPT4gJ1VybGF1YicsCiAgICAgICAgJ0Rpc2Vhc2VkJyA9PiAnRXJrcmFua3QnLAogICAgICAgICdPdmVydGltZScgPT4gJ9xiZXJzdHVuZGVuJywKICAgICAgICAnVG90YWwnID0+ICdTdW1tZScsCiAgICAgICAgJ092ZXJ2aWV3IG9mICcgPT4gJ9xiZXJzaWNodCAtICcsCiAgICAgICAgJ1RpbWVBY2NvdW50aW5nIG9mJyA9PiAnWmVpdGVyZmFzc3VuZyB2b20nLAogICAgICAgICdTdWNjZXNzZnVsIGluc2VydCEnID0+ICdFcmZvbGdyZWljaCBlaW5nZWb8Z3QhJywKICAgICAgICAnRG8geW91IHJlYWxseSB3YW50IHRvIGRlbGV0ZSB0aGlzIE9iamVjdCcgPT4gJ1dvbGxlbiBTaWUgZGllc2VuIEVpbnRyYWcgd2lya2xpY2ggbPZzY2hlbicsCiAgICAgICAgJ0NhblwndCBpbnNlcnQgV29ya2luZyBVbml0cyEnID0+ICdLYW5uIEFyYmVpdHNzdHVuZGVuIG5pY2h0IGVpbmb8Z2VuIScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIG9mIG1pc3NpbmcgQWN0aW9uIScgPT4gJ05pY2h0IHNwZWljaGVyYmFyIC0gQWt0aW9uIGZlaGx0IScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIG9mIG1pc3NpbmcgUHJvamVjdCEnID0+ICdOaWNodCBzcGVpY2hlcmJhciAtIFByb2pla3RhbmdhYmUgZmVobHQhJywKICAgICAgICAnQ2FuXCd0IHNhdmUgc2V0dGluZ3MsIGJlY2F1c2UgdGhlIFBlcmlvZCBpcyBiaWdnZXIgdGhhbiB0aGUgaW50ZXJ2YWwgYmV0d2VlbiBTdGFydHRpbWUgYW5kIEVuZHRpbWUhJyA9PiAnTmljaHQgc3BlaWNoZXJiYXIgLSBEYXVlciBpc3QgZ3L232VyIGFscyBkZXIgWmVpdHJhdW0gendpc2NoZW4gQmVnaW5uIHVuZCBFbmRlIScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIFN0YXJ0dGltZSBpcyBvbGRlciB0aGFuIEVuZHRpbWUhJyA9PiAnTmljaHQgc3BlaWNoZXJiYXIgLSBCZWdpbm4gbGllZ3QgbmFjaCBFbmRlIScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIG9mIG1pc3NpbmcgcGVyaW9kIScgPT4gJ05pY2h0IHNwZWljaGVyYmFyIC0gRGF1ZXIgKGVyZ2lidCBzaWNoIGF1cyBTdGFydC0gdW5kIEVuZHplaXQpIGlzdCBuaWNodCBhbmdlZ2ViZW4hJywKICAgICAgICAnQ2FuXCd0IHNhdmUgc2V0dGluZ3MsIGJlY2F1c2UgUGVyaW9kIGlzIG5vdCBnaXZlbiBnaXZlbiEnID0+ICdOaWNodCBzcGVpY2hlcmJhciAtIERhdWVyIChlcmdpYnQgc2ljaCBhdXMgU3RhcnQtIHVuZCBFbmR6ZWl0KSBpc3QgbmljaHQgYW5nZWdlYmVuIScsCiAgICAgICAgJ0FyZSB5b3Ugc3VyZSwgdGhhdCB5b3Ugd29ya2VkIHdoaWxlIHlvdSBhcmUgRGlzZWFzZWQ/JyA9PiAnU2llIHdhcmVuIGtyYW5rIHVuZCBoYWJlbiBnZWFyYmVpdGV0PyBXaXIgYnJhdWNoZW4gbWVociBzb2xjaGUgTWl0YXJiZWl0ZXIuJywKICAgICAgICAnQXJlIHlvdSBzdXJlLCB0aGF0IHlvdSB3b3JrZWQgd2hpbGUgeW91IGhhdmUgYSBsZWF2ZSBkYXk/JyA9PiAnU2llIGhhdHRlbiBVcmxhdWIgdW5kIGhhYmVuIGdlYXJiZWl0ZXQ/IFdpciBicmF1Y2hlbiBtZWhyIHNvbGNoZSBNaXRhcmJlaXRlci4nLAogICAgICAgICdBcmUgeW91IHN1cmUsIHRoYXQgeW91IHdvcmtlZCB3aGlsZSB5b3UgaGF2ZSBvdmVydGltZT8nID0+ICdIYWJlbiBTaWUgd+RocmVuZCBkZXIg3GJlcnN0dW5kZW4gYXVjaCBnZWFyYmVpdGV0PycsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIGEgZGF5IGhhcyBvbmx5IDI0IGhvdXJzIScgPT4gJ0VpbiBUYWcgaGF0IG51ciAyNCBTdHVuZGVuIScsCiAgICAgICAgJ0NhblwndCBkZWxldGUgV29ya2luZyBVbml0cyEnID0+ICdLYW5uIEFyYmVpdHNzdHVuZGVuIG5pY2h0IGz2c2NoZW4hJywKICAgICAgICAnUGxlYXNlIGluc2VydCB5b3VyIHdvcmtpbmcgaG91cnMhJyA9PiAnQml0dGUgdHJhZ2VuIFNpZSBJaHJlIEFyYmVpdHN6ZWl0ZW4gZWluIScsCiAgICAgICAgJ1lvdSBoYXZlIHRvIGluc2VydCBTdGFydC0gYW5kIEVuZHRpbWUgb3IgYSBQZXJpb2QnID0+ICdTaWUgbfxzc2VuIEJlZ2lubi0gdW5kIEVuZGV6ZWl0IGFuZ2ViZW4uJywKICAgICAgICAnWW91IGNhbiBvbmx5IHNlbGVjdCBvbmUgY2hlY2tib3ggZWxlbWVudCEnID0+ICdTaWUga/ZubmVuIG51ciBlaW5lIENoZWNrYm94IG1hcmtpZXJlbiEnLAogICAgICAgICdJZiB5b3UgdXNlIGFjdGlvbiBtaXNjLCBwbGVhc2UgZ2l2ZSBhbiBleHBsYW5hdGlvbiBpbiB0aGUgZmllbGQgcmVtYXJrJyA9PiAnV2VubiBTaWUgQWt0aW9uIFNvbnN0aWdlcyBhdXN35GhsZW4sIGdlYmVuIFNpZSBlaW5lIEJlc2NocmVpYnVuZyB1bSBGZWxkIEFubWVya3VuZyBhbi4nLAogICAgICAgICNhY3R1YWxseSB0aGUgZm9sbG93aW5nIHNob3VsZCBiZSBpbmNsdWRlZCBpbiBmaWxlIGRlLnBtLCBob3dldmVyIHRoZXkncmUgbm90IHNvIEkgcHV0J2VtIGhlcmUuLi4KICAgICAgICAnTW9uJyA9PiAnTW8nLAogICAgICAgICdUdWUnID0+ICdEaScsCiAgICAgICAgJ1dlZCcgPT4gJ01pJywKICAgICAgICAnVGh1JyA9PiAnRG8nLAogICAgICAgICdGcmknID0+ICdGcicsCiAgICAgICAgJ1NhdCcgPT4gJ1NhJywKICAgICAgICAnU3VuJyA9PiAnU28nLAogICAgICAgICdKYW51YXJ5JyA9PiAnSmFudWFyJywKICAgICAgICAnRmVicnVhcnknID0+ICdGZWJydWFyJywKICAgICAgICAnTWFyY2gnID0+ICdN5HJ6JywKICAgICAgICAnTWF5JyA9PiAnTWFpJywKICAgICAgICAnSnVuZScgPT4gJ0p1bmknLAogICAgICAgICdKdWx5JyA9PiAnSmFudWFyJywKICAgICAgICAnT2N0b2JlcicgPT4gJ09rdG9iZXInLAogICAgICAgICdEZWNlbWJlcicgPT4gJ0RlemVtYmVyJywKICAgIH07Cn0KCjE7Cg==
# --
# Kernel/Modules/AgentTimeAccounting.pm - time accounting module
# Copyright (C) 2003-2007 OTRS GmbH, http://otrs.com/
# --
# $Id: AgentTimeAccounting.pm,v 1.14 2007/06/05 14:15:48 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl.txt.
# --

package Kernel::Modules::AgentTimeAccounting;

use strict;
use Kernel::System::TimeAccounting;
use Date::Pcalc qw(Today Days_in_Month Day_of_Week Add_Delta_YMD);
use Time::Local;

use vars qw($VERSION);
$VERSION = '$Revision: 1.14 $';
$VERSION =~ s/^\$.*:\W(.*)\W.+?$/$1/;

sub new {
    my $Type = shift;
    my %Param = @_;

    # allocate new hash for object
    my $Self = {};
    bless ($Self, $Type);

    # get common objects
    foreach (keys %Param) {
        $Self->{$_} = $Param{$_};
    }

    # check needed Opjects
    foreach (qw(ParamObject DBObject ModuleReg LogObject
        ConfigObject TicketObject TimeObject GroupObject)
    ) {
        $Self->{LayoutObject}->FatalError(Message => "Got no $_!") if (!$Self->{$_});
    }

    # create required objects...
    $Self->{TimeAccountingObject} = Kernel::System::TimeAccounting->new(%Param);
    return $Self;
}

sub PreRun {
    my $Self = shift;
    my %Param = @_;
    my ($Sec, $Min, $Hour, $Day, $Month, $Year) = $Self->{TimeObject}->SystemTime2Date(
        SystemTime => $Self->{TimeObject}->SystemTime(),
    );
    my %User = $Self->{TimeAccountingObject}->UserCurrentPeriodGet(
        Year  => $Year,
        Month => $Month,
        Day   => $Day,
    );
    if ($User{$Self->{UserID}}){
        my %IncompleteWorkingDays = $Self->{TimeAccountingObject}->WorkingUnitsCompletnessCheck();
        # redirect if incomplete working day are out of range
        if ($IncompleteWorkingDays{EnforceInsert}
            && $Self->{Action} ne 'AgentTimeAccounting'
            && $Self->{Action} ne 'AgentCalendarSmall')
        {
            return $Self->{LayoutObject}->Redirect(OP => "Action=AgentTimeAccounting&Subaction=Edit");
        }
    }
    return;
}

sub Run {
    my $Self  = shift;
    my %Param = @_;
    my @MonthArray = (
        '',
        'January',
        'February',
        'March',
        'April',
        'May',
        'June',
        'July',
        'August',
        'September',
        'October',
        'November',
        'December',
    );
    my @WeekdayArray = (
        'Mon',
        'Tue',
        'Wed',
        'Thu',
        'Fri',
        'Sat',
        'Sun',
    );

    # ---------------------------------------------------------- #
    # edit the time accounting elements
    # ---------------------------------------------------------- #
    if ($Self->{Subaction} eq 'Edit') {
        my $Output       = '';
        my %Frontend     = ();
        my %ProjectList  = ();
        my %ActionList   = ();
        my %Data         = ();
        my %Action       = $Self->{TimeAccountingObject}->ActionSettingsGet();
        my ($Sec, $Min, $Hour, $Day, $Month, $Year) = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
        );

        # get params
        foreach (qw(Status Year Month Day Delete)) {
            $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }
        # permission check
        if (!$Self->{AccessRo}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        # Check Date
        if (!$Param{Year} || !$Param{Month} || !$Param{Day}) {
            $Param{Year}  = $Year;
            $Param{Month} = $Month;
            $Param{Day}   = $Day;
        }
        else {
            $Param{Year}  = sprintf("%02d",$Param{Year});
            $Param{Month} = sprintf("%02d",$Param{Month});
            $Param{Day}   = sprintf("%02d",$Param{Day});
        }

        my %User = $Self->{TimeAccountingObject}->UserCurrentPeriodGet(
            Year  => $Param{Year},
            Month => $Param{Month},
            Day   => $Param{Day},
        );

        # for inital useing, the first agent with rw-right will be redirected
        # to 'Setting'. Then he can do the initial settings
        if (!$User{$Self->{UserID}}){
            my %GroupList = $Self->{GroupObject}->GroupMemberList(
                UserID => $Self->{UserID},
                Type   => 'rw',
                Result => 'HASH',
            );

            foreach ((keys %GroupList)) {
                if ($GroupList{$_} eq 'time_accounting') {
                    return $Self->{LayoutObject}->Redirect(OP => "Action=AgentTimeAccounting&" .
                        "Subaction=Setting"
                    );
                }
            }
            return $Self->{LayoutObject}->ErrorScreen(
                Message => "No UserPeriod available, please contact the time accounting admin to insert your UserPeriod!"
            );
        }

        my %IncompleteWorkingDays = $Self->{TimeAccountingObject}->WorkingUnitsCompletnessCheck();
        my $MaxAllowedInsertDays  = $Self->{ConfigObject}->Get('TimeAccounting::MaxAllowedInsertDays') || '10';
        ($Param{YearAllowed}, $Param{MonthAllowed}, $Param{DayAllowed}) = Add_Delta_YMD($Year, $Month, $Day, 0, 0, -$MaxAllowedInsertDays);
        if (timelocal(1, 0, 0 , $Param{Day}, $Param{Month} - 1, $Param{Year} - 1900) <
            timelocal(1, 0, 0 , $Param{DayAllowed}, $Param{MonthAllowed} - 1, $Param{YearAllowed} - 1900)
        ) {
            if (!$IncompleteWorkingDays{Incomplete}{$Param{Year}}{$Param{Month}}{$Param{Day}}) {
                return $Self->{LayoutObject}->Redirect(OP => "Action=$Self->{Action}&Subaction=View&Year=$Param{Year}&Month=$Param{Month}&Day=$Param{Day}");
            }
        }

        # store last screen
        $Self->{SessionObject}->UpdateSessionID(
            SessionID => $Self->{SessionID},
            Key       => 'LastScreen',
            Value     => "Action=$Self->{Action}&Subaction=Edit&Year=$Param{Year}&Month=$Param{Month}&Day=$Param{Day}",
        );

        $Param{Month_to_Text}   = $MonthArray[$Param{Month}];

        ($Param{YearBack}, $Param{MonthBack}, $Param{DayBack}) = Add_Delta_YMD($Param{Year}, $Param{Month}, $Param{Day}, 0, 0, -1);
        ($Param{YearNext}, $Param{MonthNext}, $Param{DayNext}) = Add_Delta_YMD($Param{Year}, $Param{Month}, $Param{Day}, 0, 0, 1);

    # Edit Working Units
        if ($Param{Status}) {
            if ($Param{Delete}) {
                $Output .= $Self->{LayoutObject}->Header(Title => "Delete");
                $Output .= $Self->{LayoutObject}->NavigationBar();
                $Output .= $Self->{LayoutObject}->Output(Data => {%Param, %Frontend}, TemplateFile => 'AgentTimeAccountingDelete');
                $Output .= $Self->{LayoutObject}->Footer();
                return $Output;
            }
            my $WorkingUnitID = 0;
            for ($WorkingUnitID = 1; $WorkingUnitID < 10; $WorkingUnitID++) {
                foreach (qw(ProjectID ActionID Remark StartTime EndTime Period)) {
                    $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_ . '[' . $WorkingUnitID . ']');
                }
                if ($Param{ProjectID} || $Param{ActionID}) {
                    $Data{$WorkingUnitID}{ProjectID} = $Param{ProjectID};
                    $Data{$WorkingUnitID}{ActionID}  = $Param{ActionID};
                    $Data{$WorkingUnitID}{Remark}    = $Param{Remark};
                    $Data{$WorkingUnitID}{StartTime} = $Param{StartTime};
                    $Data{$WorkingUnitID}{EndTime}   = $Param{EndTime};
                    $Data{$WorkingUnitID}{Date}      = $Param{Date};
                    if ($Param{Period} =~ /^(\d+),(\d+)/) {
                        $Data{$WorkingUnitID}{Period}    = $1 . "." . $2;
                    }
                    else {
                        $Data{$WorkingUnitID}{Period} = $Param{Period};
                    }

                    my %ReduceTime = ();
                    if ($Self->{ConfigObject}->Get('TimeAccounting::ReduceTime')) {
                        %ReduceTime = %{$Self->{ConfigObject}->Get('TimeAccounting::ReduceTime')};
                    }
                    #if ($Param{StartTime} && $Param{EndTime} && !$Param{Period}) {
                    #overwrite Period when Start and Endtime is given...
                    if ($Param{StartTime} && $Param{EndTime}) {
                        if ($Param{StartTime} =~ /^(\d+):(\d+)/) {
                            my $StartTime = $1*60 + $2;
                            if ($Param{EndTime} =~ /^(\d+):(\d+)/) {
                                my $EndTime = $1*60 + $2;
                                if ($ReduceTime{$Action{$Param{ActionID}}{Action}}) {
                                    $Data{$WorkingUnitID}{Period} = ($EndTime - $StartTime)/60 * $ReduceTime{$Action{$Param{ActionID}}{Action}} / 100;
                                }
                                else {
                                    $Data{$WorkingUnitID}{Period} = ($EndTime - $StartTime)/60;
                                }
                            }
                        }
                    }
                }
            }
            my $CheckboxCheck = 0;
            foreach (qw(LeaveDay Diseased Overtime)) {
                $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_);
                if ($Param{$_}) {
                    $WorkingUnitID++;
                    $Data{$WorkingUnitID}{ProjectID} = '-1';
                    $Data{$WorkingUnitID}{ActionID}  = $Param{$_};
                    $Data{$WorkingUnitID}{Remark}    = '';
                    $Data{$WorkingUnitID}{StartTime} = '';
                    $Data{$WorkingUnitID}{EndTime}   = '';
                    $Data{$WorkingUnitID}{Period}    = '';
                    $CheckboxCheck++;
                }
            }
            if ($CheckboxCheck > 1) {
                $Param{RequiredDescription} = "You can only select one checkbox element!";
            }
            $Data{Year}  = $Param{Year};
            $Data{Month} = $Param{Month};
            $Data{Day}   = $Param{Day};
            if (!$Self->{TimeAccountingObject}->WorkingUnitsInsert(%Data)) {
                return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t insert Working Units!');
            }
            else {
                $Param{SuccessfulInsert} = 1;
            }
        }

        # Show Working Units
        # get existing working units
        %Data    = $Self->{TimeAccountingObject}->WorkingUnitsGet(
            Year  => $Param{Year},
            Month => $Param{Month},
            Day   => $Param{Day},
        );

        if ($Self->{ConfigObject}->Get('TimeAccounting::InputHoursWithoutStartEndTime')) {
            $Param{TextPosition}  = 'left';
            $Param{PeriodBlock}   = 'UnitInputPeriod';
            $Frontend{ClassTime}  = 'footnote';
            $Frontend{PeriodNote} = '*';
            $Self->{LayoutObject}->Block(
                Name => 'FootNote',
                Data => {%Param, %Frontend},
            );
        }
        else {
            $Param{TextPosition}  = 'right';
            $Param{PeriodBlock}   = 'UnitPeriodWithoutInput';
            $Frontend{ClassTime}  = 'required';
            $Frontend{PeriodNote} = '';
        }

        # get project settings
        my %Project = $Self->{TimeAccountingObject}->ProjectSettingsGet();
        foreach my $ProjectID (keys %{$Project{Project}}) {
            if ($Project{ProjectStatus}{$ProjectID}) {
                $ProjectList{$ProjectID} = $Project{Project}{$ProjectID};
            }
        }
        $ProjectList{''} = '';

        # get action settings
        foreach my $ActionID (keys %Action) {
            if ($Action{$ActionID}{ActionStatus}) {
                $ActionList{$ActionID} = $Action{$ActionID}{Action};
            }
        }
        $ActionList{''} = '';

        if (time() > timelocal(1, 0, 0 , $Param{Day}, $Param{Month} - 1, $Param{Year} - 1900)) {
            $Self->{LayoutObject}->Block(
                Name => 'UnitBlock',
                Data => {%Param, %Frontend},
            );
        }
        # build units
        my $WorkingUnitIDMax = 10;
        my $WorkingUnitID    = 0;
        $Param{Total}        = 0;
        for ($WorkingUnitID = 1; $WorkingUnitID < $WorkingUnitIDMax; $WorkingUnitID++) {
            if ($Data{$WorkingUnitID}{ProjectID} && $Data{$WorkingUnitID}{ProjectID} == -1) {
                if ($Data{$WorkingUnitID}{ActionID}    == -1) {
                    $Param{Diseased} = 'checked';
                }
                elsif ($Data{$WorkingUnitID}{ActionID} == -2) {
                    $Param{LeaveDay} = 'checked';
                }
                elsif ($Data{$WorkingUnitID}{ActionID} == -3) {
                    $Param{Overtime} = 'checked';
                }
                $WorkingUnitIDMax++;
            }
            else {
                $Param{ID} = $WorkingUnitID ;
                # get option fields
                $Frontend{'ActionOption'} = $Self->{LayoutObject}->OptionStrgHashRef(
                    Data       => \%ActionList,
                    SelectedID => $Data{$WorkingUnitID}{ActionID} || '',
                    Name       => "ActionID[$WorkingUnitID]",
                );
                $Frontend{'ProjectOption'} = $Self->{LayoutObject}->OptionStrgHashRef(
                    Data       => \%ProjectList,
                    SelectedID => $Data{$WorkingUnitID}{ProjectID} || '',
                    Name       => "ProjectID[$WorkingUnitID]",
                );

                $Param{Remark}    = $Data{$WorkingUnitID}{Remark} || '';
                if ($Data{$WorkingUnitID}{ProjectID} && $Data{$WorkingUnitID}{ActionID}) {
                    if ($Data{$WorkingUnitID}{Period}) {
                        if ($Data{$WorkingUnitID}{Period} > 0) {
                            $Param{Period}    = $Data{$WorkingUnitID}{Period};
                            $Param{Total}    += $Param{Period};
                        }
                        elsif( $Data{$WorkingUnitID}{Period} == 0) {
                            $Param{UnitRequiredDescription} = 'Can\'t save settings, because Period is not given given!';
                        }

                        else {
                            $Param{UnitRequiredDescription} = 'Can\'t save settings, because Starttime is older than Endtime!';
                        }
                    }
                    else {
                        $Param{Period} = '';
                        $Param{UnitRequiredDescription} = 'Can\'t save settings, because of missing period!';
                    }
                }
                else {
                    $Param{Period}    = '';
                    $Param{StartTime} = '';
                    $Param{EndTime}   = '';
                }

#                foreach (qw(Start End)) {
#                    if ($Data{$WorkingUnitID}{$_ . "Time"} && $Data{$WorkingUnitID}{$_ . "Time"} eq '00:00:00') {
#                        $Param{$_ . "Hour"} = '00';
#                        $Param{$_ . "Minute"} = '00';
#                    }
#                    elsif ($Data{$WorkingUnitID}{$_ . "Time"}) {
#                        $Param{$_. "Hour"}   = substr($Data{$WorkingUnitID}{$_}, 0, 2) || '';
#                        $Param{$_. "Minute"} = substr($Data{$WorkingUnitID}{$_}, 3, 2) || '';
#                    }
#                    $Param{$_ . "Time"} = BuildTimeSelection(
#                        Hour   => $Param{$_. "Hour"},
#                        Minute => $Param{$_. "Minute"}
#                        Prefix => $_,
#                        Format => 'TimeInputFormat',
#                    );
#                }

                foreach (qw(StartTime EndTime)) {
                    if ($Data{$WorkingUnitID}{$_} && $Data{$WorkingUnitID}{$_} eq '00:00') {
                        $Param{$_} = '';
                    }
                    else {
                        $Param{$_} = $Data{$WorkingUnitID}{$_};
                    }
                }

                $Self->{LayoutObject}->Block(
                    Name => 'Unit',
                    Data => {%Param, %Frontend},
                );

                $Self->{LayoutObject}->Block(
                    Name => $Param{PeriodBlock},
                    Data => {%Param, %Frontend},
                );

                # Validity checks start
                if ($Data{$WorkingUnitID}{ProjectID} && $Data{$WorkingUnitID}{ActionID} && $Param{Diseased}) {
                    $Param{ReadOnlyDescription} = 'Are you sure, that you worked while you are Diseased?';
                }
                elsif ($Data{$WorkingUnitID}{ProjectID} && $Data{$WorkingUnitID}{ActionID} && $Param{LeaveDay}) {
                    $Param{ReadOnlyDescription} = 'Are you sure, that you worked while you have a leave day?';
                }
                elsif ($Data{$WorkingUnitID}{ProjectID} && $Data{$WorkingUnitID}{ActionID} && $Param{Overtime}) {
                    $Param{ReadOnlyDescription} = 'Are you sure, that you worked while you have overtime?';
                }
                if ($Data{$WorkingUnitID}{ProjectID} && !$Data{$WorkingUnitID}{ActionID}) {
                    $Param{UnitRequiredDescription} = 'Can\'t save settings, because of missing Action!';
                }
                if (!$Data{$WorkingUnitID}{ProjectID} && $Data{$WorkingUnitID}{ActionID}) {
                    $Param{UnitRequiredDescription} = 'Can\'t save settings, because of missing Project!';
                }
                if ($Data{$WorkingUnitID}{StartTime} && $Data{$WorkingUnitID}{StartTime} ne '00:00' &&
                    $Data{$WorkingUnitID}{EndTime} && $Data{$WorkingUnitID}{EndTime}  ne '00:00'
                ) {
                    if ($Data{$WorkingUnitID}{StartTime} =~ /^(\d+):(\d+)/) {
                        my $StartTime = $1*60 + $2;
                        if ($Data{$WorkingUnitID}{EndTime} =~ /^(\d+):(\d+)/) {
                            my $EndTime = $1*60 + $2;
                            if ($Data{$WorkingUnitID}{Period} > ($EndTime - $StartTime)/60 + 0.01) {
                                $Param{UnitRequiredDescription} = 'Can\'t save settings, because the Period is bigger' .
                                    ' than the interval between Starttime and Endtime!';
                            }
                            if ($EndTime > 60 * 24 || $StartTime > 60 * 24) {
                                $Param{UnitRequiredDescription} = 'Can\'t save settings, because a day has only 24 hours!';
                            }
                        }
                    }
                }

                if ($Param{UnitRequiredDescription}){
                    $Self->{LayoutObject}->Block(
                        Name => 'UnitRequired',
                        Data => {Description => $Param{UnitRequiredDescription}},
                    );
                    if (!$Self->{TimeAccountingObject}->WorkingUnitsDelete(
                        Year  => $Param{Year},
                        Month => $Param{Month},
                        Day   => $Param{Day},
                    )) {
                        return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t delete Working Units!');
                    }
                    $Param{UnitRequiredDescription}     = '';
                    $Param{UnitRequiredDescriptionTrue} = 1;
                }
                if ($Param{UnitReadOnlyDescription}){
                    $Self->{LayoutObject}->Block(
                        Name => 'UnitReadonly',
                        Data => {Description => $Param{UnitReadOnlyDescription}},
                    );
                    $Param{UnitReadOnlyDescription} = '';
                }
                # Validity checks end
            }
        }

        if ($Self->{TimeObject}->SystemTime() > timelocal(1, 0, 0 , $Param{Day}, $Param{Month} - 1, $Param{Year} - 1900)) {
            $Param{Total}           = sprintf ("%.2f", $Param{Total});
            $Self->{LayoutObject}->Block(
                Name => 'Total',
                Data => {%Param, %Frontend},
            );
        }
        # validity checks start
        if ($Param{Total} && $Param{Total} > 24) {
            $Param{RequiredDescription} = 'Can\'t save settings, because of more than 24 working hours!';
        }
        elsif ($Param{Total} && $Param{Total} > 16) {
            $Param{ReadOnlyDescription} = 'Are you sure, that you worked more than 16 hours?';
        }
        if ($Param{RequiredDescription}){
            $Self->{LayoutObject}->Block(
                Name => 'Required',
                Data => {Description => $Param{RequiredDescription}},
            );
            if (!$Self->{TimeAccountingObject}->WorkingUnitsDelete(
                Year  => $Param{Year},
                Month => $Param{Month},
                Day   => $Param{Day},
            )) {
                return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t delete Working Units!');
            }
        }
        if ($Param{ReadOnlyDescription}){
            $Self->{LayoutObject}->Block(
                Name => 'Readonly',
                Data => {Description => $Param{ReadOnlyDescription}},
            );
        }

        # validity checks end

        $Param{Date} = $Self->{LayoutObject}->BuildDateSelection(
            %Param,
            Prefix => '',
            Format => 'DateInputFormat',
        );

        if (timelocal(1, 0, 0 , $Param{Day}, $Param{Month} - 1, $Param{Year} - 1900) <
            timelocal(1, 0, 0 , $Param{DayAllowed}, $Param{MonthAllowed} - 1, $Param{YearAllowed} - 1900)
        ) {
            if ($IncompleteWorkingDays{Incomplete}{$Param{Year}}{$Param{Month}}{$Param{Day}} && !$Param{SuccessfulInsert}) {
                $Self->{LayoutObject}->Block(
                    Name => 'Required',
                    Data => {Description => 'This Date is out of limit, but you haven\'t insert this day yet, so you get one(!) chance to insert'},
                );
            }
        }

        foreach my $YearID (sort keys %{$IncompleteWorkingDays{Incomplete}}) {
            foreach my $MonthID (sort keys %{$IncompleteWorkingDays{Incomplete}{$YearID}}) {
                foreach my $DayID (sort keys %{$IncompleteWorkingDays{Incomplete}{$YearID}{$MonthID}}) {
                    if (!$Param{Incomplete}) {
                        $Self->{LayoutObject}->Block(
                            Name => 'IncompleteText',
                        );
                    }
                    my $BoldStart = '';
                    my $BoldEnd   = '';
                    if ($YearID eq $Param{Year}
                        && $MonthID eq $Param{Month}
                        && $DayID eq $Param{Day}
                    ) {
                        $BoldStart = '<b>';
                        $BoldEnd   = '</b>';
                    }

                    $Self->{LayoutObject}->Block(
                        Name => 'IncompleteWorkingDays',
                        Data => {
                            Year      => $YearID,
                            Month     => $MonthID,
                            Day       => $DayID,
                            BoldStart => $BoldStart,
                            BoldEnd   => $BoldEnd,
                        },
                    );
                    $Param{Incomplete} = 1;
                }
            }
        }

        my $VacationCheck = $Self->{TimeAccountingObject}->VacationCheck(
            Year  => $Param{Year},
            Month => $Param{Month},
            Day   => $Param{Day},
        );

        $Param{Weekday}   = Day_of_Week($Param{Year}, $Param{Month}, $Param{Day});
        if ($Param{Weekday} != 6 && $Param{Weekday} != 7 && !$VacationCheck) {
            $Self->{LayoutObject}->Block(
                Name => 'OtherTimes',
                Data => {%Param, %Frontend},
            );
        }

        $Param{Weekday_to_Text} = $WeekdayArray[$Param{Weekday} - 1];

        # build output
        $Output = $Self->{LayoutObject}->Header(Title => "Edit");
        if (!$IncompleteWorkingDays{EnforceInsert}) {
            $Output .= $Self->{LayoutObject}->NavigationBar();
            $Self->{LayoutObject}->Block(
                Name => 'Overview',
                Data => {%Param, %Frontend},
            );
            # show create project link, if allowed
            my %UserBasics = $Self->{TimeAccountingObject}->UserGet();
            if ($UserBasics{$Self->{UserID}}{CreateProject}) {
                $Self->{LayoutObject}->Block(
                    Name => 'CreateProject',
                );
            }
        }
        if (!$Param{RequiredDescription} && !$Param{UnitRequiredDescriptionTrue} && $Param{SuccessfulInsert}) {
            $Output .= $Self->{LayoutObject}->Notify(
                Info => 'Successful insert!',
            );
        }
        $Output .= $Self->{LayoutObject}->Output(
            Data         => {%Param, %Frontend},
            TemplateFile => 'AgentTimeAccountingEdit'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # view older day insterts
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'View') {
        # permission check
        if (!$Self->{AccessRo}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        # get params
        foreach (qw(Day Month Year UserID)) {
            $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        # check needed params
        foreach (qw(Day Month Year)) {
            if (!$Param{$_}) {
                return $Self->{LayoutObject}->ErrorScreen(
                    Message => "View: Need $_"
                );
            }
        }

        # if no UserID posted use the current user
        if (!$Param{UserID}) {
            $Param{UserID} = $Self->{UserID};
        }

        # show the naming of the agent which time accounting is visited
        if ($Param{UserID} != $Self->{UserID}) {
            my %ShownUsers = $Self->{UserObject}->UserList(Type => 'Long', Valid => 1);
            $Param{User}   = $ShownUsers{$Param{UserID}};
            $Self->{LayoutObject}->Block(
                Name => 'User',
                Data => {%Param},
            );
        }

        $Param{Weekday}         = Day_of_Week($Param{Year}, $Param{Month}, $Param{Day});
        $Param{Weekday_to_Text} = $WeekdayArray[$Param{Weekday} - 1];
        $Param{Month_to_Text}   = $MonthArray[$Param{Month}];
        # Values for the link icons <>
        ($Param{YearBack}, $Param{MonthBack}, $Param{DayBack}) = Add_Delta_YMD($Param{Year}, $Param{Month}, $Param{Day}, 0, 0, -1);
        ($Param{YearNext}, $Param{MonthNext}, $Param{DayNext}) = Add_Delta_YMD($Param{Year}, $Param{Month}, $Param{Day}, 0, 0, 1);

        $Param{DateSelection} = $Self->{LayoutObject}->BuildDateSelection(
            %Param,
            Prefix => '',
            Format => 'DateInputFormat',
        );

    # Show Working Units
        # get existing working units
        my %Data    = $Self->{TimeAccountingObject}->WorkingUnitsGet(
            Year   => $Param{Year},
            Month  => $Param{Month},
            Day    => $Param{Day},
            UserID => $Param{UserID},
        );

        $Param{Date} = sprintf("%04d-%02d-%02d", $Param{Year}, $Param{Month}, $Param{Day});

        # get project and action settings
        my %Project = $Self->{TimeAccountingObject}->ProjectSettingsGet();
        my %Action  = $Self->{TimeAccountingObject}->ActionSettingsGet();
        my $Flag = 0;

        foreach my $ID (keys %Data) {
            if ($Data{$ID}{ProjectID} && $Data{$ID}{ProjectID} == -1) {
                if ($Data{$ID}{ActionID}    == -1) {
                    $Param{Diseased} = 'checked';
                }
                elsif ($Data{$ID}{ActionID} == -2) {
                    $Param{LeaveDay} = 'checked';
                }
                elsif ($Data{$ID}{ActionID} == -3) {
                    $Param{Overtime} = 'checked';
                }
            }
            else {
                if (!$Flag) {
                    $Self->{LayoutObject}->Block(
                        Name => 'UnitBlock',
                    );
                }

                $Self->{LayoutObject}->Block(
                    Name => 'Unit',
                    Data => {
                        Project   => $Project{Project}{$Data{$ID}{ProjectID}},
                        Action    => $Action {$Data{$ID}{ActionID}}{Action},
                        Remark    => $Data{$ID}{Remark},
                        StartTime => $Data{$ID}{StartTime},
                        EndTime   => $Data{$ID}{EndTime},
                        Period    => $Data{$ID}{Period},
                    }
                );
                $Flag = 1;
                $Param{Total} += $Data{$ID}{Period};
            }
        }
        if ($Flag) {
            $Self->{LayoutObject}->Block(
                Name => 'Total',
                Data => {Total => sprintf ("%.2f", $Param{Total})}
            );
        }
        if ($Param{Diseased} || $Param{LeaveDay} || $Param{Overtime}) {
            $Self->{LayoutObject}->Block(
                Name => 'OtherTimes',
                Data => {
                    Diseased => $Param{Diseased},
                    LeaveDay => $Param{LeaveDay},
                    Overtime => $Param{Overtime},
                }
            );
        }

        my $Vacation = $Self->{TimeAccountingObject}->VacationCheck(
            Year  => $Param{Year},
            Month => $Param{Month},
            Day   => $Param{Day},
        );

        if ($Vacation) {
            $Self->{LayoutObject}->Block(
                Name => 'Vacation',
                Data => {Vacation => $Vacation},
            );
        }

        # presentation
        my $Output = $Self->{LayoutObject}->Header(Title => "View");
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => \%Param,
            TemplateFile => 'AgentTimeAccountingView'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # delete object from database
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Delete') {
        foreach (qw(Day Month Year)) {
            $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        if (!$Self->{AccessRo}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        if (!$Self->{TimeAccountingObject}->WorkingUnitsDelete(
            Year  => $Param{Year},
            Month => $Param{Month},
            Day   => $Param{Day},
        )) {
            return $Self->{LayoutObject}->ErrorScreen();
        }
        return $Self->{LayoutObject}->Redirect(OP => "Action=$Self->{Action}&Subaction=Edit&Year=$Param{Year}&Month=$Param{Month}&Day=$Param{Day}");
    }

    # ---------------------------------------------------------- #
    # overview about the users time accounting
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Overview') {
        my $Output   = '';
        my %Frontend = ();
        my ($Sec, $Min, $Hour, $CurrentDay, $Month, $Year) = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
        );

        # permission check
        if (!$Self->{AccessRo}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        foreach (qw(Status Day Month Year UserID ProjectStatusShow)) {
            $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }
        $Param{Subaction} = 'Edit';

        if (!$Param{UserID}) {
            $Param{UserID} = $Self->{UserID};
        }
        else {
            if ($Param{UserID} != $Self->{UserID} && !$Self->{AccessRw}) {
                return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
            }
            $Param{Subaction} = 'View';
        }

        if ($Param{UserID} != $Self->{UserID}) {
            my %ShownUsers = $Self->{UserObject}->UserList(Type => 'Long', Valid => 1);
            $Param{User}   = $ShownUsers{$Param{UserID}};
            $Self->{LayoutObject}->Block(
                Name => 'User',
                Data => {%Param},
            );
        }

        # Check Date
        if (!$Param{Year} || !$Param{Month}) {
            $Param{Year}  = $Year;
            $Param{Month} = $Month;
        }
        else {
            $Param{Month} = sprintf("%02d", $Param{Month});
        }

        # store last screen
        $Self->{SessionObject}->UpdateSessionID(
            SessionID => $Self->{SessionID},
            Key       => 'LastScreen',
            Value     => "Action=$Self->{Action}&Subaction=Overview&Year=$Param{Year}&Month=$Param{Month}",
        );

        $Param{Month_to_Text}   = $MonthArray[$Param{Month}];

        ($Param{YearBack}, $Param{MonthBack}, $Param{DayBack}) = Add_Delta_YMD($Param{Year}, $Param{Month}, 1, 0, -1, 0);
        ($Param{YearNext}, $Param{MonthNext}, $Param{DayNext}) = Add_Delta_YMD($Param{Year}, $Param{Month}, 1, 0, 1, 0);

        # Overview per day
        my $DaysOfMonth             = Days_in_Month($Param{Year},  $Param{Month});

        for (my $Day = 1; $Day <= $DaysOfMonth; $Day++) {
            $Param{Day}     = sprintf("%02d", $Day);
            $Param{Weekday} = Day_of_Week($Param{Year}, $Param{Month}, $Day)-1;
            my $VacationCheck = $Self->{TimeAccountingObject}->VacationCheck(
                Year  => $Param{Year},
                Month => $Param{Month},
                Day   => $Day,
            );

            if ($Param{Year} eq $Year && $Param{Month} eq $Month && $CurrentDay eq $Day){
                $Param{Style} = 'bgcolor="orange"';
            }
            elsif ($VacationCheck) {
                $Param{Style}   = 'bgcolor="#EBCCCC"';
                $Param{Comment} = $VacationCheck;
            }
            elsif ($Param{Weekday} == 6 || $Param{Weekday} == 5) {
                $Param{Style} = 'bgcolor="#FFE0E0"';
            }
            else {
                $Param{Style} = 'bgcolor="#E5F0FF"';
            }

            my %Data = $Self->{TimeAccountingObject}->WorkingUnitsGet(
                Year   => $Param{Year},
                Month  => $Param{Month},
                Day    => $Param{Day},
                UserID => $Param{UserID},
            );

            my $WorkingHours = 0;
            my %OtherTime    = ();
            $OtherTime{'-1'} = 'Diseased';
            $OtherTime{'-2'} = 'LeaveDay';
            $OtherTime{'-3'} = 'Overtime';

            foreach my $ID (keys %Data) {
                $WorkingHours += $Data{$ID}{Period};
                if ($Data{$ID}{ProjectID} == -1) {
                    $Param{Comment} = $OtherTime{$Data{$ID}{ActionID}};
                }
            }
            if ($WorkingHours) {
                $Param{WorkingHours} = sprintf ("%.2f", $WorkingHours);
            }
            else {
                $Param{WorkingHours} = '';
            }

            $Param{Weekday_to_Text} = $WeekdayArray[$Param{Weekday}];
            $Self->{LayoutObject}->Block(
                Name => 'Row',
                Data => {%Param},
            );
            $Param{Comment} = '';
        }

        my %UserReport = $Self->{TimeAccountingObject}->UserReporting(
            Year  => $Param{Year},
            Month => $Param{Month},
        );

        foreach (qw(TargetState TargetStateTotal WorkingHoursTotal WorkingHours
            Overtime OvertimeTotal OvertimeUntil LeaveDay LeaveDayTotal
            LeaveDayRemaining Diseased DiseasedTotal DiseasedRemaining)
        ) {
            if (!$UserReport{$Param{UserID}}{$_}) {
                $UserReport{$Param{UserID}}{$_} = 0;
            }
            $Param{$_} = sprintf ("%.2f", $UserReport{$Param{UserID}}{$_});
        }

        # Show Overtime if allowed
        my %UserBasics = $Self->{TimeAccountingObject}->UserGet();
        if ($UserBasics{$Param{UserID}}{ShowOvertime}) {
            $Self->{LayoutObject}->Block(
                Name => 'Overtime',
                Data => {%Param, %Frontend},
            );
        }

        # Overview per project and action
        my %ProjectData = $Self->{TimeAccountingObject}->ProjectActionReporting(
            Year   => $Param{Year},
            Month  => $Param{Month},
            UserID => $Param{UserID},
        );

        # show the report sort by projects
        if (!$Param{ProjectStatusShow} || $Param{ProjectStatusShow} eq 'valid') {
            $Param{ProjectStatusShow} = 'all';
        }
        elsif ($Param{ProjectStatusShow} eq 'all') {
            $Param{ProjectStatusShow} = 'valid';
        }

        my %Project     = $Self->{TimeAccountingObject}->ProjectSettingsGet();
        my %Action      = $Self->{TimeAccountingObject}->ActionSettingsGet();
        foreach my $ProjectID (sort {$Project{Project}{$a} cmp $Project{Project}{$b}} keys %{$Project{Project}}) {
            if ($ProjectData{Total}{$ProjectID}) {
                if ($ProjectID ne '-1') {
                    $Param{Project}= '';
                    $Param{Class}            = 'contentvalue';
                    if ($Project{ProjectStatus}{$ProjectID}) {
                        $Param{Status} = '';
                    }
                    else {
                        $Param{Status} = 'passiv';
                    }
                    my $Total      = 0;
                    my $TotalTotal = 0;

                    if (($Param{ProjectStatusShow} eq 'all' && !$Param{Status})
                        || $Param{ProjectStatusShow} eq 'valid'
                    ) {

                        # action sort wrapper - be careful its not simular with project!!
                        my %SortedActions = ();
                        foreach my $ActionID (keys %{$ProjectData{Total}{$ProjectID}}) {
                            $SortedActions{$ActionID} = $Action{$ActionID}{Action};
                            $Param{RowSpan}++;
                        }
                        foreach my $ActionID (sort {$SortedActions{$a} cmp $SortedActions{$b}} keys %SortedActions) {
                        #foreach my $ActionID (keys %{$ProjectData{Total}{$ProjectID}}) {

                            $Param{Action}     = $Action{$ActionID}{Action};
                            $Param{Hours}      = sprintf ("%.2f", $ProjectData{PerMonth}{$ProjectID}{$ActionID}{Hours} || 0);
                            $Param{HoursTotal} = sprintf ("%.2f", $ProjectData{Total}{$ProjectID}{$ActionID}{Hours}    || 0);
                            $Total      += $Param{Hours};
                            $TotalTotal += $Param{HoursTotal};
                            $Self->{LayoutObject}->Block(
                                Name => 'Action',
                                Data => {%Param, %Frontend},
                            );
                            if (!$Param{Project}) {
                                $Param{Project}            = $Project{Project}{$ProjectID};
                                $Param{ProjectDescription} = $Self->{LayoutObject}->Ascii2Html(
                                    Text           => $Project{ProjectDescription}{$ProjectID},
                                    HTMLResultMode => 1,
                                    NewLine        => 50,
                                );
                                $Param{RowSpan}++;
                                $Self->{LayoutObject}->Block(
                                    Name => 'Project',
                                    Data => {%Param, %Frontend},
                                );
                                if ($UserBasics{$Self->{UserID}}{CreateProject}) {
                                    # persons how are allowed to see the create object link are allowed to see the project reporting
                                    $Self->{LayoutObject}->Block(
                                        Name => 'ProjectLink',
                                        Data => {
                                            Project => $Param{Project},
                                            ProjectID => $ProjectID,
                                        },
                                    );
                                }
                                else {
                                    $Self->{LayoutObject}->Block(
                                        Name => 'ProjectNoLink',
                                        Data => {Project => $Param{Project}},
                                    );
                                }

                                $Param{RowSpan} = 0;
                            }
                        }
                        $Param{Class}            = 'contentkey';
                        $Param{Action}           = 'Total';
                        $Param{Hours}            = sprintf ("%.2f", $Total);
                        $Param{HoursTotal}       = sprintf ("%.2f", $TotalTotal);
                        $Param{TotalHours}      += $Total;
                        $Param{TotalHoursTotal} += $TotalTotal;
                        $Self->{LayoutObject}->Block(
                            Name => 'Action',
                            Data => {%Param, %Frontend},
                        );
                        $Param{Class} = '';
                    }
                }
            }
        }
        if( defined($Param{TotalHours})) {
            $Param{TotalHours}      = sprintf ("%.2f", $Param{TotalHours});
        }
        if( defined($Param{TotalHoursTotal})) {
            $Param{TotalHoursTotal} = sprintf ("%.2f", $Param{TotalHoursTotal});
        }
        # build output
        $Output  = $Self->{LayoutObject}->Header(Title => "Overview");
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => {%Param, %Frontend},
            TemplateFile => 'AgentTimeAccountingOverview'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }
    # ---------------------------------------------------------- #
    # settings for handling time accounting
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Setting') {
        my $Output      = '';
        my %Frontend    = ();
        my %Action      = ();
        my %StatusList  = ();
        my %User        = ();
        my %Data        = ();
        my %UserBasics  = ();

        foreach (qw(ActionAction ActionUser NewAction NewUser)) {
            $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        # permission check
        if (!$Self->{AccessRw}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        $Self->{LayoutObject}->Block(
            Name => 'Setting',
        );

        # store last screen
        $Self->{SessionObject}->UpdateSessionID(
            SessionID => $Self->{SessionID},
            Key       => 'LastScreen',
            Value     => "Action=$Self->{Action}&Subaction=Setting",
        );

        if ($Param{ActionAction} || $Param{NewAction}) {
            %Action = $Self->{TimeAccountingObject}->ActionSettingsGet();
            my $ActionEmpty = 0;
            my %ActionCheck = ();
            foreach my $ActionID (keys %Action) {
                foreach (qw(Action ActionStatus)) {
                    $Data{$ActionID}{$_} = $Self->{ParamObject}->GetParam(Param => $_ . '[' . $ActionID . ']');
                }
                if (!$Data{$ActionID}{Action}) {
                    $ActionEmpty = 1;
                }
                if ($Data{$ActionID}{Action} && $ActionCheck{$Data{$ActionID}{Action}}  && $ActionCheck{$Data{$ActionID}{Action}} == 1) {
                    return $Self->{LayoutObject}->ErrorScreen(Message => 'The actionnaming must be unique!');
                }
                else {
                    $ActionCheck{$Data{$ActionID}{Action}} = 1;
                }
            }
            if (!$Self->{TimeAccountingObject}->ActionSettingsUpdate(%Data)) {
                return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t update action data!');
            }
            if ($Param{NewAction} && !$ActionEmpty) {
                if (!$Self->{TimeAccountingObject}->ActionSettingsInsert()) {
                    return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t insert action data!');
                }
            }
        }
        else {
            %User = $Self->{TimeAccountingObject}->UserSettingsGet();
            my %LastPeriod = ();
            foreach my $UserID (keys %User) {
                $LastPeriod{$UserID} = 0;
                foreach my $Period (keys %{$User{$UserID}}) {
                    if ($LastPeriod{$UserID} < $Period) {
                        $LastPeriod{$UserID} = $Period;
                    }
                }
                if ($Self->{ParamObject}->GetParam(Param => 'NewUserSetting[' . $UserID . ']')) {
                    my %InsertData = ();
                    $InsertData{UserID}      = $UserID;
                    $InsertData{Period}      = $LastPeriod{$UserID} + 1;
                    $Param{ActionUser} = 'true';
                    if (!$Self->{TimeAccountingObject}->UserSettingsInsert(%InsertData)) {
                        return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t insert user data!');
                    }
                }
            }
            if ($Param{ActionUser} || $Param{NewUser}) {
                %UserBasics = $Self->{TimeAccountingObject}->UserGet();
                foreach my $UserID (keys %User) {
                    foreach (qw(ShowOvertime CreateProject)) {
                        $Data{$UserID}{$_} = $Self->{ParamObject}->GetParam(Param => $_ . '[' . $UserID . ']');
                    }

                    my $Break = '';
                    if ($UserBasics{$UserID}{Description} && $Self->{ParamObject}->GetParam(Param => 'Description[' . $UserID . ']')) {
                        $Break = "\n";
                    }
                    my $Description = $Self->{ParamObject}->GetParam(Param => 'Description[' . $UserID . ']');
                    $Data{$UserID}{Description} = $UserBasics{$UserID}{Description} . $Break . $Description;

                    $Data{$UserID}{UserID} = $UserID;
                    foreach my $Period (keys %{$User{$UserID}}) {
                        foreach (qw(WeeklyHours LeaveDays UserStatus DateStart DateEnd Overtime)) {
                            $Data{$UserID}{$Period}{$_} = $Self->{ParamObject}->GetParam(Param => $_ . '[' . $UserID . '][' . $Period . ']');
                        }
                    $Data{$UserID}{$Period}{UserID} = $UserID;
                    $LastPeriod{$UserID} = $Period;
                    }
                }
                if (!$Self->{TimeAccountingObject}->UserSettingsUpdate(%Data)) {
                    return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t update user data!');
                }

                if ($Param{NewUser} && $Self->{ParamObject}->GetParam(Param => 'NewUserID')) {
                    %Data = ();
                    $Data{UserID}      = $Self->{ParamObject}->GetParam(Param => 'NewUserID');
                    $Data{Period}      = '1';
                    if (!$Self->{TimeAccountingObject}->UserSettingsInsert(%Data)) {
                        return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t insert user data!');
                    }
                    my %Groups = $Self->{GroupObject}->GroupList(Valid => 1);
                    my %GroupData =$Self->{GroupObject}->GroupMemberList(
                        UserID => $Data{UserID},
                        Type => 'ro',
                        Result => 'HASH',
                    );
                    foreach (keys %Groups) {
                        if ($Groups{$_} eq 'time_accounting' && !$GroupData{$_}) {

                            $Self->{GroupObject}->GroupMemberAdd(
                                GID => $_,
                                UID => $Data{UserID},
                                Permission => {
                                    ro        => 1,
                                    move_into => 0,
                                    create    => 0,
                                    owner     => 0,
                                    priority  => 0,
                                    rw        => 0,
                                },
                                UserID => $Self->{UserID},
                            );
                        }
                    }
                }
            }
        }

        # Show TimeAccounting Preferences
        $StatusList{1} = 'valid';
        $StatusList{0} = 'invalid';

        # Show action data
        %Action = $Self->{TimeAccountingObject}->ActionSettingsGet();
        my $ActionEmpty = 0;
        my %SortAction  = ();
        foreach my $ActionID (keys %Action) {
            $SortAction{$ActionID} = $Action{$ActionID}{Action};
        }
        foreach my $ActionID (sort {$SortAction{$a} cmp $SortAction{$b}} keys %SortAction) {
            $Param{Action}   = $Action{$ActionID}{Action};
            $Param{ActionID} = $ActionID;

            $Frontend{'StatusOption'} = $Self->{LayoutObject}->OptionStrgHashRef(
                Data       => \%StatusList,
                SelectedID => $Action{$ActionID}{ActionStatus} || '',
                Name       => "ActionStatus[$Param{ActionID}]",
            );

            $Self->{LayoutObject}->Block(
                Name => 'Action',
                Data => {%Param, %Frontend},
            );
        }

        # Show user data
        %User = $Self->{TimeAccountingObject}->UserSettingsGet();
        %UserBasics = $Self->{TimeAccountingObject}->UserGet();
        my %ShownUsers = $Self->{UserObject}->UserList(Type => 'Long', Valid => 1);
        foreach my $UserID (sort {$ShownUsers{$a} cmp $ShownUsers{$b}} keys %ShownUsers) {
            if ($User{$UserID}) {
                $Param{User}        = $ShownUsers{$UserID};
                $Param{UserID}      = $UserID;

                my $Description = '';
                if ($UserBasics{$UserID}{Description}) {
                    $Description = $Self->{LayoutObject}->Ascii2Html(
                        Text           => $UserBasics{$UserID}{Description},
                        HTMLResultMode => 1,
                        NewLine        => 50,
                    ) . "<br>";
                }

                if ($UserBasics{$UserID}{ShowOvertime}) {
                    $UserBasics{$UserID}{ShowOvertime} = "checked";
                }
                if ($UserBasics{$UserID}{CreateProject}) {
                    $UserBasics{$UserID}{CreateProject} = "checked";
                }

                $Self->{LayoutObject}->Block(
                    Name => 'User',
                    Data => {%Param, %Frontend,
                        Description   => $Description,
                        ShowOvertime  => $UserBasics{$UserID}{ShowOvertime},
                        CreateProject => $UserBasics{$UserID}{CreateProject},
                    },
                );

                delete $ShownUsers{$UserID};

                foreach my $Period (sort keys %{$User{$UserID}}) {
                    $Param{WeeklyHours} = $User{$UserID}{$Period}{WeeklyHours};
                    $Param{LeaveDays}   = $User{$UserID}{$Period}{LeaveDays};
                    $Param{Overtime}    = $User{$UserID}{$Period}{Overtime};
                    $Param{DateStart}   = $User{$UserID}{$Period}{DateStart};
                    $Param{DateEnd}     = $User{$UserID}{$Period}{DateEnd};
                    $Param{Period}      = $Period;

                    $Frontend{'StatusOption'} = $Self->{LayoutObject}->OptionStrgHashRef(
                        Data       => \%StatusList,
                        SelectedID => $User{$UserID}{$Period}{UserStatus} || '',
                        Name       => "UserStatus[$UserID][$Period]",
                    );

                    $Self->{LayoutObject}->Block(
                        Name => 'Period',
                        Data => {%Param, %Frontend},
                    );
                }
            }
        }

        if (%ShownUsers) {
            $ShownUsers{''} = '';
            $Frontend{'NewUserOption'} = $Self->{LayoutObject}->OptionStrgHashRef(
                Data       => \%ShownUsers,
                SelectedID => '',
                Name       => "NewUserID",
            );
            $Self->{LayoutObject}->Block(
                Name => 'NewUserOption',
                Data => {%Param, %Frontend},
            );
        }

        # build output
        $Output  = $Self->{LayoutObject}->Header(Title => "Setting");
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => {%Param, %Frontend},
            TemplateFile => 'AgentTimeAccountingSetting'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }
    # ---------------------------------------------------------- #
    # settings for handling time accounting
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'ProjectSetting') {
        my $Output     = '';
        my %Frontend   = ();
        my %Project    = ();
        my %StatusList = ();
        my %Data       = ();

        foreach (qw(ActionProject NewProject)) {
            $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        # permission check
        if (!$Self->{AccessRo}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        $Self->{LayoutObject}->Block(
            Name => 'ProjectSetting',
        );

        # Edit TimeAccounting Preferences
        if ($Param{ActionProject} || $Param{NewProject}) {
            %Project = $Self->{TimeAccountingObject}->ProjectSettingsGet();
            my $ProjectEmpty = 0;
            my %ProjectCheck = ();
            foreach my $ProjectID (keys %{$Project{Project}}) {
                foreach (qw(Project ProjectStatus ProjectDescription)) {
                    $Data{$ProjectID}{$_} = $Self->{ParamObject}->GetParam(Param => $_ . '[' . $ProjectID . ']');
                }
                if (!$Data{$ProjectID}{Project}) {
                    $ProjectEmpty = 1;
                }
                if ($Data{$ProjectID}{Project} && $ProjectCheck{$Data{$ProjectID}{Project}} && $ProjectCheck{$Data{$ProjectID}{Project}} == 1) {
                    return $Self->{LayoutObject}->ErrorScreen(
                        Message => 'The projectnaming must be unique!'
                    );
                }
                else {
                    if ($Data{$ProjectID}{Project}) {
                        $ProjectCheck{$Data{$ProjectID}{Project}} = 1;
                    }
                }
            }
            if (!$Self->{TimeAccountingObject}->ProjectSettingsUpdate(%Data)) {
                return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t update project data!');
            }
            if ($Param{NewProject} && !$ProjectEmpty) {
                if (!$Self->{TimeAccountingObject}->ProjectSettingsInsert()) {
                    return $Self->{LayoutObject}->ErrorScreen(Message => 'Can\'t insert project data!');
                }
            }
        }

        # Show TimeAccounting Preferences
        $StatusList{1} = 'valid';
        $StatusList{0} = 'invalid';

        # Show project data
        %Project = $Self->{TimeAccountingObject}->ProjectSettingsGet();
        my $ProjectEmpty = 0;
        foreach my $ProjectID (sort {$Project{Project}{$a} cmp $Project{Project}{$b}} keys %{$Project{Project}}) {

            $Param{Project}            = $Project{Project}{$ProjectID};
            $Param{ProjectDescription} = $Project{ProjectDescription}{$ProjectID};
            $Param{ProjectID}          = $ProjectID;

            $Frontend{'StatusOption'} = $Self->{LayoutObject}->OptionStrgHashRef(
                Data       => \%StatusList,
                SelectedID => $Project{ProjectStatus}{$ProjectID} || '',
                Name       => "ProjectStatus[$Param{ProjectID}]",
            );

            $Self->{LayoutObject}->Block(
                Name => 'Project',
                Data => {%Param, %Frontend},
            );
        }

        # build output
        $Output  = $Self->{LayoutObject}->Header(Title => "Setting");
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => {%Param, %Frontend},
            TemplateFile => 'AgentTimeAccountingSetting'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }
    # ---------------------------------------------------------- #
    # time accounting reporting
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Reporting') {
        my $Output   = '';
        my %Frontend = ();
        my %ShownUsers = $Self->{UserObject}->UserList(Type => 'Long', Valid => 0);
        my ($Sec, $Min, $Hour, $CurrentDay, $Month, $Year) = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
        );

        # permission check
        if (!$Self->{AccessRw}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        foreach (qw(Status Month Year ProjectStatusShow)) {
            $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        # Check Date
        if (!$Param{Year} || !$Param{Month}) {
            $Param{Year}  = $Year;
            $Param{Month} = $Month;
        }
        else {
            $Param{Month} = sprintf("%02d", $Param{Month});
        }

        # store last screen
        $Self->{SessionObject}->UpdateSessionID(
            SessionID => $Self->{SessionID},
            Key       => 'LastScreen',
            Value     => "Action=$Self->{Action}&Subaction=Reporting&Year=$Param{Year}&Month=$Param{Month}",
        );

        $Param{Month_to_Text}   = $MonthArray[$Param{Month}];

        my %Month = ();
        for (my $ID = 1; $ID <= 12; $ID++) {
            $Month{sprintf("%02d",$ID)}{'Value'}    = $MonthArray[$ID];
            $Month{sprintf("%02d",$ID)}{'Position'} = $ID;
            if ($Param{Month} == $ID) {
                $Month{sprintf("%02d",$ID)}{'Selected'} = 1;
            }
        }

        $Frontend{'MonthOption'} = $Self->{LayoutObject}->OptionElement(
            Data       => \%Month,
            Name       => "Month",
        );

        my %Year = ();
        for (my $ID = 2005; $ID <= 2006; $ID++) {
            $Year{$ID} = $ID;
        }
        $Frontend{'YearOption'} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data       => \%Year,
            SelectedID => $Param{Year} || '',
            Name       => "Year",
        );

        ($Param{YearBack}, $Param{MonthBack}, $Param{DayBack}) = Add_Delta_YMD($Param{Year}, $Param{Month}, 1, 0, -1, 0);
        ($Param{YearNext}, $Param{MonthNext}, $Param{DayNext}) = Add_Delta_YMD($Param{Year}, $Param{Month}, 1, 0, 1, 0);

        my %UserReport        = $Self->{TimeAccountingObject}->UserReporting(
            Year   => $Param{Year},
            Month  => $Param{Month},
            UserID => $Param{UserID},
        );

        my %UserBasics = $Self->{TimeAccountingObject}->UserGet();
        foreach my $UserID (sort {$ShownUsers{$a} cmp $ShownUsers{$b}} keys %ShownUsers) {
            if ($UserReport{$UserID}) {
                foreach (qw(LeaveDay Overtime WorkingHours Diseased LeaveDayRemaining OvertimeTotal)) {
                    $Param{$_} = sprintf ("%.2f", $UserReport{$UserID}{$_});
                    $Param{'Total' . $_} += $Param{$_};
                }

                # Show Overtime if allowed
                if (!$UserBasics{$UserID}{ShowOvertime}) {
                    $Param{Overtime}      = '';
                    $Param{OvertimeTotal} = '';
                }

                $Param{User}   = $ShownUsers{$UserID};
                $Param{UserID} = $UserID;
                $Self->{LayoutObject}->Block(
                    Name => 'User',
                    Data => {%Param, %Frontend},
                );
            }
        }
        foreach (qw(TotalLeaveDay TotalOvertime TotalWorkingHours
            TotalDiseased TotalLeaveDayRemaining TotalOvertimeTotal)
        ) {
            $Param{$_} = sprintf ("%.2f", $Param{$_});
        }

        # show the report sort by projects
        if (!$Param{ProjectStatusShow} || $Param{ProjectStatusShow} eq 'valid') {
            $Param{ProjectStatusShow} = 'all';
        }
        elsif ($Param{ProjectStatusShow} eq 'all') {
            $Param{ProjectStatusShow} = 'valid';
        }

        my %ProjectData = $Self->{TimeAccountingObject}->ProjectActionReporting(
            Year  => $Param{Year},
            Month => $Param{Month},
        );

        my %Project     = $Self->{TimeAccountingObject}->ProjectSettingsGet();
        my %Action      = $Self->{TimeAccountingObject}->ActionSettingsGet();
        foreach my $ProjectID (sort {$Project{Project}{$a} cmp $Project{Project}{$b}} keys %{$Project{Project}}) {
            if ($ProjectData{Total}{$ProjectID}) {
                if ($ProjectID ne '-1') {
                    $Param{Project}= '';
                    $Param{Class}            = 'contentvalue';
                    if ($Project{ProjectStatus}{$ProjectID}) {
                        $Param{Status} = '';
                    }
                    else {
                        $Param{Status} = 'passiv';
                    }
                    my $Total      = 0;
                    my $TotalTotal = 0;

                    if (($Param{ProjectStatusShow} eq 'all' && !$Param{Status})
                        || $Param{ProjectStatusShow} eq 'valid'
                    ) {
                        # action sort wrapper - be careful its not simular with project!!
                        my %SortedActions = ();
                        foreach my $ActionID (keys %{$ProjectData{Total}{$ProjectID}}) {
                            $SortedActions{$ActionID} = $Action{$ActionID}{Action};
                            $Param{RowSpan}++;
                        }
                        foreach my $ActionID (sort {$SortedActions{$a} cmp $SortedActions{$b}} keys %SortedActions) {
                        #foreach my $ActionID (keys %{$ProjectData{Total}{$ProjectID}}) {
                            $Param{Action}     = $Action{$ActionID}{Action};
                            $Param{Hours}      = sprintf ("%.2f", $ProjectData{PerMonth}{$ProjectID}{$ActionID}{Hours} || 0);
                            $Param{HoursTotal} = sprintf ("%.2f", $ProjectData{Total}   {$ProjectID}{$ActionID}{Hours}    || 0);
                            $Total      += $Param{Hours};
                            $TotalTotal += $Param{HoursTotal};

                            $Self->{LayoutObject}->Block(
                                Name => 'Action',
                                Data => {%Param, %Frontend},
                            );
                            if (!$Param{Project}) {
                                $Param{Project}            = $Project{Project}{$ProjectID};
                                $Param{ProjectID}          = $ProjectID;
                                $Param{ProjectDescription} = $Self->{LayoutObject}->Ascii2Html(
                                    Text           => $Project{ProjectDescription}{$ProjectID},
                                    HTMLResultMode => 1,
                                    NewLine        => 50,
                                );
                                $Param{RowSpan}++;
                                $Self->{LayoutObject}->Block(
                                    Name => 'Project',
                                    Data => {%Param, %Frontend},
                                );
                                $Param{RowSpan} = 0;
                            }
                        }

                        $Param{Class}            = 'contentkey';
                        $Param{Action}           = 'Total';
                        $Param{Hours}            = sprintf ("%.2f", $Total);
                        $Param{HoursTotal}       = sprintf ("%.2f", $TotalTotal);
                        $Param{TotalHours}      += $Total;
                        $Param{TotalHoursTotal} += $TotalTotal;
                        $Self->{LayoutObject}->Block(
                            Name => 'Action',
                            Data => {%Param, %Frontend},
                        );
                    }
                }
            }
        }
        if (!$Param{TotalHours}) {
            $Param{TotalHours} = 0;
        }
        if (!$Param{TotalHoursTotal}) {
            $Param{TotalHoursTotal} = 0;
        }

        $Param{TotalHours}      = sprintf ("%.2f", $Param{TotalHours});
        $Param{TotalHoursTotal} = sprintf ("%.2f", $Param{TotalHoursTotal});

        # build output
        $Output .= $Self->{LayoutObject}->Header(Title => "Reporting");
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => {%Param, %Frontend},
            TemplateFile => 'AgentTimeAccountingReporting'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }
    # ---------------------------------------------------------- #
    # time accounting project reporting
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'ProjectReporting') {
        my $Output   = '';
        my %Frontend = ();

        # permission check
        if (!$Self->{AccessRo}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        # get params
        foreach (qw(ProjectID)) {
            $Param{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        # check needed params
        foreach (qw(ProjectID)) {
            if (!$Param{$_}) {
                return $Self->{LayoutObject}->ErrorScreen(
                    Message => "ProjectReporting: Need $_"
                );
            }
        }

        my %Action      = $Self->{TimeAccountingObject}->ActionSettingsGet();
        my %Project     = $Self->{TimeAccountingObject}->ProjectSettingsGet();
        $Param{Project} = $Project{Project}{$Param{ProjectID}};

        my %ShownUsers = $Self->{UserObject}->UserList(Type => 'Long', Valid => 0);
        # necassary because the ProjectActionReporting is not reworked
        my ($Sec, $Min, $Hour, $CurrentDay, $Month, $Year) = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
        );
        my %ProjectData = ();
        my %ProjectTime     = ();

        # Only one function should be enough
        foreach my $UserID (keys %ShownUsers) {
            # Overview per project and action
            %ProjectData = $Self->{TimeAccountingObject}->ProjectActionReporting(
                Year   => $Year,
                Month  => $Month,
                UserID => $UserID,
            );
            if ($ProjectData{Total}{$Param{ProjectID}}) {
                foreach my $ActionID (keys %{$ProjectData{Total}{$Param{ProjectID}}}) {
                    $ProjectTime{$ActionID}{$UserID} = $ProjectData{Total}{$Param{ProjectID}}{$ActionID};
                }
            }
            else {
                delete ($ShownUsers{$UserID});
            }
        }

        # show the headerline
        foreach my $UserID (sort {$ShownUsers{$a} cmp $ShownUsers{$b}} keys %ShownUsers) {
            $Self->{LayoutObject}->Block(
                Name => 'UserName',
                Data => {User => $ShownUsers{$UserID}},
            );
        }

        # better solution for sort actions necessary
        my %NewAction = ();
        foreach my $ActionID (keys %ProjectTime) {
            $NewAction{$ActionID} = $Action{$ActionID}{Action};
        }
        %Action = %NewAction;

        # show the results
        my %Total = ();
        foreach my $ActionID (sort {$Action{$a} cmp $Action{$b}} keys %Action) {
            my $TotalHours = 0;
            $Self->{LayoutObject}->Block(
                Name => 'Action',
                Data => {
                    Action => $Action{$ActionID},
                },
            );

            foreach my $UserID (sort {$ShownUsers{$a} cmp $ShownUsers{$b}} keys %ShownUsers) {
                $TotalHours += $ProjectTime{$ActionID}{$UserID}{Hours};
                $Total{$UserID} += $ProjectTime{$ActionID}{$UserID}{Hours};
                $Self->{LayoutObject}->Block(
                    Name => 'User',
                    Data => {
                        Hours => sprintf ("%.2f", $ProjectTime{$ActionID}{$UserID}{Hours} || 0),
                    },
                );
            }
            # Total
            $Self->{LayoutObject}->Block(
                Name => 'User',
                Data => {
                    Hours => sprintf ("%.2f", $TotalHours),
                },
            );
        }
        $Param{TotalAll} = 0;
        foreach my $UserID (sort {$ShownUsers{$a} cmp $ShownUsers{$b}} keys %ShownUsers) {
            $Param{TotalAll} += $Total{$UserID};
            $Self->{LayoutObject}->Block(
                Name => 'UserTotal',
                Data => {
                    Total => sprintf ("%.2f", $Total{$UserID}),
                },
            );
        }

        $Param{TotalAll} = sprintf ("%.2f", $Param{TotalAll});

        # build output
        $Output .= $Self->{LayoutObject}->Header(Title => "ProjectReporting");
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => {%Param, %Frontend},
            TemplateFile => 'AgentTimeAccountingProjectReporting'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }
    # ---------------------------------------------------------- #
    # show error screen
    # ---------------------------------------------------------- #
    return $Self->{LayoutObject}->ErrorScreen(Message => "Invalid Subaction process!");
}

1;

# --
# AgentTimeAccountingEdit.dtl - provides HTML form for time accounting edit
# Copyright (C) 2003-2007 OTRS GmbH, http://otrs.com/
# --
# $Id: AgentTimeAccountingEdit.dtl,v 1.7 2007/06/05 14:18:12 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl.txt.
# --
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td colspan="3" class="mainhead">
      $Env{"Box0"}$Text{"Edit Time Accounting"}$Env{"Box1"}
    </td>
  </tr>
<!-- dtl:block:Overview -->
  <tr>
    <td colspan="3" class="menu">
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Overview&Year=$LQData{"Year"}&Month=$LQData{"Month"}" class="menuitem">$Text{"Overview"}</a>
<!-- dtl:block:CreateProject -->
      -
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=ProjectSetting" class="menuitem">
      $Text{"ProjectSetting"}
      </a>
<!-- dtl:block:CreateProject -->
    </td>
  </tr>
<!-- dtl:block:Overview -->
  <tr>
    <td class="mainbody">
      <br>
      <table border="0" width="700" align="center" cellspacing="0" cellpadding="0">
        <tr>
          <td>
            <table class="contentbody" border="0" width="100%" cellspacing="0" cellpadding="4">
              <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="compose">
                <input type="hidden" name="Action"    value="$Env{"Action"}">
                <input type="hidden" name="Subaction" value="Edit">
                <tr>
                  <td colspan="2" class="contenthead">$Text{"Time selection"}</td>
                </tr>
                <tr class="contentvalue">
                  <td>
                    $Data{"Date"}
                    <input class="button" type="submit" value="$Text{"Submit"}">
                  </td>
                  <td>
                    <table>
                      <tr>
                        <td>
                          $QData{"Year"}-$QData{"Month"}-$QData{"Day"}&nbsp;
                        </td>
                        <td>
                          <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Edit&Year=$LQData{"YearBack"}&Month=$LQData{"MonthBack"}&Day=$LQData{"DayBack"}"><img border="0" src="$Env{"Images"}left-big.png" alt="$Text{"back"}"></a>
                          &nbsp;
                        </td>
                        <td>
                          <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Edit&Year=$LQData{"YearNext"}&Month=$LQData{"MonthNext"}&Day=$LQData{"DayNext"}"><img border="0" src="$Env{"Images"}right-big.png" alt="$Text{"next"}"></a>
                          &nbsp;
                        </td>
                      </tr>
                    </table>
                  </td>
                </tr>
<!-- dtl:block:IncompleteText -->
                <tr class="contentvalue">
                  <td colspan="2">
                    $Text{"Incomplete Days"}:
<!-- dtl:block:IncompleteWorkingDays -->
                    &nbsp;
                    <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Edit&Year=$LQData{"Year"}&Month=$LQData{"Month"}&Day=$LQData{"Day"}">
                    $Data{"BoldStart"}
                    $QData{"Year"}-$QData{"Month"}-$QData{"Day"}
                    $Data{"BoldEnd"}
                    </a>
<!-- dtl:block:IncompleteWorkingDays -->
                    &nbsp;
                  </td>
                </tr>
<!-- dtl:block:IncompleteText -->
                <tr>
                  <td colspan="2" class="contentfooter">&nbsp;</td>
                </tr>
              </form>
            </table>
            <br>
            <table border="0" width="100%" cellspacing="0" cellpadding="4">
              <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data">
                <input type="hidden" name="Action"    value="$Env{"Action"}">
                <input type="hidden" name="Subaction" value="Edit">
                <input type="hidden" name="Status"    value="Action">
                <input type="hidden" name="Year"      value="$Data{"Year"}">
                <input type="hidden" name="Month"     value="$Data{"Month"}">
                <input type="hidden" name="Day"       value="$Data{"Day"}">
                <tr>
                  <td class="contenthead">
                    $Text{"Date"}: $Text{"$QData{"Weekday_to_Text"}"} $QData{"Year"}-$QData{"Month"}-$QData{"Day"}
                  </td>
                </tr>
                <tr>
                  <td class="contentbody">
                    <table border="0" cellspacing="0" cellpadding="3" align="center">
<!-- dtl:block:UnitBlock -->
                      <tr class="contentkey">
                        <td>$Text{"Project"}<span class="required">*</span></td>
                        <td>$Text{"Action"}<span class="required">*</span></td>
                        <td>$Text{"Remark"}</td>
                        <td>$Text{"Starttime"}<span class="$QData{"ClassTime"}">*</span></td>
                        <td>$Text{"Endtime"}<span class="$QData{"ClassTime"}">*</span></td>
                        <td>$Text{"Period"}<span class="footnote">$QData{"PeriodNote"}</span></td>
                      </tr>
<!-- dtl:block:Unit -->
                      <tr class="contentvalue">
                        <td>$Data{"ProjectOption"}</td>
                        <td>$Data{"ActionOption"}</td>
                        <td><input type="text" name="Remark[$QData{"ID"}]"    value="$QData{"Remark"}"    size="35"></td>
#                        <td>$Data{"StartTime"}"</td>
#                        <td>$Data{"EndTime"}"</td>
                        <td><input type="text" name="StartTime[$QData{"ID"}]" value="$QData{"StartTime"}" size="7"></td>
                        <td><input type="text" name="EndTime[$QData{"ID"}]"   value="$QData{"EndTime"}"   size="7"></td>
<!-- dtl:block:UnitInputPeriod -->
                        <td align="$QData{"TextPosition"}"><input type="text" name="Period[$QData{"ID"}]"    value="$QData{"Period"}"    size="7"></td>
<!-- dtl:block:UnitInputPeriod -->
<!-- dtl:block:UnitPeriodWithoutInput -->
                        <td align="$QData{"TextPosition"}">$QData{"Period"}</td>
<!-- dtl:block:UnitPeriodWithoutInput -->
                      </tr>
<!-- dtl:block:UnitRequired -->
                      <tr class="required">
                        <td colspan="6">
                          $Text{"$QData{"Description"}"}
                        </td>
                      </tr>
<!-- dtl:block:UnitRequired -->
<!-- dtl:block:UnitReadonly -->
                      <tr class="readonly">
                        <td colspan="6">
                          $Text{"$QData{"Description"}"}
                        </td>
                      </tr>
<!-- dtl:block:UnitReadonly -->
<!-- dtl:block:Unit -->
<!-- dtl:block:UnitBlock -->
                      <tr class="contentkey">
                        <td colspan="4">
<!-- dtl:block:OtherTimes -->
                          <input type="checkbox" value="-2" name="LeaveDay" $Data{"LeaveDay"}>$Text{"Leave Day"}&nbsp;
                          <input type="checkbox" value="-1" name="Diseased" $Data{"Diseased"}>$Text{"Diseased"}&nbsp;
                          <input type="checkbox" value="-3" name="Overtime" $Data{"Overtime"}>$Text{"Overtime"}
<!-- dtl:block:OtherTimes -->
                          &nbsp;
                        </td>
<!-- dtl:block:Total -->
                        <td>$Text{"Total"}:</td>
                        <td align="$QData{"TextPosition"}">$QData{"Total"}&nbsp;</td>
<!-- dtl:block:Total -->
                      </tr>
<!-- dtl:block:Required -->
                      <tr>
                        <td colspan="6">
                          <div class="required">$Text{"$QData{"Description"}"}</div>
                        </td>
                      </tr>
<!-- dtl:block:Required -->
<!-- dtl:block:Readonly -->
                      <tr>
                        <td colspan="6">
                          <div class="readonly">$Text{"$QData{"Description"}"}</div>
                        </td>
                      </tr>
<!-- dtl:block:Readonly -->
                    </table>
                  </td>
                </tr>
                <tr>
                  <td class="contentfooter">
                    <input class="button" style="display:none;" type="submit" value="$Text{"Submit"}" >
                    <input class="button" type="submit" value="$Text{"Delete"}" name="Delete">
                    <input class="button" accesskey="g" type="submit" value="$Text{"Submit"}">
                  </td>
                </tr>
                <tr>
                  <td>
                    <div class="required">* $Text{"Required Field"}</div>
<!-- dtl:block:FootNote -->
                    <div class="footnote">* $Text{"You have to insert Start- and Endtime or a Period"}</div>
<!-- dtl:block:FootNote -->
                    <div class="footnote">* $Text{"If you use action misc, please give an explanation in the field remark"}</div>
                  </td>
                </tr>
              </form>
            </table>
          </td>
        </tr>
      </table>
      <br>
    </td>
  </tr>
</table>

IyAtLQojIEFnZW50VGltZUFjY291bnRpbmdWaWV3LmR0bCAtIHByb3ZpZGVzIEhUTUwgZm9yIHZpZXcgaW5zZXJ0cyBvZiBvbmUgZGF5CiMgQ29weXJpZ2h0IChDKSAyMDAzLTIwMDYgT1RSUyBHbWJILCBodHRwOi8vb3Rycy5jb20vCiMgLS0KIyAkSWQ6IEFnZW50VGltZUFjY291bnRpbmdWaWV3LmR0bCx2IDEuMyAyMDA2LzEyLzA4IDE1OjA3OjIzIHRyIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLnR4dC4KIyAtLQo8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICA8dHI+CiAgICA8dGQgY29sc3Bhbj0iMyIgY2xhc3M9Im1haW5oZWFkIj4KICAgICAgJEVudnsiQm94MCJ9JFRleHR7IlZpZXcgVGltZSBBY2NvdW50aW5nIn0kRW52eyJCb3gxIn0KICAgIDwvdGQ+CiAgPC90cj4KICA8dHI+CiAgICA8dGQgY29sc3Bhbj0iMyIgY2xhc3M9Im1lbnUiPgogICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9JEVudnsiTGFzdFNjcmVlbiJ9IiBjbGFzcz0ibWVudWl0ZW0iPiRUZXh0eyJCYWNrIn08L2E+CiAgICA8L3RkPgogIDwvdHI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluYm9keSI+CiAgICAgIDxicj4KICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjcwMCIgYWxpZ249ImNlbnRlciIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIwIj4KPCEtLSBkdGw6YmxvY2s6VXNlciAtLT4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY29sc3Bhbj0iNCIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJWaWV3IG9mICJ9ICRRRGF0YXsiVXNlciJ9PGJyPjxicj48L3RkPgogICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOlVzZXIgLS0+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkPgogICAgICAgICAgICA8dGFibGUgY2xhc3M9ImNvbnRlbnRib2R5IiBib3JkZXI9IjAiIHdpZHRoPSIxMDAlIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjQiPgogICAgICAgICAgICAgIDxmb3JtIGFjdGlvbj0iJEVudnsiQ0dJSGFuZGxlIn0iIG1ldGhvZD0icG9zdCIgZW5jdHlwZT0ibXVsdGlwYXJ0L2Zvcm0tZGF0YSIgbmFtZT0iY29tcG9zZSI+CiAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iaGlkZGVuIiBuYW1lPSJBY3Rpb24iICAgIHZhbHVlPSIkRW52eyJBY3Rpb24ifSI+CiAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iaGlkZGVuIiBuYW1lPSJTdWJhY3Rpb24iIHZhbHVlPSJWaWV3Ij4KICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJoaWRkZW4iIG5hbWU9IlVzZXJJRCIgICAgdmFsdWU9IiRRRGF0YXsiVXNlcklEIn0iPgogICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50aGVhZCI+CiAgICAgICAgICAgICAgICAgIDx0ZCBjb2xzcGFuPSIyIj4kVGV4dHsiVGltZSBzZWxlY3Rpb24ifTwvdGQ+CiAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICA8dGQ+CiAgICAgICAgICAgICAgICAgICAgJERhdGF7IkRhdGVTZWxlY3Rpb24ifQogICAgICAgICAgICAgICAgICAgIDxpbnB1dCBjbGFzcz0iYnV0dG9uIiBhY2Nlc3NrZXk9ImciIHR5cGU9InN1Ym1pdCIgdmFsdWU9IiRUZXh0eyJTdWJtaXQifSI+CiAgICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgICAgICAgICA8dGFibGU+CiAgICAgICAgICAgICAgICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+CiAgICAgICAgICAgICAgICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAkUURhdGF7IkRhdGUifSZuYnNwOwogICAgICAgICAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgICAgICAgICAgICA8dGQ+CiAgICAgICAgICAgICAgICAgICAgICAgICAgPGEgaHJlZj0iJEVudnsiQmFzZWxpbmsifUFjdGlvbj0kRW52eyJBY3Rpb24ifSZTdWJhY3Rpb249VmlldyZZZWFyPSRRRGF0YXsiWWVhckJhY2sifSZNb250aD0kUURhdGF7Ik1vbnRoQmFjayJ9JkRheT0kUURhdGF7IkRheUJhY2sifSZVc2VySUQ9JFFEYXRheyJVc2VySUQifSI+PGltZyBib3JkZXI9IjAiIHNyYz0iJEVudnsiSW1hZ2VzIn1sZWZ0LWJpZy5wbmciIGFsdD0iJFRleHR7ImJhY2sifSI+PC9hPgogICAgICAgICAgICAgICAgICAgICAgICAgICZuYnNwOwogICAgICAgICAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgICAgICAgICAgICA8dGQ+CiAgICAgICAgICAgICAgICAgICAgICAgICAgPGEgaHJlZj0iJEVudnsiQmFzZWxpbmsifUFjdGlvbj0kRW52eyJBY3Rpb24ifSZTdWJhY3Rpb249VmlldyZZZWFyPSRRRGF0YXsiWWVhck5leHQifSZNb250aD0kUURhdGF7Ik1vbnRoTmV4dCJ9JkRheT0kUURhdGF7IkRheU5leHQifSZVc2VySUQ9JFFEYXRheyJVc2VySUQifSI+PGltZyBib3JkZXI9IjAiIHNyYz0iJEVudnsiSW1hZ2VzIn1yaWdodC1iaWcucG5nIiBhbHQ9IiRUZXh0eyJuZXh0In0iPjwvYT4KICAgICAgICAgICAgICAgICAgICAgICAgICAmbmJzcDsKICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgICAgICAgPC90YWJsZT4KICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnRmb290ZXIiPgogICAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iMiI+Jm5ic3A7PC90ZD4KICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgPC9mb3JtPgogICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgICA8YnI+CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSI0Ij4KICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRoZWFkIj4kVGV4dHsiRGF0ZSJ9OiAkUURhdGF7IldlZWtkYXlfdG9fVGV4dCJ9ICRRRGF0YXsiRGF0ZSJ9PC90ZD4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGJvZHkiPgogICAgICAgICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIxIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiIGFsaWduPSJjZW50ZXIiIHdpZHRoPSI5MCUiPgo8IS0tIGR0bDpibG9jazpVbml0QmxvY2sgLS0+CiAgICAgICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50a2V5Ij4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiUHJvamVjdCJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiQWN0aW9uIn08L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkPiRUZXh0eyJSZW1hcmsifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICA8dGQ+JFRleHR7IlN0YXJ0dGltZSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiRW5kdGltZSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiUGVyaW9kIn08L3RkPgogICAgICAgICAgICAgICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOlVuaXQgLS0+CiAgICAgICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50dmFsdWUiPgogICAgICAgICAgICAgICAgICAgICAgPHRkPiRRRGF0YXsiUHJvamVjdCJ9Jm5ic3A7PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kUURhdGF7IkFjdGlvbiJ9Jm5ic3A7PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kUURhdGF7IlJlbWFyayJ9Jm5ic3A7PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBhbGlnbj0icmlnaHQiPiRRRGF0YXsiU3RhcnRUaW1lIn0mbmJzcDs8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGFsaWduPSJyaWdodCI+JFFEYXRheyJFbmRUaW1lIn0mbmJzcDs8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGFsaWduPSJyaWdodCI+JFFEYXRheyJQZXJpb2QifSZuYnNwOzwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6VW5pdCAtLT4KPCEtLSBkdGw6YmxvY2s6VW5pdEJsb2NrIC0tPgo8IS0tIGR0bDpibG9jazpUb3RhbCAtLT4KICAgICAgICAgICAgICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnRrZXkiPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGNvbHNwYW49IjQiPgogICAgICAgICAgICAgICAgICAgICAgICAmbmJzcDsKICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICA8dGQ+JFRleHR7IlRvdGFsIn06PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBhbGlnbj0icmlnaHQiPiRRRGF0YXsiVG90YWwifSZuYnNwOzwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6VG90YWwgLS0+CjwhLS0gZHRsOmJsb2NrOk90aGVyVGltZXMgLS0+CiAgICAgICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50a2V5IiBhbGlnbj0iY2VudGVyIj4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjb2xzcGFuPSI2Ij4KICAgICAgICAgICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImNoZWNrYm94IiB2YWx1ZT0iLTIiIG5hbWU9IkxlYXZlRGF5IiAkRGF0YXsiTGVhdmVEYXkifSBkaXNhYmxlZD4kVGV4dHsiTGVhdmUgRGF5In0mbmJzcDsKICAgICAgICAgICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImNoZWNrYm94IiB2YWx1ZT0iLTEiIG5hbWU9IkRpc2Vhc2VkIiAkRGF0YXsiRGlzZWFzZWQifSBkaXNhYmxlZD4kVGV4dHsiRGlzZWFzZWQifSZuYnNwOwogICAgICAgICAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iY2hlY2tib3giIHZhbHVlPSItMyIgbmFtZT0iT3ZlcnRpbWUiICREYXRheyJPdmVydGltZSJ9IGRpc2FibGVkPiRUZXh0eyJPdmVydGltZSJ9CiAgICAgICAgICAgICAgICAgICAgICAgICZuYnNwOwogICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpPdGhlclRpbWVzIC0tPgo8IS0tIGR0bDpibG9jazpWYWNhdGlvbiAtLT4KICAgICAgICAgICAgICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnRrZXkiIGFsaWduPSJjZW50ZXIiPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGNvbHNwYW49IjYiPgogICAgICAgICAgICAgICAgICAgICAgICAkUURhdGF7IlZhY2F0aW9uIn0KICAgICAgICAgICAgICAgICAgICAgICAgJm5ic3A7CiAgICAgICAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOlZhY2F0aW9uIC0tPgogICAgICAgICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgIDx0cj48dGQgY2xhc3M9ImNvbnRlbnRmb290ZXIiPiZuYnNwOzwvdGQ+PC90cj4KICAgICAgICAgICAgPC90YWJsZT4KICAgICAgICAgIDwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgPC90YWJsZT4KICAgICAgPGJyPgogICAgPC90ZD4KICA8L3RyPgo8L3RhYmxlPgo=
# --
# AgentTimeAccountingSetting.dtl - provides HTML form for time accounting settings
# Copyright (C) 2003-2006 OTRS GmbH, http://otrs.com/
# --
# $Id: AgentTimeAccountingSetting.dtl,v 1.6 2006/12/08 15:07:23 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl.txt.
# --
<!-- dtl:block:ProjectSetting -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Edit Time Accounting Project Preference"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td class="menu">
      <a href="$Env{"Baselink"}$Env{"LastScreen"}" class="menuitem">
      $Text{"Back"}
      </a>
    </td>
  </tr>
  <tr>
    <td class="mainbody">
      <br>
      <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="ProjectSetting">
        <table border="0" width="700" align="center" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Project Setting"}</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3" align="center">
                <tr  class="contentkey">
                  <td>$Text{"Project"}<span class="required">*</span></td>
                  <td>$Text{"Description"}</td>
                  <td>$Text{"Status"}<span class="required">*</span></td>
                </tr>
<!-- dtl:block:Project -->
                <tr class="contentvalue">
                  <td>
                    <input type="text" name="Project[$QData{"ProjectID"}]" value="$QData{"Project"}" size="30">
                  </td>
                  <td>
                    <textarea rows="2" name="ProjectDescription[$QData{"ProjectID"}]" cols="40">$QData{"ProjectDescription"}</textarea>
                  </td>
                  <td>$Data{"StatusOption"}</td>
                </tr>
<!-- dtl:block:Project -->
                <tr>
                  <td colspan="2">
                    <input class="button" accesskey="g" type="submit" value="$Text{"New"}" name="NewProject">
                  </td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" accesskey="g" type="submit" value="$Text{"Submit"}" name="ActionProject">
            </td>
          </tr>
        </table>
      </form>
      <table border="0" width="700" align="center" cellspacing="0" cellpadding="4">
        <tr class="required"><td>* $Text{"Required Field"}</td></tr>
      </table>
      <br>
    </td>
  </tr>
</table>
<!-- dtl:block:ProjectSetting -->
<!-- dtl:block:Setting -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Edit Time Accounting Preference"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td class="menu">
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=ProjectSetting" class="menuitem">
      $Text{"ProjectSetting"}
      </a>
    </td>
  </tr>
  <tr>
    <td class="mainbody">
      <br>
      <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="Setting">
        <table border="0" width="700" align="center" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Action Setting"}</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3" align="center">
                <tr>
                  <td class="contentkey">$Text{"Action"}<span class="required">*</span></td>
                  <td class="contentkey">$Text{"Status"}<span class="required">*</span></td>
                </tr>
<!-- dtl:block:Action -->
                <tr>
                  <td class="contentvalue">
                    <input type="text" name="Action[$QData{"ActionID"}]" value="$QData{"Action"}" size="40">
                  </td>
                  <td class="contentvalue">$Data{"StatusOption"}</td>
                </tr>
<!-- dtl:block:Action -->
                <tr>
                  <td class="contentkey" colspan="2">
                    <input class="button" accesskey="g" type="submit" value="$Text{"New"}" name="NewAction">
                  </td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" accesskey="g" type="submit" value="$Text{"Submit"}" name="ActionAction">
            </td>
          </tr>
        </table>
        <br>
        <table border="0" width="700" align="center" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"User Setting"}</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3" align="center">
<!-- dtl:block:User -->
                <tr>
                  <td class="contentkey" colspan="3">$QData{"User"}</td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Description"}:</td>
                  <td class="contentvalue">
                    $Data{"Description"}
                    <textarea rows="1" name="Description[$QData{"UserID"}]" cols="40"></textarea>
                  </td>
                  <td class="contentkey">
                    <input type="Checkbox" name="ShowOvertime[$QData{"UserID"}]" value="1" $Data{"ShowOvertime"}>
                    $Text{"Show Overtime"}<br>
                    <input type="Checkbox" name="CreateProject[$QData{"UserID"}]" value="1" $Data{"CreateProject"}>
                    $Text{"Create Project"}
                  </td>
                </tr>
                <tr>
                  <td colspan="3">
                    <table border="0" cellspacing="0" cellpadding="3" align="center" width="100%">
                      <tr class="contentkey">
                        <td>$Text{"Startdate"}<span class="required">*</span></td>
                        <td>$Text{"Enddate"}<span class="required">*</span></td>
                        <td>$Text{"Leave Day"}<span class="required">*</span></td>
                        <td>$Text{"Weekly Hours"}</td>
                        <td>$Text{"Overtime"}</td>
                        <td>$Text{"Status"}<span class="required">*</span></td>
                      </tr>
<!-- dtl:block:Period -->
                      <tr class="contentvalue">
                        <td>
                          <input type="text" name="DateStart[$QData{"UserID"}][$QData{"Period"}]" value="$QData{"DateStart"}" size="12">
                        </td>
                        <td>
                          <input type="text" name="DateEnd[$QData{"UserID"}][$QData{"Period"}]" value="$QData{"DateEnd"}" size="12">
                        </td>
                        <td>
                          <input type="text" name="LeaveDays[$QData{"UserID"}][$QData{"Period"}]" value="$QData{"LeaveDays"}" size="5">
                        </td>
                        <td>
                          <input type="text" name="WeeklyHours[$QData{"UserID"}][$QData{"Period"}]" value="$QData{"WeeklyHours"}" size="5">
                        </td>
                        <td>
                          <input type="text" name="Overtime[$QData{"UserID"}][$QData{"Period"}]" value="$QData{"Overtime"}" size="5">
                        </td>
                        <td>$Data{"StatusOption"}</td>
                      </tr>
<!-- dtl:block:Period -->
                    </table>
                  </td>
                </tr>
                <tr>
                  <td class="contentvalue" colspan="3">
                    <input class="button" type="submit" value="$Text{"New user setting"}"  name="NewUserSetting[$QData{"UserID"}]">
                  </td>
                </tr>
                <tr>
                  <td colspan="3"><hr></td>
                </tr>
<!-- dtl:block:User -->
<!-- dtl:block:NewUserOption -->
                <tr>
                  <td class="contentvalue" colspan="3">
                    $Data{"NewUserOption"}
                    <input class="button" accesskey="g" type="submit" value="$Text{"New"}"  name="NewUser">
                  </td>
                </tr>
<!-- dtl:block:NewUserOption -->
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" accesskey="g" type="submit" value="$Text{"Submit"}" name="ActionUser">
            </td>
          </tr>
        </table>
      </form>
      <table border="0" width="700" align="center" cellspacing="0" cellpadding="4">
        <tr class="required"><td>* $Text{"Required Field"}</td></tr>
      </table>
      <br>
    </td>
  </tr>
</table>
<!-- dtl:block:Setting -->

# --
# AgentTimeAccountingOverview.dtl - provides HTML form for time accounting overview
# Copyright (C) 2003-2007 OTRS GmbH, http://otrs.com/
# --
# $Id: AgentTimeAccountingOverview.dtl,v 1.8 2007/06/05 14:18:38 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl.txt.
# --
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Time Accounting Overview per Month"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td class="mainbody">
      <br>
      <table border="0" width="33%" cellspacing="2" cellpadding="2" align="center">
<!-- dtl:block:User -->
        <tr>
          <td colspan="4" class="contentkey">$Text{"Overview of "} $QData{"User"}<br></td>
        </tr>
<!-- dtl:block:User -->
        <tr>
          <td colspan="4" align="center">
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Overview&Year=$LQData{"YearBack"}&Month=$LQData{"MonthBack"}&UserID=$LQData{"UserID"}">
            &#060;&#060;
            </a>
            &nbsp;&nbsp;
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Overview&Year=$LQData{"YearNext"}&Month=$LQData{"MonthNext"}&UserID=$LQData{"UserID"}">
            &#062;&#062;
            </a>
          </td>
        </tr>
        <tr>
          <td colspan="4" class="contenthead" align="center">
            $Text{"$QData{"Month_to_Text"}"} $QData{"Year"}
          </td>
        </tr>
<!-- dtl:block:Row -->
        <tr $Data{"Style"}>
          <td align="center">
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=$LQData{"Subaction"}&Year=$LQData{"Year"}&Month=$LQData{"Month"}&Day=$LQData{"Day"}&UserID=$LQData{"UserID"}">
            $QData{"Day"}
            </a>
          </td>
          <td>$Text{"$QData{"Weekday_to_Text"}"}</td>
          <td>$Text{"$QData{"Comment"}"}</td>
          <td align="right">$QData{"WorkingHours"}</td>
        </tr>
<!-- dtl:block:Row -->
        <tr>
          <td colspan="3" class="contenthead" align="left">$Text{"WorkingHours"}:</td>
          <td class="contenthead" align="right">$QData{"WorkingHours"}</td>
        </tr>
      </table>
      <br>
      <table border="0" width="33%" cellspacing="0" cellpadding="2" align="center">
<!-- dtl:block:Overtime -->
        <tr class="contenthead">
          <td colspan="2">$Text{"Overtime"}:</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"this month"}:</td>
          <td class="contentvalue" align="right">$QData{"Overtime"}</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"until"}:</td>
          <td class="contentvalue" align="right">$QData{"OvertimeUntil"}</td>
        </tr>
        <tr class="contenthead">
          <td>$Text{"Overtime total"}:</td>
          <td align="right">$QData{"OvertimeTotal"}</td>
        </tr>
        <tr>
          <td colspan="2">&nbsp;</td>
        </tr>
<!-- dtl:block:Overtime -->
        <tr class="contenthead">
          <td colspan="2">$Text{"Leave Days"}:</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"LeaveDay this month"}:</td>
          <td class="contentvalue" align="right">$QData{"LeaveDay"}</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"LeaveDayTotal"}:</td>
          <td class="contentvalue" align="right">$QData{"LeaveDayTotal"}</td>
        </tr>
        <tr class="contenthead">
          <td>$Text{"LeaveDay remaining"}:</td>
          <td align="right">$QData{"LeaveDayRemaining"}</td>
        </tr>
        <tr>
          <td colspan="2">&nbsp;</td>
        </tr>
        <tr class="contenthead">
          <td colspan="2">$Text{"Diseased"}:</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"Diseased this month"}:</td>
          <td class="contentvalue" align="right">$QData{"Diseased"}</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"DiseasedTotal"}:</td>
          <td class="contentvalue" align="right">$QData{"DiseasedTotal"}</td>
        </tr>
        <tr class="contentfooter">
          <td colspan="2">&nbsp;</td>
        </tr>
      </table>
      <br>
      <table class="contentbody" border="0" width="50%" cellpadding="3" cellspacing="0" align="center">
        <tr>
          <td class="contenthead">$Text{"Project Reporting"}:
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Overview&Year=$LQData{"Year"}&Month=$LQData{"Month"}&ProjectStatusShow=$LQData{"ProjectStatusShow"}&UserID=$LQData{"UserID"}">
            ($Text{"Show"} $Text{"$QData{"ProjectStatusShow"}"} $Text{"Projects"})
            </a>
          </td>
        </tr>
        <tr>
          <td>
            <table border="1" width="100%" align="center">
              <tr class="contentkey">
                <td colspan="2">&nbsp;</td>
                <td>$Text{"per Month"}</td>
                <td>$Text{"Total"}</td>
              </tr>
              <tr class="contentkey">
                <td>$Text{"Project"}</td>
                <td>$Text{"Action"}</td>
                <td>$Text{"Hours"}</td>
                <td>$Text{"Hours"}</td>
              </tr>
<!-- dtl:block:Action -->
              <tr>
<!-- dtl:block:Project -->
                <td rowspan="$Data{"RowSpan"}" valign="top">
                  <span class="contentkey$QData{"Status"}">
<!-- dtl:block:ProjectNoLink -->
                  $Data{"Project"}
<!-- dtl:block:ProjectNoLink -->
<!-- dtl:block:ProjectLink -->
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=ProjectReporting&ProjectID=$QData{"ProjectID"}">
                  $Data{"Project"}
                  </a>
<!-- dtl:block:ProjectLink -->
                  </span>
                  <br>
                  <span class="contentvalue$QData{"Status"}">$Data{"ProjectDescription"}</span>
                </td>
<!-- dtl:block:Project -->
                <td class="$QData{"Class"}$QData{"Status"}" >$QData{"Action"}</td>
                <td class="$QData{"Class"}$QData{"Status"}" align="right">$QData{"Hours"}</td>
                <td class="$QData{"Class"}$QData{"Status"}" align="right">$QData{"HoursTotal"}</td>
              </tr>
<!-- dtl:block:Action -->
              <tr class="contentkey">
                <td colspan="2">$Text{"Total"}</td>
                <td align="right">$QData{"TotalHours"}</td>
                <td align="right">$QData{"TotalHoursTotal"}</td>
              </tr>
            </table>
          </td>
        </tr>
        <tr>
          <td class="contentfooter">&nbsp;</td>
        </tr>
      </table>
      <br>
    </td>
  </tr>
</table>

# --
# AgentTimeAccountingReporting.dtl - provides HTML form for time accounting reporting
# Copyright (C) 2003-2007 OTRS GmbH, http://otrs.com/
# --
# $Id: AgentTimeAccountingReporting.dtl,v 1.7 2007/05/11 14:36:37 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl.txt.
# --

<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Time Preporting"}: $Text{"$QData{"Month_to_Text"}"} $QData{"Year"} $Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td class="mainbody">
      <br>
      <table class="contentbody" border="0" width="50%" cellpadding="3" cellspacing="0" align="center">
        <tr>
          <td colspan="2" class="contenthead" >$Text{"Time selection"}</td>
        </tr>
        <tr class="contentvalue">
          <td>
            <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="compose">
              <input type="hidden" name="Action"    value="$Env{"Action"}">
              <input type="hidden" name="Subaction" value="Reporting">
              $Data{"MonthOption"}
              $Data{"YearOption"}
              <input class="button" accesskey="g" type="submit" value="$Text{"Submit"}">
            </form>
          </td>
          <td>
            <table>
              <tr>
                <td>
                  $Text{"$QData{"Month_to_Text"}"} $QData{"Year"}
                </td>
                <td>
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Reporting&Year=$LQData{"YearBack"}&Month=$LQData{"MonthBack"}"><img border="0" src="$Env{"Images"}left-big.png" alt="$Text{"One day back"}"></a>
                </td>
                <td>
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Reporting&Year=$LQData{"YearNext"}&Month=$LQData{"MonthNext"}"><img border="0" src="$Env{"Images"}right-big.png" alt="$Text{"Next day"}"></a>
                </td>
              </tr>
            </table>
          </td>
        </tr>
        <tr>
          <td colspan="2" class="contentfooter">&nbsp;</td>
        </tr>
      </table>
      <br>
      <table class="contentbody" border="0" width="50%" cellpadding="3" cellspacing="0" align="center">
        <tr>
          <td class="contenthead">$Text{"User Reporting"}:</td>
        </tr>
        <tr>
          <td>
            <table border="1" width="100%" align="center" cellspacing="2" cellpadding="2">
              <tr class="contentkey">
                <td>&nbsp;</td>
                <td colspan="4">$Text{"per Month"}</td>
                <td colspan="2">$Text{"Total"}</td>
              </tr>
              <tr class="contentkey">
                <td>$Text{"User"}</td>
                <td>$Text{"Working Hours"}</td>
                <td>$Text{"Overtime"}</td>
                <td>$Text{"Leave Days"}</td>
                <td>$Text{"Diseased"}</td>
                <td>$Text{"Overtime"}</td>
                <td>$Text{"LeaveDay Remaining"}</td>
              </tr>
<!-- dtl:block:User -->
              <tr class="contentvalue">
                <td>
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Overview&Year=$LQData{"Year"}&Month=$LQData{"Month"}&UserID=$LQData{"UserID"}">
                  $QData{"User"}
                  </a>
                </td>
                <td align="right">$QData{"WorkingHours"}</td>
                <td align="right">&nbsp;$QData{"Overtime"}</td>
                <td align="right">$QData{"LeaveDay"}</td>
                <td align="right">$QData{"Diseased"}</td>
                <td align="right">&nbsp;$QData{"OvertimeTotal"}</td>
                <td align="right">$QData{"LeaveDayRemaining"}</td>
              </tr>
<!-- dtl:block:User -->
              <tr class="contentkey">
                <td>$Text{"Total"}</td>
                <td align="right">$QData{"TotalWorkingHours"}</td>
                <td align="right">$QData{"TotalOvertime"}</td>
                <td align="right">$QData{"TotalLeaveDay"}</td>
                <td align="right">$QData{"TotalDiseased"}</td>
                <td align="right">$QData{"TotalOvertimeTotal"}</td>
                <td align="right">$QData{"TotalLeaveDayRemaining"}</td>
              </tr>
            </table>
          </td>
        </tr>
        <tr>
          <td class="contentfooter">&nbsp;</td>
        </tr>
      </table>
      <br>
      <table class="contentbody" border="0" width="50%" cellpadding="3" cellspacing="0" align="center">
        <tr>
          <td class="contenthead">
            $Text{"Project Reporting"}:
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Reporting&Year=$LQData{"Year"}&Month=$LQData{"Month"}&ProjectStatusShow=$LQData{"ProjectStatusShow"}">
            ($Text{"Show"} $Text{"$QData{"ProjectStatusShow"}"} $Text{"Projects"})
            </a>
          </td>
        </tr>
        <tr>
          <td>
            <table border="1" width="100%" align="center">
              <tr class="contentkey">
                <td colspan="2">&nbsp;</td>
                <td>$Text{"per Month"}</td>
                <td>$Text{"Total"}</td>
              </tr>
              <tr class="contentkey">
                <td>$Text{"Project"}</td>
                <td>$Text{"Action"}</td>
                <td>$Text{"Hours"}</td>
                <td>$Text{"Hours"}</td>
              </tr>
<!-- dtl:block:Action -->
              <tr>
<!-- dtl:block:Project -->
                <td rowspan="$Data{"RowSpan"}" valign="top">
                  <span class="contentkey$QData{"Status"}">
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=ProjectReporting&ProjectID=$QData{"ProjectID"}">
                  $Data{"Project"}
                  </a>
                  </span>
                  <br>
                  <span class="contentvalue$QData{"Status"}">$Data{"ProjectDescription"}</span>
                </td>
<!-- dtl:block:Project -->
                <td class="$QData{"Class"}$QData{"Status"}" >$QData{"Action"}</td>
                <td class="$QData{"Class"}$QData{"Status"}" align="right">$QData{"Hours"}</td>
                <td class="$QData{"Class"}$QData{"Status"}" align="right">$QData{"HoursTotal"}</td>
              </tr>
<!-- dtl:block:Action -->
              <tr  class="contentkey">
                <td colspan="2">$Text{"Total"}</td>
                <td align="right">$QData{"TotalHours"}</td>
                <td align="right">$QData{"TotalHoursTotal"}</td>
              </tr>
            </table>
          </td>
        </tr>
        <tr>
          <td class="contentfooter">&nbsp;</td>
        </tr>
      </table>
      <br>
    </td>
  </tr>
</table>

IyAtLQojIEFnZW50VGltZUFjY291bnRpbmdQcm9qZWN0UmVwb3J0aW5nLmR0bCAtIHByb3ZpZGVzIEhUTUwgZm9ybSBmb3IgdGltZSBhY2NvdW50aW5nIHJlcG9ydGluZwojIENvcHlyaWdodCAoQykgMjAwMy0yMDA2IE9UUlMgR21iSCwgaHR0cDovL290cnMuY29tLwojIC0tCiMgJElkOiBBZ2VudFRpbWVBY2NvdW50aW5nUHJvamVjdFJlcG9ydGluZy5kdGwsdiAxLjIgMjAwNi8xMi8wOCAxNTowNzoyMyB0ciBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC50eHQuCiMgLS0KCjx0YWJsZSBib3JkZXI9IjAiIHdpZHRoPSIxMDAlIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiPgogIDx0cj4KICAgIDx0ZCBjbGFzcz0ibWFpbmhlYWQiPgogICAgICAkRW52eyJCb3gwIn0kVGV4dHsiVGltZSBQcm9qZWN0IFByZXBvcnRpbmcifTogJFFEYXRheyJQcm9qZWN0In0gJEVudnsiQm94MSJ9CiAgICA8L3RkPgogIDwvdHI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtZW51Ij4KICAgICAgPGEgaHJlZj0iJEVudnsiQmFzZWxpbmsifSRFbnZ7Ikxhc3RTY3JlZW4ifSIgY2xhc3M9Im1lbnVpdGVtIj4kVGV4dHsiQmFjayJ9PC9hPgogICAgPC90ZD4KICA8L3RyPgogIDx0cj4KICAgIDx0ZCBjbGFzcz0ibWFpbmJvZHkiPgogICAgICA8YnI+CiMgICAgICA8dGFibGUgY2xhc3M9ImNvbnRlbnRib2R5IiBib3JkZXI9IjAiIHdpZHRoPSI1MCUiIGNlbGxwYWRkaW5nPSIzIiBjZWxsc3BhY2luZz0iMCIgYWxpZ249ImNlbnRlciI+CiMgICAgICAgIDx0cj4KIyAgICAgICAgICA8dGQgY29sc3Bhbj0iMiIgY2xhc3M9ImNvbnRlbnRoZWFkIiA+JFRleHR7IlRpbWUgc2VsZWN0aW9uIn08L3RkPgojICAgICAgICA8L3RyPgojICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+CiMgICAgICAgICAgPHRkPgojICAgICAgICAgICAgPGZvcm0gYWN0aW9uPSIkRW52eyJDR0lIYW5kbGUifSIgbWV0aG9kPSJwb3N0IiBlbmN0eXBlPSJtdWx0aXBhcnQvZm9ybS1kYXRhIiBuYW1lPSJjb21wb3NlIj4KIyAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iQWN0aW9uIiAgICB2YWx1ZT0iJEVudnsiQWN0aW9uIn0iPgojICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iaGlkZGVuIiBuYW1lPSJTdWJhY3Rpb24iIHZhbHVlPSJSZXBvcnRpbmciPgojICAgICAgICAgICAgICAkRGF0YXsiTW9udGhPcHRpb24ifQojICAgICAgICAgICAgICAkRGF0YXsiWWVhck9wdGlvbiJ9CiMgICAgICAgICAgICAgIDxpbnB1dCBjbGFzcz0iYnV0dG9uIiBhY2Nlc3NrZXk9ImciIHR5cGU9InN1Ym1pdCIgdmFsdWU9IiRUZXh0eyJTdWJtaXQifSI+CiMgICAgICAgICAgICA8L2Zvcm0+CiMgICAgICAgICAgPC90ZD4KIyAgICAgICAgICA8dGQ+CiMgICAgICAgICAgICA8dGFibGU+CiMgICAgICAgICAgICAgIDx0cj4KIyAgICAgICAgICAgICAgICA8dGQ+CiMgICAgICAgICAgICAgICAgICAkVGV4dHsiJFFEYXRheyJNb250aF90b19UZXh0In0ifSAkUURhdGF7IlllYXIifQojICAgICAgICAgICAgICAgIDwvdGQ+CiMgICAgICAgICAgICAgICAgPHRkPgojICAgICAgICAgICAgICAgICAgPGEgaHJlZj0iJEVudnsiQmFzZWxpbmsifUFjdGlvbj0kRW52eyJBY3Rpb24ifSZTdWJhY3Rpb249UmVwb3J0aW5nJlllYXI9JFFEYXRheyJZZWFyQmFjayJ9Jk1vbnRoPSRRRGF0YXsiTW9udGhCYWNrIn0iPjxpbWcgYm9yZGVyPSIwIiBzcmM9IiRFbnZ7IkltYWdlcyJ9bGVmdC1iaWcucG5nIiBhbHQ9IiRUZXh0eyJPbmUgZGF5IGJhY2sifSI+PC9hPgojICAgICAgICAgICAgICAgIDwvdGQ+CiMgICAgICAgICAgICAgICAgPHRkPgojICAgICAgICAgICAgICAgICAgPGEgaHJlZj0iJEVudnsiQmFzZWxpbmsifUFjdGlvbj0kRW52eyJBY3Rpb24ifSZTdWJhY3Rpb249UmVwb3J0aW5nJlllYXI9JFFEYXRheyJZZWFyTmV4dCJ9Jk1vbnRoPSRRRGF0YXsiTW9udGhOZXh0In0iPjxpbWcgYm9yZGVyPSIwIiBzcmM9IiRFbnZ7IkltYWdlcyJ9cmlnaHQtYmlnLnBuZyIgYWx0PSIkVGV4dHsiTmV4dCBkYXkifSI+PC9hPgojICAgICAgICAgICAgICAgIDwvdGQ+CiMgICAgICAgICAgICAgIDwvdHI+CiMgICAgICAgICAgICA8L3RhYmxlPgojICAgICAgICAgIDwvdGQ+CiMgICAgICAgIDwvdHI+CiMgICAgICAgIDx0cj4KIyAgICAgICAgICA8dGQgY29sc3Bhbj0iMiIgY2xhc3M9ImNvbnRlbnRmb290ZXIiPiZuYnNwOzwvdGQ+CiMgICAgICAgIDwvdHI+CiMgICAgICA8L3RhYmxlPgojICAgICAgPGJyPgogICAgICA8dGFibGUgY2xhc3M9ImNvbnRlbnRib2R5IiBib3JkZXI9IjAiIHdpZHRoPSI1MCUiIGNlbGxwYWRkaW5nPSIzIiBjZWxsc3BhY2luZz0iMCIgYWxpZ249ImNlbnRlciI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50aGVhZCI+CiAgICAgICAgICAgICRUZXh0eyJQcm9qZWN0IFJlcG9ydGluZyJ9OiAkUURhdGF7IlByb2plY3QifQogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgPHRhYmxlIGJvcmRlcj0iMSIgd2lkdGg9IjEwMCUiIGFsaWduPSJjZW50ZXIiPgogICAgICAgICAgICAgIDx0ciBjbGFzcz0iY29udGVudGtleSI+CiAgICAgICAgICAgICAgICA8dGQ+JFRleHR7IkFjdGlvbiJ9PC90ZD4KPCEtLSBkdGw6YmxvY2s6VXNlck5hbWUgLS0+CiAgICAgICAgICAgICAgICA8dGQ+JFFEYXRheyJVc2VyIn08L3RkPgo8IS0tIGR0bDpibG9jazpVc2VyTmFtZSAtLT4KICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiVG90YWwifTwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6QWN0aW9uIC0tPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7IkFjdGlvbiJ9PC90ZD4KPCEtLSBkdGw6YmxvY2s6VXNlciAtLT4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIiBhbGlnbj0icmlnaHQiPiRRRGF0YXsiSG91cnMifTwvdGQ+CjwhLS0gZHRsOmJsb2NrOlVzZXIgLS0+CiAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6QWN0aW9uIC0tPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IlRvdGFsIn08L3RkPgo8IS0tIGR0bDpibG9jazpVc2VyVG90YWwgLS0+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIGFsaWduPSJyaWdodCI+JFFEYXRheyJUb3RhbCJ9PC90ZD4KPCEtLSBkdGw6YmxvY2s6VXNlclRvdGFsIC0tPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5IiBhbGlnbj0icmlnaHQiPiRRRGF0YXsiVG90YWxBbGwifTwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgPC90YWJsZT4KICAgICAgICAgIDwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRmb290ZXIiPiZuYnNwOzwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgPC90YWJsZT4KICAgICAgPGJyPgogICAgPC90ZD4KICA8L3RyPgo8L3RhYmxlPgo=
IyAtLQojIEFnZW50VGltZUFjY291bnRpbmdEZWxldGUuZHRsIC0gcHJvdmlkZXMgSFRNTCBmb3JtIGZvciBkZWxldGUgYSBkYXkgb2YgdGltZWFjY291bnRpbmcKIyBDb3B5cmlnaHQgKEMpIDIwMDMtMjAwNSBPVFJTIEdtYkgsIGh0dHA6Ly93d3cub3Rycy5kZS8KIyAtLQojICRJZDogQWdlbnRUaW1lQWNjb3VudGluZ0RlbGV0ZS5kdGwsdiAxLjEuMS4xIDIwMDUvMDkvMDUgMTI6MTg6NDMgdHIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwudHh0LgojIC0tCjwhLS0gc3RhcnQgZm9ybSAtLT4KPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjEwMCUiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iMyI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluaGVhZCI+CiAgICAgICRFbnZ7IkJveDAifSRUZXh0eyJEZWxldGUifTogJFRleHR7IlRpbWVBY2NvdW50aW5nIG9mIn0gJFFEYXRheyJZZWFyIn0tJFFEYXRheyJNb250aCJ9LSRRRGF0YXsiRGF5In0kRW52eyJCb3gxIn0KICAgIDwvdGQ+CiAgPC90cj4KIyAgPHRyPgojICAgIDx0ZCBjbGFzcz0ibWVudSI+CiMgICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9JEVudnsiTGFzdFNjcmVlblZpZXcifSIgY2xhc3M9Im1lbnVpdGVtIj4kVGV4dHsiQmFjayJ9PC9hPgojICAgIDwvdGQ+CiMgIDwvdHI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluYm9keSI+CiAgICAgIDxicj4KICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjM1MCIgYWxpZ249ImNlbnRlciIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSI0Ij4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY29sc3Bhbj0iMiIgY2xhc3M9ImNvbnRlbnRoZWFkIj4kVGV4dHsiT3B0aW9ucyJ9PC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGJvZHkiPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSIgID4kVGV4dHsiRGVsZXRlIn06IDwvdGQ+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+JFRleHR7IlRpbWVBY2NvdW50aW5nIG9mIn0gJFFEYXRheyJZZWFyIn0tJFFEYXRheyJNb250aCJ9LSRRRGF0YXsiRGF5In08L3RkPgogICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgPHRkIGNvbHNwYW49IjIiPiZuYnNwOzwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iMiIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJEbyB5b3UgcmVhbGx5IHdhbnQgdG8gZGVsZXRlIHRoaXMgT2JqZWN0In0/PC90ZD4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGZvb3RlciIgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+CiAgICAgICAgICAgIDxhIGhyZWY9IiRFbnZ7IkJhc2VsaW5rIn1BY3Rpb249JEVudnsiQWN0aW9uIn0mU3ViYWN0aW9uPUVkaXQmWWVhcj0kUURhdGF7IlllYXIifSZNb250aD0kUURhdGF7Ik1vbnRoIn0mRGF5PSRRRGF0YXsiRGF5In0iIj4kVGV4dHsiTm8ifTwvYT4gPGEgaHJlZj0iJEVudnsiQmFzZWxpbmsifUFjdGlvbj0kRW52eyJBY3Rpb24ifSZTdWJhY3Rpb249RGVsZXRlJlllYXI9JFFEYXRheyJZZWFyIn0mTW9udGg9JFFEYXRheyJNb250aCJ9JkRheT0kUURhdGF7IkRheSJ9Ij4kVGV4dHsiWWVzIn08L2E+CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgIDwvdGFibGU+CiAgICAgIDxicj4KICAgIDwvdGQ+CiAgPC90cj4KPC90YWJsZT4KPCEtLSBlbmQgZm9ybSAtLT4K
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9Ob3RpZmljYXRpb25UaW1lQWNjb3VudGluZy5wbQojIENvcHlyaWdodCAoQykgMjAwMy0yMDA3IE9UUlMgR21iSCwgaHR0cDovL290cnMuY29tLwojIC0tCiMgJElkOiBOb3RpZmljYXRpb25UaW1lQWNjb3VudGluZy5wbSx2IDEuMyAyMDA3LzA2LzA1IDE0OjE3OjQxIHRyIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6Tm90aWZpY2F0aW9uVGltZUFjY291bnRpbmc7Cgp1c2Ugc3RyaWN0Owp1c2UgS2VybmVsOjpTeXN0ZW06OlRpbWVBY2NvdW50aW5nOwp1c2UgS2VybmVsOjpTeXN0ZW06OlRpbWU7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gJyRSZXZpc2lvbjogMS4zICQnOwokVkVSU0lPTiA9fiBzL15cJC4qOlxXKC4qKVxXLis/JC8kMS87CgpzdWIgbmV3IHsKICAgIG15ICRUeXBlID0gc2hpZnQ7CiAgICBteSAlUGFyYW0gPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyAoJFNlbGYsICRUeXBlKTsKCiAgICAjIGdldCBuZWVkZWQgb2JqZWN0cwogICAgZm9yZWFjaCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBEQk9iamVjdCBMYXlvdXRPYmplY3QgVXNlcklEKSkgewogICAgICAgICRTZWxmLT57JF99ID0gJFBhcmFteyRffSB8fCBkaWUgIkdvdCBubyAkXyEiOwogICAgfQoKICAgICRTZWxmLT57VGltZU9iamVjdH0gICAgPSBLZXJuZWw6OlN5c3RlbTo6VGltZSAgICAgICAgICAtPm5ldyglUGFyYW0pOwogICAgJFNlbGYtPntUaW1lQWNjb3VudGluZ09iamVjdH0gPSBLZXJuZWw6OlN5c3RlbTo6VGltZUFjY291bnRpbmctPm5ldyglUGFyYW0pOwogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CiAgICBteSAlUGFyYW0gPSBAXzsKICAgIG15ICgkU2VjLCAkTWluLCAkSG91ciwgJERheSwgJE1vbnRoLCAkWWVhcikgPSAkU2VsZi0+e1RpbWVPYmplY3R9LT5TeXN0ZW1UaW1lMkRhdGUoCiAgICAgICAgU3lzdGVtVGltZSA9PiAkU2VsZi0+e1RpbWVPYmplY3R9LT5TeXN0ZW1UaW1lKCksCiAgICApOwogICAgbXkgJVVzZXIgPSAkU2VsZi0+e1RpbWVBY2NvdW50aW5nT2JqZWN0fS0+VXNlckN1cnJlbnRQZXJpb2RHZXQoCiAgICAgICAgWWVhciAgPT4gJFllYXIsCiAgICAgICAgTW9udGggPT4gJE1vbnRoLAogICAgICAgIERheSAgID0+ICREYXksCiAgICApOwogICAgaWYgKCRVc2VyeyRTZWxmLT57VXNlcklEfX0pewogICAgICAgIG15ICVJbmNvbXBsZXRlV29ya2luZ0RheXMgPSAkU2VsZi0+e1RpbWVBY2NvdW50aW5nT2JqZWN0fS0+V29ya2luZ1VuaXRzQ29tcGxldG5lc3NDaGVjaygpOwogICAgICAgICMgcmVkaXJlY3QgaWYgaW5jb21wbGV0ZSB3b3JraW5nIGRheSBhcmUgb3V0IG9mIHJhbmdlCiAgICAgICAgaWYgKCRJbmNvbXBsZXRlV29ya2luZ0RheXN7V2FybmluZ30pIHsKICAgICAgICAgICAgcmV0dXJuICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+Tm90aWZ5KAogICAgICAgICAgICAgICAgSW5mbyAgICAgPT4gJ1BsZWFzZSBpbnNlcnQgeW91ciB3b3JraW5nIGhvdXJzIScsCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnRXJyb3InCiAgICAgICAgICAgICk7CiAgICAgICAgfQogICAgICAgIGVsc2UgewogICAgICAgICAgICByZXR1cm4gJyc7CiAgICAgICAgfQogICAgfQogICAgcmV0dXJuICcnOwp9CgoxOwo=
iVBORw0KGgoAAAANSUhEUgAAABYAAAAWCAYAAADEtGw7AAAABmJLR0QA/wD/AP+gvaeTAAAEo0lEQVQ4y42VW2yURRTHf99lu7tdSr/laoBCa28qERaRRChSUWKEYKhcEqMEKoIKIdbGEDHRaIhCMJK2MZHEB60aHlRCMEDkQSlGFC1Bti1eCoFuuZV7lwK97DdzxoduSxERTnIyk8nM7/wzc+YcyxjDnSxv0U7PaIkZpRGt4yd3LUze6Yx1O3Dh0u9jRusKUbrMKO2J0oivMEojSifFV9vFVzXn9y+L3xW4+MU6z4hUidblxsskK38Y9xVEKRwVIdO1ONDSwemjFzlZ34Lfch6jVK0oqbzcuCp5W3Dx8jrPiKkj6MTGlOTycslIysaFEAPraw+TEw0wd04xKQMHL/p8/kMr+z/dh7rYERelZ145UtkPt/uhK/Z6xpg6OysYm7ckxmfzx/JUToiUgG9g3NAMRkSDCCAGxkcDvLeggBUflOGOyI6Jr+sGjd3o3QI2IlV20I3NWjSeiknZBB2LlECPQErg6dlFTJmaiza94FOtCcTA3GKP8vXzcLMzY6J01U3gomV7YkZL+Zjpebw0IRtlelWmBFLp0Tc3/HBjnPklk9iw5lVSGqbneUxd9TjG1+WhIe/G+sFGpCI8PMKCKSMI2L1KB3pPH1zgcEOcJbNnIiLEZjxB8loX3RrufziXwcX3IEpX3ABrKRtaMIwJQzPw04CBavu8aQD0rc1fMGTiozRetdjdepXWjh5Glz6A+LoMwM1/breHMV5hwRC0segWCNhgG7Ct3vtqbopzujXB2ytfQIuwurqWzvxHaE8F6dGKoANiLEZNzKFBa8+yX/dsIxITpYlGw9R+9Se7vjvCNQVdGro1JBIJ1jz/DIn6H7l6JcmMxSu50K05+3cDgwI2gwI2l5rP0/ztIYaN9hClwRBzjdIYLbi2RfJckizCXEmlVVtw9HgCz/MoLS2lurqa3Zs39uf9up9PEMkehnO9E3O5g7BrQfpfuEYkLkrTmezkyaXTCNqQTIMdC+6dUkpR6WzWrtvAKxs+YvDQkWgDSgwR18Z1bWKlBRSW5NN2or0vZtwyxpAz55v20oUTvIem5ZHpWoRdi7BjEXYhaFu4NvjdXThuAByXlBh6tKFLGTqV4boyXPeFX7f9zo53tiWN2RS1AUTp7Ud+ayVoG8IORFzICsCggEVWhsXggM3wwRGimRlkBWyyAjYR175JRMixOLzzEMD2/nQTX9Wc+OssZ46eJ+iQVmsRcW0irkXE7Q0WSQeNuBaRgEXYtQk5FkHH4sTBBC0HWgBq+sFtdYvjxle1Wz/eh+7yCaYVhB0I25DppKFO7zzkQCi9J+hY+Ne72fLmVoBaYzbFb6oVoqXyUltH/MM3dpDqTJFhWwQsCKbBe/afpqmpjZDduxZIe8/Vbt5/9hMunmqPA5W3FKEL9cuTovTMRPO5+GtLttB48CSuBQELMmxobjxN27EL/dniAA2/HGP1rCoSf5yJAzON2ZS8baHPLqryjNJV4uvyyZNHs2jBgzxWMo6CXA9toPn4Zer2HuPrLw9Q/9MxgFqgciD0f1tTZNT6mPi6QnxVJkp7RinE1xgRgGT69Wv67vSue95Ac0Nrvb6vjyH+b3X/Zf8AFuFkUCNJ29gAAAAASUVORK5CYII=