TimeAccounting
2.2.2
OTRS AG
http://otrs.org/
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
Build for TimeAccounting 2.2.2.
Build for TimeAccouting 2.2.1.
Build for TimeAccouting 2.2.0 rc1.
Build for TimeAccouting 2.2.0 beta2.
First release for OTRS 3.2.x.
A Time Registration Module.
Ein Zeiterfassungsmodul.
3.2.x
<br>
<b>WELCOME</b>
<br>
<br>
You are about to install the OTRS package TimeAccounting.<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>
((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>
((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>
((enjoy))<br>
<br>
<br>
<b>WILLKOMMEN</b>
<br>
<br>
Sie sind im Begriff das OTRS-Paket TimeAccounting zu aktualisieren.<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>
((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 method
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 method
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:15:34
opms.otrs.com
<?xml version="1.0" encoding="utf-8" ?>
<otrs_config version="1.0" init="Application">
    <ConfigItem Name="Frontend::Module###AgentTimeAccounting" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent::ModuleRegistration</SubGroup>
        <Setting>
            <FrontendModuleReg>
                <Group>time_accounting</Group>
                <GroupRo>time_accounting</GroupRo>
                <Description Translatable="1">Time accounting.</Description>
                <NavBarName>TimeAccounting</NavBarName>
                <Title>TimeAccountingTitle</Title>
                <NavBar>
                    <Description Translatable="1">Time accounting.</Description>
                    <Name>Time Accounting</Name>
                    <Link>Action=AgentTimeAccounting;Subaction=Edit</Link>
                    <NavBar>TimeAccounting</NavBar>
                    <Type>Menu</Type>
                    <Prio>6000</Prio>
                    <AccessKey></AccessKey>
                    <Block>ItemArea</Block>
                </NavBar>
                <NavBar>
                <GroupRo>time_accounting</GroupRo>
                    <Description Translatable="1">Overview</Description>
                    <Name>Overview</Name>
                    <Link>Action=AgentTimeAccounting;Subaction=Overview</Link>
                    <Type></Type>
                    <Block></Block>
                    <NavBar>TimeAccounting</NavBar>
                    <AccessKey>o</AccessKey>
                    <Prio>100</Prio>
                </NavBar>
                <NavBar>
                <GroupRo>time_accounting</GroupRo>
                    <Description Translatable="1">Edit time record</Description>
                    <Name>Edit</Name>
                    <Link>Action=AgentTimeAccounting;Subaction=Edit</Link>
                    <Type></Type>
                    <Block></Block>
                    <NavBar>TimeAccounting</NavBar>
                    <AccessKey>n</AccessKey>
                    <Prio>200</Prio>
                </NavBar>
                <NavBar>
                <Group>time_accounting</Group>
                    <Description Translatable="1">Project time reporting</Description>
                    <Name>Reporting</Name>
                    <Link>Action=AgentTimeAccounting;Subaction=Reporting</Link>
                    <Type></Type>
                    <Block></Block>
                    <NavBar>TimeAccounting</NavBar>
                    <AccessKey></AccessKey>
                    <Prio>500</Prio>
                </NavBar>
                <NavBar>
                    <Group>time_accounting</Group>
                    <Description Translatable="1">Edit time accounting settings</Description>
                    <Name>Settings</Name>
                    <Link>Action=AgentTimeAccounting;Subaction=Setting</Link>
                    <Type></Type>
                    <Block></Block>
                    <NavBar>TimeAccounting</NavBar>
                    <AccessKey></AccessKey>
                    <Prio>900</Prio>
                </NavBar>
                <Loader>
                    <CSS>TimeAccounting.Defaults.css</CSS>
                    <CSS>TimeAccounting.Table.css</CSS>
                    <JavaScript>thirdparty/jquery-ui-button-1.8.16/jquery-ui-button.js</JavaScript>
                    <JavaScript>thirdparty/jquery-combobox/jquery-combobox.js</JavaScript>
                    <JavaScript>Core.UI.ComboBox.js</JavaScript>
                    <JavaScript>TimeAccounting.Agent.ConfirmationDialog.js</JavaScript>
                    <JavaScript>TimeAccounting.Agent.EditTimeRecords.js</JavaScript>
                    <JavaScript>TimeAccounting.Agent.Datepicker.js</JavaScript>
                </Loader>
            </FrontendModuleReg>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultProjectName" Required="1" Valid="1">
        <Description Translatable="1">Default name for new projects.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex=""></String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultProjectStatus" Required="1" Valid="1">
        <Description Translatable="1">Default status for new projects.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <Option SelectedID="1">
                <Item Key="0">invalid</Item>
                <Item Key="1">valid</Item>
            </Option>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultActionName" Required="1" Valid="1">
        <Description Translatable="1">Default name for new actions.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex=""></String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultActionStatus" Required="1" Valid="1">
        <Description Translatable="1">Default status for new actions.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <Option SelectedID="1">
                <Item Key="0">invalid</Item>
                <Item Key="1">valid</Item>
            </Option>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultUserWeeklyHours" Required="1" Valid="1">
        <Description Translatable="1">Default setting for the standard weekly hours.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">40</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultUserLeaveDays" Required="1" Valid="1">
        <Description Translatable="1">Default setting for leave days.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">23</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultUserOvertime" Required="1" Valid="1">
        <Description Translatable="1">Default setting for overtime.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">0</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultUserDateStart" Required="1" Valid="1">
        <Description Translatable="1">Default setting for date start.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">2013-01-01</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultUserDateEnd" Required="1" Valid="1">
        <Description Translatable="1">Default setting for date end.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">2013-12-31</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultUserDescription" Required="1" Valid="1">
        <Description Translatable="1">Default setting for description.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">Put your description here.</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::DefaultUserStatus" Required="1" Valid="1">
        <Description Translatable="1">Default status for new users.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <Option SelectedID="1">
                <Item Key="0">invalid</Item>
                <Item Key="1">valid</Item>
            </Option>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::MaxIntervalOfIncompleteDays" Required="1" Valid="1">
        <Description Translatable="1">Maximum number of working days after which the working units have to be inserted.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">4</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::MaxIntervalOfIncompleteDaysBeforeWarning" Required="1" Valid="1">
        <Description Translatable="1">Maximum number of working days withouth working units entry after which a warning will be shown.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">2</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="Frontend::NotifyModule###888-TimeAccounting" Required="1" Valid="1">
        <Description Translatable="1">This notification module gives a warning if there are too many incomplete working days.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent::ModuleNotify</SubGroup>
        <Setting>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::NotificationTimeAccounting</Item>
            </Hash>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::MaxAllowedInsertDays" Required="1" Valid="1">
        <Description Translatable="1">For how many days ago you can insert working units.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">10</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::ReduceTime" Required="0" Valid="1">
        <Description Translatable="1">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).</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <Hash>
                <Item Key="traveling">50</Item>
            </Hash>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::InputHoursWithoutStartEndTime" Required="0" Valid="1">
        <Description Translatable="1">Specifies if working hours can be inserted without start and end times.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <Option SelectedID="1">
                <Item Key="0">No</Item>
                <Item Key="1">Yes</Item>
            </Option>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="PreApplicationModule###AgentTimeAccounting" Required="0" Valid="1">
        <Description Translatable="1">This module forces inserts in TimeAccounting.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">Kernel::Modules::AgentTimeAccounting</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::Project2RemarkRegExp" Required="0" Valid="1">
        <Description Translatable="1">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.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <String Regex="">^(?: PROJECT\# | INFRA: | PRODUCT: )</String>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::ProjectListConstraints" Required="0" Valid="1">
        <Description Translatable="1">Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <Hash>
            </Hash>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::ActionListConstraints" Required="0" Valid="1">
        <Description Translatable="1">Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <Hash>
            </Hash>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::EnableAutoCompletion" Required="0" Valid="1">
        <Description Translatable="1">If enabled, the dropdown elements in the edit screen are changed to autocompletion fields.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <Option SelectedID="1">
                <Item Key="0">No</Item>
                <Item Key="1">Yes</Item>
            </Option>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="TimeAccounting::AllowMassEntryForUser" Required="0" Valid="1">
        <Description Translatable="1">If enabled, the user is allowed to enter "on vacation leave", "on sick leave" and "on overtime leave" to multiple dates at once.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent</SubGroup>
        <Setting>
            <Option SelectedID="0">
                <Item Key="0">No</Item>
                <Item Key="1">Yes</Item>
            </Option>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="Frontend::ToolBarModule###201-TimeAccounting::IncompleteWorkingDays" Required="0" Valid="1">
        <Description Translatable="1">Agent interface notification module to see the number of incomplete working days for the user.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Frontend::Agent::ToolBarModule</SubGroup>
        <Setting>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ToolBarIncompleteWorkingDays</Item>
                <Item Key="CssClass">IncompleteWorkingDays</Item>
                <Item Key="Icon">icon-time</Item>
            </Hash>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="Stats::DynamicObjectRegistration###TimeAccounting" Required="0" Valid="1">
        <Description Translatable="1">Determines if the statistics module may generate time accounting information.</Description>
        <Group>TimeAccounting</Group>
        <SubGroup>Core::Stats</SubGroup>
        <Setting>
            <Hash>
                <Item Key="Module">Kernel::System::Stats::Dynamic::TimeAccounting</Item>
            </Hash>
        </Setting>
    </ConfigItem>
</otrs_config>

﻿# --
# Kernel/Language/de_TimeAccounting.pm - translation file
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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::de_TimeAccounting;

use utf8;

use strict;
use warnings;

sub Data {
    my $Self = shift;

    # Template: AgentTimeAccountingDelete
    $Self->{Translation}->{'Do you really want to delete the Time Accounting of this day?'} = 'Möchten Sie wirklich den Eintrag für den aktuellen Tag löschen?';

    # Template: AgentTimeAccountingEdit
    $Self->{Translation}->{'Edit Time Record'} = 'Zeiterfassung bearbeiten';
    $Self->{Translation}->{'Project settings'} = 'Projektkonfiguration';
    $Self->{Translation}->{'Date Navigation'} = 'Datumsauswahl';
    $Self->{Translation}->{'Previous day'} = 'Vorheriger Tag';
    $Self->{Translation}->{'Next day'} = 'Nächster Tag';
    $Self->{Translation}->{'Days without entries'} = 'Nicht ausgefüllte Tage';
    $Self->{Translation}->{'Required fields are marked with a "*".'} = 'Felder, die ausgefüllt werden müssen, sind mit einem Stern "*" gekennzeichnet.';
    $Self->{Translation}->{'You have to fill in start and end time or a time period.'} = 'Sie müssen eine Start- und Endzeit oder eine Zeitspanne angeben.';
    $Self->{Translation}->{'Project'} = 'Projekt';
    $Self->{Translation}->{'Task'} = 'Tätigkeit';
    $Self->{Translation}->{'Remark'} = 'Anmerkung';
    $Self->{Translation}->{'Date Navigation'} = 'Datumsnavigation';
    $Self->{Translation}->{'Please add a remark with more than 8 characters!.'} = 'Bitte geben Sie eine Anmerkung mit mindestens 8 Zeichen Länge ein.';
    $Self->{Translation}->{'Negative times are not allowed.'} = 'Negative Angaben sind nicht erlaubt.';
    $Self->{Translation}->{'Repeated hours are not allowed. Start time matches another interval.'} = 'Diese Startzeit wurde bereits in einem anderen Eintrag angegeben.';
    $Self->{Translation}->{'End time must be after start time.'} = 'Die Endzeit muss nach der Startzeit sein.';
    $Self->{Translation}->{'Repeated hours are not allowed. End time matches another interval.'} = 'Diese Endzeit wurde bereits in einem anderen Eintrag angegeben.';
    $Self->{Translation}->{'Period is bigger than the interval between start and end times!'} = 'Die angegebene Dauer ist länger als die Zeit zwischen Start- und Endzeit!';
    $Self->{Translation}->{'Invalid period! A day has only 24 hours.'} = 'Ungültige Angabe. Ein Tag hat nur 24 Stunden.';
    $Self->{Translation}->{'A valid period must be greater than zero.'} = 'Eine gültige Zeitdauer muss größer als Null sein.';
    $Self->{Translation}->{'Invalid period! Negative periods are not allowed.'} = 'Ungültige Angabe. Negative Zeitspannen sind nicht möglich.';
    $Self->{Translation}->{'Add one row'} = 'Eine Zeile hinzufügen';
    $Self->{Translation}->{'Total'} = 'Summe';
    $Self->{Translation}->{'On vacation'} = 'im Urlaub';
    $Self->{Translation}->{'You can only select one checkbox element!'} = 'Sie können nur eine Checkbox markieren!';
    $Self->{Translation}->{'On sick leave'} = 'Erkrankt';
    $Self->{Translation}->{'On overtime leave'} = 'Überstunden';
    $Self->{Translation}->{'Show all items'} = 'Alle Einträge anzeigen';
    $Self->{Translation}->{'Delete Time Accounting Entry'} = 'Eintrag löschen';
    $Self->{Translation}->{'Confirm insert'} = 'Eingabe bestätigen';
    $Self->{Translation}->{'Are you sure that you worked while you were on sick leave?'} = 'Sind Sie sicher, dass Sie gearbeitet haben, obwohl Sie erkrankt sind?';
    $Self->{Translation}->{'Are you sure that you worked while you were on vacation?'} = 'Sind Sie sicher, dass Sie gearbeitet haben, obwohl Sie im Urlaub sind?';
    $Self->{Translation}->{'Are you sure that you worked while you were on overtime leave?'} = 'Sind Sie sicher, dass Sie gearbeitet haben, obwohl Sie Überstunden genommen haben?';
    $Self->{Translation}->{'Are you sure that you worked more than 16 hours?'} = 'Sind Sie sicher, dass Sie mehr als 16 Stunden gearbeitet haben?';

    # Template: AgentTimeAccountingOverview
    $Self->{Translation}->{'Time reporting monthly overview'} = 'Monatsübersicht Zeiterfassung';
    $Self->{Translation}->{'Overtime (Hours)'} = 'Überstunden (in Stunden)';
    $Self->{Translation}->{'Overtime (this month)'} = 'Überstunden (dieser Monat)';
    $Self->{Translation}->{'Overtime (total)'} = 'Überstunden (Summe)';
    $Self->{Translation}->{'Remaining overtime leave'} = 'Überstunden (verbleibend)';
    $Self->{Translation}->{'Vacation (Days)'} = 'Urlaub (in Tagen)';
    $Self->{Translation}->{'Vacation taken (this month)'} = 'Urlaubstage (dieser Monat)';
    $Self->{Translation}->{'Vacation taken (total)'} = 'Urlaubstage (Summe)';
    $Self->{Translation}->{'Remaining vacation'} = 'Urlaubstage (verbleibend)';
    $Self->{Translation}->{'Sick Leave (Days)'} = 'Erkrankt (in Tagen)';
    $Self->{Translation}->{'Sick leave taken (this month)'} = 'Erkrankt (dieser Monat)';
    $Self->{Translation}->{'Sick leave taken (total)'} = 'Erkrankt (Summe)';
    $Self->{Translation}->{'Previous month'} = 'Vorheriger Monat';
    $Self->{Translation}->{'Next month'} = 'Nächster Monat';
    $Self->{Translation}->{'Day'} = 'Tag';
    $Self->{Translation}->{'Weekday'} = 'Wochentag';
    $Self->{Translation}->{'Working Hours'} = 'Arbeitsstunden';
    $Self->{Translation}->{'Total worked hours'} = 'Arbeitsstunden (gesamt)';
    $Self->{Translation}->{'User\'s project overview'} = 'Nutzerberichte';
    $Self->{Translation}->{'Hours (monthly)'} = 'Stunden (im Monat)';
    $Self->{Translation}->{'Hours (Lifetime)'} = 'Stunden (gesamt)';
    $Self->{Translation}->{'Grand total'} = 'Summe';

    # Template: AgentTimeAccountingProjectReporting
    $Self->{Translation}->{'Project report'} = 'Projektübersicht';

    # Template: AgentTimeAccountingReporting
    $Self->{Translation}->{'Time reporting'} = 'Zeitberichte';
    $Self->{Translation}->{'Month Navigation'} = 'Datumsauswahl';
    $Self->{Translation}->{'User reports'} = 'Nutzerberichte';
    $Self->{Translation}->{'Monthly total'} = 'pro Monat';
    $Self->{Translation}->{'Lifetime total'} = 'Summe';
    $Self->{Translation}->{'Overtime leave'} = 'Überstunden';
    $Self->{Translation}->{'Vacation'} = 'Urlaub';
    $Self->{Translation}->{'Sick leave'} = 'Erkrankt';
    $Self->{Translation}->{'Vacation remaining'} = 'Verbleibende Urlaubstage';
    $Self->{Translation}->{'Project reports'} = 'Projektberichte';

    # Template: AgentTimeAccountingSetting
    $Self->{Translation}->{'Edit Time Accounting Project Settings'} = 'Zeiterfassung: Bearbeitung der Projektkonfiguration';
    $Self->{Translation}->{'Add project'} = 'Projekt hinzufügen';
    $Self->{Translation}->{'Add Project'} = 'Projekt hinzufügen';
    $Self->{Translation}->{'Edit Project Settings'} = 'Projektkonfiguration bearbeiten';
    $Self->{Translation}->{'There is already a project with this name. Please, choose a different one.'} = 'Ein Projekt mit gleichem Namen existiert bereits. Bitte wählen Sie einen anderen Namen.';
    $Self->{Translation}->{'Edit Time Accounting Settings'} = 'Zeiterfassungseinstellungen bearbeiten';
    $Self->{Translation}->{'Add task'} = 'Tätigkeit hinzufügen';
    $Self->{Translation}->{'New user'} = 'Neuer Benutzer';
    $Self->{Translation}->{'Filter for Projects'} = 'Filter für Projekte';
    $Self->{Translation}->{'Filter for Tasks'} = 'Filter für Tätigkeiten';
    $Self->{Translation}->{'Filter for Users'} = 'Filter für Benutzer';
    $Self->{Translation}->{'Project List'} = 'Projektliste';
    $Self->{Translation}->{'Task List'} = 'Tätigkeitsliste';
    $Self->{Translation}->{'Add Task'} = 'Tätigkeit hinzufügen';
    $Self->{Translation}->{'Edit Task Settings'} = 'Tätigkeitseinstellungen bearbeiten';
    $Self->{Translation}->{'There is already a task with this name. Please, choose a different one.'} = 'Eine Tätigkeit mit gleichem Namen existiert bereits. Bitte wählen Sie einen anderen Namen.';
    $Self->{Translation}->{'User List'} = 'Benutzerliste';
    $Self->{Translation}->{'New User Settings'} = 'Einstellungen für neuen Benutzer';
    $Self->{Translation}->{'Edit User Settings'} = 'Benutzereinstellungen bearbeiten';
    $Self->{Translation}->{'Comments'} = 'Kommentare';
    $Self->{Translation}->{'Show Overtime'} = 'Überstunden anzeigen';
    $Self->{Translation}->{'Allow project creation'} = 'Projekt erstellen';
    $Self->{Translation}->{'Period Begin'} = 'Datum Beginn';
    $Self->{Translation}->{'Period End'} = 'Datum Ende';
    $Self->{Translation}->{'Days of Vacation'} = 'Urlaubstage';
    $Self->{Translation}->{'Hours per Week'} = 'Wochenstunden';
    $Self->{Translation}->{'Authorized Overtime'} = 'autorisierte Überstunden';
    $Self->{Translation}->{'Period end must be after period begin.'} = 'Das Ende der Zeitspanne muss nach dem Anfang sein.';
    $Self->{Translation}->{'No time periods found.'} = 'Keine Zeitspanne gefunden.';
    $Self->{Translation}->{'Add time period'} = 'Zeitspanne hinzufügen';

    # Template: AgentTimeAccountingView
    $Self->{Translation}->{'View Time Record'} = 'Datensatz anzeigen';
    $Self->{Translation}->{'View of '} = 'Ansicht von';
    $Self->{Translation}->{'Date navigation'} = 'Auswahl Datum';
    $Self->{Translation}->{'No data found for this day.'} = 'Kein Eintrag für diesen Tag gefunden.';

    # SysConfig
    $Self->{Translation}->{'Agent interface notification module to see the number of incomplete working days for the user.'} = 'Modul zum Anzeigen der Anzahl der unvollständigen Arbeitstage des Benutzers.';
    $Self->{Translation}->{'Default name for new actions.'} = 'Initialer Name einer neuen Tätigkeit in der Zeiterfassung.';
    $Self->{Translation}->{'Default name for new projects.'} = 'Anfangsstatus eines neuen Projekts.';
    $Self->{Translation}->{'Default setting for date end.'} = 'Enddatum für die Eingaben.';
    $Self->{Translation}->{'Default setting for date start.'} = 'Startdatum für die Eingaben.';
    $Self->{Translation}->{'Default setting for leave days.'} = 'Standard Urlaubstage.';
    $Self->{Translation}->{'Default setting for overtime.'} = 'Eventuell vorhandener Überstundenübertrag.';
    $Self->{Translation}->{'Default setting for the standard weekly hours.'} = 'Standard Wochenarbeitszeit.';
    $Self->{Translation}->{'Default status for new actions.'} = 'Anfangsstatus einer neuen Tätigkeit.';
    $Self->{Translation}->{'Default status for new projects.'} = 'Anfangsstatus eines neuen Projekts.';
    $Self->{Translation}->{'Default status for new users.'} = 'Anfangsstatus eines neuen Benutzers.';
    $Self->{Translation}->{'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.'} = 'Innerhalb dieser Konfigurationsoption kann eine RegExp definiert werden, die festlegt, bei welchen Projekten eine Bemerkung eingetragen werden muss (die RegExp arbeitet mit smx-Parametern).';
    $Self->{Translation}->{'Edit time accounting settings'} = 'Zeiterfassungseinstellungen bearbeiten';
    $Self->{Translation}->{'Edit time record'} = 'Zeiterfassung bearbeiten';
    $Self->{Translation}->{'For how many days ago you can insert working units.'} = 'Legt fest, bis wann man in ältere Zeiteinträge bearbeiten kann (z. B. 10 Tage zurückliegend).';
    $Self->{Translation}->{'If enabled, the dropdown elements in the edit screen are changed to autocompletion fields.'} = 'Aktiviert die Autocompletion auf den Eingabefeldern im Bearbeiten-Bereich.';
    $Self->{Translation}->{'Maximum number of working days after which the working units have to be inserted.'} = 'Maximale Anzahl von Arbeitstagen, nach der die Arbeitszeit eingetragen werden muss.';
    $Self->{Translation}->{'Maximum number of working days withouth working units entry after which a warning will be shown.'} = 'Maximale Anzahl von Arbeitstagen ohne Eintrag, nach der eine Warnung angezeigt wird.';
    $Self->{Translation}->{'Project time reporting'} = 'Projekt Zeiterfassung';
    $Self->{Translation}->{'Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).'} = 'Regulärer Ausdruck, um die Liste der Tätigkeiten bezüglich des ausgewählten Projekts einzuschränken. Der Schlüssel enthält einen Regulären Ausdruck für Projekte, der Wert einen Regulären Ausdruck für die Tätigkeiten.';
    $Self->{Translation}->{'Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.'} = 'Regulärer Ausdruck, um die Liste der Projekte bezüglich des aktiven Benutzers einzuschränken. Der Schlüssel enthält einen Regulären Ausdruck für Projekte, der Wert enthält eine komma-separierte Liste von Benutzergruppen.';
    $Self->{Translation}->{'Specifies if working hours can be inserted without start and end times.'} = 'Gibt an, ob Arbeitsstunden ohne Start- und Endzeit eingegeben werden können.';
    $Self->{Translation}->{'This module forces inserts in TimeAccounting.'} = 'Dieses Modul zwingt zur Eingabe von Stunden.';
    $Self->{Translation}->{'This notification module gives a warning if there are too many incomplete working days.'} = 'Modul, dass den Agent im Notification-Bereich des Agent-Interfaces darüber informiert, wenn schon zu lange keine Stunden mehr eingetragen wurden.';
    $Self->{Translation}->{'Time accounting.'} = 'Zeiterfassung.';
    $Self->{Translation}->{'To use if some actions reduced the working hours (for example, if only half of the traveling time is paid Key => traveling; Content => 50).'} = 'Hier kann man eingeben, ob für eine bestimmte Tätigkeit, die zu verrechnenden Stunden gekürzt werden. Z. B wenn Reisezeiten nur zur Hälfte vergütet werden (Key =&gt; journey; Content =&gt; 50).';
    $Self->{Translation}->{'Determines if the statistics module may generate time accounting information.'} = 'Bestimmt, ob das Statistik-Modul Informationen zur Zeiterfassung generieren kann.';

    $Self->{Translation}->{'Mon'} = 'Mo';
    $Self->{Translation}->{'Tue'} = 'Di';
    $Self->{Translation}->{'Wed'} = 'Mi';
    $Self->{Translation}->{'Thu'} = 'Do';
    $Self->{Translation}->{'Fri'} = 'Fr';
    $Self->{Translation}->{'Sat'} = 'Sa';
    $Self->{Translation}->{'Sun'} = 'So';
    $Self->{Translation}->{'January'} = 'Januar';
    $Self->{Translation}->{'February'} = 'Februar';
    $Self->{Translation}->{'March'} = 'März';
    $Self->{Translation}->{'April'} = 'April';
    $Self->{Translation}->{'May'} = 'Mai';
    $Self->{Translation}->{'June'} = 'Juni';
    $Self->{Translation}->{'July'} = 'Juli';
    $Self->{Translation}->{'August'} = 'August';
    $Self->{Translation}->{'September'} = 'September';
    $Self->{Translation}->{'October'} = 'Oktober';
    $Self->{Translation}->{'November'} = 'November';
    $Self->{Translation}->{'December'} = 'Dezember';

    $Self->{Translation}->{'Show all projects'} = 'Alle Projekte anzeigen';
    $Self->{Translation}->{'Show valid projects'} = 'Alle gültigen Projekte anzeigen';
    $Self->{Translation}->{'TimeAccounting'} = 'Zeiterfassung';
    $Self->{Translation}->{'Actions'} = 'Aktionen';
    $Self->{Translation}->{'User updated!'} = 'Benutzer aktualisiert!';
    $Self->{Translation}->{'User added!'} = 'Benutzer hinzugefügt!';
    $Self->{Translation}->{'Project added!'} = 'Projekt hinzugefügt!';
    $Self->{Translation}->{'Project updated!'} = 'Projekt aktualisiert!';
    $Self->{Translation}->{'Task added!'} = 'Tätigkeit hinzugefügt!';
    $Self->{Translation}->{'Task updated!'} = 'Tätigkeit aktualisiert!';

    #
    # OBSOLETE ENTRIES FOR REFERENCE, DO NOT TRANSLATE!
    #
    $Self->{Translation}->{'Can\'t delete Working Units!'} = 'Kann Arbeitsstunden nicht löschen!';
    $Self->{Translation}->{'Can\'t save settings, because a day has only 24 hours!'} = 'Ein Tag hat nur 24 Stunden!';
    $Self->{Translation}->{'Please insert your working hours!'} = 'Bitte tragen Sie Ihre Arbeitszeiten ein!';
    $Self->{Translation}->{'Reporting'} = 'Berichtswesen';
    $Self->{Translation}->{'Successful insert!'} = 'Erfolgreich eingefügt!';

}

1;

﻿# --
# Kernel/Language/da_TimeAccounting.pm - translation file
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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::da_TimeAccounting;

use utf8;

use strict;
use warnings;

sub Data {
    my $Self = shift;

    # Template: AgentTimeAccountingDelete
    $Self->{Translation}->{'Do you really want to delete the Time Accounting of this day?'} = 'Vil du virkelig slette tidsregnskabet for denne dag?';

    # Template: AgentTimeAccountingEdit
    $Self->{Translation}->{'Edit Time Record'} = 'Rediger tidsregistrering';
    $Self->{Translation}->{'Project settings'} = 'Projektindstillinger';
    $Self->{Translation}->{'Date Navigation'} = 'Datonavigation';
    $Self->{Translation}->{'Previous day'} = 'Forrige dag';
    $Self->{Translation}->{'Next day'} = 'Næste dag';
    $Self->{Translation}->{'Days without entries'} = 'Dage uden optegnelser';
    $Self->{Translation}->{'Required fields are marked with a "*".'} = 'De påkrævede felter er markeret med "*".';
    $Self->{Translation}->{'You have to fill in start and end time or a time period.'} = 'Du skal indtaste en start- og slutdato eller en tidsperiode.';
    $Self->{Translation}->{'Project'} = 'Projekt';
    $Self->{Translation}->{'Task'} = 'Opgave';
    $Self->{Translation}->{'Remark'} = 'Bemærkning';
    $Self->{Translation}->{'Date Navigation'} = 'Datonavigation';
    $Self->{Translation}->{'Please add a remark with more than 8 characters!.'} = 'Tilføj venligst en bemærkning med mere end 8 tegn!';
    $Self->{Translation}->{'Negative times are not allowed.'} = 'Negativ tid er ikke tilladt.';
    $Self->{Translation}->{'Repeated hours are not allowed. Start time matches another interval.'} = 'Gentagne timer er ikke tilladt. Starttiden svarer til et andet interval.';
    $Self->{Translation}->{'End time must be after start time.'} = 'Sluttiden skal være senere end starttiden.';
    $Self->{Translation}->{'Repeated hours are not allowed. End time matches another interval.'} = 'Gentagne timer er ikke tilladt. Sluttiden svarer til et andet interval.';
    $Self->{Translation}->{'Period is bigger than the interval between start and end times!'} = 'Perioden er længere end intervallet mellem start- og sluttid!';
    $Self->{Translation}->{'Invalid period! A day has only 24 hours.'} = 'Ugyldig periode! En dag har kun 24 timer.';
    $Self->{Translation}->{'A valid period must be greater than zero.'} = 'En gyldig periode skal være større end 0';
    $Self->{Translation}->{'Invalid period! Negative periods are not allowed.'} = 'Ugyldig periode! Negative perioder er ikke tilladt.';
    $Self->{Translation}->{'Add one row'} = 'Tilføj en række';
    $Self->{Translation}->{'Total'} = 'I alt';
    $Self->{Translation}->{'On vacation'} = 'På ferie';
    $Self->{Translation}->{'You can only select one checkbox element!'} = 'Du kan kun vælge et afkrydsningsfelt!';
    $Self->{Translation}->{'On sick leave'} = 'Sygefravær';
    $Self->{Translation}->{'On overtime leave'} = 'Afspadsering';
    $Self->{Translation}->{'Show all items'} = 'Vis alle punkter';
    $Self->{Translation}->{'Delete Time Accounting Entry'} = 'Slet tidsregnskabsindtastning';
    $Self->{Translation}->{'Confirm insert'} = 'Bekræft indsættelse';
    $Self->{Translation}->{'Are you sure that you worked while you were on sick leave?'} = 'Er du sikker på, at du arbejdede under dit sygefravær?';
    $Self->{Translation}->{'Are you sure that you worked while you were on vacation?'} = 'Er du sikker på, at du arbejdede, mens du var på ferie?';
    $Self->{Translation}->{'Are you sure that you worked while you were on overtime leave?'} = 'Er du sikker på, at du arbejdede, mens du var på afspadsering?';
    $Self->{Translation}->{'Are you sure that you worked more than 16 hours?'} = 'Er du sikker på, at du arbejdede mere end 16 timer?';

    # Template: AgentTimeAccountingOverview
    $Self->{Translation}->{'Time reporting monthly overview'} = 'Månedlig oversigt over tidsraportering';
    $Self->{Translation}->{'Overtime (Hours)'} = 'Overtid (timer)';
    $Self->{Translation}->{'Overtime (this month)'} = 'Overtid (denne måned)';
    $Self->{Translation}->{'Overtime (total)'} = 'Overtid (i alt)';
    $Self->{Translation}->{'Remaining overtime leave'} = 'Resterende afspadsering';
    $Self->{Translation}->{'Vacation (Days)'} = 'Ferie (dage)';
    $Self->{Translation}->{'Vacation taken (this month)'} = 'Feriedage brugt (denne måned)';
    $Self->{Translation}->{'Vacation taken (total)'} = 'Feriedage brugt (i alt)';
    $Self->{Translation}->{'Remaining vacation'} = 'Resterende feriedage';
    $Self->{Translation}->{'Sick Leave (Days)'} = 'Sygefravær (dage)';
    $Self->{Translation}->{'Sick leave taken (this month)'} = 'Sygefravær brugt (denne måned)';
    $Self->{Translation}->{'Sick leave taken (total)'} = 'Sygefravær brugt (i alt)';
    $Self->{Translation}->{'Previous month'} = 'Forrige måned';
    $Self->{Translation}->{'Next month'} = 'Næste måned';
    $Self->{Translation}->{'Day'} = 'Dag';
    $Self->{Translation}->{'Weekday'} = 'Ugedag';
    $Self->{Translation}->{'Working Hours'} = 'Arbejdstimer';
    $Self->{Translation}->{'Total worked hours'} = 'Arbejdstimer i alt';
    $Self->{Translation}->{'User\'s project overview'} = 'Brugers projektoverblik';
    $Self->{Translation}->{'Hours (monthly)'} = 'Timer (månedlig)';
    $Self->{Translation}->{'Hours (Lifetime)'} = 'Samlet antal timer';
    $Self->{Translation}->{'Grand total'} = 'Alt i alt';

    # Template: AgentTimeAccountingProjectReporting
    $Self->{Translation}->{'Project report'} = 'Projektrapport';

    # Template: AgentTimeAccountingReporting
    $Self->{Translation}->{'Time reporting'} = 'Tidsraportering';
    $Self->{Translation}->{'Month Navigation'} = 'Månedsnavigation';
    $Self->{Translation}->{'User reports'} = 'Brugerrapporter';
    $Self->{Translation}->{'Monthly total'} = 'I alt for måned';
    $Self->{Translation}->{'Lifetime total'} = 'Samlet total';
    $Self->{Translation}->{'Overtime leave'} = 'Afspadsering';
    $Self->{Translation}->{'Vacation'} = 'Ferie';
    $Self->{Translation}->{'Sick leave'} = 'Sygefravær';
    $Self->{Translation}->{'Vacation remaining'} = 'Resterende fraværsdage';
    $Self->{Translation}->{'Project reports'} = 'Projektrapporter';

    # Template: AgentTimeAccountingSetting
    $Self->{Translation}->{'Edit Time Accounting Project Settings'} = 'Rediger indstillinger for tidsregnskabsprojekt';
    $Self->{Translation}->{'Add project'} = 'Tilføj projekt';
    $Self->{Translation}->{'Add Project'} = 'Tilføj projekt';
    $Self->{Translation}->{'Edit Project Settings'} = 'Rediger projektindstillinger';
    $Self->{Translation}->{'There is already a project with this name. Please, choose a different one.'} = 'Der er allerede et projekt med dette navn. Vælg venligst et andet.';
    $Self->{Translation}->{'Edit Time Accounting Settings'} = 'Rediger tidsregnskabsindstilinger';
    $Self->{Translation}->{'Add task'} = 'Tilføj opgave';
    $Self->{Translation}->{'New user'} = 'Ny bruger';
    $Self->{Translation}->{'Filter for Projects'} = 'Filtrér projekter';
    $Self->{Translation}->{'Filter for Tasks'} = 'Filtrér opgaver';
    $Self->{Translation}->{'Filter for Users'} = 'Filtrér brugere';
    $Self->{Translation}->{'Project List'} = 'Projektliste';
    $Self->{Translation}->{'Task List'} = 'Opgaveliste';
    $Self->{Translation}->{'Add Task'} = 'Tilføj opgave';
    $Self->{Translation}->{'Edit Task Settings'} = 'Rediger opgaveindstillinger';
    $Self->{Translation}->{'There is already a task with this name. Please, choose a different one.'} = 'Der er allerede en opgave med dette navn. Vælg venligst et andet.';
    $Self->{Translation}->{'User List'} = 'Brugerliste';
    $Self->{Translation}->{'New User Settings'} = 'Nye brugerindstillinger';
    $Self->{Translation}->{'Edit User Settings'} = 'Rediger brugerindstillinger';
    $Self->{Translation}->{'Comments'} = 'Kommentarer';
    $Self->{Translation}->{'Show Overtime'} = 'Vis overtid';
    $Self->{Translation}->{'Allow project creation'} = 'Tillad oprettelse af projekt';
    $Self->{Translation}->{'Period Begin'} = 'Begynd periode';
    $Self->{Translation}->{'Period End'} = 'Afslut periode';
    $Self->{Translation}->{'Days of Vacation'} = 'Feriedage';
    $Self->{Translation}->{'Hours per Week'} = 'Timer pr. uge';
    $Self->{Translation}->{'Authorized Overtime'} = 'Godkendt overtid';
    $Self->{Translation}->{'Period end must be after period begin.'} = 'Periodens afslutning skal være efter periodens start.';
    $Self->{Translation}->{'No time periods found.'} = 'Ingen tidsperioder fundet.';
    $Self->{Translation}->{'Add time period'} = 'Tilføj tidsperiode';

    # Template: AgentTimeAccountingView
    $Self->{Translation}->{'View Time Record'} = 'Se tidsregistrering';
    $Self->{Translation}->{'View of '} = 'Se ';
    $Self->{Translation}->{'Date navigation'} = 'Datonavigation';
    $Self->{Translation}->{'No data found for this day.'} = 'Der er ikke fundet data for denne dag.';

    # SysConfig
    $Self->{Translation}->{'Agent interface notification module to see the number of incomplete working days for the user.'} = 'Agent interface-notificationsmodulet skal se antallet af uafsluttede arbejdsdage for brugeren.';
    $Self->{Translation}->{'Default name for new actions.'} = 'Standardnavn for nye handlinger.';
    $Self->{Translation}->{'Default name for new projects.'} = 'Standardnavn for nye projekter.';
    $Self->{Translation}->{'Default setting for date end.'} = 'Standardindstilling for slutdato.';
    $Self->{Translation}->{'Default setting for date start.'} = 'Standardindstilling for begyndelsesdato.';
    $Self->{Translation}->{'Default setting for leave days.'} = 'Standardindstilling for fraværsdage.';
    $Self->{Translation}->{'Default setting for overtime.'} = 'Standardindstilling for overtid.';
    $Self->{Translation}->{'Default setting for the standard weekly hours.'} = 'Standardindstilling for normale ugentlige timer.';
    $Self->{Translation}->{'Default status for new actions.'} = 'Standardindstilling for nye handlinger.';
    $Self->{Translation}->{'Default status for new projects.'} = 'Standardindstilling for nye projekter.';
    $Self->{Translation}->{'Default status for new users.'} = 'Standardindstilling for nye brugere.';
    $Self->{Translation}->{'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.'} = 'Definerer de projekter, for hvilke en bemærkning er påkrævet. Hvis RegExp matcher på projektet, skal du også indføje en bemærkning. RegExp bruger smx parameteret.';
    $Self->{Translation}->{'Edit time accounting settings'} = 'Rediger indstillinger for tidsregistrering';
    $Self->{Translation}->{'Edit time record'} = 'Rediger tidsoptegnelse';
    $Self->{Translation}->{'For how many days ago you can insert working units.'} = 'Angiver for hvor mange dage, du kan indsætte arbejdsenheder.';
    $Self->{Translation}->{'If enabled, the dropdown elements in the edit screen are changed to autocompletion fields.'} = 'Hvis denne er aktiveret, ændres dropdown-elementerne på redigeringsskærmbilledet til autofuldføringsfelter.';
    $Self->{Translation}->{'Maximum number of working days after which the working units have to be inserted.'} = 'Maksimalt antal af arbejdsdage efter hvilket arbejdsenhederne skal indsættes.';
    $Self->{Translation}->{'Maximum number of working days withouth working units entry after which a warning will be shown.'} = 'Maksimalt antal af arbejdsdage uden indtastning af arbejdsenheder efter hvilket en advarsel vil blive vist.';
    $Self->{Translation}->{'Project time reporting'} = 'Projekttidsrapportering';
    $Self->{Translation}->{'Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).'} = 'Regulære udtryk for begrænsning af handlingsliste ifølge det valgte projekt. Nøglen indeholder et regulært udtryk for projekt(er), indholdet rummer regulære udtryk for handling(er).';
    $Self->{Translation}->{'Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.'} = 'Regulære udtryk for begrænsning af projektliste ifølge brugergrupper. Nøglen indeholder et regulært udtryk for projekt(er), indholdet rummer kommasepareret liste over grupper.';
    $Self->{Translation}->{'Specifies if working hours can be inserted without start and end times.'} = 'Angiver, om arbejdstimer kan indsættes uden start- og sluttider.';
    $Self->{Translation}->{'This module forces inserts in TimeAccounting.'} = 'Dette modul gennemtvinger indsættelser i tidsregnskab.';
    $Self->{Translation}->{'This notification module gives a warning if there are too many incomplete working days.'} = 'Dette adviseringsmodul giver en advarsel, hvis der er for mange ufuldstændige arbejdsdage.';
    $Self->{Translation}->{'Time accounting.'} = 'Tidsregnskab';
    $Self->{Translation}->{'To use if some actions reduced the working hours (for example, if only half of the traveling time is paid Key => traveling; Content => 50).'} = 'Bruges, hvis nogle handlinger har reduceret arbejdstimerne (f.eks. hvis kun halvdelen af rejsetiden er betalt Key => traveling; Content => 50)';

    $Self->{Translation}->{'Mon'} = 'Man';
    $Self->{Translation}->{'Tue'} = 'Tirs';
    $Self->{Translation}->{'Wed'} = 'Ons';
    $Self->{Translation}->{'Thu'} = 'Tors';
    $Self->{Translation}->{'Fri'} = 'Fre';
    $Self->{Translation}->{'Sat'} = 'Lør';
    $Self->{Translation}->{'Sun'} = 'Søn';
    $Self->{Translation}->{'January'} = 'Januar';
    $Self->{Translation}->{'February'} = 'Februar';
    $Self->{Translation}->{'March'} = 'Marts';
    $Self->{Translation}->{'April'} = 'April';
    $Self->{Translation}->{'May'} = 'Maj';
    $Self->{Translation}->{'June'} = 'Juni';
    $Self->{Translation}->{'July'} = 'Juli';
    $Self->{Translation}->{'August'} = 'August';
    $Self->{Translation}->{'September'} = 'September';
    $Self->{Translation}->{'October'} = 'Oktober';
    $Self->{Translation}->{'November'} = 'November';
    $Self->{Translation}->{'December'} = 'December';

    $Self->{Translation}->{'Show all projects'} = 'Vis alle projekter';
    $Self->{Translation}->{'Show valid projects'} = 'Vis gyldige projekter';
    $Self->{Translation}->{'TimeAccounting'} = 'Tidsregnskab';
    $Self->{Translation}->{'Actions'} = 'Handlinger';
    $Self->{Translation}->{'User updated!'} = 'Bruger opdateret!';
    $Self->{Translation}->{'User added!'} = 'Bruger tilføjet!';
    $Self->{Translation}->{'Project added!'} = 'Projekt tilføjet!';
    $Self->{Translation}->{'Project updated!'} = 'Projekt opdateret!';
    $Self->{Translation}->{'Task added!'} = 'Opgave tilføjet!';
    $Self->{Translation}->{'Task updated!'} = 'Opgave opdateret!';

    #
    # OBSOLETE ENTRIES FOR REFERENCE, DO NOT TRANSLATE!
    #
    $Self->{Translation}->{'Can\'t delete Working Units!'} = 'Tidsenheder kan ikke slettes!';
    $Self->{Translation}->{'Can\'t save settings, because a day has only 24 hours!'} = 'Indstillinger kan ikke gemmes, fordi et døgn kun har 24 timer!';
    $Self->{Translation}->{'Please insert your working hours!'} = 'Indtast venligst dine arbejdstimer!';
    $Self->{Translation}->{'Reporting'} = 'Raportering';
    $Self->{Translation}->{'Successful insert!'} = 'Vellykket indtastning!';

}

1;

﻿# --
# Kernel/Language/es_TimeAccounting.pm - translation file
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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::es_TimeAccounting;

use utf8;

use strict;
use warnings;

sub Data {
    my $Self = shift;

    # Template: AgentTimeAccountingDelete
    $Self->{Translation}->{'Do you really want to delete the Time Accounting of this day?'} = '¿Realmente desea eliminar la Contabilidad de Tiempo de este día?';

    # Template: AgentTimeAccountingEdit
    $Self->{Translation}->{'Edit Time Record'} = 'Modificar Registro de Tiempo';
    $Self->{Translation}->{'Project settings'} = 'Configuraciones de proyecto';
    $Self->{Translation}->{'Previous day'} = 'Día anterior';
    $Self->{Translation}->{'Next day'} = 'Día siguiente';
    $Self->{Translation}->{'Days without entries'} = 'Días sin registro';
    $Self->{Translation}->{'Required fields are marked with a "*".'} = 'Los campos requeridos están marcados con un "*".';
    $Self->{Translation}->{'You have to fill in start and end time or a time period.'} = 'Es necesario que proporcione una hora de inicio y una de finalización o un periodo.';
    $Self->{Translation}->{'Project'} = 'Proyecto';
    $Self->{Translation}->{'Task'} = 'Tarea';
    $Self->{Translation}->{'Remark'} = 'Comentario';
    $Self->{Translation}->{'Date Navigation'} = 'Navegación por Fecha';
    $Self->{Translation}->{'Please add a remark with more than 8 characters!.'} = '¡Por favor, escriba un comentario de más de 8 caracteres!';
    $Self->{Translation}->{'Negative times are not allowed.'} = 'No se permite la inserción de horas negativas.';
    $Self->{Translation}->{'Repeated hours are not allowed. Start time matches another interval.'} = 'No se permite la inserción de horas repetidas. La hora de inicio está incluída en otro intervalo.';
    $Self->{Translation}->{'End time must be after start time.'} = 'La hora de finalización debe ser posterior a la de inicio.';
    $Self->{Translation}->{'Repeated hours are not allowed. End time matches another interval.'} = 'No se permite la inserción de horas repetidas. La hora de finalización está incluída en otro intervalo.';
    $Self->{Translation}->{'Period is bigger than the interval between start and end times!'} = '¡El periodo es mayor al intervalo entre las horas de inicio y finalización!';
    $Self->{Translation}->{'Invalid period! A day has only 24 hours.'} = '¡Periodo inválido! Un día sólo tiene 24 horas.';
    $Self->{Translation}->{'A valid period must be greater than zero.'} = 'Un periodo válido debe ser mayor a cero.';
    $Self->{Translation}->{'Invalid period! Negative periods are not allowed.'} = '¡Periodo inválido! No se permiten periodos negativos.';
    $Self->{Translation}->{'Add one row'} = 'Añadir una fila';
    $Self->{Translation}->{'Total'} = 'Total';
    $Self->{Translation}->{'On vacation'} = 'De vacaciones';
    $Self->{Translation}->{'You can only select one checkbox element!'} = '¡Sólo se permite elegir un elemento checkbox!';
    $Self->{Translation}->{'On sick leave'} = 'Ausente por enfermedad';
    $Self->{Translation}->{'On overtime leave'} = 'Ausente por haber trabajado horas extras';
    $Self->{Translation}->{'Show all items'} = 'Mostrar todos los elementos';
    $Self->{Translation}->{'Delete Time Accounting Entry'} = 'Eliminar Registro de Contabilidad de Tiempo';
    $Self->{Translation}->{'Confirm insert'} = 'Confirmar inserción';
    $Self->{Translation}->{'Are you sure that you worked while you were on sick leave?'} = '¿Está seguro que trabajó durante su ausencia por enfermedad?';
    $Self->{Translation}->{'Are you sure that you worked while you were on vacation?'} = '¿Está seguro que trabajó durante sus vacaciones?';
    $Self->{Translation}->{'Are you sure that you worked while you were on overtime leave?'} = '¿Está seguro que trabajó durante su ausencia por haber trabajado horas extras?';
    $Self->{Translation}->{'Are you sure that you worked more than 16 hours?'} = '¿Está seguro que trabajó más de 16 horas?';

    # Template: AgentTimeAccountingOverview
    $Self->{Translation}->{'Time reporting monthly overview'} = 'Vista mensual de tiempos reportados';
    $Self->{Translation}->{'Overtime (Hours)'} = 'Número de horas extras';
    $Self->{Translation}->{'Overtime (this month)'} = 'Horas extras trabajadas (este mes)';
    $Self->{Translation}->{'Overtime (total)'} = 'Total de horas extras';
    $Self->{Translation}->{'Remaining overtime leave'} = 'Horas restantes para ausentarse por haber trabajado horas extras';
    $Self->{Translation}->{'Vacation (Days)'} = 'Vacaciones (días)';
    $Self->{Translation}->{'Vacation taken (this month)'} = 'Vacaciones (este mes)';
    $Self->{Translation}->{'Vacation taken (total)'} = 'Vacaciones tomadas (total)';
    $Self->{Translation}->{'Remaining vacation'} = 'Días de vacaciones restantes';
    $Self->{Translation}->{'Sick Leave (Days)'} = 'Ausencia por enfermedad (días)';
    $Self->{Translation}->{'Sick leave taken (this month)'} = 'Ausencias por enfermedad (este mes)';
    $Self->{Translation}->{'Sick leave taken (total)'} = 'Ausencias por enfermedad (total)';
    $Self->{Translation}->{'Previous month'} = 'Mes anterior';
    $Self->{Translation}->{'Next month'} = 'Mes siguiente';
    $Self->{Translation}->{'Day'} = 'Día';
    $Self->{Translation}->{'Weekday'} = 'Día de la semana';
    $Self->{Translation}->{'Working Hours'} = 'Horas trabajadas';
    $Self->{Translation}->{'Total worked hours'} = 'Total de horas trabajadas';
    $Self->{Translation}->{'User\'s project overview'} = 'Vista de proyecto de usuario';
    $Self->{Translation}->{'Hours (monthly)'} = 'Horas (mensuales)';
    $Self->{Translation}->{'Hours (Lifetime)'} = 'Horas (tiempo de vida)';
    $Self->{Translation}->{'Grand total'} = 'Total neto';

    # Template: AgentTimeAccountingProjectReporting
    $Self->{Translation}->{'Project report'} = 'Reporte de proyecto';

    # Template: AgentTimeAccountingReporting
    $Self->{Translation}->{'Time reporting'} = 'Reporte de tiempos';
    $Self->{Translation}->{'Month Navigation'} = 'Navegación por mes';
    $Self->{Translation}->{'User reports'} = 'Reportes de usuario';
    $Self->{Translation}->{'Monthly total'} = 'Total mensual';
    $Self->{Translation}->{'Lifetime total'} = 'Tiempo total de vida';
    $Self->{Translation}->{'Overtime leave'} = 'Ausencia por haber trabajado horas extras';
    $Self->{Translation}->{'Vacation'} = 'Vacaciones';
    $Self->{Translation}->{'Sick leave'} = 'Ausencia por enfermedad';
    $Self->{Translation}->{'Vacation remaining'} = 'Días libres restantes';
    $Self->{Translation}->{'Project reports'} = 'Reportes de proyecto';

    # Template: AgentTimeAccountingSetting
    $Self->{Translation}->{'Edit Time Accounting Project Settings'} = 'Modificar la configuración del proyecto de contabilidad de tiempo.';
    $Self->{Translation}->{'Add project'} = 'Agregar proyecto';
    $Self->{Translation}->{'Add Project'} = 'Agregar Proyecto';
    $Self->{Translation}->{'Edit Project Settings'} = 'Modificar la Configuración del Proyecto';
    $Self->{Translation}->{'There is already a project with this name. Please, choose a different one.'} = 'Ya existe un proyecto con este nombre. Por favor, elija uno diferente.';
    $Self->{Translation}->{'Edit Time Accounting Settings'} = 'Modificar la Configuración de la Contabilidad de Tiempo';
    $Self->{Translation}->{'Add task'} = 'Agregar tarea';
    $Self->{Translation}->{'New user'} = 'Usuario nuevo';
    $Self->{Translation}->{'Filter for Projects'} = 'Filtro para Proyectos';
    $Self->{Translation}->{'Filter for Tasks'} = 'Filtro para Tareas';
    $Self->{Translation}->{'Filter for Users'} = 'Filtro para Usuarios';
    $Self->{Translation}->{'Project List'} = 'Lista de Proyectos';
    $Self->{Translation}->{'Task List'} = 'Lista de Tareas';
    $Self->{Translation}->{'Add Task'} = 'Agregar Tarea';
    $Self->{Translation}->{'Edit Task Settings'} = 'Modificar la Configuración de la Tarea';
    $Self->{Translation}->{'There is already a task with this name. Please, choose a different one.'} = 'Ya existe una tarea con este nombre. Por favor, elija uno diferente.';
    $Self->{Translation}->{'User List'} = 'Lista de Usuarios';
    $Self->{Translation}->{'New User Settings'} = 'Configuración de Usuario Nuevo';
    $Self->{Translation}->{'Edit User Settings'} = 'Modificar la Configuración del Usuario';
    $Self->{Translation}->{'Comments'} = 'Comentarios';
    $Self->{Translation}->{'Show Overtime'} = 'Mostrar horas extra';
    $Self->{Translation}->{'Allow project creation'} = 'Permitir creación de proyectos';
    $Self->{Translation}->{'Period Begin'} = 'Periodo de inicio';
    $Self->{Translation}->{'Period End'} = 'Periodo de finalización';
    $Self->{Translation}->{'Days of Vacation'} = 'Días de Vacaciones';
    $Self->{Translation}->{'Hours per Week'} = 'Horas por Semana';
    $Self->{Translation}->{'Authorized Overtime'} = 'Tiempo Extra Autorizado';
    $Self->{Translation}->{'Period end must be after period begin.'} = 'El periodo de finalización debe ser posterior al de inicio.';
    $Self->{Translation}->{'No time periods found.'} = 'No se encontraron periodos de tiempo.';
    $Self->{Translation}->{'Add time period'} = 'Agregar periodo de tiempo';

    # Template: AgentTimeAccountingView
    $Self->{Translation}->{'View Time Record'} = 'Ver registro de tiempo';
    $Self->{Translation}->{'View of '} = 'Resumen de ';
    $Self->{Translation}->{'Date navigation'} = 'Navegación por fecha';
    $Self->{Translation}->{'No data found for this day.'} = 'No se encontraron datos de este día.';

    # SysConfig
    $Self->{Translation}->{'Agent interface notification module to see the number of incomplete working days for the user.'} = 'Módulo de notificación de la interfaz del agente, necesario para visualizar el número de días incompletos de un usuario.';
    $Self->{Translation}->{'Default name for new actions.'} = 'Nombre por defecto para acciones nuevas.';
    $Self->{Translation}->{'Default name for new projects.'} = 'Nombre por defecto para proyectos nuevos.';
    $Self->{Translation}->{'Default setting for date end.'} = 'Configuración por defecto para la fecha de finalización.';
    $Self->{Translation}->{'Default setting for date start.'} = 'Configuración por defecto para la fecha de inicio.';
    $Self->{Translation}->{'Default setting for leave days.'} = 'Configuración por defecto para las faltas (días de ausencia).';
    $Self->{Translation}->{'Default setting for overtime.'} = 'Configuración por defecto para las horas extras.';
    $Self->{Translation}->{'Default setting for the standard weekly hours.'} = 'Configuración por defecto para las horas semanales estándar.';
    $Self->{Translation}->{'Default status for new actions.'} = 'Estado por defecto para acciones nuevas.';
    $Self->{Translation}->{'Default status for new projects.'} = 'Estado por defecto para proyectos nuevos.';
    $Self->{Translation}->{'Default status for new users.'} = 'Estado por defecto para usuarios nuevos.';
    $Self->{Translation}->{'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.'} = 'Define para cuáles proyectos es obligatoria la inserción de comentarios. Si la expresión regular coincide con el proyecto, se tiene que escribir un comentario. La expresión regular usa el parámetro smx.';
    $Self->{Translation}->{'Edit time accounting settings'} = 'Modificar la configuración de la contabilidad de tiempo';
    $Self->{Translation}->{'Edit time record'} = 'Modificar registro de tiempo';
    $Self->{Translation}->{'For how many days ago you can insert working units.'} = 'Especifica el número máximo de días para registrar horas de trabajo atrasadas (por ejemplo: 10 días de antigüedad).';
    $Self->{Translation}->{'If enabled, the dropdown elements in the edit screen are changed to autocompletion fields.'} = 'Si se habilita, los elementos desplegables de la ventana de edición se cambian por campos de autocompletado.';
    $Self->{Translation}->{'Maximum number of working days after which the working units have to be inserted.'} = 'Número máximo de días laborales después de los cuales es obligatorio reportar las horas trabajadas.';
    $Self->{Translation}->{'Maximum number of working days withouth working units entry after which a warning will be shown.'} = 'Número máximo de días laborales sin registro de horas trabajadas después de los cuales se mostrará una advertencia.';
    $Self->{Translation}->{'Project time reporting'} = 'Reporte de tiempos de proyecto';
    $Self->{Translation}->{'Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).'} = 'Expresiones regulares para restringir la lista de acciones, de acuerdo al proyecto seleccionado. Key contiene la expresión regular para los proyectos y Content, las expresiones regulares para las acciones.';
    $Self->{Translation}->{'Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.'} = 'Expresiones regulares para restringir la lista de proyectos, de acuerdo a los grupos de usuarios. Key contiene la expresión regular para los proyectos y Content, una lista de grupos, separados por comas.';
    $Self->{Translation}->{'Specifies if working hours can be inserted without start and end times.'} = 'Especifica si es posible insertar las horas trabajadas sin definir los tiempos de inicio y finalización de las tareas.';
    $Self->{Translation}->{'This module forces inserts in TimeAccounting.'} = 'Este módulo hace obligatoria la inserción de horas trabajadas.';
    $Self->{Translation}->{'This notification module gives a warning if there are too many incomplete working days.'} = 'Módulo de notificación que muestra una advertencia si hay demasiados días sin registro de horas trabajadas.';
    $Self->{Translation}->{'Time accounting.'} = 'Contabilidad de tiempo.';
    $Self->{Translation}->{'To use if some actions reduced the working hours (for example, if only half of the traveling time is paid Key => traveling; Content => 50).'} = 'Se usa si algunas acciones reducen el número de horas trabajadas (por ejemplo: si sólo se paga la mitad del tiempo de viaje puede especificarse Key => Viaje; Content => 50).';

    $Self->{Translation}->{'Mon'} = 'Lun';
    $Self->{Translation}->{'Tue'} = 'Mar';
    $Self->{Translation}->{'Wed'} = 'Miér';
    $Self->{Translation}->{'Thu'} = 'Jue';
    $Self->{Translation}->{'Fri'} = 'Vier';
    $Self->{Translation}->{'Sat'} = 'Sáb';
    $Self->{Translation}->{'Sun'} = 'Dom';
    $Self->{Translation}->{'January'} = 'Enero';
    $Self->{Translation}->{'February'} = 'Febrero';
    $Self->{Translation}->{'March'} = 'Marzo';
    $Self->{Translation}->{'April'} = 'Abril';
    $Self->{Translation}->{'May'} = 'Mayo';
    $Self->{Translation}->{'June'} = 'Junio';
    $Self->{Translation}->{'July'} = 'Julio';
    $Self->{Translation}->{'August'} = 'Agosto';
    $Self->{Translation}->{'September'} = 'Septiembre';
    $Self->{Translation}->{'October'} = 'Octubre';
    $Self->{Translation}->{'November'} = 'Noviembre';
    $Self->{Translation}->{'December'} = 'Diciembre';

    $Self->{Translation}->{'Show all projects'} = 'Mostrar todos los proyectos';
    $Self->{Translation}->{'Show valid projects'} = 'Mostrar todos los proyectos válidos';
    $Self->{Translation}->{'TimeAccounting'} = 'Contabilidad De Tiempo';
    $Self->{Translation}->{'Actions'} = 'Acciones';
    $Self->{Translation}->{'User updated!'} = '¡Usuario actualizado!';
    $Self->{Translation}->{'User added!'} = '¡Usuario añadido!';
    $Self->{Translation}->{'Project added!'} = '¡Proyecto añadido!';
    $Self->{Translation}->{'Project updated!'} = '¡Proyecto actualizado!';
    $Self->{Translation}->{'Task added!'} = '¡Tarea añadida!';
    $Self->{Translation}->{'Task updated!'} = '¡Tarea actualizada!';

    #
    # OBSOLETE ENTRIES FOR REFERENCE, DO NOT TRANSLATE!
    #

    $Self->{Translation}->{'Reporting'} = 'Reporte';
    $Self->{Translation}->{'Successful insert!'} = '¡Inserción exitosa!';
    $Self->{Translation}->{'Can\'t save settings, because a day has only 24 hours!'} = '¡Imposible guardar la configuración porque un día sólo tiene 24 horas!';
    $Self->{Translation}->{'Can\'t delete Working Units!'} = '¡Imposible eliminar las unidades de trabajo!';
    $Self->{Translation}->{'Please insert your working hours!'} = '¡Favor de insertar sus horas de trabajo!';
}

1;

﻿# --
# Kernel/Language/hu_TimeAccounting.pm - translation file
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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::hu_TimeAccounting;

use utf8;

use strict;
use warnings;

sub Data {
    my $Self = shift;

    # Template: AgentTimeAccountingDelete
    $Self->{Translation}->{'Do you really want to delete the Time Accounting of this day?'} = 'Biztosan törölni akarja a mai nap IdõElszámolását?';

    # Template: AgentTimeAccountingEdit
    $Self->{Translation}->{'Edit Time Record'} = 'Idõ bejegyzés szerkesztése';
    $Self->{Translation}->{'Project settings'} = 'Projekt beállítások';
    $Self->{Translation}->{'Date Navigation'} = 'Dátum navigáció';
    $Self->{Translation}->{'Previous day'} = 'Elõzõ nap';
    $Self->{Translation}->{'Next day'} = 'Következõ nap';
    $Self->{Translation}->{'Days without entries'} = 'Bejegyzés nélküli napok';
    $Self->{Translation}->{'Required fields are marked with a "*".'} = 'A "*"-al jelölt mezõk kitöltése kötelezõ.';
    $Self->{Translation}->{'You have to fill in start and end time or a time period.'} = 'Ki kell töltenie a kezdõ és befejezõ idõpontot vagy az idõtartamot.';
    $Self->{Translation}->{'Project'} = 'Projekt';
    $Self->{Translation}->{'Task'} = 'Feladat';
    $Self->{Translation}->{'Remark'} = 'Megjegyzés';
    $Self->{Translation}->{'Date Navigation'} = 'Dátum navigáció';
    $Self->{Translation}->{'Please add a remark with more than 8 characters!.'} = 'Kérem adjon meg egy 8 karakternél hosszabb megjegyzést!';
    $Self->{Translation}->{'Negative times are not allowed.'} = 'Negatív idõk nem megengedettek.';
    $Self->{Translation}->{'Repeated hours are not allowed. Start time matches another interval.'} = 'Ismétlõdõ órák megadása nem megengedett. A kezdõ idõpont egyezik egy másik intervalummal.';
    $Self->{Translation}->{'End time must be after start time.'} = 'A befejezõ idõpont késõbbi kell legyen, mint a kezdõ idõpont.';
    $Self->{Translation}->{'Repeated hours are not allowed. End time matches another interval.'} = 'Ismétlõdõ órák megadása nem lehetséges. A befejezõ idõpont egyezik más idõszakkal.';
    $Self->{Translation}->{'Period is bigger than the interval between start and end times!'} = 'Az idõtartam nagyobb, mint a kezdõ és bejezõ idõpont közti idõ!';
    $Self->{Translation}->{'Invalid period! A day has only 24 hours.'} = 'Érvénytelen idõtartam! A nap csak 24 órából áll!';
    $Self->{Translation}->{'A valid period must be greater than zero.'} = 'Az érvényes idõtartamnak nagyobbnak kell lennie nullánál.';
    $Self->{Translation}->{'Invalid period! Negative periods are not allowed.'} = 'Érvénytelen idõtartam! Negatív idõtartam nem megengedett.';
    $Self->{Translation}->{'Add one row'} = 'Sor hozzáadása';
    $Self->{Translation}->{'Total'} = 'Összesen';
    $Self->{Translation}->{'On vacation'} = 'Szabadságon';
    $Self->{Translation}->{'You can only select one checkbox element!'} = 'Csak egy jelölõnégyzetet választhat!';
    $Self->{Translation}->{'On sick leave'} = 'Betegszabadságon';
    $Self->{Translation}->{'On overtime leave'} = 'Túlóra letöltése';
    $Self->{Translation}->{'Show all items'} = 'Összes bejegyzés megjelenítése';
    $Self->{Translation}->{'Delete Time Accounting Entry'} = 'IdõElszámolás bejegyzés törlése';
    $Self->{Translation}->{'Confirm insert'} = 'Beszúrás jóváhagyása';
    $Self->{Translation}->{'Are you sure that you worked while you were on sick leave?'} = 'Biztosan dolgozott, miközben betegszabadásgon volt?';
    $Self->{Translation}->{'Are you sure that you worked while you were on vacation?'} = 'Biztosan dolgozott, miközben betegszabadásgon volt?';
    $Self->{Translation}->{'Are you sure that you worked while you were on overtime leave?'} = 'Biztosan dolgozott, miközben túlórát töltött le?';
    $Self->{Translation}->{'Are you sure that you worked more than 16 hours?'} = 'Biztosan többet dolgozott mint 16 óra?';

    # Template: AgentTimeAccountingOverview
    $Self->{Translation}->{'Time reporting monthly overview'} = 'Idõ-kimutatás havi összesítõ';
    $Self->{Translation}->{'Overtime (Hours)'} = 'Túlóra (órák)';
    $Self->{Translation}->{'Overtime (this month)'} = 'Túlóra (ebben a hónapban)';
    $Self->{Translation}->{'Overtime (total)'} = 'Túlóra (összesen)';
    $Self->{Translation}->{'Remaining overtime leave'} = 'Letölthetõ túlóra';
    $Self->{Translation}->{'Vacation (Days)'} = 'Szabadság (nap)';
    $Self->{Translation}->{'Vacation taken (this month)'} = 'Szabadság (ebben a hónapban)';
    $Self->{Translation}->{'Vacation taken (total)'} = 'Szabadság (összesen)';
    $Self->{Translation}->{'Remaining vacation'} = 'Fennmaradó szabadság';
    $Self->{Translation}->{'Sick Leave (Days)'} = 'Betegszabadság (nap)';
    $Self->{Translation}->{'Sick leave taken (this month)'} = 'Betegszabadság (ebben a hónapban)';
    $Self->{Translation}->{'Sick leave taken (total)'} = 'Betegszabadság (összesen)';
    $Self->{Translation}->{'Previous month'} = 'Elõzõ hónap';
    $Self->{Translation}->{'Next month'} = 'Következõ hónap';
    $Self->{Translation}->{'Day'} = 'Nap';
    $Self->{Translation}->{'Weekday'} = 'Hétköznap';
    $Self->{Translation}->{'Working Hours'} = 'Munkaidõ';
    $Self->{Translation}->{'Total worked hours'} = 'Összes munkaidõ';
    $Self->{Translation}->{'User\'s project overview'} = 'Felhasználói projekt összefoglaló';
    $Self->{Translation}->{'Hours (monthly)'} = 'Órák (havi)';
    $Self->{Translation}->{'Hours (Lifetime)'} = 'Órák (élettartam)';
    $Self->{Translation}->{'Grand total'} = 'Mindösszesen';

    # Template: AgentTimeAccountingProjectReporting
    $Self->{Translation}->{'Project report'} = 'Projekt kuimutatás';

    # Template: AgentTimeAccountingReporting
    $Self->{Translation}->{'Time reporting'} = 'Idõ kimutatás';
    $Self->{Translation}->{'Month Navigation'} = 'Hónap navigáció';
    $Self->{Translation}->{'User reports'} = 'Felhasználó kimutatás';
    $Self->{Translation}->{'Monthly total'} = 'Havi összesítõ';
    $Self->{Translation}->{'Lifetime total'} = 'Élettartam összesítõ';
    $Self->{Translation}->{'Overtime leave'} = 'Túlóra letöltés';
    $Self->{Translation}->{'Vacation'} = 'Szabadság';
    $Self->{Translation}->{'Sick leave'} = 'Betegszabadság';
    $Self->{Translation}->{'Vacation remaining'} = 'Hátralévõ távollét';
    $Self->{Translation}->{'Project reports'} = 'Projekt kimutatások';

    # Template: AgentTimeAccountingSetting
    $Self->{Translation}->{'Edit Time Accounting Project Settings'} = 'IdõElszámolás Projekt Beállítások';
    $Self->{Translation}->{'Add project'} = 'Új projekt';
    $Self->{Translation}->{'Add Project'} = 'Új Projekt';
    $Self->{Translation}->{'Edit Project Settings'} = 'Projekt Beállítások Szerkesztése';
    $Self->{Translation}->{'There is already a project with this name. Please, choose a different one.'} = 'Már van ilyen nevû projekt, válasszon más nevet.';
    $Self->{Translation}->{'Edit Time Accounting Settings'} = 'IdõElszámolás Beállítások Szerkesztése';
    $Self->{Translation}->{'Add task'} = 'Új feladat';
    $Self->{Translation}->{'New user'} = 'Új felhasználó';
    $Self->{Translation}->{'Filter for Projects'} = 'Projektek Szûrése';
    $Self->{Translation}->{'Filter for Tasks'} = 'Feladatok Szûrése';
    $Self->{Translation}->{'Filter for Users'} = 'Felhasználók Szûrése';
    $Self->{Translation}->{'Project List'} = 'Projekt Lista';
    $Self->{Translation}->{'Task List'} = 'Feladat Lista';
    $Self->{Translation}->{'Add Task'} = 'Új Feladat';
    $Self->{Translation}->{'Edit Task Settings'} = 'Feladat Beállítások Szerkesztése';
    $Self->{Translation}->{'There is already a task with this name. Please, choose a different one.'} = 'Már van ilyen nevû feladat, válasszon más nevet.';
    $Self->{Translation}->{'User List'} = 'Felhasználó Lista';
    $Self->{Translation}->{'New User Settings'} = 'Új Felhasználó Beállításai';
    $Self->{Translation}->{'Edit User Settings'} = 'Felhasználói Beállítások Szerkesztése';
    $Self->{Translation}->{'Comments'} = 'Megjegyzések';
    $Self->{Translation}->{'Show Overtime'} = 'Mutassa a Túlórát';
    $Self->{Translation}->{'Allow project creation'} = 'Projekt létrehozásának engedélyezése';
    $Self->{Translation}->{'Period Begin'} = 'Idõszak Kezdete';
    $Self->{Translation}->{'Period End'} = 'Idõszak Vége';
    $Self->{Translation}->{'Days of Vacation'} = 'Szabadnapok';
    $Self->{Translation}->{'Hours per Week'} = 'Órák hetente';
    $Self->{Translation}->{'Authorized Overtime'} = 'Jóváhagyott Túlóra';
    $Self->{Translation}->{'Period end must be after period begin.'} = 'Az idõszak végének a kezdete után kell lennie.';
    $Self->{Translation}->{'No time periods found.'} = 'Idõszak nem található.';
    $Self->{Translation}->{'Add time period'} = 'Idõszak hozzáadása';

    # Template: AgentTimeAccountingView
    $Self->{Translation}->{'View Time Record'} = 'Idõ Bejegyzés megtekintése';
    $Self->{Translation}->{'View of '} = 'Megtekintés ';
    $Self->{Translation}->{'Date navigation'} = 'Dátum navigáció';
    $Self->{Translation}->{'No data found for this day.'} = 'Nincs adat ehhez a naphoz.';

    # SysConfig
    $Self->{Translation}->{'Agent interface notification module to see the number of incomplete working days for the user.'} = 'Ügyintézõ interface figyelmeztetõ modul a kitöltetlen munkanapok megjelenítéséhez.';
    $Self->{Translation}->{'Default name for new actions.'} = 'Az új akciók alapértelmezett neve.';
    $Self->{Translation}->{'Default name for new projects.'} = 'Az új projektek alapértelmezett neve.';
    $Self->{Translation}->{'Default setting for date end.'} = 'Alapbeállítás a dátum végéhez.';
    $Self->{Translation}->{'Default setting for date start.'} = 'Alapbeállítás a dátum kezdetéhez.';
    $Self->{Translation}->{'Default setting for leave days.'} = 'Alapbeállítás a távollét napjaihoz.';
    $Self->{Translation}->{'Default setting for overtime.'} = 'Alapbeállítás a túlórához.';
    $Self->{Translation}->{'Default setting for the standard weekly hours.'} = 'Alapbeállítás a normál heti órákhoz.';
    $Self->{Translation}->{'Default status for new actions.'} = 'Az új akciók alapértelmezett státusza.';
    $Self->{Translation}->{'Default status for new projects.'} = 'Az új projektek alapértelmezett státusza.';
    $Self->{Translation}->{'Default status for new users.'} = 'Az új felhasználók alapértelmezett státusza.';
    $Self->{Translation}->{'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.'} = 'Adja meg a projekteket, amelyekhez kötelezõ a megjegyzés. Ha a RegExp kifejezés illeszkedik egy projektre, meg kell adnia megjegyzést. A RegExp az smx paramétert használja.';
    $Self->{Translation}->{'Edit time accounting settings'} = 'IdõElszámolás beállítások szerkesztése';
    $Self->{Translation}->{'Edit time record'} = 'Idõ bejegyzés szerkesztése';
    $Self->{Translation}->{'For how many days ago you can insert working units.'} = 'Hány napra visszamenõleg tudja megadni a munkaidõt.';
    $Self->{Translation}->{'If enabled, the dropdown elements in the edit screen are changed to autocompletion fields.'} = 'Ha engedélyezett, a legördülõ elemek helyett automatikus kiegészítés lesz a mezõknél.';
    $Self->{Translation}->{'Maximum number of working days after which the working units have to be inserted.'} = 'Maximum hány nap után kötelezõ megadni a munkaidõt.';
    $Self->{Translation}->{'Maximum number of working days withouth working units entry after which a warning will be shown.'} = 'Maximum hány munkaidõ megadása nélküli munkanap után jelenjen meg figyelmeztetés.';
    $Self->{Translation}->{'Project time reporting'} = 'Projekt idõ kimutatás';
    $Self->{Translation}->{'Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).'} = 'Reguláris kifejezés a projekthez megfelelõ akciók listájának szükítéséhez. A kulcs reguláris kifejezést tartalmaz a projektekhez, a tartalom reguláris kifejezést tartalmaz az akciókhoz.';
    $Self->{Translation}->{'Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.'} = 'Reguláris kifejezés a felhasználók csoportjának megfelelõ projektek listájának szükítéséhez. A kulcs reguláris kifejezést tartalmaz a projektekhez, a tartalom reguláris kifejezést tartalmaz a csoportok listájához.';
    $Self->{Translation}->{'Specifies if working hours can be inserted without start and end times.'} = 'Adja meg, ha a munkaidõ megadható kezdõ és befejezõ idõpont nélkül.';
    $Self->{Translation}->{'This module forces inserts in TimeAccounting.'} = 'Ez a modul kényszeríti az IdõElszámolás beszúrását..';
    $Self->{Translation}->{'This notification module gives a warning if there are too many incomplete working days.'} = 'Ez a figyelmeztetõ modul jelzi, ha túl sok kitöltetlen munkanap van.';
    $Self->{Translation}->{'Time accounting.'} = 'IdõElszámolás.';
    $Self->{Translation}->{'To use if some actions reduced the working hours (for example, if only half of the traveling time is paid Key => traveling; Content => 50).'} = 'Használja, ha valamelyik akció csökkenti a munkaórákat (például ha csak az utazás fele kerül kifizetésre, Kulcs => Utazás; Tartalom =>50).';
    $Self->{Translation}->{'Determines if the statistics module may generate time accounting information.'} = 'Meghatározza, ha a statisztika modul generálhat IdõElszámolás információkat.';

    $Self->{Translation}->{'Mon'} = 'H';
    $Self->{Translation}->{'Tue'} = 'K';
    $Self->{Translation}->{'Wed'} = 'Sze';
    $Self->{Translation}->{'Thu'} = 'Cs';
    $Self->{Translation}->{'Fri'} = 'P';
    $Self->{Translation}->{'Sat'} = 'Szo';
    $Self->{Translation}->{'Sun'} = 'V';
    $Self->{Translation}->{'January'} = 'Január';
    $Self->{Translation}->{'February'} = 'Február';
    $Self->{Translation}->{'March'} = 'Március';
    $Self->{Translation}->{'April'} = 'Április';
    $Self->{Translation}->{'May'} = 'Május';
    $Self->{Translation}->{'June'} = 'Június';
    $Self->{Translation}->{'July'} = 'Július';
    $Self->{Translation}->{'August'} = 'Augusztus';
    $Self->{Translation}->{'September'} = 'Szeptember';
    $Self->{Translation}->{'October'} = 'Október';
    $Self->{Translation}->{'November'} = 'November';
    $Self->{Translation}->{'December'} = 'December';

    $Self->{Translation}->{'Show all projects'} = 'Az összes projekt megjelenítése';
    $Self->{Translation}->{'Show valid projects'} = 'Az érvényes projektek megjelenítése';
    $Self->{Translation}->{'TimeAccounting'} = 'IdõElszámolás';
    $Self->{Translation}->{'Actions'} = 'Akciók';
    $Self->{Translation}->{'User updated!'} = 'Felhasználó módosítva!';
    $Self->{Translation}->{'User added!'} = 'Felhasználó hozzáadva!';
    $Self->{Translation}->{'Project added!'} = 'Projekt hozzáadva!';
    $Self->{Translation}->{'Project updated!'} = 'Projekt módosítva!';
    $Self->{Translation}->{'Task added!'} = 'Feladat hozzáadva!';
    $Self->{Translation}->{'Task updated!'} = 'Feladat módosítva!';

    #
    # OBSOLETE ENTRIES FOR REFERENCE, DO NOT TRANSLATE!
    #
    $Self->{Translation}->{'Can\'t delete Working Units!'} = 'Nem lehet törölni a munkaidõt!';
    $Self->{Translation}->{'Can\'t save settings, because a day has only 24 hours!'} = 'Nem lehet menteni a beállításokat, a nap csak 24 órából áll!';
    $Self->{Translation}->{'Please insert your working hours!'} = 'Kérem adja meg a munkaidejét!';
    $Self->{Translation}->{'Reporting'} = 'Kimutatás';
    $Self->{Translation}->{'Successful insert!'} = 'Sikeresen hozzáadva!';

}

1;

﻿# --
# Kernel/Language/it_TimeAccounting.pm - translation file
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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::it_TimeAccounting;

use utf8;

use strict;
use warnings;

sub Data {
    my $Self = shift;

    # Template: AgentTimeAccountingDelete
    $Self->{Translation}->{'Do you really want to delete the Time Accounting of this day?'} = 'Vuoi veramente cancellare la rendicontazione temporale di oggi ?';

    # Template: AgentTimeAccountingEdit
    $Self->{Translation}->{'Edit Time Record'} = 'Modifica la rendicontazione';
    $Self->{Translation}->{'Project settings'} = 'Impostazioni del progetto';
    $Self->{Translation}->{'Previous day'} = 'Giorno precedente';
    $Self->{Translation}->{'Next day'} = 'Giorno successivo';
    $Self->{Translation}->{'Days without entries'} = 'Giorni senza informazioni';
    $Self->{Translation}->{'Required fields are marked with a "*".'} = 'I campi obbligatori sono indicati con "*".';
    $Self->{Translation}->{'You have to fill in start and end time or a time period.'} = 'E\' obbligatorio inserire inizio e fine oppure un periodo.';
    $Self->{Translation}->{'Project'} = 'Progetto';
    $Self->{Translation}->{'Task'} = 'Compito';
    $Self->{Translation}->{'Remark'} = 'Commento';
    $Self->{Translation}->{'Date Navigation'} = 'Navigazione per data';
    $Self->{Translation}->{'Please add a remark with more than 8 characters!.'} = 'Inserire un commeto più lungo di 8 caratteri!';
    $Self->{Translation}->{'Negative times are not allowed.'} = 'Non possono essere inseriti valori negativi.';
    $Self->{Translation}->{'Repeated hours are not allowed. Start time matches another interval.'} = 'Non è permesso inserire ore ripetute. L\'orario di inizio coincide con un altro intervallo.';
    $Self->{Translation}->{'End time must be after start time.'} = 'L\'orario di fine è posteriore all\'orario di inizio.';
    $Self->{Translation}->{'Repeated hours are not allowed. End time matches another interval.'} = 'Non è permesso inserire ore ripetute. L\'orario di fine coincide con un altro intervallo.';
    $Self->{Translation}->{'Period is bigger than the interval between start and end times!'} = 'Il periodo è superiore all\'intervallo di inizio\\fine!';
    $Self->{Translation}->{'Invalid period! A day has only 24 hours.'} = 'Periodo non valido. Una giornata ha solo 24 ore.';
    $Self->{Translation}->{'A valid period must be greater than zero.'} = 'Un periodo valido deve essere maggiore di zero.';
    $Self->{Translation}->{'Invalid period! Negative periods are not allowed.'} = 'Periodo non valido. Valori negativi non ammessi.';
    $Self->{Translation}->{'Add one row'} = 'Aggiungere una riga';
    $Self->{Translation}->{'Total'} = 'Totale';
    $Self->{Translation}->{'On vacation'} = 'Ferie';
    $Self->{Translation}->{'You can only select one checkbox element!'} = 'Puoi selezionare solo un elemento!';
    $Self->{Translation}->{'On sick leave'} = 'Assente per malattia';
    $Self->{Translation}->{'On overtime leave'} = 'Assente per recupero';
    $Self->{Translation}->{'Show all items'} = 'Mostra tutti gli elementi';
    $Self->{Translation}->{'Delete Time Accounting Entry'} = 'Cancellare voce di rendicontazione';
    $Self->{Translation}->{'Confirm insert'} = 'Confermare inserimento';
    $Self->{Translation}->{'Are you sure that you worked while you were on sick leave?'} = 'Sei sicuro di aver lavorato mentre eri assente per malattia?';
    $Self->{Translation}->{'Are you sure that you worked while you were on vacation?'} = 'Sei sicuro di aver lavorato mentre eri in ferie?';
    $Self->{Translation}->{'Are you sure that you worked while you were on overtime leave?'} = 'Sei sicuro di aver lavorato mentre eri assente per recupero ?';
    $Self->{Translation}->{'Are you sure that you worked more than 16 hours?'} = 'Sei sicuro di aver lavorato più di 16 ore?';

    # Template: AgentTimeAccountingOverview
    $Self->{Translation}->{'Time reporting monthly overview'} = 'Cruscotto mensile di rendicontazione temporale';
    $Self->{Translation}->{'Overtime (Hours)'} = 'Straordinario (ore)';
    $Self->{Translation}->{'Overtime (this month)'} = 'Straordinario (questo mese)';
    $Self->{Translation}->{'Overtime (total)'} = 'Straordinario (totale)';
    $Self->{Translation}->{'Remaining overtime leave'} = 'Totale ore di recupero disponibili';
    $Self->{Translation}->{'Vacation (Days)'} = 'Ferie (giorni)';
    $Self->{Translation}->{'Vacation taken (this month)'} = 'Ferie utilizzate (questo mese)';
    $Self->{Translation}->{'Vacation taken (total)'} = 'Ferie utilizzate (totale)';
    $Self->{Translation}->{'Remaining vacation'} = 'Ferie residue';
    $Self->{Translation}->{'Sick Leave (Days)'} = 'Assenze per malattia (giorni)';
    $Self->{Translation}->{'Sick leave taken (this month)'} = 'Assenze per malattia utilizzate (questo mese)';
    $Self->{Translation}->{'Sick leave taken (total)'} = 'Assenze per malattia (totale)';
    $Self->{Translation}->{'Previous month'} = 'Mese precedente';
    $Self->{Translation}->{'Next month'} = 'Mese seguente';
    $Self->{Translation}->{'Day'} = 'Giorno';
    $Self->{Translation}->{'Weekday'} = 'Giorno della settimana';
    $Self->{Translation}->{'Working Hours'} = 'Ore lavorative';
    $Self->{Translation}->{'Total worked hours'} = 'Totale ore lavorate';
    $Self->{Translation}->{'User\'s project overview'} = 'Visualizzazione del progetto - Utente';
    $Self->{Translation}->{'Hours (monthly)'} = 'Ore (mensili)';
    $Self->{Translation}->{'Hours (Lifetime)'} = 'Ore (globali)';
    $Self->{Translation}->{'Grand total'} = 'Totale omnicomprensivo';

    # Template: AgentTimeAccountingProjectReporting
    $Self->{Translation}->{'Project report'} = 'Riassunto progetto';

    # Template: AgentTimeAccountingReporting
    $Self->{Translation}->{'Time reporting'} = 'Report del tempo';
    $Self->{Translation}->{'Month Navigation'} = 'Navigazione per mese';
    $Self->{Translation}->{'User reports'} = 'Report utente';
    $Self->{Translation}->{'Monthly total'} = 'Totale mensile';
    $Self->{Translation}->{'Lifetime total'} = 'Totale gLobale';
    $Self->{Translation}->{'Overtime leave'} = 'Assenze per recupero';
    $Self->{Translation}->{'Vacation'} = 'Ferie';
    $Self->{Translation}->{'Sick leave'} = 'Malattia';
    $Self->{Translation}->{'Vacation remaining'} = 'Giorni rimanenti';
    $Self->{Translation}->{'Project reports'} = 'Riassunti progetto';

    # Template: AgentTimeAccountingSetting
    $Self->{Translation}->{'Edit Time Accounting Project Settings'} = 'Modificare le configurazioni di gestione del tempo del progetto.';
    $Self->{Translation}->{'Add project'} = 'Aggiungere progetto';
    $Self->{Translation}->{'Add Project'} = 'Aggiungere progetto';
    $Self->{Translation}->{'Edit Project Settings'} = 'Modificare la configurazione del progetto';
    $Self->{Translation}->{'There is already a project with this name. Please, choose a different one.'} = 'Esiste già un progetto con questo nome. Scegleire un nome diverso.';
    $Self->{Translation}->{'Edit Time Accounting Settings'} = 'Modificare le configurazioni della rendicontazione';
    $Self->{Translation}->{'Add task'} = 'Aggiungere compito';
    $Self->{Translation}->{'New user'} = 'Nuovo utente';
    $Self->{Translation}->{'Filter for Projects'} = 'Filtro per progetti';
    $Self->{Translation}->{'Filter for Tasks'} = 'Filtro per compiti';
    $Self->{Translation}->{'Filter for Users'} = 'Filtro per utenti';
    $Self->{Translation}->{'Project List'} = 'Elenco dei progetti';
    $Self->{Translation}->{'Task List'} = 'Elenco dei compiti';
    $Self->{Translation}->{'Add Task'} = 'Aggiungere compito';
    $Self->{Translation}->{'Edit Task Settings'} = 'Modificare la configurazione dei compiti';
    $Self->{Translation}->{'There is already a task with this name. Please, choose a different one.'} = 'Esiste già un compito con questo nome. Scegliere un nome diverso.';
    $Self->{Translation}->{'User List'} = 'Elenco utenti';
    $Self->{Translation}->{'New User Settings'} = 'Configurazioni nuovo utente';
    $Self->{Translation}->{'Edit User Settings'} = 'Modificare configurazione utente';
    $Self->{Translation}->{'Comments'} = 'Commenti';
    $Self->{Translation}->{'Show Overtime'} = 'Mostrare straordinari';
    $Self->{Translation}->{'Allow project creation'} = 'Cosentire la creazione del progetto';
    $Self->{Translation}->{'Period Begin'} = 'Periodo di inizio';
    $Self->{Translation}->{'Period End'} = 'Periodo di termine';
    $Self->{Translation}->{'Days of Vacation'} = 'Giorni di Assenza';
    $Self->{Translation}->{'Hours per Week'} = 'Ore alla settimana';
    $Self->{Translation}->{'Authorized Overtime'} = 'Straordinari autorizzati';
    $Self->{Translation}->{'Period end must be after period begin.'} = 'Il periodo finale deve essere posteriore al periodo iniziale.';
    $Self->{Translation}->{'No time periods found.'} = 'Non ci sono periodi.';
    $Self->{Translation}->{'Add time period'} = 'Aggiungi periodo';

    # Template: AgentTimeAccountingView
    $Self->{Translation}->{'View Time Record'} = 'Visualizzare registro orario';
    $Self->{Translation}->{'View of '} = 'Visualizzazione di  ';
    $Self->{Translation}->{'Date navigation'} = 'Navigazione per data';
    $Self->{Translation}->{'No data found for this day.'} = 'Nessun dato trovato per questo giorno.';

    # SysConfig
    $Self->{Translation}->{'Agent interface notification module to see the number of incomplete working days for the user.'} = '';
    $Self->{Translation}->{'Default name for new actions.'} = '';
    $Self->{Translation}->{'Default name for new projects.'} = '';
    $Self->{Translation}->{'Default setting for date end.'} = '';
    $Self->{Translation}->{'Default setting for date start.'} = '';
    $Self->{Translation}->{'Default setting for leave days.'} = '';
    $Self->{Translation}->{'Default setting for overtime.'} = '';
    $Self->{Translation}->{'Default setting for the standard weekly hours.'} = '';
    $Self->{Translation}->{'Default status for new actions.'} = '';
    $Self->{Translation}->{'Default status for new projects.'} = '';
    $Self->{Translation}->{'Default status for new users.'} = '';
    $Self->{Translation}->{'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.'} = '';
    $Self->{Translation}->{'Edit time accounting settings'} = '';
    $Self->{Translation}->{'Edit time record'} = '';
    $Self->{Translation}->{'For how many days ago you can insert working units.'} = '';
    $Self->{Translation}->{'If enabled, the dropdown elements in the edit screen are changed to autocompletion fields.'} = '';
    $Self->{Translation}->{'Maximum number of working days after which the working units have to be inserted.'} = '';
    $Self->{Translation}->{'Maximum number of working days withouth working units entry after which a warning will be shown.'} = '';
    $Self->{Translation}->{'Project time reporting'} = '';
    $Self->{Translation}->{'Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).'} = '';
    $Self->{Translation}->{'Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.'} = '';
    $Self->{Translation}->{'Specifies if working hours can be inserted without start and end times.'} = '';
    $Self->{Translation}->{'This module forces inserts in TimeAccounting.'} = '';
    $Self->{Translation}->{'This notification module gives a warning if there are too many incomplete working days.'} = '';
    $Self->{Translation}->{'Time accounting.'} = '';
    $Self->{Translation}->{'To use if some actions reduced the working hours (for example, if only half of the traveling time is paid Key => traveling; Content => 50).'} = '';

    $Self->{Translation}->{'Mon'} = 'Lun';
    $Self->{Translation}->{'Tue'} = 'Mar';
    $Self->{Translation}->{'Wed'} = 'Mer';
    $Self->{Translation}->{'Thu'} = 'Gio';
    $Self->{Translation}->{'Fri'} = 'Ven';
    $Self->{Translation}->{'Sat'} = 'Sáb';
    $Self->{Translation}->{'Sun'} = 'Dom';
    $Self->{Translation}->{'January'} = 'Gennaio';
    $Self->{Translation}->{'February'} = 'Febbraio';
    $Self->{Translation}->{'March'} = 'Marzo';
    $Self->{Translation}->{'April'} = 'Aprile';
    $Self->{Translation}->{'May'} = 'Maggio';
    $Self->{Translation}->{'June'} = 'Giugno';
    $Self->{Translation}->{'July'} = 'Luglio';
    $Self->{Translation}->{'August'} = 'Agosto';
    $Self->{Translation}->{'September'} = 'Settembre';
    $Self->{Translation}->{'October'} = 'Ottobre';
    $Self->{Translation}->{'November'} = 'Novembre';
    $Self->{Translation}->{'December'} = 'Dicembre';

    $Self->{Translation}->{'Show all projects'} = 'Mostrare tutti i progetti';
    $Self->{Translation}->{'Show valid projects'} = 'Mostrare solo i progetti validi';
    $Self->{Translation}->{'TimeAccounting'} = 'Rendicontazione';
    $Self->{Translation}->{'Actions'} = 'Azioni';
    $Self->{Translation}->{'User updated!'} = 'Utente aggiornato!';
    $Self->{Translation}->{'User added!'} = 'Utente aggiunto!';
    $Self->{Translation}->{'Project added!'} = 'Progetto aggiunto!';
    $Self->{Translation}->{'Project updated!'} = 'Progetto aggiornato!';
    $Self->{Translation}->{'Task added!'} = 'Compito aggiunto!';
    $Self->{Translation}->{'Task updated!'} = 'Compito aggiornato!';

    #
    # OBSOLETE ENTRIES FOR REFERENCE, DO NOT TRANSLATE!
    #

    $Self->{Translation}->{'Reporting'} = 'Reporte';
    $Self->{Translation}->{'Successful insert!'} = 'Inserimento terminato positivamente';
    $Self->{Translation}->{'Can\'t save settings, because a day has only 24 hours!'} = '¡Imposible guardar la configuración porque un día sólo tiene 24 horas!';
    $Self->{Translation}->{'Can\'t delete Working Units!'} = '¡Imposible eliminar las unidades de trabajo!';
    $Self->{Translation}->{'Please insert your working hours!'} = '¡Favor de insertar sus horas de trabajo!';
}

1;

# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# Copyright (C) 2011/12/08 Kaoru Hayama TIS Inc.
# --
# 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::ja_TimeAccounting;

use utf8;

use strict;
use warnings;

sub Data {
    my $Self = shift;

    # Template: AgentTimeAccountingDelete
    $Self->{Translation}->{'Do you really want to delete the Time Accounting of this day?'} = '本当に該当日の情報を削除しますか？';

    # Template: AgentTimeAccountingEdit
    $Self->{Translation}->{'Edit Time Record'} = 'タイムレコードの編集';
    $Self->{Translation}->{'Project settings'} = 'プロジェクトの編集';
    $Self->{Translation}->{'Date Navigation'} = '日別ナビゲーション';
    $Self->{Translation}->{'Previous day'} = '前の日';
    $Self->{Translation}->{'Next day'} = '次の日';
    $Self->{Translation}->{'Days without entries'} = 'エントリーのない日';
    $Self->{Translation}->{'Required fields are marked with a "*".'} = '* の項目は入力必須です。';
    $Self->{Translation}->{'You have to fill in start and end time or a time period.'} = '開始時間と終了時間、または期間を指定してください。';
    $Self->{Translation}->{'Project'} = 'プロジェクト';
    $Self->{Translation}->{'Task'} = 'タスク';
    $Self->{Translation}->{'Remark'} = '注釈';
    $Self->{Translation}->{'Please add a remark with more than 8 characters!.'} = '8文字以上の注釈を追加してください。';
    $Self->{Translation}->{'Negative times are not allowed.'} = '－（マイナス）の時間は入力できません';
    $Self->{Translation}->{'Repeated hours are not allowed. Start time matches another interval.'} = '繰り返された時間は許可されません。開始時刻は別の間隔と一致します。';
    $Self->{Translation}->{'End time must be after start time.'} = '終了時間の前に開始時間を設定することはできません。';
    $Self->{Translation}->{'Repeated hours are not allowed. End time matches another interval.'} = 'Diese Endzeit wurde bereits in einem anderen Eintrag angegeben.';
    $Self->{Translation}->{'Period is bigger than the interval between start and end times!'} = '「期間」の設定が不正です。開始時間と終了時間の範囲を超過しています。';
    $Self->{Translation}->{'Invalid period! A day has only 24 hours.'} = '「期間」の設定が不正です。（24時間以上は許可されません）';
    $Self->{Translation}->{'A valid period must be greater than zero.'} = '「期間」の設定が不正です。（0は許可されません）';
    $Self->{Translation}->{'Invalid period! Negative periods are not allowed.'} = '「期間」の設定が不正です。（－（マイナス）は許可されません）';
    $Self->{Translation}->{'Add one row'} = '行の追加';
    $Self->{Translation}->{'Total'} = '計';
    $Self->{Translation}->{'On vacation'} = '休暇取得中';
    $Self->{Translation}->{'You can only select one checkbox element!'} = '1項目のみ選択できます。';
    $Self->{Translation}->{'On sick leave'} = '病欠中';
    $Self->{Translation}->{'On overtime leave'} = '代休中';
    $Self->{Translation}->{'Show all items'} = 'すべてのアイテムを表示';
    $Self->{Translation}->{'Delete Time Accounting Entry'} = 'エントリーを削除';
    $Self->{Translation}->{'Confirm insert'} = '挿入の確認';
    $Self->{Translation}->{'Are you sure that you worked while you were on sick leave?'} = '病欠に設定された期間中に稼働しましたか？';
    $Self->{Translation}->{'Are you sure that you worked while you were on vacation?'} = '休暇に設定された期間中に稼働しましたか？';
    $Self->{Translation}->{'Are you sure that you worked while you were on overtime leave?'} = '代休に設定された期間中に稼働しましたか？';
    $Self->{Translation}->{'Are you sure that you worked more than 16 hours?'} = '16時間以上稼働しましたか？';
    $Self->{Translation}->{'Select all days'} = '全ての日を選択';

    # Template: AgentTimeAccountingOverview
    $Self->{Translation}->{'Time reporting monthly overview'} = '月別一覧';
    $Self->{Translation}->{'Overtime (Hours)'} = '超過勤務（：時間）';
    $Self->{Translation}->{'Overtime (this month)'} = '超過勤務（今月）';
    $Self->{Translation}->{'Overtime (total)'} = '超過勤務（累計）';
    $Self->{Translation}->{'Remaining overtime leave'} = '超過勤務可能残（時間）';
    $Self->{Translation}->{'Vacation (Days)'} = '休暇（日数）';
    $Self->{Translation}->{'Vacation taken (this month)'} = '休暇取得日数（今月）';
    $Self->{Translation}->{'Vacation taken (total)'} = '休暇取得日数（累計）';
    $Self->{Translation}->{'Remaining vacation'} = '休暇取得残日数';
    $Self->{Translation}->{'Sick Leave (Days)'} = '病欠（日数）';
    $Self->{Translation}->{'Sick leave taken (this month)'} = '病欠日数（今月）';
    $Self->{Translation}->{'Sick leave taken (total)'} = '病欠日数（累計）';
    $Self->{Translation}->{'Previous month'} = '前月';
    $Self->{Translation}->{'Next month'} = '次月';
    $Self->{Translation}->{'Day'} = '日';
    $Self->{Translation}->{'Weekday'} = '曜日';
    $Self->{Translation}->{'Working Hours'} = '稼働時間';
    $Self->{Translation}->{'Total worked hours'} = '総稼働時間';
    $Self->{Translation}->{'User\'s project overview'} = 'ユーザのプロジェクト一覧';
    $Self->{Translation}->{'Hours (monthly)'} = '時間（今月）';
    $Self->{Translation}->{'Hours (Lifetime)'} = '時間（通算）';
    $Self->{Translation}->{'Grand total'} = '合計';

    # Template: AgentTimeAccountingProjectReporting
    $Self->{Translation}->{'Project report'} = 'プロジェクト・レポート';

    # Template: AgentTimeAccountingReporting
    $Self->{Translation}->{'Time reporting'} = '時間会計レポート';
    $Self->{Translation}->{'Month Navigation'} = '月別ナビゲーション';
    $Self->{Translation}->{'User reports'} = 'ユーザ・レポート';
    $Self->{Translation}->{'Monthly total'} = '月合計';
    $Self->{Translation}->{'Lifetime total'} = '通算合計';
    $Self->{Translation}->{'Overtime leave'} = '代休';
    $Self->{Translation}->{'Vacation'} = '休暇';
    $Self->{Translation}->{'Sick leave'} = '病欠';
    $Self->{Translation}->{'Vacation remaining'} = '休日残日数';
    $Self->{Translation}->{'Project reports'} = 'プロジェクト・レポート';

    # Template: AgentTimeAccountingSetting
    $Self->{Translation}->{'Edit Time Accounting Project Settings'} = '時間会計 プロジェクト設定の編集';
    $Self->{Translation}->{'Add Project'} = '新規プロジェクトの追加';
    $Self->{Translation}->{'Add project'} = '新規プロジェクトの追加';
    $Self->{Translation}->{'Edit Project Settings'} = 'プロジェクトの編集';
    $Self->{Translation}->{'There is already a project with this name. Please, choose a different one.'} = '同名のプロジェクトが存在します。名称を変更してください。';
    $Self->{Translation}->{'Edit Time Accounting Settings'} = '設定の編集';
    $Self->{Translation}->{'Add Task'} = '新規タスクの追加';
    $Self->{Translation}->{'Add task'} = '新規タスクの追加';
    $Self->{Translation}->{'New user'} = '新規ユーザ';
    $Self->{Translation}->{'Filter for Projects'} = 'プロジェクトの絞り込み';
    $Self->{Translation}->{'Filter for Tasks'} = 'タスクの絞り込み';
    $Self->{Translation}->{'Filter for Users'} = 'ユーザの絞り込み';
    $Self->{Translation}->{'Project List'} = 'プロジェクト一覧';
    $Self->{Translation}->{'Task List'} = 'タスク一覧';
    $Self->{Translation}->{'Edit Task Settings'} = 'タスク設定';
    $Self->{Translation}->{'There is already a task with this name. Please, choose a different one.'} = '同名のタスクが存在します。名称を変更してください。';
    $Self->{Translation}->{'User List'} = 'ユーザ一覧';
    $Self->{Translation}->{'New User Settings'} = '新規ユーザ設定';
    $Self->{Translation}->{'Edit User Settings'} = 'ユーザ設定の編集';
    $Self->{Translation}->{'Comments'} = 'コメント';
    $Self->{Translation}->{'Show Overtime'} = '超過勤務を表示';
    $Self->{Translation}->{'Allow project creation'} = 'プロジェクトの新規追加を許可する';
    $Self->{Translation}->{'Period Begin'} = '開始点';
    $Self->{Translation}->{'Period End'} = '終了点';
    $Self->{Translation}->{'Days of Vacation'} = '休暇';
    $Self->{Translation}->{'Hours per Week'} = '時間 / 週';
    $Self->{Translation}->{'Authorized Overtime'} = '承認済の超過勤務';
    $Self->{Translation}->{'Period end must be after period begin.'} = '終了点の前に開始点を設定することはできません。';
    $Self->{Translation}->{'No time periods found.'} = '期間が未設定です。';
    $Self->{Translation}->{'Add time period'} = '期間を追加してください。';

    # Template: AgentTimeAccountingView
    $Self->{Translation}->{'View Time Record'} = 'タイム・レコードを表示';
    $Self->{Translation}->{'View of '} = '一覧';
    $Self->{Translation}->{'No data found for this day.'} = '該当するデータがありません。';

    # SysConfig
    $Self->{Translation}->{'Agent interface notification module to see the number of incomplete working days for the user.'} = 'ユーザの不完全な稼働日数をカウントし通知するエージェント・インターフェイスです。';
    $Self->{Translation}->{'Default name for new actions.'} = '新規操作に対するデフォルトの名称';
    $Self->{Translation}->{'Default name for new projects.'} = '新規プロジェクトに対するデフォルトの名称';
    $Self->{Translation}->{'Default setting for date end.'} = '「終了日」のデフォルト値';
    $Self->{Translation}->{'Default setting for date start.'} = '「開始日」のデフォルト値';

    $Self->{Translation}->{'Default setting for description.'} = '「説明文」のデフォルト値';

    $Self->{Translation}->{'Default setting for leave days.'} = '「休暇」のデフォルト値';
    $Self->{Translation}->{'Default setting for overtime.'} = '「超過勤務」のデフォルト値';
    $Self->{Translation}->{'Default setting for the standard weekly hours.'} = '週あたり基本稼働時間のデフォルト値';
    $Self->{Translation}->{'Default status for new actions.'} = '新規操作に対するデフォルトのステータス';
    $Self->{Translation}->{'Default status for new projects.'} = '新規プロジェクトに対するデフォルトのステータス';
    $Self->{Translation}->{'Default status for new users.'} = '新規ユーザに対するデフォルトのステータス';
    $Self->{Translation}->{'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.'} = '「注釈」エントリーを必要とする項目を定義します。 もしプロジェクト名がここで設定した正規表現にマッチする場合、「注釈｝は必須項目となります。※正規表現にはSMXパラメータを使います。';
    $Self->{Translation}->{'Edit time accounting settings'} = '時間会計の設定を編集';
    $Self->{Translation}->{'Edit time record'} = 'タイムレコードを編集';
    $Self->{Translation}->{'For how many days ago you can insert working units.'} = '何日前からワーキング・ユニットの新規登録が可能であるかの設定です。';
    $Self->{Translation}->{'If enabled, the dropdown elements in the edit screen are changed to autocompletion fields.'} = 'この項目を「許可」にすると、編集画面内のプルダウンメニューをオートコンプリート機能付きのフォームに変更します。';
    $Self->{Translation}->{'If enabled, the user is allowed to enter "on vacation leave'} = 'この項目を「許可」にすると、ユーザが休暇申請できるようになります。';
    $Self->{Translation}->{'Maximum number of working days after which the working units have to be inserted.'} = '1つ以上のワーキング・ユニットを設定すべき最大の稼働日数。';
    $Self->{Translation}->{'Maximum number of working days withouth working units entry after which a warning will be shown.'} = '1つ以上のワーキング・ユニットが設定されていない場合、アラートを発するべき最大の稼働日数。';
    $Self->{Translation}->{'Project time reporting'} = 'プロジェクト別時間会計レポート';
    $Self->{Translation}->{'Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).'} = 'プロジェクトによって行動リストを絞り込むための正規表現。「鍵」ではプロジェクトに対する正規表現を、「内容」では`行動`に対する正規表現を指定する。';
    $Self->{Translation}->{'Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.'} = 'ユーザグループによってプロジェクトリストを絞り込むための正規表現。「鍵」ではプロジェクトに対する正規表現を、「内容」ではカンマ区切りのユーザリストを指定する。';
    $Self->{Translation}->{'Specifies if working hours can be inserted without start and end times.'} = '業務時間を「開始時間」と「終了時間」の指定ナシで新規登録できるかどうか指定する。';
    $Self->{Translation}->{'This module forces inserts in TimeAccounting.'} = 'TimeAccountingモジュールと同等の項目が設定必須となります。';
    $Self->{Translation}->{'This notification module gives a warning if there are too many incomplete working days.'} = 'あまりにも多くの不完全な稼働日がある場合、この通知モジュールは警告を与えます。';
    $Self->{Translation}->{'Time accounting.'} = '時間会計';
    $Self->{Translation}->{'To use if some actions reduced the working hours (for example, if only half of the traveling time is paid Key => traveling; Content => 50).'} = '業務上の行動において、稼働時間として計上する時間を調整する必要がある場合に、この設定を使用します。（例：「移動時間」の50%のみ勤務時間相当とする場合「鍵」に`journey`、「内容」に`50`と設定)';

    $Self->{Translation}->{'Mon'} = '月';
    $Self->{Translation}->{'Tue'} = '火';
    $Self->{Translation}->{'Wed'} = '水';
    $Self->{Translation}->{'Thu'} = '木';
    $Self->{Translation}->{'Fri'} = '金';
    $Self->{Translation}->{'Sat'} = '土';
    $Self->{Translation}->{'Sun'} = '日';
    $Self->{Translation}->{'January'} = '1月';
    $Self->{Translation}->{'February'} = '2月';
    $Self->{Translation}->{'March'} = '3月';
    $Self->{Translation}->{'April'} = '4月';
    $Self->{Translation}->{'May'} = '5月';
    $Self->{Translation}->{'June'} = '6月';
    $Self->{Translation}->{'July'} = '7月';
    $Self->{Translation}->{'August'} = '8月';
    $Self->{Translation}->{'September'} = '9月';
    $Self->{Translation}->{'October'} = '10月';
    $Self->{Translation}->{'November'} = '11月';
    $Self->{Translation}->{'December'} = '12月';

    $Self->{Translation}->{'Show all projects'} = 'すべてのプロジェクトを表示';
    $Self->{Translation}->{'Show valid projects'} = '有効なプロジェクトを表示';
    $Self->{Translation}->{'TimeAccounting'} = '時間会計';
    $Self->{Translation}->{'Actions'} = '操作';
    $Self->{Translation}->{'User updated!'} = 'ユーザ情報が更新されました';
    $Self->{Translation}->{'User added!'} = 'ユーザ情報が追加されました';
    $Self->{Translation}->{'Project added!'} = 'プロジェクトが追加されました';
    $Self->{Translation}->{'Project updated!'} = 'プロジェクトが更新されました';
    $Self->{Translation}->{'Task added!'} = 'タスクが追加されました';
    $Self->{Translation}->{'Task updated!'} = 'タスクが更新されました';

    #
    # OBSOLETE ENTRIES FOR REFERENCE, DO NOT TRANSLATE!
    #
    $Self->{Translation}->{'Can\'t delete Working Units!'} = 'ワーキング・ユニットを削除できません';
    $Self->{Translation}->{'Can\'t save settings, because a day has only 24 hours!'} = '時間指定が不正です。設定を保存できませんでした。';
    $Self->{Translation}->{'Please insert your working hours!'} = '稼働時間を入力してください。';
    $Self->{Translation}->{'Reporting'} = '結果表示';
    $Self->{Translation}->{'Successful insert!'} = '挿入が成功しました。';

}

1;

﻿# --
# Kernel/Language/pl_TimeAccounting.pm - translation file
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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::pl_TimeAccounting;

use utf8;

use strict;
use warnings;

sub Data {
    my $Self = shift;

    # Template: AgentTimeAccountingDelete
    $Self->{Translation}->{'Do you really want to delete the Time Accounting of this day?'} = 'Czy na pewno chcesz usunąć zaraportowany czas pracy dla tego dnia?';

    # Template: AgentTimeAccountingEdit
    $Self->{Translation}->{'Edit Time Record'} = 'Edytuj wpis czasu';
    $Self->{Translation}->{'Project settings'} = 'Ustawienia projektu';
    $Self->{Translation}->{'Date Navigation'} = 'Nawigacja po dacie';
    $Self->{Translation}->{'Previous day'} = 'Poprzedni dzień';
    $Self->{Translation}->{'Next day'} = 'Następny dzień';
    $Self->{Translation}->{'Days without entries'} = 'Dni bez raportów';
    $Self->{Translation}->{'Required fields are marked with a "*".'} = 'Wymagane pola są oznaczone "*".';
    $Self->{Translation}->{'You have to fill in start and end time or a time period.'} = 'Musisz wprowadzić dzień początku oraz końca lub czas trwania w okresie.';
    $Self->{Translation}->{'Project'} = 'Projekt';
    $Self->{Translation}->{'Task'} = 'Zadanie';
    $Self->{Translation}->{'Remark'} = 'Uwaga';
    $Self->{Translation}->{'Date Navigation'} = 'Nawigacja po dacie';
    $Self->{Translation}->{'Please add a remark with more than 8 characters!.'} = 'Proszę dodać uwagę składającą się z więcej niż 8 znaków!';
    $Self->{Translation}->{'Negative times are not allowed.'} = 'Ujemne czasy są niedozwolone.';
    $Self->{Translation}->{'Repeated hours are not allowed. Start time matches another interval.'} = 'Powtarzające się czasy nie są dozwolone. Czas rozpoczęcia pasuje do innego przedziału.';
    $Self->{Translation}->{'End time must be after start time.'} = 'Czas zakończenia musi być po czasie rozpoczęcia.';
    $Self->{Translation}->{'Repeated hours are not allowed. End time matches another interval.'} = 'Powtarzające się czasy nie są dozwolone. Czas zakończenia pasuje do innego przedziału.';
    $Self->{Translation}->{'Period is bigger than the interval between start and end times!'} = 'Okres jest większy niż odstęp między początkiem i końcem czasów!';
    $Self->{Translation}->{'Invalid period! A day has only 24 hours.'} = 'Nieprawidłowy okres! Dzień ma tylko 24 godziny.';
    $Self->{Translation}->{'A valid period must be greater than zero.'} = 'Prawidłowy okres musi być większy niż zero.';
    $Self->{Translation}->{'Invalid period! Negative periods are not allowed.'} = 'Nieprawidłowy okres! Ujemny okres nie jest dozwolony.';
    $Self->{Translation}->{'Add one row'} = 'Dodaj jeden wiersz';
    $Self->{Translation}->{'Total'} = 'Suma';
    $Self->{Translation}->{'On vacation'} = 'Urlop';
    $Self->{Translation}->{'You can only select one checkbox element!'} = 'Możesz zaznaczyć tylko jeden element!';
    $Self->{Translation}->{'On sick leave'} = 'Zwolnienie lekarskie';
    $Self->{Translation}->{'On overtime leave'} = 'Wolne za nadgodziny';
    $Self->{Translation}->{'Show all items'} = 'Pokaż wszystkie elementy';
    $Self->{Translation}->{'Delete Time Accounting Entry'} = 'Usuń pozycję zaraportowanego czasu';
    $Self->{Translation}->{'Confirm insert'} = 'Zatwierdź wstawienie';
    $Self->{Translation}->{'Are you sure that you worked while you were on sick leave?'} = 'Jesteś pewien że pracowałeś na zwolnieniu lekarskim?';
    $Self->{Translation}->{'Are you sure that you worked while you were on vacation?'} = 'Jesteś pewien że pracowałeś podczas urlopu?';
    $Self->{Translation}->{'Are you sure that you worked while you were on overtime leave?'} = 'Jesteś pewien że pracowałeś na wolnym z nadgodzin?';
    $Self->{Translation}->{'Are you sure that you worked more than 16 hours?'} = 'Jesteś pewien że pracowałeś więcej niż 16 godzin?';
    $Self->{Translation}->{'Select all days'} = 'Zaznacz wszystkie dni';
    $Self->{Translation}->{'Mass entry'} = 'Masowy wpis';
    $Self->{Translation}->{'Please choose the reason for your absence for the selected days'} = 'Proszę wybrać powód braku obecności dla wybranych dni';
    $Self->{Translation}->{'Please choose at least one day!'} = 'Proszę wybrać conajmniej jeden dzień!';
    $Self->{Translation}->{'Please choose a reason for absence!'} = 'Proszę wybrać powód nieobecności!';
    $Self->{Translation}->{'Mass Entry'} = 'Masowy wpis';
    $Self->{Translation}->{'Invalid format! Please enter a time with the format HH:MM.'} = 'Nieprawidłowy format! Proszę podać czas w godzinach GG:MM.';
    $Self->{Translation}->{'24:00 is only permitted as end time.'} = '24:00 jest dozwolone jedynie jako koniec czasu.';
    $Self->{Translation}->{'Invalid time! A day has only 24 hours.'} = 'Nieprawidłowy czas! Dzień ma tylko 24 godziny.';

    # Template: AgentTimeAccountingOverview
    $Self->{Translation}->{'Time reporting monthly overview'} = 'Przegląd miesięcznego raportu czasu';
    $Self->{Translation}->{'Overtime (Hours)'} = 'Nadgodziny (Godziny)';
    $Self->{Translation}->{'Overtime (this month)'} = 'Nadgodziny (ten misiąc)';
    $Self->{Translation}->{'Overtime (total)'} = 'Nadgodziny (łącznie)';
    $Self->{Translation}->{'Remaining overtime leave'} = 'Pozostałe nadgodziny do odebrania';
    $Self->{Translation}->{'Vacation (Days)'} = 'Urlop (Dni)';
    $Self->{Translation}->{'Vacation taken (this month)'} = 'Wykorzystane dni urlopu (ten miesiąc)';
    $Self->{Translation}->{'Vacation taken (total)'} = 'Wykorzystane dni urlopu (łącznie)';
    $Self->{Translation}->{'Remaining vacation'} = 'Pozostało urlopu';
    $Self->{Translation}->{'Sick Leave (Days)'} = 'Urlop chorobowy (Dni)';
    $Self->{Translation}->{'Sick leave taken (this month)'} = 'Wykorzystane dni urlopu chorobowego (ten miesiąc)';
    $Self->{Translation}->{'Sick leave taken (total)'} = 'Wykorzystane dni urlopu chorobowego (łącznie)';
    $Self->{Translation}->{'Previous month'} = 'Poprzedni miesiąc';
    $Self->{Translation}->{'Next month'} = 'Następny miesiąc';
    $Self->{Translation}->{'Day'} = 'Dzień';
    $Self->{Translation}->{'Weekday'} = 'Dzień tygodnia';
    $Self->{Translation}->{'Working Hours'} = 'Godziny pracy';
    $Self->{Translation}->{'Total worked hours'} = 'Łączne godziny wypracowane';
    $Self->{Translation}->{'User\'s project overview'} = 'Przegląd projektu użytkownika/ów';
    $Self->{Translation}->{'Hours (monthly)'} = 'Godziny (miesięcznie)';
    $Self->{Translation}->{'Hours (Lifetime)'} = 'Godziny (łącznie)';
    $Self->{Translation}->{'Grand total'} = 'Ogółem';

    # Template: AgentTimeAccountingProjectReporting
    $Self->{Translation}->{'Project report'} = 'Raport projektu';

    # Template: AgentTimeAccountingReporting
    $Self->{Translation}->{'Time reporting'} = 'Raport czasu';
    $Self->{Translation}->{'Month Navigation'} = 'Nawigacja miesięczna';
    $Self->{Translation}->{'User reports'} = 'Raporty użytkownika';
    $Self->{Translation}->{'Monthly total'} = 'Łącznie - miesięcznie';
    $Self->{Translation}->{'Lifetime total'} = 'Łącznie - ogółem';
    $Self->{Translation}->{'Overtime leave'} = 'Wolne za nadgodziny';
    $Self->{Translation}->{'Vacation'} = 'Urlop';
    $Self->{Translation}->{'Sick leave'} = 'Urlop chorobowy';
    $Self->{Translation}->{'Vacation remaining'} = 'Pozostałe dni poza pracą';
    $Self->{Translation}->{'Project reports'} = 'Raporty projektu';

    # Template: AgentTimeAccountingSetting
    $Self->{Translation}->{'Edit Time Accounting Project Settings'} = 'Edytuj ustawienia Raportowania Czasu';
    $Self->{Translation}->{'Add project'} = 'Dodaj projekt';
    $Self->{Translation}->{'Add Project'} = 'Dodaj projekt';
    $Self->{Translation}->{'Edit Project Settings'} = 'Edytuj ustawienia projektu';
    $Self->{Translation}->{'There is already a project with this name. Please, choose a different one.'} = 'W systemie znajduje się już projekt o takiej nazwie. Wybierz inną nazwę.';
    $Self->{Translation}->{'Edit Time Accounting Settings'} = 'Edytuj ustawienia Raportowania Czasu';
    $Self->{Translation}->{'Add task'} = 'Dodaj zadanie';
    $Self->{Translation}->{'New user'} = 'Nowy użytkownik';
    $Self->{Translation}->{'Filter for Projects'} = 'Filtr na projekty';
    $Self->{Translation}->{'Filter for Tasks'} = 'Filtr na zadania';
    $Self->{Translation}->{'Filter for Users'} = 'Filtr na użytkowników';
    $Self->{Translation}->{'Project List'} = 'Lista projektów';
    $Self->{Translation}->{'Task List'} = 'Lista zadań';
    $Self->{Translation}->{'Add Task'} = 'Dodaj zadanie';
    $Self->{Translation}->{'Edit Task Settings'} = 'Edytuj ustawienia zadania';
    $Self->{Translation}->{'There is already a task with this name. Please, choose a different one.'} = 'W systemie znajduje się już zadanie o takiej nazwie. Wybierz inną nazwę.';
    $Self->{Translation}->{'User List'} = 'Lista użytkowników';
    $Self->{Translation}->{'New User Settings'} = 'Ustawienia nowego użytkownika';
    $Self->{Translation}->{'Edit User Settings'} = 'Edytuj ustawienia użytkownika';
    $Self->{Translation}->{'Comments'} = 'Komentarz';
    $Self->{Translation}->{'Show Overtime'} = 'Pokaż nadgodziny';
    $Self->{Translation}->{'Allow project creation'} = 'Zezwól na tworzenie projektu';
    $Self->{Translation}->{'Period Begin'} = 'Okres -  początek';
    $Self->{Translation}->{'Period End'} = 'Okres - koniec';
    $Self->{Translation}->{'Days of Vacation'} = 'Dni urlopu';
    $Self->{Translation}->{'Hours per Week'} = 'Godzin w tygodniu';
    $Self->{Translation}->{'Authorized Overtime'} = 'Zezwól na nadgodziny';
    $Self->{Translation}->{'Period end must be after period begin.'} = 'Okres końcowy musi być po okresie rozpoczęcia.';
    $Self->{Translation}->{'No time periods found.'} = 'Brak okresów czasu.';
    $Self->{Translation}->{'Add time period'} = 'Dodaj okres czasu';

    # Template: AgentTimeAccountingView
    $Self->{Translation}->{'View Time Record'} = 'Pokaż wpisz czasu';
    $Self->{Translation}->{'View of '} = 'Widok ';
    $Self->{Translation}->{'No data found for this day.'} = 'Brak danych odnośnie tego dnia.';

    # SysConfig
    $Self->{Translation}->{'Agent interface notification module to see the number of incomplete working days for the user.'} = 'Interfejs agenta informujący o dniach bez zaraportowanego czasu pracy.';
    $Self->{Translation}->{'Default data to use on attribute for ticket search screen. Example: "TicketCreateTimeStartYear=2010;TicketCreateTimeStartMonth=10;TicketCreateTimeStartDay=4;TicketCreateTimeStopYear=2010;TicketCreateTimeStopMonth=11;TicketCreateTimeStopDay=3;".'} = '';
    $Self->{Translation}->{'Default name for new actions.'} = 'Domyślna nazwa dla nowej akcji.';
    $Self->{Translation}->{'Default name for new projects.'} = 'Domyślna nazwa dla nowego projektu.';
    $Self->{Translation}->{'Default setting for date end.'} = 'Domyślna data zakończenia.';
    $Self->{Translation}->{'Default setting for date start.'} = 'Domyślna data rozpoczęcia.';
    $Self->{Translation}->{'Default setting for description.'} = 'Domyślne ustawienie dla opisu.';
    $Self->{Translation}->{'Default setting for leave days.'} = 'Domyślne ustawienie dla urlopu.';
    $Self->{Translation}->{'Default setting for overtime.'} = 'Domyślne ustawienie dla nadgodzin.';
    $Self->{Translation}->{'Default setting for the standard weekly hours.'} = 'Domyślnie ustawienie dla standardowych godzin tygodniowych.';
    $Self->{Translation}->{'Default status for new actions.'} = 'Domyślny status dla nowej akcji.';
    $Self->{Translation}->{'Default status for new projects.'} = 'Domyślny status dla nowego projektu.';
    $Self->{Translation}->{'Default status for new users.'} = 'Domyślny status dla nowego użytkownika.';
    $Self->{Translation}->{'Defines if time accounting must be set to all tickets in bulk action.'} = '';
    $Self->{Translation}->{'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.'} = 'Dentro desta opção de configuração, uma regexp pode ser definida que determina quais projetos são registrados com uma observação deve ser (o regexp funciona com SMX-parâmetros).';
    $Self->{Translation}->{'Edit time accounting settings'} = 'Edytuj ustawienia czasu raportowania';
    $Self->{Translation}->{'Edit time record'} = 'Edytuj wpis czasu';
    $Self->{Translation}->{'For how many days ago you can insert working units.'} = 'Dla ilu dni wstecz można wstawić jednostki czasu pracy';
    $Self->{Translation}->{'If enabled, the dropdown elements in the edit screen are changed to autocompletion fields.'} = 'Jeśli aktywowane, lista wybierana jest zastąpioną autowypełnianiem.';
    $Self->{Translation}->{'If enabled, the user is allowed to enter "on vacation leave", "on sick leave" and "on overtime leave" to multiple dates at once.'} = 'Jeśli aktywowane, użytkownik może wstawić "urlop", "urlop chorobowy" oraz "urlop za nadgodziny" do kilku dat jednocześnie.';
    $Self->{Translation}->{'Maximum number of working days after which the working units have to be inserted.'} = 'Maksymalna liczba dni roboczych, po których jednostki czasu powinny zostać wprowadzone.';
    $Self->{Translation}->{'Maximum number of working days withouth working units entry after which a warning will be shown.'} = 'Maksymalna liczba dni roboczych bez wprowadzania raportu czasu, po których zostanie wyświetlone ostrzeżenie.';
    $Self->{Translation}->{'Project time reporting'} = 'O relatório de Tempo do projeto ';
    $Self->{Translation}->{'Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).'} = '';
    $Self->{Translation}->{'Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.'} = '';
    $Self->{Translation}->{'Saves the login and password on the session table in the database, if "DB" was selected for SessionModule.'} = 'Zapisuje login oraz hasło, do tabeli sesji w bazie danych, jeśli baza danych została wybrana w SessionModule.';
    $Self->{Translation}->{'Show article as rich text even if rich text writing is disabled.'} = '';
    $Self->{Translation}->{'Specifies if working hours can be inserted without start and end times.'} = 'Określa, czy czas pracy może zostać wprowadzony bez podawania czasu rozpoczęcia oraz zakończenia.';
    $Self->{Translation}->{'This module forces inserts in TimeAccounting.'} = 'Ten moduł wymusza wprowadzenia w Raportach Czasu.';
    $Self->{Translation}->{'This notification module gives a warning if there are too many incomplete working days.'} = 'Ten moduł daje ostrzeżenie, jeśli jest zbyt niepełnych dni roboczych.';
    $Self->{Translation}->{'Time accounting.'} = '';
    $Self->{Translation}->{'To use if some actions reduced the working hours (for example, if only half of the traveling time is paid Key => traveling; Content => 50).'} = 'Używane jeśli niektóre działania zmniejszyły godziny pracy (np. jeśli tylko połowa czas podróży jest wypłacana klucz => podróże; Zawartość => 50).';

    $Self->{Translation}->{'Mon'} = 'Pon';
    $Self->{Translation}->{'Tue'} = 'Wt';
    $Self->{Translation}->{'Wed'} = 'Śr';
    $Self->{Translation}->{'Thu'} = 'Czw';
    $Self->{Translation}->{'Fri'} = 'Pią';
    $Self->{Translation}->{'Sat'} = 'Sob';
    $Self->{Translation}->{'Sun'} = 'Niedz';
    $Self->{Translation}->{'January'} = 'Styczeń';
    $Self->{Translation}->{'February'} = 'Luty';
    $Self->{Translation}->{'March'} = 'Marzec';
    $Self->{Translation}->{'April'} = 'Kwiecień';
    $Self->{Translation}->{'May'} = 'Maj';
    $Self->{Translation}->{'June'} = 'Czerwiec';
    $Self->{Translation}->{'July'} = 'Lipiec';
    $Self->{Translation}->{'August'} = 'Sierpień';
    $Self->{Translation}->{'September'} = 'Wrzesień';
    $Self->{Translation}->{'October'} = 'Październik';
    $Self->{Translation}->{'November'} = 'Listopad';
    $Self->{Translation}->{'December'} = 'Grudzień';

    $Self->{Translation}->{'Show all projects'} = 'Pokaż wszystkie projekty';
    $Self->{Translation}->{'Show valid projects'} = 'Pokaż aktualne projekty';
    $Self->{Translation}->{'TimeAccounting'} = 'Raportowanie Czasu';
    $Self->{Translation}->{'Actions'} = 'Akcje';
    $Self->{Translation}->{'User updated!'} = 'Użytkownik zaaktualizowany!';
    $Self->{Translation}->{'User added!'} = 'Użytkownik dodany!';
    $Self->{Translation}->{'Project added!'} = 'Projekt dodany!';
    $Self->{Translation}->{'Project updated!'} = 'Projekt zaaktualizowany!';
    $Self->{Translation}->{'Task added!'} = 'Zadanie dodane!';
    $Self->{Translation}->{'Task updated!'} = 'Zadanie zaaktualizowane!';

    #
    # OBSOLETE ENTRIES FOR REFERENCE, DO NOT TRANSLATE!
    #
    $Self->{Translation}->{'Can\'t delete Working Units!'} = 'Nie można usunąć Jednostek Czasu!';
    $Self->{Translation}->{'Can\'t save settings, because a day has only 24 hours!'} = 'Nie można zapisać ustawień ponieważ dzień ma tylko 24 godziny!';
    $Self->{Translation}->{'Please insert your working hours!'} = 'Proszę wprowadzić godziny pracy!';
    $Self->{Translation}->{'Reporting'} = 'Raportowanie';
    $Self->{Translation}->{'Successful insert!'} = 'Pomyślnie wstawiono!';

}

1;

﻿# --
# Kernel/Language/pt_BR_TimeAccounting.pm - translation file
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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::pt_BR_TimeAccounting;

use utf8;

use strict;
use warnings;

sub Data {
    my $Self = shift;

    # Template: AgentTimeAccountingDelete
    $Self->{Translation}->{'Do you really want to delete the Time Accounting of this day?'} = 'Você realmente deseja excluir a contabilização desse dia?';

    # Template: AgentTimeAccountingEdit
    $Self->{Translation}->{'Edit Time Record'} = 'Editar registro de tempo';
    $Self->{Translation}->{'Project settings'} = 'Definições de Projeto';
    $Self->{Translation}->{'Date Navigation'} = 'Escolha de Data';
    $Self->{Translation}->{'Previous day'} = 'Dia anterior';
    $Self->{Translation}->{'Next day'} = 'Próximo Dia';
    $Self->{Translation}->{'Days without entries'} = 'Dias sem entradas';
    $Self->{Translation}->{'Required fields are marked with a "*".'} = 'Os campos que são obrigatórios são marcados com um asterisco "*"';
    $Self->{Translation}->{'You have to fill in start and end time or a time period.'} = 'Você tem que preencher o tempo de início e final ou um período de tempo.';
    $Self->{Translation}->{'Project'} = 'Projeto';
    $Self->{Translation}->{'Task'} = 'Tarefa';
    $Self->{Translation}->{'Remark'} = 'Marcar';
    $Self->{Translation}->{'Date Navigation'} = 'Escolha de Data';
    $Self->{Translation}->{'Please add a remark with more than 8 characters!.'} = 'Por favor insira um comentário com pelo menos 8 caracteres';
    $Self->{Translation}->{'Negative times are not allowed.'} = 'Valores negativos não são permitidos.';
    $Self->{Translation}->{'Repeated hours are not allowed. Start time matches another interval.'} = 'Este horário de início é indicado em outra entrada.';
    $Self->{Translation}->{'End time must be after start time.'} = 'Hora de término deve ser posterior à hora de início.';
    $Self->{Translation}->{'Repeated hours are not allowed. End time matches another interval.'} = 'Horas repetidas não são permitidas. Hora de término coincide com outro intervalo.';
    $Self->{Translation}->{'Period is bigger than the interval between start and end times!'} = 'Período é maior do que o intervalo entre os horários de início e fim!';
    $Self->{Translation}->{'Invalid period! A day has only 24 hours.'} = 'Período inválido! Um dia só tem 24 horas.';
    $Self->{Translation}->{'A valid period must be greater than zero.'} = 'Um período válido deve ser maior que zero.';
    $Self->{Translation}->{'Invalid period! Negative periods are not allowed.'} = 'Período inválido! Períodos negativos não são permitidos.';
    $Self->{Translation}->{'Add one row'} = 'Adicionar uma linha';
    $Self->{Translation}->{'Total'} = 'Total';
    $Self->{Translation}->{'On vacation'} = 'Em férias';
    $Self->{Translation}->{'You can only select one checkbox element!'} = 'Você pode selecionar apenas um elemento da caixa de seleção!';
    $Self->{Translation}->{'On sick leave'} = 'Em licença médica';
    $Self->{Translation}->{'On overtime leave'} = 'Em sobrehora';
    $Self->{Translation}->{'Show all items'} = 'Mostrar todos os itens';
    $Self->{Translation}->{'Delete Time Accounting Entry'} = 'Excluir Entrada de Contabilização de Tempo';
    $Self->{Translation}->{'Confirm insert'} = 'Confirme inserir';
    $Self->{Translation}->{'Are you sure that you worked while you were on sick leave?'} = 'Tem certeza que você trabalhou enquanto esteve em licença médica?';
    $Self->{Translation}->{'Are you sure that you worked while you were on vacation?'} = 'Tem certeza que você trabalhou enquanto esteve em férias?';
    $Self->{Translation}->{'Are you sure that you worked while you were on overtime leave?'} = 'Tem certeza que você trabalhou em período de sobrehora?';
    $Self->{Translation}->{'Are you sure that you worked more than 16 hours?'} = 'Tem certeza que você trabalhou mais de 16 horas?';
    $Self->{Translation}->{'Select all days'} = '';
    $Self->{Translation}->{'Mass entry'} = '';
    $Self->{Translation}->{'Please choose the reason for your absence for the selected days'} = '';
    $Self->{Translation}->{'Please choose at least one day!'} = '';
    $Self->{Translation}->{'Please choose a reason for absence!'} = '';
    $Self->{Translation}->{'Mass Entry'} = '';
    $Self->{Translation}->{'Invalid format! Please enter a time with the format HH:MM.'} = '';
    $Self->{Translation}->{'24:00 is only permitted as end time.'} = '';
    $Self->{Translation}->{'Invalid time! A day has only 24 hours.'} = '';

    # Template: AgentTimeAccountingOverview
    $Self->{Translation}->{'Time reporting monthly overview'} = 'Visão Geral do Relatório de Contabilização Mensal de Tempo';
    $Self->{Translation}->{'Overtime (Hours)'} = 'Hora-extra (em Horas)';
    $Self->{Translation}->{'Overtime (this month)'} = 'Hora-extra (este mês)';
    $Self->{Translation}->{'Overtime (total)'} = 'Hora-extra (total)';
    $Self->{Translation}->{'Remaining overtime leave'} = 'Hora-extra remanescente';
    $Self->{Translation}->{'Vacation (Days)'} = 'Férias (em Dias)';
    $Self->{Translation}->{'Vacation taken (this month)'} = 'Férias gozadas (este mês)';
    $Self->{Translation}->{'Vacation taken (total)'} = 'Férias gozadas (total)';
    $Self->{Translation}->{'Remaining vacation'} = 'Férias remanescentes';
    $Self->{Translation}->{'Sick Leave (Days)'} = 'Dias Afastados (em Dias)';
    $Self->{Translation}->{'Sick leave taken (this month)'} = 'Afastamento por doença (este Mês)';
    $Self->{Translation}->{'Sick leave taken (total)'} = 'Afastamento por doença (total)';
    $Self->{Translation}->{'Previous month'} = 'Mês anterior';
    $Self->{Translation}->{'Next month'} = 'Próximo mês';
    $Self->{Translation}->{'Day'} = 'Dia';
    $Self->{Translation}->{'Weekday'} = 'Dia de semana';
    $Self->{Translation}->{'Working Hours'} = 'Horas de Trabalho';
    $Self->{Translation}->{'Total worked hours'} = 'Total de horas trabalhadas';
    $Self->{Translation}->{'User\'s project overview'} = 'Visão geral de projetos de usuário';
    $Self->{Translation}->{'Hours (monthly)'} = 'Horas (por mês)';
    $Self->{Translation}->{'Hours (Lifetime)'} = 'Horas (total)';
    $Self->{Translation}->{'Grand total'} = 'Total Geral';

    # Template: AgentTimeAccountingProjectReporting
    $Self->{Translation}->{'Project report'} = 'Relatório do projeto';

    # Template: AgentTimeAccountingReporting
    $Self->{Translation}->{'Time reporting'} = 'Relatório de Horas';
    $Self->{Translation}->{'Month Navigation'} = 'Escolha de Mês';
    $Self->{Translation}->{'User reports'} = 'Relatórios de usuário';
    $Self->{Translation}->{'Monthly total'} = 'Total mensal';
    $Self->{Translation}->{'Lifetime total'} = 'Soma Total';
    $Self->{Translation}->{'Overtime leave'} = 'Sobrehora';
    $Self->{Translation}->{'Vacation'} = 'Férias';
    $Self->{Translation}->{'Sick leave'} = 'Licença médica';
    $Self->{Translation}->{'Vacation remaining'} = 'Dias de Afastamento Remanescentes';
    $Self->{Translation}->{'Project reports'} = 'Relatórios de projeto';

    # Template: AgentTimeAccountingSetting
    $Self->{Translation}->{'Edit Time Accounting Project Settings'} = 'Editar definições de projeto para contabilização de tempo';
    $Self->{Translation}->{'Add project'} = 'Adicionar projeto';
    $Self->{Translation}->{'Add Project'} = 'Adicionar Projeto';
    $Self->{Translation}->{'Edit Project Settings'} = 'Editar as configurações do projeto';
    $Self->{Translation}->{'There is already a project with this name. Please, choose a different one.'} = 'Já existe um projeto com este nome. Por favor, escolha um diferente..';
    $Self->{Translation}->{'Edit Time Accounting Settings'} = 'Editar definições de contabilização de tempo';
    $Self->{Translation}->{'Add task'} = 'Adicionar tarefa';
    $Self->{Translation}->{'New user'} = 'Novo usuário';
    $Self->{Translation}->{'Filter for Projects'} = 'Filtro para Projetos';
    $Self->{Translation}->{'Filter for Tasks'} = 'Filtro para Tarefas';
    $Self->{Translation}->{'Filter for Users'} = 'Filtro para Usuários';
    $Self->{Translation}->{'Project List'} = 'Lista de Projetos';
    $Self->{Translation}->{'Task List'} = 'Lista de Tarefas';
    $Self->{Translation}->{'Add Task'} = 'Adicionar Tarefa';
    $Self->{Translation}->{'Edit Task Settings'} = 'Editar Definições da Tarefa';
    $Self->{Translation}->{'There is already a task with this name. Please, choose a different one.'} = 'Já existe uma tarefa com este nome. Por favor, escolha um diferente.';
    $Self->{Translation}->{'User List'} = 'Lista de usuários';
    $Self->{Translation}->{'New User Settings'} = 'Novas Configurações do Usuário';
    $Self->{Translation}->{'Edit User Settings'} = 'Editar as configurações do usuário';
    $Self->{Translation}->{'Comments'} = 'Comentários';
    $Self->{Translation}->{'Show Overtime'} = 'Mostrar Sobrehoras';
    $Self->{Translation}->{'Allow project creation'} = 'Permitir criação de projeto';
    $Self->{Translation}->{'Period Begin'} = 'Início de Período';
    $Self->{Translation}->{'Period End'} = 'Fim de Período';
    $Self->{Translation}->{'Days of Vacation'} = 'Dias de Férias';
    $Self->{Translation}->{'Hours per Week'} = 'Horas por semana';
    $Self->{Translation}->{'Authorized Overtime'} = 'Sobrehora autorizada';
    $Self->{Translation}->{'Period end must be after period begin.'} = 'Final do período deve ser após o período de início.';
    $Self->{Translation}->{'No time periods found.'} = 'Sem períodos de tempo encontrados.';
    $Self->{Translation}->{'Add time period'} = 'Adicionar período de tempo';

    # Template: AgentTimeAccountingView
    $Self->{Translation}->{'View Time Record'} = 'Ver Registro de Tempo';
    $Self->{Translation}->{'View of '} = 'Visão de';
    $Self->{Translation}->{'No data found for this day.'} = 'Não foram encontrados dados para este dia.';

    # SysConfig
    $Self->{Translation}->{'Agent interface notification module to see the number of incomplete working days for the user.'} = 'Agente de notificação módulo de interface para ver o número de dias de trabalho incompleto para o usuário.';
    $Self->{Translation}->{'Default data to use on attribute for ticket search screen. Example: "TicketCreateTimeStartYear=2010;TicketCreateTimeStartMonth=10;TicketCreateTimeStartDay=4;TicketCreateTimeStopYear=2010;TicketCreateTimeStopMonth=11;TicketCreateTimeStopDay=3;".'} = '';
    $Self->{Translation}->{'Default name for new actions.'} = 'O nome padrão para novas ações.';
    $Self->{Translation}->{'Default name for new projects.'} = 'O nome padrão para novos projetos.';
    $Self->{Translation}->{'Default setting for date end.'} = 'Configuração padrão para a data de término.';
    $Self->{Translation}->{'Default setting for date start.'} = 'Configuração padrão para a data de início.';
    $Self->{Translation}->{'Default setting for description.'} = '';
    $Self->{Translation}->{'Default setting for leave days.'} = 'Configuração padrão para dias de ausência.';
    $Self->{Translation}->{'Default setting for overtime.'} = 'Configuração padrão para horas extraordinárias.';
    $Self->{Translation}->{'Default setting for the standard weekly hours.'} = 'Configuração padrão para o horário semanal normal.';
    $Self->{Translation}->{'Default status for new actions.'} = 'Estado padrão para novas ações.';
    $Self->{Translation}->{'Default status for new projects.'} = 'Estado padrão para novos projectos.';
    $Self->{Translation}->{'Default status for new users.'} = 'Estado padrão para os novos usuários.';
    $Self->{Translation}->{'Defines if time accounting must be set to all tickets in bulk action.'} = '';
    $Self->{Translation}->{'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.'} = 'Dentro desta opção de configuração, uma regexp pode ser definida que determina quais projetos são registrados com uma observação deve ser (o regexp funciona com SMX-parâmetros).';
    $Self->{Translation}->{'Edit time accounting settings'} = 'Editar definições de controle de tempo';
    $Self->{Translation}->{'Edit time record'} = 'Editar registro de tempo';
    $Self->{Translation}->{'For how many days ago you can insert working units.'} = 'Define-se quando é possível editar as entradas de tempo em idosos (por exemplo, 10 dias antes.';
    $Self->{Translation}->{'If enabled, the dropdown elements in the edit screen are changed to autocompletion fields.'} = 'Se ativado, os elementos de lista pendente na tela de edição são alteradas para campos autocompletar.';
    $Self->{Translation}->{'If enabled, the user is allowed to enter "on vacation leave", "on sick leave" and "on overtime leave" to multiple dates at once.'} = '';
    $Self->{Translation}->{'Maximum number of working days after which the working units have to be inserted.'} = 'O número máximo de dias de trabalho após o qual as unidades de trabalho devem ser inseridas.';
    $Self->{Translation}->{'Maximum number of working days withouth working units entry after which a warning will be shown.'} = 'O número máximo de dias de trabalho, sem uma entrada, após o qual será exibido um aviso.';
    $Self->{Translation}->{'Project time reporting'} = 'O relatório de Tempo do projeto ';
    $Self->{Translation}->{'Regular expressions for constraining action list according to selected project. Key contains regular expression for project(s), content contains regular expressions for action(s).'} = 'Expressões regulares para restringir a lista de ações de acordo com o projeto selecionado. A chave contém a expressão regular para o projeto(s), o conteúdo contém expressões regulares para ação(ões).';
    $Self->{Translation}->{'Regular expressions for constraining project list according to user groups. Key contains regular expression for project(s), content contains comma separated list of groups.'} = 'Expressões regulares para restringir a lista de projetos de acordo com grupos de usuários. A chave contém a expressão regular para o(s) projeto(s), o conteúdo contém a lista de grupos separados por vírgulas.';
    $Self->{Translation}->{'Saves the login and password on the session table in the database, if "DB" was selected for SessionModule.'} = '';
    $Self->{Translation}->{'Show article as rich text even if rich text writing is disabled.'} = '';
    $Self->{Translation}->{'Specifies if working hours can be inserted without start and end times.'} = 'Especifica se as horas de trabalho podem ser introduzidas sem horários de início e fim.';
    $Self->{Translation}->{'This module forces inserts in TimeAccounting.'} = 'Este módulo obriga inserções na Contabilização de Tempo.';
    $Self->{Translation}->{'This notification module gives a warning if there are too many incomplete working days.'} = 'Este módulo de notificação dá um aviso se houver muitos dias úteis incompletos.';
    $Self->{Translation}->{'Time accounting.'} = 'Contabilidade de Tempo.';
    $Self->{Translation}->{'To use if some actions reduced the working hours (for example, if only half of the traveling time is paid Key => traveling; Content => 50).'} = 'Para usar se alguma ações reduziram as horas úteis (por exemplo, se apenas metade do tempo de viagem é pago, Chave => Viagem; Conteúdo => 50).';

    $Self->{Translation}->{'Mon'} = 'Seg';
    $Self->{Translation}->{'Tue'} = 'Ter';
    $Self->{Translation}->{'Wed'} = 'Qua';
    $Self->{Translation}->{'Thu'} = 'Qui';
    $Self->{Translation}->{'Fri'} = 'Sex';
    $Self->{Translation}->{'Sat'} = 'Sáb';
    $Self->{Translation}->{'Sun'} = 'Dom';
    $Self->{Translation}->{'January'} = 'Janeiro';
    $Self->{Translation}->{'February'} = 'Fevereiro';
    $Self->{Translation}->{'March'} = 'Março';
    $Self->{Translation}->{'April'} = 'Abril';
    $Self->{Translation}->{'May'} = 'Maio';
    $Self->{Translation}->{'June'} = 'Junho';
    $Self->{Translation}->{'July'} = 'Julho';
    $Self->{Translation}->{'August'} = 'Agosto';
    $Self->{Translation}->{'September'} = 'Setembro';
    $Self->{Translation}->{'October'} = 'Outubro';
    $Self->{Translation}->{'November'} = 'Novembro';
    $Self->{Translation}->{'December'} = 'Dezembro';

    $Self->{Translation}->{'Show all projects'} = 'Mostrar todos os projetos';
    $Self->{Translation}->{'Show valid projects'} = 'Mostrar projetos válidos';
    $Self->{Translation}->{'TimeAccounting'} = 'Contabilização de Tempo';
    $Self->{Translation}->{'Actions'} = 'Ações';
    $Self->{Translation}->{'User updated!'} = 'Usuário atualizado!';
    $Self->{Translation}->{'User added!'} = 'Usuário adicionado!';
    $Self->{Translation}->{'Project added!'} = 'Projeto adicionado!';
    $Self->{Translation}->{'Project updated!'} = 'Projeto atualizado!';
    $Self->{Translation}->{'Task added!'} = 'Tarefa adicionada!';
    $Self->{Translation}->{'Task updated!'} = 'Tarefa atualizada!';

    #
    # OBSOLETE ENTRIES FOR REFERENCE, DO NOT TRANSLATE!
    #
    $Self->{Translation}->{'Can\'t delete Working Units!'} = 'Não é possível excluir Unidades de Trabalho!';
    $Self->{Translation}->{'Can\'t save settings, because a day has only 24 hours!'} = 'Um dia só tem 24 horas!';
    $Self->{Translation}->{'Please insert your working hours!'} = 'Informe suas horas trabalhadas!';
    $Self->{Translation}->{'Reporting'} = 'Reportando';
    $Self->{Translation}->{'Successful insert!'} = 'Sucesso na inserção!';

}

1;

# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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 Kernel::System::CheckItem;
use Date::Pcalc qw(Today Days_in_Month Day_of_Week Add_Delta_YMD check_date);
use Time::Local;

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

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

    # check needed Objects
    for my $Needed (
        qw(ParamObject DBObject ModuleReg LogObject UserObject
        ConfigObject TicketObject TimeObject GroupObject)
        )
    {
        $Self->{LayoutObject}->FatalError( Message => "Got no $Needed!" ) if !$Self->{$Needed};
    }

    # create required objects...
    $Self->{TimeAccountingObject} = Kernel::System::TimeAccounting->new(%Param);
    $Self->{CheckItemObject}      = Kernel::System::CheckItem->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', );

    # ---------------------------------------------------------- #
    # show confirmation dialog to delete the entry of this day
    # ---------------------------------------------------------- #
    if ( $Self->{ParamObject}->GetParam( Param => 'DeleteDay' ) ) {

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

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

        # 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}->Output(
            Data         => {%Param},
            TemplateFile => 'AgentTimeAccountingDelete',
        );

        # build the returned data structure
        my %Data = (
            HTML       => $Output,
            DialogType => 'Confirmation',
        );

        # return JSON-String because of AJAX-Mode
        my $OutputJSON = $Self->{LayoutObject}->JSONEncode( Data => \%Data );

        return $Self->{LayoutObject}->Attachment(
            ContentType => 'application/json; charset=' . $Self->{LayoutObject}->{Charset},
            Content     => $OutputJSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }

    # expression add time period was pressed
    elsif (
        $Self->{ParamObject}->GetParam( Param => 'AddPeriod' )
        || $Self->{ParamObject}->GetParam( Param => 'SubmitUserData' )
        )
    {
        my %GetParam = ();

        $GetParam{UserID} = $Self->{ParamObject}->GetParam( Param => 'UserID' );
        my $Periods = $Self->{TimeAccountingObject}->UserLastPeriodNumberGet( UserID => $GetParam{UserID} );

        # check validity of periods
        my %Errors = $Self->_CheckValidityUserPeriods( Period => $Periods );

        # if the period data is ok
        if ( !%Errors ) {

            # get all parameters
            for my $Parameter (qw(Subaction Description Calendar)) {
                $GetParam{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter );
            }
            for my $Parameter (qw(ShowOvertime CreateProject)) {
                $GetParam{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter ) || 0;
            }

            my $Period = 1;
            my %PeriodData;

            my %UserData = $Self->{TimeAccountingObject}->SingleUserSettingsGet(
                UserID => $GetParam{UserID}
            );

            # get parameters for all registered periods
            while ( $UserData{$Period} ) {
                for my $Parameter (qw(WeeklyHours Overtime DateStart DateEnd LeaveDays)) {
                    $PeriodData{$Period}{$Parameter}
                        = $Self->{ParamObject}->GetParam( Param => $Parameter . "[$Period]" )
                        || $UserData{$Period}{$Parameter};
                }
                $PeriodData{$Period}{UserStatus}
                    = $Self->{ParamObject}->GetParam( Param => "PeriodStatus[$Period]" ) || 0;
                $Period++;
            }
            $GetParam{Period} = \%PeriodData;

            # update periods
            if ( !$Self->{TimeAccountingObject}->UserSettingsUpdate(%GetParam) ) {
                return $Self->{LayoutObject}->ErrorScreen(
                    Message => 'Unable to update user settings! Please contact your administrator.'
                );
            }
            if ( $Self->{ParamObject}->GetParam( Param => 'AddPeriod' ) ) {

                # show the edit time settiengs again, but now with a new empty time period line
                return $Self->{LayoutObject}->Redirect(
                    OP =>
                        "Action=AgentTimeAccounting;Subaction=$GetParam{Subaction};UserID=$GetParam{UserID};"
                        . "NewTimePeriod=1",
                );
            }
            else {

                # show the overview of tasks and users
                return $Self->{LayoutObject}->Redirect(
                    OP => "Action=AgentTimeAccounting;Subaction=Setting;User=$Self->{Subaction}",
                );
            }
        }
    }

    # ---------------------------------------------------------- #
    # 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 my $Parameter (qw(Status Year Month Day Notification)) {
            $Param{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter ) || '';
        }
        $Param{RecordsNumber} = $Self->{ParamObject}->GetParam( Param => 'RecordsNumber' ) || 8;
        $Param{InsertWorkingUnits} = $Self->{ParamObject}->GetParam( Param => 'InsertWorkingUnits' );

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

        # check if the given date is a valid date
        # if not valid, set the date to today
        if ( !check_date( $Param{Year}, $Param{Month}, $Param{Day} ) ) {
            $Param{Year}        = $Year;
            $Param{Month}       = $Month;
            $Param{Day}         = $Day;
            $Param{'WrongDate'} = 1;
        }

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

        # for initial use, the first agent with rw-right will be redirected
        # to 'Setting', so 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');

        # hashes to store server side errors
        my %Errors          = ();
        my %ServerErrorData = ();
        my $ErrorIndex      = 1;

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

            # arrays to save all start and end times for some checks
            my ( @StartTimes, @EndTimes );

            # delete previous entries for this day and user
            $Self->{TimeAccountingObject}->WorkingUnitsDelete(
                Year  => $Param{Year},
                Month => $Param{Month},
                Day   => $Param{Day},
            );

            my %CheckboxCheck = ();
            for my $Element (qw(LeaveDay Sick Overtime)) {
                my $Value = $Self->{ParamObject}->GetParam( Param => $Element );
                if ($Value) {
                    $CheckboxCheck{$Element} = 1;
                    $Param{$Element}         = 'checked="checked"';
                }
                else {
                    $Param{$Element} = ''
                }
            }

            # if more than one check box was checked it is a server error
            if ( scalar keys %CheckboxCheck > 1 ) {
                for my $Checkbox ( sort keys %CheckboxCheck ) {
                    $Errors{ $Checkbox . 'Invalid' } = 'ServerError';
                }
            }
            else {

                # insert values (if any)
                if ( scalar keys %CheckboxCheck > 0 ) {
                    if (
                        !$Self->{TimeAccountingObject}->WorkingUnitsInsert(
                            Year     => $Param{Year},
                            Month    => $Param{Month},
                            Day      => $Param{Day},
                            LeaveDay => $CheckboxCheck{LeaveDay} || 0,
                            Sick     => $CheckboxCheck{Sick} || 0,
                            Overtime => $CheckboxCheck{Overtime} || 0,
                        )
                        )
                    {
                        return $Self->{LayoutObject}->ErrorScreen(
                            Message => 'Can\'t insert Working Units!'
                        );
                    }
                    $Param{SuccessfulInsert} = 1;
                }
            }

            ID:
            for my $ID ( 1 .. $Param{RecordsNumber} ) {

                # arrays to save the server errors block to show the error msgs
                my ( @StartTimeServerErrorBlock, @EndTimeServerErrorBlock, @PeriodServerErrorBlock ) = ();

                for my $Parameter (qw(ProjectID ActionID Remark StartTime EndTime Period)) {
                    $Param{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter . '[' . $ID . ']' );
                    if ( $Param{$Parameter} ) {
                        my $ParamRef = \$Param{$Parameter};

                        # delete leading and tailing spaces
                        $ParamRef = $Self->{CheckItemObject}->StringClean(
                            StringRef => $ParamRef,
                            TrimLeft  => 1,
                            TrimRight => 1,
                        );
                    }
                }

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

                # check for missing values
                if ( $Param{ProjectID} && !$Param{ActionID} ) {
                    $Errors{$ErrorIndex}{ActionIDInvalid} = 'ServerError';
                }
                if ( !$Param{ProjectID} && $Param{ActionID} ) {
                    $Errors{$ErrorIndex}{ProjectIDInvalid} = 'ServerError';
                }

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

                # if start or end times are missing, delete the other entry
                if ( !$Param{StartTime} || !$Param{EndTime} ) {
                    $Param{StartTime} = '';
                    $Param{EndTime}   = '';
                }

                # add ':00' to the time, if it doesn't have it yet
                if ( $Param{StartTime} && $Param{StartTime} !~ /^(\d+):(\d+)$/ ) {
                    $Param{StartTime} .= ':00';
                }
                if ( $Param{EndTime} && $Param{EndTime} !~ /^(\d+):(\d+)$/ ) {
                    $Param{EndTime} .= ':00';
                }
                if (
                    ( $Param{StartTime} =~ /^(\d+):(\d+)$/ )
                    && ( $Param{EndTime} =~ /^(\d+):(\d+)$/ )
                    )
                {
                    $Param{StartTime} =~ /^(\d+):(\d+)$/;
                    my $StartTime = $1 * 60 + $2;
                    $Param{EndTime} =~ /^(\d+):(\d+)$/;
                    my $EndTime = $1 * 60 + $2;
                    if ( $ReduceTimeRef->{ $ActionList{ $Param{ActionID} } } ) {
                        $Period = ( $EndTime - $StartTime ) / 60
                            * $ReduceTimeRef->{ $ActionList{ $Param{ActionID} } } / 100;
                    }
                    else {
                        $Period = ( $EndTime - $StartTime ) / 60;
                    }

                    # end time must be after start time
                    if ( $EndTime <= $StartTime ) {
                        $Errors{$ErrorIndex}{EndTimeInvalid} = 'ServerError';
                        push @EndTimeServerErrorBlock, 'EndTimeBeforeStartTimeServerError';
                    }

                    $StartTimes[$ID] = $StartTime;
                    $EndTimes[$ID]   = $EndTime;
                }
                else {
                    if ( $Param{StartTime} && $Param{StartTime} !~ /^(\d+):(\d+)$/ ) {
                        $Errors{$ErrorIndex}{StartTimeInvalid} = 'ServerError';
                        push @StartTimeServerErrorBlock, 'StartTimeInvalidFormatServerError';
                    }
                    if ( $Param{EndTime} && $Param{EndTime} !~ /^(\d+):(\d+)$/ ) {
                        $Errors{$ErrorIndex}{EndTimeInvalid} = 'ServerError';
                        push @EndTimeServerErrorBlock, 'EndTimeInvalidFormatServerError';
                    }
                }

                # negative times are not allowed
                if ( $Param{StartTime} =~ /^-(\d+):(\d+)$/ ) {
                    $Errors{$ErrorIndex}{StartTimeInvalid} = 'ServerError';
                    push @StartTimeServerErrorBlock, 'StartTimeNegativeServerError';
                }
                if ( $Param{EndTime} =~ /^-(\d+):(\d+)$/ ) {
                    $Errors{$ErrorIndex}{EndTimeInvalid} = 'ServerError';
                    push @EndTimeServerErrorBlock, 'EndTimeNegativeServerError';
                }

                # repeated hours are not allowed
                for ( my $Position = $ID - 1; $Position >= 1; $Position-- ) {
                    next if !defined $StartTimes[$Position] || !defined $StartTimes[$ID];

                    if (
                        $StartTimes[$Position] > $StartTimes[$ID]
                        && $StartTimes[$Position] < $EndTimes[$ID]
                        )
                    {
                        $Errors{$ErrorIndex}{EndTimeInvalid} = 'ServerError';
                        push @EndTimeServerErrorBlock, 'EndTimeRepeatedHourServerError'
                            if !grep( /^EndTimeRepeatedHourServerError/,
                            @EndTimeServerErrorBlock );
                    }

                    if (
                        $EndTimes[$Position] > $StartTimes[$ID]
                        && $EndTimes[$Position] < $EndTimes[$ID]
                        )
                    {
                        if ( $EndTimes[$ID] > $EndTimes[$Position] ) {
                            $Errors{$ErrorIndex}{StartTimeInvalid} = 'ServerError';
                            push @StartTimeServerErrorBlock, 'StartTimeRepeatedHourServerError'
                                if !
                                grep( /^StartTimeRepeatedHourServerError$/,
                                @StartTimeServerErrorBlock );
                        }
                        else {
                            $Errors{$ErrorIndex}{EndTimeInvalid} = 'ServerError';
                            push @EndTimeServerErrorBlock, 'EndTimeRepeatedHourServerError'
                                if !
                                grep( /^EndTimeRepeatedHourServerError$/,
                                @EndTimeServerErrorBlock );
                        }
                    }

                    if ( $StartTimes[$Position] == $StartTimes[$ID] ) {
                        $Errors{$ErrorIndex}{StartTimeInvalid} = 'ServerError';
                        push @StartTimeServerErrorBlock, 'StartTimeRepeatedHourServerError'
                            if !
                            grep( /^StartTimeRepeatedHourServerError$/,
                            @StartTimeServerErrorBlock );
                    }

                    if ( $EndTimes[$Position] == $EndTimes[$ID] ) {
                        $Errors{$ErrorIndex}{EndTimeInvalid} = 'ServerError';
                        push @EndTimeServerErrorBlock, 'EndTimeRepeatedHourServerError'
                            if !
                            grep( /^EndTimeRepeatedHourServerError$/,
                            @EndTimeServerErrorBlock );
                    }

                    if (
                        $StartTimes[$ID] > $StartTimes[$Position]
                        && $StartTimes[$ID] < $EndTimes[$Position]
                        )
                    {
                        $Errors{$ErrorIndex}{StartTimeInvalid} = 'ServerError';
                        push @StartTimeServerErrorBlock, 'StartTimeRepeatedHourServerError'
                            if !
                            grep( /^StartTimeRepeatedHourServerError$/,
                            @StartTimeServerErrorBlock );
                    }

                    if (
                        $EndTimes[$ID] > $StartTimes[$Position]
                        && $EndTimes[$ID] < $EndTimes[$Position]
                        )
                    {
                        $Errors{$ErrorIndex}{EndTimeInvalid} = 'ServerError';
                        push @EndTimeServerErrorBlock, 'EndTimeRepeatedHourServerError'
                            if !
                            grep( /^EndTimeRepeatedHourServerError$/,
                            @EndTimeServerErrorBlock );
                    }
                }

                # '24:00' is only permitted as end time
                if ( $StartTimes[$ID] && $StartTimes[$ID] == 1440 ) {
                    $Errors{$ErrorIndex}{StartTimeInvalid} = 'ServerError';
                    push @StartTimeServerErrorBlock, 'StartTime24Hours';
                }

                # times superior to 24:00 are not allowed
                if ( $StartTimes[$ID] && $StartTimes[$ID] > 1440 ) {
                    $Errors{$ErrorIndex}{StartTimeInvalid} = 'ServerError';
                    push @StartTimeServerErrorBlock, 'StartTimeInvalid';
                }
                if ( $EndTimes[$ID] && $EndTimes[$ID] > 1440 ) {
                    $Errors{$ErrorIndex}{EndTimeInvalid} = 'ServerError';
                    push @EndTimeServerErrorBlock, 'EndTimeInvalid';
                }

                # add reference to the server error msgs to be shown
                if ( $Errors{$ErrorIndex} && $Errors{$ErrorIndex}{StartTimeInvalid} ) {
                    $Errors{$ErrorIndex}{StartTimeServerErrorBlock} = \@StartTimeServerErrorBlock;
                }
                if ( $Errors{$ErrorIndex} && $Errors{$ErrorIndex}{EndTimeInvalid} ) {
                    $Errors{$ErrorIndex}{EndTimeServerErrorBlock} = \@EndTimeServerErrorBlock;
                }

                # overwrite period if there are start and end times
                if ( $StartTimes[$ID] && $EndTimes[$ID] ) {
                    $Period = $EndTimes[$ID] - $StartTimes[$ID];

                    # convert period from minutes to hours
                    $Period /= 60;
                }

                # check for errors in the period
                if ( $Period == 0 ) {
                    push @PeriodServerErrorBlock, 'ZeroHoursPeriodServerError';
                    if (
                        $Self->{ConfigObject}->Get('TimeAccounting::InputHoursWithoutStartEndTime')
                        )
                    {
                        $Errors{$ErrorIndex}{PeriodInvalid} = 'ServerError';
                    }
                    else {
                        $Errors{$ErrorIndex}{StartTimeInvalid} = 'ServerError';
                        $Errors{$ErrorIndex}{EndTimeInvalid}   = 'ServerError';
                    }
                }
                else {
                    if ( $Period < 0 ) {
                        $Errors{$ErrorIndex}{PeriodInvalid} = 'ServerError';
                        push @PeriodServerErrorBlock, 'NegativePeriodServerError';
                    }
                }
                if ( $Period > 24 ) {
                    $Errors{$ErrorIndex}{PeriodInvalid} = 'ServerError';
                    push @PeriodServerErrorBlock, 'InvalidHoursPeriodServerError';
                }

                if ( $Errors{$ErrorIndex} && $Errors{$ErrorIndex}{PeriodInvalid} ) {
                    $Errors{$ErrorIndex}{PeriodServerErrorBlock} = \@PeriodServerErrorBlock;
                }

                # if there was an error on this row, save all data in the server error hash
                if ( defined $Errors{$ErrorIndex} ) {
                    for my $Parameter (qw(ProjectID ActionID Remark StartTime EndTime)) {
                        $ServerErrorData{$ErrorIndex}{$Parameter} = $Param{$Parameter};
                    }
                    $ServerErrorData{$ErrorIndex}{Period} = $Period;
                }

                # otherwise, save row on the DB
                else {

                    # initialize the array of working units
                    @{ $Data{WorkingUnits} } = ();

                    # replace 24:00 for 23:59:59 to be a valid entry in the DB
                    $Param{EndTime} = $Param{EndTime} eq '24:00' ? '23:59:59' : $Param{EndTime};

                    my %WorkingUnit = (
                        ProjectID => $Param{ProjectID},
                        ActionID  => $Param{ActionID},
                        Remark    => $Param{Remark},
                        StartTime => $Param{StartTime},
                        EndTime   => $Param{EndTime},
                        Period    => $Period,
                    );
                    push @{ $Data{WorkingUnits} }, \%WorkingUnit;

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

                # increment the error index if there was an error on this row
                $ErrorIndex++ if ( defined $Errors{$ErrorIndex} );
            }

            if (%ServerErrorData) {
                $Param{SuccessfulInsert} = undef;
            }
        }

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

        # get number of working units (=records)
        # if bigger than RecordsNumber, more than the number of default records were saved for this date
        if ( $Data{WorkingUnits} ) {
            my $WorkingUnitsCount = @{ $Data{WorkingUnits} };
            if ( $WorkingUnitsCount > $Param{RecordsNumber} ) {
                $Param{RecordsNumber} = $WorkingUnitsCount;
            }
        }

        if ( $Self->{ConfigObject}->Get('TimeAccounting::InputHoursWithoutStartEndTime') ) {
            $Param{PeriodBlock}   = 'UnitInputPeriod';
            $Frontend{PeriodNote} = '*';
        }
        else {
            $Param{PeriodBlock}   = 'UnitPeriodWithoutInput';
            $Frontend{PeriodNote} = '';
        }

        if (
            $Self->{TimeObject}->SystemTime()
            > 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="checked"' : '';
        $Param{LeaveDay} = $Data{LeaveDay} ? 'checked="checked"' : '';
        $Param{Overtime} = $Data{Overtime} ? 'checked="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;
        for my $ActionID (@ActionIDs) {
            push @JSActions, "['$ActionID', '$ActionList{$ActionID}']";
        }
        $Param{JSActionList} = '[' . ( join ', ', @JSActions ) . ']';

        my $ActionListConstraints = $Self->{ConfigObject}->Get('TimeAccounting::ActionListConstraints');
        my @JSActionListConstraints;
        for my $ProjectNameRegExp ( sort 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} }
        }

        $ErrorIndex = 0;

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

            if ( !$UnitRef ) {
                $ErrorIndex++;
                $ShowError = 1;
            }

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

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

            # generate JavaScript array which will be output to the template
            my @JSProjectList;
            for my $Project ( @{$ProjectList} ) {
                push @JSProjectList,
                    '{id:' . ( $Project->{Key} || '0' ) . ' , name:\'' . $Project->{Value} . '\'}';

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

            $Frontend{ProjectOption} = $Self->{LayoutObject}->BuildSelection(
                Data        => $ProjectList,
                Name        => "ProjectID[$ID]",
                ID          => "ProjectID$ID",
                Translation => 0,
                Class       => 'Validate_TimeAccounting_Project ProjectSelection '
                    . ( $Errors{$ErrorIndex}{ProjectIDInvalid} || '' ),
                OnChange => "TimeAccounting.Agent.EditTimeRecords.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} || $ServerErrorData{$ErrorIndex}{ProjectID},
                ProjectList           => $ProjectList,
                ActionList            => \%ActionList,
                ActionListConstraints => $ActionListConstraints,
            );
            $ActionData->{''} = '';

            if ( $UnitRef && $UnitRef->{ActionID} && $ActionList{ $UnitRef->{ActionID} } ) {
                $ActionData->{ $UnitRef->{ActionID} } = $ActionList{ $UnitRef->{ActionID} };
            }
            elsif (
                $ServerErrorData{$ErrorIndex}
                && $ServerErrorData{$ErrorIndex}{ActionID}
                && $ActionList{ $ServerErrorData{$ErrorIndex}{ActionID} }
                )
            {
                $ActionData->{ $ServerErrorData{$ErrorIndex}{ActionID} }
                    = $ActionList{ $ServerErrorData{$ErrorIndex}{ActionID} };
            }

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

                Data        => $ActionData,
                SelectedID  => $UnitRef->{ActionID} || $ServerErrorData{$ErrorIndex}{ActionID} || '',
                Name        => "ActionID[$ID]",
                ID          => "ActionID$ID",
                Translation => 0,
                Class       => 'Validate_DependingRequiredAND Validate_Depending_ProjectID'
                    . $ID
                    . ' ActionSelection '
                    . ( $Errors{$ErrorIndex}{ActionIDInvalid} || '' ),
            );

            $Param{Remark} = $UnitRef->{Remark} || $ServerErrorData{$ErrorIndex}{Remark} || '';

            my $Period;
            if (
                ( $UnitRef->{Period} && $UnitRef->{Period} == 0 )
                || (
                    defined $ServerErrorData{$ErrorIndex}{Period}
                    && $ServerErrorData{$ErrorIndex}{Period} == 0
                )
                )
            {
                $Period = 0;
            }
            else {
                $Period = $UnitRef->{Period} || $ServerErrorData{$ErrorIndex}{Period} || '';
            }

            for my $TimePeriod (qw(StartTime EndTime)) {
                if ($ShowError) {
                    if ( defined $ServerErrorData{$ErrorIndex}{$TimePeriod} ) {
                        $Param{$TimePeriod} = $ServerErrorData{$ErrorIndex}{$TimePeriod} eq '00:00'
                            ? ''
                            : $ServerErrorData{$ErrorIndex}{$TimePeriod};
                    }
                    else {
                        $Param{$TimePeriod} = '';
                    }
                }
                else {
                    if ( $UnitRef->{$TimePeriod} ) {
                        $Param{$TimePeriod} = $UnitRef->{$TimePeriod} eq '00:00' ? '' : $UnitRef->{$TimePeriod};
                    }
                    else {
                        $Param{$TimePeriod} = '';
                    }
                }
            }

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

            # add proper server error msg for the start and end times
            my $ServerErrorBlockName;
            if ( $Errors{$ErrorIndex} && $Errors{$ErrorIndex}{StartTimeInvalid} ) {
                if ( scalar @{ $Errors{$ErrorIndex}{StartTimeServerErrorBlock} } > 0 ) {
                    while ( @{ $Errors{$ErrorIndex}{StartTimeServerErrorBlock} } ) {
                        $ServerErrorBlockName = shift @{ $Errors{$ErrorIndex}{StartTimeServerErrorBlock} };
                        $Self->{LayoutObject}->Block( Name => $ServerErrorBlockName );
                    }
                }
                else {
                    $Self->{LayoutObject}->Block( Name => 'StartTimeGenericServerError' );
                }
            }
            if ( $Errors{$ErrorIndex} && $Errors{$ErrorIndex}{EndTimeInvalid} ) {
                if ( scalar @{ $Errors{$ErrorIndex}{EndTimeServerErrorBlock} } > 0 ) {
                    while ( @{ $Errors{$ErrorIndex}{EndTimeServerErrorBlock} } ) {
                        $ServerErrorBlockName = shift @{ $Errors{$ErrorIndex}{EndTimeServerErrorBlock} };
                        $Self->{LayoutObject}->Block( Name => $ServerErrorBlockName );
                    }
                }
                else {
                    $Self->{LayoutObject}->Block( Name => 'EndTimeGenericServerError' );
                }
            }

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

            # add proper server error msg for the period
            if ( $Errors{$ErrorIndex} && $Errors{$ErrorIndex}{PeriodInvalid} ) {
                if ( scalar @{ $Errors{$ErrorIndex}{PeriodServerErrorBlock} } > 0 ) {
                    while ( @{ $Errors{$ErrorIndex}{PeriodServerErrorBlock} } ) {
                        $ServerErrorBlockName = shift @{ $Errors{$ErrorIndex}{PeriodServerErrorBlock} };
                        $Self->{LayoutObject}->Block( Name => $ServerErrorBlockName );
                    }
                }
                else {
                    $Self->{LayoutObject}->Block( Name => 'PeriodGenericServerError' );
                }
            }
            else {
                $Self->{LayoutObject}->Block( Name => 'PeriodGenericServerError' );
            }

            # validity check
            if (
                $Param{InsertWorkingUnits}
                && $UnitRef->{ProjectID}
                && $UnitRef->{ActionID}
                && $Param{Sick}
                )
            {
                $Param{BlockName} = 'SickLeaveMessage';
            }
            elsif (
                $Param{InsertWorkingUnits}
                && $UnitRef->{ProjectID}
                && $UnitRef->{ActionID}
                && $Param{LeaveDay}
                )
            {
                $Param{BlockName} = 'VacationMessage';
            }
            elsif (
                $Param{InsertWorkingUnits}
                && $UnitRef->{ProjectID}
                && $UnitRef->{ActionID}
                && $Param{Overtime}
                )
            {
                $Param{BlockName} = 'OvertimeMessage';
            }
        }

        if (
            $Self->{TimeObject}->SystemTime()
            > timelocal( 1, 0, 0, $Param{Day}, $Param{Month} - 1, $Param{Year} - 1900 )
            )
        {
            $Param{Total} = sprintf( "%.2f", ( $Param{Total} || 0 ) );
            $Self->{LayoutObject}->Block(
                Name => 'Total',
                Data => { %Param, %Frontend },
            );
        }

        # validity checks start
        my $ErrorNote;
        if ( $Param{Total} && $Param{Total} > 24 ) {
            $ErrorNote = 'Can\'t save settings, because a day has only 24 hours!';
        }
        elsif ( $Param{InsertWorkingUnits} && $Param{Total} && $Param{Total} > 16 ) {
            $Param{BlockName} = 'More16HoursMessage';
        }
        if ($ErrorNote) {
            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{BlockName} && $Param{SuccessfulInsert} ) {
            $Self->{LayoutObject}->Block(
                Name => 'ShowConfirmation',
                Data => {
                    BlockName => $Param{BlockName},
                    Year      => $Param{Year},
                    Month     => $Param{Month},
                    Day       => $Param{Day},
                },
            );
        }

        $Param{Date} = $Self->{LayoutObject}->BuildDateSelection(
            %Param,
            Validate => 1,
            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 => 'Readonly',
                    Data => {
                        Description =>
                            'This Date is out of limit, but you haven\'t insert this day yet, so you get one(!) chance to insert'
                    },
                );
            }
        }

        # get incomplete working days
        my %IncompleteWorkingDaysList;
        %IncompleteWorkingDays = $Self->{TimeAccountingObject}->WorkingUnitsCompletnessCheck();

        for my $YearID ( sort keys %{ $IncompleteWorkingDays{Incomplete} } ) {
            for my $MonthID ( sort keys %{ $IncompleteWorkingDays{Incomplete}{$YearID} } ) {
                for my $DayID (
                    sort keys %{ $IncompleteWorkingDays{Incomplete}{$YearID}{$MonthID} }
                    )
                {
                    $IncompleteWorkingDaysList{"$YearID-$MonthID-$DayID"} = "$YearID-$MonthID-$DayID";
                    $Param{Incomplete} = 1;
                }
            }
        }

        # Show text, if incomplete working days are available
        if ( $Param{Incomplete} ) {

            # if mass entry option is enabled, show list of working days
            if ( $Self->{ConfigObject}->Get("TimeAccounting::AllowMassEntryForUser") ) {

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

                for my $WorkingDays ( sort keys %IncompleteWorkingDaysList ) {
                    my ( $Year, $Month, $Day ) = split( /-/, $IncompleteWorkingDaysList{$WorkingDays} );
                    $Self->{LayoutObject}->Block(
                        Name => 'IncompleteWorkingDaysMassEntrySingleDay',
                        Data => {
                            Date  => $IncompleteWorkingDaysList{$WorkingDays},
                            Year  => $Year,
                            Month => $Month,
                            Day   => $Day,
                        },
                    );
                }

            }

            # otherwise show incomplete working days as a dropdown
            else {

                # check if current day is incomplete
                # if yes, select the current day in the dropdown
                # otherwise select the empty element
                my $SelectedID;

                if ( $IncompleteWorkingDaysList{"$Param{Year}-$Param{Month}-$Param{Day}"} ) {
                    $SelectedID = "$Param{Year}-$Param{Month}-$Param{Day}";
                }

                my $IncompleWorkingDaysSelect = $Self->{LayoutObject}->BuildSelection(
                    Data         => \%IncompleteWorkingDaysList,
                    SelectedID   => $SelectedID,
                    Name         => "IncompleteWorkingDaysList",
                    PossibleNone => 1,
                );

                $Self->{LayoutObject}->Block(
                    Name => 'IncompleteWorkingDays',
                    Data => {
                        IncompleteWorkingDaysSelect => $IncompleWorkingDaysSelect,
                    },
                );
            }
        }

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

        # get working days of the user's calendar
        my $CalendarName = 'TimeWorkingHours';
        $CalendarName .= $UserData{Calendar} ? "::Calendar$UserData{Calendar}" : '';
        my $CalendarWorkingHours = $Self->{ConfigObject}->Get($CalendarName);

        # show "other times" block, if necessary
        if (
            @{ $CalendarWorkingHours->{ $WeekdayArray[ $Param{Weekday} - 1 ] } }
            && !$VacationCheck
            )
        {
            $Self->{LayoutObject}->Block(
                Name => 'OtherTimes',
                Data => {
                    %Param,
                    %Frontend,
                    %Errors,
                },
            );
        }

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

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

        # enable autocompletion?
        $Param{EnableAutocompletion} = $Self->{ConfigObject}->Get("TimeAccounting::EnableAutoCompletion");

        # build output
        my $Output = $Self->{LayoutObject}->Header( Title => 'Edit' );

        if ( !$IncompleteWorkingDays{EnforceInsert} ) {
            $Output .= $Self->{LayoutObject}->NavigationBar();
            $Self->{LayoutObject}->Block(
                Name => 'OverviewProject',
                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',
                );
            }
        }
        else {
            if ( $IncompleteWorkingDays{Warning} ) {
                $Output .= $Self->{LayoutObject}->Notify(
                    Info     => 'Please insert your working hours!',
                    Priority => 'Error'
                );
            }
        }

        if ($ErrorNote) {
            $Output .= $Self->{LayoutObject}->Notify(
                Info     => $ErrorNote,
                Priority => 'Error'
            );
        }
        elsif ( defined $Param{SuccessfulInsert} )
        {
            $Output .= $Self->{LayoutObject}->Notify(
                Info => 'Successful insert!',
            );
        }

        # show mass entry notification
        if ( $Param{Notification} eq 'Error' ) {
            $Output .= $Self->{LayoutObject}->Notify(
                Info     => 'Error while inserting multiple dates!',
                Priority => 'Error'
            );
        }
        elsif ( $Param{Notification} eq 'Successful' ) {
            $Output .= $Self->{LayoutObject}->Notify(
                Info => 'Successfully inserted entries for several dates!',
            );
        }

        # show notification if wrong date was selected
        if ( $Param{WrongDate} ) {
            $Output .= $Self->{LayoutObject}->Notify(
                Info     => 'Entered date was invalid! Date was changed to today.',
                Priority => 'Error'
            );
        }

        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingEdit',
            Data         => {
                %Param,
                %Frontend,
            },
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # view older day inserts
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'View' ) {

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

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

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

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

        # get current date and time
        my ( $Sec, $Min, $Hour, $Day, $Month, $Year ) = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
        );

        my $MaxAllowedInsertDays = $Self->{ConfigObject}->Get('TimeAccounting::MaxAllowedInsertDays') || '10';
        ( $Param{YearAllowed}, $Param{MonthAllowed}, $Param{DayAllowed} )
            = Add_Delta_YMD( $Year, $Month, $Day, 0, 0, -$MaxAllowedInsertDays );

        # redirect to the edit screen, if necessary
        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
            ) && $Param{UserID} == $Self->{UserID}
            )
        {
            return $Self->{LayoutObject}->Redirect(
                OP =>
                    "Action=$Self->{Action};Subaction=Edit;Year=$Param{Year};Month=$Param{Month};Day=$Param{Day}"
            );
        }

        # 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',
            Validate => 1,
            Class    => $Param{Errors}->{DateInvalid},
        );

        # 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] ) {

            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} ) }
            );
        }
        else {
            $Self->{LayoutObject}->Block( Name => 'NoDataFound' );
        }

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

        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 my $Parameter (qw(Status Day Month Year UserID ProjectStatusShow)) {
            $Param{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter );
        }
        $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},
            );

            my $Date = sprintf( "%04d-%02d-%02d", $Param{Year}, $Param{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 $UserCalendar = $UserData{Calendar} || '';
            my $Zone = $Self->{ConfigObject}->Get( "TimeZone::Calendar" . $UserCalendar );
            if ($Zone) {
                my $ZoneSeconds = $Zone * 60 * 60;
                $DayStartTime = $DayStartTime - $ZoneSeconds;
                $DayStopTime  = $DayStopTime - $ZoneSeconds;
            }

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

            if ( $Param{Year} eq $Year && $Param{Month} eq $Month && $CurrentDay eq $Day ) {
                $Param{Class} = 'Active';
            }
            elsif ($VacationCheck) {
                $Param{Class}   = 'Vacation';
                $Param{Comment} = $VacationCheck;
            }
            elsif ($ThisDayWorkingTime) {
                $Param{Class} = 'WorkingDay';
            }
            else {
                $Param{Class} = 'NonWorkingDay';
            }

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

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

        $Param{ShowProjects} = 'Show ' . $Param{ProjectStatusShow} . ' projects';

        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{Status} = $ProjectRef->{Status} ? '' : 'passiv';

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

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

            if ($ActionsRef) {
                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},
                            },
                        );

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

                        if ( $UserData{CreateProject} ) {

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

                # Now show row with total result of all actions of this project
                $Param{Hours}      = sprintf( "%.2f", $Total );
                $Param{HoursTotal} = sprintf( "%.2f", $TotalTotal );
                $Param{TotalHours}      += $Total;
                $Param{TotalHoursTotal} += $TotalTotal;
                $Self->{LayoutObject}->Block(
                    Name => 'ActionTotal',
                    Data => {%Param},
                );
            }
        }
        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' ) {

        # get user data
        my %UserData = $Self->{TimeAccountingObject}->UserGet(
            UserID => $Self->{UserID},
        );

        # permission check
        if ( $UserData{CreateProject} || $Self->{AccessRw} ) {

            # get the user action to show a msg if an user was updated or added
            my $Note = $Self->{ParamObject}->GetParam( Param => 'User' );

            # build output
            $Self->_SettingOverview();
            my $Output = $Self->{LayoutObject}->Header( Title => 'Setting' );
            $Output .= $Self->{LayoutObject}->NavigationBar();

            # show a notification msg if proper
            if ($Note) {
                $Output .= $Note eq 'EditUser'
                    ? $Self->{LayoutObject}->Notify( Info => 'User updated!' )
                    : $Self->{LayoutObject}->Notify( Info => 'User added!' );
            }

            $Output .= $Self->{LayoutObject}->Output(
                Data         => \%Param,
                TemplateFile => 'AgentTimeAccountingSetting'
            );
            $Output .= $Self->{LayoutObject}->Footer();
            return $Output;
        }

        # return no permission screen
        return $Self->{LayoutObject}->NoPermission( WithHeader => 'yes' );
    }

    # ---------------------------------------------------------- #
    # 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 my $Parameter (qw(Status Month Year ProjectStatusShow)) {
            $Param{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter );
        }

        # 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 ) } = $MonthArray[$ID];
        }

        $Frontend{MonthOption} = $Self->{LayoutObject}->BuildSelection(
            Data        => \%Month,
            SelectedID  => $Param{Month} || '',
            Name        => 'Month',
            Sort        => 'NumericKey',
            Translation => 1,
        );

        my @Year = ( $Year - 4 .. $Year + 1 );

        $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 my $Parameter (
                qw(LeaveDay Overtime WorkingHours Sick LeaveDayRemaining OvertimeTotal)
                )
            {
                $Param{$Parameter} = sprintf( "%.2f", ( $UserReport{$UserID}{$Parameter} || 0 ) );
                $Param{ 'Total' . $Parameter } += $Param{$Parameter};
            }

            # 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 my $Parameter (
            qw(TotalLeaveDay TotalOvertime TotalWorkingHours
            TotalSick TotalLeaveDayRemaining TotalOvertimeTotal)
            )
        {
            $Param{$Parameter} = sprintf( "%.2f", ( $Param{$Parameter} ) || 0 );
        }

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

        $Param{ShowProjects} = 'Show ' . $Param{ProjectStatusShow} . ' projects';

        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{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{Hours}      = sprintf( "%.2f", $Total );
            $Param{HoursTotal} = sprintf( "%.2f", $TotalTotal );
            $Param{TotalHours}      += $Total;
            $Param{TotalHoursTotal} += $TotalTotal;
            $Self->{LayoutObject}->Block(
                Name => 'ActionTotal',
                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 ( sort 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 ( sort 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 ( sort 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 => sprintf( "%.2f", $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 = sprintf(
            "%.2f",
            $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;
    }

    # ---------------------------------------------------------- #
    # add project
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'AddProject' ) {
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Self->_ProjectSettingsEdit( Action => 'AddProject' );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # add project action
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'AddProjectAction' ) {

        # challenge token check for write action
        $Self->{LayoutObject}->ChallengeTokenCheck();

        my $Note = '';
        my $ProjectID;
        my ( %GetParam, %Errors );

        # get parameters
        for my $Parameter (qw(Project ProjectDescription)) {
            $GetParam{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter ) || '';
        }
        $GetParam{ProjectStatus} = $Self->{ParamObject}->GetParam( Param => 'ProjectStatus' )
            || '0';

        # check for needed data
        if ( !$GetParam{Project} ) {
            $Errors{ProjectInvalid}   = 'ServerError';
            $Errors{ProjectErrorType} = 'ProjectMissingValue';
        }
        else {

            # check that the name is unique
            my %ExistingProject = $Self->{TimeAccountingObject}->ProjectGet( Project => $GetParam{Project} );
            if (%ExistingProject) {
                $Errors{ProjectInvalid}   = 'ServerError';
                $Errors{ProjectErrorType} = 'ProjectDuplicateName';
            }
        }

        # if no errors occurred
        if ( !%Errors ) {

            # add project
            $ProjectID = $Self->{TimeAccountingObject}->ProjectSettingsInsert(%GetParam);

            if ($ProjectID) {

                # build the output
                $Self->_SettingOverview();
                my $Output = $Self->{LayoutObject}->Header();
                $Output .= $Self->{LayoutObject}->NavigationBar();
                $Output .= $Self->{LayoutObject}->Notify( Info => 'Project added!' );
                $Output .= $Self->{LayoutObject}->Output(
                    TemplateFile => 'AgentTimeAccountingSetting',
                    Data         => {%GetParam},
                );
                $Output .= $Self->{LayoutObject}->Footer();
                return $Output;
            }
            else {
                $Note = $Self->{LogObject}->GetLogEntry(
                    Type => 'Error',
                    What => 'Message',
                );
            }
        }

        # something went wrong
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Note
            ? $Self->{LayoutObject}->Notify(
            Priority => 'Error',
            Info     => $Note,
            )
            : '';
        $Self->_ProjectSettingsEdit(
            Action => 'AddProject',
            %GetParam,
            %Errors,
        );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # edit project
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'EditProject' ) {
        my $ID = $Self->{ParamObject}->GetParam( Param => 'ID' );

        # get project data
        my %Project = $Self->{TimeAccountingObject}->ProjectGet( ID => $ID );

        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Self->_ProjectSettingsEdit(
            Action => 'EditProject',
            %Project,
        );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # edit project action
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'EditProjectAction' ) {

        # challenge token check for write action
        $Self->{LayoutObject}->ChallengeTokenCheck();

        my $Note = '';
        my ( %GetParam, %Errors );

        # get parameters
        $GetParam{ID} = $Self->{ParamObject}->GetParam( Param => 'ID' ) || '';
        $GetParam{ProjectStatus} = $Self->{ParamObject}->GetParam( Param => 'ProjectStatus' )
            || '0';
        for my $Parameter (qw(Project ProjectDescription)) {
            $GetParam{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter ) || '';
        }

        # check for needed data
        if ( !$GetParam{Project} ) {
            $Errors{ProjectInvalid} = 'ServerError';
        }
        else {

            # check that the name is unique
            my %ExistingProject = $Self->{TimeAccountingObject}->ProjectGet( Project => $GetParam{Project} );

            # if the project name is found, check that the ID is different
            if ( %ExistingProject && $ExistingProject{ID} ne $GetParam{ID} ) {
                $Errors{ProjectInvalid}   = 'ServerError';
                $Errors{ProjectErrorType} = 'ProjectDuplicateName';
            }
        }

        # if no errors occurred
        if ( !%Errors ) {

            # edit project
            if ( $Self->{TimeAccountingObject}->ProjectSettingsUpdate(%GetParam) ) {

                $Self->_SettingOverview();
                my $Output = $Self->{LayoutObject}->Header();
                $Output .= $Self->{LayoutObject}->NavigationBar();
                $Output .= $Self->{LayoutObject}->Notify( Info => 'Project updated!' );
                $Output .= $Self->{LayoutObject}->Output(
                    TemplateFile => 'AgentTimeAccountingSetting',
                    Data         => \%Param,
                );
                $Output .= $Self->{LayoutObject}->Footer();
                return $Output;
            }
            else {
                $Note = $Self->{LogObject}->GetLogEntry(
                    Type => 'Error',
                    What => 'Message',
                );
            }
        }

        # something went wrong
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Note
            ? $Self->{LayoutObject}->Notify(
            Priority => 'Error',
            Info     => $Note,
            )
            : '';
        $Self->_ProjectSettingsEdit(
            Action => 'EditProject',
            %GetParam,
            %Param,
            %Errors,
        );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # add task
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'AddTask' ) {
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Self->_TaskSettingsEdit( Action => 'AddTask' );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # add task action
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'AddTaskAction' ) {

        # challenge token check for write action
        $Self->{LayoutObject}->ChallengeTokenCheck();

        my $Note = '';
        my $TaskID;
        my ( %GetParam, %Errors );

        # get parameters
        $GetParam{Task} = $Self->{ParamObject}->GetParam( Param => 'Task' ) || '';
        $GetParam{TaskStatus} = $Self->{ParamObject}->GetParam( Param => 'TaskStatus' )
            || '0';

        # check for needed data
        if ( !$GetParam{Task} ) {
            $Errors{TaskInvalid}   = 'ServerError';
            $Errors{TaskErrorType} = 'TaskMissingValue';
        }
        else {

            # check that the name is unique
            my %ExistingTask = $Self->{TimeAccountingObject}->ActionGet( Action => $GetParam{Task} );
            if (%ExistingTask) {
                $Errors{TaskInvalid}   = 'ServerError';
                $Errors{TaskErrorType} = 'TaskDuplicateName';
            }
        }

        # if no errors occurred
        if ( !%Errors ) {

            # add task
            $TaskID = $Self->{TimeAccountingObject}->ActionSettingsInsert(
                Action       => $GetParam{Task},
                ActionStatus => $GetParam{TaskStatus},
            );

            if ($TaskID) {

                # build the output
                $Self->_SettingOverview();
                my $Output = $Self->{LayoutObject}->Header();
                $Output .= $Self->{LayoutObject}->NavigationBar();
                $Output .= $Self->{LayoutObject}->Notify( Info => 'Task added!' );
                $Output .= $Self->{LayoutObject}->Output(
                    TemplateFile => 'AgentTimeAccountingSetting',
                    Data         => {},
                );
                $Output .= $Self->{LayoutObject}->Footer();
                return $Output;
            }
            else {
                $Note = $Self->{LogObject}->GetLogEntry(
                    Type => 'Error',
                    What => 'Message',
                );
            }
        }

        # something went wrong
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Note
            ? $Self->{LayoutObject}->Notify(
            Priority => 'Error',
            Info     => $Note,
            )
            : '';
        $Self->_TaskSettingsEdit(
            Action => 'AddTask',
            %GetParam,
            %Errors,
        );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # edit task
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'EditTask' ) {
        my $ID = $Self->{ParamObject}->GetParam( Param => 'ActionID' ) || '';

        # get project data
        my %Task = $Self->{TimeAccountingObject}->ActionGet( ID => $ID );

        my %TaskData = (
            Task       => $Task{Action},
            TaskStatus => $Task{ActionStatus},
        );

        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Self->_TaskSettingsEdit(
            Action   => 'EditTask',
            ActionID => $ID,
            %TaskData,
        );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # edit project action
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'EditTaskAction' ) {

        # challenge token check for write action
        $Self->{LayoutObject}->ChallengeTokenCheck();

        my $Note = '';
        my ( %GetParam, %Errors );

        # get parameters
        $GetParam{ActionID}   = $Self->{ParamObject}->GetParam( Param => 'ActionID' )   || '';
        $GetParam{TaskStatus} = $Self->{ParamObject}->GetParam( Param => 'TaskStatus' ) || '0';
        $GetParam{Task}       = $Self->{ParamObject}->GetParam( Param => 'Task' )       || '';

        # check for needed data
        if ( !$GetParam{Task} ) {
            $Errors{TaskInvalid} = 'ServerError';
        }
        else {

            # check that the name is unique
            my %ExistingTask = $Self->{TimeAccountingObject}->ActionGet( Action => $GetParam{Task} );

            # if the task name is found, check that the ID is different
            if ( %ExistingTask && $ExistingTask{ID} ne $GetParam{ActionID} ) {
                $Errors{TaskInvalid}   = 'ServerError';
                $Errors{TaskErrorType} = 'TaskDuplicateName';
            }
        }

        # if no errors occurred
        if ( !%Errors ) {

            # edit action (task)
            my $ActionUpdate = $Self->{TimeAccountingObject}->ActionSettingsUpdate(
                ActionID     => $GetParam{ActionID},
                Action       => $GetParam{Task},
                ActionStatus => $GetParam{TaskStatus},
            );

            if ($ActionUpdate) {
                $Self->_SettingOverview();
                my $Output = $Self->{LayoutObject}->Header();
                $Output .= $Self->{LayoutObject}->NavigationBar();
                $Output .= $Self->{LayoutObject}->Notify( Info => 'Task updated!' );
                $Output .= $Self->{LayoutObject}->Output(
                    TemplateFile => 'AgentTimeAccountingSetting',
                    Data         => \%Param,
                );
                $Output .= $Self->{LayoutObject}->Footer();
                return $Output;
            }
            else {
                $Note = $Self->{LogObject}->GetLogEntry(
                    Type => 'Error',
                    What => 'Message',
                );
            }
        }

        # something went wrong
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Note
            ? $Self->{LayoutObject}->Notify(
            Priority => 'Error',
            Info     => $Note,
            )
            : '';
        $Self->_TaskSettingsEdit(
            Action => 'EditTask',
            %GetParam,
            %Param,
            %Errors,
        );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # add user
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'AddUser' ) {

        # get parameters
        my $NewUserID = $Self->{ParamObject}->GetParam( Param => 'NewUserID' )
            || $Self->{ParamObject}->GetParam( Param => 'UserID' )
            || '';
        if ( !$NewUserID ) {
            return $Self->{LayoutObject}->ErrorScreen(
                Message => 'The UserID is not valid!'
            );
        }

        my $NewTimePeriod = $Self->{ParamObject}->GetParam( Param => 'NewTimePeriod' );

        my $LastPeriodNumber = $Self->{TimeAccountingObject}->UserLastPeriodNumberGet(
            UserID => $NewUserID,
        );

        my $Success = $Self->{TimeAccountingObject}->UserSettingsInsert(
            UserID => $NewUserID,
            Period => $LastPeriodNumber + 1,
        );

        # if it is not an action about adding a new time period
        if ( !$NewTimePeriod ) {
            if ( !$Success ) {
                return $Self->{LayoutObject}->ErrorScreen(
                    Message => 'Can\'t insert user data!'
                );
            }

            my %Groups = $Self->{GroupObject}->GroupList( Valid => 1 );
            my %GroupData = $Self->{GroupObject}->GroupMemberList(
                UserID => $NewUserID,
                Type   => 'ro',
                Result => 'HASH',
            );
            for my $GroupKey ( sort keys %Groups ) {
                if ( $Groups{$GroupKey} eq 'time_accounting' && !$GroupData{$GroupKey} ) {

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

        my %User = $Self->{UserObject}->GetUserData( UserID => $NewUserID );
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Self->_UserSettingsEdit(
            Action    => 'AddUser',
            Subaction => 'AddUser',
            %User,
        );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    # ---------------------------------------------------------- #
    # edit user settings
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'EditUser' ) {
        my $ID = $Self->{ParamObject}->GetParam( Param => 'UserID' ) || '';
        if ( !$ID ) {
            return $Self->{LayoutObject}->ErrorScreen(
                Message => 'The UserID is not valid!'
            );
        }

        my $NewTimePeriod = $Self->{ParamObject}->GetParam( Param => 'NewTimePeriod' );
        my $LastPeriodNumber = $Self->{TimeAccountingObject}->UserLastPeriodNumberGet(
            UserID => $ID,
        );

        # if it is an action about adding a new time period, insert it
        if ($NewTimePeriod) {
            my $Success = $Self->{TimeAccountingObject}->UserSettingsInsert(
                UserID => $ID,
                Period => $LastPeriodNumber + 1,
            );
            if ( !$Success ) {
                return $Self->{LayoutObject}->ErrorScreen(
                    Message => 'Unable to add time period! Please contact your administrator.',
                );
            }
        }

        my %Errors = ();

        if (
            $Self->{ParamObject}->GetParam( Param => 'AddPeriod' )
            || $Self->{ParamObject}->GetParam( Param => 'SubmitUserData' )
            )
        {

            # check validity of periods
            %Errors = $Self->_CheckValidityUserPeriods( Period => $LastPeriodNumber );
        }

        # get user data
        my %User = $Self->{UserObject}->GetUserData( UserID => $ID );

        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Self->_UserSettingsEdit(
            Action    => 'EditUser',
            Subaction => 'EditUser',
            UserID    => $ID,
            Errors    => \%Errors,
            Periods   => $LastPeriodNumber,
            %User,
        );
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentTimeAccountingSetting',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();
        return $Output;
    }

    #---------------------------------------------------------- #
    # mass entry
    # ---------------------------------------------------------- #
    elsif ( $Self->{Subaction} eq 'MassEntry' ) {

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

        # get params
        for my $Parameter (qw(Dates LeaveDay Sick Overtime)) {
            $Param{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter ) || '';
        }

        # split up dates
        my @Dates = split /[|]/, $Param{Dates};
        my $InsertError = 0;

        # save entries in the db
        for my $Date (@Dates) {

            my ( $Year, $Month, $Day ) = split /[-]/, $Date;

            if (
                !$Self->{TimeAccountingObject}->WorkingUnitsInsert(
                    Year     => $Year,
                    Month    => $Month,
                    Day      => $Day,
                    LeaveDay => $Param{LeaveDay} || 0,
                    Sick     => $Param{Sick} || 0,
                    Overtime => $Param{Overtime} || 0,
                )
                )
            {
                $InsertError = 1;
            }

        }

        # redirect to edit screen with log message
        return $Self->{LayoutObject}->Redirect(
            OP => 'Action=AgentTimeAccounting;Subaction=Edit;Notification='
                . ( $InsertError ? 'Error' : 'Successful' )
        );

    }

    # ---------------------------------------------------------- #
    # show error screen
    # ---------------------------------------------------------- #
    return $Self->{LayoutObject}->ErrorScreen( Message => 'Invalid Subaction process!' );
}

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

    my %Errors = ();
    my %GetParam;

    for ( my $Period = 1; $Period <= $Param{Period}; $Period++ ) {

        # check for needed data
        for my $Parameter (qw(DateStart DateEnd LeaveDays)) {
            $GetParam{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter . "[$Period]" );
            if ( !$GetParam{$Parameter} ) {
                $Errors{ $Parameter . '-' . $Period . 'Invalid' }   = 'ServerError';
                $Errors{ $Parameter . '-' . $Period . 'ErrorType' } = 'MissingValue';
            }
        }
        my ( $Year, $Month, $Day ) = split( '-', $GetParam{DateStart} );
        my $StartDate = $Self->{TimeObject}->Date2SystemTime(
            Year   => $Year,
            Month  => $Month,
            Day    => $Day,
            Hour   => 0,
            Minute => 0,
            Second => 0,
        );
        ( $Year, $Month, $Day ) = split( '-', $GetParam{DateEnd} );
        my $EndDate = $Self->{TimeObject}->Date2SystemTime(
            Year   => $Year,
            Month  => $Month,
            Day    => $Day,
            Hour   => 0,
            Minute => 0,
            Second => 0,
        );
        if ( !$StartDate ) {
            $Errors{ 'DateStart-' . $Period . 'Invalid' }   = 'ServerError';
            $Errors{ 'DateStart-' . $Period . 'ErrorType' } = 'Invalid';
        }
        if ( !$EndDate ) {
            $Errors{ 'DateEnd-' . $Period . 'Invalid' }   = 'ServerError';
            $Errors{ 'DateEnd-' . $Period . 'ErrorType' } = 'Invalid';
        }
        if ( $StartDate && $EndDate && $StartDate >= $EndDate ) {
            $Errors{ 'DateEnd-' . $Period . 'Invalid' }   = 'ServerError';
            $Errors{ 'DateEnd-' . $Period . 'ErrorType' } = 'BeforeDateStart';
        }
    }

    return %Errors;
}

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 my $GroupKey ( sort keys %GroupList ) {
        if ( $GroupList{$GroupKey} 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 ( sort 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 ( sort keys %{ $Param{ActionList} } ) {

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

                REGEXP:
                for my $ProjectNameRegExp ( sort 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 ( sort 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 ( sort 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 ( sort keys %{ $ProjectData{Project} } ) {
        if ( $ProjectData{Project}{$ProjectID} =~ m{$Project2RemarkRegExp}smx ) {
            push @Projects2Remark, $ProjectID;
        }
    }

    return join '|', @Projects2Remark;
}

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

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

    $Self->{LayoutObject}->Block( Name => 'ActionListProject' );
    $Self->{LayoutObject}->Block( Name => 'ActionSettingOverview' );

    # define status list
    my %StatusList = (
        1 => 'valid',
        0 => 'invalid',
    );

    my $ProjectStatus = 1;
    if ( defined $Param{ProjectStatus} ) {
        $ProjectStatus = $Param{ProjectStatus}
    }

    $Param{StatusOption} = $Self->{LayoutObject}->BuildSelection(
        Data       => \%StatusList,
        SelectedID => $ProjectStatus,
        Name       => 'ProjectStatus',
    );

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

    # shows header
    if ( $Param{Action} eq 'EditProject' ) {
        $Self->{LayoutObject}->Block( Name => 'HeaderEditProject' );
    }
    else {
        $Self->{LayoutObject}->Block( Name => 'HeaderAddProject' );
    }

    # show server error msg (if any) for the project name
    if ( $Param{ProjectErrorType} ) {
        $Self->{LayoutObject}->Block( Name => $Param{ProjectErrorType} );
    }

    return 1;
}

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

    my %Project = ();
    my %Data    = ();

    # build output
    $Self->{LayoutObject}->Block(
        Name => 'Setting',
    );
    $Self->{LayoutObject}->Block( Name => 'ActionListSetting' );
    $Self->{LayoutObject}->Block( Name => 'ActionAddProject' );

    # hash to save registered users
    my %User;

    if ( $Self->{AccessRw} ) {
        $Self->{LayoutObject}->Block( Name => 'ActionAddTask' );

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

        # get list of registered users (if any)
        %User = $Self->{TimeAccountingObject}->UserList();

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

            # delete already registered user from the 'new' list
            delete $ShownUsers{$UserInfo};
        }

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

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

    if ( $Self->{AccessRw} ) {
        $Self->{LayoutObject}->Block( Name => 'TaskFilter' );
        $Self->{LayoutObject}->Block( Name => 'UserFilter' );
    }

    # Show project data
    %Project = $Self->{TimeAccountingObject}->ProjectSettingsGet();

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

    # define status list
    my %StatusList = (
        1 => 'valid',
        0 => 'invalid',
    );

    # show list of available projects (if any)
    if ( $Project{Project} ) {
        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;
            $Param{Status}             = $StatusList{ $Project{ProjectStatus}{$ProjectID} };

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

    # otherwise, show a no data found msg
    else {
        $Self->{LayoutObject}->Block( Name => 'NoProjectDataFoundMsg' );
    }

    if ( $Self->{AccessRw} ) {

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

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

        # show list of available tasks/actions (if any)
        if (%Action) {
            for my $ActionID ( sort { $Action{$a}{Action} cmp $Action{$b}{Action} } keys %Action ) {
                $Param{Action}   = $Action{$ActionID}{Action};
                $Param{ActionID} = $ActionID;
                $Param{Status}   = $StatusList{ $Action{$ActionID}{ActionStatus} };

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

        # otherwise, show a no data found msg
        else {
            $Self->{LayoutObject}->Block( Name => 'NoSettingDataFoundMsg' );
        }

        # show user data
        $Self->{LayoutObject}->Block(
            Name => 'OverviewResultUser',
            Data => \%Param,
        );

        # show list of registered users (if any)
        if (%User) {
            for my $UserID ( sort { $User{$a} cmp $User{$b} } keys %User ) {

                # get missing user data
                my %UserData = $Self->{TimeAccountingObject}->UserGet( UserID => $UserID );
                my %UserGeneralData = $Self->{UserObject}->GetUserData( UserID => $UserID );

                $Param{User}
                    = "$UserGeneralData{UserFirstname} $UserGeneralData{UserLastname} ($UserGeneralData{UserLogin})";
                $Param{UserID}     = $UserID;
                $Param{Comment}    = $UserData{Description};
                $Param{CalendarNo} = $UserData{Calendar};
                $Param{Calendar}   = $Self->{ConfigObject}->Get(
                    "TimeZone::Calendar"
                        . ( $Param{CalendarNo} || '' ) . "Name"
                ) || 'Default';

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

        # otherwise, show a no data found msg
        else {
            $Self->{LayoutObject}->Block( Name => 'NoUserDataFoundMsg' );
        }
    }

    return 1;
}

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

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

    $Self->{LayoutObject}->Block( Name => 'ActionListSetting' );
    $Self->{LayoutObject}->Block( Name => 'ActionOverviewSetting' );

    # define status list
    my %StatusList = (
        1 => 'valid',
        0 => 'invalid',
    );

    my $TaskStatus = 1;
    if ( defined $Param{TaskStatus} ) {
        $TaskStatus = $Param{TaskStatus}
    }

    $Param{StatusOption} = $Self->{LayoutObject}->BuildSelection(
        Data       => \%StatusList,
        SelectedID => $TaskStatus,
        Name       => 'TaskStatus',
    );

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

    # shows header
    if ( $Param{Action} eq 'EditTask' ) {
        $Self->{LayoutObject}->Block( Name => 'HeaderEditTask' );
    }
    else {
        $Self->{LayoutObject}->Block( Name => 'HeaderAddTask' );
    }

    # show server error msg (if any) for the task name
    if ( $Param{TaskErrorType} ) {
        $Self->{LayoutObject}->Block( Name => $Param{TaskErrorType} );
    }

    return 1;
}

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

    # get parameters
    for my $Parameter (qw(Description ShowOvertime CreateProject Calendar)) {
        $GetParam{$Parameter} = $Self->{ParamObject}->GetParam( Param => $Parameter );
    }

    # the datepicker used in this screen is non-standard
    # Therefor we have to define this var in the LayoutObject for automatic config generation in the JS footer
    $Self->{LayoutObject}->{HasDatepicker} = 1;

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

    $Self->{LayoutObject}->Block( Name => 'ActionListSetting' );
    $Self->{LayoutObject}->Block( Name => 'ActionOverviewSetting' );
    $Self->{LayoutObject}->Block( Name => 'Reference' );

    # define status list
    my %StatusList = (
        1 => 'valid',
        0 => 'invalid',
    );

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

    # get user data
    my %UserData = $Self->{TimeAccountingObject}->UserGet( UserID => $Param{UserID} );

    $Param{CalendarOption} = $Self->{LayoutObject}->BuildSelection(
        Data        => $CalendarListRef,
        Name        => 'Calendar',
        Translation => 1,
        SelectedID  => $GetParam{Calendar} || $UserData{Calendar} || 0,
    );

    $Param{Description} = $GetParam{Description} || $UserData{Description} || '';

    $Self->{LayoutObject}->Block(
        Name => 'OverviewUpdateUser',
        Data => {
            %Param,
            ShowOvertime => ( $GetParam{ShowOvertime} || $UserData{ShowOvertime} )
            ? 'checked="checked"'
            : '',
            CreateProject => ( $GetParam{CreateProject} || $UserData{CreateProject} )
            ? 'checked="checked"'
            : '',
            }
    );

    # shows header
    if ( $Param{Action} eq 'EditUser' ) {
        $Self->{LayoutObject}->Block( Name => 'HeaderEditUser' );
    }
    else {
        $Self->{LayoutObject}->Block(
            Name => 'HeaderAddUser',
            Data => \%Param,
        );
    }

    # if there are errors to show
    if ( $Param{Errors} && %{ $Param{Errors} } ) {

        # show all existing periods
        for ( my $Period = 1; $Period <= $Param{Periods}; $Period++ ) {

            for my $Parameter (qw(DateStart DateEnd LeaveDays WeeklyHours Overtime PeriodStatus )) {
                $GetParam{$Parameter} = $Self->{ParamObject}->GetParam( Param => "$Parameter\[$Period\]" );
            }

            $Param{$Period}{PeriodStatusOption} = $Self->{LayoutObject}->BuildSelection(
                Data       => \%StatusList,
                SelectedID => $GetParam{PeriodStatus} || $Param{$Period}{PeriodStatus},
                Name       => "PeriodStatus[$Period]",
                ID         => "PeriodStatus-$Period",
            );

            $Self->{LayoutObject}->Block(
                Name => 'PeriodOverviewRow',
                Data => {
                    Period           => $Period,
                    DateStartInvalid => $Param{Errors}->{ 'DateStart-' . $Period . 'Invalid' }
                        || '',
                    DateEndInvalid => $Param{Errors}->{ 'DateEnd-' . $Period . 'Invalid' } || '',
                    LeaveDaysInvalid => $Param{Errors}->{ 'LeaveDays-' . $Period . 'Invalid' }
                        || '',
                    DateStart          => $GetParam{DateStart},
                    DateEnd            => $GetParam{DateEnd},
                    LeaveDays          => $GetParam{LeaveDays},
                    WeeklyHours        => $GetParam{WeeklyHours},
                    Overtime           => $GetParam{Overtime},
                    PeriodStatusOption => $Param{$Period}{PeriodStatusOption},
                },
            );

            $Self->{LayoutObject}->Block(
                Name => 'DateStart'
                    . (
                    $Param{Errors}->{ 'DateStart-' . $Period . 'ErrorType' }
                        || 'MissingValue'
                    ),
                Data => { Period => $Period },
            );
            $Self->{LayoutObject}->Block(
                Name => 'DateEnd'
                    . ( $Param{Errors}->{ 'DateEnd-' . $Period . 'ErrorType' } || 'MissingValue' ),
                Data => { Period => $Period },
            );
        }
    }
    else {
        my %User = $Self->{TimeAccountingObject}->SingleUserSettingsGet( UserID => $Param{UserID} );

        # show user data
        if (%User) {
            my $LastPeriodNumber = $Self->{TimeAccountingObject}->UserLastPeriodNumberGet(
                UserID => $Param{UserID}
            );

            for ( my $Period = 1; $Period <= $LastPeriodNumber; $Period++ ) {
                my %PeriodParam = ();

                # get all needed data to display
                for my $Parameter (qw(DateStart DateEnd LeaveDays WeeklyHours Overtime)) {
                    $PeriodParam{$Parameter} = $User{$Period}{$Parameter};
                }
                $PeriodParam{Period} = $Period;

                $PeriodParam{PeriodStatusOption} = $Self->{LayoutObject}->BuildSelection(
                    Data       => \%StatusList,
                    SelectedID => $User{$Period}{UserStatus},
                    Name       => "PeriodStatus[$Period]",
                    ID         => "PeriodStatus-$Period",
                );

                $Self->{LayoutObject}->Block(
                    Name => 'PeriodOverviewRow',
                    Data => \%PeriodParam,
                );
            }
        }

        # show a no data found msg
        else {
            $Self->{LayoutObject}->Block( Name => 'PeriodOverviewRowNoData' );
        }
    }

    return 1;
}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE3IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCjxmaWVsZHNldCBjbGFzcz0iVGFibGVMaWtlIFczMDBweCI+CiAgICA8bGFiZWw+JFRleHR7IlllYXIifTo8L2xhYmVsPgogICAgPHAgY2xhc3M9IlZhbHVlIj4KICAgICAgICAkUURhdGF7IlllYXIifQogICAgPC9wPgogICAgPGRpdiBjbGFzcz0iQ2xlYXIiPjwvZGl2PgoKICAgIDxsYWJlbD4kVGV4dHsiTW9udGgifTo8L2xhYmVsPgogICAgPHAgY2xhc3M9IlZhbHVlIj4KICAgICAgICAkVGV4dHsiJERhdGF7Ik1vbnRoIn0ifQogICAgPC9wPgogICAgPGRpdiBjbGFzcz0iQ2xlYXIiPjwvZGl2PgoKICAgIDxsYWJlbD4kVGV4dHsiRGF5In06PC9sYWJlbD4KICAgIDxwIGNsYXNzPSJWYWx1ZSI+CiAgICAgICAgJFFEYXRheyJEYXkifQogICAgPC9wPgogICAgPGRpdiBjbGFzcz0iQ2xlYXIiPjwvZGl2Pgo8L2ZpZWxkc2V0Pgo8cCBjbGFzcz0iU3BhY2luZ1RvcCBDZW50ZXIiPgogICAgJFRleHR7IkRvIHlvdSByZWFsbHkgd2FudCB0byBkZWxldGUgdGhlIFRpbWUgQWNjb3VudGluZyBvZiB0aGlzIGRheT8ifQo8L3A+Cg==
# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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.
# --

<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1 class="Header">$Text{"Edit Time Record"}</h1>

    <div class="SidebarColumn">
<!-- dtl:block:OverviewProject -->
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Actions"}</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
                    <li>
                        <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Overview;Year=$LQData{"Year"};Month=$LQData{"Month"}" class="CallForAction"><span>$Text{"Go to overview"}</span></a>
                    </li>
<!-- dtl:block:CreateProject -->
                    <li>
                        <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Setting" class="CallForAction"><span>$Text{"Project settings"}</span></a>
                    </li>
<!-- dtl:block:CreateProject -->
                </ul>
            </div>
        </div>
<!-- dtl:block:OverviewProject -->
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Date Navigation"}</h2>
            </div>
            <div class="Content">
                <div class="MonthSelection">
                    <a class="Arrow Left" title="$Text{"Previous day"}" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Edit;Year=$LQData{"YearBack"};Month=$LQData{"MonthBack"};Day=$LQData{"DayBack"}"></a>
                    <a class="Arrow Right" title="$Text{"Next day"}" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Edit;Year=$LQData{"YearNext"};Month=$LQData{"MonthNext"};Day=$LQData{"DayNext"}"></a>
                    $QData{"Year"}-$QData{"Month"}-$QData{"Day"}
                </div>
                <div class="Clear"></div>
                <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" class="Validate" name="DateSelection">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="Edit" />
                    $Data{"Date"}
                    <input class="button" type="submit" value="$Text{"Submit"}" />
                </form>
<!-- dtl:block:IncompleteWorkingDays -->
                <h2 class="SpacingTop">$Text{"Days without entries"}:</h2>
                $Data{"IncompleteWorkingDaysSelect"}
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
    $('#IncompleteWorkingDaysList').unbind('change.IncompleteWorkingDays').bind('change.IncompleteWorkingDays', function () {
        var Date = $(this).val().split('-');
        if (Date[0] && Date[1] && Date[2]) {
            location.href = "$Env{"Baselink"}Action=$Env{"Action"};Subaction=Edit;Year=" + Date[0] + ";Month=" + Date[1] + ";Day=" + Date[2];
        }
    });
//]]></script>
<!--dtl:js_on_document_complete-->
<!-- dtl:block:IncompleteWorkingDays -->
<!-- dtl:block:IncompleteWorkingDaysMassEntry -->
                <h2 class="SpacingTop">$Text{"Days without entries"}:</h2>
                <ul class="IncompleteWorkingDays">
                    <li id="SelectAllDays"><input type="checkbox" id="IncompleteWorkingDay-All" name="IncompleteWorkingDay-All" /><label for="IncompleteWorkingDay-All"><em>$Text{"Select all days"}</em></label></li>
<!-- dtl:block:IncompleteWorkingDaysMassEntrySingleDay -->
                    <li><input type="checkbox" id="IncompleteWorkingDay-$QData{"Date"}" name="IncompleteWorkingDay-$QData{"Date"}" /><label for="IncompleteWorkingDay-$QData{"Date"}">$QData{"Date"}</label> - <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Edit;Year=$LQData{"Year"};Month=$LQData{"Month"};Day=$LQData{"Day"}">$Text{"Edit"}</a></li>
<!-- dtl:block:IncompleteWorkingDaysMassEntrySingleDay -->
                </ul>
                <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" id="MassEntry" name="MassEntry" class="SpacingTop">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="MassEntry" />
                    <input type="hidden" name="Dates" value="" />
                    <input type="hidden" name="LeaveDay" value="" />
                    <input type="hidden" name="Sick" value="" />
                    <input type="hidden" name="Overtime" value="" />
                    <input class="button" type="button" id="MassEntrySubmit" value="$Text{"Mass entry"}" />
                </form>
                <div id="MassEntryConfirmDialog" class="Hidden">
                    <h2>$Text{"Please choose the reason for your absence for the selected days"}.</h2>
                    <ul id="MassEntryConfirmRadio">
                        <li><input type="radio" name="LeaveDay" id="ConfirmLeaveDay" /><label for="ConfirmLeaveDay">$Text{"On vacation"}</label></li>
                        <li><input type="radio" name="Sick" id="ConfirmSick" /><label for="ConfirmSick">$Text{"On sick leave"}</label></li>
                        <li><input type="radio" name="Overtime" id="ConfirmOvertime" /><label for="ConfirmOvertime">$Text{"On overtime leave"}</label></li>
                    </ul>
                </div>
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
    TimeAccounting.Agent.EditTimeRecords.MassEntryInit({
        MsgChooseOneDay:    '$JSText{"Please choose at least one day!"}',
        MsgAbsenceReason:   '$JSText{"Please choose a reason for absence!"}',
        MassEntry:          '$JSText{"Mass Entry"}',
        Submit:             '$JSText{"Submit"}',
        Cancel:             '$JSText{"Cancel"}'
    });
//]]></script>
<!--dtl:js_on_document_complete-->
<!-- dtl:block:IncompleteWorkingDaysMassEntry -->
            </div>
        </div>
        <div class="WidgetSimple SpacingTop">
            <div class="Header">
                <h2>$Text{"Reference"}</h2>
            </div>
            <div class="Content">
                <ul>
                    <li>$Text{"Required fields are marked with a "*"."}</li>
                    <li>$Text{"You have to fill in start and end time or a time period."}</li>
                </ul>
            </div>
        </div>
    </div>
    <div class="ContentColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Date"}: $Text{"$Data{"Weekday_to_Text"}"} $QData{"Year"}-$QData{"Month"}-$QData{"Day"}</h2>
            </div>
            <div class="Content HorizontalScroll">
                <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="WorkingUnits" id="WorkingUnits" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="Edit" />
                    <input type="hidden" name="RecordsNumber" id="RecordsNumber" value="$QData{"RecordsNumber"}" />
                    <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"}" />
<!-- dtl:block:UnitBlock -->
                    <table class="DataTable NoHover HighCells" id="InsertWorkingHours">
                        <thead>
                            <tr>
                                <th><span class="Marker">*</span> $Text{"Project"}</th>
                                <th><span class="Marker">*</span> $Text{"Task"}</th>
                                <th>$Text{"Remark"}</th>
                                <th>$Text{"Start"} (HH:MM)</th>
                                <th>$Text{"End"} (HH:MM)</th>
                                <th><span class="Marker">$QData{"PeriodNote"}</span> $Text{"Period"}</th>
                            </tr>
                        </thead>
                        <tbody>
<!-- dtl:block:Unit -->
                            <tr class="WorkingHours">
                                <td>
                                    $Data{"ProjectOption"}
                                    <div id="ProjectID$QData{"ID"}Error" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="ProjectID$QData{"ID"}ServerError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="Combo_ProjectID$QData{"ID"}Error" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="Combo_ProjectID$QData{"ID"}ServerError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                </td>
                                <td>
                                    $Data{"ActionOption"}
                                    <div id="ActionID$QData{"ID"}Error" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="ActionID$QData{"ID"}ServerError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="Combo_ActionID$QData{"ID"}Error" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="Combo_ActionID$QData{"ID"}ServerError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                </td>
                                <td>
                                    <input type="text" name="Remark[$QData{"ID"}]" id="Remark$QData{"ID"}" class="Validate_TimeAccounting_Remark $QData{"RemarkInvalid"}" value="$QData{"Remark"}" size="35" />
                                    <div id="Remark$QData{"ID"}Error" class="TooltipErrorMessage"><p>$Text{"Please add a remark with more than 8 characters!."}</p></div>
                                    <div id="Remark$QData{"ID"}ServerError" class="TooltipErrorMessage">
                                        <p>$Text{"This field is required."}</p>
                                    </div>
                                </td>
                                <td>
                                    <input type="text" name="StartTime[$QData{"ID"}]" id="StartTime$QData{"ID"}" class="StartTime Validate_TimeAccounting_StartTime $QData{"StartTimeRequired"} $QData{"StartTimeInvalid"}" value="$QData{"StartTime"}" size="7" />
                                    <div id="StartTime$QData{"ID"}Error" class="TooltipErrorMessage">
                                        <p>$Text{"You have to fill in start and end time or a time period."}</p>
                                    </div>
                                    <div id="StartTime$QData{"ID"}ServerError" class="TooltipErrorMessage">
<!-- dtl:block:StartTimeGenericServerError -->
                                        <p>$Text{"This field is required."}</p>
<!-- dtl:block:StartTimeGenericServerError -->
<!-- dtl:block:StartTimeNegativeServerError -->
                                        <p>$Text{"Negative times are not allowed."}</p>
<!-- dtl:block:StartTimeNegativeServerError -->
<!-- dtl:block:StartTimeRepeatedHourServerError -->
                                        <p>$Text{"Repeated hours are not allowed. Start time matches another interval."}</p>
<!-- dtl:block:StartTimeRepeatedHourServerError -->
<!-- dtl:block:StartTimeInvalidFormatServerError -->
                                        <p>$Text{"Invalid format! Please enter a time with the format HH:MM."}</p>
<!-- dtl:block:StartTimeInvalidFormatServerError -->
<!-- dtl:block:StartTime24Hours -->
                                        <p>$Text{"24:00 is only permitted as end time."}</p>
<!-- dtl:block:StartTime24Hours -->
<!-- dtl:block:StartTimeInvalid -->
                                        <p>$Text{"Invalid time! A day has only 24 hours."}</p>
<!-- dtl:block:StartTimeInvalid -->
                                    </div>
                                </td>
                                <td>
                                    <input type="text" name="EndTime[$QData{"ID"}]" id="EndTime$QData{"ID"}" class="EndTime Validate_DependingRequiredAND Validate_Depending_StartTime$QData{"ID"} $QData{"EndTimeRequired"} $QData{"EndTimeInvalid"}" value="$QData{"EndTime"}" size="7" />
                                    <div id="EndTime$QData{"ID"}Error" class="TooltipErrorMessage">
                                        <p>$Text{"You have to fill in start and end time or a time period."}</p>
                                    </div>
                                    <div id="EndTime$QData{"ID"}ServerError" class="TooltipErrorMessage">
<!-- dtl:block:EndTimeGenericServerError -->
                                        <p>$Text{"This field is required."}</p>
<!-- dtl:block:EndTimeGenericServerError -->
<!-- dtl:block:EndTimeBeforeStartTimeServerError -->
                                        <p>$Text{"End time must be after start time."}</p>
<!-- dtl:block:EndTimeBeforeStartTimeServerError -->
<!-- dtl:block:EndTimeNegativeServerError -->
                                        <p>$Text{"Negative times are not allowed."}</p>
<!-- dtl:block:EndTimeNegativeServerError -->
<!-- dtl:block:EndTimeRepeatedHourServerError -->
                                        <p>$Text{"Repeated hours are not allowed. End time matches another interval."}</p>
<!-- dtl:block:EndTimeRepeatedHourServerError -->
<!-- dtl:block:EndTimeInvalidFormatServerError -->
                                        <p>$Text{"Invalid format! Please enter a time with the format HH:MM."}</p>
<!-- dtl:block:EndTimeInvalidFormatServerError -->
<!-- dtl:block:EndTimeInvalid -->
                                        <p>$Text{"Invalid time! A day has only 24 hours."}</p>
<!-- dtl:block:EndTimeInvalid -->
                                    </div>
                                </td>
<!-- dtl:block:UnitInputPeriod -->
                                <td>
                                    <input type="text" name="Period[$QData{"ID"}]" id="Period$QData{"ID"}" class="Period Validate_TimeAccounting_Period Validate_TimeUnits $QData{"PeriodRequired"} $QData{"PeriodInvalid"}" value="$QData{"Period"}" size="7" />
                                    <div id="Period$QData{"ID"}Error" class="TooltipErrorMessage">
                                        <p>$Text{"You have to fill in start and end time or a time period."}</p>
                                    </div>
                                    <div id="Period$QData{"ID"}ServerError" class="TooltipErrorMessage">
<!-- dtl:block:PeriodGenericServerError -->
                                        <p>$Text{"This field is required."}</p>
<!-- dtl:block:PeriodGenericServerError -->
<!-- dtl:block:InvalidHoursPeriodServerError -->
                                        <p>$Text{"Invalid period! A day has only 24 hours."}</p>
<!-- dtl:block:InvalidHoursPeriodServerError -->
<!-- dtl:block:ZeroHoursPeriodServerError -->
                                        <p>$Text{"A valid period must be greater than zero."}</p>
<!-- dtl:block:ZeroHoursPeriodServerError -->
<!-- dtl:block:NegativePeriodServerError -->
                                        <p>$Text{"Invalid period! Negative periods are not allowed."}</p>
<!-- dtl:block:NegativePeriodServerError -->
                                    </div>

                                </td>
<!-- dtl:block:UnitInputPeriod -->
<!-- dtl:block:UnitPeriodWithoutInput -->
                                <td>$QData{"Period"}</td>
<!-- dtl:block:UnitPeriodWithoutInput -->
                            </tr>
<!-- dtl:block:Unit -->
                            <tr>
                                <td colspan="6" class="Right">
                                    <button class="CallForAction Plus" type="button" id="MoreInputFields" name="MoreInputFields"><span>$Text{"Add one row"}</span></button>
                                </td>
                            </tr>
<!-- dtl:block:UnitBlock -->
                            <tr>
                                <td colspan="4"></td>
<!-- dtl:block:Total -->
                                <td>$Text{"Total"}:</td>
                                <td><span class="TotalHours">$QData{"Total"}</span></td>
<!-- dtl:block:Total -->
                            </tr>
<!-- dtl:block:Readonly -->
                            <tr>
                                <td colspan="6">
                                    $Text{"$Data{"Description"}"}
                                </td>
                            </tr>
<!-- dtl:block:Readonly -->
                        </tbody>
                    </table>
<!-- dtl:block:OtherTimes -->
                    <div class="SpacingTop">
                        <input type="checkbox" value="1" id="LeaveDay" name="LeaveDay" title="$Text{"On vacation"}" class="$Data{"LeaveDayInvalid"}" $Data{"LeaveDay"} />
                        <label for="LeaveDay">$Text{"On vacation"}</label>
                        <div id="LeaveDayServerError" class="TooltipErrorMessage">
                            <p>$Text{"You can only select one checkbox element!"}</p>
                        </div>
                        <input type="checkbox" value="1" id="Sick" name="Sick" title="$Text{"On sick leave"}" class="$Data{"SickInvalid"}" $Data{"Sick"} />
                        <label for="Sick">$Text{"On sick leave"}</label>
                        <div id="SickServerError" class="TooltipErrorMessage">
                            <p>$Text{"You can only select one checkbox element!"}</p>
                        </div>
                        <input type="checkbox" value="1" id="Overtime" name="Overtime" title="$Text{"On overtime leave"}" class="$Data{"OvertimeInvalid"}" $Data{"Overtime"} />
                        <label for="Overtime">$Text{"On overtime leave"}</label>
                        <div id="OvertimeServerError" class="TooltipErrorMessage">
                            <p>$Text{"You can only select one checkbox element!"}</p>
                        </div>
                    </div>
<!-- dtl:block:OtherTimes -->
                    <div class="Center SpacingTop">
                        <button class="Primary" type="submit" name="InsertWorkingUnits" value="$Text{"Submit"}">$Text{"Submit"}</button>
                        <a id="DeleteTimeAccountingEntry" href="#">
                            $Text{"Delete"}
                        </a>
                        $Text{"or"}
                        <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Edit">$Text{"Cancel"}</a>
                    </div>
                    <div class="Clear"></div>
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
    TimeAccounting.Agent.EditTimeRecords.ActionList = $QData{"JSActionList"};
    TimeAccounting.Agent.EditTimeRecords.ActionListConstraints = $QData{"JSActionListConstraints"};
    TimeAccounting.Agent.EditTimeRecords.Init({
        RemarkRegExpContent: "$QData{"RemarkRegExp"}",
        Autocompletion: $QData{"EnableAutocompletion"},
        Language: {
            ShowAllItems: "$JSText{"Show all items"}"
        }
    });
    TimeAccounting.Agent.ConfirmationDialog.BindConfirmationDialog({
        ElementID:                  'DeleteTimeAccountingEntry',
        ElementSelector:            '#DeleteTimeAccountingEntry',
        DialogContentQueryString:   'Action=AgentTimeAccounting;DeleteDay=1;Year=$QData{"Year"};Month=$QData{"Month"};Day=$QData{"Day"}',
        ConfirmedActionQueryString: 'Action=AgentTimeAccounting;Subaction=Delete;Year=$QData{"Year"};Month=$QData{"Month"};Day=$QData{"Day"}',
        DialogTitle:                '$JSText{"Delete Time Accounting Entry"}',
        TranslatedText:             {
            Yes: '$JSText{"Yes"}',
            No:  '$JSText{"No"}'
        }
    });
//]]></script>
<!--dtl:js_on_document_complete-->

<!-- dtl:block:ShowConfirmation -->
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
    // define yes and no buttons
    var Buttons = [{
        Label: '$JSText{"Yes"}',
        Class: "Primary",
        Type: "Close",
        }, {
        Label: '$JSText{"No"}',

        // define the function that is called when the 'No' button is pressed
        Function: function(){

            // disable Yes and No buttons to prevent multiple submits
            $('div.Dialog:visible div.ContentFooter button').attr('disabled', 'disabled');

            // redirect to the module that deletes the saved entry after pressing the No button
            location.href = Core.Config.Get('Baselink') + 'Action=AgentTimeAccounting;Subaction=Delete;Year=$QData{"Year"};Month=$QData{"Month"};Day=$QData{"Day"}';
        }
    }];
    // show the confirmation dialog to confirm the action
    Core.UI.Dialog.ShowContentDialog($('#$QData{"BlockName"}'), '$JSText{"Confirm insert"}', '20%', 'Center', true, Buttons);

//]]></script>
<!--dtl:js_on_document_complete-->
<!-- dtl:block:ShowConfirmation -->

                </form>
            </div>
        </div>
    </div>
</div>

<p id="SickLeaveMessage" class="Hidden Center">
    $Text{"Are you sure that you worked while you were on sick leave?"}
</p>

<p id="VacationMessage" class="Hidden Center">
    $Text{"Are you sure that you worked while you were on vacation?"}
</p>

<p id="OvertimeMessage" class="Hidden Center">
    $Text{"Are you sure that you worked while you were on overtime leave?"}
</p>

<p id="More16HoursMessage" class="Hidden Center">
    $Text{"Are you sure that you worked more than 16 hours?"}
</p>

# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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.
# --

<div class="MainBox ARIARoleMain">
    <h1 class="Header">$Text{"Time reporting monthly overview"}</h1>

    <div class="LayoutFixedSidebar SidebarLast">
        <div class="SidebarColumn">
<!-- dtl:block:Overtime -->
            <div class="WidgetSimple">
                <div class="Header">
                    <h2>$Text{"Overtime (Hours)"}</h2>
                </div>
                <div class="Content">
                    <fieldset class="TableLike FixedLabel">
                        <label>
                            $Text{"Overtime (this month)"}:
                        </label>
                        <div class="Right">
                            <p>$QData{"Overtime"}</p>
                        </div>
                        <div class="Clear"></div>

                        <label>
                            $Text{"Overtime (total)"}:
                        </label>
                        <div class="Right">
                            <p>$QData{"OvertimeUntil"}</p>
                        </div>
                        <div class="Clear"></div>

                        <label>
                            $Text{"Remaining overtime leave"}:
                        </label>
                        <div class="Right">
                            <p>$QData{"OvertimeTotal"}</p>
                        </div>
                        <div class="Clear"></div>
                    </fieldset>
                </div>
            </div>
<!-- dtl:block:Overtime -->

            <div class="WidgetSimple">
                <div class="Header">
                    <h2>$Text{"Vacation (Days)"}</h2>
                </div>
                <div class="Content">
                    <fieldset class="TableLike FixedLabel">
                        <label>
                            $Text{"Vacation taken (this month)"}:
                        </label>
                        <div class="Right">
                            <p>$QData{"LeaveDay"}</p>
                        </div>
                        <div class="Clear"></div>

                        <label>
                            $Text{"Vacation taken (total)"}:
                        </label>
                        <div class="Right">
                            <p>$QData{"LeaveDayTotal"}</p>
                        </div>
                        <div class="Clear"></div>

                        <label>
                            $Text{"Remaining vacation"}:
                        </label>
                        <div class="Right">
                            <p>$QData{"LeaveDayRemaining"}</p>
                        </div>
                        <div class="Clear"></div>
                    </fieldset>
                </div>
            </div>

            <div class="WidgetSimple">
                <div class="Header">
                    <h2>$Text{"Sick Leave (Days)"}</h2>
                </div>
                <div class="Content">
                    <fieldset class="TableLike FixedLabel">
                        <label>
                            $Text{"Sick leave taken (this month)"}:
                        </label>
                        <div class="Right">
                            <p>$QData{"Sick"}</p>
                        </div>
                        <div class="Clear"></div>

                        <label>
                            $Text{"Sick leave taken (total)"}:
                        </label>
                        <div class="Right">
                            <p>$QData{"SickTotal"}</p>
                        </div>
                        <div class="Clear"></div>
                    </fieldset>
                </div>
            </div>
        </div>

        <div class="ContentColumn">
            <div class="WidgetSimple">
                <div class="Header">
                    <h2>$Text{"Overview"}
<!-- dtl:block:User -->
                        - $QData{"User"}
<!-- dtl:block:User -->
                    </h2>
                    <div class="AdditionalInformation">
                        <a class="Arrow Left" title="$Text{"Previous month"}" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Overview;Year=$LQData{"YearBack"};Month=$LQData{"MonthBack"};UserID=$LQData{"UserID"}"></a>
                        <a class="Arrow Right" title="$Text{"Next month"}" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Overview;Year=$LQData{"YearNext"};Month=$LQData{"MonthNext"};UserID=$LQData{"UserID"}"></a>
                        $Text{"$Data{"Month_to_Text"}"} $QData{"Year"}
                    </div>
                </div>
                <div class="Content">
                    <table class="DataTable">
                        <thead>
                            <tr>
                                <th>$Text{"Day"}</th>
                                <th>$Text{"Weekday"}</th>
                                <th>$Text{"Comment"}</th>
                                <th>$Text{"Working Hours"}</th>
                            </tr>
                        </thead>
                        <tbody>
<!-- dtl:block:Row -->
                            <tr class="$QData{"Class"} MasterAction">
                                <td class="Fixed Day">
                                    <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=$LQData{"Subaction"};Year=$LQData{"Year"};Month=$LQData{"Month"};Day=$LQData{"Day"};UserID=$LQData{"UserID"}" class="MasterActionLink">
                                        $QData{"Day"}
                                    </a>
                                </td>
                                <td class="Fixed Weekday">$Text{"$Data{"Weekday_to_Text"}"}</td>
                                <td>$Text{"$Data{"Comment"}"}</td>
                                <td class="Right">$QData{"WorkingHours"}</td>
                            </tr>
<!-- dtl:block:Row -->
                        </tbody>
                    </table>
                    <div class="SpacingTop Right"><strong>$Text{"Total worked hours"}:</strong> $QData{"WorkingHours"}</div>
                </div>
            </div>

            <div class="WidgetSimple">
                <div class="Header">
                    <h2>
                        $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{"$Data{"ShowProjects"}"})
                        </a>
                    </h2>
                </div>
                <div class="Content">
                    <table class="DataTable NoHover HighCells">
                        <thead>
                            <tr>
                                <th>$Text{"Project"}</th>
                                <th>$Text{"Task"}</th>
                                <th>$Text{"Hours (monthly)"}</th>
                                <th>$Text{"Hours (Lifetime)"}</th>
                            </tr>
                        </thead>
                        <tbody>
<!-- dtl:block:Action -->
                            <tr>
<!-- dtl:block:Project -->
                                <td rowspan="$Data{"RowSpan"}" class="RowSpan">
                                    <strong>
<!-- dtl:block:ProjectNoLink -->
                                    $Data{"Project"}
<!-- dtl:block:ProjectNoLink -->
<!-- dtl:block:ProjectLink -->
                                    <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=ProjectReporting;ProjectID=$LQData{"ProjectID"}">
                                        $Data{"Project"}
                                    </a>
<!-- dtl:block:ProjectLink -->
                                    </strong>
<!-- dtl:block:ProjectDescription -->
                                    <br/>$Data{"ProjectDescription"}
<!-- dtl:block:ProjectDescription -->
                                </td>
<!-- dtl:block:Project -->
                                <td>$QData{"Action"}</td>
                                <td class="Right">$QData{"Hours"}</td>
                                <td class="Right">$QData{"HoursTotal"}</td>
                            </tr>
<!-- dtl:block:ActionTotal -->
                            <tr>
                                <td><strong>$Text{"Total"}</strong></td>
                                <td class="Right"><strong>$QData{"Hours"}</strong></td>
                                <td class="Right"><strong>$QData{"HoursTotal"}</strong></td>
                            </tr>
<!-- dtl:block:ActionTotal -->
<!-- dtl:block:Action -->
                            <tr class="GrandTotal">
                                <td colspan="2"><strong>$Text{"Grand total"}</strong></td>
                                <td class="Right"><strong>$QData{"TotalHours"}</strong></td>
                                <td class="Right"><strong>$QData{"TotalHoursTotal"}</strong></td>
                            </tr>
                        </tbody>
                    </table>
                </div>
            </div>
        </div>
    </div>
</div>
<!-- dtl:js_on_document_complete -->
<script type="text/javascript">//<![CDATA[
$('.MasterAction').bind('click', function (Event) {
    var $MasterActionLink = $(this).find('.MasterActionLink');
    // only act if the link was not clicked directly
    if (Event.target !== $MasterActionLink.get(0)) {
        window.location = $MasterActionLink.attr('href');
        return false;
    }
});
//]]></script>
<!-- dtl:js_on_document_complete -->

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE3IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCjxkaXYgY2xhc3M9Ik1haW5Cb3ggQVJJQVJvbGVNYWluIExheW91dEZpeGVkU2lkZWJhciBTaWRlYmFyRmlyc3QiPgogICAgPGgxIGNsYXNzPSJIZWFkZXIiPiRUZXh0eyJQcm9qZWN0IHJlcG9ydCJ9OiAkUURhdGF7IlByb2plY3QifTwvaDE+CgogICAgPGRpdiBjbGFzcz0iU2lkZWJhckNvbHVtbiI+CiAgICAgICAgPGRpdiBjbGFzcz0iV2lkZ2V0U2ltcGxlIj4KICAgICAgICAgICAgPGRpdiBjbGFzcz0iSGVhZGVyIj4KICAgICAgICAgICAgICAgIDxoMj4kVGV4dHsiQWN0aW9ucyJ9PC9oMj4KICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgICAgIDxkaXYgY2xhc3M9IkNvbnRlbnQiPgogICAgICAgICAgICAgICAgPHVsIGNsYXNzPSJBY3Rpb25MaXN0Ij4KICAgICAgICAgICAgICAgICAgICA8bGk+CiAgICAgICAgICAgICAgICAgICAgICAgIDxhIGhyZWY9IiRFbnZ7IkJhc2VsaW5rIn1BY3Rpb249JEVudnsiQWN0aW9uIn07U3ViYWN0aW9uPVJlcG9ydGluZyIgY2xhc3M9IkNhbGxGb3JBY3Rpb24iPjxzcGFuPiRUZXh0eyJHbyB0byBvdmVydmlldyJ9PC9zcGFuPjwvYT4KICAgICAgICAgICAgICAgICAgICA8L2xpPgogICAgICAgICAgICAgICAgPC91bD4KICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgPC9kaXY+CiAgICA8L2Rpdj4KCiAgICA8ZGl2IGNsYXNzPSJDb250ZW50Q29sdW1uIj4KICAgICAgICA8ZGl2IGNsYXNzPSJXaWRnZXRTaW1wbGUiPgogICAgICAgICAgICA8ZGl2IGNsYXNzPSJIZWFkZXIiPgogICAgICAgICAgICAgICAgPGgyPiRUZXh0eyJQcm9qZWN0IHJlcG9ydCJ9OiAkUURhdGF7IlByb2plY3QifTwvaDI+CiAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgICA8ZGl2IGNsYXNzPSJDb250ZW50Ij4KCiAgICAgICAgICAgICAgICA8dGFibGUgY2xhc3M9IkRhdGFUYWJsZSBEYXRhVGFibGVOb0hpZ2hsaWdodCI+CiAgICAgICAgICAgICAgICAgICAgPHRoZWFkPgogICAgICAgICAgICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGg+JFRleHR7IlRhc2sifTwvdGg+CjwhLS0gZHRsOmJsb2NrOlVzZXJOYW1lIC0tPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoPiRRRGF0YXsiVXNlciJ9PC90aD4KPCEtLSBkdGw6YmxvY2s6VXNlck5hbWUgLS0+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGg+JFRleHR7IlRvdGFsIn08L3RoPgogICAgICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICAgIDwvdGhlYWQ+CiAgICAgICAgICAgICAgICAgICAgPHRib2R5PgogICAgICAgICAgICAgICAgICAgICAgICA8dHI+CjwhLS0gZHRsOmJsb2NrOkFjdGlvbiAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kUURhdGF7IkFjdGlvbiJ9PC90ZD4KPCEtLSBkdGw6YmxvY2s6VXNlciAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iUmlnaHQiPiRRRGF0YXsiSG91cnMifTwvdGQ+CjwhLS0gZHRsOmJsb2NrOlVzZXIgLS0+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOkFjdGlvbiAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJHcmFuZFRvdGFsIj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZD48c3Ryb25nPiRUZXh0eyJUb3RhbCJ9PC9zdHJvbmc+PC90ZD4KPCEtLSBkdGw6YmxvY2s6VXNlclRvdGFsIC0tPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJSaWdodCI+PHN0cm9uZz4kUURhdGF7IlRvdGFsIn08L3N0cm9uZz48L3RkPgo8IS0tIGR0bDpibG9jazpVc2VyVG90YWwgLS0+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9IlJpZ2h0Ij48c3Ryb25nPiRRRGF0YXsiVG90YWxBbGwifTwvc3Ryb25nPjwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgICAgICAgPC90Ym9keT4KICAgICAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICAgIDwvZGl2PgogICAgICAgIDwvZGl2PgoKICAgICAgICA8ZGl2IGNsYXNzPSJXaWRnZXRTaW1wbGUiPgogICAgICAgICAgICA8ZGl2IGNsYXNzPSJIZWFkZXIiPgogICAgICAgICAgICAgICAgPGgyPiRUZXh0eyJIaXN0b3J5In06ICRRRGF0YXsiUHJvamVjdCJ9PC9oMj4KICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgICAgIDxkaXYgY2xhc3M9IkNvbnRlbnQiPgoKICAgICAgICAgICAgICAgIDx0YWJsZSBjbGFzcz0iRGF0YVRhYmxlIERhdGFUYWJsZU5vSGlnaGxpZ2h0Ij4KICAgICAgICAgICAgICAgICAgICA8dGhlYWQ+CiAgICAgICAgICAgICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0aD4kVGV4dHsiRGF0ZSJ9PC90aD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0aD4kVGV4dHsiVXNlciJ9PC90aD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0aD4kVGV4dHsiVGFzayJ9PC90aD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0aD4kVGV4dHsiUmVtYXJrIn08L3RoPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoPiRUZXh0eyJQZXJpb2QifTwvdGg+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgICAgICAgPC90aGVhZD4KICAgICAgICAgICAgICAgICAgICA8dGJvZHk+CjwhLS0gZHRsOmJsb2NrOlJvdyAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkPiRRRGF0YXsiRGF0ZSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kUURhdGF7IlVzZXIifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQ+JFFEYXRheyJBY3Rpb24ifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQ+JFFEYXRheyJSZW1hcmsifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9IlJpZ2h0Ij4kUURhdGF7IlBlcmlvZCJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6Um93IC0tPgogICAgICAgICAgICAgICAgICAgICAgICA8dHIgY2xhc3M9IkdyYW5kVG90YWwiPgo8IS0tIGR0bDpibG9jazpIaXN0b3J5VG90YWwgLS0+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iNCI+PHN0cm9uZz4kVGV4dHsiVG90YWwifTwvc3Ryb25nPjwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9IlJpZ2h0Ij48c3Ryb25nPiRRRGF0YXsiSGlzdG9yeVRvdGFsIn08L3N0cm9uZz48L3RkPgo8IS0tIGR0bDpibG9jazpIaXN0b3J5VG90YWwgLS0+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgICAgICAgPC90Ym9keT4KICAgICAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICAgIDwvZGl2PgogICAgICAgIDwvZGl2PgogICAgPC9kaXY+CjwvZGl2Pgo=
# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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.
# --

<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1 class="Header">$Text{"Time reporting"}: $Text{"$Data{"Month_to_Text"}"} $Data{"Year"}</h1>

    <div class="SidebarColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Month Navigation"}</h2>
            </div>
            <div class="Content">
                <div class="MonthSelection">
                    <a class="Arrow Left" title="$Text{"Previous month"}" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Reporting;Year=$LQData{"YearBack"};Month=$LQData{"MonthBack"}"></a>
                    <a class="Arrow Right" title="$Text{"Next month"}" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Reporting;Year=$LQData{"YearNext"};Month=$LQData{"MonthNext"}"></a>
                    <strong>$Text{"$Data{"Month_to_Text"}"} $Data{"Year"}</strong>
                </div>
                <div class="Clear"></div>
                <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>
            </div>
        </div>
    </div>

    <div class="ContentColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"User reports"}</h2>
            </div>
            <div class="Content HorizontalScroll">
                <table class="DataTable">
                    <thead class="TwoRows">
                        <tr>
                            <th></th>
                            <th colspan="4">$Text{"Monthly total"}</th>
                            <th colspan="2">$Text{"Lifetime total"}</th>
                        </tr>
                        <tr>
                            <th>$Text{"User"}</th>
                            <th>$Text{"Working Hours"}</th>
                            <th>$Text{"Overtime leave"}</th>
                            <th>$Text{"Vacation"}</th>
                            <th>$Text{"Sick leave"}</th>
                            <th>$Text{"Overtime leave"}</th>
                            <th>$Text{"Vacation remaining"}</th>
                        </tr>
                    </thead>
                    <tbody>
<!-- dtl:block:User -->
                        <tr class="MasterAction">
                            <td>
                                <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Overview;Year=$LQData{"Year"};Month=$LQData{"Month"};UserID=$LQData{"UserID"}" class="MasterActionLink">
                                    $QData{"User"}
                                </a>
                            </td>
                            <td class="Right">$QData{"WorkingHours"}</td>
                            <td class="Right">$QData{"Overtime"}</td>
                            <td class="Right">$QData{"LeaveDay"}</td>
                            <td class="Right">$QData{"Sick"}</td>
                            <td class="Right">$QData{"OvertimeTotal"}</td>
                            <td class="Right">$QData{"LeaveDayRemaining"}</td>
                        </tr>
<!-- dtl:block:User -->
                        <tr class="GrandTotal">
                            <td><strong>$Text{"Total"}</strong></td>
                            <td class="Right"><strong>$QData{"TotalWorkingHours"}</strong></td>
                            <td class="Right"><strong>$QData{"TotalOvertime"}</strong></td>
                            <td class="Right"><strong>$QData{"TotalLeaveDay"}</strong></td>
                            <td class="Right"><strong>$QData{"TotalSick"}</strong></td>
                            <td class="Right"><strong>$QData{"TotalOvertimeTotal"}</strong></td>
                            <td class="Right"><strong>$QData{"TotalLeaveDayRemaining"}</strong></td>
                        </tr>
                    </tbody>
                </table>
            </div>
        </div>

        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Project reports"}
                    <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Reporting;Year=$LQData{"Year"};Month=$LQData{"Month"};ProjectStatusShow=$LQData{"ProjectStatusShow"}">
                        ($Text{"$Data{"ShowProjects"}"})
                    </a>
                </h2>
            </div>
            <div class="Content HorizontalScroll">
                <table class="DataTable NoHover">
                    <thead>
                        <tr>
                            <th>$Text{"Project"}</th>
                            <th>$Text{"Task"}</th>
                            <th>$Text{"Hours (monthly)"}</th>
                            <th>$Text{"Hours (Lifetime)"}</th>
                        </tr>
                    </thead>
                    <tbody>
<!-- dtl:block:Action -->
                        <tr class="MasterAction">
<!-- dtl:block:Project -->
                            <td rowspan="$Data{"RowSpan"}" class="RowSpan">
                                <strong><a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=ProjectReporting;ProjectID=$LQData{"ProjectID"}" class="MasterActionLink">
                                $Data{"Project"}
                            </a></strong>
<!-- dtl:block:ProjectDescription -->
                            <br />$Data{"ProjectDescription"}
<!-- dtl:block:ProjectDescription -->
                        </td>
<!-- dtl:block:Project -->
                        <td>$QData{"Action"}</td>
                        <td class="Right">$QData{"Hours"}</td>
                        <td class="Right">$QData{"HoursTotal"}</td>
                    </tr>
<!-- dtl:block:ActionTotal -->
                    <tr>
                        <td><strong>$Text{"Total"}</strong></td>
                        <td class="Right"><strong>$QData{"Hours"}</strong></td>
                        <td class="Right"><strong>$QData{"HoursTotal"}</strong></td>
                    </tr>
<!-- dtl:block:ActionTotal -->
<!-- dtl:block:Action -->
                    <tr class="GrandTotal">
                        <td colspan="2"><strong>$Text{"Grand total"}</strong></td>
                        <td class="Right"><strong>$QData{"TotalHours"}</strong></td>
                        <td class="Right"><strong>$QData{"TotalHoursTotal"}</strong></td>
                    </tr>
                </tbody>
            </table>
        </div>
    </div>
</div>
</div>
<!-- dtl:js_on_document_complete -->
<script type="text/javascript">//<![CDATA[
$('.MasterAction').bind('click', function (Event) {
    var $MasterActionLink = $(this).find('.MasterActionLink');
    // only act if the link was not clicked directly
    if (Event.target !== $MasterActionLink.get(0)) {
        window.location = $MasterActionLink.attr('href');
        return false;
    }
});
//]]></script>
<!-- dtl:js_on_document_complete -->

# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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:OverviewProject -->
<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1 class="Header">$Text{"Edit Time Accounting Project Settings"}</h1>

    <div class="SidebarColumn">
<!-- dtl:block:ActionListProject -->
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Actions"}</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
<!--dtl:block:ActionAddProject-->
                    <li>
                        <form action="$Env{"CGIHandle"}" method="post">
                            <input type="hidden" name="Action" value="$Env{"Action"}"/>
                            <input type="hidden" name="Subaction" value="AddProject"/>
                            <button class="CallForAction Plus" type="submit" value="$Text{"Add"}">
                                <span>$Text{"Add project"}</span>
                            </button>
                        </form>
                    </li>
<!--dtl:block:ActionAddProject-->

<!-- dtl:block:ActionSettingOverview -->
                    <li>
                        <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Setting" class="CallForAction"><span>$Text{"Go to overview"}</span></a>
                    </li>
<!-- dtl:block:ActionSettingOverview-->
                </ul>
            </div>
        </div>
<!-- dtl:block:ActionListProject -->
    </div>

    <div class="ContentColumn">

<!-- dtl:block:OverviewUpdateProject -->
        <div class="WidgetSimple">
            <div class="Header">
<!-- dtl:block:HeaderAddProject -->
                <h2>$Text{"Add Project"}</h2>
<!-- dtl:block:HeaderAddProject -->
<!-- dtl:block:HeaderEditProject -->
                <h2>$Text{"Edit Project Settings"}</h2>
<!-- dtl:block:HeaderEditProject -->
            </div>
            <div class="Content">
                <form action="$Env{"CGIHandle"}" method="post" name="ProjectUpdate" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}">
                    <input type="hidden" name="Subaction" value="$QData{"Action"}Action">
                    <input type="hidden" name="ID" value="$QData{"ID"}"/>
                    <fieldset class="TableLike">

                        <label class="Mandatory" for="Project"><span class="Marker">*</span> $Text{"Name"}:</label>
                        <div class="Field">
                            <input type="text" name="Project" id="Project" value="$QData{"Project"}" class="W50pc Validate_Required $QData{"ProjectInvalid"}" maxlength="50"/>
                            <div id="ProjectError" class="TooltipErrorMessage">
                                <p>$Text{"This field is required."}</p>
                            </div>

<!-- dtl:block:ProjectMissingValue -->
                            <div id="ProjectServerError" class="TooltipErrorMessage">
                                <p>$Text{"This field is required."}</p>
                            </div>
<!-- dtl:block:ProjectMissingValue -->

<!-- dtl:block:ProjectDuplicateName -->
                            <div id="ProjectServerError" class="TooltipErrorMessage">
                                <p>$Text{"There is already a project with this name. Please, choose a different one."}</p>
                            </div>
<!-- dtl:block:ProjectDuplicateName -->

                        </div>
                        <div class="Clear"></div>

                        <label for="ProjectDescription">$Text{"Description"}:</label>
                        <div class="Field">
                            <input type="text" name="ProjectDescription" id="ProjectDescription" value="$QData{"ProjectDescription"}" class="W50pc" maxlength="150"/>
                        </div>
                        <div class="Clear"></div>

                        <label for="ProjectStatus">$Text{"Status"}:</label>
                        <div class="Field">
                            $Data{"StatusOption"}
                        </div>
                        <div class="Clear"></div>

                        <div class="Field SpacingTop">
                            <button class="Primary" type="submit" id="Submit" value="$Text{"Submit"}">$Text{"Submit"}</button>
                            $Text{"or"}
                            <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Setting">$Text{"Cancel"}</a>
                        </div>
                        <div class="Clear"></div>
                    </fieldset>
                </form>
            </div>
        </div>
<!-- dtl:block:OverviewUpdateProject -->
    </div>
</div>
<!-- dtl:block:OverviewProject -->

<!-- dtl:block:Setting -->
<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1 class="Header">$Text{"Edit Time Accounting Settings"}</h1>

    <div class="SidebarColumn">

<!-- dtl:block:ActionListSetting -->
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Actions"}</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
<!--dtl:block:ActionAddProject-->
                    <li>
                        <form action="$Env{"CGIHandle"}" method="post">
                            <input type="hidden" name="Action" value="$Env{"Action"}"/>
                            <input type="hidden" name="Subaction" value="AddProject"/>
                            <button class="CallForAction Plus" type="submit" value="$Text{"Add"}">
                                <span>$Text{"Add project"}</span>
                            </button>
                        </form>
                    </li>
<!--dtl:block:ActionAddProject-->
<!--dtl:block:ActionAddTask-->
                    <li>
                        <form action="$Env{"CGIHandle"}" method="post">
                            <input type="hidden" name="Action" value="$Env{"Action"}"/>
                            <input type="hidden" name="Subaction" value="AddTask"/>
                            <button class="CallForAction Plus" type="submit" value="$Text{"Add task"}">
                                <span>$Text{"Add task"}</span>
                            </button>
                        </form>
                    </li>
<!--dtl:block:ActionAddTask-->
<!--dtl:block:ActionAddUser-->
                    <li>
                        <form action="$Env{"CGIHandle"}" method="post">
                            <input type="hidden" name="Action" value="$Env{"Action"}"/>
                            <input type="hidden" name="Subaction" value="AddUser"/>
                            $Data{"NewUserOption"}
                            <button class="CallForAction" type="submit" value="$Text{"New user"}">
                                <span>$Text{"New user"}</span>
                            </button>
                        </form>
                    </li>
<!--dtl:block:ActionAddUser-->
<!-- dtl:block:ActionOverviewSetting -->
                    <li>
                        <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Setting" class="CallForAction"><span>$Text{"Go to overview"}</span></a>
                    </li>
<!-- dtl:block:ActionOverviewSetting -->

                </ul>
            </div>
        </div>
<!-- dtl:block:ActionListSetting -->

<!-- dtl:block:ProjectFilter -->
        <div class="WidgetSimple">
            <div class="Header">
                <h2><label for="FilterProjects">$Text{"Filter for Projects"}</label></h2>
            </div>
            <div class="Content">
                <input type="text" id="FilterProjects" class="W50pc" name="FilterProjects" value="" title="$Text{"Filter for Projects"}" />
            </div>
        </div>
<!-- dtl:block:ProjectFilter -->

<!-- dtl:block:TaskFilter -->
        <div class="WidgetSimple">
            <div class="Header">
                <h2><label for="FilterTasks">$Text{"Filter for Tasks"}</label></h2>
            </div>
            <div class="Content">
                <input type="text" id="FilterTasks" class="W50pc" name="FilterTasks" value="" title="$Text{"Filter for Tasks"}" />
            </div>
        </div>
<!-- dtl:block:TaskFilter -->

<!-- dtl:block:UserFilter -->
        <div class="WidgetSimple">
            <div class="Header">
                <h2><label for="FilterUsers">$Text{"Filter for Users"}</label></h2>
            </div>
            <div class="Content">
                <input type="text" id="FilterUsers" class="W50pc" name="FilterUsers" value="" title="$Text{"Filter for Users"}" />
            </div>
        </div>
<!-- dtl:block:UserFilter -->

<!-- dtl:block:Reference -->
        <div class="WidgetSimple SpacingTop">
            <div class="Header">
                <h2>$Text{"Reference"}</h2>
            </div>
            <div class="Content">
                <ul>
                    <li>$Text{"Required fields are marked with a "*"."}</li>
                </ul>
            </div>
        </div>
<!-- dtl:block:Reference -->

    </div>

    <div class="ContentColumn">
<!-- dtl:block:OverviewResultProject -->
        <div class="WidgetSimple">
            <div class="Header">
                <div class="WidgetAction Toggle"><a href="#" title="$Text{"Show or hide the content"}"><span></span></a></div>
                <h2>$Text{"Project List"}</h2>
            </div>
            <div class="Content">

                <table class="DataTable" id="Projects">
                    <thead>
                        <tr>
                            <th>$Text{"Project"}</th>
                            <th>$Text{"Comment"}</th>
                            <th>$Text{"Status"}</th>
                        </tr>
                    </thead>
                    <tbody>
                        <tr class="FilterMessage Hidden">
                            <td colspan="2">
                                $Text{"No matches found."}
                            </td>
                        </tr>
<!-- dtl:block:NoProjectDataFoundMsg -->
                        <tr>
                            <td colspan="3">
                                $Text{"No data found."}
                            </td>
                        </tr>
<!-- dtl:block:NoProjectDataFoundMsg -->

<!-- dtl:block:OverviewResultProjectRow -->
                        <tr>
                            <td><a class="AsBlock" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=EditProject;ID=$LQData{"ProjectID"}">$QData{"Project"}</a></td>
                            <td title="$QData{"ProjectDescription"}">$QData{"ProjectDescription","40"}</td>
                            <td>$Text{"$Data{"Status"}"}</td>
                        </tr>
<!-- dtl:block:OverviewResultProjectRow -->
                    </tbody>
                </table>
            </div>
        </div>
<!-- dtl:block:OverviewResultProject -->

<!-- dtl:js_on_document_complete -->
<script type="text/javascript">//<![CDATA[
    Core.UI.Table.InitTableFilter($('#FilterProjects'), $('#Projects'));
//]]></script>
<!-- dtl:js_on_document_complete -->

<!-- dtl:block:OverviewResultSetting -->
        <div class="WidgetSimple">
            <div class="Header">
                <div class="WidgetAction Toggle"><a href="#" title="$Text{"Show or hide the content"}"><span></span></a></div>
                <h2>$Text{"Task List"}</h2>
            </div>
            <div class="Content">
                <table class="DataTable" id="Tasks">
                    <thead>
                        <tr>
                            <th>$Text{"Task"}</th>
                            <th>$Text{"Status"}</th>
                        </tr>
                    </thead>
                    <tbody>
                        <tr class="FilterMessage Hidden">
                            <td colspan="2">
                                $Text{"No matches found."}
                            </td>
                        </tr>
<!-- dtl:block:NoSettingDataFoundMsg -->
                        <tr>
                            <td colspan="2">
                                $Text{"No data found."}
                            </td>
                        </tr>
<!-- dtl:block:NoSettingDataFoundMsg -->

<!-- dtl:block:OverviewResultSettingRow -->
                        <tr>
                            <td><a class="AsBlock" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=EditTask;ActionID=$LQData{"ActionID"}">$QData{"Action"}</a></td>
                            <td>$Text{"$Data{"Status"}"}</td>
                        </tr>
<!-- dtl:block:OverviewResultSettingRow -->
                    </tbody>
                </table>
            </div>
        </div>
<!-- dtl:block:OverviewResultSetting -->

<!-- dtl:js_on_document_complete -->
<script type="text/javascript">//<![CDATA[
    Core.UI.Table.InitTableFilter($('#FilterTasks'), $('#Tasks'));
//]]></script>
<!-- dtl:js_on_document_complete -->

<!-- dtl:block:OverviewUpdateTask -->
        <div class="WidgetSimple">
            <div class="Header">
<!-- dtl:block:HeaderAddTask -->
                <h2>$Text{"Add Task"}</h2>
<!-- dtl:block:HeaderAddTask -->
<!-- dtl:block:HeaderEditTask -->
                <h2>$Text{"Edit Task Settings"}</h2>
<!-- dtl:block:HeaderEditTask -->
            </div>
            <div class="Content">
                <form action="$Env{"CGIHandle"}" method="post" name="TaskUpdate" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}">
                    <input type="hidden" name="Subaction" value="$QData{"Action"}Action">
                    <input type="hidden" name="ActionID" value="$QData{"ActionID"}"/>
                    <fieldset class="TableLike">

                        <label class="Mandatory" for="Task"><span class="Marker">*</span> $Text{"Task"}:</label>
                        <div class="Field">
                            <input type="text" name="Task" id="Task" value="$QData{"Task"}" class="W50pc Validate_Required $QData{"TaskInvalid"}" maxlength="50"/>
                            <div id="TaskError" class="TooltipErrorMessage">
                                <p>$Text{"This field is required."}</p>
                            </div>

<!-- dtl:block:TaskMissingValue -->
                            <div id="TaskServerError" class="TooltipErrorMessage">
                                <p>$Text{"This field is required."}</p>
                            </div>
<!-- dtl:block:TaskMissingValue -->

<!-- dtl:block:TaskDuplicateName -->
                            <div id="TaskServerError" class="TooltipErrorMessage">
                                <p>$Text{"There is already a task with this name. Please, choose a different one."}</p>
                            </div>
<!-- dtl:block:TaskDuplicateName -->
                        </div>
                        <div class="Clear"></div>

                        <label for="TaskStatus">$Text{"Status"}:</label>
                        <div class="Field">
                            $Data{"StatusOption"}
                        </div>
                        <div class="Clear"></div>

                        <div class="Field SpacingTop">
                            <button class="Primary" type="submit" id="Submit" value="$Text{"Submit"}">$Text{"Submit"}</button>
                            $Text{"or"}
                            <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Setting">$Text{"Cancel"}</a>
                        </div>
                        <div class="Clear"></div>
                    </fieldset>
                </form>
            </div>
        </div>
<!-- dtl:block:OverviewUpdateTask -->

<!-- dtl:block:OverviewResultUser -->
        <div class="WidgetSimple">
            <div class="Header">
                <div class="WidgetAction Toggle"><a href="#" title="$Text{"Show or hide the content"}"><span></span></a></div>
                <h2>$Text{"User List"}</h2>
            </div>
            <div class="Content">
                <table class="DataTable" id="Users">
                    <thead>
                        <tr>
                            <th>$Text{"User"}</th>
                            <th>$Text{"Comment"}</th>
                            <th>$Text{"Calendar"}</th>
                        </tr>
                    </thead>
                    <tbody>
                        <tr class="FilterMessage Hidden">
                            <td colspan="3">
                                $Text{"No matches found."}
                            </td>
                        </tr>
<!-- dtl:block:NoUserDataFoundMsg -->
                        <tr>
                            <td colspan="3">
                                $Text{"No data found."}
                            </td>
                        </tr>
<!-- dtl:block:NoUserDataFoundMsg -->

<!-- dtl:block:OverviewResultUserRow -->
                        <tr>
                            <td><a class="AsBlock" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=EditUser;UserID=$LQData{"UserID"}">$QData{"User"}</a></td>
                            <td>$Data{"Comment"}</td>
                            <td>$Text{"$Data{"Calendar"}"}</td>
                        </tr>
<!-- dtl:block:OverviewResultUserRow -->
                    </tbody>
                </table>
            </div>
        </div>
<!-- dtl:block:OverviewResultUser -->

<!-- dtl:js_on_document_complete -->
<script type="text/javascript">//<![CDATA[
    Core.UI.Table.InitTableFilter($('#FilterUsers'), $('#Users'));
    Core.UI.InitWidgetActionToggle();
//]]></script>
<!-- dtl:js_on_document_complete -->

<!-- dtl:block:OverviewUpdateUser -->
        <div class="WidgetSimple">
            <div class="Header">
<!-- dtl:block:HeaderAddUser -->
                <h2>$Text{"New User Settings"}: $QData{"UserFirstname"} $QData{"UserLastname"} ($QData{"UserLogin"})</h2>
<!-- dtl:block:HeaderAddUser -->
<!-- dtl:block:HeaderEditUser -->
                <h2>$Text{"Edit User Settings"}</h2>
<!-- dtl:block:HeaderEditUser -->
            </div>
            <div class="Content">
                <form action="$Env{"CGIHandle"}" method="post" name="UserUpdate" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="$QData{"Subaction"}" />
                    <input type="hidden" name="UserID" value="$QData{"UserID"}" />

                    <fieldset class="TableLike">
                        <label for="Name">$Text{"Name"}:</label>
                        <div class="Value">$QData{"UserFirstname"} $QData{"UserLastname"}</div>
                        <div class="Clear"></div>

                        <label for="Name">$Text{"Username"}:</label>
                        <div class="Value">$QData{"UserLogin"}</div>
                        <div class="Clear"></div>

                        <label for="Description">$Text{"Comments"}:</label>
                        <div class="Field">
                            <textarea rows="1" name="Description" id="Description" cols="40">$Data{"Description"}</textarea>
                        </div>
                        <div class="Clear"></div>

                        <div class="Field">
                            <input type="checkbox" name="ShowOvertime" id="ShowOvertime" value="1" $Data{"ShowOvertime"} />$Text{"Show Overtime"}
                        </div>
                        <div class="Clear"></div>

                        <div class="Field">
                            <input type="checkbox" name="CreateProject" id="CreateProject" value="1" $Data{"CreateProject"} />$Text{"Allow project creation"}
                        </div>
                        <div class="Clear"></div>

                        <label for="Calendar">$Text{"Calendar"}:</label>
                        <div class="Field">
                            $Data{"CalendarOption"}
                        </div>
                        <div class="Clear"></div>
                    </fieldset>

                    <table class="DataTable NoHover">
                        <thead>
                            <tr>
                                <th><span class="Marker">*</span> $Text{"Period Begin"}</th>
                                <th><span class="Marker">*</span> $Text{"Period End"}</th>
                                <th><span class="Marker">*</span> $Text{"Days of Vacation"}</th>
                                <th>$Text{"Hours per Week"}</th>
                                <th>$Text{"Authorized Overtime"}</th>
                                <th>$Text{"Status"}</th>
                            </tr>
                        </thead>
                        <tbody>
<!-- dtl:block:PeriodOverviewRow -->
                            <tr>
                                <td>
                                    <input type="text" id="DateStart-$QData{"Period"}" name="DateStart[$QData{"Period"}]" value="$QData{"DateStart"}" class="DatepickerElement Validate_Required $QData{"DateStartInvalid"}" />
                                    <div id="DateStart-$QData{"Period"}Error" class="TooltipErrorMessage"><p>$Text{"This field is required."} $Text{"Please insert a valid date."}</p></div>
<!-- dtl:block:DateStartInvalid -->
                                    <div id="DateStart-$QData{"Period"}ServerError" class="TooltipErrorMessage"><p>$Text{"Invalid date!"}</p></div>
<!-- dtl:block:DateStartInvalid -->
<!-- dtl:block:DateStartMissingValue -->
                                    <div id="DateStart-$QData{"Period"}ServerError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
<!-- dtl:block:DateStartMissingValue -->
                                </td>
                                <td>
                                    <input type="text" id="DateEnd-$QData{"Period"}" name="DateEnd[$QData{"Period"}]" value="$QData{"DateEnd"}" class="DatepickerElement Validate_Required $QData{"DateEndInvalid"}" />
                                    <div id="DateEnd-$QData{"Period"}Error" class="TooltipErrorMessage"><p>$Text{"This field is required."} $Text{"Please insert a valid date."}</p></div>
<!-- dtl:block:DateEndInvalid -->
                                    <div id="DateEnd-$QData{"Period"}ServerError" class="TooltipErrorMessage"><p>$Text{"Invalid date!"}</p></div>
<!-- dtl:block:DateEndInvalid -->
<!-- dtl:block:DateEndMissingValue -->
                                    <div id="DateEnd-$QData{"Period"}ServerError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
<!-- dtl:block:DateEndMissingValue -->
<!-- dtl:block:DateEndBeforeDateStart -->
                                    <div id="DateEnd-$QData{"Period"}ServerError" class="TooltipErrorMessage"><p>$Text{"Period end must be after period begin."}</p></div>
<!-- dtl:block:DateEndBeforeDateStart -->
                                </td>
                                <td>
                                    <input type="text" id="LeaveDays-$QData{"Period"}" name="LeaveDays[$QData{"Period"}]" value="$QData{"LeaveDays"}" class="Validate_Required $QData{"LeaveDaysInvalid"}" />
                                    <div id="LeaveDays-$QData{"Period"}Error" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="LeaveDays-$QData{"Period"}ServerError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                </td>
                                <td>
                                    <input type="text" id="WeeklyHours-$QData{"Period"}" name="WeeklyHours[$QData{"Period"}]" value="$QData{"WeeklyHours"}" />
                                </td>
                                <td>
                                    <input type="text" id="Overtime-$QData{"Period"}" name="Overtime[$QData{"Period"}]" value="$QData{"Overtime"}" />
                                </td>
                                <td>
                                    $Data{"PeriodStatusOption"}
                                </td>
                            </tr>
<!-- dtl:block:PeriodOverviewRow -->

<!-- dtl:block:PeriodOverviewRowNoData -->
                            <tr>
                                <td colspan="6">
                                    $Text{"No time periods found."}
                                </td>
                            </tr>
<!-- dtl:block:PeriodOverviewRowNoData -->
                        </tbody>
                    </table>

                    <div class="W100pc Right SpacingTopSmall">
                        <button type="submit" title="$Text{"Add time period"}" value="$Text{"Add time period"}" name="AddPeriod" class="CallForAction Plus"><span>$Text{"Add time period"}</span></button>
                    </div>

                    <div class="Center SpacingTop">
                        <button class="Primary" type="submit" value="$Text{"Submit"}" name="SubmitUserData">$Text{"Submit"}</button>
                        $Text{"or"}
                        <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Setting">$Text{"Cancel"}</a>
                    </div>
                </form>
            </div>
        </div>
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
    TimeAccounting.Agent.Datepicker.Init($('.DatepickerElement'), {
        DateInFuture: false,
        WeekDayStart: 1
    });
//]]></script>
<!--dtl:js_on_document_complete-->
<!-- dtl:block:OverviewUpdateUser -->
    </div>
</div>

<!-- dtl:block:Setting -->

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE3IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCjxkaXYgY2xhc3M9Ik1haW5Cb3ggQVJJQVJvbGVNYWluIExheW91dEZpeGVkU2lkZWJhciBTaWRlYmFyRmlyc3QiPgogICAgPGgxIGNsYXNzPSJIZWFkZXIiPiRUZXh0eyJWaWV3IFRpbWUgUmVjb3JkIn0KPCEtLSBkdGw6YmxvY2s6VXNlciAtLT4KICAgICAgICAtICRUZXh0eyJWaWV3IG9mICJ9ICRRRGF0YXsiVXNlciJ9CjwhLS0gZHRsOmJsb2NrOlVzZXIgLS0+CiAgICA8L2gxPgoKICAgIDxkaXYgY2xhc3M9IlNpZGViYXJDb2x1bW4iPgogICAgICAgIDxkaXYgY2xhc3M9IldpZGdldFNpbXBsZSI+CiAgICAgICAgICAgIDxkaXYgY2xhc3M9IkhlYWRlciI+CiAgICAgICAgICAgICAgICA8aDI+JFRleHR7IkFjdGlvbnMifTwvaDI+CiAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgICA8ZGl2IGNsYXNzPSJDb250ZW50Ij4KICAgICAgICAgICAgICAgIDx1bCBjbGFzcz0iQWN0aW9uTGlzdCI+CiAgICAgICAgICAgICAgICAgICAgPGxpPgogICAgICAgICAgICAgICAgICAgICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9QWN0aW9uPSRFbnZ7IkFjdGlvbiJ9O1N1YmFjdGlvbj1PdmVydmlldztZZWFyPSRMUURhdGF7IlllYXIifTtNb250aD0kTFFEYXRheyJNb250aCJ9IiBjbGFzcz0iQ2FsbEZvckFjdGlvbiI+PHNwYW4+JFRleHR7IkdvIHRvIG92ZXJ2aWV3In08L3NwYW4+PC9hPgogICAgICAgICAgICAgICAgICAgIDwvbGk+CiAgICAgICAgICAgICAgICA8L3VsPgogICAgICAgICAgICA8L2Rpdj4KICAgICAgICA8L2Rpdj4KICAgICAgICA8ZGl2IGNsYXNzPSJXaWRnZXRTaW1wbGUiPgogICAgICAgICAgICA8ZGl2IGNsYXNzPSJIZWFkZXIiPgogICAgICAgICAgICAgICAgPGgyPiRUZXh0eyJEYXRlIE5hdmlnYXRpb24ifTwvaDI+CiAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgICA8ZGl2IGNsYXNzPSJDb250ZW50Ij4KICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9Ik1vbnRoU2VsZWN0aW9uIj4KICAgICAgICAgICAgICAgICAgICA8YSBjbGFzcz0iQXJyb3cgTGVmdCIgdGl0bGU9IiRUZXh0eyJQcmV2aW91cyBkYXkifSIgaHJlZj0iJEVudnsiQmFzZWxpbmsifUFjdGlvbj0kRW52eyJBY3Rpb24ifTtTdWJhY3Rpb249VmlldztZZWFyPSRMUURhdGF7IlllYXJCYWNrIn07TW9udGg9JExRRGF0YXsiTW9udGhCYWNrIn07RGF5PSRMUURhdGF7IkRheUJhY2sifTtVc2VySUQ9JExRRGF0YXsiVXNlcklEIn0iPjwvYT4KICAgICAgICAgICAgICAgICAgICA8YSBjbGFzcz0iQXJyb3cgUmlnaHQiIHRpdGxlPSIkVGV4dHsiTmV4dCBkYXkifSIgaHJlZj0iJEVudnsiQmFzZWxpbmsifUFjdGlvbj0kRW52eyJBY3Rpb24ifTtTdWJhY3Rpb249VmlldztZZWFyPSRMUURhdGF7IlllYXJOZXh0In07TW9udGg9JExRRGF0YXsiTW9udGhOZXh0In07RGF5PSRMUURhdGF7IkRheU5leHQifTtVc2VySUQ9JExRRGF0YXsiVXNlcklEIn0iPjwvYT4KICAgICAgICAgICAgICAgICAgICAkUURhdGF7IlllYXIifS0kUURhdGF7Ik1vbnRoIn0tJFFEYXRheyJEYXkifQogICAgICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPSJDbGVhciI+PC9kaXY+CiAgICAgICAgICAgICAgICA8Zm9ybSBhY3Rpb249IiRFbnZ7IkNHSUhhbmRsZSJ9IiBtZXRob2Q9InBvc3QiIGVuY3R5cGU9Im11bHRpcGFydC9mb3JtLWRhdGEiIG5hbWU9IkRhdGVTZWxlY3Rpb24iIGNsYXNzPSJWYWxpZGF0ZSI+CiAgICAgICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iQWN0aW9uIiB2YWx1ZT0iJEVudnsiQWN0aW9uIn0iIC8+CiAgICAgICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iU3ViYWN0aW9uIiB2YWx1ZT0iVmlldyIgLz4KICAgICAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iaGlkZGVuIiBuYW1lPSJVc2VySUQiICAgIHZhbHVlPSIkUURhdGF7IlVzZXJJRCJ9IiAvPgogICAgICAgICAgICAgICAgICAgICREYXRheyJEYXRlU2VsZWN0aW9uIn0KICAgICAgICAgICAgICAgICAgICA8aW5wdXQgY2xhc3M9ImJ1dHRvbiIgdHlwZT0ic3VibWl0IiB2YWx1ZT0iJFRleHR7IlN1Ym1pdCJ9IiAvPgogICAgICAgICAgICAgICAgPC9mb3JtPgogICAgICAgICAgICA8L2Rpdj4KICAgICAgICA8L2Rpdj4KICAgIDwvZGl2PgoKICAgIDxkaXYgY2xhc3M9IkNvbnRlbnRDb2x1bW4iPgogICAgICAgIDxkaXYgY2xhc3M9IldpZGdldFNpbXBsZSI+CiAgICAgICAgICAgIDxkaXYgY2xhc3M9IkhlYWRlciI+CiAgICAgICAgICAgICAgICA8aDI+JFRleHR7IkRhdGUifTogJFRleHR7IiREYXRheyJXZWVrZGF5X3RvX1RleHQifSJ9ICRRRGF0YXsiRGF0ZSJ9PC9oMj4KICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgICAgIDxkaXYgY2xhc3M9IkNvbnRlbnQiPgogICAgICAgICAgICAgICAgPHRhYmxlIGNsYXNzPSJEYXRhVGFibGUgRGF0YVRhYmxlTm9IaWdobGlnaHQiPgogICAgICAgICAgICAgICAgICAgIDx0aGVhZD4KICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoPiRUZXh0eyJQcm9qZWN0In08L3RoPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoPiRUZXh0eyJUYXNrIn08L3RoPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoPiRUZXh0eyJSZW1hcmsifTwvdGg+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGg+JFRleHR7IlN0YXJ0In08L3RoPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoPiRUZXh0eyJFbmQifTwvdGg+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGg+JFRleHR7IlBlcmlvZCJ9PC90aD4KICAgICAgICAgICAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICAgICAgICA8L3RoZWFkPgogICAgICAgICAgICAgICAgICAgIDx0Ym9keT4KPCEtLSBkdGw6YmxvY2s6VW5pdCAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkPiRRRGF0YXsiUHJvamVjdCJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kUURhdGF7IkFjdGlvbiJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZD4kUURhdGF7IlJlbWFyayJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iUmlnaHQiPiRRRGF0YXsiU3RhcnRUaW1lIn08L3RkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJSaWdodCI+JFFEYXRheyJFbmRUaW1lIn08L3RkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJSaWdodCI+JFFEYXRheyJQZXJpb2QifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOlVuaXQgLS0+Cgo8IS0tIGR0bDpibG9jazpUb3RhbCAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkIGNvbHNwYW49IjQiPjwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQ+PHN0cm9uZz4kVGV4dHsiVG90YWwifTo8L3N0cm9uZz48L3RkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJSaWdodCI+PHN0cm9uZz4kUURhdGF7IlRvdGFsIn08L3N0cm9uZz48L3RkPgoKICAgICAgICAgICAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6VG90YWwgLS0+Cgo8IS0tIGR0bDpibG9jazpOb0RhdGFGb3VuZCAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkIGNvbHNwYW49IjYiPiRUZXh0eyJObyBkYXRhIGZvdW5kIGZvciB0aGlzIGRheS4ifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOk5vRGF0YUZvdW5kIC0tPgoKICAgICAgICAgICAgICAgICAgICA8L3Rib2R5PgogICAgICAgICAgICAgICAgPC90YWJsZT4KCjwhLS0gZHRsOmJsb2NrOk90aGVyVGltZXMgLS0+CiAgICAgICAgICAgICAgICA8ZGl2PgogICAgICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJjaGVja2JveCIgdmFsdWU9Ii0yIiBpZD0iTGVhdmVEYXkiIG5hbWU9IkxlYXZlRGF5IiB0aXRsZT0iJFRleHR7Ik9uIHZhY2F0aW9uIn0iICREYXRheyJMZWF2ZURheSJ9IGRpc2FibGVkIC8+CiAgICAgICAgICAgICAgICAgICAgPGxhYmVsIGZvcj0iTGVhdmVEYXkiPiRUZXh0eyJPbiB2YWNhdGlvbiJ9PC9sYWJlbD4KICAgICAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iY2hlY2tib3giIHZhbHVlPSItMSIgaWQ9IlNpY2siIG5hbWU9IlNpY2siIHRpdGxlPSIkVGV4dHsiT24gc2ljayBsZWF2ZSJ9IiAkRGF0YXsiU2ljayJ9IGRpc2FibGVkIC8+CiAgICAgICAgICAgICAgICAgICAgPGxhYmVsIGZvcj0iU2ljayI+JFRleHR7Ik9uIHNpY2sgbGVhdmUifTwvbGFiZWw+CiAgICAgICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImNoZWNrYm94IiB2YWx1ZT0iLTMiIGlkPSJPdmVydGltZSIgbmFtZT0iT3ZlcnRpbWUiIHRpdGxlPSIkVGV4dHsiT24gb3ZlcnRpbWUgbGVhdmUifSIgJERhdGF7Ik92ZXJ0aW1lIn0gZGlzYWJsZWQgLz4KICAgICAgICAgICAgICAgICAgICA8bGFiZWwgZm9yPSJPdmVydGltZSI+JFRleHR7Ik9uIG92ZXJ0aW1lIGxlYXZlIn08L2xhYmVsPgogICAgICAgICAgICAgICAgPC9kaXY+CjwhLS0gZHRsOmJsb2NrOk90aGVyVGltZXMgLS0+CgogICAgICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ2xlYXIiPjwvZGl2Pgo8IS0tIGR0bDpibG9jazpWYWNhdGlvbiAtLT4KICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9IlNwYWNpbmdUb3BTbWFsbCI+CiAgICAgICAgICAgICAgICAgICAgPHA+CiAgICAgICAgICAgICAgICAgICAgICAgICRRRGF0YXsiVmFjYXRpb24ifQogICAgICAgICAgICAgICAgICAgIDwvcD4KICAgICAgICAgICAgICAgIDwvZGl2Pgo8IS0tIGR0bDpibG9jazpWYWNhdGlvbiAtLT4KICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9IkNsZWFyIj48L2Rpdj4KICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgPC9kaXY+CiAgICA8L2Rpdj4KPC9kaXY+Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE3IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5vdGlmaWNhdGlvblRpbWVBY2NvdW50aW5nOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpUaW1lQWNjb3VudGluZzsKdXNlIEtlcm5lbDo6U3lzdGVtOjpUaW1lOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGdldCBuZWVkZWQgb2JqZWN0cwogICAgZm9yIChxdyhDb25maWdPYmplY3QgTG9nT2JqZWN0IERCT2JqZWN0IExheW91dE9iamVjdCBVc2VySUQpKSB7CiAgICAgICAgJFNlbGYtPnskX30gPSAkUGFyYW17JF99IHx8IGRpZSAiR290IG5vICRfISI7CiAgICB9CgogICAgJFNlbGYtPntUaW1lT2JqZWN0fSAgICAgICAgICAgPSBLZXJuZWw6OlN5c3RlbTo6VGltZS0+bmV3KCVQYXJhbSk7CiAgICAkU2VsZi0+e1RpbWVBY2NvdW50aW5nT2JqZWN0fSA9IEtlcm5lbDo6U3lzdGVtOjpUaW1lQWNjb3VudGluZy0+bmV3KCVQYXJhbSk7CiAgICByZXR1cm4gJFNlbGY7Cn0KCnN1YiBSdW4gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAoICRTZWMsICRNaW4sICRIb3VyLCAkRGF5LCAkTW9udGgsICRZZWFyICkgPSAkU2VsZi0+e1RpbWVPYmplY3R9LT5TeXN0ZW1UaW1lMkRhdGUoCiAgICAgICAgU3lzdGVtVGltZSA9PiAkU2VsZi0+e1RpbWVPYmplY3R9LT5TeXN0ZW1UaW1lKCksCiAgICApOwogICAgbXkgJVVzZXIgPSAkU2VsZi0+e1RpbWVBY2NvdW50aW5nT2JqZWN0fS0+VXNlckN1cnJlbnRQZXJpb2RHZXQoCiAgICAgICAgWWVhciAgPT4gJFllYXIsCiAgICAgICAgTW9udGggPT4gJE1vbnRoLAogICAgICAgIERheSAgID0+ICREYXksCiAgICApOwogICAgaWYgKCAkVXNlcnsgJFNlbGYtPntVc2VySUR9IH0gKSB7CiAgICAgICAgbXkgJUluY29tcGxldGVXb3JraW5nRGF5cyA9ICRTZWxmLT57VGltZUFjY291bnRpbmdPYmplY3R9LT5Xb3JraW5nVW5pdHNDb21wbGV0bmVzc0NoZWNrKCk7CgogICAgICAgICMgcmVkaXJlY3QgaWYgaW5jb21wbGV0ZSB3b3JraW5nIGRheSBhcmUgb3V0IG9mIHJhbmdlCiAgICAgICAgaWYgKCAkSW5jb21wbGV0ZVdvcmtpbmdEYXlze1dhcm5pbmd9ICkgewogICAgICAgICAgICByZXR1cm4gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5Ob3RpZnkoCiAgICAgICAgICAgICAgICBJbmZvICAgICA9PiAnUGxlYXNlIGluc2VydCB5b3VyIHdvcmtpbmcgaG91cnMhJywKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdFcnJvcicKICAgICAgICAgICAgKTsKICAgICAgICB9CiAgICB9CiAgICByZXR1cm4gJyc7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE3IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OlRvb2xCYXJJbmNvbXBsZXRlV29ya2luZ0RheXM7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpTeXN0ZW06OlRpbWVBY2NvdW50aW5nOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGdldCBuZWVkZWQgb2JqZWN0cwogICAgZm9yICgKICAgICAgICBxdyhDb25maWdPYmplY3QgTG9nT2JqZWN0IERCT2JqZWN0IFRpY2tldE9iamVjdCBVc2VyT2JqZWN0IEdyb3VwT2JqZWN0IExheW91dE9iamVjdCBVc2VySUQgVGltZU9iamVjdCkKICAgICAgICApCiAgICB7CiAgICAgICAgJFNlbGYtPnskX30gPSAkUGFyYW17JF99IHx8IGRpZSAiR290IG5vICRfISI7CiAgICB9CgogICAgIyBjcmVhdGUgbmVlZGVkIG9iamVjdHMKICAgICRTZWxmLT57VGltZUFjY291bnRpbmdPYmplY3R9ID0gS2VybmVsOjpTeXN0ZW06OlRpbWVBY2NvdW50aW5nLT5uZXcoJVBhcmFtKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCnN1YiBSdW4gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGRlZmluZSBhY3Rpb24sIGdyb3VwLCBsYWJlbCwgaW1hZ2UgYW5kIHByaW8KICAgIG15ICRBY3Rpb24gICAgPSAnQWdlbnRUaW1lQWNjb3VudGluZyc7CiAgICBteSAkU3ViYWN0aW9uID0gJ0VkaXQnOwogICAgbXkgJEdyb3VwICAgICA9ICd0aW1lX2FjY291bnRpbmcnOwoKICAgICMgZG8gbm90IHNob3cgaWNvbiBpZiBmcm9udGVuZCBtb2R1bGUgaXMgbm90IHJlZ2lzdGVyZWQKICAgIHJldHVybiBpZiAhJFNlbGYtPntDb25maWdPYmplY3R9LT5HZXQoJ0Zyb250ZW5kOjpNb2R1bGUnKS0+eyRBY3Rpb259OwoKICAgICMgZ2V0IHRoZSBncm91cCBpZAogICAgbXkgJEdyb3VwSUQgPSAkU2VsZi0+e0dyb3VwT2JqZWN0fS0+R3JvdXBMb29rdXAoIEdyb3VwID0+ICRHcm91cCApOwoKICAgICMgZGVueSBhY2Nlc3MsIHdoZW4gdGhlIGdyb3VwIGlzIG5vdCBmb3VuZAogICAgcmV0dXJuIGlmICEkR3JvdXBJRDsKCiAgICAjIGdldCB1c2VyIGdyb3Vwcywgd2hlcmUgdGhlIHVzZXIgaGFzIHRoZSBhcHByb3ByaWF0ZSBwcml2aWxlZ2UKICAgIG15ICVHcm91cHMgPSAkU2VsZi0+e0dyb3VwT2JqZWN0fS0+R3JvdXBNZW1iZXJMaXN0KAogICAgICAgIFVzZXJJRCA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICAgICAgVHlwZSAgID0+ICdydycsCiAgICAgICAgUmVzdWx0ID0+ICdIQVNIJywKICAgICk7CgogICAgIyBkZW55IGFjY2VzcyBpZiB0aGUgYWdlbnQgZG9lc24ndCBoYXZlIHRoZSBhcHByb3ByaWF0ZSB0eXBlIGluIHRoZSBhcHByb3ByaWF0ZSBncm91cAogICAgcmV0dXJuIGlmICEkR3JvdXBzeyRHcm91cElEfTsKCiAgICBteSAoICRTZWMsICRNaW4sICRIb3VyLCAkRGF5LCAkTW9udGgsICRZZWFyICkgPSAkU2VsZi0+e1RpbWVPYmplY3R9LT5TeXN0ZW1UaW1lMkRhdGUoCiAgICAgICAgU3lzdGVtVGltZSA9PiAkU2VsZi0+e1RpbWVPYmplY3R9LT5TeXN0ZW1UaW1lKCksCiAgICApOwoKICAgIG15ICVVc2VyQ3VycmVudFBlcmlvZCA9ICRTZWxmLT57VGltZUFjY291bnRpbmdPYmplY3R9LT5Vc2VyQ3VycmVudFBlcmlvZEdldCgKICAgICAgICBZZWFyICA9PiAkWWVhciwKICAgICAgICBNb250aCA9PiAkTW9udGgsCiAgICAgICAgRGF5ICAgPT4gJERheSwKICAgICk7CgogICAgIyBkZW55IGFjY2VzcywgaWYgdXNlciBoYXMgbm8gdmFsaWQgcGVyaW9kCiAgICByZXR1cm4gaWYgISRVc2VyQ3VycmVudFBlcmlvZHsgJFNlbGYtPntVc2VySUR9IH07CgogICAgIyBnZXQgdGhlIG51bWJlciBvZiBpbmNvbXBsZXRlIHdvcmtpbmcgZGF5cwogICAgbXkgJENvdW50ICAgICAgICAgICAgICAgICA9IDA7CiAgICBteSAlSW5jb21wbGV0ZVdvcmtpbmdEYXlzID0gJFNlbGYtPntUaW1lQWNjb3VudGluZ09iamVjdH0tPldvcmtpbmdVbml0c0NvbXBsZXRuZXNzQ2hlY2soKTsKCiAgICBZRUFSSUQ6CiAgICBmb3IgbXkgJFllYXJJRCAoIHNvcnQga2V5cyAleyAkSW5jb21wbGV0ZVdvcmtpbmdEYXlze0luY29tcGxldGV9IH0gKSB7CgogICAgICAgIG5leHQgWUVBUklEIGlmICEkWWVhcklEOwogICAgICAgIG5leHQgWUVBUklEIGlmICEkSW5jb21wbGV0ZVdvcmtpbmdEYXlze0luY29tcGxldGV9eyRZZWFySUR9OwogICAgICAgIG5leHQgWUVBUklEIGlmIHJlZiAkSW5jb21wbGV0ZVdvcmtpbmdEYXlze0luY29tcGxldGV9eyRZZWFySUR9IG5lICdIQVNIJzsKCiAgICAgICAgIyBleHRyYWN0IHllYXIKICAgICAgICBteSAlWWVhciA9ICV7ICRJbmNvbXBsZXRlV29ya2luZ0RheXN7SW5jb21wbGV0ZX17JFllYXJJRH0gfTsKCiAgICAgICAgTU9OVEg6CiAgICAgICAgZm9yIG15ICRNb250aElEICggc29ydCBrZXlzICVZZWFyICkgewoKICAgICAgICAgICAgbmV4dCBNT05USCBpZiAhJE1vbnRoSUQ7CiAgICAgICAgICAgIG5leHQgTU9OVEggaWYgISRZZWFyeyRNb250aElEfTsKICAgICAgICAgICAgbmV4dCBNT05USCBpZiByZWYgJFllYXJ7JE1vbnRoSUR9IG5lICdIQVNIJzsKCiAgICAgICAgICAgICMgZXh0cmFjdCBtb250aAogICAgICAgICAgICBteSAlTW9udGggPSAleyAkWWVhcnskTW9udGhJRH0gfTsKCiAgICAgICAgICAgICRDb3VudCArPSBzY2FsYXIga2V5cyAlTW9udGg7CiAgICAgICAgfQogICAgfQoKICAgICMgcmVtb3ZlIGN1cnJlbnQgZGF5IGJlY2F1c2UgaXQgbWFrZXMgbm8gc2Vuc2UgdG8gc2hvdyB0aGUgY3VycmVudCBkYXkgYXMgaW5jb21wbGV0ZWQKICAgIGlmICggJENvdW50ID4gMCApIHsKICAgICAgICAkQ291bnQtLTsKICAgIH0KCiAgICAjIGdldCBUb29sQmFyIG9iamVjdCBwYXJhbWV0ZXJzCiAgICBteSAkQ2xhc3MgPSAkUGFyYW17Q29uZmlnfS0+e0Nzc0NsYXNzfTsKICAgIG15ICRUZXh0ICA9ICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+e0xhbmd1YWdlT2JqZWN0fS0+R2V0KCdJbmNvbXBsZXRlIHdvcmtpbmcgZGF5cycpOwogICAgbXkgJFVSTCAgID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT57QmFzZWxpbmt9OwogICAgbXkgJEljb24gID0gJFBhcmFte0NvbmZpZ30tPntJY29ufTsKCiAgICByZXR1cm4gKCkgaWYgISRDb3VudDsKCiAgICBteSAlUmV0dXJuID0gKAogICAgICAgIDEwMDA4MTAgPT4gewogICAgICAgICAgICBCbG9jayAgICAgICA9PiAnVG9vbEJhckl0ZW0nLAogICAgICAgICAgICBEZXNjcmlwdGlvbiA9PiAkVGV4dCwKICAgICAgICAgICAgQ291bnQgICAgICAgPT4gJENvdW50LAogICAgICAgICAgICBDbGFzcyAgICAgICA9PiAkQ2xhc3MsCiAgICAgICAgICAgIEljb24gICAgICAgID0+ICRJY29uLAogICAgICAgICAgICBMaW5rICAgICAgICA9PiAkVVJMIC4gJ0FjdGlvbj0nIC4gJEFjdGlvbiAuICc7U3ViYWN0aW9uPScgLiAkU3ViYWN0aW9uLAogICAgICAgICAgICBBY2Nlc3NLZXkgICA9PiAnJywKICAgICAgICAgICAgfQogICAgKTsKCiAgICByZXR1cm4gJVJldHVybjsKfQoKMTsK
# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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);

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

=head1 NAME

Kernel::System::TimeAccounting - time accounting lib

=head1 SYNOPSIS

All time accounting 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 current period data of the specified user

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

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

    # db select
    return if !$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, ],
    );

    # fetch the 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 valid 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         = 1970;
    my $MonthStart        = 1;
    my $DayStart          = 1;
    my $YearEnd           = $Param{Year};
    my $MonthEnd          = $Param{Month};
    my $DayEnd            = $Param{Day};

    my %Data;

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

        if ( $UserCurrentPeriod{$UserID}{DateStart} =~ m{ \A (\d{4})-(\d{2})-(\d{2}) }xms ) {
            $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,
        );

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

        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,
                        Calendar => $Calendar || '',
                    );
                    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 all the projects' data

    my %ProjectData = $TimeAccountingObject->ProjectSettingsGet(
        Status => 'valid' || 'invalid', optional default valid && invalid
    );

=cut

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

    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 the data
    my %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 ProjectGet()

returns a hash with the requested project data

    my %ProjectData = $TimeAccountingObject->ProjectGet( ID => 2 );

This returns something like:

    $TimeAccountingObject = (
        Project            => 'internal',
        ProjectDescription => 'description',
        ProjectStatus      => 1,
    );

    or

    my %ProjectData = $TimeAccountingObject->ProjectGet( Project => 'internal' );

This returns something like:

    $TimeAccountingObject = (
        ID                 => 2,
        ProjectDescription => 'description',
        ProjectStatus      => 1,
    );

=cut

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

    # check needed stuff
    if ( !$Param{ID} && !$Param{Project} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Need ID or project name!'
        );
        return;
    }

    my %Project;

    # look for the task data with the ID
    if ( $Param{ID} ) {

        # sql
        return if !$Self->{DBObject}->Prepare(
            SQL => 'SELECT project, description, status '
                . 'FROM time_accounting_project WHERE id = ?',
            Bind => [ \$Param{ID} ],
        );
        while ( my @Data = $Self->{DBObject}->FetchrowArray() ) {
            %Project = (
                ID                 => $Param{ID},
                Project            => $Data[0],
                ProjectDescription => $Data[1],
                ProjectStatus      => $Data[2],
            );
        }
    }

    # look for the task data with the task name
    else {

        # sql
        return if !$Self->{DBObject}->Prepare(
            SQL => 'SELECT id, description, status '
                . 'FROM time_accounting_project WHERE project = ?',
            Bind => [ \$Param{Project} ],
        );
        while ( my @Data = $Self->{DBObject}->FetchrowArray() ) {
            %Project = (
                Project            => $Param{Project},
                ID                 => $Data[0],
                ProjectDescription => $Data[1],
                ProjectStatus      => $Data[2],
            );
        }
    }

    return %Project;
}

=item ProjectSettingsInsert()

inserts a new project 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{ProjectDescription} ||= '';

    if ( $Param{ProjectStatus} ne '0' && $Param{ProjectStatus} ne '1' ) {
        $Param{ProjectStatus} = $Self->{ConfigObject}->Get('TimeAccounting::DefaultProjectStatus');
    }

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

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

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

    return $ProjectID;
}

=item ProjectSettingsUpdate()

updates a project

    my $Success = $TimeAccountingObject->ProjectSettingsUpdate(
        ID                 => 123,
        Project            => 'internal',
        ProjectDescription => 'description',
        ProjectStatus      => 1,
    );

=cut

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

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

    # sql
    return if !$Self->{DBObject}->Do(
        SQL => 'UPDATE time_accounting_project SET project = ?, description = ?, status = ?'
            . ' WHERE id = ?',
        Bind => [
            \$Param{Project}, \$Param{ProjectDescription}, \$Param{ProjectStatus}, \$Param{ID},
        ],
    );

    return 1;
}

=item ActionSettingsGet()

returns a hash with all the actions settings

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

=cut

sub ActionSettingsGet {
    my $Self = shift;

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

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

    return %Data;
}

=item ActionGet()

returns a hash with the requested action (task) data

    my %ActionData = $TimeAccountingObject->ActionGet( ID => 2 );

This returns something like:

    $TimeAccountingObject = (
        Action       => 'My task',
        ActionStatus => 1,
    );

    or

    my %ActionData = $TimeAccountingObject->ActionGet( Action => 'My task' );

This returns something like:

    $TimeAccountingObject = (
        ID           => 2,
        ActionStatus => 1,
    );

=cut

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

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

    my %Task;

    # look for the task data with the ID
    if ( $Param{ID} ) {

        # sql
        return if !$Self->{DBObject}->Prepare(
            SQL => 'SELECT action, status '
                . 'FROM time_accounting_action WHERE id = ?',
            Bind => [ \$Param{ID} ],
        );
        while ( my @Data = $Self->{DBObject}->FetchrowArray() ) {
            %Task = (
                ID           => $Param{ID},
                Action       => $Data[0],
                ActionStatus => $Data[1],
            );
        }
    }

    # look for the task data with the task name
    else {

        # sql
        return if !$Self->{DBObject}->Prepare(
            SQL => 'SELECT id, status '
                . 'FROM time_accounting_action WHERE action = ?',
            Bind => [ \$Param{Action} ],
        );
        while ( my @Data = $Self->{DBObject}->FetchrowArray() ) {
            %Task = (
                Action       => $Param{Action},
                ID           => $Data[0],
                ActionStatus => $Data[1],
            );
        }
    }

    return %Task;
}

=item ActionSettingsInsert()

inserts a new action in the db

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

=cut

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

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

    # db insert
    return if !$Self->{DBObject}->Do(
        SQL => 'INSERT INTO time_accounting_action (action, status) '
            . 'VALUES (?, ?)',
        Bind => [ \$Param{Action}, \$Param{ActionStatus}, ],
    );

    return 1;
}

=item ActionSettingsUpdate()

updates an action (task)

    my $Success = $TimeAccountingObject->ActionSettingsUpdate(
        ActionID     => 123,
        Action       => 'internal',
        ActionStatus => 1,
    );

=cut

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

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

    # sql
    return if !$Self->{DBObject}->Do(
        SQL => 'UPDATE time_accounting_action SET action = ?, status = ?'
            . ' WHERE id = ?',
        Bind => [
            \$Param{Action}, \$Param{ActionStatus}, \$Param{ActionID}
        ],
    );

    return 1;
}

=item UserList()

returns a hash with the user data of all users

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

=cut

sub UserList {
    my $Self = shift;

    # db select
    $Self->{DBObject}->Prepare(
        SQL =>
            'SELECT user_id, description, show_overtime, create_project, calendar FROM time_accounting_user',
    );

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

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

    # fetch the data
    my %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 complete user period data for all users

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

returns:
    %UserData = (
        3 => {
            1 => {
                DateEnd     => "2012-12-31",
                DateStart   => "2012-10-01",
                LeaveDays   => "23.00",
                Overtime    => "0.00",
                Period      => 1,
                UserID      => 3,
                UserStatus  => 1,
                WeeklyHours => "40.00",
            },
            2 => {
                DateEnd     => "2013-12-31",
                DateStart   => "2013-01-01",
                LeaveDays   => "23.00",
                Overtime    => "0.00",
                Period      => 2,
                UserID      => 3,
                UserStatus  => 1,
                WeeklyHours => "32.00",
            },
        },
        4 => {
            1 => {
                DateEnd     => "2013-12-31",
                DateStart   => "2012-01-01",
                LeaveDays   => "23.00",
                Overtime    => "0.00",
                Period      => 1,
                UserID      => 4,
                UserStatus  => 1,
                WeeklyHours => "40.00",
            },
        },
    };

=cut

sub UserSettingsGet {
    my $Self = shift;

    # 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 the data
    my %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 SingleUserSettingsGet()

returns a hash with the requested user's period data

    my %UserData = $TimeAccountingObject->SingleUserSettingsGet( UserID => 1 );

=cut

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

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

    # db select
    $Self->{DBObject}->Prepare(
        SQL =>
            'SELECT user_id, preference_period, date_start, date_end, weekly_hours, leave_days, overtime, status '
            . 'FROM time_accounting_user_period WHERE user_id = ?',
        Bind => [ \$Param{UserID} ],
    );

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

    return %UserData;
}

=item UserLastPeriodNumberGet()

returns the number of the last registered period for the specified user

    my $LastPeriodNumber = $TimeAccountingObject->UserLastPeriodNumberGet( UserID => 1 );

=cut

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

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

    # db select
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT max(preference_period) '
            . 'FROM time_accounting_user_period WHERE user_id = ?',
        Bind => [ \$Param{UserID} ],
    );

    # fetch the data
    my @Row = $Self->{DBObject}->FetchrowArray();
    my $LastPeriodNumber = $Row[0] || 0;

    return $LastPeriodNumber;
}

=item UserSettingsInsert()

insert new user data in the db

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

=cut

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

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

    # check if user exists
    if ( !$Self->{UserObject}->UserLookup( UserID => $Param{UserID} ) ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "UserID $Param{UserID} does not exist!"
        );
        return;
    }

    $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')
        || '2011-12-31';
    $Param{DateStart} = $Self->{ConfigObject}->Get('TimeAccounting::DefaultUserDateStart')
        || '2011-01-01';
    $Param{Description} = $Self->{ConfigObject}->Get('TimeAccounting::DefaultUserDescription')
        || 'Put your description here.';

    $Param{DateStart} .= ' 00:00:00';
    $Param{DateEnd}   .= ' 00:00:00';

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

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

    # select userid
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT user_id FROM time_accounting_user WHERE user_id = ?',
        Bind  => [ \$Param{UserID}, ],
        Limit => 1,
    );

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

    if ( !defined $UserID ) {

        # db insert
        return if !$Self->{DBObject}->Do(
            SQL => 'INSERT INTO time_accounting_user (user_id, description)'
                . ' VALUES (?, ?)',
            Bind => [ \$Param{UserID}, \$Param{Description}, ],
        );
    }

    return 1;
}

=item UserSettingsUpdate()

updates user data in the db

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

=cut

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

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

    my $UserID = $Param{UserID};

    if ( !defined $Param{Period}->{1}{DateStart} && !defined $Param{Period}->{1}{DateEnd} ) {
        return $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "UserSettingUpdate: No data for user id $UserID!"
        );
    }

    # set default values
    $Param{ShowOvertime}  ||= 0;
    $Param{CreateProject} ||= 0;
    $Param{Calendar}      ||= 0;

    # db insert
    return if !$Self->{DBObject}->Do(
        SQL => 'UPDATE time_accounting_user '
            . ' SET description = ?, show_overtime = ?, create_project = ?, calendar = ?'
            . ' WHERE user_id = ?',
        Bind => [
            \$Param{Description}, \$Param{ShowOvertime},
            \$Param{CreateProject}, \$Param{Calendar}, \$Param{UserID}
        ],
    );

    # update all periods
    for my $Period ( sort keys %{ $Param{Period} } ) {

        # db insert
        return if !$Self->{DBObject}->Do(
            SQL => "UPDATE time_accounting_user_period "
                . "SET leave_days = ?, date_start = ?"
                . ", date_end = ?, overtime = ?"
                . ", weekly_hours = ?, status = ? "
                . "WHERE user_id = ? AND preference_period = ?",
            Bind => [
                \$Param{Period}->{$Period}{LeaveDays},   \$Param{Period}->{$Period}{DateStart},
                \$Param{Period}->{$Period}{DateEnd},     \$Param{Period}->{$Period}{Overtime},
                \$Param{Period}->{$Period}{WeeklyHours}, \$Param{Period}->{$Period}{UserStatus},
                \$UserID, \$Period,
                ]
        );
    }

    return 1;
}

=item WorkingUnitsCompletnessCheck()

returns a hash with the incomplete working days and
the information if the incomplete working days 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 the 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   = 1970;
    my $MonthStart  = 1;
    my $DayStart    = 1;
    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 ( sort keys %{ $Data{Incomplete} } ) {

        for my $Month ( sort keys %{ $Data{Incomplete}{$Year} } ) {

            for my $Day ( sort 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 ) = @_;

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

    # ask the database
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT user_id, project_id, action_id, remark, time_start, time_end, period '
            . 'FROM time_accounting_table '
            . 'WHERE time_start >= ? AND time_start <= ? AND user_id = ? '
            . 'ORDER by id',
        Bind => [ \$DateStart, \$DateStop, \$Param{UserID} ],
    );

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

    # fetch the result
    ROW:
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {

        next ROW if $Row[4] !~ m{^ (.+?) \s (\d+:\d+) : (\d+) }xms;

        # 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+)}xms ) {
            $EndTime = $2;

            # replace 23:59:59 with 24:00
            if ( $EndTime eq '23:59' && $3 eq '59' ) {
                $EndTime = '24:00';
            }
        }

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

    # 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} } ) {

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

deletes working units in the db

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

=cut

sub WorkingUnitsDelete {
    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};
    my $StartTime = $Date . ' 00:00:00';
    my $EndTime   = $Date . ' 23:59:59';

    return if !$Self->{DBObject}->Do(
        SQL => 'DELETE FROM time_accounting_table'
            . ' WHERE time_start >= ?'
            . ' AND time_start <= ?'
            . ' AND user_id = ?',
        Bind => [ \$StartTime, \$EndTime, \$Self->{UserID}, ],
    );
    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 users'
    );

=cut

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

    for my $Parameter (qw(Year Month)) {
        $Param{$Parameter} = $Self->{DBObject}->Quote( $Param{$Parameter} ) || '';
        if ( !$Param{$Parameter} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "ProjectActionReporting: Need $Parameter!"
            );
            return;
        }
    }

    my $IDSelect = '';
    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 $SQLQueryTimeStart = "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 $SQLQueryTimeStart",
    );

    # fetch the data
    my %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 $SQLQueryTimeStart",
    );

    # fetch the 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 ( sort 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 ( sort keys %{$ActionsRef} ) {
            $Data{$ProjectID}->{Actions}->{$ActionID}->{Name} = $Action{$ActionID}->{Action} || '';
        }
    }

    return %Data;
}

=item ProjectTotalHours()

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

    # ask the database
    return if !$Self->{DBObject}->Prepare(
        SQL   => 'SELECT SUM(period) FROM time_accounting_table WHERE project_id = ?',
        Bind  => [ \$Param{ProjectID} ],
        Limit => 1,
    );

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

    return $Total;
}

=item ProjectHistory()

returns an array with all WorkingUnits related to a project

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

This would return

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

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

    # ask the database
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT id, user_id, action_id, remark, time_start, time_end, period, created'
            . ' FROM time_accounting_table WHERE project_id = ? ORDER BY time_start',
        Bind => [ \$Param{ProjectID} ],
    );

    # fetch the result
    my @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}xms;

        push @Data, $UserRef;
    }

    return @Data;
}

=item LastProjectsOfUser()

returns an array with the last projects of the current 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
    return if !$Self->{DBObject}->Prepare(
        SQL => 'SELECT project_id FROM time_accounting_table '
            . 'WHERE user_id = ? AND project_id <> -1 ORDER BY time_start DESC',
        Bind  => [ \$Self->{UserID} ],
        Limit => 40,
    );

    # fetch the result
    my %Projects;
    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>.

=cut

# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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::Stats::Dynamic::TimeAccounting;

use strict;
use warnings;

use Kernel::System::TimeAccounting;
use Date::Pcalc qw( Add_Delta_Days Add_Delta_YMD );

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 UserObject TimeObject MainObject EncodeObject)
        )
    {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }

    # create additional needed objects
    $Self->{TimeAccountingObject} = Kernel::System::TimeAccounting->new(
        %{$Self},
        UserID => 1,
    );

    return $Self;
}

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

    return 'TimeAccounting';
}

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

    # set predefined start time
    my $TimeStamp = $Self->{TimeObject}->CurrentTimestamp();
    my ($Date) = split /\s+/, $TimeStamp;
    my $Today = sprintf "%s 23:59:59", $Date;

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

    # get action list
    my %ActionListSource = $Self->{TimeAccountingObject}->ActionSettingsGet();
    my %ActionList;

    for my $Action ( sort keys %ActionListSource ) {
        $ActionList{$Action} = $ActionListSource{$Action}->{Action};
    }

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

    my @Attributes = (
        {
            Name             => 'Project',
            UseAsXvalue      => 1,
            UseAsValueSeries => 0,
            UseAsRestriction => 1,
            Element          => 'Project',
            Block            => 'MultiSelectField',
            Translation      => 0,
            Values           => $ProjectList{Project},
        },
        {
            Name             => 'User',
            UseAsXvalue      => 1,
            UseAsValueSeries => 0,
            UseAsRestriction => 1,
            Element          => 'User',
            Block            => 'MultiSelectField',
            Translation      => 0,
            Values           => \%UserList,
        },
        {
            Name             => 'Sort sequence',
            UseAsXvalue      => 0,
            UseAsValueSeries => 1,
            UseAsRestriction => 0,
            Element          => 'SortSequence',
            Block            => 'SelectField',
            Translation      => 0,
            Values           => {
                Up   => 'ascending',
                Down => 'descending',
            },
        },
        {
            Name             => 'Task',
            UseAsXvalue      => 0,
            UseAsValueSeries => 0,
            UseAsRestriction => 1,
            Element          => 'ProjectAction',
            Block            => 'MultiSelectField',
            Translation      => 0,
            Values           => \%ActionList,
        },
        {
            Name             => 'Period',
            UseAsXvalue      => 0,
            UseAsValueSeries => 0,
            UseAsRestriction => 1,
            Element          => 'Period',
            TimePeriodFormat => 'DateInputFormat',    # 'DateInputFormatLong',
            Block            => 'Time',
            Values           => {
                TimeStart => 'TimeAccountingPeriodStart',
                TimeStop  => 'TimeAccountingPeriodStop',
            },
        },
    );

    return @Attributes;
}

sub GetHeaderLine {
    my ( $Self, %Param ) = @_;
    my @HeaderLine = ("");

    # Users as X-value
    if ( $Param{XValue}{Element} && $Param{XValue}{Element} eq 'User' ) {

        # user have been selected as x-value
        my @UserIDs = @{ $Param{XValue}{SelectedValues} };

        # iterate over selected users
        USERID:
        for my $UserID (@UserIDs) {
            my $UserName = $Self->{UserObject}->UserName(
                UserID => $UserID,
            );

            push @HeaderLine, $UserName;
        }
    }

    # Projects as X-value
    else {
        # projects have been selevted as x-value
        my @ProjectIDs = @{ $Param{XValue}{SelectedValues} };

        # iterate over selected projects
        PROJECTID:
        for my $ProjectID (@ProjectIDs) {
            my %ProjectData = $Self->{TimeAccountingObject}->ProjectGet(
                ID => $ProjectID,
            );

            push @HeaderLine, $ProjectData{Project};
        }
    }

    return \@HeaderLine;
}

sub GetStatTable {
    my ( $Self, %Param ) = @_;
    my @StatArray;
    my @UserIDs;

    # Users as X-value
    if ( $Param{XValue}{Element} && $Param{XValue}{Element} eq 'User' ) {

        # user have been selected as x-value
        @UserIDs = @{ $Param{XValue}{SelectedValues} };

        # get stat data
        my $StatData = $Self->_GetStatData(
            Param   => \%Param,
            UserIDs => \@UserIDs,
        );

        # check stat data
        return if !$StatData;
        return if ref $StatData ne 'ARRAY';

        my @RawStatArray = @{$StatData};
        return if !@RawStatArray;

        # get list of needed data
        my %ProjectData = $Self->{TimeAccountingObject}->ProjectSettingsGet();
        my %ProjectList = %{ $ProjectData{Project} || {} };

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

        my @SortedProjectIDs = sort { $ProjectList{$a} cmp $ProjectList{$b} } keys %ProjectList;
        my @SortedActionIDs  = sort { $ActionList{$a} cmp $ActionList{$b} } keys %ActionList;

        # re-sort projects depending on selected sequence
        if ( $Param{ValueSeries} && $Param{ValueSeries}[0]{SelectedValues}[0] eq 'Down' ) {
            @SortedProjectIDs = reverse @SortedProjectIDs;
        }

        # iterate over sorted project list
        SORTEDPROJECTID:
        for my $SortedProjectID (@SortedProjectIDs) {

            # check for unselected projects
            next SORTEDPROJECTID if $Param{Restrictions}->{Project} && !grep {
                $_ == $SortedProjectID
            } @{ $Param{Restrictions}->{Project} || [] };

            # get the current project data of current project
            my @ProjectStatData = grep { $_->{ProjectID} == $SortedProjectID } @RawStatArray;

            # iterate over sorted action list
            SORTEDACTIONID:
            for my $SortedActionID (@SortedActionIDs) {

                # check for unselected actions
                next SORTEDACTIONID if $Param{Restrictions}->{ProjectAction} && !grep {
                    $_ == $SortedActionID
                } @{ $Param{Restrictions}->{ProjectAction} || [] };

                # get the current action out of the current project
                my @ActionStatData = grep { $_->{ActionID} == $SortedActionID } @ProjectStatData;

                my @RowData;

                # add descriptive first column
                my $RowLabel = "$ProjectList{$SortedProjectID}::$ActionList{$SortedActionID}";
                push @RowData, $RowLabel;

                # iterate over selected users
                USERID:
                for my $UserID (@UserIDs) {

                    # at least get '0' for user data
                    my $UserPeriodSum = 0;

                    # iterate over period data of user
                    for my $PeriodData ( grep { $_->{UserID} == $UserID } @ActionStatData ) {
                        $UserPeriodSum += $PeriodData->{Period};
                    }

                    # safe user data to row data
                    push @RowData, $UserPeriodSum;
                }

                # store current row to global stat array
                push @StatArray, \@RowData;
            }
        }
    }

    # Projects as X-value
    else {

        # projects have been selected as x-value
        my @ProjectIDs = @{ $Param{XValue}{SelectedValues} };

        # we need to get all users
        my %UserIDs = $Self->{UserObject}->UserList(
            Type  => 'Short',
            Valid => 1,
        );

        @UserIDs = keys %UserIDs;

        # get calculated stats data
        my $StatData = $Self->_GetStatData(
            Param   => \%Param,
            UserIDs => \@UserIDs,
        );

        # check stat data
        return if !$StatData;
        return if ref $StatData ne 'ARRAY';

        my @RawStatArray = @{$StatData};
        return if !@RawStatArray;

        # get list of needed data
        my %UserList = $Self->{UserObject}->UserList(
            Type  => 'Long',
            Valid => 1,
        );

        my @SortedUserIDs = sort { $UserList{$a} cmp $UserList{$b} } keys %UserList;

        # re-sort users depending on selected sequence
        if ( $Param{ValueSeries} && $Param{ValueSeries}[0]{SelectedValues}[0] eq 'Down' ) {
            @SortedUserIDs = reverse @SortedUserIDs;
        }

        # iterate over sorted user list
        SORTEDUSERID:
        for my $SortedUserID (@SortedUserIDs) {

            # check for unselected users
            next SORTEDUSERID if $Param{Restrictions}->{User} && !grep {
                $_ == $SortedUserID
            } @{ $Param{Restrictions}->{User} || [] };

            # get the current user data of current user
            my @UserStatData = grep { $_->{UserID} == $SortedUserID } @RawStatArray;

            my @RowData;

            # add descriptive first column
            my $RowLabel = $UserList{$SortedUserID};
            push @RowData, $RowLabel;

            # iterate over selected projects
            PROJECTID:
            for my $ProjectID (@ProjectIDs) {

                # at least get '0' for user data
                my $ProjectPeriodSum = 0;

                # iterate over period data of user
                for my $PeriodData ( grep { $_->{ProjectID} == $ProjectID } @UserStatData ) {
                    $ProjectPeriodSum += $PeriodData->{Period};
                }

                # safe user data to row data
                push @RowData, $ProjectPeriodSum;
            }

            # store current row to global stat array
            push @StatArray, \@RowData;
        }

    }

    return @StatArray;
}

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

    # wrap ids to used spelling
    for my $Use (qw(UseAsValueSeries UseAsRestriction UseAsXvalue)) {
        ELEMENT:
        for my $Element ( @{ $Param{$Use} } ) {
            next ELEMENT if !$Element || !$Element->{SelectedValues};
            my $ElementName = $Element->{Element};
            my $Values      = $Element->{SelectedValues};

            if ( $ElementName eq 'User' ) {
                ID:
                for my $ID ( @{$Values} ) {
                    next ID if !$ID;

                    $ID->{Content} = $Self->{UserObject}->UserLookup( UserID => $ID->{Content} );
                }
            }
            elsif ( $ElementName eq 'Project' ) {
                ID:
                for my $ID ( @{$Values} ) {
                    next ID if !$ID;

                    my %TmpProjectData = $Self->{TimeAccountingObject}->ProjectGet( ID => $ID->{Content} );
                    $ID->{Content} = $TmpProjectData{Project};
                }
            }
            elsif ( $ElementName eq 'ProjectAction' ) {
                ID:
                for my $ID ( @{$Values} ) {
                    next ID if !$ID;

                    my %TmpActionData = $Self->{TimeAccountingObject}->ActionGet( ID => $ID->{Content} );
                    $ID->{Content} = $TmpActionData{Action};
                }
            }
        }
    }
    return \%Param;
}

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

    # wrap used spelling to ids
    for my $Use (qw(UseAsValueSeries UseAsRestriction UseAsXvalue)) {
        ELEMENT:
        for my $Element ( @{ $Param{$Use} } ) {
            next ELEMENT if !$Element || !$Element->{SelectedValues};

            my $ElementName = $Element->{Element};
            my $Values      = $Element->{SelectedValues};

            if ( $ElementName eq 'User' ) {
                ID:
                for my $ID ( @{$Values} ) {
                    next ID if !$ID;

                    if ( $Self->{UserObject}->UserLookup( UserLogin => $ID->{Content} ) ) {
                        $ID->{Content} = $Self->{UserObject}->UserLookup( UserLogin => $ID->{Content} );
                    }
                    else {
                        $Self->{LogObject}->Log(
                            Priority => 'error',
                            Message  => "Import: Can' find the user $ID->{Content}!"
                        );
                        $ID = undef;
                    }
                }
            }
            elsif ( $ElementName eq 'Project' ) {
                ID:
                for my $ID ( @{$Values} ) {
                    next ID if !$ID;

                    my %Project = $Self->{TimeAccountingObject}->ProjectGet(
                        Project => $ID->{Content},
                    );
                    if ( $Project{ID} ) {
                        $ID->{Content} = $Project{ID};
                    }
                    else {
                        $Self->{LogObject}->Log(
                            Priority => 'error',
                            Message  => "Import: Can' find project $ID->{Content}!"
                        );
                        $ID = undef;
                    }
                }
            }
            elsif ( $ElementName eq 'ProjectAction' ) {
                ID:
                for my $ID ( @{$Values} ) {
                    next ID if !$ID;

                    my %Action = $Self->{TimeAccountingObject}->ActionGet(
                        Action => $ID->{Content},
                    );
                    if ( $Action{ID} ) {
                        $ID->{Content} = $Action{ID};
                    }
                    else {
                        $Self->{LogObject}->Log(
                            Priority => 'error',
                            Message  => "Import: Can' find action $ID->{Content}!"
                        );
                        $ID = undef;
                    }
                }
            }
        }
    }
    return \%Param;
}

sub _GetStatData {

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

    my @Return;
    my @UserIDs = @{ $Param{UserIDs} || [] };

    my %DateIndexToName = (
        'Second' => 0,
        'Minute' => 1,
        'Hour'   => 2,
        'Day'    => 3,
        'Month'  => 4,
        'Year'   => 5,
    );

    # looping over all or selected users
    for my $UserID (@UserIDs) {

        my $StartDate;
        my $StopDate;

        # check if time period has been selected
        if ( $Param{Param}{Restrictions}{TimeAccountingPeriodStart} ) {

            # get unix timestamp of start and end values
            $StartDate = $Self->{TimeObject}->TimeStamp2SystemTime(
                String => $Param{Param}{Restrictions}{TimeAccountingPeriodStart},
            );
            $StopDate = $Self->{TimeObject}->TimeStamp2SystemTime(
                String => $Param{Param}{Restrictions}{TimeAccountingPeriodStop},
            );
        }
        else {

            # IMPORTANT:
            # If no time period had been selected previous month will be used as period!

            # get current date values
            my @CurrentDate = $Self->{TimeObject}->SystemTime2Date(
                SystemTime => $Self->{TimeObject}->SystemTime(),
            );

            # get first day of previous month
            my @NewStartDate = Add_Delta_YMD(
                $CurrentDate[ $DateIndexToName{'Year'} ],
                $CurrentDate[ $DateIndexToName{'Month'} ],
                1,
                0,
                -1,
                0,
            );

            # get first day of next month relative to previous month
            my @NewStopDate = Add_Delta_YMD(
                $NewStartDate[0],
                $NewStartDate[1],
                $NewStartDate[2],
                0,
                +1,
                0,
            );

            # get last of day previous month
            @NewStopDate = Add_Delta_YMD(
                $NewStopDate[0],
                $NewStopDate[1],
                $NewStopDate[2],
                0,
                0,
                -1,
            );

            # calculate unix timestamp for start and stop date
            $StartDate = $Self->{TimeObject}->Date2SystemTime(
                Year   => $NewStartDate[0],
                Month  => $NewStartDate[1],
                Day    => $NewStartDate[2],
                Hour   => 0,
                Minute => 0,
                Second => 0,
            );
            $StopDate = $Self->{TimeObject}->Date2SystemTime(
                Year   => $NewStopDate[0],
                Month  => $NewStopDate[1],
                Day    => $NewStopDate[2],
                Hour   => 23,
                Minute => 59,
                Second => 59,
            );
        }

        # calculate number of days within the given range
        my $Days = int( ( $StopDate - $StartDate ) / 86400 ) + 1;

        DAY:
        for my $Day ( 0 .. $Days ) {

            # get day relative to start date
            my $DateOfPeriod = $StartDate + $Day * 86400;

            # get needed date values out of timestamp
            my @DateValues = $Self->{TimeObject}->SystemTime2Date(
                SystemTime => $DateOfPeriod,
            );

            # get working unit for user and day
            my %WorkingUnit = $Self->{TimeAccountingObject}->WorkingUnitsGet(
                Year   => $DateValues[ $DateIndexToName{'Year'} ],
                Month  => $DateValues[ $DateIndexToName{'Month'} ],
                Day    => $DateValues[ $DateIndexToName{'Day'} ],
                UserID => $UserID,
            );

            # extract detailed information
            my @DayWorkingUnits = @{ $WorkingUnit{WorkingUnits} || [] };

            # check for project restrictions
            if (
                $Param{Param}{Restrictions}->{Project}
                && ref $Param{Param}{Restrictions}->{Project} eq 'ARRAY'
                )
            {

                # build matching hash for selected projects
                my %SelectedProjectIDs = map { ( $_ => 1 ) } @{ $Param{Param}{Restrictions}->{Project} };

                # filter only selected projects
                my @FilteredProjectWUs = grep {
                    $SelectedProjectIDs{ $_->{ProjectID} }
                } @DayWorkingUnits;

                @DayWorkingUnits = @FilteredProjectWUs;
            }

            # check for task restrictions
            if (
                $Param{Param}{Restrictions}->{ProjectAction}
                && ref $Param{Param}{Restrictions}->{ProjectAction} eq 'ARRAY'
                )
            {

                # build matching hash for selected actions
                my %SelectedActionIDs = map { ( $_ => 1 ) } @{ $Param{Param}{Restrictions}->{ProjectAction} };

                # filter only selected actions
                my @FilteredActionWUs = grep {
                    $SelectedActionIDs{ $_->{ActionID} }
                } @DayWorkingUnits;

                @DayWorkingUnits = @FilteredActionWUs;
            }

            # check for user restrictions
            if (
                $Param{Param}{Restrictions}->{User}
                && ref $Param{Param}{Restrictions}->{User} eq 'ARRAY'
                )
            {

                # build matching hash for selected actions
                my %SelectedUserIDs = map { ( $_ => 1 ) } @{ $Param{Param}{Restrictions}->{User} };

                # filter only selected actions
                my @FilteredUserWUs = grep {
                    $SelectedUserIDs{ $_->{UserID} }
                } @DayWorkingUnits;

                @DayWorkingUnits = @FilteredUserWUs;
            }

            # do not store data if no data is available
            next DAY if !@DayWorkingUnits;

            # add data to global result set
            push @Return, @DayWorkingUnits;
        }
    }

    return \@Return;

}

1;

# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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.
# --

## nofilter(TidyAll::Plugin::OTRS::Perl::PerlCritic)

use strict;
use warnings;

# declare externally defined variables to avoid errors under 'use strict'
use vars qw($Self);

use Kernel::System::TimeAccounting;
use Kernel::System::User;

# create local object
my $UserObject = Kernel::System::User->new( %{$Self} );

# data for new user
my $RandomNumber = int( rand(10000) );
my %UserData     = (
    UserFirstname => 'Unit Test',
    UserLastname  => 'User',
    UserLogin     => 'UnitTest' . $RandomNumber,
    UserPw        => 'pass',
    UserEmail     => 'email@mydomain.com',
    ValidID       => 1,
    ChangeUserID  => 1,
);

# create test user
my $UserID = $UserObject->UserAdd(%UserData);

# crete local object
my $TimeAccountingObject = Kernel::System::TimeAccounting->new(
    %{$Self},
    UserID     => $UserID,
    UserObject => $UserObject,
);

# data for the new action (task)
my %NewActionData = (
    Action       => 'TestAction' . $RandomNumber,
    ActionStatus => 1,
);

# create a new action (task)
my $Insert = $TimeAccountingObject->ActionSettingsInsert(%NewActionData);

# verify that the action was successfully inserted
$Self->True(
    $Insert,
    'Insert test action settings into database',
);

# get the action data that was inserted above
my %ActionData = $TimeAccountingObject->ActionGet( Action => $NewActionData{Action} );

# verify that the data saved in the DB is the same that was inserted
$Self->Is(
    $ActionData{ActionStatus},
    $NewActionData{ActionStatus},
    'Compare action status saved in DB with the inserted one',
);

# modify the action name
my $Update = $TimeAccountingObject->ActionSettingsUpdate(
    ActionID     => $ActionData{ID},
    Action       => $NewActionData{Action} . 'modified',
    ActionStatus => 1,
);

# verify that the action was updated
$Self->True(
    $Update,
    'Update test action settings',
);

# get all actions data
my %AllActions = $TimeAccountingObject->ActionSettingsGet();

# verify the last modification
$Self->Is(
    $AllActions{ $ActionData{ID} }{Action},
    $NewActionData{Action} . 'modified',
    'Compare action name saved in DB with the specified in the update',
);

# data for the new project
my %NewProjectData = (
    Project            => 'TestProject' . $RandomNumber,
    ProjectDescription => 'Test',
    ProjectStatus      => 1,
);

# create a new project
my $ProjectID = $TimeAccountingObject->ProjectSettingsInsert(%NewProjectData);

# verify that the action was successfully inserted
$Self->True(
    $ProjectID,
    'Insert test project settings into database',
);

# get the action data that was inserted above
my %ProjectData = $TimeAccountingObject->ProjectGet( ID => $ProjectID );

# verify that the data saved in the DB is the same that was inserted
$Self->Is(
    $ProjectData{Project},
    $NewProjectData{Project},
    'Compare project name saved in DB with the inserted one',
);
$Self->Is(
    $ProjectData{ProjectDescription},
    $NewProjectData{ProjectDescription},
    'Compare project description saved in DB with the inserted one',
);
$Self->Is(
    $ProjectData{ProjectStatus},
    $NewProjectData{ProjectStatus},
    'Compare project status saved in DB with the inserted one',
);

# modify the project name
$Update = $TimeAccountingObject->ProjectSettingsUpdate(
    ID                 => $ProjectID,
    Project            => $NewProjectData{Project} . 'modified',
    ProjectDescription => 'Test',
    ProjectStatus      => 1,
);

# verify that the action was updated
$Self->True(
    $Update,
    'Update test project settings',
);

# get all projects data
my %AllProjects = $TimeAccountingObject->ProjectSettingsGet( Status => 'valid' );

# verify the last modification
$Self->Is(
    $AllProjects{Project}{$ProjectID},
    $NewProjectData{Project} . 'modified',
    'Compare project name saved in DB with the specified in the update',
);

# obtain the last registered period of the test user
my $LastPeriodNumber = $TimeAccountingObject->UserLastPeriodNumberGet( UserID => $UserID ) + 1;

# create registry for the new user in TA
$Insert = $TimeAccountingObject->UserSettingsInsert(
    UserID => $UserID,
    Period => $LastPeriodNumber,
);

# verify that the user was successfully inserted
$Self->True(
    $Insert,
    'Insert test user settings into database',
);

# insert another period entry for the same user
$Insert = $TimeAccountingObject->UserSettingsInsert(
    UserID => $UserID,
    Period => $LastPeriodNumber + 1,
);

# verify that the user was successfully inserted
$Self->True(
    $Insert,
    'Insert time period for test user settings into database',
);

$RandomNumber = int( rand(100) );

# update user data
$Update = $TimeAccountingObject->UserSettingsUpdate(
    UserID        => $UserID,
    Description   => 'Test user' . $RandomNumber,
    CreateProject => 1,
    ShowOvertime  => 1,
    Period        => {
        1 => {
            DateStart   => '2011-01-01',
            DateEnd     => '2011-01-31',
            WeeklyHours => $RandomNumber,
            LeaveDays   => '10',
            Overtime    => '20',
            UserStatus  => 1,
        },
        2 => {
            DateStart   => '2011-02-01',
            DateEnd     => '2011-12-31',
            WeeklyHours => $RandomNumber + 10,
            LeaveDays   => '5',
            Overtime    => '10',
            UserStatus  => 1,
        },
    },
);

# verify that the action was updated
$Self->True(
    $Update,
    'Update test user settings',
);

# get user settings
my %SingleUserData = $TimeAccountingObject->SingleUserSettingsGet( UserID => $UserID );

# verify the last modification
$Self->Is(
    int $SingleUserData{1}{WeeklyHours},
    $RandomNumber,
    'Compare weekly hours for period 1 saved in DB with the specified in the update',
);

$Self->Is(
    int $SingleUserData{2}{WeeklyHours},
    $RandomNumber + 10,
    'Compare weekly hours for period 2 saved in DB with the specified in the update',
);

# get user data
%SingleUserData = $TimeAccountingObject->UserGet( UserID => $UserID );

# compare data saved on the DB with the inserted one
$Self->Is(
    $SingleUserData{Description},
    'Test user' . $RandomNumber,
    'Compare description saved on the DB with the specified in the insertion',
);

# get all users data
my %AllUsersData = $TimeAccountingObject->UserList();

# verify that the test user is in the list
$Self->True(
    $AllUsersData{$UserID}{Description},
    'Verify the existence of the test user in the list'
);

# get all periods of data of all users
my %AllUsersPeriodData = $TimeAccountingObject->UserSettingsGet();

# verify the period data of the test user
my $CorrectUserData =
    (
    $AllUsersPeriodData{$UserID}{1}{WeeklyHours} == $RandomNumber
        && $AllUsersPeriodData{$UserID}{1}{LeaveDays} == 10
        && $AllUsersPeriodData{$UserID}{1}{Overtime} == 20
        && $AllUsersPeriodData{$UserID}{1}{UserStatus}
        && $AllUsersPeriodData{$UserID}{2}{WeeklyHours} == $RandomNumber + 10
        && $AllUsersPeriodData{$UserID}{2}{LeaveDays} == 5
        && $AllUsersPeriodData{$UserID}{2}{Overtime} == 10
        && $AllUsersPeriodData{$UserID}{2}{UserStatus}
    ) ? 1 : 0;

$Self->True(
    $CorrectUserData,
    'Verify the period data of the test user in the list'
);

# get current period of user
my %UserCurrentPeriod = $TimeAccountingObject->UserCurrentPeriodGet(
    Year  => '2011',
    Month => '01',
    Day   => '15',
);

# check the period
$Self->Is(
    $UserCurrentPeriod{$UserID}->{Period},
    1,
    'Verify current period',
);

$Self->Is(
    $UserCurrentPeriod{$UserID}->{DateStart},
    '2011-01-01',
    'Verify start date of current period',
);

$Self->Is(
    $UserCurrentPeriod{$UserID}->{DateEnd},
    '2011-01-31',
    'Verify end date of current period',
);

# hash with the working units for Jan. 15th, 2011
my %WorkingUnits = (
    Year         => '2011',
    Month        => '01',
    Day          => '15',
    LeaveDay     => 0,
    Sick         => 0,
    Overtime     => 0,
    WorkingUnits => [
        {
            ProjectID => $ProjectID,
            ActionID  => $ActionData{ID},
            Remark    => 'My comment',
            StartTime => '7:00',
            EndTime   => '10:00',
            Period    => 3.0,
        },
        {
            ProjectID => $ProjectID,
            ActionID  => $ActionData{ID},
            Remark    => 'My comment',
            StartTime => '13:00',
            EndTime   => '15:00',
            Period    => 2.0,
        },
    ],
);

# insert working units in the DB
$Insert = $TimeAccountingObject->WorkingUnitsInsert(%WorkingUnits);

# verify that the working units were successfully inserted
$Self->True(
    $Insert,
    'Insert working units for test user into database',
);

# get all days without working units entry
my %WorkingUnitsCheck = $TimeAccountingObject->WorkingUnitsCompletnessCheck(
    UserID => $UserID,
);

# verify that Jan 15th, 2011 is not in the list of days without entry
$Self->False(
    defined $WorkingUnitsCheck{Incomplete}{2011}{01}{15},
    'Verify completion of working units'
);

# get working units of all users
my %Data = $TimeAccountingObject->UserReporting(
    Year  => '2011',
    Month => '01',
    Day   => '15',
);

# verify the correctness of the working units for the test user
$Self->Is(
    $Data{$UserID}{WorkingHoursTotal},
    5,
    'Verify number of working hours of the test user',
);

# get projects in which the test user has worked on
my @LastProjects = $TimeAccountingObject->LastProjectsOfUser();

my $TestProjectExistence;

for my $ID (@LastProjects) {
    next if $ID != $ProjectID;
    $TestProjectExistence = 1;
}

# verify that existence of the test project into the user's list
$Self->True(
    $TestProjectExistence,
    'Verify that existence of the test project into the user\'s list',
);

# get project - action working hours
my %ProjectActionWorkingHours = $TimeAccountingObject->ProjectActionReporting(
    Year   => 2011,
    Month  => 1,
    UserID => $UserID,
);

# verify total reported hours for the test action (task) and test project
$Self->Is(
    $ProjectActionWorkingHours{$ProjectID}{Actions}{ $ActionData{ID} }{Total},
    '5',
    'Verify total reported hours for the test action (task) and test project',
);

# get project working units
my @ProjectHistoryArray = $TimeAccountingObject->ProjectHistory( ProjectID => $ProjectID );

my $TotalHours;

# get sum of all working unit for the test project
for my $Project (@ProjectHistoryArray) {
    $TotalHours += $Project->{Period};
}

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

# verify total working units for the test project
$Self->Is(
    int $TotalHours,
    int $TestProjectTotalHours,
    'Verify total hours of test project',
);

# delete working units for Jan. 15th, 2011
my $Delete = $TimeAccountingObject->WorkingUnitsDelete(
    Year  => '2011',
    Month => '1',
    Day   => '15',
);

# verify that the working units were successfully deleted
$Self->True(
    $Delete,
    'Delete working units for test',
);

# get working units for Jan. 15th, 2011
my %WorkingUnitsData = $TimeAccountingObject->WorkingUnitsGet(
    Year   => '2011',
    Month  => '1',
    Day    => '15',
    UserID => $UserID,
);

# verify inexistence of the deleted working units
$Self->Is(
    $WorkingUnitsData{Total},
    '0',
    'Verify inexistence of the deleted working units',
);

# set to invalid all registries used for the tests
$TimeAccountingObject->ActionSettingsUpdate(
    ActionID     => $ActionData{ID},
    Action       => $NewActionData{Action} . 'modified',
    ActionStatus => 0,
);

$TimeAccountingObject->ProjectSettingsUpdate(
    ID                 => $ProjectID,
    Project            => $NewProjectData{Project} . 'modified',
    ProjectDescription => 'Test',
    ProjectStatus      => 0,
);

$UserData{ValidID} = 2;
$UserData{UserID}  = $UserID;
$UserObject->UserUpdate(%UserData);
$TimeAccountingObject->UserSettingsUpdate(
    UserID        => $UserID,
    Description   => 'Test user',
    CreateProject => 0,
    ShowOvertime  => 0,
    Period        => {
        1 => {
            DateStart   => '2011-01-01',
            DateEnd     => '2011-01-31',
            WeeklyHours => '50',
            LeaveDays   => '10',
            Overtime    => '20',
            UserStatus  => 0,
        },
        2 => {
            DateStart   => '2011-02-01',
            DateEnd     => '2011-02-28',
            WeeklyHours => '30',
            LeaveDays   => '5',
            Overtime    => '10',
            UserStatus  => 0,
        },
    },
);

1;

Ly8gLS0KLy8ganF1ZXJ5LWNvbWJvYm94LmpzIC0gc3BlY2lhbCBqcXVlcnkgdWkgY29tYm9ib3gKLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnL1xuIjsKLy8gLS0KLy8gJElkOiBqcXVlcnktY29tYm9ib3guanMsdiAxLjQgMjAxMi0wMS0zMSAxMzo1NDozOCBtbiBFeHAgJAovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91Ci8vIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCi8vIC0tCgooZnVuY3Rpb24oJCkgewogICAgJC53aWRnZXQoInVpLmNvbWJvYm94IiwgewogICAgICAgIG9wdGlvbnM6IHsKICAgICAgICAgICAgQ2xhc3M6ICcnLAogICAgICAgICAgICBJRFByZWZpeElucHV0OiAnQ29tYm9fJywKICAgICAgICAgICAgSURQcmVmaXhCdXR0b246ICdDb21ib0J0bl8nLAogICAgICAgICAgICBWYWxpZGF0aW9uVG9vbHRpcDogJycsCiAgICAgICAgICAgIExhbmc6IHsKICAgICAgICAgICAgICAgIFNob3dBbGxJdGVtczogJ1Nob3cgYWxsIGl0ZW1zJwogICAgICAgICAgICB9CiAgICAgICAgfSwKICAgICAgICBfY3JlYXRlOiBmdW5jdGlvbigpIHsKICAgICAgICAgICAgdmFyIFNlbGYgPSB0aGlzLAogICAgICAgICAgICAgICAgJFNlbGVjdCA9IHRoaXMuZWxlbWVudC5oaWRlKCksCiAgICAgICAgICAgICAgICAkU2VsZWN0ZWQgPSAkU2VsZWN0LmNoaWxkcmVuKCI6c2VsZWN0ZWQiKSwKICAgICAgICAgICAgICAgIFZhbHVlID0gJFNlbGVjdGVkLnZhbCgpID8gJFNlbGVjdGVkLnRleHQoKSA6ICIiLAogICAgICAgICAgICAgICAgJElucHV0ID0gdGhpcy5pbnB1dCA9ICQoIjxpbnB1dD4iKQogICAgICAgICAgICAgICAgLmFkZENsYXNzKHRoaXMub3B0aW9ucy5DbGFzcykKICAgICAgICAgICAgICAgIC5hdHRyKCdpZCcsIHRoaXMub3B0aW9ucy5JRFByZWZpeElucHV0ICsgJFNlbGVjdFswXS5pZCkKICAgICAgICAgICAgICAgIC5pbnNlcnRBZnRlcigkU2VsZWN0KQogICAgICAgICAgICAgICAgLmNzcygnd2lkdGgnLCAkU2VsZWN0LndpZHRoKCkpCiAgICAgICAgICAgICAgICAudmFsKFZhbHVlKQogICAgICAgICAgICAgICAgLmF1dG9jb21wbGV0ZSh7CiAgICAgICAgICAgICAgICAgICAgZGVsYXk6IDAsCiAgICAgICAgICAgICAgICAgICAgbWluTGVuZ3RoOiAwLAogICAgICAgICAgICAgICAgICAgIHNvdXJjZTogZnVuY3Rpb24oUmVxdWVzdCwgUmVzcG9uc2UpIHsKICAgICAgICAgICAgICAgICAgICAgICAgdmFyIE1hdGNoZXIgPSBuZXcgUmVnRXhwKCQudWkuYXV0b2NvbXBsZXRlLmVzY2FwZVJlZ2V4KFJlcXVlc3QudGVybSksICJpIiApOwoKICAgICAgICAgICAgICAgICAgICAgICAgUmVzcG9uc2UoJFNlbGVjdC5jaGlsZHJlbigib3B0aW9uIikubWFwKGZ1bmN0aW9uKCkgewogICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFyIFRleHQgPSAkKHRoaXMpLnRleHQoKTsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmICh0aGlzLnZhbHVlICYmICghUmVxdWVzdC50ZXJtIHx8IE1hdGNoZXIudGVzdChUZXh0KSkpCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbGFiZWw6IFRleHQucmVwbGFjZSgKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5ldyBSZWdFeHAoCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIig/IVteJjtdKzspKD8hPFtePD5dKikoIiArCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJC51aS5hdXRvY29tcGxldGUuZXNjYXBlUmVnZXgoUmVxdWVzdC50ZXJtKSArCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIikoPyFbXjw+XSo+KSg/IVteJjtdKzspIiwgImdpIgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKSwgIjxzdHJvbmc+JDE8L3N0cm9uZz4iICksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlOiBUZXh0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvcHRpb246IHRoaXMKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9OwogICAgICAgICAgICAgICAgICAgICAgICB9KSk7CiAgICAgICAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgICAgICAgICBzZWxlY3Q6IGZ1bmN0aW9uKEV2ZW50LCBVSSkgewogICAgICAgICAgICAgICAgICAgICAgICBpZiAoVUkuaXRlbS5vcHRpb24uZGlzYWJsZWQpIHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKICAgICAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgICAgICBVSS5pdGVtLm9wdGlvbi5zZWxlY3RlZCA9IHRydWU7CiAgICAgICAgICAgICAgICAgICAgICAgIFNlbGYuX3RyaWdnZXIoInNlbGVjdGVkIiwgRXZlbnQsIHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGl0ZW06IFVJLml0ZW0ub3B0aW9uCiAgICAgICAgICAgICAgICAgICAgICAgIH0pOwogICAgICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICAgICAgICAgb3BlbjogZnVuY3Rpb24oRXZlbnQsIFVJKSB7CiAgICAgICAgICAgICAgICAgICAgICAgICQoJ3VsLnVpLWF1dG9jb21wbGV0ZS51aS1tZW51JykuZWFjaChmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICB2YXIgd2lkdGggPSAkKHRoaXMpLndpZHRoKCk7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAkKHRoaXMpLndpZHRoKHdpZHRoICsgMzApOwogICAgICAgICAgICAgICAgICAgICAgICB9KTsKICAgICAgICAgICAgICAgICAgICB9LAogICAgICAgICAgICAgICAgICAgIGNoYW5nZTogZnVuY3Rpb24oRXZlbnQsIFVJKSB7CiAgICAgICAgICAgICAgICAgICAgICAgIGlmICghVUkuaXRlbSkgewogICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFyIE1hdGNoZXIgPSBuZXcgUmVnRXhwKCAiXiIgKyAkLnVpLmF1dG9jb21wbGV0ZS5lc2NhcGVSZWdleCggJCh0aGlzKS52YWwoKSApICsgIiQiLCAiaSIgKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBWYWxpZCA9IGZhbHNlOwogICAgICAgICAgICAgICAgICAgICAgICAgICAgJFNlbGVjdC5jaGlsZHJlbigib3B0aW9uIikuZWFjaChmdW5jdGlvbigpIHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoJCh0aGlzKS50ZXh0KCkubWF0Y2goTWF0Y2hlcikpIHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5zZWxlY3RlZCA9IFZhbGlkID0gdHJ1ZTsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0pOwogICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCFWYWxpZCkgewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIHJlbW92ZSBpbnZhbGlkIHZhbHVlLCBhcyBpdCBkaWRuJ3QgbWF0Y2ggYW55dGhpbmcKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAkKHRoaXMpLnZhbCgiIik7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJFNlbGVjdC52YWwoIiIpOwogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICRJbnB1dC5kYXRhKCJhdXRvY29tcGxldGUiKS50ZXJtID0gIiI7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwogICAgICAgICAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgfSkKICAgICAgICAgICAgICAgIC5hZGRDbGFzcygidWktd2lkZ2V0IHVpLXdpZGdldC1jb250ZW50IHVpLWNvcm5lci1sZWZ0IikKICAgICAgICAgICAgICAgIC5hZnRlcih0aGlzLm9wdGlvbnMuVmFsaWRhdGlvblRvb2x0aXApOwoKICAgICAgICAgICAgJElucHV0LmRhdGEoImF1dG9jb21wbGV0ZSIpLl9yZW5kZXJJdGVtID0gZnVuY3Rpb24oVUwsIEl0ZW0pIHsKICAgICAgICAgICAgICAgIHJldHVybiAkKCI8bGk+PC9saT4iKQogICAgICAgICAgICAgICAgICAgIC5kYXRhKCJpdGVtLmF1dG9jb21wbGV0ZSIsIEl0ZW0pCiAgICAgICAgICAgICAgICAgICAgLmFwcGVuZCgiPGE+IiArIEl0ZW0ubGFiZWwgKyAiPC9hPiIpCiAgICAgICAgICAgICAgICAgICAgLmFwcGVuZFRvKFVMKTsKICAgICAgICAgICAgfTsKCiAgICAgICAgICAgIHRoaXMuYnV0dG9uID0gJCgiPGJ1dHRvbiB0eXBlPSdidXR0b24nPiZuYnNwOzwvYnV0dG9uPiIpCiAgICAgICAgICAgICAgICAuYXR0cigidGFiSW5kZXgiLCAtMSkKICAgICAgICAgICAgICAgIC5hdHRyKCJ0aXRsZSIsIHRoaXMub3B0aW9ucy5MYW5nLlNob3dBbGxJdGVtcykKICAgICAgICAgICAgICAgIC5hdHRyKCdpZCcsIHRoaXMub3B0aW9ucy5JRFByZWZpeEJ1dHRvbiArICRTZWxlY3RbMF0uaWQpCiAgICAgICAgICAgICAgICAuaW5zZXJ0QWZ0ZXIoJElucHV0KQogICAgICAgICAgICAgICAgLmJ1dHRvbih7CiAgICAgICAgICAgICAgICAgICAgaWNvbnM6IHsKICAgICAgICAgICAgICAgICAgICAgICAgcHJpbWFyeTogInVpLWljb24tdHJpYW5nbGUtMS1zIgogICAgICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICAgICAgICAgdGV4dDogZmFsc2UKICAgICAgICAgICAgICAgIH0pCiAgICAgICAgICAgICAgICAucmVtb3ZlQ2xhc3MoInVpLWNvcm5lci1hbGwiKQogICAgICAgICAgICAgICAgLmFkZENsYXNzKCJ1aS1jb3JuZXItcmlnaHQgdWktYnV0dG9uLWljb24iKQogICAgICAgICAgICAgICAgLmNsaWNrKGZ1bmN0aW9uKEV2ZW50KSB7CiAgICAgICAgICAgICAgICAgICAgLy8gY2xvc2UgaWYgYWxyZWFkeSB2aXNpYmxlCiAgICAgICAgICAgICAgICAgICAgaWYgKCRJbnB1dC5hdXRvY29tcGxldGUoIndpZGdldCIpLmlzKCI6dmlzaWJsZSIpKSB7CiAgICAgICAgICAgICAgICAgICAgICAgICRJbnB1dC5hdXRvY29tcGxldGUoImNsb3NlIik7CiAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybjsKICAgICAgICAgICAgICAgICAgICB9CgogICAgICAgICAgICAgICAgICAgIC8vIHBhc3MgZW1wdHkgc3RyaW5nIGFzIHZhbHVlIHRvIHNlYXJjaCBmb3IsIGRpc3BsYXlpbmcgYWxsIHJlc3VsdHMKICAgICAgICAgICAgICAgICAgICAkSW5wdXQuYXV0b2NvbXBsZXRlKCJzZWFyY2giLCAiIikuZm9jdXMoKTsKICAgICAgICAgICAgICAgICAgICBFdmVudC5wcmV2ZW50RGVmYXVsdCgpOwogICAgICAgICAgICAgICAgICAgIEV2ZW50LnN0b3BQcm9wYWdhdGlvbigpOwogICAgICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKICAgICAgICAgICAgICAgIH0pOwogICAgICAgIH0sCgogICAgICAgIGRlc3Ryb3k6IGZ1bmN0aW9uKCkgewogICAgICAgICAgICB0aGlzLmlucHV0LnJlbW92ZSgpOwogICAgICAgICAgICB0aGlzLmJ1dHRvbi5yZW1vdmUoKTsKICAgICAgICAgICAgdGhpcy5lbGVtZW50LnNob3coKTsKICAgICAgICAgICAgJC5XaWRnZXQucHJvdG90eXBlLmRlc3Ryb3kuY2FsbCh0aGlzKTsKICAgICAgICB9CiAgICB9KTsKfSkoalF1ZXJ5KTs=
/*
 * jQuery UI Button 1.8.16
 *
 * Copyright 2011, AUTHORS.txt (http://jqueryui.com/about)
 * Dual licensed under the MIT or GPL Version 2 licenses.
 * http://jquery.org/license
 *
 * http://docs.jquery.com/UI/Button
 *
 * Depends:
 *	jquery.ui.core.js
 *	jquery.ui.widget.js
 */
(function(b){var h,i,j,g,l=function(){var a=b(this).find(":ui-button");setTimeout(function(){a.button("refresh")},1)},k=function(a){var c=a.name,e=a.form,f=b([]);if(c)f=e?b(e).find("[name='"+c+"']"):b("[name='"+c+"']",a.ownerDocument).filter(function(){return!this.form});return f};b.widget("ui.button",{options:{disabled:null,text:true,label:null,icons:{primary:null,secondary:null}},_create:function(){this.element.closest("form").unbind("reset.button").bind("reset.button",l);if(typeof this.options.disabled!==
"boolean")this.options.disabled=this.element.propAttr("disabled");this._determineButtonType();this.hasTitle=!!this.buttonElement.attr("title");var a=this,c=this.options,e=this.type==="checkbox"||this.type==="radio",f="ui-state-hover"+(!e?" ui-state-active":"");if(c.label===null)c.label=this.buttonElement.html();if(this.element.is(":disabled"))c.disabled=true;this.buttonElement.addClass("ui-button ui-widget ui-state-default ui-corner-all").attr("role","button").bind("mouseenter.button",function(){if(!c.disabled){b(this).addClass("ui-state-hover");
this===h&&b(this).addClass("ui-state-active")}}).bind("mouseleave.button",function(){c.disabled||b(this).removeClass(f)}).bind("click.button",function(d){if(c.disabled){d.preventDefault();d.stopImmediatePropagation()}});this.element.bind("focus.button",function(){a.buttonElement.addClass("ui-state-focus")}).bind("blur.button",function(){a.buttonElement.removeClass("ui-state-focus")});if(e){this.element.bind("change.button",function(){g||a.refresh()});this.buttonElement.bind("mousedown.button",function(d){if(!c.disabled){g=
false;i=d.pageX;j=d.pageY}}).bind("mouseup.button",function(d){if(!c.disabled)if(i!==d.pageX||j!==d.pageY)g=true})}if(this.type==="checkbox")this.buttonElement.bind("click.button",function(){if(c.disabled||g)return false;b(this).toggleClass("ui-state-active");a.buttonElement.attr("aria-pressed",a.element[0].checked)});else if(this.type==="radio")this.buttonElement.bind("click.button",function(){if(c.disabled||g)return false;b(this).addClass("ui-state-active");a.buttonElement.attr("aria-pressed","true");
var d=a.element[0];k(d).not(d).map(function(){return b(this).button("widget")[0]}).removeClass("ui-state-active").attr("aria-pressed","false")});else{this.buttonElement.bind("mousedown.button",function(){if(c.disabled)return false;b(this).addClass("ui-state-active");h=this;b(document).one("mouseup",function(){h=null})}).bind("mouseup.button",function(){if(c.disabled)return false;b(this).removeClass("ui-state-active")}).bind("keydown.button",function(d){if(c.disabled)return false;if(d.keyCode==b.ui.keyCode.SPACE||
d.keyCode==b.ui.keyCode.ENTER)b(this).addClass("ui-state-active")}).bind("keyup.button",function(){b(this).removeClass("ui-state-active")});this.buttonElement.is("a")&&this.buttonElement.keyup(function(d){d.keyCode===b.ui.keyCode.SPACE&&b(this).click()})}this._setOption("disabled",c.disabled);this._resetButton()},_determineButtonType:function(){this.type=this.element.is(":checkbox")?"checkbox":this.element.is(":radio")?"radio":this.element.is("input")?"input":"button";if(this.type==="checkbox"||this.type===
"radio"){var a=this.element.parents().filter(":last"),c="label[for='"+this.element.attr("id")+"']";this.buttonElement=a.find(c);if(!this.buttonElement.length){a=a.length?a.siblings():this.element.siblings();this.buttonElement=a.filter(c);if(!this.buttonElement.length)this.buttonElement=a.find(c)}this.element.addClass("ui-helper-hidden-accessible");(a=this.element.is(":checked"))&&this.buttonElement.addClass("ui-state-active");this.buttonElement.attr("aria-pressed",a)}else this.buttonElement=this.element},
widget:function(){return this.buttonElement},destroy:function(){this.element.removeClass("ui-helper-hidden-accessible");this.buttonElement.removeClass("ui-button ui-widget ui-state-default ui-corner-all ui-state-hover ui-state-active  ui-button-icons-only ui-button-icon-only ui-button-text-icons ui-button-text-icon-primary ui-button-text-icon-secondary ui-button-text-only").removeAttr("role").removeAttr("aria-pressed").html(this.buttonElement.find(".ui-button-text").html());this.hasTitle||this.buttonElement.removeAttr("title");
b.Widget.prototype.destroy.call(this)},_setOption:function(a,c){b.Widget.prototype._setOption.apply(this,arguments);if(a==="disabled")c?this.element.propAttr("disabled",true):this.element.propAttr("disabled",false);else this._resetButton()},refresh:function(){var a=this.element.is(":disabled");a!==this.options.disabled&&this._setOption("disabled",a);if(this.type==="radio")k(this.element[0]).each(function(){b(this).is(":checked")?b(this).button("widget").addClass("ui-state-active").attr("aria-pressed",
"true"):b(this).button("widget").removeClass("ui-state-active").attr("aria-pressed","false")});else if(this.type==="checkbox")this.element.is(":checked")?this.buttonElement.addClass("ui-state-active").attr("aria-pressed","true"):this.buttonElement.removeClass("ui-state-active").attr("aria-pressed","false")},_resetButton:function(){if(this.type==="input")this.options.label&&this.element.val(this.options.label);else{var a=this.buttonElement.removeClass("ui-button-icons-only ui-button-icon-only ui-button-text-icons ui-button-text-icon-primary ui-button-text-icon-secondary ui-button-text-only"),
c=b("<span></span>").addClass("ui-button-text").html(this.options.label).appendTo(a.empty()).text(),e=this.options.icons,f=e.primary&&e.secondary,d=[];if(e.primary||e.secondary){if(this.options.text)d.push("ui-button-text-icon"+(f?"s":e.primary?"-primary":"-secondary"));e.primary&&a.prepend("<span class='ui-button-icon-primary ui-icon "+e.primary+"'></span>");e.secondary&&a.append("<span class='ui-button-icon-secondary ui-icon "+e.secondary+"'></span>");if(!this.options.text){d.push(f?"ui-button-icons-only":
"ui-button-icon-only");this.hasTitle||a.attr("title",c)}}else d.push("ui-button-text-only");a.addClass(d.join(" "))}}});b.widget("ui.buttonset",{options:{items:":button, :submit, :reset, :checkbox, :radio, a, :data(button)"},_create:function(){this.element.addClass("ui-buttonset")},_init:function(){this.refresh()},_setOption:function(a,c){a==="disabled"&&this.buttons.button("option",a,c);b.Widget.prototype._setOption.apply(this,arguments)},refresh:function(){var a=this.element.css("direction")===
"ltr";this.buttons=this.element.find(this.options.items).filter(":ui-button").button("refresh").end().not(":ui-button").button().end().map(function(){return b(this).button("widget")[0]}).removeClass("ui-corner-all ui-corner-left ui-corner-right").filter(":first").addClass(a?"ui-corner-left":"ui-corner-right").end().filter(":last").addClass(a?"ui-corner-right":"ui-corner-left").end().end()},destroy:function(){this.element.removeClass("ui-buttonset");this.buttons.map(function(){return b(this).button("widget")[0]}).removeClass("ui-corner-left ui-corner-right").end().button("destroy");
b.Widget.prototype.destroy.call(this)}})})(jQuery);
;
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMTcgT1RSUyBBRywgaHR0cDovL290cnMuY29tL1xuIjsKLy8gLS0KLy8gVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKLy8gdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQovLyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgovLyAtLQoKInVzZSBzdHJpY3QiOwoKdmFyIENvcmUgPSBDb3JlIHx8IHt9OwpDb3JlLlVJID0gQ29yZS5VSSB8fCB7fTsKCi8qKgogKiBAbmFtZXNwYWNlCiAqIEBleHBvcnRzIFRhcmdldE5TIGFzIENvcmUuVUkuQ29tYm9Cb3gKICogQGRlc2NyaXB0aW9uCiAqICAgICAgVGhpcyBuYW1lc3BhY2UgY29udGFpbnMgdGhlIGZ1bmN0aW9ucyBmb3IgdGhlIGNvbWJvYm94IGVsZW1lbnQuCiAqLwpDb3JlLlVJLkNvbWJvQm94ID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewogICAgLyoqCiAgICAgKiBAZnVuY3Rpb24KICAgICAqIEBwYXJhbSB7U3RyaW5nfSBTZWxlY3RvciBUaGUgalF1ZXJ5IHNlbGVjdG9yIGZvciB0aGUgZWxlbWVudHMgdGhhdCBzaG91bGQgYmUgaW5pdGlhbGl6ZWQgYXMgY29tYm9ib3hlcwogICAgICogQHBhcmFtIHtPYmplY3R9IE9wdGlvbnMgVGhlIGFkZGl0aW9uYWwgb3B0aW9ucyBmb3IgdGhlIGNvbWJvYm94OgogICAgICogICAgICAgICAgICAgICAgICAgICAgQ2xhc3M6IGEgc3RyaW5nIHdpdGggdGhlIGNsYXNzKGVzKSB0byBiZSBhZGRlZCB0byB0aGUgaW5wdXQgZmllbGQgb2YgdGhlIGNvbWJvYm94CiAgICAgKiAgICAgICAgICAgICAgICAgICAgICBMYW5nOiBhbiBvYmplY3Qgd2l0aCBrZXkvdmFsdWUgcGFpcnMgb2YgdHJhbnNsYXRpb25zIGZvciB0aGUgY29tYm9ib3gKICAgICAqIEBkZXNjcmlwdGlvbgogICAgICogICAgICBUaGlzIGZ1bmN0aW9uIGluaXRpYWxpemVzIHRoZSBjb21ib2JveGVzCiAgICAgKiBAcmV0dXJuIG5vdGhpbmcKICAgICAqLwogICAgVGFyZ2V0TlMuSW5pdCA9IGZ1bmN0aW9uIChTZWxlY3RvciwgT3B0aW9ucykgewogICAgICAgIFNlbGVjdG9yID0gU2VsZWN0b3IgfHwgJ3NlbGVjdCc7CiAgICAgICAgJChTZWxlY3RvcikuY29tYm9ib3goT3B0aW9ucyk7CiAgICB9OwoKICAgIHJldHVybiBUYXJnZXROUzsKfShDb3JlLlVJLkNvbWJvQm94IHx8IHt9KSk7Cg==
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMTcgT1RSUyBBRywgaHR0cDovL290cnMuY29tL1xuIjsKLy8gLS0KLy8gVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKLy8gdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQovLyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgovLyAtLQoKInVzZSBzdHJpY3QiOwoKdmFyIFRpbWVBY2NvdW50aW5nID0gVGltZUFjY291bnRpbmcgfHwge307ClRpbWVBY2NvdW50aW5nLkFnZW50ID0gVGltZUFjY291bnRpbmcuQWdlbnQgfHwge307CgovKioKICogQG5hbWVzcGFjZQogKiBAZXhwb3J0cyBUYXJnZXROUyBhcyBUaW1lQWNjb3VudGluZy5BZ2VudC5Db25maXJtYXRpb25EaWFsb2cKICogQGRlc2NyaXB0aW9uCiAqICAgICAgVGhpcyBuYW1lc3BhY2UgY29udGFpbnMgdGhlIHNwZWNpYWwgbW9kdWxlIGZ1bmN0aW9ucyBmb3IgQ29uZmlybWF0aW9uRGlhbG9nLgogKi8KVGltZUFjY291bnRpbmcuQWdlbnQuQ29uZmlybWF0aW9uRGlhbG9nID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewoKICAgIC8qKgogICAgICogQHZhcmlhYmxlCiAgICAgKiBAcHJpdmF0ZQogICAgICogICAgIFRoaXMgdmFyaWFibGUgc3RvcmVzIHRoZSBwYXJhbWV0ZXJzIHRoYXQgYXJlIHBhc3NlZCBmcm9tIHRoZSBEVEwgYW5kIGNvbnRhaW4gYWxsIHRoZSBkYXRhIHRoYXQgdGhlIGRpYWxvZyBuZWVkcy4KICAgICAqLwogICAgdmFyIERpYWxvZ0RhdGEgPSBbXTsKCiAgICAvKioKICAgICAqIEBmdW5jdGlvbgogICAgICogQHByaXZhdGUKICAgICAqIEByZXR1cm4gbm90aGluZwogICAgICogQGRlc2NyaXB0aW9uIFNob3dzIHdhaXRpbmcgZGlhbG9nIHVudGlsIHNlYXJjaCBzY3JlZW4gaXMgcmVhZHkuCiAgICAgKi8KICAgIGZ1bmN0aW9uIFNob3dXYWl0aW5nRGlhbG9nKFBvc2l0aW9uVG9wKXsKICAgICAgICBDb3JlLlVJLkRpYWxvZy5TaG93Q29udGVudERpYWxvZygnPGRpdiBjbGFzcz0iU3BhY2luZyBDZW50ZXIiPjxzcGFuIGNsYXNzPSJBSkFYTG9hZGVyIiB0aXRsZT0iJyArIENvcmUuQ29uZmlnLkdldCgnTG9hZGluZ01zZycpICsgJyI+PC9zcGFuPjwvZGl2PicsICcnLCBQb3NpdGlvblRvcCwgJ0NlbnRlcicsIHRydWUpOwogICAgfQoKICAgIC8qKgogICAgICogQGZ1bmN0aW9uCiAgICAgKiBAcGFyYW0ge0V2ZW50T2JqZWN0fSBldmVudCBvYmplY3Qgb2YgdGhlIGNsaWNrZWQgZWxlbWVudC4KICAgICAqIEByZXR1cm4gbm90aGluZwogICAgICogICAgICBUaGlzIGZ1bmN0aW9uIHNob3dzIGEgY29uZmlybWF0aW9uIGRpYWxvZyB3aXRoIDIgYnV0dG9uczogWWVzIGFuZCBObyBvciBhIG1lc3NhZ2UgZGlhbG9nIHdpdGggb25lIGJ1dHRvbjogT2suCiAgICAgKi8KICAgIFRhcmdldE5TLlNob3dDb25maXJtYXRpb25EaWFsb2cgPSBmdW5jdGlvbiAoRXZlbnQpIHsKCiAgICAgICAgdmFyIExvY2FsRGlhbG9nRGF0YSwKICAgICAgICAgICAgUG9zaXRpb25Ub3AsCiAgICAgICAgICAgIERhdGEsCiAgICAgICAgICAgIEJ1dHRvbnM7CgogICAgICAgIC8vIGdldCBnbG9iYWwgc2F2ZWQgRGlhbG9nRGF0YSBmb3IgdGhpcyBmdW5jdGlvbgogICAgICAgIExvY2FsRGlhbG9nRGF0YSA9IERpYWxvZ0RhdGFbJChFdmVudC50YXJnZXQpLmF0dHIoJ2lkJyldOwoKICAgICAgICAvLyBkZWZpbmUgdGhlIHBvc2l0aW9uIG9mIHRoZSBkaWFsb2cKICAgICAgICBQb3NpdGlvblRvcCA9ICQod2luZG93KS5oZWlnaHQoKSAqIDAuMzsKCiAgICAgICAgLy8gc2hvdyB3YWl0aW5nIGRpYWxvZwogICAgICAgIFNob3dXYWl0aW5nRGlhbG9nKFBvc2l0aW9uVG9wKTsKCiAgICAgICAgLy8gYWpheCBjYWxsIHRvIHRoZSBtb2R1bGUgdGhhdCBleGVjdXRlcyB0aGUgYWN0aW9uIHdoZW4gcHJlc3NpbmcgdGhlIGNvbmZpcm1hdGlvbiBidXR0b24KICAgICAgICBEYXRhID0gTG9jYWxEaWFsb2dEYXRhLkRpYWxvZ0NvbnRlbnRRdWVyeVN0cmluZzsKICAgICAgICBDb3JlLkFKQVguRnVuY3Rpb25DYWxsKENvcmUuQ29uZmlnLkdldCgnQmFzZWxpbmsnKSwgRGF0YSwgZnVuY3Rpb24gKFJlc3BvbnNlKSB7CgogICAgICAgICAgICAvLyAnQ29uZmlybWF0aW9uJyBvcGVucyBhIGRpYWxvZyB3aXRoIDIgYnV0dG9uczogWWVzIGFuZCBObwogICAgICAgICAgICBpZiAoUmVzcG9uc2UuRGlhbG9nVHlwZSA9PT0gJ0NvbmZpcm1hdGlvbicpIHsKCiAgICAgICAgICAgICAgICAvLyBkZWZpbmUgeWVzIGFuZCBubyBidXR0b25zCiAgICAgICAgICAgICAgICBCdXR0b25zID0gW3sKICAgICAgICAgICAgICAgICAgICBMYWJlbDogTG9jYWxEaWFsb2dEYXRhLlRyYW5zbGF0ZWRUZXh0LlllcywKICAgICAgICAgICAgICAgICAgICBDbGFzczogIlByaW1hcnkiLAoKICAgICAgICAgICAgICAgICAgICAvLyBkZWZpbmUgdGhlIGZ1bmN0aW9uIHRoYXQgaXMgY2FsbGVkIHdoZW4gdGhlICdZZXMnIGJ1dHRvbiBpcyBwcmVzc2VkCiAgICAgICAgICAgICAgICAgICAgRnVuY3Rpb246IGZ1bmN0aW9uKCl7CgogICAgICAgICAgICAgICAgICAgICAgICAvLyBkaXNhYmxlIFllcyBhbmQgTm8gYnV0dG9ucyB0byBwcmV2ZW50IG11bHRpcGxlIHN1Ym1pdHMKICAgICAgICAgICAgICAgICAgICAgICAgJCgnZGl2LkRpYWxvZzp2aXNpYmxlIGRpdi5Db250ZW50Rm9vdGVyIGJ1dHRvbicpLmF0dHIoJ2Rpc2FibGVkJywgJ2Rpc2FibGVkJyk7CgogICAgICAgICAgICAgICAgICAgICAgICAvLyByZWRpcmVjdCB0byB0aGUgbW9kdWxlIHRoYXQgZG9lcyB0aGUgY29uZmlybWVkIGFjdGlvbiBhZnRlciBwcmVzc2luZyB0aGUgWWVzIGJ1dHRvbgogICAgICAgICAgICAgICAgICAgICAgICBsb2NhdGlvbi5ocmVmID0gQ29yZS5Db25maWcuR2V0KCdCYXNlbGluaycpICsgTG9jYWxEaWFsb2dEYXRhLkNvbmZpcm1lZEFjdGlvblF1ZXJ5U3RyaW5nOwogICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIH0sIHsKICAgICAgICAgICAgICAgICAgICBMYWJlbDogTG9jYWxEaWFsb2dEYXRhLlRyYW5zbGF0ZWRUZXh0Lk5vLAogICAgICAgICAgICAgICAgICAgIFR5cGU6ICJDbG9zZSIKICAgICAgICAgICAgICAgIH1dOwogICAgICAgICAgICB9CgogICAgICAgICAgICAvLyAnTWVzc2FnZScgb3BlbnMgYSBkaWFsb2cgd2l0aCAxIGJ1dHRvbjogT2sKICAgICAgICAgICAgZWxzZSBpZiAoUmVzcG9uc2UuRGlhbG9nVHlwZSA9PT0gJ01lc3NhZ2UnKSB7CgogICAgICAgICAgICAgICAgLy8gZGVmaW5lIE9rIGJ1dHRvbgogICAgICAgICAgICAgICAgQnV0dG9ucyA9IFt7CiAgICAgICAgICAgICAgICAgICAgTGFiZWw6IExvY2FsRGlhbG9nRGF0YS5UcmFuc2xhdGVkVGV4dC5PaywKICAgICAgICAgICAgICAgICAgICBDbGFzczogIlByaW1hcnkiLAogICAgICAgICAgICAgICAgICAgIFR5cGU6ICJDbG9zZSIKICAgICAgICAgICAgICAgIH1dOwogICAgICAgICAgICB9CgogICAgICAgICAgICAvLyBzaG93IHRoZSBjb25maXJtYXRpb24gZGlhbG9nIHRvIGNvbmZpcm0gdGhlIGFjdGlvbgogICAgICAgICAgICBDb3JlLlVJLkRpYWxvZy5TaG93Q29udGVudERpYWxvZyhSZXNwb25zZS5IVE1MLCBMb2NhbERpYWxvZ0RhdGEuRGlhbG9nVGl0bGUsIFBvc2l0aW9uVG9wLCAiQ2VudGVyIiwgdHJ1ZSwgQnV0dG9ucyk7CiAgICAgICAgfSwgJ2pzb24nKTsKICAgICAgICByZXR1cm4gZmFsc2U7CiAgICB9OwoKICAgIC8qKgogICAgICogQGZ1bmN0aW9uCiAgICAgKiBAcGFyYW0ge0V2ZW50T2JqZWN0fSBldmVudCBvYmplY3Qgb2YgdGhlIGNsaWNrZWQgZWxlbWVudC4KICAgICAqIEByZXR1cm4gbm90aGluZwogICAgICogICAgICBUaGlzIGZ1bmN0aW9uIHNob3dzIGEgY29uZmlybWF0aW9uIGRpYWxvZyB3aXRoIDIgYnV0dG9uczogWWVzIGFuZCBObwogICAgICovCiAgICBUYXJnZXROUy5CaW5kQ29uZmlybWF0aW9uRGlhbG9nID0gZnVuY3Rpb24gKERhdGEpIHsKICAgICAgICBEaWFsb2dEYXRhW0RhdGEuRWxlbWVudElEXSA9IERhdGE7CgogICAgICAgIC8vIGJpbmRpbmcgYSBjbGljayBldmVudCB0byB0aGUgZGVmaW5lZCBlbGVtZW50CiAgICAgICAgJChEaWFsb2dEYXRhW0RhdGEuRWxlbWVudElEXS5FbGVtZW50U2VsZWN0b3IpLmJpbmQoJ2NsaWNrJywgVGltZUFjY291bnRpbmcuQWdlbnQuQ29uZmlybWF0aW9uRGlhbG9nLlNob3dDb25maXJtYXRpb25EaWFsb2cpOwogICAgfTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oVGltZUFjY291bnRpbmcuQWdlbnQuQ29uZmlybWF0aW9uRGlhbG9nIHx8IHt9KSk7Cg==
// --
// Copyright (C) 2001-2017 OTRS AG, http://otrs.com/\n";
// --
// 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.
// --

"use strict";

var TimeAccounting = TimeAccounting || {};
TimeAccounting.Agent = TimeAccounting.Agent || {};

/**
 * @namespace
 * @exports TargetNS as TimeAccounting.Agent.Datepicker
 * @description
 *      This namespace contains the special datepicker functiond for TA.
 */
TimeAccounting.Agent.Datepicker = (function (TargetNS) {

    var VacationDays,
        VacationDaysOneTime,
        LocalizationData,
        DatepickerCount = 0;

    if (!Core.Debug.CheckDependency('Core.UI.Datepicker', '$([]).datepicker', 'jQuery UI datepicker')) {
        return;
    }

    /**
     * @function
     * @private
     * @param A boolean value
     * @param {jQueryObject} Element that will be checked
     * @description Review if a date object have correct values
     */
    function CheckDate(DateObject) {
        var DayDescription = '',
            DayClass = '';

        // Get defined days from Config, if not done already
        if (typeof VacationDays === 'undefined') {
            VacationDays = Core.Config.Get('Datepicker.VacationDays').TimeVacationDays;
        }
        if (typeof VacationDaysOneTime === 'undefined') {
            VacationDaysOneTime = Core.Config.Get('Datepicker.VacationDays').TimeVacationDaysOneTime;
        }

        // Check if date is one of the vacation days
        if (typeof VacationDays[DateObject.getMonth() + 1] !== 'undefined' &&
            typeof VacationDays[DateObject.getMonth() + 1][DateObject.getDate()] !== 'undefined') {
            DayDescription += VacationDays[DateObject.getMonth() + 1][DateObject.getDate()];
            DayClass = 'Highlight ';
        }

        // Check if date is one of the one time vacation days
        if (typeof VacationDaysOneTime[DateObject.getFullYear()] !== 'undefined' &&
            typeof VacationDaysOneTime[DateObject.getFullYear()][DateObject.getMonth() + 1] !== 'undefined' &&
            typeof VacationDaysOneTime[DateObject.getFullYear()][DateObject.getMonth() + 1][DateObject.getDate()] !== 'undefined') {
            DayDescription += VacationDaysOneTime[DateObject.getFullYear()][DateObject.getMonth() + 1][DateObject.getDate()];
            DayClass = 'Highlight ';
        }

        if (DayClass.length) {
            return [true, DayClass, DayDescription];
        }
        else {
            return [true, ''];
        }
    }

    /**
     * @function
     * @description
     *      This function initializes the datepicker on single elements for TA.
     *      This function is not yet available in the OTRS Core.
     *      If it will be implemented in OTRS Core we could remove this file here.
     * @param {Object} $Element The jQuery object(s) of a text input field which should get a datepicker.
     * @param {Object} DatepickerOptions Object with two possible keys:
     *                      DateInFuture: true|false,
     *                      WeekDayStart: 0-7 (sunday - saturday)
     * @return nothing
     */
    TargetNS.Init = function ($Element, DatepickerOptions) {
        function LeadingZero(Number) {
            if (Number.toString().length === 1) {
                return '0' + Number;
            }
            else {
                return Number;
            }
        }

        var $DatepickerElement,
            Options,
            I,
            ErrorMessage;

        if (typeof LocalizationData === 'undefined') {
            LocalizationData = Core.Config.Get('Datepicker.Localization');
            if (typeof LocalizationData === 'undefined') {
                throw new Core.Exception.ApplicationError('Datepicker localization data could not be found!', 'InternalError');
            }
        }

        // Define options hash
        Options = {
                beforeShowDay: function (DateObject) {
                    return CheckDate(DateObject);
                },
                showOn: 'focus',
                prevText: LocalizationData.PrevText,
                nextText: LocalizationData.NextText,
                showMonthAfterYear: 0,
                dateFormat: 'yy-mm-dd',
                monthNames: LocalizationData.MonthNames,
                monthNamesShort: LocalizationData.MonthNamesShort,
                dayNames: LocalizationData.DayNames,
                dayNamesShort: LocalizationData.DayNamesShort,
                dayNamesMin: LocalizationData.DayNamesMin,
                isRTL: LocalizationData.IsRTL
        };

        Options.beforeShow = function (Input, Instance) {
            var Value = $(Input).prevAll('input:text').val();
            $(Input).val('');
            return {
                defaultDate: new Date(Value)
            };
        };

        Options.onSelect = function (DateText, Instance) {
            $(this).prevAll('input:text').val(DateText);
        };

        if (isJQueryObject($Element)) {
            $Element.each(function () {
                var $SingleElement = $(this);
                if ($SingleElement.is('input:text')) {
                    // Increment number of initialized datepickers on this site
                    DatepickerCount++;

                    $DatepickerElement = $('<input>').attr('type', 'hidden').attr('id', 'Datepicker' + DatepickerCount);
                    $SingleElement.after($DatepickerElement);

                    Options.firstDay = DatepickerOptions.WeekDayStart;
                    $DatepickerElement.datepicker(Options);

                    // add datepicker icon and click event
                    $DatepickerElement.after('<a href="#" class="DatepickerIcon DatepickerNumber' + DatepickerCount + '" title="' + LocalizationData.IconText + '"></a>').next('a.DatepickerIcon').click(function () {
                        var Classes = $(this).attr('class'),
                            DatepickerNumber = Classes.replace(/^.*DatepickerNumber([0-9]+).*/g, "$1");

                        $('#Datepicker' + DatepickerNumber).trigger('focus');
                        return false;
                    });

                    // do not show the datepicker container div.
                    $('#ui-datepicker-div').hide();
                }
            });
        }
    };

    return TargetNS;
}(TimeAccounting.Agent.Datepicker || {}));

// --
// Copyright (C) 2001-2017 OTRS AG, http://otrs.com/\n";
// --
// 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.
// --

"use strict";

var TimeAccounting = TimeAccounting || {};
TimeAccounting.Agent = TimeAccounting.Agent || {};

/**
 * @namespace
 * @exports TargetNS as TimeAccounting.Agent.EditTimeRecords
 * @description
 *      This namespace contains the special module functions for the edit screen.
 */
TimeAccounting.Agent.EditTimeRecords = (function (TargetNS) {

    TargetNS.ActionList = "";
    TargetNS.ActionListConstraints = "";

    /**
     * @function
     * @param {String} The ID of the working unit we are in
     * @return nothing
     *      Fills options of action selection according to selected project
     */
    TargetNS.FillActionList = function (WorkingUnitID) {
        var ProjectName = $('#ProjectID' + WorkingUnitID + ' option:selected').text(),
            ActionList = TargetNS.ActionList,
            ActionListConstraints = TargetNS.ActionListConstraints,
            $ActionSelection = $("#ActionID" + WorkingUnitID),
            SelectedActionID = $ActionSelection.val(),
            OptionCount = 0;

        // remove previous actions, leave first element (empty element)
        $ActionSelection.find('option').filter(':not(:first)').remove();


        $.each(ActionList, function () {
            var ActionID = this[0],
                ActionName = this[1];

            $.each(ActionListConstraints, function () {
                var ProjectNameRegExp = new RegExp(this[0]);
                var ActionNameRegExp = new RegExp(this[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
            $.each(ActionList, function (Index) {
                var ActionID = this[0],
                    ActionName = this[1];
                if (Index > 0) {
                    AddSelectionOption($ActionSelection, ActionName, ActionID, SelectedActionID);
                }
            });
        }
    }

    // Adds option to a selection
    function AddSelectionOption($SelectionElement, OptionText, OptionValue, SelectedOption) {
        var $Option = $('<option value="' + OptionValue + '">' + OptionText + '</option>');

        if (OptionValue == SelectedOption) {
            $Option.attr('selected', 'selected');
        }

        $SelectionElement.append($Option);
    }

    function InitAutoCompletion(Language) {
        // Initialize ComboBox on Project dropdown
        Core.UI.ComboBox.Init('.ProjectSelection', {
            Class: "Validate_TimeAccounting_Project",
            Lang: {
                ShowAllItems: Language.ShowAllItems
            }
        });
        // Initialize ComboBox on task dropdown
        Core.UI.ComboBox.Init('.ActionSelection', {
            Class: "Validate_DependingRequiredAND",
            Lang: {
                ShowAllItems: Language.ShowAllItems
            }
        });
        // Add special validation class to ActionSelection
        $('.ActionSelection').next('input:text').each(function () {
            var ID = this.id.replace('Combo_ActionID', '');
            $(this).addClass('Validate_Depending_Combo_ProjectID' + ID);
        });
        // Remove validation classes from underlying select
        $('.ProjectSelection').removeClass('Validate_TimeAccounting_Project');
        $('.ActionSelection').removeClass('Validate_DependingRequiredAND');

        // Copy ServerError classes to according comboboxes
        $('.ProjectSelection.ServerError').next('input[id^=Combo_').addClass('ServerError');
        $('.ActionSelection.ServerError').next('input[id^=Combo_').addClass('ServerError');

        Core.Form.Validate.Init();
    }

    function InitAddRow(Language) {
        $('#MoreInputFields').unbind('click.MoreInputFields').bind('click.MoreInputFields', function () {
            var $LastRow = $('#InsertWorkingHours tbody tr.WorkingHours:last'),
                $NewRow = $LastRow.clone(),
                NewRowHTML = $NewRow.html(),
                RecordNumber = parseInt($('#RecordsNumber').val(), 10) + 1;

            // Now take the last row and replace all numbers with the new record number
            NewRowHTML = NewRowHTML.replace(/ProjectID[0-9]+/g, 'ProjectID' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/ProjectID\[[0-9]+/g, 'ProjectID[' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/FillActionList\([0-9]+/g, 'FillActionList(' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/ActionID[0-9]+/g, 'ActionID' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/ActionID\[[0-9]+/g, 'ActionID[' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/Remark[0-9]+/g, 'Remark' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/Remark\[[0-9]+/g, 'Remark[' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/StartTime[0-9]+/g, 'StartTime' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/StartTime\[[0-9]+/g, 'StartTime[' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/EndTime[0-9]+/g, 'EndTime' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/EndTime\[[0-9]+/g, 'EndTime[' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/Period[0-9]+/g, 'Period' + RecordNumber);
            NewRowHTML = NewRowHTML.replace(/Period\[[0-9]+/g, 'Period[' + RecordNumber);

            // Now write this HTML back to the jquery object
            $NewRow.html(NewRowHTML);

            // If last row contained values, these must be removed
            $NewRow
                .find('input:text').val('').end()
                .find('select option').removeAttr('selected');

            // Remove autocompletion from this row (will be re-initiated later)
            $NewRow
                .find('input[id^=Combo_]').remove().end()
                .find('button[id^=ComboBtn_]').remove();

            // Now add this row to the table
            $LastRow.after($NewRow);

            // Save new RecordNumber
            $('#RecordsNumber').val(RecordNumber);

            // Re-initiate the 'odd/even' colours of the table
            $('#InsertWorkingHours tbody tr')
                .removeClass('Even')
                .filter(':odd').addClass('Even');

            // Re-initiate autocompletion combobox
            InitAutoCompletion(Language);
        });
    }

    function RecalculateTotalHours() {
        var Total = 0;
        $('.Period').each(function () {
            var Value = parseFloat($(this).val());
            if (!isNaN(Value)) {
                Total += Value;
            }
        });
        $('.TotalHours').text(Total.toFixed(2));
    }

    function InitPeriodCalculation() {
        // init calculation of period field after working hours where added/substracted/whatever...
        $('.Period').unbind('change.PeriodCalculation').bind('change.PeriodCalculation', function () {
           var FieldValue = $(this).val();

           // replace , with .
           FieldValue = FieldValue.replace(/,/g, ".");

           // check if the entered value only consists of allowed values
           // if not, we do not eval for security reasons
           if (FieldValue.match(/^[0-9\.+\- ]+$/)) {
               // Calculation
               try {
                   FieldValue = eval(FieldValue);
               }
               catch (CalcError) {
                   FieldValue = 0;
               }

               // set new value
               $(this).val(FieldValue.toFixed(2));

               // now StartTime and EndTime will not match the period anymore
               // delete these entries
               $(this).closest('tr').find('.StartTime, .EndTime').val('');

           }
           else {
               $(this).val('');
           }

           // now re-calculate the total hours sum
           RecalculateTotalHours();
        });

        // init period calculation on starttime and endtime fields
        $('.StartTime, .EndTime').unbind('change.PeriodCalculation').bind('change.PeriodCalculation', function () {
            var $StartTime, $EndTime,
                StartTimeHour, StartTimeMinute, EndTimeHour, EndTimeMinute,
                StartDate, EndDate, Today,
                Period, PeriodHour, PeriodMinute;

            if ($(this).hasClass('StartTime')) {
                $StartTime = $(this);
                $EndTime = $StartTime.closest('tr').find('.EndTime');
            }
            else {
                $StartTime = $(this).closest('tr').find('.StartTime');
                $EndTime = $(this);
            }

            // only calculate if both fields are filled
            if ($StartTime.val() && $EndTime.val()) {
                // extract hours and minutes
                StartTimeHour = $StartTime.val().split(/:/)[0] || 0;
                StartTimeMinute = $StartTime.val().split(/:/)[1] || 0;
                EndTimeHour = $EndTime.val().split(/:/)[0] || 0;
                EndTimeMinute = $EndTime.val().split(/:/)[1] || 0;

                // define date objects
                Today = new Date();
                StartDate = new Date(Today.getYear(), Today.getMonth(), Today.getDate(), StartTimeHour, StartTimeMinute, 0);
                EndDate = new Date(Today.getYear(), Today.getMonth(), Today.getDate(), EndTimeHour, EndTimeMinute, 0);

                // calculate differences
                Period = EndDate.getTime() - StartDate.getTime();
                Period = Period / (1000 * 60 * 60);

                // set new value
                if (!isNaN(Period)) {
                    $StartTime.closest('tr').find('.Period').val(Period.toFixed(2));
                }
                else {
                    $StartTime.closest('tr').find('.Period').val('');
                }

                RecalculateTotalHours();
            }
        });
    }

    /**
     * @function
     * @param {Object} Options thedifferent possible options:
     *                  RemarkRegExpContent - the regular expression for the remark validation check
     *                  Language - object with text translations
     *                  Autocompletion - boolean
     * @return nothing
     *      This function initializes all needed JS for the Edit screen
     */
    TargetNS.Init = function (Options) {
        var Options = Options || {},
            RemarkRegExpContent = Options.RemarkRegExpContent,
            Language = Options.Language;
        // Add some special validation methods for the edit screen
        // Define all available elements (only the prefixes) in a row
        var ElementPrefixes = ['ProjectID', 'ActionID', 'Remark', 'StartTime', 'EndTime', 'Period'];

        // Validates the project: if some other field in this row is filled, the project select must be filled, too
        Core.Form.Validate.AddMethod('Validate_TimeAccounting_Project', function (Value, Element) {
            var ID,
                Result = true;

            // Get ID
            // Our Element can be the select element or the autocompletion input field
            if ($(Element).is('select')) {
                ID = $(Element).attr('id').replace('ProjectID', '');
            }
            else {
                ID = $(Element).prev('select').attr('id').replace('ProjectID', '');
            }

            if (!Value) {
                $.each(ElementPrefixes, function () {
                    if (!Result) {
                        return;
                    }

                    if (this !== 'ProjectID' && $('#' + this + ID).val()) {
                        Result = false;
                    }
                });
            }

            return Result;
        });

        Core.Form.Validate.AddRule('Validate_TimeAccounting_Project', { Validate_TimeAccounting_Project: true });

        // Validates the remarks: depending on the project, remarks must be entered or not
        Core.Form.Validate.AddMethod('Validate_TimeAccounting_Remark', function (Value, Element) {
            var ID = $(Element).attr('id').replace('Remark', ''),
                RemarkRegExp = new RegExp("^(" + RemarkRegExpContent + ")$"),
                RemarkCheck = RemarkRegExp.test($('#ProjectID' + ID).val());

            if ($('#ProjectID' + ID).val()) {
                return !(RemarkCheck && Value.length < 8);
            }

            return true;
        });

        Core.Form.Validate.AddRule('Validate_TimeAccounting_Remark', { Validate_TimeAccounting_Remark: true });

        // Validates the start time: if a project is given and no time period is given, this field is required
        Core.Form.Validate.AddMethod('Validate_TimeAccounting_StartTime', function (Value, Element) {
            var ID = $(Element).attr('id').replace('StartTime', '');

            if (!Value && $('#ProjectID' + ID).val() && !$('#Period' + ID).val()) {
                return false;
            }

            return true;
        });

        Core.Form.Validate.AddRule('Validate_TimeAccounting_StartTime', { Validate_TimeAccounting_StartTime: true });

        // Validates the time period: if a project is given and no start time is given, this field is required
        Core.Form.Validate.AddMethod('Validate_TimeAccounting_Period', function (Value, Element) {
            var ID = $(Element).attr('id').replace('Period', '');

            if (!Value && $('#ProjectID' + ID).val() && !$('#StartTime' + ID).val()) {
                return false;
            }

            return true;
        });

        Core.Form.Validate.AddRule('Validate_TimeAccounting_Period', { Validate_TimeAccounting_Period: true });

        // Enable autocompletion, if configured
        if (Options.Autocompletion) {
            InitAutoCompletion(Language);
        }

        // initiate "more input fields" functionality
        InitAddRow(Language);

        // initiate period calculation
        InitPeriodCalculation();
    };

    /**
     * @function
     * @param {Object} Language object with text translations
     * @return nothing
     *      This function initializes the javascript for the mass entry functionality
     */
    TargetNS.MassEntryInit = function(Language) {
        $('#IncompleteWorkingDay-All').unbind('click.SelectAllDays').bind('click.SelectAllDays', function (Event) {
            var State = $(this).attr('checked');
            $('ul.IncompleteWorkingDays li input:checkbox').attr('checked', State);
        });
        $('#MassEntrySubmit').unbind('click.MassEntrySubmit').bind('click.MassEntrySubmit', function () {
            var $WorkingDayCheckboxes = $('ul.IncompleteWorkingDays li input:checkbox:checked').filter('[name!=IncompleteWorkingDay-All]');

            if (!$WorkingDayCheckboxes.length) {
                alert(Language.MsgChooseOneDay);
                return false;
            }

            // Show overlay
            Core.UI.Dialog.ShowContentDialog($('#MassEntryConfirmDialog'), Language.MassEntry, '150px', 'Center', true, [
                {
                    Label: Language.Submit,
                    Function: function () {
                        var $SelectedRadio = $('#MassEntryConfirmRadio li input:radio:checked'),
                            AbsenceReason,
                            CollectedDates = '';

                        if (!$SelectedRadio.length) {
                            alert(Language.MsgAbsenceReason);
                            return false;
                        }

                        // set absence reason
                        AbsenceReason = $SelectedRadio.attr('name');
                        if (AbsenceReason === 'LeaveDay') {
                            $('#MassEntry input[name=LeaveDay]').val(1);
                        }
                        else if (AbsenceReason === 'Sick') {
                            $('#MassEntry input[name=Sick]').val(1);
                        }
                        else if (AbsenceReason === 'Overtime') {
                            $('#MassEntry input[name=Overtime]').val(1);
                        }

                        // collect dates
                        $('ul.IncompleteWorkingDays li input:checkbox:checked').each(function () {
                            var Date = $(this).attr('name').replace(/IncompleteWorkingDay-/, "");
                            if (Date !== 'All') {
                                CollectedDates += Date + '|';
                            }
                        });
                        $('#MassEntry input[name=Dates]').val(CollectedDates);

                        // Show waiting icon
                        $('.Dialog:visible')
                        .find('.ContentFooter')
                        .empty()
                        .end()
                        .find('.InnerContent')
                        .width($('.Dialog:visible').find('.InnerContent').width())
                        .empty()
                        .append('<span class="AJAXLoader"></span>');

                        // Submit form
                        $('#MassEntry').submit();
                    },
                    Class: 'Primary'
                },
                {
                    Label: Language.Cancel,
                    Function: function () {
                        Core.UI.Dialog.CloseDialog($('.Dialog:visible'));
                    }
                },
            ]);
            return false;
        });
    };

    return TargetNS;
}(TimeAccounting.Agent.EditTimeRecords || {}));

PCFET0NUWVBFIGh0bWwgUFVCTElDICItLy9XM0MvL0RURCBYSFRNTCAxLjAgVHJhbnNpdGlvbmFsLy9FTiIKICAgICJodHRwOi8vd3d3LnczLm9yZy9UUi94aHRtbDEvRFREL3hodG1sMS10cmFuc2l0aW9uYWwuZHRkIj4KPGh0bWwgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGh0bWwiIHhtbDpsYW5nPSJlbiIgbGFuZz0iZW4iPgo8aGVhZD4KICAgIDxsaW5rIHJlbD0ic3R5bGVzaGVldCIgaHJlZj0iLi4vdGhpcmRwYXJ0eS9xdW5pdC9xdW5pdC5jc3MiIHR5cGU9InRleHQvY3NzIi8+CgogICAgPHRpdGxlPlRpbWVBY2NvdW50aW5nIGZvciBPVFJTIDMuMCAmbmRhc2g7IFVuaXRUZXN0IFJlc3VsdHM8L3RpdGxlPgo8L2hlYWQ+Cjxib2R5PgogICAgPGgxIGlkPSJxdW5pdC1oZWFkZXIiPlRpbWVBY2NvdW50aW5nIGZvciBPVFJTIDMuMCAmbmRhc2g7IFVuaXRUZXN0IFJlc3VsdHM8L2gxPgogICAgPGgyIGlkPSJxdW5pdC1iYW5uZXIiPjwvaDI+CiAgICA8aDIgaWQ9InF1bml0LXVzZXJBZ2VudCI+PC9oMj4KICAgIDxvbCBpZD0icXVuaXQtdGVzdHMiPjwvb2w+CgoKICAgIDwhLS0gVW5pdFRlc3QgY29kZSBnb2VzIGhlcmUgLS0+CiAgICA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSIuLi90aGlyZHBhcnR5L3F1bml0L3F1bml0LmpzIj48L3NjcmlwdD4KICAgIDwhLS0KICAgICAgICBOb3cgYWx3YXlzIGxvYWQgbGlicmFyeSBmaXJzdCBhbmQgdGhlbiB0aGUgVW5pdFRlc3QgZmlsZSBmb3IgaXQuCiAgICAgICAgVGhlIFVuaXRUZXN0IGZpbGUgaXMgcmVzcG9uc2libGUgZm9yIHJ1bm5pbnQgdGhlIHRlc3RzLgogICAgLS0+CiAgICA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSIuLi90aGlyZHBhcnR5L2pxdWVyeS0xLjQuNC9qcXVlcnkuanMiPjwvc2NyaXB0PgoKICAgIDxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0IiBzcmM9Ii4uL0NvcmUuRXhjZXB0aW9uLmpzIj48L3NjcmlwdD4KICAgIDxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0IiBzcmM9Ii4uL0NvcmUuRGVidWcuanMiPjwvc2NyaXB0PgogICAgPHNjcmlwdCB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiIHNyYz0iLi4vQ29yZS5EYXRhLmpzIj48L3NjcmlwdD4KCiAgICA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSIuLi90aGlyZHBhcnR5L2pzb24vanNvbjIuanMiPjwvc2NyaXB0PgogICAgPHNjcmlwdCB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiIHNyYz0iLi4vQ29yZS5KU09OLmpzIj48L3NjcmlwdD4KICAgIDxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0IiBzcmM9Ii4uL0NvcmUuQ29uZmlnLmpzIj48L3NjcmlwdD4KCiAgICA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSIuLi90aGlyZHBhcnR5L2pxdWVyeS11aS0xLjguNy9qcXVlcnktdWkuanMiPjwvc2NyaXB0PgogICAgPHNjcmlwdCB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiIHNyYz0iLi4vdGhpcmRwYXJ0eS9qcXVlcnktdmFsaWRhdGUtMS43L2pxdWVyeS52YWxpZGF0ZS5qcyI+PC9zY3JpcHQ+CiAgICA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSIuLi9Db3JlLlVJLkRpYWxvZy5qcyI+PC9zY3JpcHQ+CiAgICA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSIuLi9Db3JlLkZvcm0uRXJyb3JUb29sdGlwcy5qcyI+PC9zY3JpcHQ+CiAgICA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSIuLi9Db3JlLlVJLkFjY2Vzc2liaWxpdHkuanMiPjwvc2NyaXB0PgogICAgPHNjcmlwdCB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiIHNyYz0iLi4vQ29yZS5VSS5SaWNoVGV4dEVkaXRvci5qcyI+PC9zY3JpcHQ+CiAgICA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSIuLi9Db3JlLkZvcm0uVmFsaWRhdGUuanMiPjwvc2NyaXB0PgogICAgPHNjcmlwdCB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiIHNyYz0iLi4vVGltZUFjY291bnRpbmcuQWdlbnQuRWRpdFRpbWVSZWNvcmRzLmpzIj48L3NjcmlwdD4KICAgIDxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0IiBzcmM9IlRpbWVBY2NvdW50aW5nLkFnZW50LkVkaXRUaW1lUmVjb3Jkcy5Vbml0VGVzdC5qcyI+PC9zY3JpcHQ+CiAgICA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCI+CiAgICAgICAgVGltZUFjY291bnRpbmcuQWdlbnQuRWRpdFRpbWVSZWNvcmRzLlJ1blVuaXRUZXN0cygpOwogICAgPC9zY3JpcHQ+CjwvYm9keT4KPC9odG1sPgo=
// --
// Copyright (C) 2001-2017 OTRS AG, http://otrs.com/\n";
// --
// 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.
// --

"use strict";

var TimeAccounting = TimeAccounting || {};
TimeAccounting.Agent = TimeAccounting.Agent || {};

TimeAccounting.Agent.EditTimeRecords = (function (Namespace) {
    Namespace.RunUnitTests = function(){
        module('TimeAccounting.Agent.EditTimeRecords');
        test('client-side time period calculations', function(){

            expect(20);

            /*
             * Create a form container for the tests
             */
            var $TestForm = $('<form id="TestForm"></form>'),
                $Table;
            $TestForm.append('<table><tr><td></td></tr><tr><td></td></tr><tr><td></td></tr></table>');
            $Table = $TestForm.find('tr:eq(0) td');
            $Table.append('<input type="text" value="" id="StartTime1" name="StartTime1" class="StartTime" />');
            $Table.append('<input type="text" value="" id="EndTime1" name="EndTime1" class="EndTime" />');
            $Table.append('<input type="text" value="" id="Period1" name="Period1" class="Period" />');
            $Table = $TestForm.find('tr:eq(1) td');
            $Table.append('<input type="text" value="" id="StartTime2" name="StartTime2" class="StartTime" />');
            $Table.append('<input type="text" value="" id="EndTime2" name="EndTime2" class="EndTime" />');
            $Table.append('<input type="text" value="" id="Period2" name="Period2" class="Period" />');
            $Table = $TestForm.find('tr:eq(2) td');
            $Table.append('<input type="text" value="" id="StartTime3" name="StartTime3" class="StartTime" />');
            $Table.append('<input type="text" value="" id="EndTime3" name="EndTime3" class="EndTime" />');
            $Table.append('<input type="text" value="" id="Period3" name="Period3" class="Period" />');
            $Table.append('<span id="TotalHours" name="TotalHours" class="TotalHours"></span>');
            $('body').append($TestForm);

            /*
             * Run the tests
             */

            TimeAccounting.Agent.EditTimeRecords.Init();

            // Test: time calculation on time period field
            $('#Period1').val('2');
            $('#Period1').trigger('change');
            equals($('#Period1').val(), '2.00', 'Normal number entry');

            $('#Period1').val('foobar');
            $('#Period1').trigger('change');
            equals($('#Period1').val(), '', 'Not a number (="foobar")');

            $('#Period1').val('2 + 3');
            $('#Period1').trigger('change');
            equals($('#Period1').val(), '5.00', 'Easy calculation');

            $('#Period1').val('alert("huhu");');
            $('#Period1').trigger('change');
            equals($('#Period1').val(), '', 'Evil javascript code');

            $('#Period1').val('2.23 + 1.09');
            $('#Period1').trigger('change');
            equals($('#Period1').val(), '3.32', 'Calculation with non-integers');

            $('#Period1').val('2.0001 + 1.5');
            $('#Period1').trigger('change');
            equals($('#Period1').val(), '3.50', 'Calculation with a number with more than 2 decimals');

            $('#Period1').val('1,2 + 1,3');
            $('#Period1').trigger('change');
            equals($('#Period1').val(), '2.50', 'Numbers with , instead of .');

            $('#Period1').val('5.8 - 1.2');
            $('#Period1').trigger('change');
            equals($('#Period1').val(), '4.60', 'Substraction');

            $('#Period1').val('');

            // Test: calculate period from entered start and end time
            $('#StartTime1').val('1');
            $('#StartTime1').trigger('change');
            $('#EndTime1').val('');
            $('#EndTime1').trigger('change');
            equals($('#Period1').val(), '', 'Only start time entered');

            $('#StartTime1').val('');
            $('#StartTime1').trigger('change');
            $('#EndTime1').val('2');
            $('#EndTime1').trigger('change');
            equals($('#Period1').val(), '', 'Only end time entered');

            $('#StartTime1').val('1');
            $('#StartTime1').trigger('change');
            $('#EndTime1').val('2');
            $('#EndTime1').trigger('change');
            equals($('#Period1').val(), '1.00', 'Both times entered');

            $('#StartTime1').val('9:06');
            $('#StartTime1').trigger('change');
            $('#EndTime1').val('10:00');
            $('#EndTime1').trigger('change');
            equals($('#Period1').val(), '0.90', 'Easy calculation with two times of the format HH:MM');

            $('#StartTime1').val('9:00');
            $('#StartTime1').trigger('change');
            $('#EndTime1').val('11:30');
            $('#EndTime1').trigger('change');
            equals($('#Period1').val(), '2.50', 'Easy calculation with two times of the format HH:MM');

            $('#StartTime1').val('9,5');
            $('#StartTime1').trigger('change');
            $('#EndTime1').val('10,5');
            $('#EndTime1').trigger('change');
            equals($('#Period1').val(), '', 'StartTime and EndTime format are wrong');

            $('#StartTime1').val('9:5');
            $('#StartTime1').trigger('change');
            $('#EndTime1').val('10:5');
            $('#EndTime1').trigger('change');
            equals($('#Period1').val(), '1.00', 'StartTime and EndTime format are slightly wrong');

            // Test: Calculation of total hours
            $('#StartTime1').val('9:5');
            $('#StartTime1').trigger('change');
            $('#EndTime1').val('10:5');
            $('#EndTime1').trigger('change');
            equals($('#TotalHours').text(), '1.00', 'Easy start');

            $('#Period2').val('2.5');
            $('#Period2').trigger('change');
            equals($('#TotalHours').text(), '3.50', 'Second period field');

            $('#Period2').val('1.5 + 9,5');
            $('#Period2').trigger('change');
            equals($('#TotalHours').text(), '12.00', 'Second period field gets an additional calculation with one wrong comma');

            $('#Period3').val('4.35');
            $('#Period3').trigger('change');
            equals($('#TotalHours').text(), '16.35', 'Third field');

            $('#StartTime1').val('14:00');
            $('#StartTime1').trigger('change');
            $('#EndTime1').val('openend');
            $('#EndTime1').trigger('change');
            equals($('#TotalHours').text(), '15.35', 'First row: starttime and endtime fields have wrong input');


            /*
             * Cleanup div container and contents
             */
            $('#TestForm').remove();
        });
    };

    return Namespace;
}(TimeAccounting.Agent.EditTimeRecords || {}));

LyoqCiAqIEBwcm9qZWN0ICAgICBPVFJTIChodHRwOi8vd3d3Lm90cnMub3JnKSAtIEFnZW50IEZyb250ZW5kCiAqIEBjb3B5cmlnaHQgICBPVFJTIEFHCiAqIEBsaWNlbnNlICAgICBBR1BMIChodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQpCiAqLwoKLyoqCiAqIEBwYWNrYWdlICAgICBTa2luICJEZWZhdWx0IgogKiBAc2VjdGlvbiAgICAgVGltZSBBY2NvdW50aW5nIERlZmF1bHQgQ1NTCiAqLwoKQG1lZGlhIHNjcmVlbixwcm9qZWN0aW9uLHR2LGhhbmRoZWxkIHsKCi8qKgogKiBAc3Vic2VjdGlvbiAgbGVmdCBhbmQgcmlnaHQgYXJyb3dzCiAqLwoKLkFycm93IHsKICAgIHRleHQtaW5kZW50OiAtOTk5OXB4OwogICAgZGlzcGxheTogYmxvY2s7CiAgICB3aWR0aDogMTJweDsKICAgIGhlaWdodDogMTJweDsKICAgIGZsb2F0OiBsZWZ0OwogICAgbWFyZ2luLXRvcDogNXB4OwogICAgbWFyZ2luLXJpZ2h0OiA0cHg7Cn0KCi5BcnJvdy5MZWZ0IHsKICAgIGJhY2tncm91bmQtaW1hZ2U6IHVybCguLi9pbWcvYXJyb3dfbGVmdHJpZ2h0LnBuZyk7Cn0KCi5BcnJvdy5SaWdodCB7CiAgICBiYWNrZ3JvdW5kLWltYWdlOiB1cmwoLi4vaW1nL2Fycm93X2xlZnRyaWdodC5wbmcpOwogICAgYmFja2dyb3VuZC1wb3NpdGlvbjogLTEycHggMHB4Owp9CgovKioKICogQHN1YnNlY3Rpb24gIHNwZWNpYWwgc2V0dGluZ3M6IHJlcG9ydGluZwogKi8KCmRpdi5Nb250aFNlbGVjdGlvbiB7CiAgICBsaW5lLWhlaWdodDogMjRweDsKICAgIG1hcmdpbi1ib3R0b206IDEwcHg7Cn0KCi8qKgogKiBAc3Vic2VjdGlvbiAgc3BlY2lhbCBvdmVyd3JpdGluZyBzZXR0aW5ncyBmb3IgYXV0b2NvbXBsZXRpb24gYnV0dG9uCiAqLwoKdGFibGUjSW5zZXJ0V29ya2luZ0hvdXJzIHRib2R5IHRkIHsKICAgIHdoaXRlLXNwYWNlOiBub3dyYXA7CiAgICBwYWRkaW5nLXRvcDogNXB4Owp9CgoudWktYnV0dG9uLWljb24tb25seSAudWktYnV0dG9uLXRleHQgewogICAgcGFkZGluZzogMCAhaW1wb3J0YW50OwogICAgaGVpZ2h0OiAxOHB4ICFpbXBvcnRhbnQ7Cn0KCi51aS1idXR0b24gLnVpLWJ1dHRvbi10ZXh0IHsKICAgIGxpbmUtaGVpZ2h0OiAwICFpbXBvcnRhbnQ7Cn0KCmJ1dHRvbi51aS1idXR0b24taWNvbi1vbmx5IHsKICAgIHdpZHRoOiAyNHB4ICFpbXBvcnRhbnQ7CiAgICB0b3A6IC05cHg7Cn0KCi51aS1hdXRvY29tcGxldGUgLnVpLW1lbnUtaWV0bSB7CiAgICBmb250LWZhbWlseTogIkhlbHZldGljYSBOZXVlIiwgSGVsdmV0aWNhLCBBcmlhbCwgc2Fucy1zZXJpZjsKICAgIGZvbnQtc2l6ZTogMTJweDsKICAgIHRleHQtYWxpZ246IGxlZnQ7Cn0KCi5Ib3Jpem9udGFsU2Nyb2xsIHsKICAgIG92ZXJmbG93LXg6IHNjcm9sbDsKfQoKLyoqCiAqIEBzdWJzZWN0aW9uICBJbmNvbXBsZXRlV29ya2luZ0RheXMgc2VsZWN0CiAqLwojSW5jb21wbGV0ZVdvcmtpbmdEYXlzTGlzdCBvcHRpb25bc2VsZWN0ZWQ9c2VsZWN0ZWRdIHsKICAgIGZvbnQtd2VpZ2h0OiBib2xkOwp9CgovKioKICogQHN1YnNlY3Rpb24gIHByb2plY3QgYW5kIGFjdGlvbiBkcm9wZG93bnMKICovCi5Qcm9qZWN0U2VsZWN0aW9uIHsKICAgIHdpZHRoOiAzMDBweDsKfQoKLkFjdGlvblNlbGVjdGlvbiB7CiAgICB3aWR0aDogMzAwcHg7Cn0KCnVsLnVpLWF1dG9jb21wbGV0ZS51aS1tZW51IHsKICAgIG1heC1oZWlnaHQ6IDMwMHB4OwogICAgb3ZlcmZsb3cteTogYXV0bzsKfQoKfSAvKiBlbmQgQG1lZGlhICov
LyoqCiAqIEBwcm9qZWN0ICAgICBPVFJTIChodHRwOi8vd3d3Lm90cnMub3JnKSAtIEFnZW50IEZyb250ZW5kCiAqIEBjb3B5cmlnaHQgICBPVFJTIEFHCiAqIEBsaWNlbnNlICAgICBBR1BMIChodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQpCiAqLwoKLyoqCiAqIEBwYWNrYWdlICAgICBTa2luICJEZWZhdWx0IgogKiBAc2VjdGlvbiAgICAgVGltZSBBY2NvdW50aW5nIEFkZGl0aW9uYWwgVGFibGUgU2V0dGluZ3MKICovCgpAbWVkaWEgc2NyZWVuLHByb2plY3Rpb24sdHYsaGFuZGhlbGQgewoKLyoqCiAqIEBzdWJzZWN0aW9uICBkYXRhIHRhYmxlIGVsZW1lbnRzIHdpdGhvdXQgaG92ZXIKICovCgouTm9Ib3ZlciB0Ym9keSB0cjpob3ZlciB0ZCB7CiAgICBiYWNrZ3JvdW5kOiB0cmFuc3BhcmVudDsKICAgIC1tb3otYm94LXNoYWRvdzogbm9uZTsKICAgIC13ZWJraXQtYm94LXNoYWRvdzogbm9uZTsKICAgIGJveC1zaGFkb3c6IG5vbmU7Cn0KCi5Ob0hvdmVyIHRyLkV2ZW46aG92ZXIgdGQgewogICAgYmFja2dyb3VuZC1jb2xvcjogI0Y1RjVGNTsKfQoKLk5vSG92ZXIgdHIuR3JhbmRUb3RhbDpob3ZlciB0ZCB7CiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjQ0ZDRkNGOwp9CgovKioKICogQHN1YnNlY3Rpb24gIHNwZWNpYWwgdGFibGUgaGVpZ2h0CiAqLwoKLkhpZ2hDZWxscyB0Ym9keSB0ZCB7CiAgICBoZWlnaHQ6IDMwcHg7Cn0KCi8qKgogKiBAc3Vic2VjdGlvbiAgZml4ZWQgZGF0YSB0YWJsZSB3aWR0aHMKICovCgouRGF0YVRhYmxlIHRkLkZpeGVkLkRheSB7CiAgICB3aWR0aDogNTBweDsKfQoKLkRhdGFUYWJsZSB0ZC5GaXhlZC5XZWVrZGF5IHsKICAgIHdpZHRoOiA4MHB4Owp9CgovKioKICogQHN1YnNlY3Rpb24gIHRhYmxlIGNvbG91cnMgZm9yIG92ZXJ2aWV3CiAqLwoKLkRhdGFUYWJsZSB0ci5WYWNhdGlvbiB0ZCB7CiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjRUJDQ0NDOwp9CgouRGF0YVRhYmxlIHRyLk5vbldvcmtpbmdEYXkgdGQgewogICAgYmFja2dyb3VuZC1jb2xvcjogI0ZGRTBFMDsKfQoKLkRhdGFUYWJsZSB0ci5BY3RpdmUgdGQgewogICAgYm9yZGVyLWJvdHRvbTogMXB4IHNvbGlkICMwMDAwMDA7CiAgICBib3JkZXItdG9wOiAxcHggc29saWQgIzAwMDAwMDsKICAgIGNvbG9yOiAjMUExQTFBOwp9CgouRGF0YVRhYmxlIHRyLkFjdGl2ZSB0ZDpmaXJzdC1jaGlsZCB7CiAgICBib3JkZXItbGVmdDogMXB4IHNvbGlkICMwMDAwMDA7Cn0KCi5EYXRhVGFibGUgdHIuQWN0aXZlIHRkOmxhc3QtY2hpbGQsCi5EYXRhVGFibGUgdHIuQWN0aXZlIHRkLkxhc3QgewogICAgYm9yZGVyLXJpZ2h0OiAxcHggc29saWQgIzAwMDAwMDsKfQoKLkRhdGFUYWJsZSB0ci5HcmFuZFRvdGFsIHRkIHsKICAgIGJhY2tncm91bmQtY29sb3I6ICNDRkNGQ0Y7Cn0KCi8qKgogKiBAc3Vic2VjdGlvbiAgZGF0YSB0YWJsZSB3aXRoIHR3byBoZWFkZXIgcm93cwogKi8KCi5EYXRhVGFibGUgdGhlYWQuVHdvUm93cyB7CiAgICBiYWNrZ3JvdW5kLWltYWdlOiBub25lOwp9CgouRGF0YVRhYmxlIHRoZWFkLlR3b1Jvd3MgdHI6Zmlyc3QtY2hpbGQgewogICAgYmFja2dyb3VuZDogI0VFRSB1cmwoIi4uL2ltZy9TbWFsbERhcmtHcmFkaWVudC5wbmciKSByZXBlYXQteDsKICAgIGJhY2tncm91bmQtcG9zaXRpb246IDBweCAwcHg7Cn0KCi5EYXRhVGFibGUgdGhlYWQuVHdvUm93cyB0ciB7CiAgICBiYWNrZ3JvdW5kOiAjRUVFIHVybCgiLi4vaW1nL1NtYWxsRGFya0dyYWRpZW50LnBuZyIpIHJlcGVhdC14OwogICAgYmFja2dyb3VuZC1wb3NpdGlvbjogMHB4IC0xNnB4Owp9CgouRGF0YVRhYmxlIHRoZWFkLlR3b1Jvd3MgdHI6Zmlyc3QtY2hpbGQgdGggewogICAgYm9yZGVyLWJvdHRvbTogbm9uZTsKICAgIHBhZGRpbmctYm90dG9tOiAycHg7Cn0KCi5EYXRhVGFibGUgdGhlYWQuVHdvUm93cyB0cjpsYXN0LWNoaWxkIHRoLAouRGF0YVRhYmxlIHRoZWFkLlR3b1Jvd3MgdHIuTGFzdCB0aCB7CiAgICBib3JkZXItdG9wOiBub25lOwogICAgcGFkZGluZy10b3A6IDNweDsKfQoKdHIuTWFzdGVyQWN0aW9uIHRkLlJvd1NwYW4gewogICAgdmVydGljYWwtYWxpZ246IHRvcDsKfQoKfSAvKiBlbmQgQG1lZGlhICov
iVBORw0KGgoAAAANSUhEUgAAABgAAAANCAYAAACzbK7QAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAOpJREFUeNpi/P//PwMtASMRasRABNAhr1A0MjIKA6nfQPwJn2YmPHKsQJwOxIeAWAGLvDIQHwXiSHJ85gLEx4D4GRSbgYISGYPEkOR3Q/kEfaAGxEuBeBEOV+MC2kC8AYinArE0Ngv4gLgeiPcCsSMFcRoIxIeBuAiIOWAWcAPxWmh4M1Mh4YAMLgHizSCzQRZ8BWJfIO4B4h9UsOAvEM8E4mCQ2bAgAhncB8S2QLyeAsP3A7EzEDcSSr5m0JRBbCo6Bk15JANQGr+Mx4Ib0LhjpSQng1IYK9DQt2g5WQzKfIW3qKB1WQQQYAAqT1loJfuiDwAAAABJRU5ErkJggg==
# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# 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 var::packagesetup::TimeAccounting;

use strict;
use warnings;

use Kernel::System::SysConfig;
use Kernel::System::CacheInternal;
use Kernel::System::Group;
use Kernel::System::Valid;

use vars qw(@ISA);

=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;
    use var::packagesetup::TimeAccounting;

    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 EncodeObject LogObject MainObject TimeObject DBObject XMLObject)
        )
    {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }

    # create needed sysconfig object
    $Self->{SysConfigObject} = Kernel::System::SysConfig->new( %{$Self} );

    # rebuild ZZZ* files
    $Self->{SysConfigObject}->WriteDefault();

    # define the ZZZ files
    my @ZZZFiles = (
        'ZZZAAuto.pm',
        'ZZZAuto.pm',
    );

    # reload the ZZZ files (mod_perl workaround)
    for my $ZZZFile (@ZZZFiles) {

        PREFIX:
        for my $Prefix (@INC) {
            my $File = $Prefix . '/Kernel/Config/Files/' . $ZZZFile;
            next PREFIX if !-f $File;
            do $File;
            last PREFIX;
        }
    }

    $Self->{GroupObject}         = Kernel::System::Group->new( %{$Self} );
    $Self->{ValidObject}         = Kernel::System::Valid->new( %{$Self} );
    $Self->{CacheInternalObject} = Kernel::System::CacheInternal->new(
        %{$Self},
        Type => 'Group',
        TTL  => 60 * 60 * 3,
    );

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

    # delete the group cache to avoid permission problems
    $Self->{CacheInternalObject}->CleanUp( OtherType => 'Group' );

    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;

    # get list of all groups
    my %GroupList = $Self->{GroupObject}->GroupList();

    # reverse the group list for easier lookup
    my %GroupListReverse = reverse %GroupList;

    # check if group already exists
    my $GroupID = $GroupListReverse{ $Param{Name} };

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

=cut
