TimeAccounting
1.3.3
OTRS AG
http://otrs.org/
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
Fixed bug #Bug 3849 - The default button for the Edit screen in Safari v4 is the delete button (Step 4).
Fixed bug# 3849 and 3968 - thanks to Thorsten Thau.
First version for OTRS 2.4.
A TimeAccounting Modul.
Ein Zeiterfassungsmodul.
2.4.x
<br>
<b>WELCOME</b>
<br>
<br>
You are about to install the OTRS package TimeAccounting.<br>
<br>
<br>
<b>REQUIRED OTRS PACKAGES</b>
<ul>
<li>no dependencies</li>
</ul>
<br>
<br>
((enjoy))<br>
<br>
Implemented and maintained by OTRS AG and c.a.p.e. IT.<br>
<br>
<br>
<b>WILLKOMMEN</b>
<br>
<br>
Sie sind im Begriff das OTRS-Paket TimeAccounting (Zeiterfassung) zu installieren.<br>
<br>
<br>
<b>BENÖTIGTE OTRS-PAKETE</b>
<ul>
<li>keine Abhängigkeiten</li>
</ul>
<br>
<br>
((enjoy))<br>
<br>
Entwickelt und supported von OTRS AG and c.a.p.e. IT.<br>
<br>
<br>
<b>NOTICE</b>
<br>
<br>
In order to grant users access to the time accounting module, you need to add them as member to the group 'time_accounting'.
<br>
<br>
The menu items that were added by this package will be visible after you log-in to the system again.
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>HINWEIS</b>
<br>
<br>
Um Benutzern Zugriff auf die Zeiterfassung zu gewähren, müssen diese Mitglied der neuen Gruppe 'time_accounting' sein.
<br>
<br>
Die von diesem Paket hinzugefügten Menü-Punkte sind erst nach einem erneuten Anmeldevorgang im System sichtbar.
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>ATTENTION</b>
<br>
<br>
If you uninstall this package, all database tables that were created during installation will be deleted.
All data from these tables will be irrevocably lost!
<br>
<br>
The group 'time_accounting' that was created during package installation will be deactivated.
You can activate this group again in the admin area.
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>ACHTUNG</b>
<br>
<br>
Bei der Deinstallation werden die von diesem Paket angelegten Datenbank-Tabellen gelöscht.
Alle darin enthaltenen Daten gehen unwiderruflich verloren!
<br>
<br>
Die von diesem Paket angelegte Gruppe 'time_accounting' wird deaktiviert.
Sie kann jederzeit im Admin-Bereich wieder aktiviert werden.
<br>
<br>
((enjoy))<br>
<br>
# define function name
my $FunctionName = 'CodeInstall';
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
# load the module
if ( $Self->{MainObject}->Require($CodeModule) ) {
# create new instance
my $CodeObject = $CodeModule->new( %{$Self} );
if ($CodeObject) {
# start methode
if ( !$CodeObject->$FunctionName(%{$Self}) ) {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method $FunctionName() on $CodeModule.pm."
);
}
}
# error handling
else {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method new() on $CodeModule.pm."
);
}
}
# define function name
my $FunctionName = 'CodeUninstall';
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
# load the module
if ( $Self->{MainObject}->Require($CodeModule) ) {
# create new instance
my $CodeObject = $CodeModule->new( %{$Self} );
if ($CodeObject) {
# start methode
if ( !$CodeObject->$FunctionName(%{$Self}) ) {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method $FunctionName() on $CodeModule.pm."
);
}
}
# error handling
else {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method new() on $CodeModule.pm."
);
}
}
2009-07-14 08:28:45
opms.otrs.com
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iQXBwbGljYXRpb24iPgogICAgPENWUz4kSWQ6IFRpbWVBY2NvdW50aW5nLnhtbCx2IDEuMTIgMjAwOS8wMi8yNCAxNDo0MjowMSB0ciBFeHAgJDwvQ1ZTPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRnJvbnRlbmQ6Ok1vZHVsZSMjI0FnZW50VGltZUFjY291bnRpbmciIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Gcm9udGVuZE1vZHVsZVJlZ2lzdHJhdGlvbiBmb3IgVGltZUFjY291bnRpbmdNb2R1bGUgIHRvIGRlZmluZSB0aGUgcmVnaXN0cmF0aW9uIHNldHRpbmdzIGluIHRoZSBwcm9ncmFtIGFuZCB0byBzZXQgcmlnaHRzIGFuZCBwb3NpdGlvbnMgb2YgaWNvbnMgaW4gdGhlIG5hdmlnYXRpb24gYmFyLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5Gcm9udGVuZE1vZHVsUmVnaXN0cmF0aW9uIGb8ciBkYXMgWmVpdGVyZmFzc3VuZ3NNb2R1bC4gIEhpZXIKa/ZubmVuIFNpZSBkaWUgQW5nYWJlbiBm/HIgZGllIFJlZ2lzdHJpZXJ1bmcgZGVzIE1vZHVscyBpbSBQcm9ncmFtbSB05HRpZ2VuLiBGZXJuZXIKa/ZubmVuIGRpZSBSZWNodGUgdW5kIGRpZSBQb3NpdGlvbmVuIGRlciBJY29ucyBpbiBkZXIgTmF2aWdhdGlvbnNsZWlzdGUgZ2VzZXR6dCB3ZXJkZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ6Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxGcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICAgICAgICAgIDxHcm91cD50aW1lX2FjY291bnRpbmc8L0dyb3VwPgogICAgICAgICAgICAgICAgPEdyb3VwUm8+dGltZV9hY2NvdW50aW5nPC9Hcm91cFJvPgogICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPlRpbWVBY2NvdW50aW5nPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgIDxOYXZCYXJOYW1lPlRpbWVBY2NvdW50aW5nPC9OYXZCYXJOYW1lPgogICAgICAgICAgICAgICAgPFRpdGxlPlRpbWVBY2NvdW50aW5nPC9UaXRsZT4KICAgICAgICAgICAgICAgIDxOYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPlRpbWVBY2NvdW50aW5nPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgICAgICA8TmFtZT5UaW1lQWNjb3VudGluZzwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+dGltZV9hY2NvdW50aW5nLnBuZzwvSW1hZ2U+CiAgICAgICAgICAgICAgICAgICAgPExpbms+QWN0aW9uPUFnZW50VGltZUFjY291bnRpbmcmYW1wO1N1YmFjdGlvbj1FZGl0PC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxOYXZCYXI+VGltZUFjY291bnRpbmc8L05hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8VHlwZT5NZW51PC9UeXBlPgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjYwMDA8L1ByaW8+CiAgICAgICAgICAgICAgICAgICAgPEFjY2Vzc0tleT48L0FjY2Vzc0tleT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+SXRlbUFyZWE8L0Jsb2NrPgogICAgICAgICAgICAgICAgPC9OYXZCYXI+CiAgICAgICAgICAgICAgICA8TmF2QmFyPgogICAgICAgICAgICAgICAgPEdyb3VwUm8+dGltZV9hY2NvdW50aW5nPC9Hcm91cFJvPgogICAgICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5PdmVydmlldzwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+T3ZlcnZpZXc8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPm92ZXJ2aWV3LnBuZzwvSW1hZ2U+CiAgICAgICAgICAgICAgICAgICAgPExpbms+QWN0aW9uPUFnZW50VGltZUFjY291bnRpbmcmYW1wO1N1YmFjdGlvbj1PdmVydmlldzwvTGluaz4KICAgICAgICAgICAgICAgICAgICA8VHlwZT48L1R5cGU+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPjwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPE5hdkJhcj5UaW1lQWNjb3VudGluZzwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+bzwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjEwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgIDxHcm91cFJvPnRpbWVfYWNjb3VudGluZzwvR3JvdXBSbz4KICAgICAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+RWRpdCB0aW1lIHJlY29yZDwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+RWRpdDwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+bmV3LnBuZzwvSW1hZ2U+CiAgICAgICAgICAgICAgICAgICAgPExpbms+QWN0aW9uPUFnZW50VGltZUFjY291bnRpbmcmYW1wO1N1YmFjdGlvbj1FZGl0PC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPlRpbWVBY2NvdW50aW5nPC9OYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEFjY2Vzc0tleT5uPC9BY2Nlc3NLZXk+CiAgICAgICAgICAgICAgICAgICAgPFByaW8+MjAwPC9QcmlvPgogICAgICAgICAgICAgICAgPC9OYXZCYXI+CiAgICAgICAgICAgICAgICA8TmF2QmFyPgogICAgICAgICAgICAgICAgPEdyb3VwPnRpbWVfYWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPlByb2plY3QgdGltZSByZXBvcnRpbmc8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPlJlcG9ydGluZzwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+cmVwb3J0aW5nLnBuZzwvSW1hZ2U+CiAgICAgICAgICAgICAgICAgICAgPExpbms+QWN0aW9uPUFnZW50VGltZUFjY291bnRpbmcmYW1wO1N1YmFjdGlvbj1SZXBvcnRpbmc8L0xpbms+CiAgICAgICAgICAgICAgICAgICAgPFR5cGU+PC9UeXBlPgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz48L0Jsb2NrPgogICAgICAgICAgICAgICAgICAgIDxOYXZCYXI+VGltZUFjY291bnRpbmc8L05hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8QWNjZXNzS2V5PjwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjUwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8R3JvdXA+dGltZV9hY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+RWRpdCB0aW1lIGFjY291bnRpbmcgc2V0dGluZ3M8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPlNldHRpbmc8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPm1vZHVsZV9zZXR0aW5nLnBuZzwvSW1hZ2U+CiAgICAgICAgICAgICAgICAgICAgPExpbms+QWN0aW9uPUFnZW50VGltZUFjY291bnRpbmcmYW1wO1N1YmFjdGlvbj1TZXR0aW5nPC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPlRpbWVBY2NvdW50aW5nPC9OYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEFjY2Vzc0tleT48L0FjY2Vzc0tleT4KICAgICAgICAgICAgICAgICAgICA8UHJpbz45MDA8L1ByaW8+CiAgICAgICAgICAgICAgICA8L05hdkJhcj4KICAgICAgICAgICAgPC9Gcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdFByb2plY3ROYW1lIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBuYW1lIGZvciBuZXcgcHJvamVjdHMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkluaXRpYWxlciBQcm9qZWt0bmFtZSBiZWltIEVyc3RlbGxlbiBlaW5lcyBuZXVlbiBQcm9qZWt0cyBpbiBkZXIgWmVpdGVyZmFzc3VuZy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+PC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRQcm9qZWN0U3RhdHVzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzdGF0dXMgb2YgYW4gbmV3IHByb2plY3QuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkFuZmFuZ3NzdGF0dXMgZWluZXMgbmV1ZW4gUHJvamVrdHMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8T3B0aW9uIFNlbGVjdGVkSUQ9IjEiPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIwIj5pbnZhbGlkPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIxIj52YWxpZDwvSXRlbT4KICAgICAgICAgICAgPC9PcHRpb24+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRBY3Rpb25OYW1lIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBuYW1lIGZvciBuZXcgYWN0aW9ucy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+SW5pdGlhbGVyIE5hbWUgZWluZXIgbmV1ZW4gVOR0aWdrZWl0IGluIGRlciBaZWl0ZXJmYXNzdW5nLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZyBSZWdleD0iIj48L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdEFjdGlvblN0YXR1cyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgc3RhdHVzIG9mIGFuIG5ldyBhY3Rpb24uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkFuZmFuZ3NzdGF0dXMgZWluZXIgbmV1ZW4gVOR0aWdrZWl0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPE9wdGlvbiBTZWxlY3RlZElEPSIxIj4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMCI+aW52YWxpZDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMSI+dmFsaWQ8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0VXNlcldlZWtseUhvdXJzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzZXR0aW5nIGZvciB0aGUgc3RhbmRhcmQgd2Vla2x5IGhvdXJzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5TdGFuZGFyZCBXb2NoZW5hcmJlaXRzemVpdC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+NDA8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdFVzZXJMZWF2ZURheXMiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZhdWx0IHNldHRpbmcgZm9yIGxlYXZlIGRheXMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPlN0YW5kYXJkIFVybGF1YnN0YWdlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZyBSZWdleD0iIj4yMzwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0VXNlck92ZXJ0aW1lIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzZXR0aW5nIGZvciBvdmVydGltZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RXZlbnR1ZWxsIHZvcmhhbmRlbmVyINxiZXJzdHVuZGVu/GJlcnRyYWcuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjA8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdFVzZXJEYXRlU3RhcnQiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZhdWx0IHNldHRpbmcgZm9yIGRhdGUgc3RhcnQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPlN0YXJ0ZGF0dW0gZvxyIGRpZSBFaW5nYWJlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+MjAxMC0wMS0wMTwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0VXNlckRhdGVFbmQiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZhdWx0IHNldHRpbmcgZm9yIGRhdGUgZW5kLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5FbmRkYXR1bSBm/HIgZGllIEVpbmdhYmVuLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZyBSZWdleD0iIj4yMDEwLTEyLTMxPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRVc2VyU3RhdHVzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzdGF0dXMgb2YgYW4gbmV3IHVzZXIuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkFuZmFuZ3NzdGF0dXMgZWluZXMgbmV1ZW4gQmVudXR6ZXJzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPE9wdGlvbiBTZWxlY3RlZElEPSIxIj4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMCI+aW52YWxpZDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMSI+dmFsaWQ8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpNYXhJbnRlcnZhbE9mSW5jb21wbGV0ZURheXMiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Xb3JraW5nIGRheXMgYWZ0ZXIgYXQgdGhlIGxhdGVzdCB3b3JraW5nIHVuaXRzIGhhdmUgdG8gaW5zZXJ0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5MZWd0IGZlc3QsIG5hY2ggd2lldmllbGVuIFdlcmt0YWdlbiBzcOR0ZXN0ZW5zIGRpZSBBcmJlaXRzc3R1bmRlbiBlaW5nZXRyYWdlbiB3ZXJkZW4gbfxzc2VuLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZyBSZWdleD0iIj40PC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6Ok1heEludGVydmFsT2ZJbmNvbXBsZXRlRGF5c0JlZm9yZVdhcm5pbmciIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5TaG93IHdhcm5pbmcsIGlmIHRvIG11Y2ggaW5jb21wbGV0ZSB3b3JraW5nIGRheXMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPlplaWd0IGVpbmUgV2FybnVuZywgd2VubiB6dXZpZWxlIEFyYmVpdHN0YWdlIG5pY2h0IGVpbmdldHJhZ2VuIHd1cmRlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+MjwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZyb250ZW5kOjpOb3RpZnlNb2R1bGUjIyM4ODgtVGltZUFjY291bnRpbmciIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIG5vdGlmaWNhdGlvbiBtb2R1bGUgZ2l2ZXMgYSB3YXJuaW5nIGlmIHRvIG11Y2ggaW5jb21wbGV0ZSB3b3JraW5nIGRheXMgYXZhaWxhYmxlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5Nb2R1bCwgZGFzIGRlbiBBZ2VudCBpbSBOb3RpZmljYXRpb24tQmVyZWljaCBkZXMgQWdlbnQtSW50ZXJmYWNlcyBkYXL8YmVyIGluZm9ybWllcnQsIHdlbm4gc2Nob24genUgbGFuZ2Uga2VpbmUgU3R1bmRlbiBtZWhyIGVpbmdldHJhZ2VuIHd1cmRlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDo6TW9kdWxlTm90aWZ5PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik1vZHVsZSI+S2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5vdGlmaWNhdGlvblRpbWVBY2NvdW50aW5nPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6Ok1heEFsbG93ZWRJbnNlcnREYXlzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+Rm9yIGhvdyBtYW55IGRheXMgYWdvIHlvdSBjYW4gaW5zZXJ0IHdvcmtpbmcgdW5pdHMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkxlZ3QgZmVzdCwgYmlzIHdhbm4gbWFuIGluIORsdGVyZSBaZWl0ZWludHLkZ2UgYmVhcmJlaXRlbiBrYW5uICh6LiBCLiAxMCBUYWdlIHp1cvxja2xpZWdlbmQpLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZyBSZWdleD0iIj4xMDwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpSZWR1Y2VUaW1lIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VG8gdXNlLCBpZiBzb21lIGFjdGlvbnMgYWNvdW50IHJlZHVjZWQgd29ya2luZyBob3VycyhLZXkgPSZndDsgdHJhdmVsaW5nOyBDb250ZW50ID0mZ3Q7IDUwKS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+SGllciBrYW5uIG1hbiBlaW5nZWJlbiwgb2IgZvxyIGVpbmUgYmVzdGltbXRlIFTkdGlna2VpdCwgZGllIHp1IHZlcnJlY2huZW5kZW4gU3R1bmRlbiBnZWv8cnp0IHdlcmRlbi4gWi4gQiB3ZW5uIFJlaXNlemVpdGVuIG51ciB6dXIgSORsZnRlIHZlcmf8dGV0IHdlcmRlbiAoS2V5ID0mZ3Q7IGpvdXJuZXk7IENvbnRlbnQgPSZndDsgNTApLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9InRyYXZlbGluZyI+NTA8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6SW5wdXRIb3Vyc1dpdGhvdXRTdGFydEVuZFRpbWUiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5TZXR0aW5nLCBpZiB3b3JraW5nIGhvdXJzIGNhbiBpbnNlcnQgd2l0aG91dCBzdGFydC0gYW5kIGVuZHRpbWUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkxlZ3QgZmVzdCwgb2IgbWFuIEFyYmVpdHNzdHVuZGVuIGF1Y2ggb2huZSBBbmZhbmdzLSB1bmQgRW5kemVpdCBlaW5nZWJlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPE9wdGlvbiBTZWxlY3RlZElEPSIxIj4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMCI+Tm88L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjEiPlllczwvSXRlbT4KICAgICAgICAgICAgPC9PcHRpb24+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iUHJlQXBwbGljYXRpb25Nb2R1bGUjIyNBZ2VudFRpbWVBY2NvdW50aW5nIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBtb2R1bGUgeW91IGNhbiBmb3JjZSBpbnNlcnRzIGluIHRoZSBUaW1lQWNjb3VudGluZy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+TWl0IEhpbGZlIGRpZXNlcyBNb2R1bHMga/ZubmVuIEVpbnRy5GdlIGluIGRpZSBaZWl0ZXJmYXNzdW5nIGVyend1bmdlbiB3ZXJkZW4sIGluIGRlbSBiZWltIEVpbmxvZ2dlbiBkYXMgUG9ydGFsIGdlYmxvY2t0IHdpcmQgdW5kIG51ciBkYXMgWmVpdGVyZmFzc3VuZ3NmZW5zdGVyIGVyc2NoZWludC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+S2VybmVsOjpNb2R1bGVzOjpBZ2VudFRpbWVBY2NvdW50aW5nPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OlByb2plY3QyUmVtYXJrUmVnRXhwIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+Rm9yIHdoaWNoIHByb2plY3RzIGlzIGEgcmVtYXJrIHJlcXVpcmVkLiBJZiB0aGUgUmVnRXhwIG1hdGNoIG9uIHRoZSBwcm9qZWN0LCB5b3UgaGF2ZSB0byBpbnNlcnQgYSByZW1hcmsgdG9vLiBUaGUgUmVnRXhwIHVzZSB0aGUgc214IHBhcmFtZXRlci48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+SW5uZXJoYWxiIGRpZXNlciBLb25maWd1cmF0aW9uc29wdGlvbiBrYW5uIGVpbmUgUmVnRXhwIGRlZmluaWVydCB3ZXJkZW4sIGRpZSBmZXN0bGVndCwgYmVpIHdlbGNoZW4gUHJvamVrdGVuIGVpbmUgQmVtZXJrdW5nIGVpbmdldHJhZ2VuIHdlcmRlbiBtdXNzIChkaWUgUmVnRXhwIGFyYmVpdGV0IG1pdCBzbXgtUGFyYW1ldGVybikuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPl4oPzogUFJPSkVDVFwjIHwgSU5GUkE6IHwgUFJPRFVDVDogKTwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KPC9vdHJzX2NvbmZpZz4K
#--
# Kernel/System/TimeAccounting.pm - all time accounting functions
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: TimeAccounting.pm,v 1.36 2009/07/13 13:43:01 tt Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::TimeAccounting;

