TimeAccounting
1.4.3
OTRS AG
http://otrs.org/
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
Build for TimeAccounting 1.4.3.
Optimized DB.
First stable release.
Second beta with one timezone related bugfix.
First beta release of version 1.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>ATTENTION</b>
<br>
<br>
Make sure your database accepts packages over 5 MB in size. A MySQL database for example accepts packages up to 1 MB by default. In this case, the value for max_allowed_packet must be increased. The recommended maximum size accepted is 20 MB.<br>
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>WILLKOMMEN</b>
<br>
<br>
Sie sind im Begriff das OTRS-Paket TimeAccounting zu installieren.<br>
<br>
<br>
<b>ACHTUNG</b>
<br>
<br>
Vergewissern Sie sich, dass ihre Datenbank Pakete mit einer Größe von mehr als 5 MB akzeptiert. Eine MySQL Datenbank akzeptiert beispielsweise standardmäßig Pakete mit einer Größe von maximal 1 MB. In diesem Fall muss der Wert für max_allowed_packet erhöht werden. Empfohlen werden 20 MB.<br>
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>BIENVENIDO</b>
<br>
<br>
Ud. está a punto de instalar el paquete Contabilidad de Tiempo de OTRS.<br>
<br>
<br>
<b>NOTA</b>
<br>
<br>
Asegúrese de que su base de datos acepte paquetes de tamaño superior a 5 MB. Una base de datos MySQL, por ejemplo, acepta por defecto paquetes de 1 MB, como máximo. En este caso, el valor de max_allowed_packet debe incrementarse. El tamaño que se recomienda es de 20 MB.<br>
<br>
<br>
((enjoy))<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>NOTA</b>
<br>
<br>
Para permitirles s los usuarios el acceso al módulo de contabilidad de tiempo, es necesario añadirlos como miembros del grupo 'time_accounting'.
<br>
<br>
Los elementos del menú agregados por este paquete, serán visibles luego de reiniciar sesión en el sistema.
<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>
<br>
<b>ATTENTION</b>
<br>
<br>
Si usted desinstala este paquete, todas las tablas de la base de datos que se crearon durante la instalación del mismo, se eliminarán y
el contenido de dichas tablas se perderá irrevocablemente.
<br>
<br>
El grupo 'time_accounting' que se dió de alta durante la instalación del paquete, se desactivará.
Es posible reactivar dicho grupo en el área de administración.
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>WELCOME</b>
<br>
<br>
You are about to upgrade the OTRS package TimeAccounting.<br>
<br>
<br>
</b>ATTENTION</b>
<br>
<br>
Make sure your database accepts packages over 5 MB in size. A MySQL database for example accepts packages up to 1 MB by default. In this case, the value for max_allowed_packet must be increased. The recommended maximum size accepted is 20 MB.<br>
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>WILLKOMMEN</b>
<br>
<br>
Sie sind im Begriff das OTRS-Paket TimeAccounting zu aktualisieren.<br>
<br>
<br>
<b>ACHTUNG</b>
<br>
<br>
Vergewissern Sie sich, dass ihre Datenbank Pakete mit einer Größe von mehr als 5 MB akzeptiert. Eine MySQL Datenbank akzeptiert beispielsweise standardmäßig Pakete mit einer Größe von maximal 1 MB. In diesem Fall muss der Wert für max_allowed_packet erhöht werden. Empfohlen werden 20 MB.<br>
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>BIENVENIDO</b>
<br>
<br>
Ud. está a punto de actualizar el paquete Contabilidad de Tiempo de OTRS.<br>
<br>
<br>
<b>NOTA</b>
<br>
<br>
Asegúrese de que su base de datos acepte paquetes de tamaño superior a 5 MB. Una base de datos MySQL, por ejemplo, acepta por defecto paquetes de 1 MB, como máximo. En este caso, el valor de max_allowed_packet debe incrementarse. El tamaño que se recomienda es de 20 MB.<br>
<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."
);
}
}
2017-10-09 16:01:09
opms.otrs.com
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iQXBwbGljYXRpb24iPgogICAgPENWUz4kSWQ6IFRpbWVBY2NvdW50aW5nLnhtbCx2IDEuMTQuMi4xIDIwMTAtMTEtMjYgMTM6NTE6NDMgbW4gRXhwICQ8L0NWUz4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZyb250ZW5kOjpNb2R1bGUjIyNBZ2VudFRpbWVBY2NvdW50aW5nIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RnJvbnRlbmRNb2R1bGVSZWdpc3RyYXRpb24gZm9yIFRpbWVBY2NvdW50aW5nTW9kdWxlICB0byBkZWZpbmUgdGhlIHJlZ2lzdHJhdGlvbiBzZXR0aW5ncyBpbiB0aGUgcHJvZ3JhbSBhbmQgdG8gc2V0IHJpZ2h0cyBhbmQgcG9zaXRpb25zIG9mIGljb25zIGluIHRoZSBuYXZpZ2F0aW9uIGJhci48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RnJvbnRlbmRNb2R1bFJlZ2lzdHJhdGlvbiBm/HIgZGFzIFplaXRlcmZhc3N1bmdzTW9kdWwuICBIaWVyCmv2bm5lbiBTaWUgZGllIEFuZ2FiZW4gZvxyIGRpZSBSZWdpc3RyaWVydW5nIGRlcyBNb2R1bHMgaW0gUHJvZ3JhbW0gdOR0aWdlbi4gRmVybmVyCmv2bm5lbiBkaWUgUmVjaHRlIHVuZCBkaWUgUG9zaXRpb25lbiBkZXIgSWNvbnMgaW4gZGVyIE5hdmlnYXRpb25zbGVpc3RlIGdlc2V0enQgd2VyZGVuLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50OjpNb2R1bGVSZWdpc3RyYXRpb248L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8RnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgICAgICAgICA8R3JvdXA+dGltZV9hY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICAgICAgICAgIDxHcm91cFJvPnRpbWVfYWNjb3VudGluZzwvR3JvdXBSbz4KICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5UaW1lQWNjb3VudGluZzwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICA8TmF2QmFyTmFtZT5UaW1lQWNjb3VudGluZzwvTmF2QmFyTmFtZT4KICAgICAgICAgICAgICAgIDxUaXRsZT5UaW1lQWNjb3VudGluZzwvVGl0bGU+CiAgICAgICAgICAgICAgICA8TmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5UaW1lQWNjb3VudGluZzwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+VGltZUFjY291bnRpbmc8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPnRpbWVfYWNjb3VudGluZy5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudFRpbWVBY2NvdW50aW5nJmFtcDtTdWJhY3Rpb249RWRpdDwvTGluaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPlRpbWVBY2NvdW50aW5nPC9OYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPFR5cGU+TWVudTwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8UHJpbz42MDAwPC9QcmlvPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+PC9BY2Nlc3NLZXk+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPkl0ZW1BcmVhPC9CbG9jaz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgIDxHcm91cFJvPnRpbWVfYWNjb3VudGluZzwvR3JvdXBSbz4KICAgICAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+T3ZlcnZpZXc8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPk92ZXJ2aWV3PC9OYW1lPgogICAgICAgICAgICAgICAgICAgIDxJbWFnZT5vdmVydmlldy5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudFRpbWVBY2NvdW50aW5nJmFtcDtTdWJhY3Rpb249T3ZlcnZpZXc8L0xpbms+CiAgICAgICAgICAgICAgICAgICAgPFR5cGU+PC9UeXBlPgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz48L0Jsb2NrPgogICAgICAgICAgICAgICAgICAgIDxOYXZCYXI+VGltZUFjY291bnRpbmc8L05hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8QWNjZXNzS2V5Pm88L0FjY2Vzc0tleT4KICAgICAgICAgICAgICAgICAgICA8UHJpbz4xMDA8L1ByaW8+CiAgICAgICAgICAgICAgICA8L05hdkJhcj4KICAgICAgICAgICAgICAgIDxOYXZCYXI+CiAgICAgICAgICAgICAgICA8R3JvdXBSbz50aW1lX2FjY291bnRpbmc8L0dyb3VwUm8+CiAgICAgICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPkVkaXQgdGltZSByZWNvcmQ8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPkVkaXQ8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPm5ldy5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudFRpbWVBY2NvdW50aW5nJmFtcDtTdWJhY3Rpb249RWRpdDwvTGluaz4KICAgICAgICAgICAgICAgICAgICA8VHlwZT48L1R5cGU+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPjwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPE5hdkJhcj5UaW1lQWNjb3VudGluZzwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+bjwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjIwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgIDxHcm91cD50aW1lX2FjY291bnRpbmc8L0dyb3VwPgogICAgICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5Qcm9qZWN0IHRpbWUgcmVwb3J0aW5nPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgICAgICA8TmFtZT5SZXBvcnRpbmc8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPnJlcG9ydGluZy5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudFRpbWVBY2NvdW50aW5nJmFtcDtTdWJhY3Rpb249UmVwb3J0aW5nPC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPlRpbWVBY2NvdW50aW5nPC9OYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEFjY2Vzc0tleT48L0FjY2Vzc0tleT4KICAgICAgICAgICAgICAgICAgICA8UHJpbz41MDA8L1ByaW8+CiAgICAgICAgICAgICAgICA8L05hdkJhcj4KICAgICAgICAgICAgICAgIDxOYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEdyb3VwPnRpbWVfYWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPkVkaXQgdGltZSBhY2NvdW50aW5nIHNldHRpbmdzPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgICAgICA8TmFtZT5TZXR0aW5nPC9OYW1lPgogICAgICAgICAgICAgICAgICAgIDxJbWFnZT5tb2R1bGVfc2V0dGluZy5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudFRpbWVBY2NvdW50aW5nJmFtcDtTdWJhY3Rpb249U2V0dGluZzwvTGluaz4KICAgICAgICAgICAgICAgICAgICA8VHlwZT48L1R5cGU+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPjwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPE5hdkJhcj5UaW1lQWNjb3VudGluZzwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+PC9BY2Nlc3NLZXk+CiAgICAgICAgICAgICAgICAgICAgPFByaW8+OTAwPC9QcmlvPgogICAgICAgICAgICAgICAgPC9OYXZCYXI+CiAgICAgICAgICAgIDwvRnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRQcm9qZWN0TmFtZSIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgbmFtZSBmb3IgbmV3IHByb2plY3RzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5Jbml0aWFsZXIgUHJvamVrdG5hbWUgYmVpbSBFcnN0ZWxsZW4gZWluZXMgbmV1ZW4gUHJvamVrdHMgaW4gZGVyIFplaXRlcmZhc3N1bmcuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0UHJvamVjdFN0YXR1cyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgc3RhdHVzIG9mIGFuIG5ldyBwcm9qZWN0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5BbmZhbmdzc3RhdHVzIGVpbmVzIG5ldWVuIFByb2pla3RzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPE9wdGlvbiBTZWxlY3RlZElEPSIxIj4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMCI+aW52YWxpZDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMSI+dmFsaWQ8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0QWN0aW9uTmFtZSIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgbmFtZSBmb3IgbmV3IGFjdGlvbnMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkluaXRpYWxlciBOYW1lIGVpbmVyIG5ldWVuIFTkdGlna2VpdCBpbiBkZXIgWmVpdGVyZmFzc3VuZy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+PC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRBY3Rpb25TdGF0dXMiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZhdWx0IHN0YXR1cyBvZiBhbiBuZXcgYWN0aW9uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5BbmZhbmdzc3RhdHVzIGVpbmVyIG5ldWVuIFTkdGlna2VpdC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxPcHRpb24gU2VsZWN0ZWRJRD0iMSI+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjAiPmludmFsaWQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjEiPnZhbGlkPC9JdGVtPgogICAgICAgICAgICA8L09wdGlvbj4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdFVzZXJXZWVrbHlIb3VycyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgc2V0dGluZyBmb3IgdGhlIHN0YW5kYXJkIHdlZWtseSBob3Vycy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+U3RhbmRhcmQgV29jaGVuYXJiZWl0c3plaXQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjQwPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRVc2VyTGVhdmVEYXlzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzZXR0aW5nIGZvciBsZWF2ZSBkYXlzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5TdGFuZGFyZCBVcmxhdWJzdGFnZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+MjM8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdFVzZXJPdmVydGltZSIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgc2V0dGluZyBmb3Igb3ZlcnRpbWUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkV2ZW50dWVsbCB2b3JoYW5kZW5lciDcYmVyc3R1bmRlbvxiZXJ0cmFnLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZyBSZWdleD0iIj4wPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OkRlZmF1bHRVc2VyRGF0ZVN0YXJ0IiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzZXR0aW5nIGZvciBkYXRlIHN0YXJ0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5TdGFydGRhdHVtIGb8ciBkaWUgRWluZ2FiZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjIwMTAtMDEtMDE8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6RGVmYXVsdFVzZXJEYXRlRW5kIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzZXR0aW5nIGZvciBkYXRlIGVuZC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RW5kZGF0dW0gZvxyIGRpZSBFaW5nYWJlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+MjAxMC0xMi0zMTwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpEZWZhdWx0VXNlclN0YXR1cyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgc3RhdHVzIG9mIGFuIG5ldyB1c2VyLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5BbmZhbmdzc3RhdHVzIGVpbmVzIG5ldWVuIEJlbnV0emVycy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxPcHRpb24gU2VsZWN0ZWRJRD0iMSI+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjAiPmludmFsaWQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjEiPnZhbGlkPC9JdGVtPgogICAgICAgICAgICA8L09wdGlvbj4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6TWF4SW50ZXJ2YWxPZkluY29tcGxldGVEYXlzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+V29ya2luZyBkYXlzIGFmdGVyIGF0IHRoZSBsYXRlc3Qgd29ya2luZyB1bml0cyBoYXZlIHRvIGluc2VydC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+TGVndCBmZXN0LCBuYWNoIHdpZXZpZWxlbiBXZXJrdGFnZW4gc3DkdGVzdGVucyBkaWUgQXJiZWl0c3N0dW5kZW4gZWluZ2V0cmFnZW4gd2VyZGVuIG38c3Nlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+NDwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpNYXhJbnRlcnZhbE9mSW5jb21wbGV0ZURheXNCZWZvcmVXYXJuaW5nIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+U2hvdyB3YXJuaW5nLCBpZiB0byBtdWNoIGluY29tcGxldGUgd29ya2luZyBkYXlzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5aZWlndCBlaW5lIFdhcm51bmcsIHdlbm4genV2aWVsZSBBcmJlaXRzdGFnZSBuaWNodCBlaW5nZXRyYWdlbiB3dXJkZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjI8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGcm9udGVuZDo6Tm90aWZ5TW9kdWxlIyMjODg4LVRpbWVBY2NvdW50aW5nIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBub3RpZmljYXRpb24gbW9kdWxlIGdpdmVzIGEgd2FybmluZyBpZiB0byBtdWNoIGluY29tcGxldGUgd29ya2luZyBkYXlzIGF2YWlsYWJsZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+TW9kdWwsIGRhcyBkZW4gQWdlbnQgaW0gTm90aWZpY2F0aW9uLUJlcmVpY2ggZGVzIEFnZW50LUludGVyZmFjZXMgZGFy/GJlciBpbmZvcm1pZXJ0LCB3ZW5uIHNjaG9uIHp1IGxhbmdlIGtlaW5lIFN0dW5kZW4gbWVociBlaW5nZXRyYWdlbiB3dXJkZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ6Ok1vZHVsZU5vdGlmeTwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJNb2R1bGUiPktlcm5lbDo6T3V0cHV0OjpIVE1MOjpOb3RpZmljYXRpb25UaW1lQWNjb3VudGluZzwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpNYXhBbGxvd2VkSW5zZXJ0RGF5cyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkZvciBob3cgbWFueSBkYXlzIGFnbyB5b3UgY2FuIGluc2VydCB3b3JraW5nIHVuaXRzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5MZWd0IGZlc3QsIGJpcyB3YW5uIG1hbiBpbiDkbHRlcmUgWmVpdGVpbnRy5GdlIGJlYXJiZWl0ZW4ga2FubiAoei4gQi4gMTAgVGFnZSB6dXL8Y2tsaWVnZW5kKS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+MTA8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6UmVkdWNlVGltZSIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRvIHVzZSwgaWYgc29tZSBhY3Rpb25zIGFjb3VudCByZWR1Y2VkIHdvcmtpbmcgaG91cnMoS2V5ID0mZ3Q7IHRyYXZlbGluZzsgQ29udGVudCA9Jmd0OyA1MCkuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkhpZXIga2FubiBtYW4gZWluZ2ViZW4sIG9iIGb8ciBlaW5lIGJlc3RpbW10ZSBU5HRpZ2tlaXQsIGRpZSB6dSB2ZXJyZWNobmVuZGVuIFN0dW5kZW4gZ2Vr/HJ6dCB3ZXJkZW4uIFouIEIgd2VubiBSZWlzZXplaXRlbiBudXIgenVyIEjkbGZ0ZSB2ZXJn/HRldCB3ZXJkZW4gKEtleSA9Jmd0OyBqb3VybmV5OyBDb250ZW50ID0mZ3Q7IDUwKS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJ0cmF2ZWxpbmciPjUwPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGltZUFjY291bnRpbmc6OklucHV0SG91cnNXaXRob3V0U3RhcnRFbmRUaW1lIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+U2V0dGluZywgaWYgd29ya2luZyBob3VycyBjYW4gaW5zZXJ0IHdpdGhvdXQgc3RhcnQtIGFuZCBlbmR0aW1lLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5MZWd0IGZlc3QsIG9iIG1hbiBBcmJlaXRzc3R1bmRlbiBhdWNoIG9obmUgQW5mYW5ncy0gdW5kIEVuZHplaXQgZWluZ2ViZW4ga2Fubi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxPcHRpb24gU2VsZWN0ZWRJRD0iMSI+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjAiPk5vPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIxIj5ZZXM8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlByZUFwcGxpY2F0aW9uTW9kdWxlIyMjQWdlbnRUaW1lQWNjb3VudGluZyIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgbW9kdWxlIHlvdSBjYW4gZm9yY2UgaW5zZXJ0cyBpbiB0aGUgVGltZUFjY291bnRpbmcuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPk1pdCBIaWxmZSBkaWVzZXMgTW9kdWxzIGv2bm5lbiBFaW50cuRnZSBpbiBkaWUgWmVpdGVyZmFzc3VuZyBlcnp3dW5nZW4gd2VyZGVuLCBpbiBkZW0gYmVpbSBFaW5sb2dnZW4gZGFzIFBvcnRhbCBnZWJsb2NrdCB3aXJkIHVuZCBudXIgZGFzIFplaXRlcmZhc3N1bmdzZmVuc3RlciBlcnNjaGVpbnQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPktlcm5lbDo6TW9kdWxlczo6QWdlbnRUaW1lQWNjb3VudGluZzwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpQcm9qZWN0MlJlbWFya1JlZ0V4cCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkZvciB3aGljaCBwcm9qZWN0cyBpcyBhIHJlbWFyayByZXF1aXJlZC4gSWYgdGhlIFJlZ0V4cCBtYXRjaCBvbiB0aGUgcHJvamVjdCwgeW91IGhhdmUgdG8gaW5zZXJ0IGEgcmVtYXJrIHRvby4gVGhlIFJlZ0V4cCB1c2UgdGhlIHNteCBwYXJhbWV0ZXIuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPklubmVyaGFsYiBkaWVzZXIgS29uZmlndXJhdGlvbnNvcHRpb24ga2FubiBlaW5lIFJlZ0V4cCBkZWZpbmllcnQgd2VyZGVuLCBkaWUgZmVzdGxlZ3QsIGJlaSB3ZWxjaGVuIFByb2pla3RlbiBlaW5lIEJlbWVya3VuZyBlaW5nZXRyYWdlbiB3ZXJkZW4gbXVzcyAoZGllIFJlZ0V4cCBhcmJlaXRldCBtaXQgc214LVBhcmFtZXRlcm4pLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpbWVBY2NvdW50aW5nPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZyBSZWdleD0iIj5eKD86IFBST0pFQ1RcIyB8IElORlJBOiB8IFBST0RVQ1Q6ICk8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaW1lQWNjb3VudGluZzo6UHJvamVjdExpc3RDb25zdHJhaW50cyIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlJlZ3VsYXIgZXhwcmVzc2lvbnMgZm9yIGNvbnN0cmFpbmluZyBwcm9qZWN0IGxpc3QgYWNjb3JkaW5nIHRvIHVzZXIgZ3JvdXBzLiBLZXkgY29udGFpbnMgcmVndWxhciBleHByZXNzaW9uIGZvciBwcm9qZWN0KHMpLCBjb250ZW50IGNvbnRhaW5zIGNvbW1hIHNlcGFyYXRlZCBsaXN0IG9mIGdyb3Vwcy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+UmVndWzkcmUgQXVzZHL8Y2tlIHp1ciBFaW5zY2hy5G5rdW5nIGRlciBQcm9qZWt0bGlzdGUgYWJo5G5naWcgdm9uIEdydXBwZW56dW9yZG51bmcgZGVzIEJlbnV0emVycy4gU2NobPxzc2VsIGVudGjkbHQgUmVnRXhwIGb8ciBQcm9qZWt0KGUpLCBJbmhhbHQgZW50aORsdCBrb21tYXNlcGFyaWVydGUgTGlzdGUgdm9uIEdydXBwZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGltZUFjY291bnRpbmc8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpbWVBY2NvdW50aW5nOjpBY3Rpb25MaXN0Q29uc3RyYWludHMiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5SZWd1bGFyIGV4cHJlc3Npb25zIGZvciBjb25zdHJhaW5pbmcgYWN0aW9uIGxpc3QgYWNjb3JkaW5nIHRvIHNlbGVjdGVkIHByb2plY3QuIEtleSBjb250YWlucyByZWd1bGFyIGV4cHJlc3Npb24gZm9yIHByb2plY3QocyksIGNvbnRlbnQgY29udGFpbnMgcmVndWxhciBleHByZXNzaW9ucyBmb3IgYWN0aW9uKHMpLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5SZWd1bORyZSBBdXNkcvxja2UgenVyIEVpbnNjaHLkbmt1bmcgZGVyIFTkdGlna2VpdGVubGlzdGUgYWJo5G5naWcgdm9tIGdld+RobHRlbiBQcm9qZWt0LiBTY2hs/HNzZWwgZW50aORsdCBSZWdFeHAgZvxyIFByb2pla3QsIEluaGFsdCBlbnRo5Gx0IFJlZ0V4cCBm/HIgVOR0aWdrZWl0KGVuKS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaW1lQWNjb3VudGluZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgo8L290cnNfY29uZmlnPgo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9BZ2VudFRpbWVBY2NvdW50aW5nLnBtIC0gdGhlIGRlIGxhbmd1YWdlIGZvciBBZ2VudFRpbWVBY2NvdW50aW5nCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBkZV9BZ2VudFRpbWVBY2NvdW50aW5nLnBtLHYgMS40NS4yLjEgMjAxMC0xMS0yNiAxMzo1MTo0MyBtbiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpkZV9BZ2VudFRpbWVBY2NvdW50aW5nOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufSA9IHsKICAgICAgICAleyAkU2VsZi0+e1RyYW5zbGF0aW9ufSB9LAoKICAgICAgICAjIFRlbXBsYXRlOiBBZ2VudFRpbWVBY2NvdW50aW5nCiAgICAgICAgJ1NldHRpbmcnICAgICAgICAgID0+ICdLb25maWd1cmF0aW9uJywKICAgICAgICAnUHJvamVjdCBzZXR0aW5ncycgPT4gJ1Byb2pla3Rrb25maWd1cmF0aW9uZW4nLAoKICAgICAgICAnTmV4dCBkYXknICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ07kY2hzdGVyIFRhZycsCiAgICAgICAgJ09uZSBkYXkgYmFjaycgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdFaW5lbiBUYWcgenVy/GNrJywKICAgICAgICAnRGF0ZScgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0RhdHVtJywKICAgICAgICAnQ29tbWVudHMnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0tvbW1lbnRhcmUnLAogICAgICAgICd1bnRpbCcgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnYmlzJywKICAgICAgICAnVG90YWwgaG91cnMgd29ya2VkJyAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0FyYmVpdHNzdHVuZGVuJywKICAgICAgICAnV29ya2luZyBIb3VycycgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0FyYmVpdHNzdHVuZGVuJywKICAgICAgICAnSG91cnMgcGVyIHdlZWsnICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1dvY2hlbnN0dW5kZW4nLAogICAgICAgICd0aGlzIG1vbnRoJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnZGllc2VyIE1vbmF0JywKICAgICAgICAnT3ZlcnRpbWUgKEhvdXJzKScgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ9xiZXJzdHVuZGVuIChpbiBTdHVuZGVuKScsCiAgICAgICAgJ092ZXJ0aW1lICh0b3RhbCknICAgICAgICAgICAgICAgICAgICAgICAgID0+ICfcYmVyc3R1bmRlbiAoZGllc2VyIE1vbmF0KScsCiAgICAgICAgJ092ZXJ0aW1lICh0aGlzIG1vbnRoKScgICAgICAgICAgICAgICAgICAgID0+ICfcYmVyc3R1bmRlbiAoU3VtbWUpJywKICAgICAgICAnUmVtYWluaW5nIG92ZXJ0aW1lIGxlYXZlJyAgICAgICAgICAgICAgICAgPT4gJ9xiZXJzdHVuZGVuICh2ZXJibGVpYmVuZCknLAogICAgICAgICdWYWNhdGlvbicgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnVXJsYXViJywKICAgICAgICAnVmFjYXRpb24gKERheXMpJyAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1VybGF1YiAoaW4gVGFnZW4pJywKICAgICAgICAnVmFjYXRpb24gdGFrZW4gKHRoaXMgbW9udGgpJyAgICAgICAgICAgICAgPT4gJ1VybGF1YnN0YWdlIChkaWVzZXIgTW9uYXQpJywKICAgICAgICAnVmFjYXRpb24gdGFrZW4gKHRvdGFsKScgICAgICAgICAgICAgICAgICAgPT4gJ1VybGF1YnN0YWdlIChTdW1tZSknLAogICAgICAgICdSZW1haW5pbmcgdmFjYXRpb24nICAgICAgICAgICAgICAgICAgICAgICA9PiAnVXJsYXVic3RhZ2UgKHZlcmJsZWliZW5kKScsCiAgICAgICAgJ1NpY2sgbGVhdmUgdGFrZW4gKHRoaXMgbW9udGgpJyAgICAgICAgICAgID0+ICdFcmtyYW5rdCAoZGllc2VyIE1vbmF0KScsCiAgICAgICAgJ1NpY2sgbGVhdmUgdGFrZW4gKHRvdGFsKScgICAgICAgICAgICAgICAgID0+ICdFcmtyYW5rdCAoU3VtbWUpJywKICAgICAgICAnVGltZUFjY291bnRpbmcnICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1plaXRlcmZhc3N1bmcnLAogICAgICAgICdUaW1lIHJlcG9ydGluZyBtb250aGx5IG92ZXJ2aWV3JyAgICAgICAgICA9PiAnTW9uYXRz/GJlcnNpY2h0IFplaXRlcmZhc3N1bmcnLAogICAgICAgICdFZGl0IHRpbWUgcmVjb3JkJyAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnWmVpdGVyZmFzc3VuZyBiZWFyYmVpdGVuJywKICAgICAgICAnRWRpdCB0aW1lIGFjY291bnRpbmcgc2V0dGluZ3MnICAgICAgICAgICAgPT4gJ1plaXRlcmZhc3N1bmdzZWluc3RlbGx1bmdlbiBiZWFyYmVpdGVuJywKICAgICAgICAnVXNlciByZXBvcnRzJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ051dHplcmJlcmljaHRlJywKICAgICAgICAnVXNlclwncyBwcm9qZWN0IG92ZXJ2aWV3JyAgICAgICAgICAgICAgICAgPT4gJ051dHplcmJlcmljaHRlJywKICAgICAgICAnUHJvamVjdCByZXBvcnQnICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1Byb2pla3RiZXJpY2h0JywKICAgICAgICAnUHJvamVjdCByZXBvcnRzJyAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1Byb2pla3RiZXJpY2h0ZScsCiAgICAgICAgJ1RpbWUgcmVwb3J0aW5nJyAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdaZWl0YmVyaWNodGUnLAogICAgICAgICdMZWF2ZURheSBSZW1haW5pbmcnICAgICAgICAgICAgICAgICAgICAgICA9PiAnVmVyYmxlaWJlbmRlIFVybGF1YnN0YWdlJywKICAgICAgICAnTW9udGhseSB0b3RhbCcgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ3BybyBNb25hdCcsCiAgICAgICAgJ1ZpZXcgdGltZSByZWNvcmQnICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdaZWl0cmVrb3JkIGFuc2ljaHQnLAogICAgICAgICdWaWV3IG9mICcgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnQW5zaWNodCB2b24nLAogICAgICAgICdNb250aGx5JyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAncHJvIE1vbmF0JywKICAgICAgICAnSG91cnMnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1N0dW5kZW4nLAogICAgICAgICdEYXRlIG5hdmlnYXRpb24nICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnQXVzd2FobCBEYXR1bScsCiAgICAgICAgJ01vbnRoIG5hdmlnYXRpb24nICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdBdXN3YWhsIERhdHVtJywKICAgICAgICAnRGF5cyB3aXRob3V0IGVudHJpZXMnICAgICAgICAgICAgICAgICAgICAgPT4gJ25pY2h0IGF1c2dlZvxsbHRlIFRhZ2UnLAogICAgICAgICdQcm9qZWN0JyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnUHJvamVrdCcsCiAgICAgICAgJ1Byb2plY3RzJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdQcm9qZWt0ZScsCiAgICAgICAgJ0dyYW5kIHRvdGFsJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdTdW1tZScsCiAgICAgICAgJ0xpZmV0aW1lJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdTdW1tZScsCiAgICAgICAgJ0xpZmV0aW1lIHRvdGFsJyAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdTdW1tZScsCiAgICAgICAgJ1JlcG9ydGluZycgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdCZXJpY2h0c3dlc2VuJywKICAgICAgICAnVGFzayBzZXR0aW5ncycgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ1TkdGlna2VpdHNlaW5zdGVsbHVuZ2VuJywKICAgICAgICAnVXNlciBzZXR0aW5ncycgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ051dHplcmVpbnN0ZWxsdW5nZW4nLAogICAgICAgICdTaG93IE92ZXJ0aW1lJyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAn3GJlcnN0dW5kZW4gYW56ZWlnZW4nLAogICAgICAgICdBbGxvdyBwcm9qZWN0IGNyZWF0aW9uJyAgICAgICAgICAgICAgICAgICA9PiAnUHJvamVrdCBlcnN0ZWxsZW4nLAogICAgICAgICdBZGQgdGltZSBwZXJpb2QnICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnTmV1ZSBOdXR6ZXJlaW5zdGVsbHVuZycsCiAgICAgICAgJ1JlbWFyaycgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdBbm1lcmt1bmcnLAogICAgICAgICdTdGFydCcgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnQmVnaW5uJywKICAgICAgICAnRW5kJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0VuZGUnLAogICAgICAgICdQZXJpb2QgYmVnaW4nICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnRGF0dW0gQmVnaW5uJywKICAgICAgICAnUGVyaW9kIGVuZCcgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0RhdHVtIEVuZGUnLAogICAgICAgICdQZXJpb2QnICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnRGF1ZXInLAogICAgICAgICdEYXlzIG9mIHZhY2F0aW9uJyAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnVXJsYXVic3RhZ2UnLAogICAgICAgICdPbiB2YWNhdGlvbicgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnaW0gVXJsYXViJywKICAgICAgICAnU2ljayBkYXknICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0Vya3Jhbmt0JywKICAgICAgICAnU2ljayBsZWF2ZSAoRGF5cyknICAgICAgICAgICAgICAgICAgICAgICAgPT4gJ0Vya3Jhbmt0IChpbiBUYWdlbiknLAogICAgICAgICdTaWNrIGxlYXZlJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnRXJrcmFua3QnLAogICAgICAgICdPbiBzaWNrIGxlYXZlJyAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnRXJrcmFua3QnLAogICAgICAgICdUYXNrJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9PiAnVOR0aWdrZWl0JywKICAgICAgICAnQXV0aG9yaXplZCBvdmVydGltZScgICAgICAgICAgICAgICAgICAgICAgPT4gJ2F1dG9yaXNpZXJ0ZSDcYmVyc3R1bmRlbicsCiAgICAgICAgJ09uIG92ZXJ0aW1lIGxlYXZlJyAgICAgICAgICAgICAgICAgICAgICAgID0+ICfcYmVyc3R1bmRlbicsCiAgICAgICAgJ092ZXJ0aW1lIGxlYXZlJyAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICfcYmVyc3R1bmRlbicsCiAgICAgICAgJ1RvdGFsJyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICdTdW1tZScsCiAgICAgICAgJ092ZXJ2aWV3IG9mICcgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0+ICfcYmVyc2ljaHQgLSAnLAogICAgICAgICdUaW1lQWNjb3VudGluZyBvZicgICAgICAgICAgICAgICAgICAgICAgICA9PiAnWmVpdGVyZmFzc3VuZyB2b20nLAogICAgICAgICdTdWNjZXNzZnVsIGluc2VydCEnICAgICAgICAgICAgICAgICAgICAgICA9PiAnRXJmb2xncmVpY2ggZWluZ2Vm/Gd0IScsCiAgICAgICAgJ01vcmUgaW5wdXQgZmllbGRzJyAgICAgICAgICAgICAgICAgICAgICAgID0+ICdXZWl0ZXJlIEVpbmdhYmVmZWxkZXInLAogICAgICAgICdEbyB5b3UgcmVhbGx5IHdhbnQgdG8gZGVsZXRlIHRoaXMgT2JqZWN0JyA9PiAnV29sbGVuIFNpZSBkaWVzZW4gRWludHJhZyB3aXJrbGljaCBs9nNjaGVuJywKICAgICAgICAnQ2FuXCd0IGluc2VydCBXb3JraW5nIFVuaXRzIScgICAgICAgICAgICAgPT4gJ0thbm4gZGllIEFyYmVpdHNzdHVuZGVuIG5pY2h0IGVpbmb8Z2VuIScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIG9mIG1pc3NpbmcgdGFzayEnID0+CiAgICAgICAgICAgICdOaWNodCBzcGVpY2hlcmJhciAtIFTkdGlna2VpdCBmZWhsdCEnLAogICAgICAgICdDYW5cJ3Qgc2F2ZSBzZXR0aW5ncywgYmVjYXVzZSBvZiBtaXNzaW5nIHByb2plY3QhJyA9PgogICAgICAgICAgICAnTmljaHQgc3BlaWNoZXJiYXIgLSBQcm9qZWt0YW5nYWJlIGZlaGx0IScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIHRoZSBQZXJpb2QgaXMgYmlnZ2VyIHRoYW4gdGhlIGludGVydmFsIGJldHdlZW4gU3RhcnR0aW1lIGFuZCBFbmR0aW1lIScKICAgICAgICAgICAgPT4gJ05pY2h0IHNwZWljaGVyYmFyIC0gRGF1ZXIgaXN0IGdy9t9lciBhbHMgZGVyIFplaXRyYXVtIHp3aXNjaGVuIEJlZ2lubiB1bmQgRW5kZSEnLAogICAgICAgICdDYW5cJ3Qgc2F2ZSBzZXR0aW5ncywgYmVjYXVzZSBTdGFydHRpbWUgaXMgb2xkZXIgdGhhbiBFbmR0aW1lIScgPT4KICAgICAgICAgICAgJ05pY2h0IHNwZWljaGVyYmFyIC0gQmVnaW5uIGxpZWd0IG5hY2ggRW5kZSEnLAogICAgICAgICdDYW5cJ3Qgc2F2ZSBzZXR0aW5ncywgYmVjYXVzZSBvZiBtaXNzaW5nIHBlcmlvZCEnID0+CiAgICAgICAgICAgICdOaWNodCBzcGVpY2hlcmJhciAtIERhdWVyIChlcmdpYnQgc2ljaCBhdXMgU3RhcnQtIHVuZCBFbmR6ZWl0KSBpc3QgbmljaHQgYW5nZWdlYmVuIScsCiAgICAgICAgJ0NhblwndCBzYXZlIHNldHRpbmdzLCBiZWNhdXNlIFBlcmlvZCBpcyBub3QgZ2l2ZW4gZ2l2ZW4hJyA9PgogICAgICAgICAgICAnTmljaHQgc3BlaWNoZXJiYXIgLSBEYXVlciAoZXJnaWJ0IHNpY2ggYXVzIFN0YXJ0LSB1bmQgRW5kemVpdCkgaXN0IG5pY2h0IGFuZ2VnZWJlbiEnLAogICAgICAgICdBcmUgeW91IHN1cmUsIHRoYXQgeW91IHdvcmtlZCB3aGlsZSB5b3Ugd2VyZSBvbiBzaWNrIGxlYXZlPycgPT4KICAgICAgICAgICAgJ1NpZSB3YXJlbiBrcmFuayB1bmQgaGFiZW4gZ2VhcmJlaXRldD8gV2lyIGJyYXVjaGVuIG1laHIgc29sY2hlIE1pdGFyYmVpdGVyLicsCiAgICAgICAgJ0FyZSB5b3Ugc3VyZSwgdGhhdCB5b3Ugd29ya2VkIHdoaWxlIHlvdSB3ZXJlIG9uIHZhY2F0aW9uPycgPT4KICAgICAgICAgICAgJ1NpZSBoYXR0ZW4gVXJsYXViIHVuZCBoYWJlbiBnZWFyYmVpdGV0PyBXaXIgYnJhdWNoZW4gbWVociBzb2xjaGUgTWl0YXJiZWl0ZXIuJywKICAgICAgICAnQXJlIHlvdSBzdXJlLCB0aGF0IHlvdSB3b3JrZWQgd2hpbGUgeW91IHdlcmUgb24gb3ZlcnRpbWUgbGVhdmU/JyA9PgogICAgICAgICAgICAnSGFiZW4gU2llIHfkaHJlbmQgZGVyINxiZXJzdHVuZGVuIGF1Y2ggZ2VhcmJlaXRldD8nLAogICAgICAgICdDYW5cJ3Qgc2F2ZSBzZXR0aW5ncywgYmVjYXVzZSBhIGRheSBoYXMgb25seSAyNCBob3VycyEnID0+ICdFaW4gVGFnIGhhdCBudXIgMjQgU3R1bmRlbiEnLAogICAgICAgICdDYW5cJ3QgZGVsZXRlIFdvcmtpbmcgVW5pdHMhJyAgICAgID0+ICdLYW5uIEFyYmVpdHNzdHVuZGVuIG5pY2h0IGz2c2NoZW4hJywKICAgICAgICAnUGxlYXNlIGluc2VydCB5b3VyIHdvcmtpbmcgaG91cnMhJyA9PiAnQml0dGUgdHJhZ2VuIFNpZSBJaHJlIEFyYmVpdHN6ZWl0ZW4gZWluIScsCiAgICAgICAgJ1lvdSBoYXZlIHRvIGluc2VydCBhIHN0YXJ0IGFuZCBhbiBlbmQgdGltZSBvciBhIHBlcmlvZCcgPT4KICAgICAgICAgICAgJ1NpZSBt/HNzZW4gQmVnaW5uLSB1bmQgRW5kZXplaXQgb2RlciBkaWUgRGF1ZXIgYW5nZWJlbi4nLAogICAgICAgICdZb3UgY2FuIG9ubHkgc2VsZWN0IG9uZSBjaGVja2JveCBlbGVtZW50IScgPT4gJ1NpZSBr9m5uZW4gbnVyIGVpbmUgQ2hlY2tib3ggbWFya2llcmVuIScsCiAgICAgICAgJ0VkaXQgdGltZSBhY2NvdW50aW5nIHByb2plY3Qgc2V0dGluZ3MnID0+CiAgICAgICAgICAgICdaZWl0ZXJmYXNzdW5nOiBCZWFyYmVpdHVuZyBkZXIgUHJvamVrdGtvbmZpZ3VyYXRpb24nLAogICAgICAgICdQcm9qZWN0IHNldHRpbmdzJyA9PiAnUHJvamVrdGtvbmZpZ3VyYXRpb24nLAogICAgICAgICdJZiB5b3Ugc2VsZWN0ICJNaXNjZWxsYW5lb3VzIChtaXNjKSIgdGhlIHRhc2ssIHBsZWFzZSBleHBsYWluIHRoaXMgaW4gdGhlIHJlbWFya3MgZmllbGQnID0+CiAgICAgICAgICAgICdXZW5uIFNpZSBhbHMgVOR0aWdrZWl0IFNvbnN0aWdlcyBhdXN35GhsZW4sIGdlYmVuIFNpZSBiaXR0ZSBlaW5lIEJlc2NocmVpYnVuZyB1bSBGZWxkIEFubWVya3VuZyBhbi4nLAogICAgICAgICdQbGVhc2UgYWRkIGEgcmVtYXJrIHdpdGggbW9yZSB0aGFuIDggY2hhcmFjdGVycyEnID0+CiAgICAgICAgICAgICdCaXR0ZSBnZWJlbiBTaWUgZWluZSBBbm1lcmt1bmcgZWluIGRpZSBs5G5nZXIgYWxzIDggWmVpY2hlbiBpc3QhJywKCiMgRklYTUUgYWN0dWFsbHkgdGhlIGZvbGxvd2luZyBzaG91bGQgYmUgaW5jbHVkZWQgaW4gZmlsZSBkZS5wbSwgaG93ZXZlciB0aGV5J3JlIG5vdCBzbyBJIHB1dCdlbSBoZXJlLi4uCiAgICAgICAgJ01vbicgICAgICA9PiAnTW8nLAogICAgICAgICdUdWUnICAgICAgPT4gJ0RpJywKICAgICAgICAnV2VkJyAgICAgID0+ICdNaScsCiAgICAgICAgJ1RodScgICAgICA9PiAnRG8nLAogICAgICAgICdGcmknICAgICAgPT4gJ0ZyJywKICAgICAgICAnU2F0JyAgICAgID0+ICdTYScsCiAgICAgICAgJ1N1bicgICAgICA9PiAnU28nLAogICAgICAgICdKYW51YXJ5JyAgPT4gJ0phbnVhcicsCiAgICAgICAgJ0ZlYnJ1YXJ5JyA9PiAnRmVicnVhcicsCiAgICAgICAgJ01hcmNoJyAgICA9PiAnTeRyeicsCiAgICAgICAgJ01heScgICAgICA9PiAnTWFpJywKICAgICAgICAnSnVuZScgICAgID0+ICdKdW5pJywKICAgICAgICAnSnVseScgICAgID0+ICdKdWxpJywKICAgICAgICAnT2N0b2JlcicgID0+ICdPa3RvYmVyJywKICAgICAgICAnRGVjZW1iZXInID0+ICdEZXplbWJlcicsCiAgICB9OwogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
# --
# Kernel/Language/fa_AgentTimeAccounting.pm - Persian translation for AgentTimeAccounting
# Copyright (C) 2001-2011 OTRS AG, http://otrs.org/
# --
# $Id: fa_AgentTimeAccounting.pm,v 1.2.2.2 2011-01-05 10:04:08 mn 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::Language::fa_AgentTimeAccounting;