use strict;
use warnings;

use vars qw(@ISA $VERSION);
$VERSION = qw($Revision: 1.36 $) [1];

use Date::Pcalc qw(Today Days_in_Month Day_of_Week);

=head1 NAME#

Kernel::System::TimeAccounting - timeaccounting lib

=head1 SYNOPSIS

All timeaccounting functions

=head1 PUBLIC INTERFACE#

=over 4

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Encode;
    use Kernel::System::Log;
    use Kernel::System::Main;
    use Kernel::System::DB;
    use Kernel::System::Time;
    use Kernel::System::User;
    use Kernel::System::TimeAccounting;

    my $ConfigObject = Kernel::Config->new();
    my $EncodeObject = Kernel::System::Encode->new(
        ConfigObject => $ConfigObject,
    );
    my $LogObject = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $TimeObject = Kernel::System::Time->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
    );
    my $UserObject = Kernel::System::User->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
        TimeObject   => $TimeObject,
        DBObject     => $DBObject,
        EncodeObject => $EncodeObject,
    );
    my $TimeAccountingObject = Kernel::System::TimeAccounting->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
        UserID       => 123,
        MainObject   => $MainObject,
        TimeObject   => $TimeObject,
        UserObject   => $UserObject,
    );

=cut

sub new {
    my ( $Type, %Param ) = @_;

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

    # check needed objects
    for (qw(DBObject ConfigObject LogObject UserID TimeObject UserObject MainObject)) {
        $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, %Param ) = @_;

    for (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";

    # Caching
    if ( $Self->{'Cache::UserCurrentPeriodGet'}{$Date} ) {
        return %{ $Self->{'Cache::UserCurrentPeriodGet'}{$Date} };
    }

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

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

    $Self->{'Cache::UserCurrentPeriodGet'}{$Date} = \%Data;

    return %Data;
}

=item UserReporting()

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

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

=cut

sub UserReporting {
    my ( $Self, %Param ) = @_;

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

    $Param{Day} ||= Days_in_Month( $Param{Year}, $Param{Month} );

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

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

        my $Counter = 0;
        for my $Year ( $YearStart .. $YearEnd ) {
            my $MonthStartPoint = $Year == $YearStart ? $MonthStart : 1;
            my $MonthEndPoint   = $Year == $YearEnd   ? $MonthEnd   : 12;

            for my $Month ( $MonthStartPoint .. $MonthEndPoint ) {

                my $DayStartPoint = $Year == $YearStart && $Month == $MonthStart ? $DayStart : 1;
                my $DayEndPoint = $Year == $YearEnd
                    && $Month == $MonthEnd ? $DayEnd : Days_in_Month( $Year, $Month );

                for my $Day ( $DayStartPoint .. $DayEndPoint ) {
                    my %WorkingUnit = $Self->WorkingUnitsGet(
                        Year   => $Year,
                        Month  => $Month,
                        Day    => $Day,
                        UserID => $UserID,
                    );

                    my $LeaveDay    = 0;
                    my $Sick        = 0;
                    my $Overtime    = 0;
                    my $TargetState = 0;

                    if ( $WorkingUnit{LeaveDay} ) {
                        $Data{$UserID}{LeaveDayTotal}++;
                        $LeaveDay = 1;
                    }
                    elsif ( $WorkingUnit{Sick} ) {
                        $Data{$UserID}{SickTotal}++;
                        $Sick = 1;
                    }
                    elsif ( $WorkingUnit{Overtime} ) {
                        $Data{$UserID}{OvertimeTotal}++;
                        $Overtime = 1;
                    }

                    $Data{$UserID}{WorkingHoursTotal} += $WorkingUnit{Total};
                    my $VacationCheck = $Self->{TimeObject}->VacationCheck(
                        Year  => $Year,
                        Month => $Month,
                        Day   => $Day,
                    );
                    my $Weekday = Day_of_Week( $Year, $Month, $Day );
                    if (
                        $Weekday != 6
                        && $Weekday != 7
                        && !$VacationCheck
                        && !$Sick
                        && !$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} += $WorkingUnit{Total};
                        $Data{$UserID}{LeaveDay}     += $LeaveDay;
                        $Data{$UserID}{Sick}         += $Sick;
                    }
                }
            }
        }
        $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(
        Status => 'valid' || 'invalid', optional default valid && invalid
    );

=cut

sub ProjectSettingsGet {
    my ( $Self, %Param ) = @_;

    my %Data  = ();
    my $Where = '';

    if ( $Param{Status} ) {
        $Where = ' WHERE status = ';
        $Where .= $Param{Status} eq 'invalid' ? "'0'" : "'1'";
    }

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

    # fetch Data
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        my $ID = $Row[0];
        $Data{Project}{$ID}            = $Row[1];
        $Data{ProjectDescription}{$ID} = $Row[2];
        $Data{ProjectStatus}{$ID}      = $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, %Param ) = @_;

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

    # build sql

    my $SQL
        = 'INSERT INTO time_accounting_project (project, description, status) VALUES ( ? , ? , ?)';

    my $Bind = [ \$Param{Project}, \$Param{ProjectDescription}, \$Param{ProjectStatus} ];

    # db insert
    return if !$Self->{DBObject}->Do( SQL => $SQL, Bind => $Bind );
    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, %Param ) = @_;

    PROJECTID:
    for my $ProjectID ( sort keys %Param ) {
        next if !$Param{$ProjectID}{Project};

        # build sql
        my $SQL
            = "UPDATE time_accounting_project "
            . "SET project = ? , status = ? , description = ?  WHERE id = ?";

        my $Bind = [
            \$Param{$ProjectID}{Project},            \$Param{$ProjectID}{ProjectStatus},
            \$Param{$ProjectID}{ProjectDescription}, \$ProjectID
        ];

        # db insert
        return if !$Self->{DBObject}->Do( SQL => $SQL, Bind => $Bind );
    }
    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, %Param ) = @_;

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

    $Param{Action} ||= $Self->{ConfigObject}->Get('TimeAccounting::DefaultActionName') || '';
    $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
    return if !$Self->{DBObject}->Do( SQL => $SQL );
    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, %Param ) = @_;

    ACTIONID:
    for my $ActionID ( sort keys %Param ) {
        next ACTIONID if !$Param{$ActionID}{Action};

        # build sql
        my $SQL = 'UPDATE time_accounting_action SET action = ? , status = ? WHERE id = ? ';
        my $Bind = [ \$Param{$ActionID}{Action}, \$Param{$ActionID}{ActionStatus}, \$ActionID ];

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

=item UserList()

returns a hash with the user data of all user

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

=cut

sub UserList {
    my $Self = shift;

    my %Data = ();

    # db select
    $Self->{DBObject}->Prepare(
        SQL =>
            'SELECT user_id, description, show_overtime, create_project, calendar 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];
        $Data{ $Row[0] }{Calendar}      = $Row[4];
    }
    return %Data;
}

=item UserGet()

returns a hash with the user data of one user

    my %UserData = $TimeAccountingObject->UserGet(
        UserID => 15,  #
    );

=cut

sub UserGet {
    my ( $Self, %Param ) = @_;

    # check needed data
    if ( !$Param{UserID} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'UserGet: Need UserID!'
        );
        return;
    }

    my %Data = ();

    # db select
    $Self->{DBObject}->Prepare(
        SQL =>
            'SELECT description, show_overtime, create_project, calendar FROM time_accounting_user WHERE user_id = ?',
        Bind => [ \$Param{UserID} ],
    );

    # fetch Data
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $Data{UserID}        = $Param{UserID};
        $Data{Description}   = $Row[0];
        $Data{ShowOvertime}  = $Row[1];
        $Data{CreateProject} = $Row[2];
        $Data{Calendar}      = $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, %Param ) = @_;

    # delete cache
    delete $Self->{'Cache::UserCurrentPeriodGet'};

    $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
    for ( keys %Param ) {
        $Param{$_} = $Self->{DBObject}->Quote( $Param{$_} );
        if ( !defined( $Param{$_} ) ) {
            $Param{$_} = '';
        }
    }

    # build sql
    my $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
    return if !$Self->{DBObject}->Do( SQL => $SQL );

    # 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
        return if !$Self->{DBObject}->Do( SQL => $SQL );
    }
    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, %Param ) = @_;

    # delete cache
    delete $Self->{'Cache::UserCurrentPeriodGet'};

    USERID:
    for my $UserID ( sort keys %Param ) {

        my $UserRef = $Param{$UserID};

        if ( !defined $UserRef->{1}{DateStart} && !defined $UserRef->{1}{DateEnd} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "UserSettingUpdate: There are no data for user id $UserID!"
            );
            next USERID;
        }

        #set default for ShowOverTime...
        $UserRef->{ShowOvertime} ||= 0;

        #set default for CreateProject...
        $UserRef->{CreateProject} ||= 0;
        $UserRef->{Calendar}      ||= 0;

        # build sql
        my $SQL
            = "UPDATE time_accounting_user "
            . "SET description = ?, show_overtime = ?, create_project = ?, calendar = ? "
            . "WHERE user_id = ?";

        my $Bind = [
            \$UserRef->{Description}, \$UserRef->{ShowOvertime},
            \$UserRef->{CreateProject}, \$UserRef->{Calendar}, \$UserRef->{UserID}
        ];

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

        for (qw(UserID Description ShowOvertime CreateProject Calendar)) {
            delete $UserRef->{$_};
        }

        for my $Period ( keys %{$UserRef} ) {

            my $PeriodRef = $UserRef->{$Period};

            # build sql
            my $SQL
                = "UPDATE time_accounting_user_period "
                . "SET leave_days = ?, date_start = ?"
                . ", date_end = ?, overtime = ?"
                . ", weekly_hours = ?, status = ? "
                . "WHERE user_id = ? AND preference_period = ?";

            my $Bind = [
                \$PeriodRef->{LeaveDays}, \$PeriodRef->{DateStart},   \$PeriodRef->{DateEnd},
                \$PeriodRef->{Overtime},  \$PeriodRef->{WeeklyHours}, \$PeriodRef->{UserStatus},
                \$PeriodRef->{UserID},    \$Period,
            ];

            # db insert
            return if !$Self->{DBObject}->Do( SQL => $SQL, Bind => $Bind );
        }
    }
    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, %Param ) = @_;

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

    my $UserID = $Param{UserID} || $Self->{UserID};

    # TODO: Search only in the CurrentUserPeriod
    # TODO: Search only working units where action_id and project_id is true

    $Self->{DBObject}->Prepare(
        SQL  => "SELECT DISTINCT time_start FROM time_accounting_table WHERE user_id = ?",
        Bind => [ \$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 $YearEnd     = $Year;
    my $MonthEnd    = $Month;
    my $DayEnd      = $Day;

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

    my $Calendar = { $Self->UserGet( UserID => $UserID ) }->{Calendar};

    for my $Year ( $YearStart .. $YearEnd ) {

        my $MonthStartPoint = $Year == $YearStart ? $MonthStart : 1;
        my $MonthEndPoint   = $Year == $YearEnd   ? $MonthEnd   : 12;

        for my $Month ( $MonthStartPoint .. $MonthEndPoint ) {
            my $DayStartPoint = $Year == $YearStart && $Month == $MonthStart ? $DayStart : 1;
            my $DayEndPoint = $Year == $YearEnd
                && $Month == $MonthEnd ? $DayEnd : Days_in_Month( $Year, $Month );
            my $MonthString = sprintf( "%02d", $Month );

            for my $Day ( $DayStartPoint .. $DayEndPoint ) {
                my $VacationCheck = $Self->{TimeObject}->VacationCheck(
                    Year     => $Year,
                    Month    => $Month,
                    Day      => $Day,
                    Calendar => $Calendar,
                );

                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';
    for my $Year ( keys %{ $Data{Incomplete} } ) {
        for my $Month ( keys %{ $Data{Incomplete}{$Year} } ) {
            for 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, %Param ) = @_;

    for ( keys %Param ) {
        $Param{$_} = $Self->{DBObject}->Quote( $Param{$_} ) || '';
    }

    $Param{UserID} ||= $Self->{UserID};

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

    $Self->{DBObject}->Prepare(
        SQL => "SELECT user_id, project_id, action_id, remark, time_start, time_end,"
            . " period FROM time_accounting_table "
            . " WHERE "
            . " time_start >= '$DateStart' "
            . " AND time_start <= '$DateStop' "
            . " AND user_id = '$Param{UserID}' ORDER by id",
    );

    my %Data = (
        Total => 0,
        Date  => $Date,
    );

    # fetch Data
    ROW:
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        next ROW if $Row[4] !~ m{^ (.+?) \s (\d+:\d+) : (\d+) }smx;

        # check if it is a special working unit
        if ( $Row[1] == -1 ) {
            my $ActionID = $Row[2];

            $Data{Sick}     = $ActionID == -1 ? 1 : 0;
            $Data{LeaveDay} = $ActionID == -2 ? 1 : 0;
            $Data{Overtime} = $ActionID == -3 ? 1 : 0;

            next ROW;
        }
        my $StartTime = $2;
        my $EndTime   = '';
        if ( $Row[5] =~ m{^(.+?)\s(\d+:\d+):(\d+)}smx ) {
            $EndTime = $2;
        }

        my %WorkingUnit = (
            UserID    => $Row[0],
            ProjectID => $Row[1],
            ActionID  => $Row[2],
            Remark    => $Row[3],
            StartTime => $StartTime,
            EndTime   => $EndTime,
            Period    => defined( $Row[6] ) ? sprintf( "%.2f", $Row[6] ) : 0,
        );

        # only count complete working units
        if ( $Row[1] && $Row[2] ) {
            $Data{Total} += $WorkingUnit{Period};
        }
        push @{ $Data{WorkingUnits} }, \%WorkingUnit;
    }
    return %Data;
}

=item WorkingUnitsInsert()

insert working units in the db

    $TimeAccountingObject->WorkingUnitsInsert(
        Year  => '2005',
        Month => '07',
        Day   => '02',
        LeaveDay => 1, || 0
        Sick     => 1, || 0
        Overtime => 1, || 0
        WorkingUnits => [
            {
                ProjectID => 1,
                ActionID  => 23,
                Remark    => 'SomeText,
                StartTime => '7:30',
                EndTime   => '11:00',
                Period    => '8.5',
            },
            { ...... },
        ]
    );

=cut

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

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

    # add special time working units
    my %SpecialAction = (
        'Sick'     => '-1',
        'LeaveDay' => '-2',
        'Overtime' => '-3',
    );

    ELEMENT:
    for my $Element (qw(LeaveDay Sick Overtime)) {
        next ELEMENT if !$Param{$Element};

        my %Unit = (
            ProjectID => -1,
            ActionID  => $SpecialAction{$Element},
            Remark    => '',
            StartTime => '',
            EndTime   => '',
            Period    => 0,
        );

        push @{ $Param{WorkingUnits} }, \%Unit;
    }

    #insert new working units
    UNITREF:
    for my $UnitRef ( @{ $Param{WorkingUnits} } ) {

        #next UNITREF if !$UnitRef->{ProjectID} || !$UnitRef->{ActionID};

        my $StartTime = $Date . ' ' . $UnitRef->{StartTime};
        my $EndTime   = $Date . ' ' . $UnitRef->{EndTime};

        # '' does not work in integer field of postgres
        $UnitRef->{ProjectID} ||= 0;
        $UnitRef->{ActionID}  ||= 0;
        $UnitRef->{Period}    ||= 0;

        # build sql
        my $SQL
            = "INSERT INTO time_accounting_table "
            . "(user_id, project_id, action_id, remark,"
            . " time_start, time_end, period, created )"
            . " VALUES  ( ?, ?, ?, ?, ?, ?, ?, current_timestamp)";
        my $Bind = [
            \$Self->{UserID}, \$UnitRef->{ProjectID}, \$UnitRef->{ActionID},
            \$UnitRef->{Remark}, \$StartTime, \$EndTime, \$UnitRef->{Period}
        ];

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

=item WorkingUnitsDelete()

delets working units in the db

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

=cut

sub WorkingUnitsDelete {
    my ( $Self, %Param ) = @_;

    for (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
    my $SQL = "DELETE FROM time_accounting_table "
        . "WHERE time_start <= '$Date 23:59:59'"
        . " AND time_start >= '$Date 00:00:00' "
        . " AND user_id = '$Self->{UserID}'";

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

=item ProjectActionReporting()

returns a hash with the hours dependent project and action data

    my %ProjectData = $TimeAccountingObject->ProjectActionReporting(
        Year  => 2005,
        Month => 7,
        UserID => 123, # optional; no UserID means 'of all user'
    );

=cut

sub ProjectActionReporting {
    my ( $Self, %Param ) = @_;

    my %Data     = ();
    my $IDSelect = '';
    for (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 $DaysInMonth = Days_in_Month( $Param{Year}, $Param{Month} );
    my $DateString = $Param{Year} . "-" . sprintf( "%02d", $Param{Month} );

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

    # Total hours
    $Self->{DBObject}->Prepare(
        SQL => "SELECT project_id, action_id, period FROM time_accounting_table"
            . " WHERE project_id != -1 AND $SQL_Query_TimeStart",
    );

    # fetch Data
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        next if !$Row[2];
        $Data{ $Row[0] }{Actions}{ $Row[1] }{Total} += $Row[2];
    }

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

    # fetch Data
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        next if !$Row[2];
        $Data{ $Row[0] }{Actions}{ $Row[1] }{PerMonth} += $Row[2];
    }

    # add readable components
    my %Project = $Self->ProjectSettingsGet();
    my %Action  = $Self->ActionSettingsGet();
    for my $ProjectID ( keys %Data ) {
        $Data{$ProjectID}{Name}        = $Project{Project}{$ProjectID};
        $Data{$ProjectID}{Status}      = $Project{ProjectStatus}{$ProjectID};
        $Data{$ProjectID}{Description} = $Project{ProjectDescription}{$ProjectID};
        my $ActionsRef = $Data{$ProjectID}{Actions};
        for my $ActionID ( keys %{$ActionsRef} ) {
            $Data{$ProjectID}{Actions}{$ActionID}{Name} = $Action{$ActionID}{Action};
        }
    }

    return %Data;
}

=item ProjectTotalHours()

returns the total sum of all hours related to a project

    my $ProjectTotalHours = $TimeAccountingObject->ProjectTotalHours(
        ProjectID  => 15,
    );

=cut

sub ProjectTotalHours {
    my ( $Self, %Param ) = @_;

    # check needed param
    if ( !$Param{ProjectID} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'ProjectActionReporting: Need ProjectID!'
        );
        return;
    }

    # db select
    my $Total = 0;
    my $SQL   = 'SELECT SUM(period) FROM time_accounting_table WHERE project_id = ?';
    my $Bind  = [ \$Param{ProjectID} ];
    return if !$Self->{DBObject}->Prepare( SQL => $SQL, Bind => $Bind );

    # fetch Data
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $Total = $Row[0];
    }

    return $Total;
}

=item ProjectHistory()

returns a array with all WorkingUnits related to a project

    my @ProjectHistoryArray = $TimeAccountingObject->ProjectHistory(
        ProjectID  => 15,
    );

e.g. @ProjectHistoryArray = (
        {
            ID        => 999,
            UserID    => 15,
            User      => 'Tom',
            ActionID  => 6,
            Action    => 'misc',
            Remark    => 'remark',
            TimeStart => '7:00',
            TimeEnd   => '18:00',
            Date      => '2008-10-31', # the date of the working unit
            Period    => 11,
            Created   => '2008-11-01', # the insert time of the working unit
        },
        {
            ID        => 999,
            UserID    => 16,
            User      => 'Mane',
            ActionID  => 7,
            Action    => 'development',
            Remark    => 'remark',
            TimeStart => '7:00',
            TimeEnd   => '18:00',
            Period    => 11,
            Date      => '2008-11-03',
            Created   => '2008-11-03',
        }
    );

=cut

sub ProjectHistory {
    my ( $Self, %Param ) = @_;

    # check needed param
    if ( !$Param{ProjectID} ) {
        $Self->{LogObject}->Log(
            Priority => 'error', Message => 'ProjectActionReporting: Need ProjectID!'
        );
        return;
    }

    # call action data to get the readable name of the action
    my %ActionData = $Self->ActionSettingsGet();

    my %ShownUsers = $Self->{UserObject}->UserList( Type => 'Long', Valid => 0 );

    # db select
    my @Data = ();
    my $SQL  = 'SELECT id, user_id, action_id, remark, time_start, time_end, period, created '
        . ' FROM time_accounting_table WHERE project_id = ?';
    my $Bind = [ \$Param{ProjectID} ];
    $Self->{DBObject}->Prepare( SQL => $SQL, Bind => $Bind );

    # fetch Data
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        my $UserRef = {
            ID        => $Row[0],
            UserID    => $Row[1],
            User      => $ShownUsers{ $Row[1] },
            ActionID  => $Row[2],
            Action    => $ActionData{ $Row[2] }{Action},
            Remark    => $Row[3] || '',
            TimeStart => $Row[4],
            TimeEnd   => $Row[5],
            Date      => $Row[4],
            Period    => $Row[6],
            Created   => $Row[7],
        };
        $UserRef->{Date} =~ s{(\d\d\d\d-\d\d-\d\d) \s .+ }{$1}smx;

        push @Data, $UserRef;
    }

    return @Data;

}

=item LastProjectsOfUser()

returns the a array with the last projects of the user

    my @LastProjects = $TimeAccountingObject->LastProjectsOfUser();

=cut

sub LastProjectsOfUser {
    my $Self = shift;

    # db select
    # I don't use distinct because of ORDER BY problems of postgre sql
    my %Projects = ();
    my $SQL
        = 'SELECT project_id FROM time_accounting_table WHERE user_id = ? AND project_id <> -1 ORDER BY time_start DESC';
    my $Bind  = [ \$Self->{UserID} ];
    my $Limit = 40;
    return if !$Self->{DBObject}->Prepare( SQL => $SQL, Bind => $Bind, Limit => $Limit );

    # fetch Data
    my $Counter = 0;
    ROW:
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        next ROW if $Counter > 7;
        next ROW if $Projects{ $Row[0] };

        $Projects{ $Row[0] } = 1;
        $Counter++;
    }

    return keys %Projects;
}

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 (AGPL). If you
did not receive this file, see http://www.gnu.org/licenses/agpl.txt.

=head1 VERSION

$Revision: 1.36 $ $Date: 2009/07/13 13:43:01 $

=cut

IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9BZ2VudFRpbWVBY2NvdW50aW5nLnBtIC0gdGhlIGRlIGxhbmd1YWdlIGZvciBBZ2VudFRpbWVBY2NvdW50aW5nCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBkZV9BZ2VudFRpbWVBY2NvdW50aW5nLnBtLHYgMS40NCAyMDA5LzA0LzAzIDExOjQ5OjI5IHRyIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0FnZW50VGltZUFjY291bnRpbmc7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICRTZWxmLT57VHJhbnNsYXRpb259ID0gewogICAgICAgICV7ICRTZWxmLT57VHJhbnNsYXRpb259IH0sCgogICAgICAgICMgVGVtcGxhdGU6IEFnZW50VGltZUFjY291bnRpbmcKICAgICAgICAnU2V0dGluZycgICAgICAgICAgPT4gJ0tvbmZpZ3VyYXRpb24nLAogICAgICAgICdQcm9qZWN0IHNldHRpbmdzJyA9PiAnUHJvamVrdGtvbmZpZ3VyYXRpb25lbicsCgogICAgICAgICdOZXh0IGRheScgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnTuRjaHN0ZXIgVGFnJywKICAgICAgICAnT25lIGRheSBiYWNrJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0VpbmVuIFRhZyB6dXL8Y2snLAogICAgICAgICdEYXRlJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnRGF0dW0nLAogICAgICAgICdDb21tZW50cycgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnS29tbWVudGFyZScsCiAgICAgICAgJ3VudGlsJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdiaXMnLAogICAgICAgICdUb3RhbCBob3VycyB3b3JrZWQnICAgICAgICAgICAgICAgICAgICAgICA9PiAnQXJiZWl0c3N0dW5kZW4nLAogICAgICAgICdXb3JraW5nIEhvdXJzJyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnQXJiZWl0c3N0dW5kZW4nLAogICAgICAgICdIb3VycyBwZXIgd2VlaycgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnV29jaGVuc3R1bmRlbicsCiAgICAgICAgJ3RoaXMgbW9udGgnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdkaWVzZXIgTW9uYXQnLAogICAgICAgICdPdmVydGltZSAoSG91cnMpJyAgICAgICAgICAgICAgICAgICAgICAgICA9PiAn3GJlcnN0dW5kZW4gKGluIFN0dW5kZW4pJywKICAgICAgICAnT3ZlcnRpbWUgKHRvdGFsKScgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ9xiZXJzdHVuZGVuIChkaWVzZXIgTW9uYXQpJywKICAgICAgICAnT3ZlcnRpbWUgKHRoaXMgbW9udGgpJyAgICAgICAgICAgICAgICAgICAgPT4gJ9xiZXJzdHVuZGVuIChTdW1tZSknLAogICAgICAgICdSZW1haW5pbmcgb3ZlcnRpbWUgbGVhdmUnICAgICAgICAgICAgICAgICA9PiAn3GJlcnN0dW5kZW4gKHZlcmJsZWliZW5kKScsCiAgICAgICAgJ1ZhY2F0aW9uJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdVcmxhdWInLAogICAgICAgICdWYWNhdGlvbiAoRGF5cyknICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnVXJsYXViIChpbiBUYWdlbiknLAogICAgICAgICdWYWNhdGlvbiB0YWtlbiAodGhpcyBtb250aCknICAgICAgICAgICAgICA9PiAnVXJsYXVic3RhZ2UgKGRpZXNlciBNb25hdCknLAogICAgICAgICdWYWNhdGlvbiB0YWtlbiAodG90YWwpJyAgICAgICAgICAgICAgICAgICA9PiAnVXJsYXVic3RhZ2UgKFN1bW1lKScsCiAgICAgICAgJ1JlbWFpbmluZyB2YWNhdGlvbicgICAgICAgICAgICAgICAgICAgICAgID0+ICdVcmxhdWJzdGFnZSAodmVyYmxlaWJlbmQpJywKICAgICAgICAnU2ljayBsZWF2ZSB0YWtlbiAodGhpcyBtb250aCknICAgICAgICAgICAgPT4gJ0Vya3Jhbmt0IChkaWVzZXIgTW9uYXQpJywKICAgICAgICAnU2ljayBsZWF2ZSB0YWtlbiAodG90YWwpJyAgICAgICAgICAgICAgICAgPT4gJ0Vya3Jhbmt0IChTdW1tZSknLAogICAgICAgICdUaW1lQWNjb3VudGluZycgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnWmVpdGVyZmFzc3VuZycsCiAgICAgICAgJ1RpbWUgcmVwb3J0aW5nIG1vbnRobHkgb3ZlcnZpZXcnICAgICAgICAgID0+ICdNb25hdHP8YmVyc2ljaHQgWmVpdGVyZmFzc3VuZycsCiAgICAgICAgJ0VkaXQgdGltZSByZWNvcmQnICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdaZWl0ZXJmYXNzdW5nIGJlYXJiZWl0ZW4nLAogICAgICAgICdFZGl0IHRpbWUgYWNjb3VudGluZyBzZXR0aW5ncycgICAgICAgICAgICA9PiAnWmVpdGVyZmFzc3VuZ3NlaW5zdGVsbHVuZ2VuIGJlYXJiZWl0ZW4nLAogICAgICAgICdVc2VyIHJlcG9ydHMnICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnTnV0emVyYmVyaWNodGUnLAogICAgICAgICdVc2VyXCdzIHByb2plY3Qgb3ZlcnZpZXcnICAgICAgICAgICAgICAgICA9PiAnTnV0emVyYmVyaWNodGUnLAogICAgICAgICdQcm9qZWN0IHJlcG9ydCcgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnUHJvamVrdGJlcmljaHQnLAogICAgICAgICdQcm9qZWN0IHJlcG9ydHMnICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnUHJvamVrdGJlcmljaHRlJywKICAgICAgICAnVGltZSByZXBvcnRpbmcnICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1plaXRiZXJpY2h0ZScsCiAgICAgICAgJ0xlYXZlRGF5IFJlbWFpbmluZycgICAgICAgICAgICAgICAgICAgICAgID0+ICdWZXJibGVpYmVuZGUgVXJsYXVic3RhZ2UnLAogICAgICAgICdNb250aGx5IHRvdGFsJyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAncHJvIE1vbmF0JywKICAgICAgICAnVmlldyB0aW1lIHJlY29yZCcgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1plaXRyZWtvcmQgYW5zaWNodCcsCiAgICAgICAgJ1ZpZXcgb2YgJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdBbnNpY2h0IHZvbicsCiAgICAgICAgJ01vbnRobHknICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdwcm8gTW9uYXQnLAogICAgICAgICdIb3VycycgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnU3R1bmRlbicsCiAgICAgICAgJ0RhdGUgbmF2aWdhdGlvbicgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdBdXN3YWhsIERhdHVtJywKICAgICAgICAnTW9udGggbmF2aWdhdGlvbicgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0F1c3dhaGwgRGF0dW0nLAogICAgICAgICdEYXlzIHdpdGhvdXQgZW50cmllcycgICAgICAgICAgICAgICAgICAgICA9PiAnbmljaHQgYXVzZ2Vm/GxsdGUgVGFnZScsCiAgICAgICAgJ1Byb2plY3QnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdQcm9qZWt0JywKICAgICAgICAnUHJvamVjdHMnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1Byb2pla3RlJywKICAgICAgICAnR3JhbmQgdG90YWwnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1N1bW1lJywKICAgICAgICAnTGlmZXRpbWUnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1N1bW1lJywKICAgICAgICAnTGlmZXRpbWUgdG90YWwnICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1N1bW1lJywKICAgICAgICAnUmVwb3J0aW5nJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0JlcmljaHRzd2VzZW4nLAogICAgICAgICdUYXNrIHNldHRpbmdzJyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnVOR0aWdrZWl0c2VpbnN0ZWxsdW5nZW4nLAogICAgICAgICdVc2VyIHNldHRpbmdzJyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnTnV0emVyZWluc3RlbGx1bmdlbicsCiAgICAgICAgJ1Nob3cgT3ZlcnRpbWUnICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICfcYmVyc3R1bmRlbiBhbnplaWdlbicsCiAgICAgICAgJ0FsbG93IHByb2plY3QgY3JlYXRpb24nICAgICAgICAgICAgICAgICAgID0+ICdQcm9qZWt0IGVyc3RlbGxlbicsCiAgICAgICAgJ0FkZCB0aW1lIHBlcmlvZCcgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdOZXVlIE51dHplcmVpbnN0ZWxsdW5nJywKICAgICAgICAnUmVtYXJrJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0FubWVya3VuZycsCiAgICAgICAgJ1N0YXJ0JyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdCZWdpbm4nLAogICAgICAgICdFbmQnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnRW5kZScsCiAgICAgICAgJ1BlcmlvZCBiZWdpbicgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdEYXR1bSBCZWdpbm4nLAogICAgICAgICdQZXJpb2QgZW5kJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnRGF0dW0gRW5kZScsCiAgICAgICAgJ1BlcmlvZCcgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdEYXVlcicsCiAgICAgICAgJ0RheXMgb2YgdmFjYXRpb24nICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdVcmxhdWJzdGFnZScsCiAgICAgICAgJ09uIHZhY2F0aW9uJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdpbSBVcmxhdWInLAogICAgICAgICdTaWNrIGRheScgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnRXJrcmFua3QnLAogICAgICAgICdTaWNrIGxlYXZlIChEYXlzKScgICAgICAgICAgICAgICAgICAgICAgICA9PiAnRXJrcmFua3QgKGluIFRhZ2VuKScsCiAgICAgICAgJ1NpY2sgbGVhdmUnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdFcmtyYW5rdCcsCiAgICAgICAgJ09uIHNpY2sgbGVhdmUnICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdFcmtyYW5rdCcsCiAgICAgICAgJ1Rhc2snICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdU5HRpZ2tlaXQnLAogICAgICAgICdBdXRob3JpemVkIG92ZXJ0aW1lJyAgICAgICAgICAgICAgICAgICAgICA9PiAnYXV0b3Jpc2llcnRlINxiZXJzdHVuZGVuJywKICAgICAgICAnT24gb3ZlcnRpbWUgbGVhdmUnICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ9xiZXJzdHVuZGVuJywKICAgICAgICAnT3ZlcnRpbWUgbGVhdmUnICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ9xiZXJzdHVuZGVuJywKICAgICAgICAnVG90YWwnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1N1bW1lJywKICAgICAgICAnT3ZlcnZpZXcgb2YgJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ9xiZXJzaWNodCAtICcsCiAgICAgICAgJ1RpbWVBY2NvdW50aW5nIG9mJyAgICAgICAgICAgICAgICAgICAgICAgID0+ICdaZWl0ZXJmYXNzdW5nIHZvbScsCiAgICAgICAgJ1N1Y2Nlc3NmdWwgaW5zZXJ0IScgICAgICAgICAgICAgICAgICAgICAgID0+ICdFcmZvbGdyZWljaCBlaW5nZWb8Z3QhJywKICAgICAgICAnTW9yZSBpbnB1dCBmaWVsZHMnICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1dlaXRlcmUgRWluZ2FiZWZlbGRlcicsCiAgICAgICAgJ0RvIHlvdSByZWFsbHkgd2FudCB0byBkZWxldGUgdGhpcyBPYmplY3QnID0+ICdXb2xsZW4gU2llIGRpZXNlbiBFaW50cmFnIHdpcmtsaWNoIGz2c2NoZW4nLAogICAgICAgICdDYW5cJ3QgaW5zZXJ0IFdvcmtpbmcgVW5pdHMhJyAgICAgICAgICAgICA9PiAnS2FubiBkaWUgQXJiZWl0c3N0dW5kZW4gbmljaHQgZWluZvxnZW4hJywKICAgICAgICAnQ2FuXCd0IHNhdmUgc2V0dGluZ3MsIGJlY2F1c2Ugb2YgbWlzc2luZyB0YXNrIScgPT4KICAgICAgICAgICAgJ05pY2h0IHNwZWljaGVyYmFyIC0gVOR0aWdrZWl0IGZlaGx0IScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIG9mIG1pc3NpbmcgcHJvamVjdCEnID0+CiAgICAgICAgICAgICdOaWNodCBzcGVpY2hlcmJhciAtIFByb2pla3RhbmdhYmUgZmVobHQhJywKICAgICAgICAnQ2FuXCd0IHNhdmUgc2V0dGluZ3MsIGJlY2F1c2UgdGhlIFBlcmlvZCBpcyBiaWdnZXIgdGhhbiB0aGUgaW50ZXJ2YWwgYmV0d2VlbiBTdGFydHRpbWUgYW5kIEVuZHRpbWUhJwogICAgICAgICAgICA9PiAnTmljaHQgc3BlaWNoZXJiYXIgLSBEYXVlciBpc3QgZ3L232VyIGFscyBkZXIgWmVpdHJhdW0gendpc2NoZW4gQmVnaW5uIHVuZCBFbmRlIScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIFN0YXJ0dGltZSBpcyBvbGRlciB0aGFuIEVuZHRpbWUhJyA9PgogICAgICAgICAgICAnTmljaHQgc3BlaWNoZXJiYXIgLSBCZWdpbm4gbGllZ3QgbmFjaCBFbmRlIScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIG9mIG1pc3NpbmcgcGVyaW9kIScgPT4KICAgICAgICAgICAgJ05pY2h0IHNwZWljaGVyYmFyIC0gRGF1ZXIgKGVyZ2lidCBzaWNoIGF1cyBTdGFydC0gdW5kIEVuZHplaXQpIGlzdCBuaWNodCBhbmdlZ2ViZW4hJywKICAgICAgICAnQ2FuXCd0IHNhdmUgc2V0dGluZ3MsIGJlY2F1c2UgUGVyaW9kIGlzIG5vdCBnaXZlbiBnaXZlbiEnID0+CiAgICAgICAgICAgICdOaWNodCBzcGVpY2hlcmJhciAtIERhdWVyIChlcmdpYnQgc2ljaCBhdXMgU3RhcnQtIHVuZCBFbmR6ZWl0KSBpc3QgbmljaHQgYW5nZWdlYmVuIScsCiAgICAgICAgJ0FyZSB5b3Ugc3VyZSwgdGhhdCB5b3Ugd29ya2VkIHdoaWxlIHlvdSB3ZXJlIG9uIHNpY2sgbGVhdmU/JyA9PgogICAgICAgICAgICAnU2llIHdhcmVuIGtyYW5rIHVuZCBoYWJlbiBnZWFyYmVpdGV0PyBXaXIgYnJhdWNoZW4gbWVociBzb2xjaGUgTWl0YXJiZWl0ZXIuJywKICAgICAgICAnQXJlIHlvdSBzdXJlLCB0aGF0IHlvdSB3b3JrZWQgd2hpbGUgeW91IHdlcmUgb24gdmFjYXRpb24/JyA9PgogICAgICAgICAgICAnU2llIGhhdHRlbiBVcmxhdWIgdW5kIGhhYmVuIGdlYXJiZWl0ZXQ/IFdpciBicmF1Y2hlbiBtZWhyIHNvbGNoZSBNaXRhcmJlaXRlci4nLAogICAgICAgICdBcmUgeW91IHN1cmUsIHRoYXQgeW91IHdvcmtlZCB3aGlsZSB5b3Ugd2VyZSBvbiBvdmVydGltZSBsZWF2ZT8nID0+CiAgICAgICAgICAgICdIYWJlbiBTaWUgd+RocmVuZCBkZXIg3GJlcnN0dW5kZW4gYXVjaCBnZWFyYmVpdGV0PycsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIGEgZGF5IGhhcyBvbmx5IDI0IGhvdXJzIScgPT4gJ0VpbiBUYWcgaGF0IG51ciAyNCBTdHVuZGVuIScsCiAgICAgICAgJ0NhblwndCBkZWxldGUgV29ya2luZyBVbml0cyEnICAgICAgPT4gJ0thbm4gQXJiZWl0c3N0dW5kZW4gbmljaHQgbPZzY2hlbiEnLAogICAgICAgICdQbGVhc2UgaW5zZXJ0IHlvdXIgd29ya2luZyBob3VycyEnID0+ICdCaXR0ZSB0cmFnZW4gU2llIElocmUgQXJiZWl0c3plaXRlbiBlaW4hJywKICAgICAgICAnWW91IGhhdmUgdG8gaW5zZXJ0IGEgc3RhcnQgYW5kIGFuIGVuZCB0aW1lIG9yIGEgcGVyaW9kJyA9PgogICAgICAgICAgICAnU2llIG38c3NlbiBCZWdpbm4tIHVuZCBFbmRlemVpdCBvZGVyIGRpZSBEYXVlciBhbmdlYmVuLicsCiAgICAgICAgJ1lvdSBjYW4gb25seSBzZWxlY3Qgb25lIGNoZWNrYm94IGVsZW1lbnQhJyA9PiAnU2llIGv2bm5lbiBudXIgZWluZSBDaGVja2JveCBtYXJraWVyZW4hJywKICAgICAgICAnRWRpdCB0aW1lIGFjY291bnRpbmcgcHJvamVjdCBzZXR0aW5ncycgPT4KICAgICAgICAgICAgJ1plaXRlcmZhc3N1bmc6IEJlYXJiZWl0dW5nIGRlciBQcm9qZWt0a29uZmlndXJhdGlvbicsCiAgICAgICAgJ1Byb2plY3Qgc2V0dGluZ3MnID0+ICdQcm9qZWt0a29uZmlndXJhdGlvbicsCiAgICAgICAgJ0lmIHlvdSBzZWxlY3QgIk1pc2NlbGxhbmVvdXMgKG1pc2MpIiB0aGUgdGFzaywgcGxlYXNlIGV4cGxhaW4gdGhpcyBpbiB0aGUgcmVtYXJrcyBmaWVsZCcgPT4KICAgICAgICAgICAgJ1dlbm4gU2llIGFscyBU5HRpZ2tlaXQgU29uc3RpZ2VzIGF1c3fkaGxlbiwgZ2ViZW4gU2llIGJpdHRlIGVpbmUgQmVzY2hyZWlidW5nIHVtIEZlbGQgQW5tZXJrdW5nIGFuLicsCiAgICAgICAgJ1BsZWFzZSBhZGQgYSByZW1hcmsgd2l0aCBtb3JlIHRoYW4gOCBjaGFyYWN0ZXJzIScgPT4KICAgICAgICAgICAgJ0JpdHRlIGdlYmVuIFNpZSBlaW5lIEFubWVya3VuZyBlaW4gZGllIGzkbmdlciBhbHMgOCBaZWljaGVuIGlzdCEnLAoKIyBGSVhNRSBhY3R1YWxseSB0aGUgZm9sbG93aW5nIHNob3VsZCBiZSBpbmNsdWRlZCBpbiBmaWxlIGRlLnBtLCBob3dldmVyIHRoZXkncmUgbm90IHNvIEkgcHV0J2VtIGhlcmUuLi4KICAgICAgICAnTW9uJyAgICAgID0+ICdNbycsCiAgICAgICAgJ1R1ZScgICAgICA9PiAnRGknLAogICAgICAgICdXZWQnICAgICAgPT4gJ01pJywKICAgICAgICAnVGh1JyAgICAgID0+ICdEbycsCiAgICAgICAgJ0ZyaScgICAgICA9PiAnRnInLAogICAgICAgICdTYXQnICAgICAgPT4gJ1NhJywKICAgICAgICAnU3VuJyAgICAgID0+ICdTbycsCiAgICAgICAgJ0phbnVhcnknICA9PiAnSmFudWFyJywKICAgICAgICAnRmVicnVhcnknID0+ICdGZWJydWFyJywKICAgICAgICAnTWFyY2gnICAgID0+ICdN5HJ6JywKICAgICAgICAnTWF5JyAgICAgID0+ICdNYWknLAogICAgICAgICdKdW5lJyAgICAgPT4gJ0p1bmknLAogICAgICAgICdKdWx5JyAgICAgPT4gJ0p1bGknLAogICAgICAgICdPY3RvYmVyJyAgPT4gJ09rdG9iZXInLAogICAgICAgICdEZWNlbWJlcicgPT4gJ0RlemVtYmVyJywKICAgIH07CiAgICByZXR1cm4gMTsKfQoKMTsK
# --
# Kernel/Modules/AgentTimeAccounting.pm - time accounting module
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AgentTimeAccounting.pm,v 1.37 2009/07/13 13:44:02 tt Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Modules::AgentTimeAccounting;

use strict;
use warnings;

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 = qw($Revision: 1.37 $) [1];

sub new {
    my ( $Type, %Param ) = @_;

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

    # check needed Objects
    for (
        qw(ParamObject DBObject ModuleReg LogObject UserObject
        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, %Param ) = @_;

    # permission check
    return 1 if !$Self->{AccessRo};

    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,
    );

    return 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, %Param ) = @_;

    my @MonthArray = (
        '',     'January', 'February', 'March',     'April',   'May',
        'June', 'July',    'August',   'September', 'October', 'November',
        'December',
    );
    my @WeekdayArray = ( 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun', );

    # ---------------------------------------------------------- #
    # delete the time accounting elements of one day
    # ---------------------------------------------------------- #
    if ( $Self->{ParamObject}->GetParam( Param => 'Delete' ) ) {
        my ( $Sec, $Min, $Hour, $Day, $Month, $Year )
            = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
            );

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

        # 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 $Output = $Self->{LayoutObject}->Header( Title => 'Delete' );
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => \%Param,
            TemplateFile => 'AgentTimeAccountingDelete'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # edit the time accounting elements
    # ---------------------------------------------------------- #
    if ( $Self->{Subaction} eq 'Edit' ) {

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

        my %Frontend   = ();
        my %Data       = ();
        my %ActionList = $Self->_ActionList();
        my ( $Sec, $Min, $Hour, $Day, $Month, $Year )
            = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
            );

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

        # 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 initial useing, the first agent with rw-right will be redirected
        # to 'Setting'. Then he can do the initial settings
        if ( !$User{ $Self->{UserID} } ) {
            return $Self->_FirstUserRedirect();
        }

        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 );

        my $ReduceTimeRef = $Self->{ConfigObject}->Get('TimeAccounting::ReduceTime');

        # Edit Working Units
        if ( $Param{Status} ) {

            ID:
            for my $ID ( 1 .. 16 ) {
                for (qw(ProjectID ActionID Remark StartTime EndTime Period)) {
                    $Param{$_} = $Self->{ParamObject}->GetParam( Param => $_ . '[' . $ID . ']' );
                }

                next ID if !$Param{ProjectID} && !$Param{ActionID};

                # create a valid period
                my $Period = $Param{Period};
                if ( $Period =~ /^(\d+),(\d+)/) {
                    $Period = $1 . "." . $2;
                }
                #allow format hh:mm
                elsif( $Param{Period} =~ /^(\d+):(\d+)/) {
                    $Period = $1 + $2/60;
                }

                my %WorkingUnit = (
                    ProjectID => $Param{ProjectID},
                    ActionID  => $Param{ActionID},
                    Remark    => $Param{Remark},
                    StartTime => $Param{StartTime},
                    EndTime   => $Param{EndTime},
                    Period    => $Period,
                );

                push @{ $Data{WorkingUnits} }, \%WorkingUnit;

                #if ($Param{StartTime} && $Param{EndTime} && !$Param{Period}) {
                #overwrite Period when Start and Endtime is given...
                next ID 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 ( $ReduceTimeRef->{ $ActionList{ $Param{ActionID} } } ) {
                            $WorkingUnit{Period} = ( $EndTime - $StartTime ) / 60
                                * $ReduceTimeRef->{ $ActionList{ $Param{ActionID} } } / 100;
                        }
                        else {
                            $WorkingUnit{Period} = ( $EndTime - $StartTime ) / 60;
                        }
                    }
                }
            }

            my $CheckboxCheck = 0;
            for my $Element (qw(LeaveDay Sick Overtime)) {
                my $Value = $Self->{ParamObject}->GetParam( Param => $Element );
                if ($Value) {
                    $Data{$Element} = 1;
                    $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!'
                );
            }

            $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} = '';
        }

        if ( time() > timelocal( 1, 0, 0, $Param{Day}, $Param{Month} - 1, $Param{Year} - 1900 ) ) {
            $Self->{LayoutObject}->Block(
                Name => 'UnitBlock',
                Data => { %Param, %Frontend },
            );
        }

        # get sick, leave day and overtime
        $Param{Sick}     = $Data{Sick}     ? 'checked' : '';
        $Param{LeaveDay} = $Data{LeaveDay} ? 'checked' : '';
        $Param{Overtime} = $Data{Overtime} ? 'checked' : '';

        $Param{Total} = $Data{Total};

        # build a working unit array
        my @Units = (undef);
        if ( $Data{WorkingUnits} ) {
            push @Units, @{ $Data{WorkingUnits} }
        }

        my $ShowAllInputFields = scalar @Units > 9 ? 1 : 0;

        # build units
        for my $ID ( 1 .. 16 ) {
            $Param{ID} = $ID;
            my $UnitRef = $Units[$ID];

            # get option fields
            $Frontend{ActionOption} = $Self->{LayoutObject}->BuildSelection(
                Data        => \%ActionList,
                SelectedID  => $UnitRef->{ActionID} || '',
                Name        => "ActionID[$ID]",
                Translation => 0,

                #Max         => 37,
                Class => 'ActionSelection',
            );

            $Frontend{ProjectOption} = $Self->_ProjectList(
                WorkingUnitID => $ID,
                SelectedID    => $UnitRef->{ProjectID},
            );

            $Param{Remark} = $UnitRef->{Remark} || '';
            if ( $UnitRef->{ProjectID} && $UnitRef->{ActionID} ) {
                if ( $UnitRef->{Period} == 0 ) {
                    $Param{UnitRequiredDescription}
                        = 'Can\'t save settings, because of missing period!';
                }
            }

            my $Period = $UnitRef->{Period} || '';

            for (qw(StartTime EndTime)) {
                $Param{$_} = !$UnitRef->{$_} || $UnitRef->{$_} eq '00:00' ? '' : $UnitRef->{$_};
            }

            # Define if the input fields are visible or not
            $Param{Visibility} = $ShowAllInputFields || $ID < 9 ? 'visible' : 'collapse';

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

            $Self->{LayoutObject}->Block(
                Name => $Param{PeriodBlock},
                Data => {
                    TextPosition => $Param{TextPosition},
                    Period       => $Period,
                    ID           => $ID,
                },
            );

            # Validity checks start
            if (
                $UnitRef->{ProjectID}
                && $UnitRef->{ActionID}
                && $Param{Sick}
                )
            {
                $Param{ReadOnlyDescription}
                    = 'Are you sure, that you worked while you were on sick leave?';
            }
            elsif (
                $UnitRef->{ProjectID}
                && $UnitRef->{ActionID}
                && $Param{LeaveDay}
                )
            {
                $Param{ReadOnlyDescription}
                    = 'Are you sure, that you worked while you were on vacation?';
            }
            elsif (
                $UnitRef->{ProjectID}
                && $UnitRef->{ActionID}
                && $Param{Overtime}
                )
            {
                $Param{ReadOnlyDescription}
                    = 'Are you sure, that you worked while you were on overtime leave?';
            }
            if ( $UnitRef->{ProjectID} && !$UnitRef->{ActionID} ) {
                $Param{UnitRequiredDescription}
                    = 'Can\'t save settings, because of missing task!';
            }
            if ( !$UnitRef->{ProjectID} && $UnitRef->{ActionID} ) {
                $Param{UnitRequiredDescription}
                    = 'Can\'t save settings, because of missing project!';
            }
            if (
                $UnitRef->{StartTime}
                && $UnitRef->{StartTime} ne '00:00'
                && $UnitRef->{EndTime}
                && $UnitRef->{EndTime} ne '00:00'
                )
            {
                if ( $UnitRef->{StartTime} =~ /^(\d+):(\d+)/ ) {
                    my $StartTime = $1 * 60 + $2;
                    if ( $UnitRef->{EndTime} =~ /^(\d+):(\d+)/ ) {
                        my $EndTime = $1 * 60 + $2;
                        if (
                            $UnitRef->{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} },
                );

                # REMARK: don't delete all working units
                # REMARK: better would be to delete only incomplete working units
                #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'
                    },
                );
            }
        }
        for my $YearID ( sort keys %{ $IncompleteWorkingDays{Incomplete} } ) {
            for my $MonthID ( sort keys %{ $IncompleteWorkingDays{Incomplete}{$YearID} } ) {
                for 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 %UserData = $Self->{TimeAccountingObject}->UserGet(
            UserID => $Self->{UserID},
        );

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

        $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 ];

        # integrate the handling for required remarks in relation to
        # projects
        $Param{RemarkRegExp} = $Self->_Project2RemarkRegExp();

        $Param{LinkVisibility} = $ShowAllInputFields ? 'collapse' : 'visible';

        # build output
        my $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 %UserData = $Self->{TimeAccountingObject}->UserGet(
                UserID => $Self->{UserID},
            );
            if ( $UserData{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
        return $Self->{LayoutObject}->NoPermission( WithHeader => 'yes' ) if !$Self->{AccessRo};

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

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

        # if no UserID posted use the current user
        $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} = $Data{Date};

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

        # get sick, leave day and overtime
        $Param{Sick}     = $Data{Sick}     ? 'checked' : '';
        $Param{LeaveDay} = $Data{LeaveDay} ? 'checked' : '';
        $Param{Overtime} = $Data{Overtime} ? 'checked' : '';

        # only show the unit block if there is some data
        my $UnitsRef = $Data{WorkingUnits};
        if ( $UnitsRef->[0] ) {
            $Self->{LayoutObject}->Block( Name => 'UnitBlock', );

            for my $UnitRef ( @{$UnitsRef} ) {

                $Self->{LayoutObject}->Block(
                    Name => 'Unit',
                    Data => {
                        Project   => $Project{Project}{ $UnitRef->{ProjectID} },
                        Action    => $Action{ $UnitRef->{ActionID} }{Action},
                        Remark    => $UnitRef->{Remark},
                        StartTime => $UnitRef->{StartTime},
                        EndTime   => $UnitRef->{EndTime},
                        Period    => $UnitRef->{Period},
                        }
                );
            }

            $Self->{LayoutObject}->Block(
                Name => 'Total',
                Data => { Total => sprintf( "%.2f", $Data{Total} ) }
            );
        }

        if ( $Param{Sick} || $Param{LeaveDay} || $Param{Overtime} ) {
            $Self->{LayoutObject}->Block(
                Name => 'OtherTimes',
                Data => {
                    Sick     => $Param{Sick},
                    LeaveDay => $Param{LeaveDay},
                    Overtime => $Param{Overtime},
                    }
            );
        }

        my %UserData = $Self->{TimeAccountingObject}->UserGet(
            UserID => $Param{UserID},
        );

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

        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' ) {
        for (qw(Day Month Year)) {
            $Param{$_} = $Self->{ParamObject}->GetParam( Param => $_ );
        }

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

        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 ( $Sec, $Min, $Hour, $CurrentDay, $Month, $Year )
            = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
            );

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

        for (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} );

        my %UserData = $Self->{TimeAccountingObject}->UserGet(
            UserID => $Param{UserID},
        );

        for my $Day ( 1 .. $DaysOfMonth ) {
            $Param{Day} = sprintf( "%02d", $Day );
            $Param{Weekday} = Day_of_Week( $Param{Year}, $Param{Month}, $Day ) - 1;
            my $VacationCheck = $Self->{TimeObject}->VacationCheck(
                Year     => $Param{Year},
                Month    => $Param{Month},
                Day      => $Day,
                Calendar => $UserData{Calendar},
            );

            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},
            );

            $Param{Comment} = $Data{Sick}
                ? 'Sick leave'
                : $Data{LeaveDay} ? 'On vacation'
                : $Data{Overtime} ? 'On overtime leave'
                :                   '';

            $Param{WorkingHours} = $Data{Total} ? sprintf( "%.2f", $Data{Total} ) : '';

            $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},
        );
        for (
            qw(TargetState TargetStateTotal WorkingHoursTotal WorkingHours
            Overtime OvertimeTotal OvertimeUntil LeaveDay LeaveDayTotal
            LeaveDayRemaining Sick SickTotal SickRemaining)
            )
        {
            $UserReport{ $Param{UserID} }{$_} ||= 0;
            $Param{$_} = sprintf( "%.2f", $UserReport{ $Param{UserID} }{$_} );
        }

        if ( $UserData{ShowOvertime} ) {
            $Self->{LayoutObject}->Block(
                Name => 'Overtime',
                Data => \%Param,
            );
        }

        # 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';
        }

        PROJECTID:
        for my $ProjectID (
            sort { $ProjectData{$a}{Name} cmp $ProjectData{$b}{Name} }
            keys %ProjectData
            )
        {
            my $ProjectRef = $ProjectData{$ProjectID};
            my $ActionsRef = $ProjectRef->{Actions};

            $Param{Project} = '';
            $Param{Class}   = 'contentvalue';
            $Param{Status}  = $ProjectRef->{Status} ? '' : 'passiv';

            my $Total      = 0;
            my $TotalTotal = 0;

            next PROJECTID if $Param{ProjectStatusShow} eq 'all' && $Param{Status};

            for my $ActionID (
                sort { $ActionsRef->{$a}{Name} cmp $ActionsRef->{$b}{Name} }
                keys %{$ActionsRef}
                )
            {
                my $ActionRef = $ActionsRef->{$ActionID};

                $Param{Action}     = $ActionRef->{Name};
                $Param{Hours}      = sprintf( "%.2f", $ActionRef->{PerMonth} || 0 );
                $Param{HoursTotal} = sprintf( "%.2f", $ActionRef->{Total} || 0 );
                $Total      += $Param{Hours};
                $TotalTotal += $Param{HoursTotal};
                $Self->{LayoutObject}->Block(
                    Name => 'Action',
                    Data => {%Param},
                );
                if ( !$Param{Project} ) {
                    $Param{Project} = $ProjectRef->{Name};
                    my $ProjectDescription = $Self->{LayoutObject}->Ascii2Html(
                        Text           => $ProjectRef->{Description},
                        HTMLResultMode => 1,
                        NewLine        => 50,
                    );

                    $Self->{LayoutObject}->Block(
                        Name => 'Project',
                        Data => {
                            RowSpan            => ( 1 + scalar keys %{$ActionsRef} ),
                            Status             => $Param{Status},
                            ProjectDescription => $ProjectDescription,
                        },
                    );

                    if ( $UserData{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   => $ProjectRef->{Name},
                                ProjectID => $ProjectID,
                            },
                        );
                    }
                    else {
                        $Self->{LayoutObject}->Block(
                            Name => 'ProjectNoLink',
                            Data => { Project => $ProjectRef->{Name} },
                        );
                    }
                }
            }
            $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},
            );
            $Param{Class} = '';
        }
        if ( defined( $Param{TotalHours} ) ) {
            $Param{TotalHours} = sprintf( "%.2f", $Param{TotalHours} );
        }
        if ( defined( $Param{TotalHoursTotal} ) ) {
            $Param{TotalHoursTotal} = sprintf( "%.2f", $Param{TotalHoursTotal} );
        }

        # build output
        my $Output = $Self->{LayoutObject}->Header( Title => 'Overview' );
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => \%Param,
            TemplateFile => 'AgentTimeAccountingOverview'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # settings for handling time accounting
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'Setting' ) {
        my %Data = ();

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

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

        $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} ) {
            my %Action      = $Self->{TimeAccountingObject}->ActionSettingsGet();
            my $ActionEmpty = 0;
            my %ActionCheck = ();
            for my $ActionID ( keys %Action ) {
                for (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!'
                    );
                }

                $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 {
            my %User = $Self->{TimeAccountingObject}->UserSettingsGet();

            my %LastPeriod = ();
            for my $UserID ( keys %User ) {
                $LastPeriod{$UserID} = 0;
                for 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} ) {

                my %UserBasics = $Self->{TimeAccountingObject}->UserList();

                USERID:
                for my $UserID ( keys %User ) {

                    for (qw( ShowOvertime CreateProject Calendar )) {
                        $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;

                    for my $Period ( keys %{ $User{$UserID} } ) {

                        # the following is because of deactivate user
                        if (
                            !defined $Self->{ParamObject}->GetParam(
                                Param => 'DateStart[' . $UserID . '][' . $Period . ']'
                            )
                            )
                        {
                            delete $Data{$UserID};
                            next USERID;
                        }

                        for (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',
                    );
                    for ( 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
        my %StatusList = (
            1 => 'valid',
            0 => 'invalid',
        );

        # Show action data
        my %Action      = $Self->{TimeAccountingObject}->ActionSettingsGet();
        my $ActionEmpty = 0;

        for my $ActionID ( sort { $Action{$a}{Action} cmp $Action{$b}{Action} } keys %Action ) {
            $Param{Action}   = $Action{$ActionID}{Action};
            $Param{ActionID} = $ActionID;

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

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

        # Show user data
        my %User       = $Self->{TimeAccountingObject}->UserSettingsGet();
        my %UserBasics = $Self->{TimeAccountingObject}->UserList();
        my %ShownUsers = $Self->{UserObject}->UserList( Type => 'Long', Valid => 1 );

        # fill up the calendar list
        my $CalendarListRef = { 0 => 'Default' };
        for my $Calendar ( 1 .. 9 ) {
            my $CalendarName = "TimeZone::Calendar${Calendar}Name";
            $CalendarListRef->{$Calendar} = $Self->{ConfigObject}->Get($CalendarName);
        }

        USERID:
        for my $UserID ( sort { $ShownUsers{$a} cmp $ShownUsers{$b} } keys %ShownUsers ) {
            next USERID if !$User{$UserID};

            $Param{User}   = $ShownUsers{$UserID};
            $Param{UserID} = $UserID;
            my $UserRef       = $User{$UserID};
            my $UserBasicsRef = $UserBasics{$UserID};

            my $Description = $Self->{LayoutObject}->Ascii2Html(
                Text           => $UserBasicsRef->{Description},
                HTMLResultMode => 1,
                NewLine        => 50,
            );

            $Description = $Description ? $Description . '<br>' : '';

            my $CalendarOption = $Self->{LayoutObject}->BuildSelection(
                Data        => $CalendarListRef,
                Name        => "Calendar[$UserID]",
                Translation => 0,
                SelectedID  => $UserBasicsRef->{Calendar} || 0,
            );

            $Self->{LayoutObject}->Block(
                Name => 'User',
                Data => {
                    %Param,
                    Description    => $Description,
                    ShowOvertime   => $UserBasicsRef->{ShowOvertime} ? 'checked' : '',
                    CreateProject  => $UserBasicsRef->{CreateProject} ? 'checked' : '',
                    CalendarOption => $CalendarOption,
                },
            );

            delete $ShownUsers{$UserID};

            for my $Period ( sort keys %{$UserRef} ) {

                my $StatusOption = $Self->{LayoutObject}->BuildSelection(
                    Data       => \%StatusList,
                    SelectedID => $UserRef->{$Period}{UserStatus},
                    Name       => "UserStatus[$UserID][$Period]",
                );

                $Self->{LayoutObject}->Block(
                    Name => 'Period',
                    Data => {
                        UserID       => $UserID,
                        WeeklyHours  => $UserRef->{$Period}{WeeklyHours},
                        LeaveDays    => $UserRef->{$Period}{LeaveDays},
                        Overtime     => $UserRef->{$Period}{Overtime},
                        DateStart    => $UserRef->{$Period}{DateStart},
                        DateEnd      => $UserRef->{$Period}{DateEnd},
                        Period       => $Period,
                        StatusOption => $StatusOption,
                    },
                );
            }
        }

        if (%ShownUsers) {
            $ShownUsers{''} = '';
            my $NewUserOption = $Self->{LayoutObject}->BuildSelection(
                Data        => \%ShownUsers,
                SelectedID  => '',
                Name        => 'NewUserID',
                Translation => 0,
            );
            $Self->{LayoutObject}->Block(
                Name => 'NewUserOption',
                Data => { NewUserOption => $NewUserOption, },
            );
        }

        # build output
        my $Output = $Self->{LayoutObject}->Header( Title => 'Setting' );
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => \%Param,
            TemplateFile => 'AgentTimeAccountingSetting'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # settings for handling time accounting
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'ProjectSetting' ) {
        my %Project = ();
        my %Data    = ();

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

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

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

        # Edit TimeAccounting Preferences
        if ( $Param{ActionProject} || $Param{NewProject} ) {
            %Project = $Self->{TimeAccountingObject}->ProjectSettingsGet();
            my $ProjectEmpty = 0;
            my %ProjectCheck = ();
            for my $ProjectID ( keys %{ $Project{Project} } ) {
                for (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
        my %StatusList = (
            1 => 'valid',
            0 => 'invalid',
        );

        # Show project data
        %Project = $Self->{TimeAccountingObject}->ProjectSettingsGet();
        my $ProjectEmpty = 0;
        for 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;

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

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

        # build output
        my $Output = $Self->{LayoutObject}->Header( Title => 'Setting' );
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            Data         => \%Param,
            TemplateFile => 'AgentTimeAccountingSetting'
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # time accounting reporting
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'Reporting' ) {
        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
        return $Self->{LayoutObject}->NoPermission( WithHeader => 'yes' ) if !$Self->{AccessRw};

        for (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 .. 12 ) {
            $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 = ( 2005 .. $Year );

        $Frontend{YearOption} = $Self->{LayoutObject}->BuildSelection(
            Data        => \@Year,
            SelectedID  => $Param{Year} || '',
            Name        => 'Year',
            Translation => 0,
        );

        ( $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}->UserList();

        USERID:
        for my $UserID ( sort { $ShownUsers{$a} cmp $ShownUsers{$b} } keys %ShownUsers ) {
            next USERID if !$UserReport{$UserID};

            for (qw(LeaveDay Overtime WorkingHours Sick 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 },
            );
        }

        for (
            qw(TotalLeaveDay TotalOvertime TotalWorkingHours
            TotalSick 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},
        );

        # REMARK:merge this projectreporting list with the list in overview

        PROJECTID:
        for my $ProjectID (
            sort { $ProjectData{$a}{Name} cmp $ProjectData{$b}{Name} }
            keys %ProjectData
            )
        {
            my $ProjectRef = $ProjectData{$ProjectID};
            my $ActionsRef = $ProjectRef->{Actions};

            $Param{Project} = '';
            $Param{Class}   = 'contentvalue';
            $Param{Status}  = $ProjectRef->{Status} ? '' : 'passiv';

            my $Total      = 0;
            my $TotalTotal = 0;

            next PROJECTID if $Param{ProjectStatusShow} eq 'all' && $Param{Status};

            for my $ActionID (
                sort { $ActionsRef->{$a}{Name} cmp $ActionsRef->{$b}{Name} }
                keys %{$ActionsRef}
                )
            {
                my $ActionRef = $ActionsRef->{$ActionID};

                $Param{Action}     = $ActionRef->{Name};
                $Param{Hours}      = sprintf( "%.2f", $ActionRef->{PerMonth} || 0 );
                $Param{HoursTotal} = sprintf( "%.2f", $ActionRef->{Total} || 0 );
                $Total      += $Param{Hours};
                $TotalTotal += $Param{HoursTotal};
                $Self->{LayoutObject}->Block(
                    Name => 'Action',
                    Data => {%Param},
                );

                if ( !$Param{Project} ) {
                    $Param{Project} = $ProjectRef->{Name};
                    my $ProjectDescription = $Self->{LayoutObject}->Ascii2Html(
                        Text           => $ProjectRef->{Description},
                        HTMLResultMode => 1,
                        NewLine        => 50,
                    );

                    $Self->{LayoutObject}->Block(
                        Name => 'Project',
                        Data => {
                            RowSpan            => ( 1 + scalar keys %{$ActionsRef} ),
                            Status             => $Param{Status},
                            ProjectDescription => $ProjectDescription,
                            Project            => $ProjectRef->{Name},
                            ProjectID          => $ProjectID,
                        },
                    );
                }
            }

            $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{TotalHours}      ||= 0;
        $Param{TotalHoursTotal} ||= 0;

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

        # build output
        my $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 %Frontend = ();

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

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

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

        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
        for my $UserID ( keys %ShownUsers ) {

            # Overview per project and action
            # REMARK: This is the wrong function to get this information
            %ProjectData = $Self->{TimeAccountingObject}->ProjectActionReporting(
                Year   => $Year,
                Month  => $Month,
                UserID => $UserID,
            );
            if ( $ProjectData{ $Param{ProjectID} } ) {
                my $ActionsRef = $ProjectData{ $Param{ProjectID} }{Actions};
                for my $ActionID ( keys %{$ActionsRef} ) {
                    $ProjectTime{$ActionID}{$UserID}{Hours} = $ActionsRef->{$ActionID}{Total};
                }
            }
            else {
                delete $ShownUsers{$UserID};
            }
        }

        # show the headerline
        for 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 = ();
        for my $ActionID ( keys %ProjectTime ) {
            $NewAction{$ActionID} = $Action{$ActionID}{Action};
        }
        %Action = %NewAction;

        # show the results
        my %Total = ();
        for my $ActionID ( sort { $Action{$a} cmp $Action{$b} } keys %Action ) {
            my $TotalHours = 0;
            $Self->{LayoutObject}->Block(
                Name => 'Action',
                Data => { Action => $Action{$ActionID}, },
            );
            for my $UserID ( sort { $ShownUsers{$a} cmp $ShownUsers{$b} } keys %ShownUsers ) {
                $TotalHours     += $ProjectTime{$ActionID}{$UserID}{Hours} || 0;
                $Total{$UserID} += $ProjectTime{$ActionID}{$UserID}{Hours} || 0;
                $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;
        for 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} );

        my @ProjectHistoryArray = $Self->{TimeAccountingObject}->ProjectHistory(
            ProjectID => $Param{ProjectID},
        );
        for my $Row (@ProjectHistoryArray) {
            $Self->{LayoutObject}->Block(
                Name => 'Row',
                Data => {
                    User   => $Row->{User},
                    Action => $Row->{Action},
                    Remark => $Row->{Remark} || '--',
                    Period => $Row->{Period},
                    Date   => $Row->{Date},
                    }
            );
        }

        # show the total sum of hours at the end of the history list
        # I also can use $Param{TotalAll}
        my $ProjectTotalHours = $Self->{TimeAccountingObject}->ProjectTotalHours(
            ProjectID => $Param{ProjectID},
        );

        $Self->{LayoutObject}->Block(
            Name => 'HistoryTotal',
            Data => {
                HistoryTotal => $ProjectTotalHours || 0,
                }
        );

        # build output
        my $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!' );
}

sub _FirstUserRedirect {
    my $Self = shift;

    # for initial useing, the first agent with rw-right will be redirected
    # to 'Setting'. Then he can do the initial settings

    my %GroupList = $Self->{GroupObject}->GroupMemberList(
        UserID => $Self->{UserID},
        Type   => 'rw',
        Result => 'HASH',
    );
    for ( 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!"
    );
}

sub _ActionList {
    my $Self = shift;

    my %ActionList;
    my %Action = $Self->{TimeAccountingObject}->ActionSettingsGet();

    # get action settings
    ACTIONID:
    for my $ActionID ( keys %Action ) {
        next ACTIONID if !$Action{$ActionID}{ActionStatus};
        next ACTIONID if !$Action{$ActionID}{Action};
        $ActionList{$ActionID} = $Action{$ActionID}{Action};
    }
    $ActionList{''} = '';

    return %ActionList;
}

sub _ProjectList {
    my ( $Self, %Param ) = @_;

    # check needed param
    if ( !$Param{WorkingUnitID} ) {
        $Self->{LayoutObject}->ErrorScreen(
            Message => '_ProjectList: Need WorkingUnitID',
        );
    }

    # at first a empty line
    my @List = (
        {
            Key   => '',
            Value => '',
        },
    );

    # get project settings
    my %Project = $Self->{TimeAccountingObject}->ProjectSettingsGet(
        Status => 'valid',
    );

    if ( !$Self->{LastProjectsRef} ) {

        # get the last projects
        my @LastProjects = $Self->{TimeAccountingObject}->LastProjectsOfUser();

        # add the favorits
        %{ $Self->{LastProjectsRef} } = map { $_ => 1 } @LastProjects;
    }

    PROJECTID:
    for my $ProjectID (
        sort { $Project{Project}{$a} cmp $Project{Project}{$b} }
        keys %{ $Project{Project} }
        )
    {
        next PROJECTID if !$Self->{LastProjectsRef}->{$ProjectID};
        my %Hash = (
            Key   => $ProjectID,
            Value => $Project{Project}{$ProjectID},
        );
        push @List, \%Hash;

        # at the moment it is not possilbe mark the selected project
        # in the favorit list (I think a bug in Build selection?!)
    }

    # add the seperator
    push @List, {
        Key      => '0',
        Value    => '--------------------',
        Disabled => 1,
    };

    # add all allowed projects to the list
    PROJECTID:
    for my $ProjectID (
        sort { $Project{Project}{$a} cmp $Project{Project}{$b} }
        keys %{ $Project{Project} }
        )
    {
        next PROJECTID if !$Project{Project}{$ProjectID};
        my %Hash = (
            Key   => $ProjectID,
            Value => $Project{Project}{$ProjectID},
        );
        if ( $Param{SelectedID} && $Param{SelectedID} eq $ProjectID ) {
            $Hash{Selected} = 1;
        }

        push @List, \%Hash;
    }

    return $Self->{LayoutObject}->BuildSelection(
        Data        => \@List,
        Name        => "ProjectID[$Param{WorkingUnitID}]",
        Translation => 0,

        #Max         => 62,
        Class => 'ProjectSelection',
    );

}

# integrate the handling for required remarks in relation to projects

sub _Project2RemarkRegExp {
    my $Self = shift;

    my @Projects2Remark = ();
    my %ProjectData     = $Self->{TimeAccountingObject}->ProjectSettingsGet(
        Status => 'valid',
    );

    return '' if !$Self->{ConfigObject}->Get('TimeAccounting::Project2RemarkRegExp');

    my $Project2RemarkRegExp = $Self->{ConfigObject}->Get('TimeAccounting::Project2RemarkRegExp');

    for my $ProjectID ( keys %{ $ProjectData{Project} } ) {
        if ( $ProjectData{Project}{$ProjectID} =~ m{$Project2RemarkRegExp}smx ) {
            push @Projects2Remark, $ProjectID;
        }
    }

    return join '|', @Projects2Remark;
}
1;

# --
# AgentTimeAccountingEdit.dtl - provides HTML form for time accounting edit
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AgentTimeAccountingEdit.dtl,v 1.27 2009/07/12 17:30:41 tt Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

<script language="JavaScript" type="text/javascript">
<!--
function submit_compose() {
    for (var Counter = 1;Counter < 10; Counter++) {
        var Remark  = 'Remark['    + Counter + ']';
        var Project = 'ProjectID[' + Counter + ']';

        if ( document.compose.elements[Project].value ) {

            var RemarkCheck = project_remark_check(Project);

            if (   document.compose.elements[Project].value
                && RemarkCheck
                &&  (!document.compose.elements[Remark].value
                        || document.compose.elements[Remark].value.length < 8
                    )
            ) {
                alert('$JSText{"Please add a remark with more than 8 characters!"}');
                document.compose.elements[Remark].focus();
                return false;
            }
        }
    }
    return true;
}

function project_remark_check(Project) {

    var reg = /^($QData{"RemarkRegExp"})$/;

    if ( reg.test(document.compose.elements[Project].value) == false ) {
        return false;
    }
    return true;
}

// This funciton make the invisible input fields (9 - 16) visible

function MoreInputFields() {
    if (document.getElementById) {
        document.getElementById("MoreFields[9]").style.visibility = "visible";
        document.getElementById("MoreFields[10]").style.visibility = "visible";
        document.getElementById("MoreFields[11]").style.visibility = "visible";
        document.getElementById("MoreFields[12]").style.visibility = "visible";
        document.getElementById("MoreFields[13]").style.visibility = "visible";
        document.getElementById("MoreFields[14]").style.visibility = "visible";
        document.getElementById("MoreFields[15]").style.visibility = "visible";
        document.getElementById("MoreFields[16]").style.visibility = "visible";
        document.getElementById("Link").style.visibility = "collapse";
    }
}

//-->
</script>

<style type="text/css">
.ProjectSelection { width:450px;}
.ActionSelection { width:300px;}
</style>


<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td colspan="3" class="mainhead">
      $Env{"Box0"}$Text{"Edit time record"}$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{"Project settings"}
      </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="compose2">
                <input type="hidden" name="Action"    value="$Env{"Action"}">
                <input type="hidden" name="Subaction" value="Edit">
                <tr>
                  <td colspan="2" class="contenthead">$Text{"Date navigation"}</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="$Config{"Frontend::ImagePath"}left-big.png" alt="$Text{"One day 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="$Config{"Frontend::ImagePath"}right-big.png" alt="$Text{"Next day"}"></a>
                          &nbsp;
                        </td>
                      </tr>
                    </table>
                  </td>
                </tr>
<!-- dtl:block:IncompleteText -->
                <tr class="contentvalue">
                  <td colspan="2">
                    $Text{"Days without entries"}:
<!-- 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" name="compose">
                <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="$QData{"Year"}">
                <input type="hidden" name="Month"     value="$QData{"Month"}">
                <input type="hidden" name="Day"       value="$QData{"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{"Task"}<span class="required">*</span></td>
                        <td>$Text{"Remark"}</td>
                        <td>$Text{"Start"} (HH:MM)<span class="$QData{"ClassTime"}">*</span></td>
                        <td>$Text{"End"} (HH:MM)<span class="$QData{"ClassTime"}">*</span></td>
                        <td>$Text{"Period"}<span class="footnote">$QData{"PeriodNote"}</span></td>
                      </tr>
<!-- dtl:block:Unit -->
                      <tr class="contentvalue" id="MoreFields[$QData{"ID"}]"  style="visibility:$QData{"Visibility"}">
                        <td>$Data{"ProjectOption"}</td>
                        <td>$Data{"ActionOption"}</td>
                        <td><input type="text" name="Remark[$QData{"ID"}]"    value="$QData{"Remark"}"    size="35"></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 -->
                      <tr class="contentvalue" id="Link">
                        <td colspan="5">
                          <a href="javascript:MoreInputFields()" style="visibility:$QData{"LinkVisibility"}">
                          <img border="0" src="$Config{"Frontend::ImagePath"}down-big.png" alt="$Text{"More input fields"}">
                          </a>
                        </td>
                      </tr>
<!-- dtl:block:UnitBlock -->
                      <tr>
                        <td class="contentkey" colspan="4">
<!-- dtl:block:OtherTimes -->
                          <input type="checkbox" value="1" name="LeaveDay" $Data{"LeaveDay"}>$Text{"On vacation"}&nbsp;
                          <input type="checkbox" value="1" name="Sick"     $Data{"Sick"}>$Text{"On sick leave"}&nbsp;
                          <input type="checkbox" value="1" name="Overtime" $Data{"Overtime"}>$Text{"On overtime leave"}
<!-- 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" type="submit" value="$Text{"Submit"}" onclick="return submit_compose();">
                    <input class="button" type="submit" value="$Text{"Delete"}" name="Delete">
                  </td>
                </tr>
                <tr>
                  <td>
                    <div class="required">* $Text{"Required Field"}</div>
<!-- dtl:block:FootNote -->
                    <div class="footnote">* $Text{"You have to insert a start and an end time or a period"}</div>
<!-- dtl:block:FootNote -->
                    <div class="footnote">* $Text{"If you select "Miscellaneous (misc)" the task, please explain this in the remarks field"}</div>
                  </td>
                </tr>
              </form>
            </table>
          </td>
        </tr>
      </table>
      <br>
    </td>
  </tr>
</table>

IyAtLQojIEFnZW50VGltZUFjY291bnRpbmdWaWV3LmR0bCAtIHByb3ZpZGVzIEhUTUwgZm9yIHZpZXcgaW5zZXJ0cyBvZiBvbmUgZGF5CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBBZ2VudFRpbWVBY2NvdW50aW5nVmlldy5kdGwsdiAxLjExIDIwMDkvMDQvMDMgMTE6NDk6MjkgdHIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjEwMCUiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iMyI+CiAgPHRyPgogICAgPHRkIGNvbHNwYW49IjMiIGNsYXNzPSJtYWluaGVhZCI+CiAgICAgICRFbnZ7IkJveDAifSRUZXh0eyJWaWV3IHRpbWUgcmVjb3JkIn0kRW52eyJCb3gxIn0KICAgIDwvdGQ+CiAgPC90cj4KICA8dHI+CiAgICA8dGQgY29sc3Bhbj0iMyIgY2xhc3M9Im1lbnUiPgogICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9JEVudnsiTGFzdFNjcmVlbiJ9IiBjbGFzcz0ibWVudWl0ZW0iPiRUZXh0eyJCYWNrIn08L2E+CiAgICA8L3RkPgogIDwvdHI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluYm9keSI+CiAgICAgIDxicj4KICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjcwMCIgYWxpZ249ImNlbnRlciIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIwIj4KPCEtLSBkdGw6YmxvY2s6VXNlciAtLT4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY29sc3Bhbj0iNCIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJWaWV3IG9mICJ9ICRRRGF0YXsiVXNlciJ9PGJyPjxicj48L3RkPgogICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOlVzZXIgLS0+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkPgogICAgICAgICAgICA8dGFibGUgY2xhc3M9ImNvbnRlbnRib2R5IiBib3JkZXI9IjAiIHdpZHRoPSIxMDAlIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjQiPgogICAgICAgICAgICAgIDxmb3JtIGFjdGlvbj0iJEVudnsiQ0dJSGFuZGxlIn0iIG1ldGhvZD0icG9zdCIgZW5jdHlwZT0ibXVsdGlwYXJ0L2Zvcm0tZGF0YSIgbmFtZT0iY29tcG9zZSI+CiAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iaGlkZGVuIiBuYW1lPSJBY3Rpb24iICAgIHZhbHVlPSIkRW52eyJBY3Rpb24ifSI+CiAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iaGlkZGVuIiBuYW1lPSJTdWJhY3Rpb24iIHZhbHVlPSJWaWV3Ij4KICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJoaWRkZW4iIG5hbWU9IlVzZXJJRCIgICAgdmFsdWU9IiRRRGF0YXsiVXNlcklEIn0iPgogICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50aGVhZCI+CiAgICAgICAgICAgICAgICAgIDx0ZCBjb2xzcGFuPSIyIj4kVGV4dHsiRGF0ZSBuYXZpZ2F0aW9uIn08L3RkPgogICAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgICAgPHRkPgogICAgICAgICAgICAgICAgICAgICREYXRheyJEYXRlU2VsZWN0aW9uIn0KICAgICAgICAgICAgICAgICAgICA8aW5wdXQgY2xhc3M9ImJ1dHRvbiIgYWNjZXNza2V5PSJnIiB0eXBlPSJzdWJtaXQiIHZhbHVlPSIkVGV4dHsiU3VibWl0In0iPgogICAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgICAgICA8dGQ+CiAgICAgICAgICAgICAgICAgICAgPHRhYmxlPgogICAgICAgICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50dmFsdWUiPgogICAgICAgICAgICAgICAgICAgICAgICA8dGQ+CiAgICAgICAgICAgICAgICAgICAgICAgICAgJFFEYXRheyJEYXRlIn0mbmJzcDsKICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgPHRkPgogICAgICAgICAgICAgICAgICAgICAgICAgIDxhIGhyZWY9IiRFbnZ7IkJhc2VsaW5rIn1BY3Rpb249JEVudnsiQWN0aW9uIn0mU3ViYWN0aW9uPVZpZXcmWWVhcj0kUURhdGF7IlllYXJCYWNrIn0mTW9udGg9JFFEYXRheyJNb250aEJhY2sifSZEYXk9JFFEYXRheyJEYXlCYWNrIn0mVXNlcklEPSRRRGF0YXsiVXNlcklEIn0iPjxpbWcgYm9yZGVyPSIwIiBzcmM9IiRDb25maWd7IkZyb250ZW5kOjpJbWFnZVBhdGgifWxlZnQtYmlnLnBuZyIgYWx0PSIkVGV4dHsiT25lIGRheSBiYWNrIn0iPjwvYT4KICAgICAgICAgICAgICAgICAgICAgICAgICAmbmJzcDsKICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgPHRkPgogICAgICAgICAgICAgICAgICAgICAgICAgIDxhIGhyZWY9IiRFbnZ7IkJhc2VsaW5rIn1BY3Rpb249JEVudnsiQWN0aW9uIn0mU3ViYWN0aW9uPVZpZXcmWWVhcj0kUURhdGF7IlllYXJOZXh0In0mTW9udGg9JFFEYXRheyJNb250aE5leHQifSZEYXk9JFFEYXRheyJEYXlOZXh0In0mVXNlcklEPSRRRGF0YXsiVXNlcklEIn0iPjxpbWcgYm9yZGVyPSIwIiBzcmM9IiRDb25maWd7IkZyb250ZW5kOjpJbWFnZVBhdGgifXJpZ2h0LWJpZy5wbmciIGFsdD0iJFRleHR7Ik5leHQgZGF5In0iPjwvYT4KICAgICAgICAgICAgICAgICAgICAgICAgICAmbmJzcDsKICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgICAgICAgPC90YWJsZT4KICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnRmb290ZXIiPgogICAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iMiI+Jm5ic3A7PC90ZD4KICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgPC9mb3JtPgogICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgICA8YnI+CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSI0Ij4KICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRoZWFkIj4kVGV4dHsiRGF0ZSJ9OiAkUURhdGF7IldlZWtkYXlfdG9fVGV4dCJ9ICRRRGF0YXsiRGF0ZSJ9PC90ZD4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGJvZHkiPgogICAgICAgICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIxIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiIGFsaWduPSJjZW50ZXIiIHdpZHRoPSI5MCUiPgo8IS0tIGR0bDpibG9jazpVbml0QmxvY2sgLS0+CiAgICAgICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50a2V5Ij4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiUHJvamVjdCJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiVGFzayJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiUmVtYXJrIn08L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkPiRUZXh0eyJTdGFydCJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiRW5kIn08L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkPiRUZXh0eyJQZXJpb2QifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6VW5pdCAtLT4KICAgICAgICAgICAgICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+CiAgICAgICAgICAgICAgICAgICAgICA8dGQ+JFFEYXRheyJQcm9qZWN0In0mbmJzcDs8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkPiRRRGF0YXsiQWN0aW9uIn0mbmJzcDs8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkPiRRRGF0YXsiUmVtYXJrIn0mbmJzcDs8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGFsaWduPSJyaWdodCI+JFFEYXRheyJTdGFydFRpbWUifSZuYnNwOzwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICA8dGQgYWxpZ249InJpZ2h0Ij4kUURhdGF7IkVuZFRpbWUifSZuYnNwOzwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICA8dGQgYWxpZ249InJpZ2h0Ij4kUURhdGF7IlBlcmlvZCJ9Jm5ic3A7PC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpVbml0IC0tPgo8IS0tIGR0bDpibG9jazpVbml0QmxvY2sgLS0+CjwhLS0gZHRsOmJsb2NrOlRvdGFsIC0tPgogICAgICAgICAgICAgICAgICAgIDx0ciBjbGFzcz0iY29udGVudGtleSI+CiAgICAgICAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iNCI+CiAgICAgICAgICAgICAgICAgICAgICAgICZuYnNwOwogICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiVG90YWwifTo8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGFsaWduPSJyaWdodCI+JFFEYXRheyJUb3RhbCJ9Jm5ic3A7PC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpUb3RhbCAtLT4KPCEtLSBkdGw6YmxvY2s6T3RoZXJUaW1lcyAtLT4KICAgICAgICAgICAgICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnRrZXkiIGFsaWduPSJjZW50ZXIiPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGNvbHNwYW49IjYiPgogICAgICAgICAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iY2hlY2tib3giIHZhbHVlPSItMiIgbmFtZT0iTGVhdmVEYXkiICREYXRheyJMZWF2ZURheSJ9IGRpc2FibGVkPiRUZXh0eyJPbiB2YWNhdGlvbiJ9Jm5ic3A7CiAgICAgICAgICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJjaGVja2JveCIgdmFsdWU9Ii0xIiBuYW1lPSJTaWNrIiAkRGF0YXsiU2ljayJ9IGRpc2FibGVkPiRUZXh0eyJPbiBzaWNrIGxlYXZlIn0mbmJzcDsKICAgICAgICAgICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImNoZWNrYm94IiB2YWx1ZT0iLTMiIG5hbWU9Ik92ZXJ0aW1lIiAkRGF0YXsiT3ZlcnRpbWUifSBkaXNhYmxlZD4kVGV4dHsiT24gb3ZlcnRpbWUgbGVhdmUifQogICAgICAgICAgICAgICAgICAgICAgICAmbmJzcDsKICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6T3RoZXJUaW1lcyAtLT4KPCEtLSBkdGw6YmxvY2s6VmFjYXRpb24gLS0+CiAgICAgICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50a2V5IiBhbGlnbj0iY2VudGVyIj4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjb2xzcGFuPSI2Ij4KICAgICAgICAgICAgICAgICAgICAgICAgJFFEYXRheyJWYWNhdGlvbiJ9CiAgICAgICAgICAgICAgICAgICAgICAgICZuYnNwOwogICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpWYWNhdGlvbiAtLT4KICAgICAgICAgICAgICAgICAgPC90YWJsZT4KICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICA8dHI+PHRkIGNsYXNzPSJjb250ZW50Zm9vdGVyIj4mbmJzcDs8L3RkPjwvdHI+CiAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgIDwvdGFibGU+CiAgICAgIDxicj4KICAgIDwvdGQ+CiAgPC90cj4KPC90YWJsZT4K
# --
# AgentTimeAccountingSetting.dtl - provides HTML form for time accounting settings
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AgentTimeAccountingSetting.dtl,v 1.21 2009/04/03 11:49:29 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --
<!-- dtl:block:ProjectSetting -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Edit time accounting project settings"}$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 settings"}</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{"Comments"}</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 settings"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td class="menu">
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=ProjectSetting" class="menuitem">
      $Text{"Project settings"}
      </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{"Task settings"}</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3" align="center">
                <tr>
                  <td class="contentkey">$Text{"Task"}<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 settings"}</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{"Comments"}:</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{"Allow project creation"}
                  </td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Calendar"}:</td>
                  <td class="contentvalue">$Data{"CalendarOption"}</td>
                  <td class="contentkey"></td>
                </tr>
                <tr>
                  <td colspan="3">
                    <table border="0" cellspacing="0" cellpadding="3" align="center" width="100%">
                      <tr class="contentkey">
                        <td>$Text{"Period begin"}<span class="required">*</span></td>
                        <td>$Text{"Period end"}<span class="required">*</span></td>
                        <td>$Text{"Days of vacation"}<span class="required">*</span></td>
                        <td>$Text{"Hours per week"}</td>
                        <td>$Text{"Authorized 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{"Add time period"}"  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) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AgentTimeAccountingOverview.dtl,v 1.21 2009/04/03 11:49:29 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Time reporting monthly overview"}$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{"Total hours worked"}:</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 (Hours)"}:</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"Overtime (this month)"}:</td>
          <td class="contentvalue" align="right">$QData{"Overtime"}</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"Overtime (total)"}:</td>
          <td class="contentvalue" align="right">$QData{"OvertimeUntil"}</td>
        </tr>
        <tr class="contenthead">
          <td>$Text{"Remaining overtime leave"}:</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{"Vacation (Days)"}:</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"Vacation taken (this month)"}:</td>
          <td class="contentvalue" align="right">$QData{"LeaveDay"}</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"Vacation taken (total)"}:</td>
          <td class="contentvalue" align="right">$QData{"LeaveDayTotal"}</td>
        </tr>
        <tr class="contenthead">
          <td>$Text{"Remaining vacation"}:</td>
          <td align="right">$QData{"LeaveDayRemaining"}</td>
        </tr>
        <tr>
          <td colspan="2">&nbsp;</td>
        </tr>
        <tr class="contenthead">
          <td colspan="2">$Text{"Sick leave (Days)"}:</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"Sick leave taken (this month)"}:</td>
          <td class="contentvalue" align="right">$QData{"Sick"}</td>
        </tr>
        <tr class="contentbody">
          <td class="contentkey">&nbsp;$Text{"Sick leave taken (total)"}:</td>
          <td class="contentvalue" align="right">$QData{"SickTotal"}</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{"User's project overview"}:
            <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{"Monthly"}</td>
                <td>$Text{"Lifetime"}</td>
              </tr>
              <tr class="contentkey">
                <td>$Text{"Project"}</td>
                <td>$Text{"Task"}</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{"Grand 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) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AgentTimeAccountingReporting.dtl,v 1.17 2009/04/03 11:49:29 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Time reporting"}: $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{"Month navigation"}</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="$Config{"Frontend::ImagePath"}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="$Config{"Frontend::ImagePath"}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 reports"}:</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{"Monthly total"}</td>
                <td colspan="2">$Text{"Lifetime total"}</td>
              </tr>
              <tr class="contentkey">
                <td>$Text{"User"}</td>
                <td>$Text{"Working Hours"}</td>
                <td>$Text{"Overtime leave"}</td>
                <td>$Text{"Vacation"}</td>
                <td>$Text{"Sick leave"}</td>
                <td>$Text{"Overtime leave"}</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{"Sick"}</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{"TotalSick"}</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 reports"}:
            <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{"Monthly total"}</td>
                <td>$Text{"Lifetime total"}</td>
              </tr>
              <tr class="contentkey">
                <td>$Text{"Project"}</td>
                <td>$Text{"Task"}</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>

IyAtLQojIEFnZW50VGltZUFjY291bnRpbmdQcm9qZWN0UmVwb3J0aW5nLmR0bCAtIHByb3ZpZGVzIEhUTUwgZm9ybSBmb3IgdGltZSBhY2NvdW50aW5nIHJlcG9ydGluZwojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogQWdlbnRUaW1lQWNjb3VudGluZ1Byb2plY3RSZXBvcnRpbmcuZHRsLHYgMS43IDIwMDkvMDQvMDMgMTE6NDk6MjkgdHIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCjx0YWJsZSBib3JkZXI9IjAiIHdpZHRoPSIxMDAlIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiPgogIDx0cj4KICAgIDx0ZCBjbGFzcz0ibWFpbmhlYWQiPgogICAgICAkRW52eyJCb3gwIn0kVGV4dHsiUHJvamVjdCByZXBvcnQifTogJFFEYXRheyJQcm9qZWN0In0gJEVudnsiQm94MSJ9CiAgICA8L3RkPgogIDwvdHI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtZW51Ij4KICAgICAgPGEgaHJlZj0iJEVudnsiQmFzZWxpbmsifSRFbnZ7Ikxhc3RTY3JlZW4ifSIgY2xhc3M9Im1lbnVpdGVtIj4kVGV4dHsiQmFjayJ9PC9hPgogICAgPC90ZD4KICA8L3RyPgogIDx0cj4KICAgIDx0ZCBjbGFzcz0ibWFpbmJvZHkiPgogICAgICA8YnI+CiAgICAgIDx0YWJsZSBjbGFzcz0iY29udGVudGJvZHkiIGJvcmRlcj0iMCIgd2lkdGg9IjUwJSIgY2VsbHBhZGRpbmc9IjMiIGNlbGxzcGFjaW5nPSIwIiBhbGlnbj0iY2VudGVyIj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRoZWFkIj4KICAgICAgICAgICAgJFRleHR7IlByb2plY3QgcmVwb3J0In06ICRRRGF0YXsiUHJvamVjdCJ9CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIxIiB3aWR0aD0iMTAwJSIgYWxpZ249ImNlbnRlciI+CiAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50a2V5Ij4KICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiVGFzayJ9PC90ZD4KPCEtLSBkdGw6YmxvY2s6VXNlck5hbWUgLS0+CiAgICAgICAgICAgICAgICA8dGQ+JFFEYXRheyJVc2VyIn08L3RkPgo8IS0tIGR0bDpibG9jazpVc2VyTmFtZSAtLT4KICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiVG90YWwifTwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6QWN0aW9uIC0tPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7IkFjdGlvbiJ9PC90ZD4KPCEtLSBkdGw6YmxvY2s6VXNlciAtLT4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIiBhbGlnbj0icmlnaHQiPiRRRGF0YXsiSG91cnMifTwvdGQ+CjwhLS0gZHRsOmJsb2NrOlVzZXIgLS0+CiAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6QWN0aW9uIC0tPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IlRvdGFsIn08L3RkPgo8IS0tIGR0bDpibG9jazpVc2VyVG90YWwgLS0+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIGFsaWduPSJyaWdodCI+JFFEYXRheyJUb3RhbCJ9PC90ZD4KPCEtLSBkdGw6YmxvY2s6VXNlclRvdGFsIC0tPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5IiBhbGlnbj0icmlnaHQiPiRRRGF0YXsiVG90YWxBbGwifTwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgPC90YWJsZT4KICAgICAgICAgIDwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRmb290ZXIiPiZuYnNwOzwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgPC90YWJsZT4KICAgICAgPGJyPgogICAgICA8dGFibGUgY2xhc3M9ImNvbnRlbnRib2R5IiBib3JkZXI9IjAiIHdpZHRoPSI1MCUiIGNlbGxwYWRkaW5nPSIzIiBjZWxsc3BhY2luZz0iMCIgYWxpZ249ImNlbnRlciI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50aGVhZCI+CiAgICAgICAgICAgICRUZXh0eyJIaXN0b3J5In06ICRRRGF0YXsiUHJvamVjdCJ9CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIxIiB3aWR0aD0iMTAwJSIgYWxpZ249ImNlbnRlciI+CiAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50a2V5Ij4KICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiRGF0ZSJ9PC90ZD4KICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiVXNlciJ9PC90ZD4KICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiVGFzayJ9PC90ZD4KICAgICAgICAgICAgICAgIDx0ZD4kVGV4dHsiUmVtYXJrIn08L3RkPgogICAgICAgICAgICAgICAgPHRkPiRUZXh0eyJQZXJpb2QifTwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6Um93IC0tPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7IkRhdGUifTwvdGQ+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+JFFEYXRheyJVc2VyIn08L3RkPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiQWN0aW9uIn08L3RkPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiUmVtYXJrIn08L3RkPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiIGFsaWduPSJyaWdodCI+JFFEYXRheyJQZXJpb2QifTwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6Um93IC0tPgogICAgICAgICAgICAgIDx0cj4KPCEtLSBkdGw6YmxvY2s6SGlzdG9yeVRvdGFsIC0tPgogICAgICAgICAgICAgICAgPHRkIGNvbHNwYW49IjQiIGNsYXNzPSJjb250ZW50a2V5Ij4kVGV4dHsiVG90YWwifTwvdGQ+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIGFsaWduPSJyaWdodCI+JFFEYXRheyJIaXN0b3J5VG90YWwifTwvdGQ+CjwhLS0gZHRsOmJsb2NrOkhpc3RvcnlUb3RhbCAtLT4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGZvb3RlciI+Jm5ic3A7PC90ZD4KICAgICAgICA8L3RyPgogICAgICA8L3RhYmxlPgogICAgICA8YnI+CiAgICA8L3RkPgogIDwvdHI+CjwvdGFibGU+Cg==
IyAtLQojIEFnZW50VGltZUFjY291bnRpbmdEZWxldGUuZHRsIC0gcHJvdmlkZXMgSFRNTCBmb3JtIGZvciBkZWxldGUgYSBkYXkgb2YgdGltZWFjY291bnRpbmcKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEFnZW50VGltZUFjY291bnRpbmdEZWxldGUuZHRsLHYgMS4yIDIwMDkvMDQvMDMgMTE6NDk6MjkgdHIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCjwhLS0gc3RhcnQgZm9ybSAtLT4KPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjEwMCUiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iMyI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluaGVhZCI+CiAgICAgICRFbnZ7IkJveDAifSRUZXh0eyJEZWxldGUifTogJFRleHR7IlRpbWVBY2NvdW50aW5nIG9mIn0gJFFEYXRheyJZZWFyIn0tJFFEYXRheyJNb250aCJ9LSRRRGF0YXsiRGF5In0kRW52eyJCb3gxIn0KICAgIDwvdGQ+CiAgPC90cj4KIyAgPHRyPgojICAgIDx0ZCBjbGFzcz0ibWVudSI+CiMgICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9JEVudnsiTGFzdFNjcmVlblZpZXcifSIgY2xhc3M9Im1lbnVpdGVtIj4kVGV4dHsiQmFjayJ9PC9hPgojICAgIDwvdGQ+CiMgIDwvdHI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluYm9keSI+CiAgICAgIDxicj4KICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjM1MCIgYWxpZ249ImNlbnRlciIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSI0Ij4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY29sc3Bhbj0iMiIgY2xhc3M9ImNvbnRlbnRoZWFkIj4kVGV4dHsiT3B0aW9ucyJ9PC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGJvZHkiPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSIgID4kVGV4dHsiRGVsZXRlIn06IDwvdGQ+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+JFRleHR7IlRpbWVBY2NvdW50aW5nIG9mIn0gJFFEYXRheyJZZWFyIn0tJFFEYXRheyJNb250aCJ9LSRRRGF0YXsiRGF5In08L3RkPgogICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgPHRkIGNvbHNwYW49IjIiPiZuYnNwOzwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iMiIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJEbyB5b3UgcmVhbGx5IHdhbnQgdG8gZGVsZXRlIHRoaXMgT2JqZWN0In0/PC90ZD4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGZvb3RlciIgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+CiAgICAgICAgICAgIDxhIGhyZWY9IiRFbnZ7IkJhc2VsaW5rIn1BY3Rpb249JEVudnsiQWN0aW9uIn0mU3ViYWN0aW9uPUVkaXQmWWVhcj0kUURhdGF7IlllYXIifSZNb250aD0kUURhdGF7Ik1vbnRoIn0mRGF5PSRRRGF0YXsiRGF5In0iIj4kVGV4dHsiTm8ifTwvYT4gPGEgaHJlZj0iJEVudnsiQmFzZWxpbmsifUFjdGlvbj0kRW52eyJBY3Rpb24ifSZTdWJhY3Rpb249RGVsZXRlJlllYXI9JFFEYXRheyJZZWFyIn0mTW9udGg9JFFEYXRheyJNb250aCJ9JkRheT0kUURhdGF7IkRheSJ9Ij4kVGV4dHsiWWVzIn08L2E+CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgIDwvdGFibGU+CiAgICAgIDxicj4KICAgIDwvdGQ+CiAgPC90cj4KPC90YWJsZT4KPCEtLSBlbmQgZm9ybSAtLT4K
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9Ob3RpZmljYXRpb25UaW1lQWNjb3VudGluZy5wbQojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogTm90aWZpY2F0aW9uVGltZUFjY291bnRpbmcucG0sdiAxLjYgMjAwOS8wNC8wMyAxMTo0OToyOSB0ciBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6Tm90aWZpY2F0aW9uVGltZUFjY291bnRpbmc7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpTeXN0ZW06OlRpbWVBY2NvdW50aW5nOwp1c2UgS2VybmVsOjpTeXN0ZW06OlRpbWU7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjYgJCkgWzFdOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGdldCBuZWVkZWQgb2JqZWN0cwogICAgZm9yIChxdyhDb25maWdPYmplY3QgTG9nT2JqZWN0IERCT2JqZWN0IExheW91dE9iamVjdCBVc2VySUQpKSB7CiAgICAgICAgJFNlbGYtPnskX30gPSAkUGFyYW17JF99IHx8IGRpZSAiR290IG5vICRfISI7CiAgICB9CgogICAgJFNlbGYtPntUaW1lT2JqZWN0fSAgICAgICAgICAgPSBLZXJuZWw6OlN5c3RlbTo6VGltZS0+bmV3KCVQYXJhbSk7CiAgICAkU2VsZi0+e1RpbWVBY2NvdW50aW5nT2JqZWN0fSA9IEtlcm5lbDo6U3lzdGVtOjpUaW1lQWNjb3VudGluZy0+bmV3KCVQYXJhbSk7CiAgICByZXR1cm4gJFNlbGY7Cn0KCnN1YiBSdW4gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAoICRTZWMsICRNaW4sICRIb3VyLCAkRGF5LCAkTW9udGgsICRZZWFyICkKICAgICAgICA9ICRTZWxmLT57VGltZU9iamVjdH0tPlN5c3RlbVRpbWUyRGF0ZSggU3lzdGVtVGltZSA9PiAkU2VsZi0+e1RpbWVPYmplY3R9LT5TeXN0ZW1UaW1lKCksICk7CiAgICBteSAlVXNlciA9ICRTZWxmLT57VGltZUFjY291bnRpbmdPYmplY3R9LT5Vc2VyQ3VycmVudFBlcmlvZEdldCgKICAgICAgICBZZWFyICA9PiAkWWVhciwKICAgICAgICBNb250aCA9PiAkTW9udGgsCiAgICAgICAgRGF5ICAgPT4gJERheSwKICAgICk7CiAgICBpZiAoICRVc2VyeyAkU2VsZi0+e1VzZXJJRH0gfSApIHsKICAgICAgICBteSAlSW5jb21wbGV0ZVdvcmtpbmdEYXlzID0gJFNlbGYtPntUaW1lQWNjb3VudGluZ09iamVjdH0tPldvcmtpbmdVbml0c0NvbXBsZXRuZXNzQ2hlY2soKTsKCiAgICAgICAgIyByZWRpcmVjdCBpZiBpbmNvbXBsZXRlIHdvcmtpbmcgZGF5IGFyZSBvdXQgb2YgcmFuZ2UKICAgICAgICBpZiAoICRJbmNvbXBsZXRlV29ya2luZ0RheXN7V2FybmluZ30gKSB7CiAgICAgICAgICAgIHJldHVybiAkU2VsZi0+e0xheW91dE9iamVjdH0tPk5vdGlmeSgKICAgICAgICAgICAgICAgIEluZm8gICAgID0+ICdQbGVhc2UgaW5zZXJ0IHlvdXIgd29ya2luZyBob3VycyEnLAogICAgICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ0Vycm9yJwogICAgICAgICAgICApOwogICAgICAgIH0KICAgIH0KICAgIHJldHVybiAnJzsKfQoKMTsK
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=
# --
# TimeAccounting.pm - code to excecute during package installation
# Copyright (C) 2001-2008 OTRS AG, http://otrs.org/
# --
# $Id: TimeAccounting.pm,v 1.4 2008/08/29 05:55:26 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-2.0.txt.
# --

package var::packagesetup::TimeAccounting;

use strict;
use warnings;

use Kernel::System::Group;
use Kernel::System::Valid;

use vars qw(@ISA $VERSION);
$VERSION = qw($Revision: 1.4 $) [1];

=head1 NAME

TimeAccounting.pm - code to excecute during package installation

=head1 SYNOPSIS

All functions

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Log;
    use Kernel::System::Main;
    use Kernel::System::Time;
    use Kernel::System::DB;
    use Kernel::System::XML;

    my $ConfigObject = Kernel::Config->new();
    my $LogObject    = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
    );
    my $TimeObject = Kernel::System::Time->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $XMLObject = Kernel::System::XML->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
        MainObject   => $MainObject,
    );
    my $CodeObject = var::packagesetup::TimeAccounting->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
        TimeObject   => $TimeObject,
        DBObject     => $DBObject,
        XMLObject    => $XMLObject,
    );

=cut

sub new {
    my ( $Type, %Param ) = @_;

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

    # check needed objects
    for my $Object (qw(ConfigObject LogObject MainObject TimeObject DBObject XMLObject)) {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }
    $Self->{GroupObject}          = Kernel::System::Group->new( %{$Self} );
    $Self->{ValidObject}          = Kernel::System::Valid->new( %{$Self} );

    return $Self;
}

=item CodeInstall()

run the code install part

    my $Result = $CodeObject->CodeInstall();

=cut

sub CodeInstall {
    my ( $Self, %Param ) = @_;

    # add the group time_accounting
    $Self->_GroupAdd(
        Name        => 'time_accounting',
        Description => 'Group for all time accounting user.',
    );

    return 1;
}

=item CodeUninstall()

run the code uninstall part

    my $Result = $CodeObject->CodeUninstall();

=cut

sub CodeUninstall {
    my ( $Self, %Param ) = @_;

    # deactivate the group time_accounting
    $Self->_GroupDeactivate(
        Name => 'time_accounting',
    );

    return 1;
}

=item _GroupAdd()

add a group

    my $Result = $CodeObject->_GroupAdd(
        Name        => 'the-group-name',
        Description => 'The group description.',
    );

=cut

sub _GroupAdd {
    my ( $Self, %Param ) = @_;

    # check needed stuff
    for my $Argument (qw(Name Description)) {
        if ( !$Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # get valid list
    my %ValidList = $Self->{ValidObject}->ValidList(
        UserID => 1,
    );
    my %ValidListReverse = reverse %ValidList;

    # check if group already exists
    my $GroupID = $Self->{GroupObject}->GroupLookup(
        Group  => $Param{Name},
        UserID => 1,
    );

    # reactivate the group
    if ($GroupID) {

        # get current group data
        my %GroupData = $Self->{GroupObject}->GroupGet(
            ID     => $GroupID,
            UserID => 1,
        );

        # reactivate group
        $Self->{GroupObject}->GroupUpdate(
            %GroupData,
            ValidID => $ValidListReverse{valid},
            UserID  => 1,
        );

        return 1;
    }

    # add the group
    else {
        return if !$Self->{GroupObject}->GroupAdd(
            Name    => $Param{Name},
            Comment => $Param{Description},
            ValidID => $ValidListReverse{valid},
            UserID  => 1,
        );
    }

    # lookup the new group id
    my $NewGroupID = $Self->{GroupObject}->GroupLookup(
        Group  => $Param{Name},
        UserID => 1,
    );

    # add user root to the group
    $Self->{GroupObject}->GroupMemberAdd(
        GID        => $NewGroupID,
        UID        => 1,
        Permission => {
            ro        => 1,
            move_into => 1,
            create    => 1,
            owner     => 1,
            priority  => 1,
            rw        => 1,
        },
        UserID => 1,
    );

    return 1;
}

=item _GroupDeactivate()

deactivate a group

    my $Result = $CodeObject->_GroupDeactivate(
        Name => 'the-group-name',
    );

=cut

sub _GroupDeactivate {
    my ( $Self, %Param ) = @_;

    # check needed stuff
    if ( !$Param{Name} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Need Name!',
        );
        return;
    }

    # lookup group id
    my $GroupID = $Self->{GroupObject}->GroupLookup(
        Group => $Param{Name},
    );

    return if !$GroupID;

    # get valid list
    my %ValidList = $Self->{ValidObject}->ValidList(
        UserID => 1,
    );
    my %ValidListReverse = reverse %ValidList;

    # get current group data
    my %GroupData = $Self->{GroupObject}->GroupGet(
        ID     => $GroupID,
        UserID => 1,
    );

    # deactivate group
    $Self->{GroupObject}->GroupUpdate(
        %GroupData,
        ValidID => $ValidListReverse{invalid},
        UserID  => 1,
    );

    return 1;
}

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-2.0.txt.

=cut

=head1 VERSION

$Revision: 1.4 $ $Date: 2008/08/29 05:55:26 $

=cut