use strict;
use warnings;

use vars qw($VERSION);
$VERSION = qw($Revision: 1.2.2.2 $) [1];

sub Data {
    my $Self = shift;

    my $Lang = $Self->{Translation};

    return if ref $Lang ne 'HASH';

    $Lang->{'Setting'} = 'تنظیمات';
    $Lang->{'Project settings'} = 'تنظیمات پروژه';
    $Lang->{'Next day'} = 'روز بعد';
    $Lang->{'One day back'} = 'برگشت به روز قبل';
    $Lang->{'Date'} = 'تاریخ';
    $Lang->{'Comments'} = 'توضیحات';
    $Lang->{'until'} = 'تا';
    $Lang->{'Total ras eurs worked'} = 'جمع ساعات کاری';
    $Lang->{'Working Hours'} = 'ساعات کاری';
    $Lang->{'Hours per week'} = 'میزان ساعات در هفته';
    $Lang->{'this month'} = 'این ماه';
    $Lang->{'Overtime (Hours)'} = 'ساعات اضافه کار';
    $Lang->{'Overtime (total)'} = 'مجموع اضافه کار';
    $Lang->{'Overtime (this month)'} = 'اضافه کار این ماه';
    $Lang->{'Remaining overtime leave'} = 'اضافه کاری باقیمانده مجاز';
    $Lang->{'Vacation'} = 'مرخصی';
    $Lang->{'Vacation (Days)'} = 'مرخصی )روزها(';
    $Lang->{'Vacation taken (this month)'} = 'مرخصی‌های گرفته شده این ماه';
    $Lang->{'Vacation taken (total)'} = 'کل مرخصی‌های گرفته شده';
    $Lang->{'Remaining vacation'} = 'مرخصی باقیمانده';
    $Lang->{'Sick leave taken (this month)'} = 'استعلاجی گرفته شده این ماه';
    $Lang->{'Sick leave taken (total)'} = 'کل استعلاجی گرفته شده';
    $Lang->{'TimeAccounting'} = 'حسابداری زمان';
    $Lang->{'Time accounting.'} = 'حسابداری زمان';
    $Lang->{'Time reporting monthly overview'} = 'گزارش کلی ماهیانه زمان ';
    $Lang->{'Edit time record'} = 'ویرایش سابقه زمان';
    $Lang->{'Edit time accounting settings'} = 'ویرایش تنظیمات حسابداری زمان';
    $Lang->{'User reports'} = 'گزارشات کاربری';
    $Lang->{'User\'s project overview'} = 'نمای کلی پروژه کاربر';
    $Lang->{'Project report'} = 'گزارش پروژه';
    $Lang->{'Project reports'} = 'گزارشات پروژه';
    $Lang->{'Time reporting'} = 'گزارشگیری زمان';
    $Lang->{'LeaveDay Remaining'} = 'باقیمانده ترک کار';
    $Lang->{'Monthly total'} = 'کل ماهیانه';
    $Lang->{'View time record'} = 'نمایش سابقه زمان';
    $Lang->{'View of'} = 'نمای کلی از';
    $Lang->{'Monthly'} = 'ماهیانه';
    $Lang->{'Hours'} = 'ساعات';
    $Lang->{'Date navigation'} = 'راهبری تاریخ';
    $Lang->{'Month navigation'} = 'راهبری ماه';
    $Lang->{'Days without entries'} = 'روزهای ثبت نشده';
    $Lang->{'Project'} = 'پروژه';
    $Lang->{'Projects'} = 'پروژه‌ها';
    $Lang->{'Grand total'} = 'تعداد';
    $Lang->{'Lifetime'} = 'تعداد';
    $Lang->{'Lifetime total'} = 'تعداد';
    $Lang->{'Reporting'} = 'گزارشگیری';
    $Lang->{'Task settings'} = 'تنظیمات وظایف';
    $Lang->{'User settings'} = 'تنظیمات کاربری';
    $Lang->{'Show Overtime'} = 'نمایش اضافه کاری';
    $Lang->{'Allow project creation'} = 'اجازه ساخت پروژه';
    $Lang->{'Add time period'} = 'افزودن دوره زمانی';
    $Lang->{'Remark'} = 'تذکر';
    $Lang->{'Start'} = 'آغاز';
    $Lang->{'End'} = 'پایان';
    $Lang->{'Period begin'} = 'آغاز دوره';
    $Lang->{'Period end'} = 'پایان دوره';
    $Lang->{'Period'} = 'دوره';
    $Lang->{'Days of vacation'} = 'روزهای مرخصی';
    $Lang->{'On vacation'} = 'در مرخصی';
    $Lang->{'Sick day'} = 'بیماری';
    $Lang->{'Sick leave (Days)'} = 'روزهای استعلاجی';
    $Lang->{'Sick leave'} = 'استعلاجی';
    $Lang->{'On sick leave'} = 'در استعلاجی';
    $Lang->{'Task'} = 'وظیفه';
    $Lang->{'Authorized overtime'} = 'اضافه کار مجاز';
    $Lang->{'On overtime leave'} = 'در مازاد اضافه کاری';
    $Lang->{'Overtime leave'} = 'مازاد اضافه کاری';
    $Lang->{'Total'} = 'جمع کل';
    $Lang->{'Overview of '} = 'نمای کلی از';
    $Lang->{'TimeAccounting of'} = 'حسابداری زمان مربوط به';
    $Lang->{'Successful insert!'} = 'با موفقیت ثبت شد';
    $Lang->{'More input fields'} = 'فیلدهای ورودی بیشتر';
    $Lang->{'Do you really want to delete this Object'} = 'آیا واقعا مایل به حذف این مورد هستید؟';
    $Lang->{'Can\'t insert Working Units!'} = 'عدم توانایی در ثبت واحدهای کاری';
    $Lang->{'Can\'t save settings, because of missing task!'} = 'به دلیل فراموشی وظیفه، تنظیمات ذخیره نمی‌شود!';
    $Lang->{'Can\'t save settings, because of missing project!'} = 'به دلیل فراموشی پروژه، تنظیمات ذخیره نمی‌شود!';
    $Lang->{'Can\'t save settings, because the Period is bigger than the interval between Starttime and Endtime!'}
        = 'نمی‌توان تنظیمات را ذخیره کرد زیرا دوره زمانی از مقدار ورودی بین ساعت آغاز و پایان بزرگتر است!';
    $Lang->{'Can\'t save settings, because Starttime is older than Endtime!'}
        =  'نمی‌توان تنظیمات را ذخیره کرد زیرا زمان شروع بعد از زمان پایان است!';
    $Lang->{'Can\'t save settings, because of missing period!'} = 'نمی‌توان تنظیمات را ذخیره کرد زیرا دوره زمانی فراموش شده است!';
    $Lang->{'Can\'t save settings, because Period is not given!'} = 'نمی‌توان تنظیمات را ذخیره کرد زیرا دوره زمانی داده نشده است!';
    $Lang->{'Are you sure, that you worked while you were on sick leave?'} = 'آیا اطمینان دارید در زمانی که در زمان استعلاجی بودید، کار کرده‌اید؟';
    $Lang->{'Are you sure, that you worked while you were on vacation?'} = 'آیا اطمینان دارید در زمانی که در مرخصی بودید، کار کرده‌اید؟';
    $Lang->{'Are you sure, that you worked while you were on overtime leave?'} = 'آیا اطمینان دارید در زمانی که در مازاد اضافه کاری بودید، کار کرده‌اید؟';
    $Lang->{'Can\'t save settings, because a day has only 24 hours!'} = 'هر روز فقط ۲۴ ساعت دارد!';
    $Lang->{'Can\'t delete Working Units!'} = 'واحدهای کاری قابل خذف نیست!';
    $Lang->{'Please insert your working hours!'} = 'لطفا ساعات کاری خود را وارد نمایید!';
    $Lang->{'You have to insert a start and an end time or a period'} = 'شما باید یا یک دوره زمانی و یا زمان آغاز و پایان وارد نمایید!';
    $Lang->{'You can only select one checkbox element!'} = 'شما فقط می‌توانید یک گزینه را انتخاب نمایید!';
    $Lang->{'Edit time accounting project settings'} = 'ویرایش تنظیمات حسابداری زمان پروژه';
    $Lang->{'Project settings'} = 'تنظیمات پروژه';
    $Lang->{'If you select "Miscellaneous (misc)" the task, please explain this in the remarks field'} =
        'اگر وظایف همزمان را انتخاب کردید، لطفا در فیلدهای تذکر توضیح دهید';
    $Lang->{'Please add a remark with more than 8 characters!'} = 'لطفا تذکری با بیش از ۸ کاراکتر بیفزایید!';
    $Lang->{'Mon'} = 'دوشنبه';
    $Lang->{'Tue'} = 'سه‌شنبه';
    $Lang->{'Wed'} = 'چهارشنبه';
    $Lang->{'Thu'} = 'پنجشنبه';
    $Lang->{'Fri'} = 'جمعه';
    $Lang->{'Sat'} = 'شنبه';
    $Lang->{'Sun'} = 'یکشنبه';
    $Lang->{'January'} = 'ژانویه';
    $Lang->{'February'} = 'فوریه';
    $Lang->{'March'} = 'مارس';
    $Lang->{'April'} = '';
    $Lang->{'May'} = 'می';
    $Lang->{'June'} = 'ژوئن';
    $Lang->{'July'} = 'جولای';
    $Lang->{'August'} = '';
    $Lang->{'September'} = '';
    $Lang->{'October'} = 'اکتبر';
    $Lang->{'November'} = '';
    $Lang->{'December'} = 'دسامبر';
    $Lang->{'Frontend module registration for the agent interface.'} = '';
    $Lang->{'Overview'} = '';
    $Lang->{'Project time reporting'} = '';
    $Lang->{'Default name for new projects.'} = '';
    $Lang->{'Default status for new projects.'} = '';
    $Lang->{'Default name for new actions.'} = '';
    $Lang->{'Default status for new actions.'} = '';
    $Lang->{'Default setting for the standard weekly hours.'} = '';
    $Lang->{'Default setting for leave days.'} = '';
    $Lang->{'Default setting for overtime.'} = '';
    $Lang->{'Default setting for date start.'} = '';
    $Lang->{'Default setting for date end.'} = '';
    $Lang->{'Default status for new users.'} = '';
    $Lang->{'Maximum number of working days after which the working units have to be inserted.'} = '';
    $Lang->{'Maximum number of working days withouth working units entry after which a warning will be shown.'} = '';
    $Lang->{'This notification module gives a warning if there are too many incomplete working days.'} = '';
    $Lang->{'For how many days ago you can insert working units.'} = '';
    $Lang->{'To use if some actions reduced the working hours (for example, if only half of the traveling time is paid Key =&gt; traveling; Content =&gt; 50).'}
        = '';
    $Lang->{'Specifies if working hours can be inserted without start and end times.'} = '';
    $Lang->{'This module forces inserts in TimeAccounting.'} = '';
    $Lang->{'Defines the projects for which a remark is required. If the RegExp matches on the project, you have to insert a remark too. The RegExp use the smx parameter.'}
        = '';
    $Lang->{'Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.'}
        = '';
    $Lang->{'Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).'}
        = '';

    return 1;
}

1;

# --
# Kernel/Modules/AgentTimeAccounting.pm - time accounting module
# Copyright (C) 2001-2011 OTRS AG, http://otrs.org/
# --
# $Id: AgentTimeAccounting.pm,v 1.41.2.1 2011-05-10 18:42:25 en 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.41.2.1 $) [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};

        # set action list and related constraints
        # generate a JavaScript Array which will be output to the template
        my @ActionIDs = sort { $ActionList{$a} cmp $ActionList{$b} } keys %ActionList;
        my @JSActions;
        foreach my $ActionID (@ActionIDs) {
            push @JSActions, "['$ActionID', '$ActionList{$ActionID}']";
        }
        $Param{JSActionList} = '[' . ( join ', ', @JSActions ) . ']';

        my $ActionListConstraints
            = $Self->{ConfigObject}->Get('TimeAccounting::ActionListConstraints');
        my @JSActionListConstraints;
        for my $ProjectNameRegExp ( keys %{$ActionListConstraints} ) {
            my $ActionNameRegExp = $ActionListConstraints->{$ProjectNameRegExp};
            s{(['"\\])}{\\$1}smxg for ( $ProjectNameRegExp, $ActionNameRegExp );
            push @JSActionListConstraints, "['$ProjectNameRegExp', '$ActionNameRegExp']";
        }
        $Param{JSActionListConstraints} = '[' . ( join ', ', @JSActionListConstraints ) . ']';

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

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

        # build units
        $Param{"JSProjectList"} = "var JSProjectList = new Array();\n";
        for my $ID ( 1 .. 16 ) {
            $Param{ID} = $ID;
            my $UnitRef = $Units[$ID];

            # get data of projects
            my $ProjectList = $Self->_ProjectList(
                SelectedID => $UnitRef->{ProjectID},
            );

            $Param{ProjectID} = $UnitRef->{ProjectID} || '';
            $Param{ProjectName} = '';

            # generate JavaScript array which will be output to the template
            my @JSProjectList;
            for my $Project ( @{$ProjectList} ) {

                my $ProjectNmame = $Project->{Value};
                $ProjectNmame =~ s{[']}{\\'}gxms;

                push @JSProjectList,
                    '{id:' . ( $Project->{Key} || '0' ) . ' , name:\'' . $ProjectNmame . '\'}';
                if ( $Project->{Key} eq $Param{ProjectID} ) {
                    $Param{ProjectName} = $Project->{Value};
                }
            }
            $Param{"JSProjectList"}
                .= "JSProjectList[$ID] = [" . ( join ', ', @JSProjectList ) . "];\n";

            # ProjectOption will not be used any more when project autocompletion is done
            $Frontend{ProjectOption} = $Self->{LayoutObject}->BuildSelection(
                Data        => $ProjectList,
                Name        => "ProjectID[$ID]",
                Translation => 0,

                #Max        => 62,
                Class    => 'ProjectSelection',
                OnChange => "FillActionList($ID);",
            );

            # action list initially only contains empty and selected element as well as
            # elements configured for selected project if no constraints are configured,
            # all actions will be displayed
            my $ActionData = $Self->_ActionListConstraints(
                ProjectID             => $UnitRef->{ProjectID},
                ProjectList           => $ProjectList,
                ActionList            => \%ActionList,
                ActionListConstraints => $ActionListConstraints,
            );
            $ActionData->{''} = '';
            if ( $UnitRef->{ActionID} && $ActionList{ $UnitRef->{ActionID} } ) {
                $ActionData->{ $UnitRef->{ActionID} } = $ActionList{ $UnitRef->{ActionID} };
            }

            $Frontend{ActionOption} = $Self->{LayoutObject}->BuildSelection(

                Data        => $ActionData,
                SelectedID  => $UnitRef->{ActionID} || '',
                Name        => "ActionID[$ID]",
                Translation => 0,
                Class       => 'ActionSelection',
            );

            $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' };
        my $CalendarIndex = 1;
        while ( $Self->{ConfigObject}->Get( "TimeZone::Calendar" . $CalendarIndex . "Name" ) ) {
            $CalendarListRef->{$CalendarIndex}
                = $Self->{ConfigObject}->Get( "TimeZone::Calendar" . $CalendarIndex . "Name" );
            $CalendarIndex++;
        }

        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 _ActionListConstraints {
    my ( $Self, %Param ) = @_;

    my %List;
    if ( $Param{ProjectID} && keys %{ $Param{ActionListConstraints} } ) {

        my $ProjectName;

        PROJECT:
        for my $Project ( @{ $Param{ProjectList} } ) {
            if ( $Project->{Key} eq $Param{ProjectID} ) {
                $ProjectName = $Project->{Value};
                last PROJECT;
            }
        }

        if ( defined($ProjectName) ) {

            # loop over actions to find matches for configured project
            # and action regexp pairs
            for my $ActionID ( keys %{ $Param{ActionList} } ) {

                my $ActionName = $Param{ActionList}->{$ActionID};

                REGEXP:
                for my $ProjectNameRegExp ( keys %{ $Param{ActionListConstraints} } ) {
                    my $ActionNameRegExp = $Param{ActionListConstraints}->{$ProjectNameRegExp};
                    if (
                        $ProjectName   =~ m{$ProjectNameRegExp}smx
                        && $ActionName =~ m{$ActionNameRegExp}smx
                        )
                    {
                        $List{$ActionID} = $ActionName;
                        last REGEXP;
                    }
                }
            }
        }
    }

    # all available actions will be added if no action was added above (possible misconfiguration)
    if ( !keys %List ) {
        for my $ActionID ( keys %{ $Param{ActionList} } ) {
            my $ActionName = $Param{ActionList}->{$ActionID};
            $List{$ActionID} = $ActionName;
        }
    }

    return \%List;
}

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

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

    @List = $Self->_ProjectListConstraints(
        List => \@List,
        SelectedID => $Param{SelectedID} || '',
    );

    return \@List;
}

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

    my @List;
    my $ProjectCount = 0;
    my $ProjectListConstraints
        = $Self->{ConfigObject}->Get('TimeAccounting::ProjectListConstraints');

    if ( keys %{$ProjectListConstraints} ) {

        # get groups of current user
        my %Groups = $Self->{GroupObject}->GroupMemberList(
            UserID => $Self->{UserID},
            Type   => 'ro',
            Result => 'HASH',
        );
        %Groups = map { $Groups{$_} => 1 } keys %Groups;

        # get project list constraints
        my %ProjectRegex;
        for my $ProjectRegex ( keys %{$ProjectListConstraints} ) {
            for my $ProjectGroup ( split /,\s*/, $ProjectListConstraints->{$ProjectRegex} ) {
                if ( $Groups{$ProjectGroup} ) {
                    $ProjectRegex{$ProjectRegex} = 1;
                }
            }
        }
        my @ProjectRegex = keys %ProjectRegex;

        # reduce project list according to configuration
        if ( ref( $Param{List} ) && @ProjectRegex ) {

            my $ElementCount = 0;

            for my $Project ( @{ $Param{List} } ) {
                my $ProjectName = $Project->{Value};

                # empty first element, last projects separator and currently selected project
                if ( !$ElementCount || !$Project->{Key} || $Project->{Key} eq $Param{SelectedID} ) {
                    push @List, $Project;
                }
                else {
                    PROJECTREGEXP:
                    for my $ProjectRegex (@ProjectRegex) {
                        if ( $ProjectName =~ m{$ProjectRegex}smx ) {
                            push @List, $Project;
                            $ProjectCount++;
                            last PROJECTREGEXP;
                        }
                    }
                }
                $ElementCount++;
            }
        }
    }

# get full project list if constraints resulted in empty project list or if constraints aren't configured (possible misconfiguration)
    if ( !$ProjectCount ) {
        @List = @{ $Param{List} };
    }

    return @List;
}

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

IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9Ob3RpZmljYXRpb25UaW1lQWNjb3VudGluZy5wbQojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogTm90aWZpY2F0aW9uVGltZUFjY291bnRpbmcucG0sdiAxLjYgMjAwOS0wNC0wMyAxMTo0OToyOSB0ciBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6Tm90aWZpY2F0aW9uVGltZUFjY291bnRpbmc7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpTeXN0ZW06OlRpbWVBY2NvdW50aW5nOwp1c2UgS2VybmVsOjpTeXN0ZW06OlRpbWU7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjYgJCkgWzFdOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGdldCBuZWVkZWQgb2JqZWN0cwogICAgZm9yIChxdyhDb25maWdPYmplY3QgTG9nT2JqZWN0IERCT2JqZWN0IExheW91dE9iamVjdCBVc2VySUQpKSB7CiAgICAgICAgJFNlbGYtPnskX30gPSAkUGFyYW17JF99IHx8IGRpZSAiR290IG5vICRfISI7CiAgICB9CgogICAgJFNlbGYtPntUaW1lT2JqZWN0fSAgICAgICAgICAgPSBLZXJuZWw6OlN5c3RlbTo6VGltZS0+bmV3KCVQYXJhbSk7CiAgICAkU2VsZi0+e1RpbWVBY2NvdW50aW5nT2JqZWN0fSA9IEtlcm5lbDo6U3lzdGVtOjpUaW1lQWNjb3VudGluZy0+bmV3KCVQYXJhbSk7CiAgICByZXR1cm4gJFNlbGY7Cn0KCnN1YiBSdW4gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAoICRTZWMsICRNaW4sICRIb3VyLCAkRGF5LCAkTW9udGgsICRZZWFyICkKICAgICAgICA9ICRTZWxmLT57VGltZU9iamVjdH0tPlN5c3RlbVRpbWUyRGF0ZSggU3lzdGVtVGltZSA9PiAkU2VsZi0+e1RpbWVPYmplY3R9LT5TeXN0ZW1UaW1lKCksICk7CiAgICBteSAlVXNlciA9ICRTZWxmLT57VGltZUFjY291bnRpbmdPYmplY3R9LT5Vc2VyQ3VycmVudFBlcmlvZEdldCgKICAgICAgICBZZWFyICA9PiAkWWVhciwKICAgICAgICBNb250aCA9PiAkTW9udGgsCiAgICAgICAgRGF5ICAgPT4gJERheSwKICAgICk7CiAgICBpZiAoICRVc2VyeyAkU2VsZi0+e1VzZXJJRH0gfSApIHsKICAgICAgICBteSAlSW5jb21wbGV0ZVdvcmtpbmdEYXlzID0gJFNlbGYtPntUaW1lQWNjb3VudGluZ09iamVjdH0tPldvcmtpbmdVbml0c0NvbXBsZXRuZXNzQ2hlY2soKTsKCiAgICAgICAgIyByZWRpcmVjdCBpZiBpbmNvbXBsZXRlIHdvcmtpbmcgZGF5IGFyZSBvdXQgb2YgcmFuZ2UKICAgICAgICBpZiAoICRJbmNvbXBsZXRlV29ya2luZ0RheXN7V2FybmluZ30gKSB7CiAgICAgICAgICAgIHJldHVybiAkU2VsZi0+e0xheW91dE9iamVjdH0tPk5vdGlmeSgKICAgICAgICAgICAgICAgIEluZm8gICAgID0+ICdQbGVhc2UgaW5zZXJ0IHlvdXIgd29ya2luZyBob3VycyEnLAogICAgICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ0Vycm9yJwogICAgICAgICAgICApOwogICAgICAgIH0KICAgIH0KICAgIHJldHVybiAnJzsKfQoKMTsK
IyAtLQojIEFnZW50VGltZUFjY291bnRpbmdEZWxldGUuZHRsIC0gcHJvdmlkZXMgSFRNTCBmb3JtIGZvciBkZWxldGUgYSBkYXkgb2YgdGltZWFjY291bnRpbmcKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEFnZW50VGltZUFjY291bnRpbmdEZWxldGUuZHRsLHYgMS4yIDIwMDktMDQtMDMgMTE6NDk6MjkgdHIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCjwhLS0gc3RhcnQgZm9ybSAtLT4KPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjEwMCUiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iMyI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluaGVhZCI+CiAgICAgICRFbnZ7IkJveDAifSRUZXh0eyJEZWxldGUifTogJFRleHR7IlRpbWVBY2NvdW50aW5nIG9mIn0gJFFEYXRheyJZZWFyIn0tJFFEYXRheyJNb250aCJ9LSRRRGF0YXsiRGF5In0kRW52eyJCb3gxIn0KICAgIDwvdGQ+CiAgPC90cj4KIyAgPHRyPgojICAgIDx0ZCBjbGFzcz0ibWVudSI+CiMgICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9JEVudnsiTGFzdFNjcmVlblZpZXcifSIgY2xhc3M9Im1lbnVpdGVtIj4kVGV4dHsiQmFjayJ9PC9hPgojICAgIDwvdGQ+CiMgIDwvdHI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluYm9keSI+CiAgICAgIDxicj4KICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjM1MCIgYWxpZ249ImNlbnRlciIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSI0Ij4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY29sc3Bhbj0iMiIgY2xhc3M9ImNvbnRlbnRoZWFkIj4kVGV4dHsiT3B0aW9ucyJ9PC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGJvZHkiPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSIgID4kVGV4dHsiRGVsZXRlIn06IDwvdGQ+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+JFRleHR7IlRpbWVBY2NvdW50aW5nIG9mIn0gJFFEYXRheyJZZWFyIn0tJFFEYXRheyJNb250aCJ9LSRRRGF0YXsiRGF5In08L3RkPgogICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgPHRkIGNvbHNwYW49IjIiPiZuYnNwOzwvdGQ+CiAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iMiIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJEbyB5b3UgcmVhbGx5IHdhbnQgdG8gZGVsZXRlIHRoaXMgT2JqZWN0In0/PC90ZD4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGZvb3RlciIgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+CiAgICAgICAgICAgIDxhIGhyZWY9IiRFbnZ7IkJhc2VsaW5rIn1BY3Rpb249JEVudnsiQWN0aW9uIn0mU3ViYWN0aW9uPUVkaXQmWWVhcj0kUURhdGF7IlllYXIifSZNb250aD0kUURhdGF7Ik1vbnRoIn0mRGF5PSRRRGF0YXsiRGF5In0iIj4kVGV4dHsiTm8ifTwvYT4gPGEgaHJlZj0iJEVudnsiQmFzZWxpbmsifUFjdGlvbj0kRW52eyJBY3Rpb24ifSZTdWJhY3Rpb249RGVsZXRlJlllYXI9JFFEYXRheyJZZWFyIn0mTW9udGg9JFFEYXRheyJNb250aCJ9JkRheT0kUURhdGF7IkRheSJ9Ij4kVGV4dHsiWWVzIn08L2E+CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgIDwvdGFibGU+CiAgICAgIDxicj4KICAgIDwvdGQ+CiAgPC90cj4KPC90YWJsZT4KPCEtLSBlbmQgZm9ybSAtLT4K
# --
# AgentTimeAccountingEdit.dtl - provides HTML form for time accounting edit
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: AgentTimeAccountingEdit.dtl,v 1.28 2010-06-24 12:35:58 jp 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.
# --

<!--CSS file (default YUI Sam Skin) -->
<link type="text/css" rel="stylesheet" href="$Config{"Frontend::YUIPath"}build/autocomplete/assets/skins/sam/autocomplete.css">

<!-- Dependencies -->
<script type="text/javascript" src="$Config{"Frontend::YUIPath"}build/yahoo-dom-event/yahoo-dom-event.js"></script>
<script type="text/javascript" src="$Config{"Frontend::YUIPath"}build/datasource/datasource-min.js"></script>

<!-- OPTIONAL: Animation (required only if enabling animation) -->
<script type="text/javascript" src="$Config{"Frontend::YUIPath"}build/animation/animation-min.js"></script>

<!-- Source file -->
<script type="text/javascript" src="$Config{"Frontend::YUIPath"}build/autocomplete/autocomplete-min.js"></script>

<style type="text/css">
.ProjectAutoComplete {
    width:36em; /* set width here or else widget will expand to fit its container */
    padding-bottom:0.5em;
}
.ActionAutoComplete {
    width:30em; /* set width here or else widget will expand to fit its container */
    padding-bottom:0.5em;
}
</style>

<script type="text/javascript">
<!--

//
// AutoCompletion not finished yet, it is currently deactivated
//

// Initializes Autocompletion for Input field
// FieldName contains either "Project" or "Action"
// RowNumber contains number of row (1 to 16)
function initAutoCompletionInput(FieldName, RowNumber) {

    var Dom = YAHOO.util.Dom;

    $QData{"JSProjectList"}
    var myProjects = JSProjectList[RowNumber];

    // Define a custom search function for the DataSource
    var matchNames = function(sQuery) {
        // Case insensitive matching
        var query = decodeURIComponent(sQuery.toLowerCase()),
            contact,
            i=0,
            l=myProjects.length,
            matches = [];

        // Match against each name of each contact
        for(; i<l; i++) {
            contact = myProjects[i];
            if( (contact.name && (contact.name.toLowerCase().indexOf(query) > -1)) || query == ' ' ) {
                matches[matches.length] = contact;
            }
        }

        return matches;
    };

    // Use a FunctionDataSource
    var oDS = new YAHOO.util.FunctionDataSource(matchNames);
    oDS.responseSchema = {
        fields: ["id", "name"]
    }

    // Instantiate AutoComplete
    var oAC = new YAHOO.widget.AutoComplete(FieldName + "AutoCompleteInput[" + RowNumber + "]", FieldName + "AutoCompleteContainer[" + RowNumber + "]", oDS);
    //oAC.useIFrame = true;
    oAC.useShadow = true;
    oAC.resultTypeList = false;
    //oAC.forceSelection = true;


    // Custom formatter to highlight the matching letters
    oAC.formatResult = function(oResultData, sQuery, sResultMatch) {
        var query = sQuery.toLowerCase(),

            name = oResultData.name || "", // Guard against null value
            query = sQuery.toLowerCase(),
            nameMatchIndex = name.toLowerCase().indexOf(query),
            displayname;

        if(nameMatchIndex > -1) {
            displayname = highlightMatch(name, query, nameMatchIndex);
        }
        else {
            displayname = name ? name : "";
        }

        return displayname;
    };

    // Helper function for the formatter
    var highlightMatch = function(full, snippet, matchindex) {
        return full.substring(0, matchindex) +
                "<span class='match'>" +
                full.substr(matchindex, snippet.length) +
                "</span>" +
                full.substring(matchindex + snippet.length);
    };

    // Define an event handler to populate a hidden form field
    // when an item gets selected and populate the input field
    var myHiddenField = YAHOO.util.Dom.get(FieldName + "ID[" + RowNumber + "]");
    var myHandler = function(sType, aArgs) {
        var myAC = aArgs[0]; // reference back to the AC instance
        var elLI = aArgs[1]; // reference to the selected LI element
        var oData = aArgs[2]; // object literal of selected item's result data

        // update hidden form field with the selected item's ID
        myHiddenField.value = oData.id;

        myAC.getInputEl().value = oData.name ? oData.name : "";
    };

    oAC.itemSelectEvent.subscribe(myHandler);

    return {
        oDS: oDS,
        oAC: oAC
    };
};

// Initialize Autocompletion for project and action selection
var ProjectACObjects = new Array();
var ActionACObjects = new Array();
var Event = YAHOO.util.Event;
Event.onDOMReady(function() {
    // add yui css to body tag
    document.body.className = "yui-skin-sam";

    // initialize Autocompletion for selection of projects and actions
    for (var RowNumber = 1; RowNumber < 16; RowNumber++) {
        ProjectACObjects[RowNumber] = initAutoCompletionInput("Project", RowNumber);
        //ActionACObjects[RowNumber] = initAutoCompletionInput("Action", RowNumber);
    }
});

function ClearAutoCompletionID(FieldName, RowNumber) {
    var TextElement = document.getElementById(FieldName + "AutoCompleteInput[" + RowNumber + "]");
    var IDElement = document.getElementById(FieldName + "ID[" + RowNumber + "]");
    if (TextElement.value == '') {
        IDElement.value = '';
    }
}

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

// Fills options of action selection according to selected project
function FillActionList(WorkingUnitID) {

    var ProjectSelectionIndex = document.getElementById("ProjectID[" + WorkingUnitID + "]").selectedIndex;
    var ProjectName = document.getElementById("ProjectID[" + WorkingUnitID + "]").options[ProjectSelectionIndex].text;

    var ActionList = $QData{"JSActionList"};
    var ActionListConstraints = $QData{"JSActionListConstraints"};
    var ActionSelection = document.getElementById("ActionID[" + WorkingUnitID + "]");
    var SelectedActionID = ActionSelection.value;

    // remove previous actions, leave first element (empty element)
    var ActionSelectionLength = ActionSelection.length;
    for (var ActionSelectionIndex = 1; ActionSelectionIndex < ActionSelectionLength; ActionSelectionIndex++) {
        ActionSelection.remove(1);
    }

    var OptionCount = 0;
    for (var ActionListIndex = 0; ActionListIndex < ActionList.length; ActionListIndex++) {

        var ActionID = ActionList[ActionListIndex][0];
        var ActionName = ActionList[ActionListIndex][1];

        for (var ActionListConstraintsIndex = 0; ActionListConstraintsIndex < ActionListConstraints.length; ActionListConstraintsIndex++) {
            var ProjectNameRegExp = new RegExp(ActionListConstraints[ActionListConstraintsIndex][0]);
            var ActionNameRegExp = new RegExp(ActionListConstraints[ActionListConstraintsIndex][1]);

            // add action to selection
            if (ProjectNameRegExp.test(ProjectName) && ActionNameRegExp.test(ActionName)) {
                AddSelectionOption(ActionSelection, ActionName, ActionID, SelectedActionID);
                OptionCount++;
            }
        }
    }

    // all actions will be added if no action was added above (possible misconfiguration)
    if (!OptionCount) {
        // ignore first element because it's the empty element which is already part of the list
        for (var ActionListIndex = 1; ActionListIndex < ActionList.length; ActionListIndex++) {
            var ActionID = ActionList[ActionListIndex][0];
            var ActionName = ActionList[ActionListIndex][1];
            AddSelectionOption(ActionSelection, ActionName, ActionID, SelectedActionID);
        }
    }
}

// Adds option to a selection
function AddSelectionOption(SelectionElement, OptionText, OptionValue, SelectedOption) {

    var Option = document.createElement('option');
    Option.text = OptionText;
    Option.value = OptionValue;

    if (OptionValue == SelectedOption) {
        Option.selected = true;
    }

    try {
        SelectionElement.add(Option, null); // All but IE
    }
    catch (ex) {
        SelectionElement.add(Option); // Only IE
    }
}

// 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{"$Data{"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>
<!-- the following block will be enabled when project autocompletion is implemented
                                                    <div class="ProjectAutoComplete">
                                                        <input id="ProjectAutoCompleteInput[$QData{"ID"}]" type="text" value="$QData{"ProjectName"}" onchange="ClearAutoCompletionID('Project', $QData{"ID"});">
                                                        <div id="ProjectAutoCompleteContainer[$QData{"ID"}]"></div>
                                                        <input id="ProjectID[$QData{"ID"}]" name="ProjectID[$QData{"ID"}]" type="hidden" value="$QData{"ProjectID"}">
                                                    </div>
                                                    -->
                                                    $Data{"ProjectOption"}
                                                </td>
                                                <td>
<!-- the following block will be enabled when action autocompletion is implemented
                                                    <div class="ActionAutoComplete">
                                                        <input id="ActionAutoCompleteInput[$QData{"ID"}]" type="text" value="$QData{"ActionName"}" onchange="ClearAutoCompletionID('Action', $QData{"ID"});">
                                                        <div id="ActionAutoCompleteContainer[$QData{"ID"}]"></div>
                                                        <input id="ActionIDNew[$QData{"ID"}]" name="ActionIDNew[$QData{"ID"}]" type="hidden" value="$QData{"ActionID"}">
                                                    </div>
                                                    -->
                                                    $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{"$Data{"Description"}"}
                                                </td>
                                            </tr>
<!-- dtl:block:UnitRequired -->
<!-- dtl:block:UnitReadonly -->
                                            <tr class="readonly">
                                                <td colspan="6">
                                                    $Text{"$Data{"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{"$Data{"Description"}"}</div>
                                                </td>
                                            </tr>
<!-- dtl:block:Required -->
<!-- dtl:block:Readonly -->
                                            <tr>
                                                <td colspan="6">
                                                    <div class="readonly">$Text{"$Data{"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>

# --
# 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>

IyAtLQojIEFnZW50VGltZUFjY291bnRpbmdQcm9qZWN0UmVwb3J0aW5nLmR0bCAtIHByb3ZpZGVzIEhUTUwgZm9ybSBmb3IgdGltZSBhY2NvdW50aW5nIHJlcG9ydGluZwojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogQWdlbnRUaW1lQWNjb3VudGluZ1Byb2plY3RSZXBvcnRpbmcuZHRsLHYgMS43IDIwMDktMDQtMDMgMTE6NDk6MjkgdHIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCjx0YWJsZSBib3JkZXI9IjAiIHdpZHRoPSIxMDAlIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiPgogIDx0cj4KICAgIDx0ZCBjbGFzcz0ibWFpbmhlYWQiPgogICAgICAkRW52eyJCb3gwIn0kVGV4dHsiUHJvamVjdCByZXBvcnQifTogJFFEYXRheyJQcm9qZWN0In0gJEVudnsiQm94MSJ9CiAgICA8L3RkPgogIDwvdHI+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==
# --
# 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>

# --
# 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 -->

IyAtLQojIEFnZW50VGltZUFjY291bnRpbmdWaWV3LmR0bCAtIHByb3ZpZGVzIEhUTUwgZm9yIHZpZXcgaW5zZXJ0cyBvZiBvbmUgZGF5CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBBZ2VudFRpbWVBY2NvdW50aW5nVmlldy5kdGwsdiAxLjExIDIwMDktMDQtMDMgMTE6NDk6MjkgdHIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjEwMCUiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iMyI+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
#--
# Kernel/System/TimeAccounting.pm - all time accounting functions
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: TimeAccounting.pm,v 1.40.2.1 2010-12-03 09:39:56 mn 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.40.2.1 $) [1];

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

=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 my $Object (qw(DBObject ConfigObject LogObject UserID TimeObject UserObject MainObject)) {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }

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

The returned hash contains the following elements:

    %UserData = (
        1 => {
            UserID      => 1,
            Period      => 123,
            DateStart   => '2005-12-24',
            DateEnd     => '2005-12-24',
            WeeklyHours => 40.4,
            LeaveDays   => 12,
            OverTime    => 34,
        },
    );

=cut

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

    # check needed params
    for my $NeededParam (qw(Year Month Day)) {
        if ( !$Param{$NeededParam} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "UserCurrentPeriodGet: Need $NeededParam!"
            );
            return;
        }
    }

    # build date string for given params
    my $Date = sprintf( "%04d-%02d-%02d", $Param{Year}, $Param{Month}, $Param{Day} ) . ' 00:00:00';

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

    # db select
    my $PrepareSuccess = $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 <= ? AND date_end  >= ? '
            . 'AND status = ?',
        Bind => [ \$Date, \$Date, \1, ],
    );

    # check success of prepare statement
    return if !$PrepareSuccess;

    # fetch Data
    my %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;
    }

    # check for valid user data
    return if !%Data;

    # store user data in cache
    $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 ) = @_;

    # check needed params
    for my $NeededParam (qw(Year Month Day)) {
        if ( !$Param{$NeededParam} && $NeededParam ne 'Day' ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "UserReporting: Need $NeededParam!"
            );
            return;
        }
    }

    # check valide date values
    return if !check_date( $Param{Year}, $Param{Month}, $Param{Day} || 1 );

    # get days of month if not provided
    $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};

    my %Data;

    USERID:
    for my $UserID ( keys %UserCurrentPeriod ) {
        if ( $UserCurrentPeriod{$UserID}{DateStart} =~ m{ \A (\d{4})-(\d{2})-(\d{2}) }smx ) {
            $YearStart  = $1;
            $MonthStart = $2;
            $DayStart   = $3;
        }

        if ( !check_date( $YearStart, $MonthStart, $DayStart ) ) {

            $Self->{LogObject}->Log(
                Priority => 'notice',
                Message  => 'UserReporting: Invalid start date for user '
                    . "$UserID: $UserCurrentPeriod{$UserID}{DateStart}"
            );

            next USERID;
        }

        my %CurrentUserData = (
            LeaveDate        => 0,
            Sick             => 0,
            Overtime         => 0,
            TargetState      => 0,
            LeaveDayTotal    => 0,
            SickTotal        => 0,
            SickTotal        => 0,
            TargetStateTotal => 0,
        );

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

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

                DAY:
                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} ) {
                        $CurrentUserData{LeaveDayTotal}++;
                        $LeaveDay = 1;
                    }
                    elsif ( $WorkingUnit{Sick} ) {
                        $CurrentUserData{SickTotal}++;
                        $Sick = 1;
                    }
                    elsif ( $WorkingUnit{Overtime} ) {
                        $CurrentUserData{OvertimeTotal}++;
                        $Overtime = 1;
                    }

                    $CurrentUserData{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
                        )
                    {
                        $CurrentUserData{TargetStateTotal}
                            += $UserCurrentPeriod{$UserID}{WeeklyHours} / 5;
                        $TargetState = $UserCurrentPeriod{$UserID}{WeeklyHours} / 5;
                    }

                    if ( $Month == $MonthEnd && $Year == $YearEnd ) {
                        $CurrentUserData{TargetState}  += $TargetState;
                        $CurrentUserData{WorkingHours} += $WorkingUnit{Total};
                        $CurrentUserData{LeaveDay}     += $LeaveDay;
                        $CurrentUserData{Sick}         += $Sick;
                    }
                }
            }
        }
        $CurrentUserData{Overtime} = $CurrentUserData{WorkingHours} - $CurrentUserData{TargetState};
        $CurrentUserData{OvertimeTotal}
            = $UserCurrentPeriod{$UserID}{Overtime}
            + $CurrentUserData{WorkingHoursTotal}
            - $CurrentUserData{TargetStateTotal};
        $CurrentUserData{OvertimeUntil}
            = $CurrentUserData{OvertimeTotal} - $CurrentUserData{Overtime};
        $CurrentUserData{LeaveDayRemaining}
            = $UserCurrentPeriod{$UserID}{LeaveDays} - $CurrentUserData{LeaveDayTotal};

        $Data{$UserID} = \%CurrentUserData;
    }
    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
        ProjectDescription => 'description', # optional
        ProjectStatus      => 1 || 0,        # optional
    );

    returns ID of created project

=cut

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

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

    # insert project record
    my $Result = $Self->{DBObject}->Do(
        SQL => 'INSERT INTO time_accounting_project (project, description, status) '
            . 'VALUES (?, ?, ?)',
        Bind => [ \$Param{Project}, \$Param{ProjectDescription}, \$Param{ProjectStatus} ],
    );
    return if !$Result;

    # get ID of newly created project record
    $Result = $Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM time_accounting_project WHERE project = ?',
        Bind  => [ \$Param{Project} ],
        Limit => 1,
    );
    return if !$Result;

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

    return $ProjectID;
}

=item ProjectSettingsDelete()

delete records of project from database

    $TimeAccountingObject->ProjectSettingsDelete(
        ProjectID          => 3423, # ID of record
    );

=cut

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

    if ( !exists $Param{ProjectID} || !$Param{ProjectID} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'ProjectSettingsDelete: Need ProjectID!',
        );
        return;
    }

    # delete project record
    my $Result = $Self->{DBObject}->Do(
        SQL  => 'DELETE FROM time_accounting_project WHERE id = ?',
        Bind => [ \$Param{ProjectID}, ],
    );

    return $Result;
}

=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 $Date = sprintf( "%04d-%02d-%02d", $Year, $Month, $Day );
                my $DayStartTime
                    = $Self->{TimeObject}->TimeStamp2SystemTime( String => $Date . ' 00:00:00' );
                my $DayStopTime
                    = $Self->{TimeObject}->TimeStamp2SystemTime( String => $Date . ' 23:59:59' );

                # add time zone to calculation
                my $Zone = $Self->{ConfigObject}->Get( "TimeZone::Calendar" . $Calendar );
                if ($Zone) {
                    my $ZoneSeconds = $Zone * 60 * 60;
                    $DayStartTime = $DayStartTime - $ZoneSeconds;
                    $DayStopTime  = $DayStopTime - $ZoneSeconds;
                }

                my $ThisDayWorkingTime = $Self->{TimeObject}->WorkingTime(
                    StartTime => $DayStartTime,
                    StopTime  => $DayStopTime,
                    Calendar  => $Calendar || '',
                ) || '0';

                my $DayString = sprintf( "%02d", $Day );

                if ( $ThisDayWorkingTime && !$VacationCheck ) {
                    $WorkingDays++;
                }
                if (
                    $ThisDayWorkingTime
                    && !$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 (L<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 L<http://www.gnu.org/licenses/agpl.txt>.

=head1 VERSION

$Revision: 1.40.2.1 $ $Date: 2010-12-03 09:39:56 $

=cut

IyAtLQojIHNjcmlwdHMvdGVzdC9UaW1lQWNjb3VudGluZy50IC0gVGltZUFjY291bnRpbmcgdGVzdHNjcmlwdAojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogVGltZUFjY291bnRpbmcudCx2IDEuNiAyMDEwLTA3LTI5IDA4OjQwOjE2IGpwIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgojIGRlY2xhcmUgZXh0ZXJuYWxseSBkZWZpbmVkIHZhcmlhYmxlcyB0byBhdm9pZCBlcnJvcnMgdW5kZXIgJ3VzZSBzdHJpY3QnCnVzZSB2YXJzIHF3KCRTZWxmKTsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VGltZUFjY291bnRpbmc7CnVzZSBLZXJuZWw6OlN5c3RlbTo6VXNlcjsKCm15ICRVc2VyT2JqZWN0ICAgICAgICAgICA9IEtlcm5lbDo6U3lzdGVtOjpVc2VyLT5uZXcoICV7JFNlbGZ9ICk7Cm15ICRUaW1lQWNjb3VudGluZ09iamVjdCA9IEtlcm5lbDo6U3lzdGVtOjpUaW1lQWNjb3VudGluZy0+bmV3KAogICAgJXskU2VsZn0sCiAgICBVc2VySUQgICAgID0+IDEsCiAgICBVc2VyT2JqZWN0ID0+ICRVc2VyT2JqZWN0LAopOwoKIyBJRHMgb2YgY3JlYXRlZCBwcm9qZWN0cwpteSBAUHJvamVjdElEczsKCiMgaW5zZXJ0IHRlc3QgcHJvamVjdCBzZXR0aW5ncwpteSAkUHJvamVjdElEID0gJFRpbWVBY2NvdW50aW5nT2JqZWN0LT5Qcm9qZWN0U2V0dGluZ3NJbnNlcnQoCiAgICBQcm9qZWN0ICAgICAgICAgICAgPT4gJ1Rlc3QnLAogICAgUHJvamVjdERlc2NyaXB0aW9uID0+ICdEZXNjcmlwdGlvbicsCiAgICBQcm9qZWN0U3RhdHVzICAgICAgPT4gMSwKKTsKCiRTZWxmLT5UcnVlKAogICAgJFByb2plY3RJRCwKICAgICdJbnNlcnQgdGVzdCBwcm9qZWN0IHNldHRpbmdzIGludG8gZGF0YWJhc2UnLAopOwoKcHVzaCBAUHJvamVjdElEcywgJFByb2plY3RJRDsKCiMgY2hlY2sgaWYgcHJvamVjdCBzZXR0aW5ncyBoYXZlIChyZWFsbHkpIGJlZW4gY3JlYXRlZApteSAlUHJvamVjdERhdGEgPSAkVGltZUFjY291bnRpbmdPYmplY3QtPlByb2plY3RTZXR0aW5nc0dldCgKICAgIFN0YXR1cyA9PiAndmFsaWQnLAopOwoKbXkgJFByb2plY3RGb3VuZCAgICAgICAgICAgID0gMDsKbXkgJFByb2plY3REZXNjcmlwdGlvbkZvdW5kID0gMDsKCiRTZWxmLT5UcnVlKAogICAgZXhpc3RzICRQcm9qZWN0RGF0YXtQcm9qZWN0fS0+eyRQcm9qZWN0SUR9CiAgICAgICAgJiYgJFByb2plY3REYXRhe1Byb2plY3R9LT57JFByb2plY3RJRH0gZXEgJ1Rlc3QnCiAgICAgICAgJiYgZXhpc3RzICRQcm9qZWN0RGF0YXtQcm9qZWN0RGVzY3JpcHRpb259LT57JFByb2plY3RJRH0KICAgICAgICAmJiAkUHJvamVjdERhdGF7UHJvamVjdERlc2NyaXB0aW9ufS0+eyRQcm9qZWN0SUR9IGVxICdEZXNjcmlwdGlvbicsCiAgICAndGVzdCBwcm9qZWN0IGZvdW5kJywKKTsKCiMgZGVsZXRlIHRlc3QgcHJvamVjdChzKQpmb3IgbXkgJFByb2plY3RJRCAoQFByb2plY3RJRHMpIHsKICAgIG15ICRSZXN1bHQgPSAkVGltZUFjY291bnRpbmdPYmplY3QtPlByb2plY3RTZXR0aW5nc0RlbGV0ZSgKICAgICAgICBQcm9qZWN0SUQgPT4gJFByb2plY3RJRCwKICAgICk7CgogICAgJFNlbGYtPlRydWUoCiAgICAgICAgJFJlc3VsdCwKICAgICAgICAndGVzdCBwcm9qZWN0IGRlbGV0ZWQnLAogICAgKTsKCiAgICAjIFRPRE86CiAgICAjIHRlc3QgaWYgZGVsZXRlZCBwcm9qZWN0IHJlYWxseSBkb2Vzbid0IGV4aXN0IGFueW1vcmUKfQoKMTsK
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
