OTRSMasterSlave
6.1.1
Ported to 6.1
6.1.x
Znuny
https://www.znuny.org/
GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007
Includes "Ticket Master/Slave" feature.
Enthält "Ticket Master/Slave" Funktionalität.
A „mester/alárendelt jegy” funkciót tartalmazza.
build.znuny.com
2021-09-14 11:36:22 UTC
<?xml version="1.0" encoding="utf-8" ?>
<otrs_config version="2.0" init="Changes">
    <Setting Name="MasterSlave::DynamicField" Required="0" Valid="1">
        <Description Translatable="1">Defines dynamic field name for master ticket feature.</Description>
        <Navigation>Core::MasterSlave</Navigation>
        <Value>
            <Item ValueType="String">MasterSlave</Item>
        </Value>
    </Setting>
    <Setting Name="MasterSlave::AdvancedEnabled" Required="1" Valid="1">
        <Description Translatable="1">Enables the advanced MasterSlave part of the feature.</Description>
        <Navigation>Core::MasterSlave</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="0" Translatable="1">Disabled</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Enabled</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="MasterSlave::UnsetMasterSlave" Required="1" Valid="1">
        <Description Translatable="1">Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.</Description>
        <Navigation>Core::MasterSlave</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="0" Translatable="1">Disabled</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Enabled</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="MasterSlave::UpdateMasterSlave" Required="1" Valid="1">
        <Description Translatable="1">Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.</Description>
        <Navigation>Core::MasterSlave</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="0" Translatable="1">Disabled</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Enabled</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="MasterSlave::FollowUpdatedMaster" Required="1" Valid="1">
        <Description Translatable="1">Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.</Description>
        <Navigation>Core::MasterSlave</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="0" Translatable="1">Disabled</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Enabled</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="MasterSlave::ForwardSlaves" Required="1" Valid="1">
        <Description Translatable="1">Enables the feature to forward articles from type 'forward' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type 'forward' to the slave tickets.</Description>
        <Navigation>Core::MasterSlave</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="0" Translatable="1">Disabled</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Enabled</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="MasterSlave::KeepParentChildAfterUnset" Required="1" Valid="1">
        <Description Translatable="1">Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.</Description>
        <Navigation>Core::MasterSlave</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="0" Translatable="1">Disabled</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Enabled</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="MasterSlave::KeepParentChildAfterUpdate" Required="1" Valid="1">
        <Description Translatable="1">Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.</Description>
        <Navigation>Core::MasterSlave</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="0" Translatable="1">Disabled</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Enabled</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###MasterSlave" Required="0" Valid="1">
        <Description Translatable="1">Registration of the ticket event module.</Description>
        <Navigation>Core::Event::MasterSlave</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::MasterSlave</Item>
                <Item Key="Event">ArticleCreate|ArticleSend|TicketStateUpdate|TicketPriorityUpdate|TicketPendingTimeUpdate|TicketLockUpdate|TicketOwnerUpdate|TicketResponsibleUpdate</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PreApplicationModule###AgentPreMasterSlave" Required="0" Valid="1">
        <Description Translatable="1">This module activates Master/Slave field in new email and phone ticket screens.</Description>
        <Navigation>Core::MasterSlave</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">Kernel::Modules::AgentPreMasterSlave</Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentTicketMasterSlave" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">Ticket MasterSlave.</Item>
                    <Item Key="Title" Translatable="1">MasterSlave</Item>
                    <Item Key="NavBarName">Ticket</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AgentTicketMasterSlave###004-OTRSMasterSlave" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="JavaScript">
                    <Array>
                        <Item>Core.Agent.TicketAction.js</Item>
                        <Item>Core.Agent.TicketMasterSlave.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AgentTicketMasterSlave###004-OTRSMasterSlave" Required="0" Valid="0">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::MenuModule###480-MasterSlave" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketZoom::MenuModule</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::TicketMenu::Generic</Item>
                <Item Key="Name" Translatable="1">MasterSlave</Item>
                <Item Key="Description" Translatable="1">Change the MasterSlave state of the ticket.</Item>
                <Item Key="Action">AgentTicketMasterSlave</Item>
                <Item Key="Link">Action=AgentTicketMasterSlave;TicketID=[% Data.TicketID | html %]</Item>
                <Item Key="Target"></Item>
                <Item Key="PopupType">TicketAction</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">MasterSlave</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###RequiredLock" Required="0" Valid="1">
        <Description Translatable="1">Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn't locked yet, the ticket gets locked and the current agent will be set automatically as its owner).</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">1</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###TicketType" Required="0" Valid="1">
        <Description Translatable="1">Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###Service" Required="0" Valid="1">
        <Description Translatable="1">Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###Owner" Required="0" Valid="1">
        <Description Translatable="1">Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###OwnerMandatory" Required="0" Valid="1">
        <Description Translatable="1">Sets if ticket owner must be selected by the agent.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###Responsible" Required="0" Valid="1">
        <Description Translatable="1">Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###ResponsibleMandatory" UserPreferencesGroup="Advanced" UserModificationPossible="1" Required="0" Valid="1">
        <Description Translatable="1">Sets if ticket responsible must be selected by the agent.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###State" Required="0" Valid="1">
        <Description Translatable="1">If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###StateType" Required="1" Valid="1">
        <Description Translatable="1">Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Array>
                <Item>open</Item>
                <Item>closed</Item>
                <Item>pending reminder</Item>
                <Item>pending auto</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###StateDefault" Required="0" Valid="1">
        <Description Translatable="1">Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Entity" ValueEntityType="State" ValueRegex="">pending reminder</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###Note" Required="0" Valid="1">
        <Description Translatable="1">Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">1</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###NoteMandatory" Required="0" Valid="1">
        <Description Translatable="1">Sets if note must be filled in by the agent. Can be overwritten by Ticket::Frontend::NeedAccountedTime.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">1</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###Subject" Required="0" Valid="1">
        <Description Translatable="1">Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex=""></Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###Body" Required="0" Valid="1">
        <Description Translatable="1">Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Textarea"></Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###RichTextWidth" UserPreferencesGroup="Advanced" UserModificationPossible="1" Required="0" Valid="1">
        <Description Translatable="1">Defines the width for the rich text editor component for this screen. Enter number (pixels) or percent value (relative).</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="^\d+%?$">620</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###RichTextHeight" UserPreferencesGroup="Advanced" UserModificationPossible="1" Required="0" Valid="1">
        <Description Translatable="1">Defines the height for the rich text editor component for this screen. Enter number (pixels) or percent value (relative).</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="^\d+%?$">100</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###InvolvedAgent" Required="0" Valid="0">
        <Description Translatable="1">Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###InformAgent" Required="0" Valid="0">
        <Description Translatable="1">Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###IsVisibleForCustomerDefault" Required="0" Valid="1">
        <Description Translatable="1">Defines if the MasterSlave note is visible for the customer by default.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
             <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###Priority" Required="0" Valid="0">
        <Description Translatable="1">Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###PriorityDefault" Required="0" Valid="0">
        <Description Translatable="1">Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Entity" ValueEntityType="Priority" ValueRegex="">3 normal</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###Title" Required="0" Valid="1">
        <Description Translatable="1">Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###HistoryType" Required="0" Valid="1">
        <Description Translatable="1">Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">AddNote</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###HistoryComment" Required="0" Valid="1">
        <Description Translatable="1">Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">%%MasterSlave</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::AgentTicketMasterSlave###MasterSlaveMandatory" Required="0" Valid="1">
        <Description Translatable="1">Sets if Master / Slave field must be selected by the agent.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="DashboardBackend###0900-TicketMaster" Required="0" Valid="1">
        <Description Translatable="1">Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.</Description>
        <Navigation>Frontend::Agent::View::Dashboard</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::Dashboard::TicketGeneric</Item>
                <Item Key="Title" Translatable="1">Master Tickets</Item>
                <Item Key="Description" Translatable="1">All master tickets</Item>
                <Item Key="Attributes">DynamicField_MasterSlave_Equals=Master;StateType=open;StateType=new;</Item>
                <Item Key="Filter">All</Item>
                <Item Key="Time">Age</Item>
                <Item Key="Limit">10</Item>
                <Item Key="Permission">rw</Item>
                <Item Key="Block">ContentLarge</Item>
                <Item Key="Group"></Item>
                <Item Key="Default">1</Item>
                <Item Key="CacheTTLLocal">0.5</Item>
                <Item Key="DefaultColumns">
                    <Hash>
                        <Item Key="Age">2</Item>
                        <Item Key="Changed">1</Item>
                        <Item Key="CustomerID">1</Item>
                        <Item Key="CustomerName">1</Item>
                        <Item Key="CustomerUserID">1</Item>
                        <Item Key="EscalationResponseTime">1</Item>
                        <Item Key="EscalationSolutionTime">1</Item>
                        <Item Key="EscalationTime">1</Item>
                        <Item Key="EscalationUpdateTime">1</Item>
                        <Item Key="TicketNumber">2</Item>
                        <Item Key="Lock">1</Item>
                        <Item Key="Owner">1</Item>
                        <Item Key="PendingTime">1</Item>
                        <Item Key="Queue">1</Item>
                        <Item Key="Responsible">1</Item>
                        <Item Key="Priority">1</Item>
                        <Item Key="Service">1</Item>
                        <Item Key="State">1</Item>
                        <Item Key="SLA">1</Item>
                        <Item Key="Title">2</Item>
                        <Item Key="Type">1</Item>
                    </Hash>
                </Item>
                <Item Key="Mandatory">0</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="DashboardBackend###0910-TicketSlave" Required="0" Valid="1">
        <Description Translatable="1">Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.</Description>
        <Navigation>Frontend::Agent::View::Dashboard</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::Dashboard::TicketGeneric</Item>
                <Item Key="Title" Translatable="1">Slave Tickets</Item>
                <Item Key="Description" Translatable="1">All slave tickets</Item>
                <Item Key="Attributes">DynamicField_MasterSlave_Like=Slave*;StateType=open;StateType=new;</Item>
                <Item Key="Filter">All</Item>
                <Item Key="Time">Age</Item>
                <Item Key="Limit">10</Item>
                <Item Key="Permission">rw</Item>
                <Item Key="Block">ContentLarge</Item>
                <Item Key="Group"></Item>
                <Item Key="Default">1</Item>
                <Item Key="CacheTTLLocal">0.5</Item>
                <Item Key="DefaultColumns">
                    <Hash>
                        <Item Key="Age">2</Item>
                        <Item Key="Changed">1</Item>
                        <Item Key="CustomerID">1</Item>
                        <Item Key="CustomerName">1</Item>
                        <Item Key="CustomerUserID">1</Item>
                        <Item Key="EscalationResponseTime">1</Item>
                        <Item Key="EscalationSolutionTime">1</Item>
                        <Item Key="EscalationTime">1</Item>
                        <Item Key="EscalationUpdateTime">1</Item>
                        <Item Key="TicketNumber">2</Item>
                        <Item Key="Lock">1</Item>
                        <Item Key="Owner">1</Item>
                        <Item Key="PendingTime">1</Item>
                        <Item Key="Queue">1</Item>
                        <Item Key="Responsible">1</Item>
                        <Item Key="Priority">1</Item>
                        <Item Key="Service">1</Item>
                        <Item Key="State">1</Item>
                        <Item Key="SLA">1</Item>
                        <Item Key="Title">2</Item>
                        <Item Key="Type">1</Item>
                    </Hash>
                </Item>
                <Item Key="Mandatory">0</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ReplaceCustomerRealNameOnSlaveArticleTypes" Required="0" Valid="1">
        <Description Translatable="1">This setting is deprecated and will be removed in further versions of OTRSMasterSlave.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Hash>
                <Item Key="email-external">0</Item>
                <Item Key="email-internal">0</Item>
                <Item Key="email-notification-ext">0</Item>
                <Item Key="email-notification-int">0</Item>
                <Item Key="phone">0</Item>
                <Item Key="fax">0</Item>
                <Item Key="sms">0</Item>
                <Item Key="webrequest">0</Item>
                <Item Key="note-internal">0</Item>
                <Item Key="note-external">0</Item>
                <Item Key="note-report">0</Item>
            </Hash>
        </Value>
    </Setting>

    <Setting Name="ReplaceCustomerRealNameOnSlaveArticleCommunicationChannels" Required="0" Valid="1">
        <Description Translatable="1">Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.</Description>
        <Navigation>Frontend::Agent::View::TicketMasterSlave</Navigation>
        <Value>
            <Hash>
                <Item Key="Email">0</Item>
            </Hash>
        </Value>
    </Setting>

    <Setting Name="ACLKeysLevel3::Actions###888-OTRSMasterSlave" Required="0" Valid="1">
        <Description Translatable="1">Defines which items are available for 'Action' in third level of the ACL structure.</Description>
        <Navigation>Core::Ticket::ACL</Navigation>
        <Value>
            <Array>
                <Item>AgentTicketMasterSlave</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="DynamicFields::Driver###MasterSlave" Required="0" Valid="1">
        <Description Translatable="1">DynamicField backend registration.</Description>
        <Navigation>Core::DynamicFields::DriverRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="DisplayName" Translatable="1">Master / Slave</Item>
                <Item Key="Module">Kernel::System::DynamicField::Driver::MasterSlave</Item>
                <Item Key="ConfigDialog">AdminDynamicFieldMasterSlave</Item>
                <Item Key="DisabledAdd">1</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AdminDynamicFieldMasterSlave" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Admin::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Group">
                        <Array>
                            <Item>admin</Item>
                        </Array>
                    </Item>
                    <Item Key="Description">Admin</Item>
                    <Item Key="Title" Translatable="1">Dynamic Fields Drop-down Backend GUI</Item>
                    <Item Key="NavBarName"></Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AdminDynamicFieldMasterSlave###004-OTRSMasterSlave" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Frontend::Admin::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="CSS">
                    <Array>
                        <Item>Core.Agent.Admin.DynamicField.css</Item>
                    </Array>
                </Item>
                <Item Key="JavaScript">
                    <Array>
                        <Item>Core.Agent.Admin.DynamicField.js</Item>
                        <Item>Core.Agent.Admin.DynamicFieldMasterSlave.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AdminDynamicFieldMasterSlave###004-OTRSMasterSlave" Required="0" Valid="0">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Frontend::Admin::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
            </Array>
        </Value>
    </Setting>
     <Setting Name="Ticket::Frontend::BulkModule###010-MasterSlave" Required="0" Valid="1">
        <Description Translatable="1">MasterSlave module for Ticket Bulk feature.</Description>
        <Navigation>Frontend::Agent::View::TicketBulk::Module</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::TicketBulk::MasterSlave</Item>
            </Hash>
        </Value>
    </Setting>
</otrs_config>

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::cs_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = '';
    $Self->{Translation}->{'Unset Master Ticket'} = '';
    $Self->{Translation}->{'Unset Slave Ticket'} = '';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = '';
    $Self->{Translation}->{'Unset Slave Tickets'} = '';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = '';
    $Self->{Translation}->{'Slave of %s%s%s'} = '';
    $Self->{Translation}->{'Master Ticket'} = '';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = '';
    $Self->{Translation}->{'All slave tickets'} = '';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Dovoluje přidávání poznámek při zobrazení Nadřízeného / Podřízeného v detailu tiketu - AgentTicketZoom.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Změna stavu Nadřízený/Podřízený.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = '';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Určuje, zda je vyžadováno uzamknutí tiketu při zobrazení Nadřízeného / Podřízeného v detailu tiketu v AgentTicketZoom. (Není-li tiket zatím uzemčen, bude uzamčen a stávající agent bude automaticky určen jako vlastník.)';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Určuje defaultní následují stav tiketu po přidání poznámky - v zobrazení Nadřízený / Podřízený v detailu tiketu v AgentTicketZoom.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Určuje defaultní prioritu tiketu v zobrazení Nadřízený / Podřízený v AgentTicketZoom.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Určuje komentář historie pro akci zobrazení Nadřízený / Podřízený, komentář je použit v historii tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Určuje typ historie pro akci zobrazení Nadřízený / Podřízený, typ je použit v historii tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Určuje další stav tiketu po přidání poznámky v zobrazení Nadřízený / Podřízený v detailu tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = '';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Pokud je přidána poznámka agentem, určí stav tiketu v zobrazení Nadřízený / Podřízený - v detailu tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Master / Slave'} = '';
    $Self->{Translation}->{'Master Tickets'} = '';
    $Self->{Translation}->{'MasterSlave'} = '';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametry nastavení nástěnky v přehledu nadřízených tiketů v rozhraní agenta. "Limit" je množství standardně viditelných vstupů. "Group" se používá k omezení přístupu k pluginu (např.: Group: admin;group1;group2;). "Default" určuje, zda je plugin povolen standardně nebo zda jej musí uživatel povolit manuálně. "CacheTTLLocal" je cachovací čas pluginu -  v minutách.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametry nastavení nástěnky pro přehled podřízených tiketů v rozhraní agenta. "Limit" je množství standardně viditelných vstupů. "Group" se používá k omezení přístupu k pluginu (např.: Group: admin;group1;group2;). "Default" určuje, zda je plugin povolen standardně nebo zda jej musí uživatel povolit manuálně. "CacheTTLLocal" je cachovací čas pluginu - v minutách.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Registrace modulu událostí tiketu.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Požadovaná oprávnění k užívání zobrazení Nadřízený / Podřízený v detailu tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Určuje defaultní text těla poznámky přidané v zobrazení Nadřízený / Podřízený v detailu tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Určuje defaultní subjekt poznámek v zobrazení Nadřízený / Podřízený v detailu tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Určuje zodpovědného agenta v zobrazení Nadřízený / Podřízený v detailu tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Určuje servis v zobrazení Nadřízený / Podřízený v detailu tiketu v rozhraní agenta. (Musí být aktivován Ticket::Service.)';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Určuje vlastníka tiketu v zobrazení Nadřízený / Podřízený v detailu tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Určuje typ tiketu v zobrazení Nadřízený / Podřízený v detailu tiketu v rozhraní agenta. (Musí být aktivován Ticket::Type.)';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Zobrazuje link v menu,  který slouží ke změně statusu tiketu v nadřízený / podřízený, v detailu tiketu v rozhraní agenta.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Zobrazuje seznam všech agentů, kteří se podíleli na tiketu,  v zobrazení Nadřízený / Podřízený v detailu tiketu v rozhraní agenta';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Zobrazuje seznam možných agentů (všichni agenti mohoucí přidat poznámku k frontě/tiketu), aby bylo možno určit, kdo bude o poznámce informován - v zobrazení Nadřízeného/Podřízeného zazoomovaného tiketu v agentském prostředí. ';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = '';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        '';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::de_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Master/Slave-Status verwalten für %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Neues Master-Ticket';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Aufheben des Master-Tickets';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Aufheben des Slave-Tickets';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Slave von %s%s%s: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Aufheben der Master-Tickets';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Aufheben der Slave-Tickets';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Master';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Slave von %s%s%s';
    $Self->{Translation}->{'Master Ticket'} = 'Master-Ticket';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Alle Master-Tickets';
    $Self->{Translation}->{'All slave tickets'} = 'Alle Slave-Tickets';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Erlaubt das hinzufügen von Notizen in der MasterSlave-Ansicht eines aufgerufenen Tickets in der Agenten-Oberfläche.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Den MasterSlave-Status des Tickets ändern.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Dynamisches Feld für das Master-Ticket-Feature definieren.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Bestimmt, ob dieser Screen im Agenten-Interface das Sperren des Tickets voraussetzt. Das Ticket wird (falls nötig) gesperrt und der aktuelle Agent wird als Besitzer gesetzt.';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        'Definiert, ob die MasterSlave-Notiz standardmäßig für Kunden sichtbar ist.';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Bestimmt den Folgestatus für Tickets, für die im MasterSlave-Bildschirm des Agenten-Interface eine Notiz hinzugefügt wurde.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Bestimmt die standardmäßige Ticket-Priorität für Tickets im MasterSlave-Bildschirm des Agenten-Interface.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Bestimmt den Historien-Kommentar von Ticket-Aktionen im MasterSlave-Bildschirm, welcher für die Ticket-Historie im Agenten-Interface verwendet wird.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Bestimmt den Historien-Typ von Ticket-Aktionen im MasterSlave-Bildschirm, welcher für die Ticket-Historie im Agenten-Interface verwendet wird.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Bestimmt den Folgestatus für Tickets, für die im MasterSlave-Bildschirm des Agenten-Interface eine Notiz hinzugefügt wurde.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Das erweiterte Verhalten des MasterSlave-Features aktivieren.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Aktiviert die Funktionalität, dass Slave-Tickets dem Master-Ticket im erweiterten MasterSlave-Verhalten zum neuen Master folgen.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Aktiviert die Funktionalität zum Ändern des MasterSlave-Status eines Tickets im erweiterten MasterSlave-Modus.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Aktiviert die Funktion zum Weiterleiten von Artikeln des Typs \'Weiterleiten\' zu den Kunden des Slave-Tickets. Standardmäßig (deaktiviert) werden keine Artikel des Typs \'Weiterleiten\' an die Slave-Tickets weitergeleitet.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Aktiviert im die Funktion, im erweiterten MasterSlave-Verhalten eine Eltern-Kind-Beziehung nach dem Ändern des MasterSlave-Status zu behalten.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Aktiviert im die Funktion, im erweiterten MasterSlave-Verhalten eine Eltern-Kind-Beziehung nach dem Auflösen des MasterSlave-Status zu behalten.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Aktiviert die Funktion zum Aufheben des MasterSlave-Status eines Tickets im erweiterten MasterSlave-Verhalten.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Ermöglicht das Ändern des TIcket-Status beim Hinzufügen einer Notiz innerhalb des MasterSlave-Bildschirms.';
    $Self->{Translation}->{'Master / Slave'} = 'Master / Slave';
    $Self->{Translation}->{'Master Tickets'} = 'Master-Tickets';
    $Self->{Translation}->{'MasterSlave'} = 'MasterSlave';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'MasterSlave-Modul für Ticket-Sammelaktionen.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Einstellung der Übersichtsseitenparameter für Master Tickets in der Agentenoberfläche. "Limit" gibt die Anzahl der standardmäßig dargestellten Einträge an. "Group" wird verwendet, um den Zugriff auf das Plugin zu begrenzen (bspw. Group: admin;group1;group2;). "Default" bestimmt, ob das Plugin standardmäßig aktiviert ist oder ob der Benutzer es selbst aktivieren muss. "CacheTTLLocal" ist die Caching-Zeit des Plugins, angegeben in Minuten.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Einstellung der Übersichtsseitenparameter für Slave Tickets in der Agentenoberfläche. "Limit" gibt die Anzahl der standardmäßig dargestellten Einträge an. "Group" wird verwendet, um den Zugriff auf das Plugin zu begrenzen (bspw. Group: admin;group1;group2;). "Default" bestimmt, ob das Plugin standardmäßig aktiviert ist oder ob der Benutzer es selbst aktivieren muss. "CacheTTLLocal" ist die Caching-Zeit des Plugins, angegeben in Minuten.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Registrierung des Ticket-Event-Moduls.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Benötigte Berechtigungen für die Anzeige des MasterSlave-Dialogs im Ticket-Zoom-Dialog des Agenteninterface.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Legt fest, ob Master / Slave Feld durch einen Agenten ausgewählt sein muss.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Bestimmt den standardmäßigen Text einer Notiz für Tickets im MasterSlave-Bildschirm des Agenten-Interface.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Bestimmt den standardmäßigen Betreff einer Notiz für Tickets im MasterSlave-Bildschirm des Agenten-Interface.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Bestimmt den verantwortlichen Agenten eines Tickets im MasterSlave Bildschirm des Agenten-Interface für ein aufgerufenes Ticket.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Bestimmt den Service eines Tickets im MasterSlave-Bildschirm des Agenten-Interface für ein aufgerufenes Ticket (Ticket::Service muss aktiviert sein).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Bestimmt den Besitzer eines Tickets im MasterSlave-Bildschirm des Agenten-Interface für ein aufgerufenes Ticket.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Bestimmt den Ticket-Typ eines Tickets im MasterSlave-Bildschirm des Agenten-Interface für ein aufgerufenes Ticket (Ticket::Type muss aktiviert sein).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Zeigt einen Link zum Ändern des MasterSlave-Status eines Tickets im Menü der Ticket-Zoom-Ansicht des Agenten an.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Zeigt eine Liste aller involvierten Agenten für das gewählte Ticket im MasterSlave-Bildschirm des Agenten-Interface an.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Zeigt eine Liste aller möglichen Agenten (alle Agenten mit Notiz Berechtigung auf der Queue/dem Ticket) für das gewählte Ticket im MasterSlave-Bildschirm des Agenten-Interface an.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Zeigt die Priorität eines Tickets im MasterSlave-Bildschirm des Agenten-Interface für ein aufgerufenes Ticket an.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Zeigt das Titel-Feld in der MasterSlave-Oberfläche eines aufgerufenen Tickets im Agenten-Interface an. ';
    $Self->{Translation}->{'Slave Tickets'} = 'Slave-Tickets';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        'Gibt die verschiedenen Artikel-Kommunikationskanäle an, bei denen der echte Name vom Eltern-Ticket durch den im Kind-Ticket ersetzt wird.';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Dieses Modul aktiviert das Master/Slave-Feld in der Anzeige für ein neues Email- oder Telefon-Ticket.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        'Diese Einstellung ist veraltet und wird in weiteren Versionen von OTRSMasterSlave entfernt.';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Ticket MasterSlave.';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::es_MX_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Gestionar estas Maestro/Esclavo para %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Nuevo Ticket Maestro';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Remover estado Ticket Maestro';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Remover estado Ticket Esclavo';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Esclavo de %s%s%s:%s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Remover estado Maestro de Tickets';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Remver estado Esclavo de Ticket';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Maestro';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Esclavo de %s%s%s';
    $Self->{Translation}->{'Master Ticket'} = 'Ticket Maestro';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Todos los tickets maestros';
    $Self->{Translation}->{'All slave tickets'} = 'Todos los tickets esclavos';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Permite añadir notas en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Modifica el estado Maestro-Esclavo del ticket.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Define el nombre del campo dinámico para el funcionalidad de ticket maestro.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Define el bloqueo requerido in la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente (si el ticket no ha sido bloqueado aun, el ticket se bloquea y el agente actual se convertirá automáticamente en el dueño del mismo).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        'Define si la nota de Maestro-Eslavo es visible por defecto para lel cliente.';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define el siguiente estado del ticket después de añadir una nota, en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define la prioridad por defecto en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Define el comentario del historial para la acción de la pantalla del ticket Maestro-Esclavo, que es usado para el historial del ticket en la interfaz del agente.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Define el tipo de historial para la acción de la pantalla del ticket Maestro-Esclavo, que es usado para el historial del ticket en la interfaz del agente.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define el siguiente estado después de añadir una nota, en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Habilita la parte avanzada de la funcionalidad MaestroEsclavo.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Habilita la funcionalidad en la que tickets esclavos siguen al ticket maestro hacia un nuevo maestro en el modo avanzado de MasterSlave';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Habilita la funcionalidad de cambiar el estado MaestroEsclavo de un ticket en el modo avanzado de MaestroEsclavo.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Habilita la funcionalidad de re-enviar artículos de tipo \'forward\' de un ticket maestro hacia los clientes de sus ticket esclavos. Por omisión (deshabitado) no re-enviará artículos de tipo \'forward\' hacia los tickets esclavos.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Habilita la funcionalidad de mantener los vínculos padre-hijo después de cambiar el estado de MaestroEsclavo en el modo avanzado de MaestroEsclavo.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Habilita la funcionalidad de mantener los vínculos padre-hijo después de remover el estado de MaestroEsclavo  en el modo avanzado de MaestroEsclavo. ';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Habilita la funcionalidad de remover el estado de MaestroEsclavo de un ticket en el modo avanzado de MaestroEsclavo.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Si una nota es añadida por un agente, fija el estado del ticket en en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Master / Slave'} = 'Maestro / Esclavo';
    $Self->{Translation}->{'Master Tickets'} = 'Tickets Maestros';
    $Self->{Translation}->{'MasterSlave'} = 'MaestroEsclavo';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Módulo MasterSlave para la opción de Tickets por Lote';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parámetros para el backend del panel principal de las estadísticas de tickets maestros de la interfaz del agente. "Limit" es el número de entradas mostradas por defecto. "Group" se usa para restringir el acceso al plugin (por ejemplo, Group: admin;group1;group2;). "Default" determina si el plugin está habilitado por defecto o si el usuario tiene que habilitarlo manualmente. "CacheTTLLocal" es el tiempo en minutos para la caché del plugin.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parámetros para el backend del panel principal de las estadísticas de tickets esclavos de la interfaz del agente. "Limit" es el número de entradas mostradas por defecto. "Group" se usa para restringir el acceso al plugin (por ejemplo, Group: admin;group1;group2;). "Default" determina si el plugin está habilitado por defecto o si el usuario tiene que habilitarlo manualmente. "CacheTTLLocal" es el tiempo en minutos para la caché del plugin.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Registro del móduo de evento para tickets.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Permisos requeridos para usar la pantalla Maestro-Esclavo de un ticket, en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Establece si el agente debe seleccionar el campo Maestro / Esclavo.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Fija el texto del cuerpo por defecto de las notas añadidas en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Fija el asunto por defecto de las notas añadidas en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Fija el agente responsable de un ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Fija el servicio de un ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente (Ticket::Service require ser activado).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Fija el dueño de un ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Fija el tipo de ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente (Ticket::Type require ser activado).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Muestra un enlace en el menú para cambiar el estatus Maestro-Esclavo de un ticket, en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Muestra una lista de todos los agentes involucrados con este ticket, en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Muestra una lista de todos los posibles agentes (todos los agentes con permisos "nota" en la fila/ticket) para determinar quién deberá ser informado acerca de esta nota, en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Muestra las opciones de prioridad de ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        ' Muestra el campo de título en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Slave Tickets'} = 'Ticket Esclavos';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        'Especifica los diferentes canales de comunicación de artículos en los cuales se reemplazará el nombre real de ticket Maestro con el nombre del ticket Esclavo.';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Este módulo activa el campo Maestro/Esclavo en una nueva pantalla de ticket por email o por teléfono.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        'Esta configuración es obsoleta y será removida en futuras versiones de OTRSMasterSlave.';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Ticket MaestroEsclavo';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::es_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Gestionar el estado Maestro/Esclavo para  %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Nuevo ticket maestro';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Desactivar ticket maestro';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Desactivar ticket esclavo';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Esclavo de %s%s%s: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Desactivar tickets maestros';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Desactivar tickets esclavos';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Maestro';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Esclavo de %s%s%s';
    $Self->{Translation}->{'Master Ticket'} = 'Ticket Maestro';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Todos los Tickets maestros';
    $Self->{Translation}->{'All slave tickets'} = 'Todos los Tickets esclavos';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Permite añadir notas en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Modifica el estado Maestro-Esclavo del ticket.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Define el nombre del campo dinámico para la función de ticket maestro.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Define el bloqueo requerido in la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente (si el ticket no ha sido bloqueado aun, el ticket se bloquea y el agente actual se convertirá automáticamente en el dueño del mismo).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        'Define si la nota MaestroEsclavo es visible para el cliente por defecto.';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define el siguiente estado del ticket después de añadir una nota, en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define la prioridad por defecto en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Define el comentario del historial para la acción de la pantalla del ticket Maestro-Esclavo, que es usado para el historial del ticket en la interfaz del agente.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Define el tipo de historial para la acción de la pantalla del ticket Maestro-Esclavo, que es usado para el historial del ticket en la interfaz del agente.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define el siguiente estado después de añadir una nota, en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Activa la parte Maestro-Esclavo avanzada de la función.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Activa la función de que los tickets esclavos siguen el ticket maestro a un nuevo maestro en el modo Maestro-Esclavo avanzado.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Activa la función que cambie el estado Maestro-Esclavo de un ticket en el modo Maestro-Esclavo avanzado.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Habilita la característica para reenviar artículos desde el tipo \'reenviar\' de un ticket maestro a los clientes de los tickets esclavos. Por defecto (deshabilitado) no reenviará artículos del tipo \'reenviar\' a los tickets esclavos.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Permite que la característica para mantener el enlace padre-hijo después del cambio de estado MaestroEsclavo en el modo avanzado MaestroEsclavo.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Habilita la característica para mantener el enlace padre-hijo después de desactivar el estado MaestroEsclavo en el avanzado de MaestroEsclavo.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Habilita la característica para desactivar el estado de MaestroEsclado de un ticket en el modo avanzado de MaestroEsclavo.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Si una nota es añadida por un agente, fija el estado del ticket en en la pantalla de ticket Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Master / Slave'} = 'Maestro / Esclavo';
    $Self->{Translation}->{'Master Tickets'} = 'Tickets Maestros';
    $Self->{Translation}->{'MasterSlave'} = 'Maestro-Esclavo';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Módulo Esclavo Maestro para mayor características de entradas.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parámetros de el dashboard backend de la vista de los tickets maestros de la interfaz del agente. "Limite" es el numero de entradas a mostrar por defecto. "Grupo" es usado para restringir el acceso a el plugin (E.J. Grupo: admin;grupo1;grupo2;). "Defecto" determina si el plugin es activado por defecto ó si el usuario necesita activarlo manualmente. "CacheTTLLocal" es hora cache en minutos para el plugin.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametros del backend del panel principal de la vista del ticket esclavo del interface del agente. "Limite" es el número de entrada mostradas por defecto. "Grupo"es usado para restringir el acceso al plugin (Ejm. Grupo: admin;group1;group2;). "Defecto" determina si el plugin es habilitado por defecto o si el usuario necesita habilitarlo manualmente. "CacheTTLLocal" es el tiempo de cache en minutos para el plugin';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Registro del módulo de evento de tickets.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Permisos requeridos para usar la pantalla Maestro-Esclavo de un ticket, en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Establece si el agente debe seleccionar el campo Maestro / Esclavo.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Fija el texto del cuerpo por defecto de las notas añadidas en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Fija el asunto por defecto de las notas añadidas en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Fija el agente responsable de un ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Fija el servicio de un ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente (Ticket::Service require ser activado).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Fija el dueño de un ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Fija el tipo de ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket en la interfaz del agente (Ticket::Type require ser activado).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Muestra un enlace en el menú para cambiar el estatus Maestro-Esclavo de un ticket, en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Muestra una lista de todos los agentes involucrados con este ticket, en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Muestra una lista de todos los posibles agentes (todos los agentes con permisos "nota" en la fila/ticket) para determinar quién deberá ser informado acerca de esta nota, en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Muestra las opciones de prioridad de ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Muestra el campo título del ticket en la pantalla Maestro-Esclavo en la vista detallada de dicho ticket de la interfaz del agente.';
    $Self->{Translation}->{'Slave Tickets'} = 'Tickets esclavo';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Este módulo activa el campo Maestro-Esclavo en la pantalla de un nuevo ticket de email o de teléfono.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Ticket Maestro-Esclavo.';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::fr_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Gérer le statut maître-esclave pour %s %s %s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Nouveau ticket maître';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Ne plus définir ce ticket en tant que ticket maître';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Ne plus définir ce ticket en tant que ticket esclave';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Esclave du %s %s %s : %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Ne plus définir ces tickets en tant que tickets maîtres';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Ne plus définir ces tickets en tant que tickets esclaves';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Maître';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Esclave du %s %s %s';
    $Self->{Translation}->{'Master Ticket'} = 'Ticket maître';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Tous les tickets maîtres';
    $Self->{Translation}->{'All slave tickets'} = 'Tous les tickets esclaves';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Autorise l\'ajout de notes d\'un ticket maître / esclave depuis la vue détaillée de l\'interface opérateur.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Changer l\'état maître / esclave d\'un ticket.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = '';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        '';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = '';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Master / Slave'} = 'Maître / Esclave';
    $Self->{Translation}->{'Master Tickets'} = 'Tickets maîtres';
    $Self->{Translation}->{'MasterSlave'} = 'Maître / Esclave';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Registration of the ticket event module.'} = '';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        '';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = 'Tickets esclaves';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        '';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Ticket maître / esclave.';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::gl_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = '';
    $Self->{Translation}->{'Unset Master Ticket'} = '';
    $Self->{Translation}->{'Unset Slave Ticket'} = '';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = '';
    $Self->{Translation}->{'Unset Slave Tickets'} = '';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Principal';
    $Self->{Translation}->{'Slave of %s%s%s'} = '';
    $Self->{Translation}->{'Master Ticket'} = '';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = '';
    $Self->{Translation}->{'All slave tickets'} = '';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Cambiar o estado AmoEscravo do tícket.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = '';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        '';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = '';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Master / Slave'} = '';
    $Self->{Translation}->{'Master Tickets'} = '';
    $Self->{Translation}->{'MasterSlave'} = '';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Registration of the ticket event module.'} = '';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        '';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = '';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        '';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::hu_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '%s%s%s mester/alárendelt állapotának kezelése';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Új mesterjegy';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Mesterjegy törlése';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Alárendelt jegy törlése';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '%s%s%s alárendeltje: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Mesterjegyek törlése';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Alárendelt jegyek törlése';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Mester';
    $Self->{Translation}->{'Slave of %s%s%s'} = '%s%s%s alárendeltje';
    $Self->{Translation}->{'Master Ticket'} = 'Mesterjegy';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Összes mesterjegy';
    $Self->{Translation}->{'All slave tickets'} = 'Összes alárendelt jegy';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Lehetővé teszi jegyzetek hozzáadását egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Változtassa meg a jegy mester-alárendelt állapotát.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Meghatározza a dinamikus mező nevét a mesterjegy funkciónál.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Meghatározza, hogy szükséges-e jegyzárolás egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen (ha a jegy még nincs zárolva, akkor a jegy zárolva lesz, és az aktuális ügyintéző automatikusan annak tulajdonosaként lesz beállítva).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        'Meghatározza, hogy a mester-alárendelt jegyzet alapértelmezetten látható legyen az ügyfélnek.';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Meghatározza egy jegy alapértelmezett következő állapotát egy jegyzet hozzáadása után egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Meghatározza az alapértelmezett jegyprioritást egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Azt az előzmény megjegyzést határozza meg a jegy mester-alárendelt képernyő műveleténél, amelyet a jegy előzményeinél szoktak használni az ügyintézői felületen.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Azt az előzmény típust határozza meg a jegy mester-alárendelt képernyő műveleténél, amelyet a jegy előzményeinél szoktak használni az ügyintézői felületen.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Meghatározza egy jegy következő állapotát egy jegyzet hozzáadása után egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Engedélyezi a funkció speciális mester-alárendelt részét.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Engedélyezi azt a funkciót, hogy az alárendelt jegyek a mesterjegyet kövessék egy új mesterhez a speciális mester-alárendelt módban.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Engedélyezi a funkciót egy jegy mester-alárendelt állapotának megváltoztatásához a speciális mester-alárendelt módban.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Engedélyezi a funkciót a bejegyzések továbbításához egy mesterjegy „továbbítás” típusából az alárendelt jegyek ügyfeleihez. Alapértelmezetten (letiltva) nem fog bejegyzéseket továbbítani a „továbbítás” típusból az alárendelt jegyekhez.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Engedélyezi a funkciót a szülő-gyermek kapcsolat megtartásához a mester-alárendelt állapot megváltoztatása után a speciális mester-alárendelt módban.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Engedélyezi a funkciót a szülő-gyermek kapcsolat megtartásához a mester-alárendelt állapot törlése után a speciális mester-alárendelt módban.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Engedélyezi a funkciót egy jegy mester-alárendelt állapotának törléséhez a speciális mester-alárendelt módban.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Ha egy ügyintéző hozzáadott egy megjegyzést, akkor beállítja a jegy állapotát egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Master / Slave'} = 'Mester / alárendelt';
    $Self->{Translation}->{'Master Tickets'} = 'Mesterjegyek';
    $Self->{Translation}->{'MasterSlave'} = 'Mester-alárendelt';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Mester-alárendelt modul a jegy tömeges funkciójához.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Paraméterek az ügyintézői felület mesterjegyei áttekintőjének vezérlőpult háttérprogramjához. A „Limit” az alapértelmezetten megjelenített bejegyzések száma. A „Group” használható a hozzáférés korlátozásához a bővítményre (például Group: admin;csoport1;csoport2;). A „Default” jelzi, hogy a bővítmény alapértelmezetten engedélyezve van, vagy hogy a felhasználónak kézzel kell engedélyeznie azt. A „CacheTTLLocal” a bővítmény gyorsítótár ideje percben.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Paraméterek az ügyintézői felület alárendelt jegyei áttekintőjének vezérlőpult háttérprogramjához. A „Limit” az alapértelmezetten megjelenített bejegyzések száma. A „Group” használható a hozzáférés korlátozásához a bővítményre (például Group: admin;csoport1;csoport2;). A „Default” jelzi, hogy a bővítmény alapértelmezetten engedélyezve van, vagy hogy a felhasználónak kézzel kell engedélyeznie azt. A „CacheTTLLocal” a bővítmény gyorsítótár ideje percben.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'A jegy esemény modul regisztrációja.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'A szükséges jogosultságok egy nagyított jegynek a jegy mester-alárendelt képernyőjének használatához az ügyintézői felületen.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Beállítja, hogy az ügyintézőnek ki kell választania a mester/alárendelt mezőt.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Beállítja az alapértelmezett törzsszöveget egy nagyított jegynek a jegy mester-alárendelt képernyőjén hozzáadott jegyzeteknél az ügyintézői felületen.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Beállítja az alapértelmezett tárgyat egy nagyított jegynek a jegy mester-alárendelt képernyőjén hozzáadott jegyzeteknél az ügyintézői felületen.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Beállítja a jegy felelős ügyintézőjét egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Beállítja a szolgáltatást egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen (a Ticket::Service lehetőségnek aktiválva kell lennie).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Beállítja a jegy tulajdonosát egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Beállítja a jegy típusát egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen (a Ticket::Type lehetőségnek aktiválva kell lennie).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben egy jegy mester-alárendelt állapotának megváltoztatásához az ügyintézői felület jegynagyítás nézetében.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Megjeleníti a jegyen részt vevő összes ügyintéző listáját egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Megjeleníti az összes lehetséges ügyintéző (a várólistán vagy jegyen jegyzet jogosultsággal rendelkező összes ügyintéző) listáját egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen annak meghatározásához, hogy kit kell tájékoztatni erről a jegyzetről.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Megjeleníti a jegy prioritási lehetőségeket egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Megjeleníti a címmezőt egy nagyított jegynek a jegy mester-alárendelt képernyőjén az ügyintézői felületen.';
    $Self->{Translation}->{'Slave Tickets'} = 'Alárendelt jegyek';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        'Megadja a különböző bejegyzés kommunikációs csatornákat, ahol a mesterjegyben lévő valódi név ki lesz cserélve az alárendelt jegyben lévővel.';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Ez a modul bekapcsolja a mester/alárendelt mezőt az új e-mail és telefonos jegy képernyőkön.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        'Ez a beállítás elavult, és el lesz távolítva az OTRSMasterSlave későbbi verzióiból.';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Jegy mester-alárendelt.';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::id_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Tiket master baru';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Tidak perlu set tiket master';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Lepaskan set tiket slave';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Melepas tiket master';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Melepaskan tiket slave';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Induk';
    $Self->{Translation}->{'Slave of %s%s%s'} = '';
    $Self->{Translation}->{'Master Ticket'} = '';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Semua tiket master';
    $Self->{Translation}->{'All slave tickets'} = 'Semua tiket slave';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Memungkinkan menambahkan catatan dalam layar tiket master slave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Mengubah keadaan Master slave tiket.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Mendefinisikan nama field yang dinamis untuk fitur utama tiket.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Mendefinisikan jika kunci tiket diperlukan di layar MasterSlave tiket dari tiket yang diperbesar di antarmuka agen (jika tiket tidak terkunci lagi, tiket akan terkunci dan agen saat ini akan diatur secara otomatis sebagai pemiliknya).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mendefinisikan default state berikutnya tiket setelah menambahkan catatan, dalam tiket Guru layar Slave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mendefinisikan prioritas tiket default di layar tiket Master Slave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Mendefinisikan komentar sejarah untuk tindakan tiket masterslave pada layar, yang akan digunakan untuk sejarah tiket di antarmuka agen.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Mendefinisikan jenis sejarah untuk tindakan tiket master slave di layar, yang akan digunakan untuk sejarah tiket di antarmuka agen.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mendefinisikan state berikutnya tiket setelah menambahkan catatan, dalam layar tiket Master Slave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Memungkinkan Master slave bagian dari fitur tersebut.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Mengaktifkan fitur tiket slave untuk mengikuti tiket master baru dalam modus Master Slave canggih.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Memungkinkan fitur untuk mengubah keadaan Master slave dari tiket dalam modus Master Slave canggih.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Memungkinkan fitur untuk meneruskan artikel dari jenis \'maju\' dari tiket master untuk pelanggan dari tiket slave. Secara default (cacat) itu tidak akan meneruskan artikel dari jenis \'maju\' untuk tiket slave.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Memungkinkan fitur untuk menjaga link orangtua-anak setelah perubahan dari state Master slave dalam modus Master Slave canggih.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Memungkinkan fitur untuk menjaga link orangtua-anak setelah diset dari negara Master slave dalam modus Master Slave canggih.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Memungkinkan fitur yang tidak di set Master slave dari tiket dalam modus Master Slave canggih.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Jika catatan ditambahkan oleh agen, menetapkan keadaan tiket di tiket layar MasterSlave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Master / Slave'} = 'Master/Slave';
    $Self->{Translation}->{'Master Tickets'} = 'Tiket Master';
    $Self->{Translation}->{'MasterSlave'} = 'MasterSlave';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Modul MasterSlave untuk fitur tiket';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parameter untuk backend dashboard tiket menguasai gambaran dari antarmuka agen. "Batas" adalah jumlah entri yang ditampilkan secara default. "Grup" digunakan untuk membatasi akses ke plugin (e g Grup: Admin;group1,group2;). "Default" menentukan apakah plugin diaktifkan secara default atau jika pengguna perlu mengaktifkannya secara manual. "CacheTTLLocal" adalah waktu cache di menit untuk plugin.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parameter untuk backend dashboard tiket budak gambaran dari antarmuka agen. "Batas" adalah jumlah entri yang ditampilkan secara default. "Grup" digunakan untuk membatasi akses ke plugin (e g Grup: Admin;group1,group2;). "Default" menentukan apakah plugin diaktifkan secara default atau jika pengguna perlu mengaktifkannya secara manual. "CacheTTLLocal" adalah waktu cache di menit untuk plugin.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Pendaftaran modul acara tiket.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'izin yang diperlukan untuk menggunakan laya tiket MasterSlave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mengatur teks tubuh default untuk catatan ditambahkan dalam layar tiket Master Slave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mengatur Teks Tubuh bawaan untuk review Catatan ditambahkan hearts Tiket Layar Slave Dari Tiket Yang diperbesar di Antarmuka agen.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Set agen yang bertanggung jawab dari tiket di layar tiket Master Slave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Set layanan dalam tiket Guru layar Slave dari tiket yang diperbesar di antarmuka agen (Ticket::Service needs to be activated). ';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Menetapkan pemilik tiket di layar tiket Master Slave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Menetapkan jenis tiket di layar tiket Master Slave dari tiket yang diperbesar di antarmuka agen  (Ticket::Type needs to be activated).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Menunjukkan link dalam menu untuk mengubah status Master slave dari tiket dalam tampilan zoom tiket dari antarmuka agen.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Menunjukkan daftar semua agen yang terlibat pada tiket ini, dalam layar tiket Master Slave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Menunjukkan daftar semua agen yang mungkin (semua agen dengan izin catatan di antrian / tiket) untuk menentukan siapa yang harus diberitahu tentang catatan ini, di layar MasterSlave tiket dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Menunjukkan pilihan prioritas tiket di tiket Guru layar Slave dari tiket yang diperbesar di antarmuka agen.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = 'Tiket slave';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Modul ini mengaktifkan bidang Master / Slave di baru layar email dan tiket telepon.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Tiket MasterSlave.';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::it_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Gestisci lo stato Master / Slave per %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Nuovo ticket primario';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Rimuovi ticket primario';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Rimuovi ticket secondario';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Schiavo di: %s%s%s: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Ticket master non impostati';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Ticket schiavi non impostati';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Primario';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Schiavo di %s%s%s';
    $Self->{Translation}->{'Master Ticket'} = 'Ticket maestro';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Tutti i ticket maestri';
    $Self->{Translation}->{'All slave tickets'} = 'Tutti i ticket schiavi';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Consente di aggiungere delle note nella schermata MasterSlave di un ticket nell\'interfaccia Agente';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Cambia lo stato MasterSlave del ticket';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Definisce il nome del campo dinamico per la funzione ticket principale.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Definisce se è necessario un blocco ticket nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti (se il ticket non è ancora bloccato, il ticket viene bloccato e l\'agente corrente verrà impostato automaticamente come proprietario).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        'Definisce se la nota MasterSlave è visibile per il cliente per impostazione predefinita.';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Definisce lo stato successivo predefinito di un ticket dopo aver aggiunto una nota, nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Definisce la priorità predefinita del ticket nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia dell\'interfaccia agenti.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Definisce il commento cronologico per l\'azione sullo schermo MasterSlave del ticket, che viene utilizzato per la cronologia dei ticket nell\'interfaccia agenti.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Definisce il tipo di cronologia per l\'azione della schermata MasterSlave del ticket, che viene utilizzata per la cronologia dei ticket nell\'interfaccia agenti.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Definisce lo stato successivo di un ticket dopo aver aggiunto una nota, nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia dell\'interfaccia agenti.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Abilita la parte avanzata di MasterSlave della funzione.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Abilita la funzione che i ticket slave seguono il ticket master verso un nuovo master in modalità MasterSlave avanzata.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Abilita la funzione per cambiare lo stato MasterSlave di un ticket nella modalità MasterSlave avanzata.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Abilita la funzione per inoltrare articoli dal tipo \'inoltra\' di un ticket principale ai clienti dei ticket slave. Per impostazione predefinita (disabilitato) non inoltrerà articoli dal tipo "inoltra" ai ticket schiavo.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Abilita la funzione per mantenere il collegamento genitore-figlio dopo la modifica dello stato MasterSlave nella modalità MasterSlave avanzata.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Abilita la funzione per mantenere il collegamento genitore-figlio dopo aver disinserito lo stato MasterSlave nella modalità MasterSlave avanzata.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Abilita la funzione a disinserire lo stato MasterSlave di un ticket nella modalità MasterSlave avanzata.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Se una nota viene aggiunta da un agente, imposta lo stato del ticket nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Master / Slave'} = 'Primario / Secondario';
    $Self->{Translation}->{'Master Tickets'} = 'Ticket maestro';
    $Self->{Translation}->{'MasterSlave'} = 'MasterSlave';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Modulo MasterSlave per la funzione Bulk Ticket.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametri per il backend del dashboard della panoramica dei ticket principali dell\'interfaccia agenti. "Limite" è il numero di voci visualizzate per impostazione predefinita. "Gruppo" viene utilizzato per limitare l\'accesso al plug-in (ad es. Gruppo: admin; gruppo1; gruppo2;). "Predefinito" determina se il plug-in è abilitato per impostazione predefinita o se l\'utente deve abilitarlo manualmente. "CacheTTLLocal" è il tempo di cache in minuti per il plugin.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametri per il backend del dashboard della panoramica dei ticket slave dell\'interfaccia dell\'interfaccia agenti. "Limite" è il numero di voci visualizzate per impostazione predefinita. "Gruppo" viene utilizzato per limitare l\'accesso al plug-in (ad es. Gruppo: admin; gruppo1; gruppo2;). "Predefinito" determina se il plug-in è abilitato per impostazione predefinita o se l\'utente deve abilitarlo manualmente. "CacheTTLLocal" è il tempo di cache in minuti per il plugin.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Registrazione del modulo di evento del ticket.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Permessi necessari per utilizzare la schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Imposta se il campo Master / Slave deve essere selezionato dall\'agente.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Imposta il corpo del testo predefinito per le note aggiunte nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia dell\'interfaccia agenti.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Imposta l\'oggetto predefinito per le note aggiunte nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Imposta l\'agente responsabile del ticket nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Imposta il servizio nella schermata del ticket MasterSlave di un ticket ingrandito nell\'interfaccia agenti (Ticket :: Il servizio deve essere attivato).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Imposta il proprietario del ticket nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Imposta il tipo di ticket nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti (Ticket::Type deve essere attivato).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Mostra un collegamento nel menu per modificare lo stato MasterSlave di un ticket nella vista zoom ticket nell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mostra un elenco di tutti gli agenti coinvolti su questo ticket, nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mostra un elenco di tutti i possibili agenti (tutti gli agenti con i permessi note sulla coda/ticket) per determinare chi deve essere informato su questa nota, nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mostra le opzioni di priorità del ticket nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mostra il campo del titolo nella schermata MasterSlave del ticket di un ticket ingrandito nell\'interfaccia agenti.';
    $Self->{Translation}->{'Slave Tickets'} = 'Ticket schiavi';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        'Specifica i diversi canali di comunicazione dell\'articolo in cui il nome reale del ticket Master verrà sostituito con quello nel ticket Slave.';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Questo modulo attiva il campo Primario/Secondario nelle schermate dei nuovi ticket tramite email e telefono.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        'Questa impostazione è obsoleta e verrà rimossa in ulteriori versioni di OTRSMasterSlave.';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Ticket MasterSlave.';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::ja_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '%s%s%sのマスタ/スレーブステータスを管理する';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = '新しいマスターチケット';
    $Self->{Translation}->{'Unset Master Ticket'} = 'マスターチケットを解除する';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'スレーブチケットを解除する';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '%s%s%sのスレーブ: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'マスターチケットを解除する';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'スレーブチケットを解除する';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'マスター';
    $Self->{Translation}->{'Slave of %s%s%s'} = '';
    $Self->{Translation}->{'Master Ticket'} = 'マスター チケット';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'すべてのマスターチケット';
    $Self->{Translation}->{'All slave tickets'} = 'すべてのスレーブチケット';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'エージェントインタフェースのズームされたチケットのマスタースレイブ画面で、注釈の追加を許可します。';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'チケットのマスター・スレーブ状況を変更する';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'マスター・スレーブ動的フィールド名を設定する';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        '';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '担当者インターフェイス のズームされたチケットのマスター・スレーブ画面で、デフォルトの チケット 優先度を定義します。';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '担当者インタフェースのチケット履歴で使用される、チケット・マスター・スレーブ画面アクションに関する履歴コメントを定義します。';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '担当者インタフェースのチケット履歴で使用される、チケット・マスター・スレーブ画面アクションに関する履歴タイプを定義します。';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = '';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Master / Slave'} = 'マスター / スレーブ';
    $Self->{Translation}->{'Master Tickets'} = 'マスターチケット';
    $Self->{Translation}->{'MasterSlave'} = 'マスター・スレーブ';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'MasterSlave モジュールのチケット一括処理機能';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'チケットイベントモジュールの登録';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '担当者インタフェースで、ズームされたチケットのマスター・スレーブ画面を使用するための必要な許可です。';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '担当者インタフェースのズームされたチケットのマスター・スレーブ画面で、追加されたメモのデフォルトの本文を設定します。';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '担当者インタフェースのズームされたチケットのマスター・スレーブ画面で、追加されたメモのデフォルトの件名を設定します。';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '担当者インタフェースのズームされたチケットのマスター・スレーブ画面で、チケットの責任を有する担当者を設定します。';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        '担当者インタフェースのズームされたチケットのチケットのマスター・スレーブ画面で、サービスを設定します (Ticket::Serviceを有効とする必要があります)。';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '担当者インタフェースのズームされたチケットのマスター・スレーブ画面で、チケット所有者を設定します。';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        '担当者インタフェースのズームされたチケットのチケットのマスター・スレーブ画面で、チケット・タイプを設定します(Ticket::Typeを有効とする必要があります)。';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = 'スレーブチケット';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'このモジュールは、新規メール/電話チケットに、マスター/スレーブチケット欄を追加します。';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'マスター・スレーブチケット';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::ms_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Tiket Master Baharu';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Tiket Master Tidak Ditetapkan';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Tiket Slave Tidak Ditetapkan';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = '';
    $Self->{Translation}->{'Unset Slave Tickets'} = '';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Induk';
    $Self->{Translation}->{'Slave of %s%s%s'} = '';
    $Self->{Translation}->{'Master Ticket'} = '';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = '';
    $Self->{Translation}->{'All slave tickets'} = '';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Membenarkan menambah nota dalam skrin MasterSlave tiket bagi tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Ubah keadaan tiket MasterSlave.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = '';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Mentakrifkan jika kunci tiket diperlukan dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen (jika tiket itu tidak dikunci lagi, tiket yang akan dikunci dan ejen semasa akan ditetapkan secara automatik sebagai pemiliknya).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mentakrifkan keadaan default seterusnya bagi tiket selepas menambah nota, dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mentakrifkan keutamaan tiket default dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Mentakrifkan komen sejarah bagi kesan skrin MasterSlave tiket, yang akan digunakan untuk sejarah tiket dalam antara muka ejen.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Mentakrifkan jenis sejarah bagi tindakan skrin tiket MasterSlave, yang akan digunakan untuk sejarah tiket dalam antara muka ejen.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mentakrifkan keadaan seterusnya bagi tiket selepas menambah nota, dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = '';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Jika nota ditambah oleh ejen, tetapkan keadaan tiket dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Master / Slave'} = 'Master / Slave';
    $Self->{Translation}->{'Master Tickets'} = '';
    $Self->{Translation}->{'MasterSlave'} = '';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Modul MasterSlave untuk ciri Bulk Tiket.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parameter untuk bahagian belakang papan pemuka daripada gambaran keseluruhan tiket master antara muka ejen. "Had" adalah bilangan penyertaan ditunjukkan secara default. "Kumpulan" digunakan untuk menyekat akses kepada masuk (contoh: Kumpulan: admin; kumpulan1; kumpulan2;). "Default" menentukan jika plugin ini diaktifkan secara default atau jika pengguna perlu untuk membolehkan secara manual. "CacheTTLLocal" adalah masa cache dalam minit untuk plugin.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parameter untuk bahagian belakang papan pemuka daripada gambaran keseluruhan tiket slave antara muka ejen. "Had" adalah bilangan penyertaan ditunjukkan secara default. "Kumpulan" digunakan untuk menyekat akses kepada masuk (contoh: Kumpulan: admin; kumpulan1; kumpulan2;). "Default" menentukan jika plugin ini diaktifkan secara default atau jika pengguna perlu untuk membolehkan secara manual. "CacheTTLLocal" adalah masa cache dalam minit untuk plugin.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Pendaftaran tiket modul acara.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Kebenaran yang diperlukan untuk menggunakan skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Menetapkan teks badan default untuk nota ditambah dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Set subjek default untuk nota ditambah dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Set ejen yang bertanggungjawab bagi tiket dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Set perkhidmatan di skrin MasterSlave tiket tiket dizum dalam antara muka ejen (Tiket::Perkhidmatan perlu diaktifkan).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Set pemilik tiket dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Menetapkan jenis tiket yang dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen (Tiket::Jenis perlu diaktifkan).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Menunjukkan pautan dalam menu untuk menukar status MasterSlave bagi tiket dalam pandangan zum tiket bagi antara muka ejen.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Memaparkan senarai semua ejen yang terlibat dalam tiket ini, dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Memaparkan senarai semua kemungkinan ejen (semua ejen dengan kebenaran nota mengenai barisan/tiket) untuk menentukan siapa yang perlu diberitahu tentang nota ini, dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Menunjukkan pilihan keutamaan tiket dalam skrin MasterSlave tiket tiket dizum dalam antara muka ejen.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = '';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Modul ini mengaktifkan medan Master/Slave di skrin emel dan tiket telefon baru.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::pl_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Ustaw stan Nadrzędny/Podrzędny dla %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Nowe zgłoszenie nadrzędne';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Wyłącz wartość Nadrzędny';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Wyłącz wartość Podrzędny';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Podrzędne do %s%s%s: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Wyczyść wartość Nadrzędny';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Wyczyść wartość Podrzędny';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Nadrzędny';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Podrzędny do %s%s%s';
    $Self->{Translation}->{'Master Ticket'} = 'Zgłoszenie nadrzędne';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Wszystkie zgłoszenia nadrzędne';
    $Self->{Translation}->{'All slave tickets'} = 'Wszystkie zgłoszenia podrzędne';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Pozwala na dodanie notatki na ekranie Nadrzędny/Podrzędny, wywołanym z przybliżonego widoku zgłoszenia w interfejsie agenta.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Zmień stan zgłoszenia dla Nadrzędny/Podrzędny.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Określa nazwę pola dynamicznego dla funkcjonalności zgłoszenia nadrzędnego.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Określa, czy wymagana jest zablokowanie zgłoszenia na ekranie Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta (jeśli zgłoszenie nie jest jeszcze zablokowane, staje się zablokowane, a bieżący agent staje się jego właścicielem).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Określa domyślny następny stan zgłoszenia po dodaniu notatki na ekranie Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Określa domyślny priorytet zgłoszenia na ekranie Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Określa komentarz historii dla akcji zgłoszenia na ekranie Nadrzędny/Podrzędny który zostanie użyty w historii zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Określa typ historii dla akcji zgłoszenia na ekranie Nadrzędny/Podrzędny który zostanie użyty w historii zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Określa następny stan zgłoszenia po dodaniu notatki na ekranie Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Włącza zaawansowaną część funkcjonalności Nadrzędny/Podrzędny.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Włącza funkcjonalność trybu zaawansowanego Nadrzędny/Podrzędny, w której zgłoszenia podrzędne podążają za zgłoszeniem nadrzędnym do nowego zgłoszenia nadrzędnego.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Włącza funkcjonalność trybu zaawansowanego Nadrzędny/Podrzędny umożliwiającą zmianę stanu Nadrzędny/Podrzędny w zgłoszeniu.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Włącza funkcjonalność przesyłania artykułów typu "przekaż" ze zgłoszenia nadrzędnego do klientów zgłoszeń podrzędnych. Domyślnie (wyłączone) artykuły typu "przekaż" nie są przekazywane do zgłoszeń podrzędnych.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Włącza funkcjonalność trybu zaawansowanego Nadrzędny/Podrzędny, polegającą na utrzymaniu połączenia rodzic-dziecko po zmianie zależności Nadrzędny/Podrzędny.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Włącza funkcjonalność trybu zaawansowanego Nadrzędny/Podrzędny, polegającą na utrzymaniu połączenia rodzic-dziecko po usunięciu zależności Nadrzędny/Podrzędny.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Włącza funkcjonalność trybu zaawansowanego Nadrzędny/Podrzędny, usuwania stanu Nadrzędny/Podrzędny ze zgłoszenia.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Jeśli agent doda notatkę, to ustawiany jest stan zgłoszenia na ekranie Nadrzędny/Podrzędny okna szczegółów zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Master / Slave'} = 'Nadrzędny / Podrzędny';
    $Self->{Translation}->{'Master Tickets'} = 'Zgłoszenia nadrzędne';
    $Self->{Translation}->{'MasterSlave'} = 'Nadrzędny/Podrzędny';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Moduł Nadrzędnych/Podrzędnych dla funkcji grupowych zgłoszeń.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametr okna pulpitu zgłoszeń nadrzędnych interfejsu agenta. "Limit" jest liczbą wierszy pokazywaną domyślnie. "Group" jest używana do ograniczenia dostępu do wtyczki (np.: Group: admin;group1;group2;). "Default" określa czy wtyczka jest domyślnie włączona lub czy użytkownik może włączyć ją ręcznie. "CacheTTLLocal" jest czasem podanym w minutach określającym czas cache\'owania wtyczki.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametr okna pulpitu zgłoszeń podrzędnych interfejsu agenta. "Limit" jest liczbą wierszy pokazywaną domyślnie. "Group" jest używana do ograniczenia dostępu do wtyczki (np.: Group: admin;group1;group2;). "Default" określa czy wtyczka jest domyślnie włączona lub czy użytkownik może włączyć ją ręcznie. "CacheTTLLocal" jest czasem podanym w minutach określającym czas cache\'owania wtyczki.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Rejestracja modułu zdarzeń zgłoszenia,';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Uprawnienia wymagane do używania ekranu Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Określa, czy pole Nadrzędny/Podrzędny musi być wybrane przez agenta.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Ustawia domyślną treść notatki dodawanej do zgłoszenia na ekranie Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Ustawia domyślny temat dla notatek dodawanych do zgłoszenia na ekranie Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Ustawia odpowiedzialnego za zgłoszenie agenta na ekranie Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Ustawia usługę w zgłoszeniu na ekranie Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta (Ticket::Service musi być aktywowane).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Ustawia właściciela zgłoszenia na ekranie Nadrzędny/Podrzędny w oknie szczegółów zgłoszenia interfejsu agenta.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Ustawia typ zgłoszenia na ekranie Nadrzędny/Podrzędny, wywołanym z przybliżonego widoku zgłoszenia w interfejsie agent (Ticket::Type musi być aktywowany).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Na ekranie przybliżonego widoku zgłoszenia w interfejsie agenta, pokazuje link w menu do zmiany stanu Nadrzędny/Podrzędny.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Na ekranie Nadrzędny/Podrzędny, wywołanym z przybliżonego widoku zgłoszenia w interfejsie agenta, pokazuje listę wszystkich agentów zaangażowanych w zgłoszenie.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Na ekranie Nadrzędny/Podrzędny, wywołanym z przybliżonego widoku zgłoszenia w interfejsie agenta, pokazuje listę wszystkich możliwych agentów (wszystkich agentów z uprawnieniami do notatek w kolejce/zgłoszeniu), w celu określenia kto powinien być poinformowany o tej notatce.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Pokazuje opcje priorytetu zgłoszenia na ekranie Nadrzędny/Podrzędny, wywołanym z przybliżonego widoku zgłoszenia w interfejsie agenta';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Pokazuje pole tytułu na ekranie Nadrzędny/Podrzędny w szczegółach zgłoszenia w intefejsie agenta.';
    $Self->{Translation}->{'Slave Tickets'} = 'Zgłoszenia podrzędne';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Ten moduł aktywuje pole Nadrzędny/Podrzędny w oknie nowego zgłoszenia e-mail i telefonicznego';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Zgłoszenie Nadrzędne/Podrzędne';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::pt_BR_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Gerenciar estado de Mestre/Escravo para %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Novo Chamado Mestre';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Desfazer Chamado Mestre';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Desfazer Chamado Escravo';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Escravo de %s%s%s: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Limpar Chamados Mestres';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Limpar Chamados Escravos';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Mestre';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Escravo de %s%s%s';
    $Self->{Translation}->{'Master Ticket'} = 'Chamado Mestre';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Todos chamados mestres';
    $Self->{Translation}->{'All slave tickets'} = 'Todos chamados escravos';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Permite adicionar notas na tela "MasterSlave"';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Altera o estado MasterSlave de um chamado.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Define o nome do campo dinâmico para a funcionalidade de chamado mestre.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Define se um bloqueio de ticket é requerido na tela MasterSlave de um ticket detalhado na interface de agente (se o ticket ainda não estiver bloqueado, o ticket será bloqueado e o agente corrente será automaticamente definido como o seu proprietário).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        'Define se a nota MestreEscravo fica visível para o cliente como padrão.';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o próximo estado padrão de um ticket após uma nota ter sido adicionada, na tela MasterSlave de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define a prioridade padrão do ticket na tela MasterSlave de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Define o comentário de histórico para a ação de tela MasterSlave do ticket, que é utilizada para o histórico do ticket na interface de agente.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Define o tipo de histórico para a ação da tela MasterSlave do ticket, que é utilizado para o histórico do ticket na interface de agente.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o próximo estado de um ticket após uma nota ter sido adicionada, na tela MasterSlave de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Habilita as funcionalidades avançadas do MasterSlave.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade de chamados escravos seguirem chamados mestre para um novo mestre no modo avançado do MasterSlave.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade de alterar estado do MasterSlave de um chamado no modo avançado.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Habilita a funcionalidade de encaminhar artigos do tipo \'forward\' de um chamado mestre para os clientes dos chamados escravos. Por padrão (desabilitado) os artigos não são encaminhados.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade que mantem o link pai-filho após a alteração do estado de MasterSlave no modo avançado do MasterSlave.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade de manter a associação pai-filho depois de limpar o estado de MasterSlave no modo avançado.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade de limpar o estado MasterSlave de um chamado no modo avançado.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Se uma nota for adicionada por um agente, define o estado do ticket na tela de MasterSlave de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Master / Slave'} = 'Mestre / Escravo';
    $Self->{Translation}->{'Master Tickets'} = 'Chamados Mestres';
    $Self->{Translation}->{'MasterSlave'} = 'MestreEscravo';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Módulo MestreEscravo para funcionalidade de ação em massa de chamado.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parâmetros para o backend do painel de visão geral de tickets mestre da interface de agente. "Limite" é a quantidade de registros exibida por padrão. "Grupo" é usado para restringir o acesso ao plugin (ex.: Grupo: admin;grupo1;grupo2). "Padrão" determina se o plugin é habilitado por padrão ou se o usuário precisa habilitá-lo manualmente. "CacheTTLLocal" é o tempo de cache em minutos para o plugin.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parâmetros para o backend do painel de visão geral de tickets mestre da interface de agente. "Limite" é a quantidade de registros exibida por padrão. "Grupo" é usado para restringir o acesso ao plugin (ex.: Grupo: admin;grupo1;grupo2). "Padrão" determina se o plugin é habilitado por padrão ou se o usuário precisa habilitá-lo manualmente. "CacheTTLLocal" é o tempo de cache em minutos para o plugin.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Registro do módulo de eventos de ticket.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Permissões requeridas para usar a tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Define se o campo Mestre/ Escravo deve ser selecionado pelo agente.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o corpo do texto padrão para notas adicionadas na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o assunto padrão para notas adicionadas na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o agente responsável do ticket na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Define o serviço na tela MasterSlave de um ticket detalhado na interface de agente (Ticket::Service precisa ser ativado).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o proprietário do ticket na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Define o tipo do ticket na tela MasterSlave de ticket de um ticket detalhado na interface de agente (Ticket::Type precisa ser ativado).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Exibe um link no menu para alterar o estado MasterSlave de um ticket na tela de detalhe de ticket da interface de agente.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Exibe uma lista de todos os agentes envolvidos neste ticket, na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Exibe uma lista de todos os agentes possíveis (todos os agentes com permissão de nota na fila/ticket) para determinar quem deve ser informado sobre esta nota, na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Exibe as opções de prioridade de ticket na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mostra o título na tela de um chamado MestreEscravo de um chamado aberto na interface de agente.';
    $Self->{Translation}->{'Slave Tickets'} = 'Chamados Escravos';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Este módulo ativa o campo Mestre/Escravo nas telas de novo chamado fone/e-mail.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Chamado MestreEscravo';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::pt_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Gerenciar estado de Mestre/Escravo para %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Novo Chamado Mestre';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Desfazer Chamado Mestre';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Desfazer Chamado Escravo';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Escravo de %s%s%s: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Limpar Chamados Mestres';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Limpar Chamados Escravos';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Mestre';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Escravo de %s%s%s';
    $Self->{Translation}->{'Master Ticket'} = 'Chamado Mestre';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Todos chamados mestres';
    $Self->{Translation}->{'All slave tickets'} = 'Todos chamados escravos';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Permite adicionar notas na tela "MasterSlave"';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Altera o estado MasterSlave de um chamado.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Define o nome do campo dinâmico para a funcionalidade de chamado mestre.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Define se um bloqueio de ticket é requerido na tela MasterSlave de um ticket detalhado na interface de agente (se o ticket ainda não estiver bloqueado, o ticket será bloqueado e o agente corrente será automaticamente definido como o seu proprietário).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        'Define se a nota MestreEscravo fica visível para o cliente como padrão.';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o próximo estado padrão de um ticket após uma nota ter sido adicionada, na tela MasterSlave de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define a prioridade padrão do ticket na tela MasterSlave de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Define o comentário de histórico para a ação de tela MasterSlave do ticket, que é utilizada para o histórico do ticket na interface de agente.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Define o tipo de histórico para a ação da tela MasterSlave do ticket, que é utilizado para o histórico do ticket na interface de agente.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o próximo estado de um ticket após uma nota ter sido adicionada, na tela MasterSlave de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Habilita as funcionalidades avançadas do MasterSlave.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade de chamados escravos seguirem chamados mestre para um novo mestre no modo avançado do MasterSlave.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade de alterar estado do MasterSlave de um chamado no modo avançado.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Habilita a funcionalidade de encaminhar artigos do tipo \'forward\' de um chamado mestre para os clientes dos chamados escravos. Por padrão (desabilitado) os artigos não são encaminhados.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade que mantem o link pai-filho após a alteração do estado de MasterSlave no modo avançado do MasterSlave.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade de manter a associação pai-filho depois de limpar o estado de MasterSlave no modo avançado.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Habilita a funcionalidade de limpar o estado MasterSlave de um chamado no modo avançado.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Se uma nota for adicionada por um agente, define o estado do ticket na tela de MasterSlave de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Master / Slave'} = 'Mestre / Escravo';
    $Self->{Translation}->{'Master Tickets'} = 'Chamados Mestres';
    $Self->{Translation}->{'MasterSlave'} = 'MestreEscravo';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Módulo MestreEscravo para funcionalidade de ação em massa de chamado.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parâmetros para o backend do painel de visão geral de tickets mestre da interface de agente. "Limite" é a quantidade de registros exibida por padrão. "Grupo" é usado para restringir o acesso ao plugin (ex.: Grupo: admin;grupo1;grupo2). "Padrão" determina se o plugin é habilitado por padrão ou se o usuário precisa habilitá-lo manualmente. "CacheTTLLocal" é o tempo de cache em minutos para o plugin.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parâmetros para o backend do painel de visão geral de tickets mestre da interface de agente. "Limite" é a quantidade de registros exibida por padrão. "Grupo" é usado para restringir o acesso ao plugin (ex.: Grupo: admin;grupo1;grupo2). "Padrão" determina se o plugin é habilitado por padrão ou se o usuário precisa habilitá-lo manualmente. "CacheTTLLocal" é o tempo de cache em minutos para o plugin.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Registro do módulo de eventos de ticket.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Permissões requeridas para usar a tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Define se o campo Mestre/ Escravo deve ser selecionado pelo agente.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o corpo do texto padrão para notas adicionadas na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o assunto padrão para notas adicionadas na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o agente responsável do ticket na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Define o serviço na tela MasterSlave de um ticket detalhado na interface de agente (Ticket::Service precisa ser ativado).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Define o proprietário do ticket na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Define o tipo do ticket na tela MasterSlave de ticket de um ticket detalhado na interface de agente (Ticket::Type precisa ser ativado).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Exibe um link no menu para alterar o estado MasterSlave de um ticket na tela de detalhe de ticket da interface de agente.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Exibe uma lista de todos os agentes envolvidos neste ticket, na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Exibe uma lista de todos os agentes possíveis (todos os agentes com permissão de nota na fila/ticket) para determinar quem deve ser informado sobre esta nota, na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Exibe as opções de prioridade de ticket na tela MasterSlave de ticket de um ticket detalhado na interface de agente.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Mostra o título na tela de um chamado MestreEscravo de um chamado aberto na interface de agente.';
    $Self->{Translation}->{'Slave Tickets'} = 'Chamados Escravos';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Este módulo ativa o campo Mestre/Escravo nas telas de novo chamado fone/e-mail.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Chamado MestreEscravo';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::ru_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Управление статусом Главная/Ведомая для %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Новая Master заявка';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Снять значение Master заявка';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Снять значение Slave заявка';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Ведомая от %s%s%s: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Убрать главные заявки';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Убрать ведомые заявки';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Главный';
    $Self->{Translation}->{'Slave of %s%s%s'} = '';
    $Self->{Translation}->{'Master Ticket'} = 'Главная заявка';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Все главные заявки';
    $Self->{Translation}->{'All slave tickets'} = 'Все ведомые заявки';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Позволяет добавить сообщение на экране MasterSlave в интерфейсе агента.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Изменить состояние MasterSlave для этой заявки.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Задает имя динамического поля для функции главной заявки.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Требуется ли блокировка заявки при применении опции MasterSlave в интерфейсе агента (если заявка еще не заблокирована, она блокируется и текущий агент становится ее Владельцем).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Задает следующее состояние по умолчанию для заявки после добавления заметки на экране MasterSlave заявки при ее просмотре в интерфейсе агента.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Задает умалчиваемый приоритет заявки на экране MasterSlave при просмотре заявки в интерфейсе агента.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Задает текст комментария в записи истории при вызове MasterSlave экрана , в интерфейсе агента.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Задает тип записи истории при вызове MasterSlave экрана , в интерфейсе агента.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Задает следующее состояние для заявки после добавления заметки на экране MasterSlave заявки при ее просмотре в интерфейсе агента.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Включает расширенные возможности для MasterSlave.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Включает возможность передачи ведомых заявок главной заявки к новой главной в расширенном режиме MasterSlave .';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Включает возможность изменения состояния MasterSlave заявки в расширенном режиме.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Включает возможность пересылки сообщений/заметок при пересылке главной заявки  клиентам ведомых заявок. По умолчанию (выключено) в ведомые заявки ничего не пересылается.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Включает возможность сохранить связь родитель-потомок для заявок, после изменения признака MasterSlave в расширенном режиме MasterSlave.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Включает возможность сохранить связь родитель-потомок для заявок, после снятия признака MasterSlave в расширенном режиме MasterSlave.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Задает возможность сбросить установленное состояние MasterSlave заявки в расширенном режиме.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Если сообщение/заметка добавлена агентом, задает состояние заявки на экране MasterSlave  в интерфейсе агента.';
    $Self->{Translation}->{'Master / Slave'} = 'Master / Slave';
    $Self->{Translation}->{'Master Tickets'} = 'Главные заявки';
    $Self->{Translation}->{'MasterSlave'} = 'MasterSlave';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'MasterSlave модуль для функции Массовое действие.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Параметры для раздела Дайджеста с информацией о master заявках в интерфейсе агента. "Group" используется для ограничения доступа к разделу (например, Group: admin;group1;group2;). "Default" - задает, будет ли раздел доступен по умолчанию или агент должен активировать его вручную. "CacheTTLLocal" - время обновления кэша в минутах для этого раздела.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Параметры для раздела Дайджеста с информацией о slave заявках в интерфейсе агента. "Group" используется для ограничения доступа к разделу (например, Group: admin;group1;group2;). "Default" - задает, будет ли раздел доступен по умолчанию или агент должен активировать его вручную. "CacheTTLLocal" - время обновления кэша в минутах для этого раздела.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Регистрация модуля обработки событий заявки.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Права, требуемые для использования функции MasterSlave заявок в интерфейсе агента.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Устанавливает текст сообщения по умолчанию при добавлении сообщения на экране MasterSlave заявки в интерфейсе агента.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Устанавливает тему сообщения по умолчанию при добавлении сообщения на экране MasterSlave заявки в интерфейсе агента.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Задает Ответственного за заявку на экране MasterSlave заявки в интерфейсе агента.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Задает сервис для заявки на экране MasterSlave заявки в интерфейсе агента. (Параметр Ticket::Service должен быть включен).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Задает Владельца заявки на экране MasterSlave заявки в интерфейсе агента.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Задает тип заявки на экране MasterSlave заявки в интерфейсе агента. (Параметр Ticket::Service должен быть включен).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Задает ссылку в меню для изменения MasterSlave статуса  на экране просмотра заявки в интерфейсе агента.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Показывает список всех привлекаемых агентов по этой заявке на экране MasterSlave заявки в интерфейсе агента. ';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Показывает список всех доступных агентов (всех агентов с правами note для очереди/заявки), чтобы задать кого нужно информировать об этой заметке на экране MasterSlave заявки в интерфейсе агента. ';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Дает возможность изменить приоритет на экране MasterSlave заявки в интерфейсе агента. ';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = 'Ведомые заявки';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Этот модуль включает возможность выбора master/slave опции на экране создания заявки, на основе телефонного звонка или письма клиента агентом.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Заявка MasterSlave';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::sr_Cyrl_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Управљање статусом главни/зависни за %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Нови главни тикет';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Опозови подешавање главног тикета';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Опозови подешавање зависног тикета';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Зависни од %s%s%s: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Опозови подешавање главних тикета';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Опозови подешавање зависних тикета';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Главно';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Зависни од %s%s%s';
    $Self->{Translation}->{'Master Ticket'} = 'Главни тикет';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Сви главни тикети';
    $Self->{Translation}->{'All slave tickets'} = 'Сви зависни тикети';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Дозвољава додавање белешки на главни/зависни екрану детаљног приказа тикета оператерског интерфејса.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Промени главни/зависни статус тикета.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Дефинише динамички назив поља за функцију главног тикета.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Одређује да ли је потребно закључати главни/зависни екран тикета на детаљном приказу тикета у интерфејсу оператера (ако тикет још увек није закључан, тикет ће добити статус закључан и тренутни оператер ће бити аутоматски постављен као власник).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        'Дефинише уколико је MasterSlave напомена подразумевано видљива за клијент кориснике.';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Одређује подразумевани наредни статус тикета после додаваља белешке, на главни/зависни екрану детаљног приказа тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Одређује подразумевани приоритет тикета на главни/зависни екану детаљног приказа тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Одређује коментар за историјат на главни/зависни екранској акцији, што ће се користити за историјат у интерфејсу оператера.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Одређује тип историјата за главни/зависни екранску акцију, што ће се користити за историјат у интерфејсу оператера.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Одређује наредни статус тикета после додаваља белешке, на главни/зависни екрану детаљног приказа тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Активира напредни део функције главни/зависни.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Активирање својства да зависни тикет прати главни на нови главни у напредном главни/зависни моду.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Активирање функције за промену стања тикета главни/зависни  у напредном главни/зависни моду.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Активирање својства за прослеђивање чланака типа "проследи" главног тикета корисницима зависних тикета. Подразумевано је (искључено) да се ништа не прослеђује зависним тикетима.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Активирање функције за задржавање везе надређени-подређени после измене стања главни/зависни у напредном главни/зависни моду.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Активирање функције за задржавање везе надређени-подређени после опозива подешавања стања главни/зависни  у напредном главни/зависни моду.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Активирање функције за опозив подешавања стања тикета главни/зависни у напредном главни/зависни моду.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Ако је оператер додао напомену, подешава статус тикета на екрану главни/зависни тикета на детаљном приказу у интерфејсу оператера.';
    $Self->{Translation}->{'Master / Slave'} = 'Главни / зависни';
    $Self->{Translation}->{'Master Tickets'} = 'Главни тикети';
    $Self->{Translation}->{'MasterSlave'} = 'Главни/зависни';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Модул главни/зависни за функцију масовне обраде тикета.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Параметри за додатак прегледа главних тикета контролне табле у интерфејсу оператера. "Limit" је број подразумевано приказаних тикета. "Group" се користи да ограничи приступ додатку (нпр. Group: admin;group1;group2;). "Default" одређује да ли је подразумевано активиран или да је потребно да га корисник мануелно активира. "CacheTTLLocal" је време у минутима за кеш додатка. ';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Параметри за додатак прегледа зависних тикета контролне табле у интерфејсу оператера. "Limit" је број подразумевано приказаних тикета. "Group" се користи да ограничи приступ додатку (нпр. Group: admin;group1;group2;). "Default" одређује да ли је подразумевано активиран или да је потребно да га корисник мануелно активира. "CacheTTLLocal" је време у минутима за кеш додатка. ';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Регистрација модула догађаја за тикете.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Неопходна дозвола за употребу главни/зависни екрана детаљног приказа тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Дефинише да ли поље Главни / Зависни мора бити подешено од стране оператера.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Одређује подразумевани садржај за напомене додате на на Главни/Зависни екрану детаљног приказа тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Одређује подразумевани предмет за напомене додате на Главни/Зависни екрану детаљног приказа тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Одређује одговорног оператера за тикет на екрану главни/зависни  детаљног приказа тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Одређује сервис на главни/зависни екрану детаљног приказа тикета у интерфејсу оператера (Ticket::Service мора бити активиран).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Одређује власника тикета на главни/зависни екрану детаљног приказа тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Одређује тип тикета на главни/зависни екрану детаљног приказа тикета у интерфејсу оператера (Ticket::Type мора бити активиран).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'У менију приказује везу за измену главни/зависни статуса тикета на детаљном приказу тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Приказује листу свих оператера укључених у овај тикет на главни/зависни екрану детаљног приказа тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Приказује листу свих могућих оператера (сви оператери са дозволом за напомену за ред/тикет) ради утврђивања ко треба да буде информисан о овој напомени, на главни/зависни екрану тикета на детаљном приказу тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Приказује опције приоритета тикета на екрану главни/зависни тикета на детаљном приказу тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Приказује насловна поља на екрану главни/зависни тикета у интерфејсу оператера.';
    $Self->{Translation}->{'Slave Tickets'} = 'Зависни тикети';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        'Наводи разне комуникационе канале чланака где ће стварно име са главног тикета бити замењено са једним на зависном тикету.';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Овај модул активира поље главни/зависни на екрану нових имејл тикета и тикета позива.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        'Ово подешавање је застарело и биће избачено из будућих верзија модула главни/зависни.';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Тикет главни/зависни.';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::sr_Latn_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = 'Upravljanje statusom glavni/zavisni za %s%s%s';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Novi glavni tiket';
    $Self->{Translation}->{'Unset Master Ticket'} = 'Opozovi podešavanje glavnog tiketa';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'Opozovi podešavanje zavisnog tiketa';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = 'Zavisni od %s%s%s: %s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = 'Opozovi podešavanje glavnih tiketa';
    $Self->{Translation}->{'Unset Slave Tickets'} = 'Opozovi podešavanje zavisnih tiketa';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Glavno';
    $Self->{Translation}->{'Slave of %s%s%s'} = 'Zavisni od %s%s%s';
    $Self->{Translation}->{'Master Ticket'} = 'Glavni tiket';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Svi glavni tiketi';
    $Self->{Translation}->{'All slave tickets'} = 'Svi zavisni tiketi';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Dozvoljava dodavanje beleški na glavni/zavisni ekranu detaljnog prikaza tiketa operaterskog interfejsa.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Promeni glavni/zavisni status tiketa.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = 'Definiše dinamički naziv polja za funkciju glavnog tiketa.';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Određuje da li je potrebno zaključati glavni/zavisni ekran tiketa na detaljnom prikazu tiketa u interfejsu operatera (ako tiket još uvek nije zaključan, tiket će dobiti status zaključan i trenutni operater će biti automatski postavljen kao vlasnik).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        'Definiše ukoliko je MasterSlave napomena podrazumevano vidljiva za klijent korisnike.';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Određuje podrazumevani naredni status tiketa posle dodavalja beleške, na glavni/zavisni ekranu detaljnog prikaza tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Određuje podrazumevani prioritet tiketa na glavni/zavisni ekanu detaljnog prikaza tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Određuje komentar za istorijat na glavni/zavisni ekranskoj akciji, što će se koristiti za istorijat u interfejsu operatera.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Određuje tip istorijata za glavni/zavisni ekransku akciju, što će se koristiti za istorijat u interfejsu operatera.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Određuje naredni status tiketa posle dodavalja beleške, na glavni/zavisni ekranu detaljnog prikaza tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = 'Aktivira napredni deo funkcije glavni/zavisni.';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        'Aktiviranje svojstva da zavisni tiket prati glavni na novi glavni u naprednom glavni/zavisni modu.';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Aktiviranje funkcije za promenu stanja tiketa glavni/zavisni  u naprednom glavni/zavisni modu.';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        'Aktiviranje svojstva za prosleđivanje članaka tipa "prosledi" glavnog tiketa korisnicima zavisnih tiketa. Podrazumevano je (isključeno) da se ništa ne prosleđuje zavisnim tiketima.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Aktiviranje funkcije za zadržavanje veze nadređeni-podređeni posle izmene stanja glavni/zavisni u naprednom glavni/zavisni modu.';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        'Aktiviranje funkcije za zadržavanje veze nadređeni-podređeni posle opoziva podešavanja stanja glavni/zavisni  u naprednom glavni/zavisni modu.';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        'Aktiviranje funkcije za opoziv podešavanja stanja tiketa glavni/zavisni u naprednom glavni/zavisni modu.';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Ako je operater dodao napomenu, podešava status tiketa na ekranu glavni/zavisni tiketa na detaljnom prikazu u interfejsu operatera.';
    $Self->{Translation}->{'Master / Slave'} = 'Glavni / zavisni';
    $Self->{Translation}->{'Master Tickets'} = 'Glavni tiketi';
    $Self->{Translation}->{'MasterSlave'} = 'Glavni/zavisni';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'Modul glavni/zavisni za funkciju masovne obrade tiketa.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametri za dodatak pregleda glavnih tiketa kontrolne table u interfejsu operatera. "Limit" je broj podrazumevano prikazanih tiketa. "Group" se koristi da ograniči pristup dodatku (npr. Group: admin;group1;group2;). "Default" određuje da li je podrazumevano aktiviran ili da je potrebno da ga korisnik manuelno aktivira. "CacheTTLLocal" je vreme u minutima za keš dodatka. ';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametri za dodatak pregleda zavisnih tiketa kontrolne table u interfejsu operatera. "Limit" je broj podrazumevano prikazanih tiketa. "Group" se koristi da ograniči pristup dodatku (npr. Group: admin;group1;group2;). "Default" određuje da li je podrazumevano aktiviran ili da je potrebno da ga korisnik manuelno aktivira. "CacheTTLLocal" je vreme u minutima za keš dodatka. ';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Registracija modula događaja za tikete.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Neophodna dozvola za upotrebu glavni/zavisni ekrana detaljnog prikaza tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = 'Definiše da li polje Glavni / Zavisni mora biti podešeno od strane operatera.';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Određuje podrazumevani sadržaj za napomene dodate na na Glavni/Zavisni ekranu detaljnog prikaza tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Određuje podrazumevani predmet za napomene dodate na Glavni/Zavisni ekranu detaljnog prikaza tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Određuje odgovornog operatera za tiket na ekranu glavni/zavisni  detaljnog prikaza tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Određuje servis na glavni/zavisni ekranu detaljnog prikaza tiketa u interfejsu operatera (Ticket::Service mora biti aktiviran).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Određuje vlasnika tiketa na glavni/zavisni ekranu detaljnog prikaza tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Određuje tip tiketa na glavni/zavisni ekranu detaljnog prikaza tiketa u interfejsu operatera (Ticket::Type mora biti aktiviran).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'U meniju prikazuje vezu za izmenu glavni/zavisni statusa tiketa na detaljnom prikazu tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Prikazuje listu svih operatera uključenih u ovaj tiket na glavni/zavisni ekranu detaljnog prikaza tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Prikazuje listu svih mogućih operatera (svi operateri sa dozvolom za napomenu za red/tiket) radi utvrđivanja ko treba da bude informisan o ovoj napomeni, na glavni/zavisni ekranu tiketa na detaljnom prikazu tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Prikazuje opcije prioriteta tiketa na ekranu glavni/zavisni tiketa na detaljnom prikazu tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Prikazuje naslovna polja na ekranu glavni/zavisni tiketa u interfejsu operatera.';
    $Self->{Translation}->{'Slave Tickets'} = 'Zavisni tiketi';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        'Navodi razne komunikacione kanale članaka gde će stvarno ime sa glavnog tiketa biti zamenjeno sa jednim na zavisnom tiketu.';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'Ovaj modul aktivira polje glavni/zavisni na ekranu novih imejl tiketa i tiketa poziva.';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        'Ovo podešavanje je zastarelo i biće izbačeno iz budućih verzija modula glavni/zavisni.';
    $Self->{Translation}->{'Ticket MasterSlave.'} = 'Tiket glavni/zavisni.';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::sv_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'Nytt Masterärende';
    $Self->{Translation}->{'Unset Master Ticket'} = '';
    $Self->{Translation}->{'Unset Slave Ticket'} = '';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = '';
    $Self->{Translation}->{'Unset Slave Tickets'} = '';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = '';
    $Self->{Translation}->{'Slave of %s%s%s'} = '';
    $Self->{Translation}->{'Master Ticket'} = 'Masterärende';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = 'Alla masterärenden';
    $Self->{Translation}->{'All slave tickets'} = 'Alla slavärenden';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = '';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = '';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        '';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = '';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Master / Slave'} = 'Master / Slave';
    $Self->{Translation}->{'Master Tickets'} = '';
    $Self->{Translation}->{'MasterSlave'} = '';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Registration of the ticket event module.'} = '';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        '';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = '';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        '';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::sw_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = '';
    $Self->{Translation}->{'Unset Master Ticket'} = '';
    $Self->{Translation}->{'Unset Slave Ticket'} = '';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = '';
    $Self->{Translation}->{'Unset Slave Tickets'} = '';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'Fuzu';
    $Self->{Translation}->{'Slave of %s%s%s'} = '';
    $Self->{Translation}->{'Master Ticket'} = '';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = '';
    $Self->{Translation}->{'All slave tickets'} = '';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Ruhusu kuongeza vidokezi katika skrini ya MkuuMtumwa ya tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'Badilisha hali ya Mkuu/mtumwa ya tiketi.';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = '';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'Fafanua kama kufunga kwa tiketi kunahitajika katika skrini ya tiketi ya  mkuumtumwa ya tiketi iliyokuzwa katika kiolesura cha wakala (kama tiketi haijafungwa bado, tiketi zipatwe kufungwa na wakala wa sasa atafanywa automatiki kuwa mmiliki wake).';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inafafanua chaguo-msingi la hali iyajo ya tiketi baada ya kuongeza kidokezo, katika skrini ya mkuumtumwa ya tiketi ya tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inafafanua kipaumbele chaguo-msingi cha tiketi katika skrini ya mkuumtumwa ya tiketi ya tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Inafafanua historia ya maoni kwa kitendo cha skrini ya tiketi mkuumtumwa, ambayo inatumika na historia ya tiketi ya kiolesura cha wakala.';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'Fafanua aina ya historia kwa kitendo cha skrini ya mkuumtumwa cha tiketi, ambacho kinatumika na historia ya tiketi katika kiolesura cha wakala.';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Fafanua hali inayofuata ya tiketi baada ya kuongeza kidokezo, katika skrini ya mkuumtumwa ya tiketi ya tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = '';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Kama kidokezo kimeongezwa na wakala, inaweka hali ya tiketi katika skrini ya mkuumtumwa ya tiketi ya tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Master / Slave'} = '';
    $Self->{Translation}->{'Master Tickets'} = '';
    $Self->{Translation}->{'MasterSlave'} = '';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parameta kwa backend ya dasibodi ya mapitio ya tiketi kuu ya kiolesura cha wakala. \'\'Kikomo\'\' ni namba ya vipengee halisi vinavyoonyeshwa kwa mchaguo-halisi. \'\'Kikundi\'\' kinatumika kuzuia kufikia kuchomeka (mfano Kikundi:Utawala;kikundi cha 1;kikundi cha 2;). \'\'Chaguo-msingi\'\' inahakiki kama mchomeko umewezeshwa kwa mchaguo-msingi au kama mtumizi anahitaji kuwezesha kwa mkono. \'\'HifadhimudaTTLKiambo\'\' ni muda wa hifadhi muda katika dakika kwa mchomeko.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parameta kwa backend ya dasibodi ya mapitio ya tiketi tumwa ya kiolesura cha wakala. \'\'Kikomo\'\' ni namba ya vipengee halisi vinavyoonyeshwa kwa mchaguo-halisi. \'\'Kikundi\'\' kinatumika kuzuia kufikia kuchomeka (mfano Kikundi:Utawala;kikundi cha 1;kikundi cha 2;). \'\'Chaguo-msingi\'\' inahakiki kama mchomeko umewezeshwa kwa mchaguo-msingi au kama mtumizi anahitaji kuwezesha kwa mkono. \'\'HifadhimudaTTLKiambo\'\' ni muda wa hifadhi muda katika dakika kwa mchomeko.';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'Usajili wa kipimo cha tukio la tiketi.';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inahitaji ruhusa kutumia skrini ya mkuumtumwa ya tiketi ya tiketi iliyokuzwa katika kioesura cha wakala.';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inaweka matini kiini kilichoongezwa cha chaguo-msingi katika skrini ya mkuumtuwa ya tiketi katika tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inaweka somo la chaguo-msingi kwa kipengele kilichoongezwa katika skrini ya mkuumtuwa ya tiketi katika tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inaweka wakala ambaye anahusika na tiketi katika skrini ya mkuumtumwa ya tiketi ya tiketi iliyokuzwa katikat kiolesura cha wakala.';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'Inaweka huduma katika skrini ya mkuumtumwa ya tiketi ya tiketi iliyokuzwa katika kiolesura cha wakala (Tiketi::Huduma inayohitaji kuamilishwa).';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inaweka mmiliki wa tiketi katika skrini ya mkuumtumwa ya tiketi ya tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'Inaweka aina ya tiketi katika skrini ya mkuumtumwa ya tiketi ya tiketi iliyokuzwa katika kiolesura cha wakala (tiketi:: aina inayoitaji kuamilishwa).';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'Inaonyesha kiungo katika menyu cha kubadilisha hadhi ya mkuumtumwa ya tiketi katika mandhari iliyokuzwa ya tiketi ya kiolesura cha wakala.';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inaonyesha orodha ya mawakala wote wanaohusika na tiketi hii, katika skrini ya tiketi mkuu/mtumwa ya tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inaonyesha orodha ya wakala wote wanaoweza (mawakala wote wenye kidokezo cha ruhusa ya tiketi/foleni) kutambua nani ataarifiwe kuhusu kidokezo hiki, katika skrini ya tiketi mkuumtumwa ya tiketi iliyokuzwa katika kiolesura chwa wakala.';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'Inaonyesha machaguo ya kipaumbele ya tiketi katika skrini ya tiketi mkuumtumwa ya tiketi iliyokuzwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = '';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        '';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::th_TH_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = 'ตั๋วมาสเตอร์ใหม่';
    $Self->{Translation}->{'Unset Master Ticket'} = 'ยกเลิกการตั้งค่าตั๋วมาสเตอร์';
    $Self->{Translation}->{'Unset Slave Ticket'} = 'ยกเลิกการตั้งค่าตั๋วSlave';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = '';
    $Self->{Translation}->{'Unset Slave Tickets'} = '';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = 'มาสเตอร์';
    $Self->{Translation}->{'Slave of %s%s%s'} = '';
    $Self->{Translation}->{'Master Ticket'} = '';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = '';
    $Self->{Translation}->{'All slave tickets'} = '';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'อนุญาตให้เพิ่มโน้ตในหน้าจอตั๋วMasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์ ';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = 'เปลี่ยนสถานภาพของตั๋ว MasterSlave ';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = '';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        'กำหนดค่าถ้าหากจำเป็นต้องใช้ตั๋วล็อคในหน้าจอตั๋วMasterSlaveของตั๋วซูมในอินเตอร์เฟซของเอเย่นต์ (ทำการล็อคตั๋วถ้าหากตั๋วยังไม่ได้ล็อคและระบุให้เอเย่นต์ปัจจุบันเป็นเจ้าของอัตโนมัติ)';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'กำหนดค่าเริ่มต้นของสถานภาพถัดไปของตั๋วหลังจากเพิ่มโน้ตในหน้าจอตั๋วMasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์ ';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'กำหนดลำดับความสำคัญเริ่มต้นของตั๋วในหน้าจอตั๋วMasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'กำหนดประวัติการแสดงความเห็นสำหรับการกระทำหน้าจอตั๋วMasterSlave ของตั๋วซูม ซึ่งทำให้เกิดความคุ้นเคยในประวัติของตั๋วในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        'กำหนดประเภทประวัติสำหรับการกระทำสำหรับการกระทำหน้าจอตั๋วMasterSlave ของตั๋วซูม ซึ่งทำให้เกิดความคุ้นเคยในประวัติของตั๋วในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'กำหนดสถานภาพถัดไปของตั๋วหลังจากเพิ่มโน้ตในหน้าจอตั๋วMasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = '';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'เซตสถานภาพของตั๋วในหน้าจอตั๋ว MasterSlave ของตั๋วซูมในอินเตอร์เฟซของเอเย่นต์หากเอเย่นต์ได้เพิ่มโน้ต';
    $Self->{Translation}->{'Master / Slave'} = 'มาสเตอร์/Slave';
    $Self->{Translation}->{'Master Tickets'} = '';
    $Self->{Translation}->{'MasterSlave'} = '';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = 'โมดูล มาสเตอร์/Slave สำหรับคุณลักษณะกลุ่มตั๋ว';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'พารามิเตอร์สำหรับแดชบอร์ดเบื้องหลังของภาพรวมตั๋วmaster ของอินเตอร์เฟสเอเย่นต์ "จำกัด" คือกำหนดจำนวนของรายการที่แสดงโดยค่าเริ่มต้น "กลุ่ม" จะถูกนำมาใช้เพื่อจำกัดการเข้าถึงปลั๊กอิน (เช่นกลุ่ม: ผู้ดูแลระบบ; กลุ่ม 1; กลุ่ม2;) "เริ่มต้น" ระบุว่าถ้าปลั๊กอินถูกเปิดใช้งานโดยค่าเริ่มต้นหรือหากผู้ใช้ต้องการเพื่อเปิดใช้งานได้ด้วยตนเอง';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'พารามิเตอร์สำหรับแดชบอร์ดเบื้องหลังของภาพรวมตั๋วslave tของอินเตอร์เฟสเอเย่นต์ "จำกัด" คือกำหนดจำนวนของรายการที่แสดงโดยค่าเริ่มต้น "กลุ่ม" จะถูกนำมาใช้เพื่อจำกัดการเข้าถึงปลั๊กอิน (เช่นกลุ่ม: ผู้ดูแลระบบ; กลุ่ม 1; กลุ่ม2;) "เริ่มต้น" ระบุว่าถ้าปลั๊กอินถูกเปิดใช้งานโดยค่าเริ่มต้นหรือหากผู้ใช้ต้องการเพื่อเปิดใช้งานได้ด้วยตนเอง';
    $Self->{Translation}->{'Registration of the ticket event module.'} = 'ารลงทะเบียนของโมดูลตั๋วกิจกรรม';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'จำเป็นต้องมีการอนุญาติในการใช้งานในหน้าจอตั๋ว MasterSlave ในตั๋วซูมในอินเตอร์เฟซของเอเย่นต์';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'ระบุข้อความส่วนเนื้อหาเริ่มต้นสำหรับโน้ตที่ถูกเพิ่มเข้ามาในหน้าจอตั๋ว MasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'ระบุหัวข้อเริ่มต้นสำหรับโน้ตที่ถูกเพิ่มเข้ามาในหน้าจอตั๋ว MasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'กำหนดเอเย่นต์ที่รับผิดชอบตั๋วในหน้าจอตั๋ว MasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        'กำหนดการบริการในหน้าจอตั๋ว MasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์ (ตั๋ว ::การบริการจำเป็นต้องมีการเปิดใช้งาน)';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'กำหนดเจ้าของตั๋วในหน้าจอตั๋ว MasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        'กำหนดประเภทของตั๋วในหน้าจอตั๋ว MasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์ (ตั๋ว ::ประเภทตั๋วจำเป็นต้องมีการเปิดใช้งาน)';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        'แสดงลิงค์ในเมนูเพื่อเปลี่ยนสถานภาพของตั๋ว MasterSlave ในการซูมตั๋วของอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'แสดงรายชื่อของเอเย่นต์ที่เกี่ยวข้องกับตั๋วนี้ในหน้าจอตั๋ว MasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'แสดงรายชื่อเอเย่นต์ที่เป็นไปได้ทั้งหมด (เอเย่นต์ทั้งหมดที่ได้รับโน้ตอนุญาติในคิว/ตั๋ว) เพื่อกำหนดว่าใครควรรายงานเกี่ยวกับโน้ตนี้ในในหน้าจอตั๋วMasterSlave ของตั๋วซูมในอินเตอร์เฟซของเอเย่นต์';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        'แสดงตัวเลือกลำดับความสำคัญของตั๋วในหน้าจอตั๋ว MasterSlave ของตั๋วซูมในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '';
    $Self->{Translation}->{'Slave Tickets'} = '';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        'โมดูลนี้เปิดใช้งานฟิลด์ Master /Slave ในหน้าจออีเมลใหม่และตั๋วโทรศัพท์';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '';
    $Self->{Translation}->{'Ticket MasterSlave.'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::zh_CN_OTRSMasterSlave;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AgentTicketMasterSlave
    $Self->{Translation}->{'Manage Master/Slave status for %s%s%s'} = '管理%s%s%s的 主/从 状态';

    # Perl Module: Kernel/Modules/AgentTicketMasterSlave.pm
    $Self->{Translation}->{'New Master Ticket'} = '新建主工单';
    $Self->{Translation}->{'Unset Master Ticket'} = '主工单转为普通工单';
    $Self->{Translation}->{'Unset Slave Ticket'} = '从工单转为普通工单';
    $Self->{Translation}->{'Slave of %s%s%s: %s'} = '%s%s%s的从工单：%s';

    # Perl Module: Kernel/Output/HTML/TicketBulk/MasterSlave.pm
    $Self->{Translation}->{'Unset Master Tickets'} = '取消主工单设置';
    $Self->{Translation}->{'Unset Slave Tickets'} = '取消从工单设置';

    # Perl Module: Kernel/System/DynamicField/Driver/MasterSlave.pm
    $Self->{Translation}->{'Master'} = '主';
    $Self->{Translation}->{'Slave of %s%s%s'} = '%s%s%s的从工单';
    $Self->{Translation}->{'Master Ticket'} = '主工单';

    # SysConfig
    $Self->{Translation}->{'All master tickets'} = '所有的主工单';
    $Self->{Translation}->{'All slave tickets'} = '所有的从工单';
    $Self->{Translation}->{'Allows adding notes in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '服务人员界面主/从工单详情窗口允许添加备注。';
    $Self->{Translation}->{'Change the MasterSlave state of the ticket.'} = '修改工单主/从状态。';
    $Self->{Translation}->{'Defines dynamic field name for master ticket feature.'} = '定义主工单功能的动态字段名称。';
    $Self->{Translation}->{'Defines if a ticket lock is required in the ticket MasterSlave screen of a zoomed ticket in the agent interface (if the ticket isn\'t locked yet, the ticket gets locked and the current agent will be set automatically as its owner).'} =
        '定义服务人员界面在工单主从设置窗口是否需要工单锁定（如果工单还没有锁定，则工单被锁定且当前服务人员被自动设置为工单所有者）。';
    $Self->{Translation}->{'Defines if the MasterSlave note is visible for the customer by default.'} =
        '定义默认情况下客户是否可以看到MasterSlave的备注。';
    $Self->{Translation}->{'Defines the default next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '定义服务人员界面在工单主从设置窗口添加备注后的默认下一个工单状态。';
    $Self->{Translation}->{'Defines the default ticket priority in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '定义服务人员界面在工单主从设置窗口添加备注后工单的默认优先级。';
    $Self->{Translation}->{'Defines the history comment for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '定义工单主从设置窗口操作的历史注释，以用于服务人员界面的工单历史。';
    $Self->{Translation}->{'Defines the history type for the ticket MasterSlave screen action, which gets used for ticket history in the agent interface.'} =
        '定义工单主从设置窗口操作的历史类型，以用于服务人员界面的工单历史。';
    $Self->{Translation}->{'Defines the next state of a ticket after adding a note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '定义服务人员界面在工单主从设置窗口添加备注后的下一个工单状态。';
    $Self->{Translation}->{'Enables the advanced MasterSlave part of the feature.'} = '启用主从功能的高级模式。';
    $Self->{Translation}->{'Enables the feature that slave tickets follow the master ticket to a new master in the advanced MasterSlave mode.'} =
        '启用高级主从模式下从工单随主工单到一个新的主工单的功能。';
    $Self->{Translation}->{'Enables the feature to change the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '启用本功能后，可修改一个工单在高级主从模式下的主从状态。';
    $Self->{Translation}->{'Enables the feature to forward articles from type \'forward\' of a master ticket to the customers of the slave tickets. By default (disabled) it will not forward articles from type \'forward\' to the slave tickets.'} =
        '启用本功能后，将“转发”类型的主工单信件转发到从工单的客户。默认（已禁用）不转发“转发”类型的信件到从工单。';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after change of the MasterSlave state in the advanced MasterSlave mode.'} =
        '启用本功能后，在高级主从模式下修改主从状态后保持父子链接。';
    $Self->{Translation}->{'Enables the feature to keep parent-child link after unset of the MasterSlave state in the advanced MasterSlave mode.'} =
        '启用本功能后，在高级主从模式下取消主从状态后保持父子链接。';
    $Self->{Translation}->{'Enables the feature to unset the MasterSlave state of a ticket in the advanced MasterSlave mode.'} =
        '启用本功能后，可取消一个工单在高级主从模式下的主从状态。';
    $Self->{Translation}->{'If a note is added by an agent, sets the state of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '服务人员界面主/从工单详情窗口，如果服务人员添加了备注，设置工单的状态。';
    $Self->{Translation}->{'Master / Slave'} = '主/从';
    $Self->{Translation}->{'Master Tickets'} = '主工单';
    $Self->{Translation}->{'MasterSlave'} = '主从';
    $Self->{Translation}->{'MasterSlave module for Ticket Bulk feature.'} = '工单批量操作功能的主从工单批量设置模块';
    $Self->{Translation}->{'Parameters for the dashboard backend of the master tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '定义服务人员界面仪表板主工单概览后端的参数。“Limit（限制）定义默认显示的条目数；“GROUP（组）用于到本插件的访问权限限制（如 Group:admin;group1;group2）；“Default（默认）”代表这个插件是默认启用还是需要用户手动启用；“CacheTTLLocal”表明本插件的缓存过期时间（单位：分钟）。';
    $Self->{Translation}->{'Parameters for the dashboard backend of the slave tickets overview of the agent interface. "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '定义服务人员界面仪表板从工单概览后端的参数。“Limit（限制）定义默认显示的条目数；“GROUP（组）用于到本插件的访问权限限制（如 Group:admin;group1;group2）；“Default（默认）”代表这个插件是默认启用还是需要用户手动启用；“CacheTTLLocal”表明本插件的缓存过期时间（单位：分钟）。';
    $Self->{Translation}->{'Registration of the ticket event module.'} = '注册到工单事件模块。';
    $Self->{Translation}->{'Required permissions to use the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '服务人员界面使用主从工单详情窗口必需的权限。';
    $Self->{Translation}->{'Sets if Master / Slave field must be selected by the agent.'} = '设置服务人员是否必须选择主从字段。';
    $Self->{Translation}->{'Sets the default body text for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '设置服务人员界面主从工单详情窗口添加备注的的默认正文文本。';
    $Self->{Translation}->{'Sets the default subject for notes added in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '设置服务人员界面主从工单详情窗口添加备注的的默认主题。';
    $Self->{Translation}->{'Sets the responsible agent of the ticket in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '设置服务人员界面主从工单详情窗口工单的负责人。';
    $Self->{Translation}->{'Sets the service in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Service needs to be activated).'} =
        '设置服务人员界面主从工单详情窗口工单的服务（需要激活工单::服务）。';
    $Self->{Translation}->{'Sets the ticket owner in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '设置服务人员界面主从工单详情窗口工单的所有者。';
    $Self->{Translation}->{'Sets the ticket type in the ticket MasterSlave screen of a zoomed ticket in the agent interface (Ticket::Type needs to be activated).'} =
        '设置服务人员界面主从工单详情窗口设置工单的类型（需要激活工单::类型）。';
    $Self->{Translation}->{'Shows a link in the menu to change the MasterSlave status of a ticket in the ticket zoom view of the agent interface.'} =
        '在服务人员界面工单详情视图中，为修改主从工单状态菜单显示一个链接。';
    $Self->{Translation}->{'Shows a list of all the involved agents on this ticket, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '在服务人员界面主从工单详情窗口，显示这个工单相关的所有服务人员列表。';
    $Self->{Translation}->{'Shows a list of all the possible agents (all agents with note permissions on the queue/ticket) to determine who should be informed about this note, in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '在服务人员界面主从工单详情窗口，显示这个工单所有可能的服务人员（需要具有这个队列或工单的备注权限）列表，用于确定谁将收到关于这个备注的通知。';
    $Self->{Translation}->{'Shows the ticket priority options in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '在服务人员界面主从工单详情窗口是否显示工单优先级的选项。';
    $Self->{Translation}->{'Shows the title field in the ticket MasterSlave screen of a zoomed ticket in the agent interface.'} =
        '在服务人员界面主从工单详情窗口显示工单标题字段。';
    $Self->{Translation}->{'Slave Tickets'} = '从工单';
    $Self->{Translation}->{'Specifies the different article communication channels where the real name from Master ticket will be replaced with the one in the Slave ticket.'} =
        '指定不同的信件通信渠道，其中主工单的真实姓名将替换为从工单中的真实姓名。';
    $Self->{Translation}->{'This module activates Master/Slave field in new email and phone ticket screens.'} =
        '这个模块用来激活新建邮件/电话工单窗口的主从字段。';
    $Self->{Translation}->{'This setting is deprecated and will be removed in further versions of OTRSMasterSlave.'} =
        '此设置已弃用，会在将来版本的 OTRSMasterSlave 中移除。';
    $Self->{Translation}->{'Ticket MasterSlave.'} = '主从工单。';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    );

}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AdminDynamicFieldMasterSlave;

use strict;
use warnings;

our $ObjectManagerDisabled = 1;

use Kernel::System::VariableCheck qw(:all);
use Kernel::Language qw(Translatable);

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

    my $Self = {%Param};
    bless( $Self, $Type );

    return $Self;
}

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

    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    if ( $Self->{Subaction} eq 'Add' ) {
        return $Self->_Add(
            %Param,
        );
    }
    elsif ( $Self->{Subaction} eq 'AddAction' ) {

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

        return $Self->_AddAction(
            %Param,
        );
    }
    if ( $Self->{Subaction} eq 'Change' ) {
        return $Self->_Change(
            %Param,
        );
    }
    elsif ( $Self->{Subaction} eq 'ChangeAction' ) {

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

        return $Self->_ChangeAction(
            %Param,
        );
    }
    return $LayoutObject->ErrorScreen(
        Message => Translatable('Undefined subaction.'),
    );
}

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

    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    my %GetParam;
    for my $Needed (qw(ObjectType FieldType FieldOrder)) {
        $GetParam{$Needed} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Needed );
        if ( !$Needed ) {
            return $LayoutObject->ErrorScreen(
                Message => $LayoutObject->{LanguageObject}->Translate( 'Need %s', $Needed ),
            );
        }
    }

    # get the object type and field type display name
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    my $ObjectTypeName
        = $ConfigObject->Get('DynamicFields::ObjectType')->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName = $ConfigObject->Get('DynamicFields::Driver')->{ $GetParam{FieldType} }->{DisplayName} || '';

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        Mode           => 'Add',
        BreadcrumbText => $LayoutObject->{LanguageObject}
            ->Translate( 'Add %s field', $LayoutObject->{LanguageObject}->Translate($FieldTypeName) ),
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my $ParamObject  = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    for my $Needed (qw(Name Label FieldOrder)) {
        $GetParam{$Needed} = $ParamObject->GetParam( Param => $Needed );
        if ( !$GetParam{$Needed} ) {
            $Errors{ $Needed . 'ServerError' }        = 'ServerError';
            $Errors{ $Needed . 'ServerErrorMessage' } = Translatable('This field is required.');
        }
    }

    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

    if ( $GetParam{Name} ) {

        # check if name is alphanumeric
        if ( $GetParam{Name} !~ m{\A (?: [a-zA-Z] | \d )+ \z}xms ) {

            # add server error error class
            $Errors{NameServerError} = 'ServerError';
            $Errors{NameServerErrorMessage} =
                Translatable('The field does not contain only ASCII letters and numbers.');
        }

        # check if name is duplicated
        my %DynamicFieldsList = %{
            $DynamicFieldObject->DynamicFieldList(
                Valid      => 0,
                ResultType => 'HASH',
            )
        };

        %DynamicFieldsList = reverse %DynamicFieldsList;

        if ( $DynamicFieldsList{ $GetParam{Name} } ) {

            # add server error error class
            $Errors{NameServerError}        = 'ServerError';
            $Errors{NameServerErrorMessage} = Translatable('There is another field with the same name.');
        }
    }

    if ( $GetParam{FieldOrder} ) {

        # check if field order is numeric and positive
        if ( $GetParam{FieldOrder} !~ m{\A (?: \d )+ \z}xms ) {

            # add server error error class
            $Errors{FieldOrderServerError}        = 'ServerError';
            $Errors{FieldOrderServerErrorMessage} = Translatable('The field must be numeric.');
        }
    }

    for my $ConfigParam (qw( ObjectType ObjectTypeName FieldType FieldTypeName ValidID )) {
        $GetParam{$ConfigParam} = $ParamObject->GetParam( Param => $ConfigParam );
    }

    # uncorrectable errors
    if ( !$GetParam{ValidID} ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Need ValidID'),
        );
    }

    # return to add screen if errors
    if (%Errors) {
        return $Self->_ShowScreen(
            %Param,
            %Errors,
            %GetParam,
            Mode => 'Add',
        );
    }

    # set specific config
    my $FieldConfig = {
        DefaultValue       => '',
        PossibleNone       => 1,
        TranslatableValues => 1,
    };

    # create a new field
    my $FieldID = $DynamicFieldObject->DynamicFieldAdd(
        Name       => $GetParam{Name},
        Label      => $GetParam{Label},
        FieldOrder => $GetParam{FieldOrder},
        FieldType  => $GetParam{FieldType},
        ObjectType => $GetParam{ObjectType},
        Config     => $FieldConfig,
        ValidID    => $GetParam{ValidID},
        UserID     => $Self->{UserID},
    );

    if ( !$FieldID ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Could not create the new field'),
        );
    }

    return $LayoutObject->Redirect(
        OP => "Action=AdminDynamicField",
    );
}

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

    my $ParamObject  = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    my %GetParam;
    for my $Needed (qw(ObjectType FieldType)) {
        $GetParam{$Needed} = $ParamObject->GetParam( Param => $Needed );
        if ( !$Needed ) {
            return $LayoutObject->ErrorScreen(
                Message => $LayoutObject->{LanguageObject}->Translate( 'Need %s', $Needed ),
            );
        }
    }

    # get the object type and field type display name
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    my $ObjectTypeName
        = $ConfigObject->Get('DynamicFields::ObjectType')->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName = $ConfigObject->Get('DynamicFields::Driver')->{ $GetParam{FieldType} }->{DisplayName} || '';

    my $FieldID = $ParamObject->GetParam( Param => 'ID' );

    if ( !$FieldID ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Need ID'),
        );
    }

    # get dynamic field data
    my $DynamicFieldData = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
        ID => $FieldID,
    );

    # check for valid dynamic field configuration
    if ( !IsHashRefWithData($DynamicFieldData) ) {
        return $LayoutObject->ErrorScreen(
            Message =>
                $LayoutObject->{LanguageObject}->Translate( 'Could not get data for dynamic field %s', $FieldID ),
        );
    }

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        %${DynamicFieldData},
        ID             => $FieldID,
        Mode           => 'Change',
        BreadcrumbText => $LayoutObject->{LanguageObject}
            ->Translate( 'Change %s field', $LayoutObject->{LanguageObject}->Translate($FieldTypeName) ),
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my $ParamObject  = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    for my $Needed (qw(Name Label FieldOrder)) {
        $GetParam{$Needed} = $ParamObject->GetParam( Param => $Needed );
        if ( !$GetParam{$Needed} ) {
            $Errors{ $Needed . 'ServerError' }        = 'ServerError';
            $Errors{ $Needed . 'ServerErrorMessage' } = Translatable('This field is required.');
        }
    }

    my $FieldID = $ParamObject->GetParam( Param => 'ID' );
    if ( !$FieldID ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Need ID'),
        );
    }

    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

    # get dynamic field data
    my $DynamicFieldData = $DynamicFieldObject->DynamicFieldGet(
        ID => $FieldID,
    );

    # check for valid dynamic field configuration
    if ( !IsHashRefWithData($DynamicFieldData) ) {
        return $LayoutObject->ErrorScreen(
            Message =>
                $LayoutObject->{LanguageObject}->Translate( 'Could not get data for dynamic field %s', $FieldID ),
        );
    }

    if ( $GetParam{Name} ) {

        # check if name is lowercase
        if ( $GetParam{Name} !~ m{\A (?: [a-zA-Z] | \d )+ \z}xms ) {

            # add server error error class
            $Errors{NameServerError} = 'ServerError';
            $Errors{NameServerErrorMessage} =
                Translatable('The field does not contain only ASCII letters and numbers.');
        }

        # check if name is duplicated
        my %DynamicFieldsList = %{
            $DynamicFieldObject->DynamicFieldList(
                Valid      => 0,
                ResultType => 'HASH',
            )
        };

        %DynamicFieldsList = reverse %DynamicFieldsList;

        if (
            $DynamicFieldsList{ $GetParam{Name} } &&
            $DynamicFieldsList{ $GetParam{Name} } ne $FieldID
            )
        {

            # add server error class
            $Errors{NameServerError}        = 'ServerError';
            $Errors{NameServerErrorMessage} = Translatable('There is another field with the same name.');
        }

        # if it's an internal field, it's name should not change
        if (
            $DynamicFieldData->{InternalField} &&
            $DynamicFieldsList{ $GetParam{Name} } ne $FieldID
            )
        {

            # add server error class
            $Errors{NameServerError}        = 'ServerError';
            $Errors{NameServerErrorMessage} = Translatable('The name for this field should not change.');
            $Param{InternalField}           = $DynamicFieldData->{InternalField};
        }
    }

    if ( $GetParam{FieldOrder} ) {

        # check if field order is numeric and positive
        if ( $GetParam{FieldOrder} !~ m{\A (?: \d )+ \z}xms ) {

            # add server error error class
            $Errors{FieldOrderServerError}        = 'ServerError';
            $Errors{FieldOrderServerErrorMessage} = Translatable('The field must be numeric.');
        }
    }

    for my $ConfigParam (qw( ObjectType ObjectTypeName FieldType FieldTypeName ValidID )) {
        $GetParam{$ConfigParam} = $ParamObject->GetParam( Param => $ConfigParam );
    }

    # uncorrectable errors
    if ( !$GetParam{ValidID} ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Need ValidID'),
        );
    }

    # return to change screen if errors
    if (%Errors) {
        return $Self->_ShowScreen(
            %Param,
            %Errors,
            %GetParam,
            ID   => $FieldID,
            Mode => 'Change',
        );
    }

    # set specific config
    my $FieldConfig = {
        DefaultValue       => '',
        PossibleNone       => 1,
        TranslatableValues => 1,
    };

    # update dynamic field (FieldType and ObjectType cannot be changed; use old values)
    my $UpdateSuccess = $DynamicFieldObject->DynamicFieldUpdate(
        ID         => $FieldID,
        Name       => $GetParam{Name},
        Label      => $GetParam{Label},
        FieldOrder => $GetParam{FieldOrder},
        FieldType  => $DynamicFieldData->{FieldType},
        ObjectType => $DynamicFieldData->{ObjectType},
        Config     => $FieldConfig,
        ValidID    => $GetParam{ValidID},
        UserID     => $Self->{UserID},
    );

    if ( !$UpdateSuccess ) {
        return $LayoutObject->ErrorScreen(
            Message => $LayoutObject->{LanguageObject}->Translate( 'Could not update the field %s', $GetParam{Name} ),
        );
    }

    return $LayoutObject->Redirect(
        OP => "Action=AdminDynamicField",
    );
}

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

    $Param{DisplayFieldName} = 'New';

    if ( $Param{Mode} eq 'Change' ) {
        $Param{ShowWarning}      = 'ShowWarning';
        $Param{DisplayFieldName} = $Param{Name};
    }

    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    # header
    my $Output = $LayoutObject->Header();
    $Output .= $LayoutObject->NavigationBar();

    # get all fields
    my $DynamicFieldList = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        Valid => 0,
    );

    # get the list of order numbers (is already sorted).
    my @DynamicfieldOrderList;
    my %DynamicfieldNamesList;
    for my $Dynamicfield ( @{$DynamicFieldList} ) {
        push @DynamicfieldOrderList, $Dynamicfield->{FieldOrder};
        $DynamicfieldNamesList{ $Dynamicfield->{FieldOrder} } = $Dynamicfield->{Label};
    }

    # when adding we need to create an extra order number for the new field
    if ( $Param{Mode} eq 'Add' ) {

        # get the last element form the order list and add 1
        my $LastOrderNumber = $DynamicfieldOrderList[-1];
        $LastOrderNumber++;

        # add this new order number to the end of the list
        push @DynamicfieldOrderList, $LastOrderNumber;
    }

    # show the names of the other fields to ease ordering
    my %OrderNamesList;
    my $CurrentlyText = $LayoutObject->{LanguageObject}->Translate('Currently') . ': ';
    for my $OrderNumber ( sort @DynamicfieldOrderList ) {
        $OrderNamesList{$OrderNumber} = $OrderNumber;
        if ( $DynamicfieldNamesList{$OrderNumber} && $OrderNumber ne $Param{FieldOrder} ) {
            $OrderNamesList{$OrderNumber} = $OrderNumber . ' - '
                . $CurrentlyText
                . $DynamicfieldNamesList{$OrderNumber};
        }
    }

    my $DynamicFieldOrderStrg = $LayoutObject->BuildSelection(
        Data          => \%OrderNamesList,
        Name          => 'FieldOrder',
        SelectedValue => $Param{FieldOrder} || 1,
        PossibleNone  => 0,
        Translation   => 0,
        Sort          => 'NumericKey',
        Class         => 'W75pc Validate_Number Modernize',
    );

    my %ValidList = $Kernel::OM->Get('Kernel::System::Valid')->ValidList();

    # create the Validity select
    my $ValidityStrg = $LayoutObject->BuildSelection(
        Data         => \%ValidList,
        Name         => 'ValidID',
        SelectedID   => $Param{ValidID} || 1,
        PossibleNone => 0,
        Translation  => 1,
        Class        => 'W50pc Modernize',
    );

    # create the possible values template
    $LayoutObject->Block(
        Name => 'ValueTemplate',
        Data => {
            %Param,
        },
    );

    my $ReadonlyInternalField = '';

    # Internal fields can not be deleted and name should not change.
    if ( $Param{InternalField} ) {
        $LayoutObject->Block(
            Name => 'InternalField',
            Data => {%Param},
        );
        $ReadonlyInternalField = 'readonly="readonly"';
    }

    # generate output
    $Output .= $LayoutObject->Output(
        TemplateFile => 'AdminDynamicFieldMasterSlave',
        Data         => {
            %Param,
            ValidityStrg          => $ValidityStrg,
            DynamicFieldOrderStrg => $DynamicFieldOrderStrg,
            ReadonlyInternalField => $ReadonlyInternalField,
        },
    );

    $Output .= $LayoutObject->Footer();

    return $Output;
}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok1vZHVsZXM6OkFnZW50UHJlTWFzdGVyU2xhdmU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgojIHByZXZlbnQgdXNlZCBvbmNlIHdhcm5pbmcKdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwoKb3VyICRPYmplY3RNYW5hZ2VyRGlzYWJsZWQgPSAxOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7JVBhcmFtfTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCnN1YiBQcmVSdW4gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGRvIG9ubHkgdXNlIHRoaXMgaW4gcGhvbmUgYW5kIGVtYWlsIHRpY2tldAogICAgcmV0dXJuIGlmICggJFNlbGYtPntBY3Rpb259ICF+IC9eQWdlbnRUaWNrZXQoRW1haWx8UGhvbmUpJC8gKTsKCiAgICAjIGdldCBjb25maWcgb2JqZWN0CiAgICBteSAkQ29uZmlnT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKTsKCiAgICAjIGdldCBtYXN0ZXIvc2xhdmUgZHluYW1pYyBmaWVsZAogICAgbXkgJE1hc3RlclNsYXZlRHluYW1pY0ZpZWxkID0gJENvbmZpZ09iamVjdC0+R2V0KCdNYXN0ZXJTbGF2ZTo6RHluYW1pY0ZpZWxkJykgfHwgJyc7CgogICAgIyByZXR1cm4gaWYgbm8gY29uZmlnIG9wdGlvbiBpcyB1c2VkCiAgICByZXR1cm4gaWYgISRNYXN0ZXJTbGF2ZUR5bmFtaWNGaWVsZDsKCiAgICAjIHNldCBkeW5hbWljIGZpZWxkIGFzIHNob3duCiAgICAkQ29uZmlnT2JqZWN0LT57IlRpY2tldDo6RnJvbnRlbmQ6OiRTZWxmLT57QWN0aW9ufSJ9LT57RHluYW1pY0ZpZWxkfS0+eyRNYXN0ZXJTbGF2ZUR5bmFtaWNGaWVsZH0gPSAxOwoKICAgIHJldHVybjsKfQoKMTsK
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# $origin: otrs - 61764fc3cfd0e81cd5f6dbcf08115e12240f412d - Kernel/Modules/AgentTicketActionCommon.pm
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

# ---
# OTRSMasterSlave
# ---
#package Kernel::Modules::AgentTicketActionCommon;

# This module uses AgentTicketActionCommon as a base, for easy update and framework compatibility
# special markers has been set along the file to easy spot the differences introduced by
# OTRSMasterSlave package
package Kernel::Modules::AgentTicketMasterSlave;
# ---

use strict;
use warnings;

use Kernel::System::EmailParser;
use Kernel::System::VariableCheck qw(:all);
use Kernel::Language qw(Translatable);

our $ObjectManagerDisabled = 1;

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

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

    # Try to load draft if requested.
    if (
        $Kernel::OM->Get('Kernel::Config')->Get("Ticket::Frontend::$Self->{Action}")->{FormDraft}
        && $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'LoadFormDraft' )
        && $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'FormDraftID' )
        )
    {
        $Self->{LoadedFormDraftID} = $Kernel::OM->Get('Kernel::System::Web::Request')->LoadFormDraft(
            FormDraftID => $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'FormDraftID' ),
            UserID      => $Self->{UserID},
        );
    }

    # get article for whom this should be a reply, if available
    my $ReplyToArticle = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'ReplyToArticle' ) || '';
    my $TicketID       = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'TicketID' )       || '';

    # check if ReplyToArticle really belongs to the ticket
    my %ReplyToArticleContent;
    my @ReplyToAdresses;
    if ($ReplyToArticle) {

        my $ArticleBackendObject = $Kernel::OM->Get('Kernel::System::Ticket::Article')->BackendForArticle(
            TicketID  => $TicketID,
            ArticleID => $ReplyToArticle,
        );
        %ReplyToArticleContent = $ArticleBackendObject->ArticleGet(
            TicketID      => $TicketID,
            ArticleID     => $ReplyToArticle,
            DynamicFields => 0,
        );

        $Self->{ReplyToArticle}        = $ReplyToArticle;
        $Self->{ReplyToArticleContent} = \%ReplyToArticleContent;

        # get sender of original note (to inform sender about answer)
        if ( $ReplyToArticleContent{CreateBy} ) {
            my @ReplyToSenderID = ( $ReplyToArticleContent{CreateBy} );
            $Self->{ReplyToSenderUserID} = \@ReplyToSenderID;
        }

        # if article belongs to other ticket, don't use it as reply
        if ( $ReplyToArticleContent{TicketID} ne $Self->{TicketID} ) {
            $Self->{ReplyToArticle} = "";
        }

        # if article is not of type note-internal, don't use it as reply
        if (
            $ArticleBackendObject->ChannelNameGet() ne 'Internal'
            || (
                $ArticleBackendObject->ChannelNameGet() eq 'Internal'
                && $ReplyToArticleContent{SenderType} ne 'agent'
            )
            )
        {
            $Self->{ReplyToArticle} = "";
        }
    }

    # get form id
    $Self->{FormID} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'FormID' );

    # create form id
    if ( !$Self->{FormID} ) {
        $Self->{FormID} = $Kernel::OM->Get('Kernel::System::Web::UploadCache')->FormIDCreate();
    }

    return $Self;
}

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

    # get needed objects
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    my $ParamObject  = $Kernel::OM->Get('Kernel::System::Web::Request');

    # check needed stuff
    if ( !$Self->{TicketID} ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('No TicketID is given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get config of frontend module
    my $Config = $ConfigObject->Get("Ticket::Frontend::$Self->{Action}");

    # check permissions
    my $Access = $TicketObject->TicketPermission(
        Type     => $Config->{Permission},
        TicketID => $Self->{TicketID},
        UserID   => $Self->{UserID}
    );

    # error screen, don't show ticket
    if ( !$Access ) {
        return $LayoutObject->NoPermission(
            Message => $LayoutObject->{LanguageObject}->Translate( 'You need %s permissions!', $Config->{Permission} ),
            WithHeader => 'yes',
        );
    }

    # get ACL restrictions
    my %PossibleActions = ( 1 => $Self->{Action} );

    my $ACL = $TicketObject->TicketAcl(
        Data          => \%PossibleActions,
        Action        => $Self->{Action},
        TicketID      => $Self->{TicketID},
        ReturnType    => 'Action',
        ReturnSubType => '-',
        UserID        => $Self->{UserID},
    );
    my %AclAction = $TicketObject->TicketAclActionData();

    # check if ACL restrictions exist
    if ( $ACL || IsHashRefWithData( \%AclAction ) ) {

        my %AclActionLookup = reverse %AclAction;

        # show error screen if ACL prohibits this action
        if ( !$AclActionLookup{ $Self->{Action} } ) {
            return $LayoutObject->NoPermission( WithHeader => 'yes' );
        }
    }

    # Check for failed draft loading request.
    if (
        $ParamObject->GetParam( Param => 'LoadFormDraft' )
        && !$Self->{LoadedFormDraftID}
        )
    {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Loading draft failed!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    my %Ticket = $TicketObject->TicketGet(
        TicketID      => $Self->{TicketID},
        DynamicFields => 1,
    );

    my $LoadedFormDraft;
    if ( $Self->{LoadedFormDraftID} ) {
        $LoadedFormDraft = $Kernel::OM->Get('Kernel::System::FormDraft')->FormDraftGet(
            FormDraftID => $Self->{LoadedFormDraftID},
            GetContent  => 0,
            UserID      => $Self->{UserID},
        );

        my @Articles = $Kernel::OM->Get('Kernel::System::Ticket::Article')->ArticleList(
            TicketID => $Self->{TicketID},
            OnlyLast => 1,
        );

        if (@Articles) {
            my $LastArticle = $Articles[0];

            my $LastArticleSystemTime;
            if ( $LastArticle->{CreateTime} ) {
                my $LastArticleSystemTimeObject = $Kernel::OM->Create(
                    'Kernel::System::DateTime',
                    ObjectParams => {
                        String => $LastArticle->{CreateTime},
                    },
                );
                $LastArticleSystemTime = $LastArticleSystemTimeObject->ToEpoch();
            }

            my $FormDraftSystemTimeObject = $Kernel::OM->Create(
                'Kernel::System::DateTime',
                ObjectParams => {
                    String => $LoadedFormDraft->{ChangeTime},
                },
            );
            my $FormDraftSystemTime = $FormDraftSystemTimeObject->ToEpoch();

            if ( !$LastArticleSystemTime || $FormDraftSystemTime <= $LastArticleSystemTime ) {
                $Param{FormDraftOutdated} = 1;
            }
        }
    }

    if ( IsHashRefWithData($LoadedFormDraft) ) {

        $LoadedFormDraft->{ChangeByName} = $Kernel::OM->Get('Kernel::System::User')->UserName(
            UserID => $LoadedFormDraft->{ChangeBy},
        );
    }

    $LayoutObject->Block(
        Name => 'Properties',
        Data => {
            FormDraft      => $Config->{FormDraft},
            FormDraftID    => $Self->{LoadedFormDraftID},
            FormDraftTitle => $LoadedFormDraft ? $LoadedFormDraft->{Title} : '',
            FormDraftMeta  => $LoadedFormDraft,
            FormID         => $Self->{FormID},
            ReplyToArticle => $Self->{ReplyToArticle},
            %Ticket,
            %Param,
        },
    );

    # show right header
    $LayoutObject->Block(
        Name => 'Header' . $Self->{Action},
        Data => {
            %Ticket,
        },
    );

    # get lock state
    if ( $Config->{RequiredLock} ) {
        if ( !$TicketObject->TicketLockGet( TicketID => $Self->{TicketID} ) ) {

            my $Lock = $TicketObject->TicketLockSet(
                TicketID => $Self->{TicketID},
                Lock     => 'lock',
                UserID   => $Self->{UserID}
            );

            if ($Lock) {

                # Set new owner if ticket owner is different then logged user.
                if ( $Ticket{OwnerID} != $Self->{UserID} ) {

                    # Remember previous owner, which will be used to restore ticket owner on undo action.
                    $Param{PreviousOwner} = $Ticket{OwnerID};

                    $TicketObject->TicketOwnerSet(
                        TicketID  => $Self->{TicketID},
                        UserID    => $Self->{UserID},
                        NewUserID => $Self->{UserID},
                    );
                }

                # Show lock state.
                $LayoutObject->Block(
                    Name => 'PropertiesLock',
                    Data => {
                        %Param,
                        TicketID => $Self->{TicketID},
                    },
                );
            }
        }
        else {
            my $AccessOk = $TicketObject->OwnerCheck(
                TicketID => $Self->{TicketID},
                OwnerID  => $Self->{UserID},
            );
            if ( !$AccessOk ) {
                my $Output = $LayoutObject->Header(
                    Type      => 'Small',
                    Value     => $Ticket{Number},
                    BodyClass => 'Popup',
                );
                $Output .= $LayoutObject->Warning(
                    Message => Translatable('Sorry, you need to be the ticket owner to perform this action.'),
                    Comment => Translatable('Please change the owner first.'),
                );
                $Output .= $LayoutObject->Footer(
                    Type => 'Small',
                );
                return $Output;
            }

            # show back link
            $LayoutObject->Block(
                Name => 'TicketBack',
                Data => {
                    %Param,
                    TicketID => $Self->{TicketID},
                },
            );
        }
    }
    else {
        $LayoutObject->Block(
            Name => 'TicketBack',
            Data => {
                %Param,
                %Ticket,
            },
        );
    }

    # get params
    my %GetParam;
    for my $Key (
        qw(
        NewStateID NewPriorityID TimeUnits IsVisibleForCustomer Title Body Subject NewQueueID
        Year Month Day Hour Minute NewOwnerID NewResponsibleID TypeID ServiceID SLAID
        Expand ReplyToArticle StandardTemplateID CreateArticle FormDraftID Title
        )
        )
    {
        $GetParam{$Key} = $ParamObject->GetParam( Param => $Key );
    }

    # ACL compatibility translation
    my %ACLCompatGetParam = (
        StateID       => $GetParam{NewStateID},
        PriorityID    => $GetParam{NewPriorityID},
        QueueID       => $GetParam{NewQueueID},
        OwnerID       => $GetParam{NewOwnerID},
        ResponsibleID => $GetParam{NewResponsibleID},
    );

    # get dynamic field values form http request
    my %DynamicFieldValues;

    # define the dynamic fields to show based on the object type
    my $ObjectType = ['Ticket'];
# ---
# OTRSMasterSlave
# ---
    # get master/slave dynamic field
    my $MasterSlaveDynamicField = $ConfigObject->Get('MasterSlave::DynamicField') || '';
    my $MasterSlaveAdvancedEnabled = $ConfigObject->Get('MasterSlave::AdvancedEnabled') || 0;
# ---

    # only screens that add notes can modify Article dynamic fields
    if ( $Config->{Note} ) {
        $ObjectType = [ 'Ticket', 'Article' ];
    }
# ---
# OTRSMasterSlave
# ---
    if ($MasterSlaveAdvancedEnabled) {
        my $Display = $Config->{MasterSlaveMandatory} ? 2 : 1;
        $Config->{DynamicField}->{$MasterSlaveDynamicField} = $Display;
    }
# ---

    # get the dynamic fields for this screen
    my $DynamicField = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        Valid       => 1,
        ObjectType  => $ObjectType,
        FieldFilter => $Config->{DynamicField} || {},
    );

    # get dynamic field backend object
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    # cycle trough the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{$DynamicField} ) {
        next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);

        # extract the dynamic field value from the web request
        $DynamicFieldValues{ $DynamicFieldConfig->{Name} } = $DynamicFieldBackendObject->EditFieldValueGet(
            DynamicFieldConfig => $DynamicFieldConfig,
            ParamObject        => $ParamObject,
            LayoutObject       => $LayoutObject,
        );
    }

    # convert dynamic field values into a structure for ACLs
    my %DynamicFieldACLParameters;
    DYNAMICFIELD:
    for my $DynamicFieldItem ( sort keys %DynamicFieldValues ) {
        next DYNAMICFIELD if !$DynamicFieldItem;
        next DYNAMICFIELD if !defined $DynamicFieldValues{$DynamicFieldItem};

        $DynamicFieldACLParameters{ 'DynamicField_' . $DynamicFieldItem } = $DynamicFieldValues{$DynamicFieldItem};
    }
    $GetParam{DynamicField} = \%DynamicFieldACLParameters;

    # transform pending time, time stamp based on user time zone
    if (
        defined $GetParam{Year}
        && defined $GetParam{Month}
        && defined $GetParam{Day}
        && defined $GetParam{Hour}
        && defined $GetParam{Minute}
        )
    {
        %GetParam = $LayoutObject->TransformDateSelection(
            %GetParam,
        );
    }

    # rewrap body if no rich text is used
    if ( $GetParam{Body} && !$LayoutObject->{BrowserRichText} ) {
        $GetParam{Body} = $LayoutObject->WrapPlainText(
            MaxCharacters => $ConfigObject->Get('Ticket::Frontend::TextAreaNote'),
            PlainText     => $GetParam{Body},
        );
    }

    # get upload cache object
    my $UploadCacheObject = $Kernel::OM->Get('Kernel::System::Web::UploadCache');

    if (
        $Self->{Subaction} eq 'Store'
        || $Self->{LoadedFormDraftID}
        )
    {

        # challenge token check for write action
        if ( $Self->{Subaction} eq 'Store' ) {
            $LayoutObject->ChallengeTokenCheck();
        }

        $GetParam{IsVisibleForCustomer} //= 0;

        # store action
        my %Error;

        # get all attachments meta data
        my @Attachments = $UploadCacheObject->FormIDGetAllFilesMeta(
            FormID => $Self->{FormID},
        );

        # Get and validate draft action.
        my $FormDraftAction = $ParamObject->GetParam( Param => 'FormDraftAction' );
        if ( $FormDraftAction && !$Config->{FormDraft} ) {
            return $LayoutObject->ErrorScreen(
                Message => Translatable('FormDraft functionality disabled!'),
                Comment => Translatable('Please contact the administrator.'),
            );
        }

        my %FormDraftResponse;

        # Check draft name.
        if (
            $FormDraftAction
            && ( $FormDraftAction eq 'Add' || $FormDraftAction eq 'Update' )
            )
        {
            my $Title = $ParamObject->GetParam( Param => 'FormDraftTitle' );

            # A draft name is required.
            if ( !$Title ) {

                %FormDraftResponse = (
                    Success      => 0,
                    ErrorMessage => $Kernel::OM->Get('Kernel::Language')->Translate("Draft name is required!"),
                );
            }

            # Chosen draft name must be unique.
            else {
                my $FormDraftList = $Kernel::OM->Get('Kernel::System::FormDraft')->FormDraftListGet(
                    ObjectType => 'Ticket',
                    ObjectID   => $Self->{TicketID},
                    Action     => $Self->{Action},
                    UserID     => $Self->{UserID},
                );
                DRAFT:
                for my $FormDraft ( @{$FormDraftList} ) {

                    # No existing draft with same name.
                    next DRAFT if $Title ne $FormDraft->{Title};

                    # Same name for update on existing draft.
                    if (
                        $GetParam{FormDraftID}
                        && $FormDraftAction eq 'Update'
                        && $GetParam{FormDraftID} eq $FormDraft->{FormDraftID}
                        )
                    {
                        next DRAFT;
                    }

                    # Another draft with the chosen name already exists.
                    %FormDraftResponse = (
                        Success      => 0,
                        ErrorMessage => $Kernel::OM->Get('Kernel::Language')
                            ->Translate( "FormDraft name %s is already in use!", $Title ),
                    );
                    last DRAFT;
                }
            }
        }

        # Perform draft action instead of saving form data in ticket/article.
        if ( $FormDraftAction && !%FormDraftResponse ) {

            # Reset FormDraftID to prevent updating existing draft.
            if ( $FormDraftAction eq 'Add' && $GetParam{FormDraftID} ) {
                $ParamObject->{Query}->param(
                    -name  => 'FormDraftID',
                    -value => '',
                );
            }

            my $FormDraftActionOk;
            if (
                $FormDraftAction eq 'Add'
                ||
                ( $FormDraftAction eq 'Update' && $GetParam{FormDraftID} )
                )
            {
                $FormDraftActionOk = $ParamObject->SaveFormDraft(
                    UserID         => $Self->{UserID},
                    ObjectType     => 'Ticket',
                    ObjectID       => $Self->{TicketID},
                    OverrideParams => {
                        ReplyToArticle => undef,
                    },
                );
            }

            if ($FormDraftActionOk) {
                $FormDraftResponse{Success} = 1;
            }
            else {
                %FormDraftResponse = (
                    Success      => 0,
                    ErrorMessage => 'Could not perform requested draft action!',
                );
            }
        }

        if (%FormDraftResponse) {

            # build JSON output
            my $JSON = $LayoutObject->JSONEncode(
                Data => \%FormDraftResponse,
            );

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

        # get state object
        my $StateObject = $Kernel::OM->Get('Kernel::System::State');

        # check pending time
        if ( $GetParam{NewStateID} ) {
            my %StateData = $StateObject->StateGet(
                ID => $GetParam{NewStateID},
            );

            # check state type
            if ( $StateData{TypeName} =~ /^pending/i ) {

                # check needed stuff
                for my $Needed (qw(Year Month Day Hour Minute)) {
                    if ( !defined $GetParam{$Needed} ) {
                        $Error{'DateInvalid'} = 'ServerError';
                    }
                }

                # create datetime object
                my $PendingDateTimeObject = $Kernel::OM->Create(
                    'Kernel::System::DateTime',
                    ObjectParams => {
                        %GetParam,
                        Second => 0,
                    },
                );

                # get current system epoch
                my $CurSystemDateTimeObject = $Kernel::OM->Create('Kernel::System::DateTime');

                # check date
                if (
                    !$PendingDateTimeObject
                    || $PendingDateTimeObject < $CurSystemDateTimeObject
                    )
                {
                    $Error{'DateInvalid'} = 'ServerError';
                }
            }
        }

        if ( $Config->{Note} && $Config->{NoteMandatory} ) {

            # check subject
            if ( !$GetParam{Subject} ) {
                $Error{'SubjectInvalid'} = 'ServerError';
            }

            # check body
            if ( !$GetParam{Body} ) {
                $Error{'BodyInvalid'} = 'ServerError';
            }
        }

        # check owner
        if ( $Config->{Owner} && $Config->{OwnerMandatory} ) {
            if ( !$GetParam{NewOwnerID} ) {
                $Error{'NewOwnerInvalid'} = 'ServerError';
            }
        }

        # check responsible
        if ( $Config->{Responsible} && $Config->{ResponsibleMandatory} ) {
            if ( !$GetParam{NewResponsibleID} ) {
                $Error{'NewResponsibleInvalid'} = 'ServerError';
            }
        }

        # check title
        if ( $Config->{Title} && !$GetParam{Title} ) {
            $Error{'TitleInvalid'} = 'ServerError';
        }

        # check type
        if (
            ( $ConfigObject->Get('Ticket::Type') )
            &&
            ( $Config->{TicketType} ) &&
            ( !$GetParam{TypeID} )
            )
        {
            $Error{'TypeIDInvalid'} = ' ServerError';
        }

        # check service
        if (
            $ConfigObject->Get('Ticket::Service')
            && $Config->{Service}
            && $GetParam{SLAID}
            && !$GetParam{ServiceID}
            )
        {
            $Error{'ServiceInvalid'} = ' ServerError';
        }

        # check mandatory service
        if (
            $ConfigObject->Get('Ticket::Service')
            && $Config->{Service}
            && $Config->{ServiceMandatory}
            && !$GetParam{ServiceID}
            )
        {
            $Error{'ServiceInvalid'} = ' ServerError';
        }

        # check mandatory sla
        if (
            $ConfigObject->Get('Ticket::Service')
            && $Config->{Service}
            && $Config->{SLAMandatory}
            && !$GetParam{SLAID}
            )
        {
            $Error{'SLAInvalid'} = ' ServerError';
        }

        # check mandatory queue
        if ( $Config->{Queue} && $Config->{QueueMandatory} ) {
            if ( !$GetParam{NewQueueID} ) {
                $Error{'NewQueueInvalid'} = 'ServerError';
            }
        }

        # check mandatory state
        if ( $Config->{State} && $Config->{StateMandatory} ) {
            if ( !$GetParam{NewStateID} ) {
                $Error{'NewStateInvalid'} = 'ServerError';
            }
        }

        # check time units, but only if the current screen has a note
        #   (accounted time can only be stored if and article is generated)
        if (
            $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
            && $Config->{Note}
            && $GetParam{TimeUnits} eq ''
            )
        {
            $Error{'TimeUnitsInvalid'} = ' ServerError';
        }

        # check expand
        if ( $GetParam{Expand} ) {
            %Error = ();
            $Error{Expand} = 1;
        }

        # create html strings for all dynamic fields
        my @TicketTypeDynamicFields;
        my @ArticleTypeDynamicFields;

        # cycle trough the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( @{$DynamicField} ) {
            next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);
# ---
# OTRSMasterSlave
# ---
            if (
                !$MasterSlaveAdvancedEnabled
                && $DynamicFieldConfig->{Name} eq $MasterSlaveDynamicField
                )
            {
                next DYNAMICFIELD;
            }
# ---

            my $PossibleValuesFilter;

            my $IsACLReducible = $DynamicFieldBackendObject->HasBehavior(
                DynamicFieldConfig => $DynamicFieldConfig,
                Behavior           => 'IsACLReducible',
            );

            if ($IsACLReducible) {

                # get PossibleValues
                my $PossibleValues = $DynamicFieldBackendObject->PossibleValuesGet(
                    DynamicFieldConfig => $DynamicFieldConfig,
                );

                # check if field has PossibleValues property in its configuration
                if ( IsHashRefWithData($PossibleValues) ) {

                    # convert possible values key => value to key => key for ACLs using a Hash slice
                    my %AclData = %{$PossibleValues};
                    @AclData{ keys %AclData } = keys %AclData;

                    # set possible values filter from ACLs
                    my $ACL = $TicketObject->TicketAcl(
                        %GetParam,
                        Action        => $Self->{Action},
                        TicketID      => $Self->{TicketID},
                        ReturnType    => 'Ticket',
                        ReturnSubType => 'DynamicField_' . $DynamicFieldConfig->{Name},
                        Data          => \%AclData,
                        UserID        => $Self->{UserID},
                    );
                    if ($ACL) {
                        my %Filter = $TicketObject->TicketAclData();

                        # convert Filer key => key back to key => value using map
                        %{$PossibleValuesFilter} = map { $_ => $PossibleValues->{$_} }
                            keys %Filter;
                    }
                }
            }

            my $ValidationResult;
# ---
# OTRSMasterSlave
# ---
            if ( $DynamicFieldConfig->{Name} eq $MasterSlaveDynamicField ) {
               $PossibleValuesFilter = $Self->_GetMasterSlaveData(
                Ticket => \%Ticket,
                MasterSlaveDynamicField => $MasterSlaveDynamicField,
               );
            }
# ---

            # Do not validate only if object type is Article and CreateArticle value is not defined.
            if ( !( $DynamicFieldConfig->{ObjectType} eq 'Article' && !$GetParam{CreateArticle} ) ) {

                $ValidationResult = $DynamicFieldBackendObject->EditFieldValueValidate(
                    DynamicFieldConfig   => $DynamicFieldConfig,
                    PossibleValuesFilter => $PossibleValuesFilter,
                    ParamObject          => $ParamObject,
                    Mandatory =>
                        $Config->{DynamicField}->{ $DynamicFieldConfig->{Name} } == 2,
                );

                if ( !IsHashRefWithData($ValidationResult) ) {
                    return $LayoutObject->ErrorScreen(
                        Message =>
                            $LayoutObject->{LanguageObject}->Translate(
                            'Could not perform validation on field %s!', $DynamicFieldConfig->{Label}
                            ),
                        Comment => Translatable('Please contact the administrator.'),
                    );
                }

                # Propagate validation error to the Error variable to be detected by the frontend.
                if ( $ValidationResult->{ServerError} )
                {
                    $Error{ $DynamicFieldConfig->{Name} } = ' ServerError';
                }
            }

            if ( $DynamicFieldConfig->{ObjectType} eq 'Ticket' ) {

                # Get field html.
                my $DynamicFieldHTML = $DynamicFieldBackendObject->EditFieldRender(
                    DynamicFieldConfig   => $DynamicFieldConfig,
                    PossibleValuesFilter => $PossibleValuesFilter,
                    ServerError          => $ValidationResult->{ServerError} || '',
                    ErrorMessage         => $ValidationResult->{ErrorMessage} || '',
                    Mandatory            => $Config->{DynamicField}->{ $DynamicFieldConfig->{Name} } == 2,
                    LayoutObject         => $LayoutObject,
                    ParamObject          => $ParamObject,
                    AJAXUpdate           => 1,
                    UpdatableFields      => $Self->_GetFieldsToUpdate(),
                );

                push @TicketTypeDynamicFields, {
                    Name  => $DynamicFieldConfig->{Name},
                    Label => $DynamicFieldHTML->{Label},
                    Field => $DynamicFieldHTML->{Field},
                };
            }
            elsif ( $DynamicFieldConfig->{ObjectType} eq 'Article' ) {
                my $Class            = '';
                my $MandatoryTooltip = 0;

                if ( $Config->{DynamicField}->{ $DynamicFieldConfig->{Name} } == 2 ) {
                    if (
                        $Config->{NoteMandatory} ||
                        $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
                        )
                    {
                        $Class = 'Validate_Required';
                    }
                    else {
                        $Class            = 'Validate_DependingRequiredAND Validate_Depending_CreateArticle';
                        $MandatoryTooltip = 1;
                    }
                }

                # Get field html.
                my $DynamicFieldHTML = $DynamicFieldBackendObject->EditFieldRender(
                    DynamicFieldConfig   => $DynamicFieldConfig,
                    PossibleValuesFilter => $PossibleValuesFilter,
                    ServerError          => $ValidationResult->{ServerError} || '',
                    ErrorMessage         => $ValidationResult->{ErrorMessage} || '',
                    Mandatory            => ( $Class eq 'Validate_Required' ) ? 1 : 0,
                    Class                => $Class,
                    LayoutObject         => $LayoutObject,
                    ParamObject          => $ParamObject,
                    AJAXUpdate           => 1,
                    UpdatableFields      => $Self->_GetFieldsToUpdate(),
                );

                push @ArticleTypeDynamicFields, {
                    Name             => $DynamicFieldConfig->{Name},
                    Label            => $DynamicFieldHTML->{Label},
                    Field            => $DynamicFieldHTML->{Field},
                    MandatoryTooltip => $MandatoryTooltip,
                };
            }
        }

        # Make sure we don't save form if a draft was loaded.
        if ( $Self->{LoadedFormDraftID} ) {
            %Error = ( LoadedFormDraft => 1 );
        }

        # check errors
        if (%Error) {

            my $Output = $LayoutObject->Header(
                Type      => 'Small',
                Value     => $Ticket{TicketNumber},
                BodyClass => 'Popup',
            );
            $Output .= $Self->_Mask(
                Attachments       => \@Attachments,
                TimeUnitsRequired => (
                    $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
                    ? 'Validate_Required'
                    : ''
                ),
                %Ticket,
                TicketTypeDynamicFields  => \@TicketTypeDynamicFields,
                ArticleTypeDynamicFields => \@ArticleTypeDynamicFields,

                %GetParam,
                %Error,
            );
            $Output .= $LayoutObject->Footer(
                Type => 'Small',
            );
            return $Output;
        }

        # set new title
        if ( $Config->{Title} ) {
            if ( defined $GetParam{Title} ) {
                $TicketObject->TicketTitleUpdate(
                    Title    => $GetParam{Title},
                    TicketID => $Self->{TicketID},
                    UserID   => $Self->{UserID},
                );
            }
        }

        # set new type
        if ( $ConfigObject->Get('Ticket::Type') && $Config->{TicketType} ) {
            if ( $GetParam{TypeID} ) {
                $TicketObject->TicketTypeSet(
                    Action   => $Self->{Action},
                    TypeID   => $GetParam{TypeID},
                    TicketID => $Self->{TicketID},
                    UserID   => $Self->{UserID},
                );
            }
        }

        # set new service
        if ( $ConfigObject->Get('Ticket::Service') && $Config->{Service} ) {
            if ( defined $GetParam{ServiceID} ) {
                $TicketObject->TicketServiceSet(
                    %GetParam,
                    %ACLCompatGetParam,
                    Action         => $Self->{Action},
                    ServiceID      => $GetParam{ServiceID},
                    TicketID       => $Self->{TicketID},
                    CustomerUserID => $Ticket{CustomerUserID},
                    UserID         => $Self->{UserID},
                );
            }
            if ( defined $GetParam{SLAID} ) {
                $TicketObject->TicketSLASet(
                    Action   => $Self->{Action},
                    SLAID    => $GetParam{SLAID},
                    TicketID => $Self->{TicketID},
                    UserID   => $Self->{UserID},
                );
            }
        }

        my $UnlockOnAway = 1;

        # move ticket to a new queue, but only if the queue was changed
        if (
            $Config->{Queue}
            && $GetParam{NewQueueID}
            && $GetParam{NewQueueID} ne $Ticket{QueueID}
            )
        {

            # move ticket (send notification if no new owner is selected)
            my $BodyAsText = '';
            if ( $LayoutObject->{BrowserRichText} ) {
                $BodyAsText = $LayoutObject->RichText2Ascii(
                    String => $GetParam{Body} || 0,
                );
            }
            else {
                $BodyAsText = $GetParam{Body} || 0;
            }
            my $Move = $TicketObject->TicketQueueSet(
                QueueID            => $GetParam{NewQueueID},
                UserID             => $Self->{UserID},
                TicketID           => $Self->{TicketID},
                SendNoNotification => $GetParam{NewUserID},
                Comment            => $BodyAsText,
                Action             => $Self->{Action},
            );
            if ( !$Move ) {
                return $LayoutObject->ErrorScreen();
            }
        }

        # set new owner
        my @NotifyDone;
        if ( $Config->{Owner} ) {
            my $BodyText = $LayoutObject->RichText2Ascii(
                String => $GetParam{Body} || '',
            );
            if ( $GetParam{NewOwnerID} ) {
                $TicketObject->TicketLockSet(
                    TicketID => $Self->{TicketID},
                    Lock     => 'lock',
                    UserID   => $Self->{UserID},
                );
                my $Success = $TicketObject->TicketOwnerSet(
                    TicketID  => $Self->{TicketID},
                    UserID    => $Self->{UserID},
                    NewUserID => $GetParam{NewOwnerID},
                    Comment   => $BodyText,
                );
                $UnlockOnAway = 0;

                # remember to not notify owner twice
                if ( $Success && $Success eq 1 ) {
                    push @NotifyDone, $GetParam{NewOwnerID};
                }
            }
        }

        # set new responsible
        if ( $ConfigObject->Get('Ticket::Responsible') && $Config->{Responsible} ) {
            if ( $GetParam{NewResponsibleID} ) {
                my $BodyText = $LayoutObject->RichText2Ascii(
                    String => $GetParam{Body} || '',
                );
                my $Success = $TicketObject->TicketResponsibleSet(
                    TicketID  => $Self->{TicketID},
                    UserID    => $Self->{UserID},
                    NewUserID => $GetParam{NewResponsibleID},
                    Comment   => $BodyText,
                );

                # remember to not notify responsible twice
                if ( $Success && $Success eq 1 ) {
                    push @NotifyDone, $GetParam{NewResponsibleID};
                }
            }
        }

        # add note
        my $ArticleID = '';
        my $ReturnURL;

        # set priority
        if ( $Config->{Priority} && $GetParam{NewPriorityID} ) {
            $TicketObject->TicketPrioritySet(
                TicketID   => $Self->{TicketID},
                PriorityID => $GetParam{NewPriorityID},
                UserID     => $Self->{UserID},
            );
        }

        # set state
        if ( $Config->{State} && $GetParam{NewStateID} ) {
            $TicketObject->TicketStateSet(
                TicketID     => $Self->{TicketID},
                StateID      => $GetParam{NewStateID},
                UserID       => $Self->{UserID},
                DynamicField => $GetParam{DynamicField},
            );

            # unlock the ticket after close
            my %StateData = $StateObject->StateGet(
                ID => $GetParam{NewStateID},
            );

            # set unlock on close state
            if ( $StateData{TypeName} =~ /^close/i ) {
                $TicketObject->TicketLockSet(
                    TicketID => $Self->{TicketID},
                    Lock     => 'unlock',
                    UserID   => $Self->{UserID},
                );
            }

            # set pending time on pending state
            elsif ( $StateData{TypeName} =~ /^pending/i ) {

                # set pending time
                $TicketObject->TicketPendingTimeSet(
                    UserID   => $Self->{UserID},
                    TicketID => $Self->{TicketID},
                    %GetParam,
                );
            }

            # redirect parent window to last screen overview on closed tickets
            if (
                $StateData{TypeName} =~ /^close/i
                && !$ConfigObject->Get('Ticket::Frontend::RedirectAfterCloseDisabled')
                )
            {
                $ReturnURL = $Self->{LastScreenOverview} || 'Action=AgentDashboard';
            }
        }

        if (
            $GetParam{CreateArticle}
            && $Config->{Note}
            && ( $GetParam{Subject} || $GetParam{Body} )
            )
        {

            if ( !$GetParam{Subject} ) {
                if ( $Config->{Subject} ) {
                    my $Subject = $LayoutObject->Output(
                        Template => $Config->{Subject},
                    );
                    $GetParam{Subject} = $Subject;
                }
                $GetParam{Subject} = $GetParam{Subject}
                    || $LayoutObject->{LanguageObject}->Translate('No subject');
            }

            # get pre loaded attachment
            my @Attachments = $UploadCacheObject->FormIDGetAllFilesData(
                FormID => $Self->{FormID},
            );

            # get submit attachment
            my %UploadStuff = $ParamObject->GetUploadAll(
                Param => 'FileUpload',
            );
            if (%UploadStuff) {
                push @Attachments, \%UploadStuff;
            }

            my $MimeType = 'text/plain';
            if ( $LayoutObject->{BrowserRichText} ) {
                $MimeType = 'text/html';

                # remove unused inline images
                my @NewAttachmentData;
                ATTACHMENT:
                for my $Attachment (@Attachments) {
                    my $ContentID = $Attachment->{ContentID};
                    if (
                        $ContentID
                        && ( $Attachment->{ContentType} =~ /image/i )
                        && ( $Attachment->{Disposition} eq 'inline' )
                        )
                    {
                        my $ContentIDHTMLQuote = $LayoutObject->Ascii2Html(
                            Text => $ContentID,
                        );

                        # workaround for link encode of rich text editor, see bug#5053
                        my $ContentIDLinkEncode = $LayoutObject->LinkEncode($ContentID);
                        $GetParam{Body} =~ s/(ContentID=)$ContentIDLinkEncode/$1$ContentID/g;

                        # ignore attachment if not linked in body
                        next ATTACHMENT
                            if $GetParam{Body} !~ /(\Q$ContentIDHTMLQuote\E|\Q$ContentID\E)/i;
                    }

                    # remember inline images and normal attachments
                    push @NewAttachmentData, \%{$Attachment};
                }
                @Attachments = @NewAttachmentData;

                # verify html document
                $GetParam{Body} = $LayoutObject->RichTextDocumentComplete(
                    String => $GetParam{Body},
                );
            }

            my $From = "\"$Self->{UserFullname}\" <$Self->{UserEmail}>";
            my @NotifyUserIDs;

            # get list of users that will be informed without selection in informed/involved list
            my @UserListWithoutSelection
                = split( ',', $ParamObject->GetParam( Param => 'UserListWithoutSelection' ) || "" );

            # get inform user list
            my @InformUserID = $ParamObject->GetArray( Param => 'InformUserID' );

            # get involved user list
            my @InvolvedUserID = $ParamObject->GetArray( Param => 'InvolvedUserID' );

            if ( $Config->{InformAgent} ) {
                push @NotifyUserIDs, @InformUserID;
            }

            if ( $Config->{InvolvedAgent} ) {
                push @NotifyUserIDs, @InvolvedUserID;
            }

            if ( $Self->{ReplyToArticle} ) {
                push @NotifyUserIDs, @UserListWithoutSelection;
            }

            if ( $Self->{Action} eq 'AgentTicketEmailOutbound' ) {
                $ArticleID = $Kernel::OM->Get('Kernel::System::Ticket::Article::Backend::Email')->ArticleSend(
                    TicketID                        => $Self->{TicketID},
                    SenderType                      => 'agent',
                    From                            => $From,
                    MimeType                        => $MimeType,
                    Charset                         => $LayoutObject->{UserCharset},
                    UserID                          => $Self->{UserID},
                    HistoryType                     => $Config->{HistoryType},
                    HistoryComment                  => $Config->{HistoryComment},
                    ForceNotificationToUserID       => \@NotifyUserIDs,
                    ExcludeMuteNotificationToUserID => \@NotifyDone,
                    UnlockOnAway                    => $UnlockOnAway,
                    Attachment                      => \@Attachments,
                    %GetParam,
                );
            }
            else {
                $ArticleID = $Kernel::OM->Get('Kernel::System::Ticket::Article::Backend::Internal')->ArticleCreate(
                    TicketID                        => $Self->{TicketID},
                    SenderType                      => 'agent',
                    From                            => $From,
                    MimeType                        => $MimeType,
                    Charset                         => $LayoutObject->{UserCharset},
                    UserID                          => $Self->{UserID},
                    HistoryType                     => $Config->{HistoryType},
                    HistoryComment                  => $Config->{HistoryComment},
                    ForceNotificationToUserID       => \@NotifyUserIDs,
                    ExcludeMuteNotificationToUserID => \@NotifyDone,
                    UnlockOnAway                    => $UnlockOnAway,
                    Attachment                      => \@Attachments,
                    %GetParam,
                );
            }

            if ( !$ArticleID ) {
                return $LayoutObject->ErrorScreen();
            }

            # time accounting
            if ( $GetParam{TimeUnits} ) {
                $TicketObject->TicketAccountTime(
                    TicketID  => $Self->{TicketID},
                    ArticleID => $ArticleID,
                    TimeUnit  => $GetParam{TimeUnits},
                    UserID    => $Self->{UserID},
                );
            }

            # remove pre submitted attachments
            $UploadCacheObject->FormIDRemove( FormID => $Self->{FormID} );
        }

        # set dynamic fields
        # cycle through the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( @{$DynamicField} ) {
            next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);
# ---
# OTRSMasterSlave
# ---
            if (
                !$MasterSlaveAdvancedEnabled
                && $DynamicFieldConfig->{Name} eq $MasterSlaveDynamicField
                )
            {
                next DYNAMICFIELD;
            }
# ---

            # set the object ID (TicketID or ArticleID) depending on the field configration
            my $ObjectID = $DynamicFieldConfig->{ObjectType} eq 'Article' ? $ArticleID : $Self->{TicketID};

            # set the value
            my $Success = $DynamicFieldBackendObject->ValueSet(
                DynamicFieldConfig => $DynamicFieldConfig,
                ObjectID           => $ObjectID,
                Value              => $DynamicFieldValues{ $DynamicFieldConfig->{Name} },
                UserID             => $Self->{UserID},
            );
        }

        # If form was called based on a draft,
        #   delete draft since its content has now been used.
        if (
            $GetParam{FormDraftID}
            && !$Kernel::OM->Get('Kernel::System::FormDraft')->FormDraftDelete(
                FormDraftID => $GetParam{FormDraftID},
                UserID      => $Self->{UserID},
            )
            )
        {
            return $LayoutObject->ErrorScreen(
                Message => Translatable('Could not delete draft!'),
                Comment => Translatable('Please contact the administrator.'),
            );
        }

        # load new URL in parent window and close popup
        $ReturnURL ||= "Action=AgentTicketZoom;TicketID=$Self->{TicketID};ArticleID=$ArticleID";

        return $LayoutObject->PopupClose(
            URL => $ReturnURL,
        );
    }
    elsif ( $Self->{Subaction} eq 'AJAXUpdate' ) {
        my %Ticket         = $TicketObject->TicketGet( TicketID => $Self->{TicketID} );
        my $CustomerUser   = $Ticket{CustomerUserID};
        my $ElementChanged = $ParamObject->GetParam( Param => 'ElementChanged' ) || '';

        my $ServiceID;

        # get service value from param if field is visible in the screen
        if ( $ConfigObject->Get('Ticket::Service') && $Config->{Service} ) {
            $ServiceID = $GetParam{ServiceID} || '';
        }

        # otherwise use ticket service value since it can't be changed
        elsif ( $ConfigObject->Get('Ticket::Service') ) {
            $ServiceID = $Ticket{ServiceID} || '';
        }

        my $QueueID = $GetParam{NewQueueID} || $Ticket{QueueID};
        my $StateID = $GetParam{NewStateID} || $Ticket{StateID};

        # get list type
        my $TreeView = 0;
        if ( $ConfigObject->Get('Ticket::Frontend::ListType') eq 'tree' ) {
            $TreeView = 1;
        }

        my $Owners = $Self->_GetOwners(
            %GetParam,
            QueueID  => $QueueID,
            StateID  => $StateID,
            AllUsers => $GetParam{OwnerAll},
        );
        my $OldOwners = $Self->_GetOldOwners(
            %GetParam,
            QueueID  => $QueueID,
            StateID  => $StateID,
            AllUsers => $GetParam{OwnerAll},
        );
        my $ResponsibleUsers = $Self->_GetResponsible(
            %GetParam,
            QueueID  => $QueueID,
            StateID  => $StateID,
            AllUsers => $GetParam{OwnerAll},
        );
        my $Priorities = $Self->_GetPriorities(
            %GetParam,
        );
        my $Services = $Self->_GetServices(
            %GetParam,
            CustomerUserID => $CustomerUser,
            QueueID        => $QueueID,
            StateID        => $StateID,
        );
        my $Types = $Self->_GetTypes(
            %GetParam,
            CustomerUserID => $CustomerUser,
            QueueID        => $QueueID,
            StateID        => $StateID,
        );
        my $NewQueues = $Self->_GetQueues(
            %GetParam,
        );

        # reset previous ServiceID to reset SLA-List if no service is selected
        if ( !defined $ServiceID || !$Services->{$ServiceID} ) {
            $ServiceID = '';
        }
        my $SLAs = $Self->_GetSLAs(
            %GetParam,
            CustomerUserID => $CustomerUser,
            QueueID        => $QueueID,
            StateID        => $StateID,
            ServiceID      => $ServiceID,
        );
        my $NextStates = $Self->_GetNextStates(
            %GetParam,
            CustomerUserID => $CustomerUser || '',
            QueueID        => $QueueID,
            StateID        => $StateID,
        );

        # update Dynamic Fields Possible Values via AJAX
        my @DynamicFieldAJAX;

        # cycle trough the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( @{$DynamicField} ) {
            next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);

            my $IsACLReducible = $DynamicFieldBackendObject->HasBehavior(
                DynamicFieldConfig => $DynamicFieldConfig,
                Behavior           => 'IsACLReducible',
            );
            next DYNAMICFIELD if !$IsACLReducible;

            my $PossibleValues = $DynamicFieldBackendObject->PossibleValuesGet(
                DynamicFieldConfig => $DynamicFieldConfig,
            );

            # convert possible values key => value to key => key for ACLs using a Hash slice
            my %AclData = %{$PossibleValues};
            @AclData{ keys %AclData } = keys %AclData;

            # set possible values filter from ACLs
            my $ACL = $TicketObject->TicketAcl(
                %GetParam,
                Action        => $Self->{Action},
                TicketID      => $Self->{TicketID},
                QueueID       => $QueueID,
                ReturnType    => 'Ticket',
                ReturnSubType => 'DynamicField_' . $DynamicFieldConfig->{Name},
                Data          => \%AclData,
                UserID        => $Self->{UserID},
            );
            if ($ACL) {
                my %Filter = $TicketObject->TicketAclData();

                # convert Filer key => key back to key => value using map
                %{$PossibleValues} = map { $_ => $PossibleValues->{$_} } keys %Filter;
            }

            my $DataValues = $DynamicFieldBackendObject->BuildSelectionDataGet(
                DynamicFieldConfig => $DynamicFieldConfig,
                PossibleValues     => $PossibleValues,
                Value              => $DynamicFieldValues{ $DynamicFieldConfig->{Name} },
            ) || $PossibleValues;

            # add dynamic field to the list of fields to update
            push(
                @DynamicFieldAJAX,
                {
                    Name        => 'DynamicField_' . $DynamicFieldConfig->{Name},
                    Data        => $DataValues,
                    SelectedID  => $DynamicFieldValues{ $DynamicFieldConfig->{Name} },
                    Translation => $DynamicFieldConfig->{Config}->{TranslatableValues} || 0,
                    Max         => 100,
                }
            );
        }

        my $StandardTemplates = $Self->_GetStandardTemplates(
            %GetParam,
            QueueID => $QueueID || '',
        );

        my @TemplateAJAX;

        # update ticket body and attachements if needed.
        if ( $ElementChanged eq 'StandardTemplateID' ) {
            my @TicketAttachments;
            my $TemplateText;

            # remove all attachments from the Upload cache
            my $RemoveSuccess = $UploadCacheObject->FormIDRemove(
                FormID => $Self->{FormID},
            );
            if ( !$RemoveSuccess ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Form attachments could not be deleted!",
                );
            }

            # get the template text and set new attachments if a template is selected
            if ( IsPositiveInteger( $GetParam{StandardTemplateID} ) ) {
                my $TemplateGenerator = $Kernel::OM->Get('Kernel::System::TemplateGenerator');

                # set template text, replace smart tags (limited as ticket is not created)
                $TemplateText = $TemplateGenerator->Template(
                    TemplateID => $GetParam{StandardTemplateID},
                    TicketID   => $Self->{TicketID},
                    UserID     => $Self->{UserID},
                );

                # if ReplyToArticle is given, get this article to generate
                # the quoted article content
                if ( $Self->{ReplyToArticle} ) {

                    # get article to quote
                    my $Body = $LayoutObject->ArticleQuote(
                        TicketID          => $Self->{TicketID},
                        ArticleID         => $Self->{ReplyToArticle},
                        FormID            => $Self->{FormID},
                        UploadCacheObject => $UploadCacheObject,
                    );

                    # prepare quoted body content
                    $Body = $Self->_GetQuotedReplyBody(
                        %{ $Self->{ReplyToArticleContent} },
                        Body => $Body,
                    );

                    if ( $LayoutObject->{BrowserRichText} ) {
                        $TemplateText = $TemplateText . '<br><br>' . $Body;
                    }
                    else {
                        $TemplateText = $TemplateText . "\n\n" . $Body;
                    }
                }

                # create StdAttachmentObject
                my $StdAttachmentObject = $Kernel::OM->Get('Kernel::System::StdAttachment');

                # add std. attachments to ticket
                my %AllStdAttachments = $StdAttachmentObject->StdAttachmentStandardTemplateMemberList(
                    StandardTemplateID => $GetParam{StandardTemplateID},
                );
                for ( sort keys %AllStdAttachments ) {
                    my %AttachmentsData = $StdAttachmentObject->StdAttachmentGet( ID => $_ );
                    $UploadCacheObject->FormIDAddFile(
                        FormID      => $Self->{FormID},
                        Disposition => 'attachment',
                        %AttachmentsData,
                    );
                }

                # send a list of attachments in the upload cache back to the clientside JavaScript
                # which renders then the list of currently uploaded attachments
                @TicketAttachments = $UploadCacheObject->FormIDGetAllFilesMeta(
                    FormID => $Self->{FormID},
                );

                for my $Attachment (@TicketAttachments) {
                    $Attachment->{Filesize} = $LayoutObject->HumanReadableDataSize(
                        Size => $Attachment->{Filesize},
                    );
                }
            }

            @TemplateAJAX = (
                {
                    Name => 'UseTemplateNote',
                    Data => '0',
                },
                {
                    Name => 'RichText',
                    Data => $TemplateText || '',
                },
                {
                    Name     => 'TicketAttachments',
                    Data     => \@TicketAttachments,
                    KeepData => 1,
                },
            );
        }

        my $JSON = $LayoutObject->BuildSelectionJSON(
            [

                {
                    Name         => 'NewOwnerID',
                    Data         => $Owners,
                    SelectedID   => $GetParam{NewOwnerID},
                    Translation  => 0,
                    PossibleNone => 1,
                    Max          => 100,
                },
                {
                    Name         => 'NewResponsibleID',
                    Data         => $ResponsibleUsers,
                    SelectedID   => $GetParam{NewResponsibleID},
                    Translation  => 0,
                    PossibleNone => 1,
                    Max          => 100,
                },
                {
                    Name         => 'NewStateID',
                    Data         => $NextStates,
                    SelectedID   => $GetParam{NewStateID},
                    Translation  => 1,
                    PossibleNone => $Config->{StateDefault} ? 0 : 1,
                    Max          => 100,
                },
                {
                    Name         => 'NewPriorityID',
                    Data         => $Priorities,
                    SelectedID   => $GetParam{NewPriorityID},
                    PossibleNone => 0,
                    Translation  => 1,
                    Max          => 100,
                },
                {
                    Name         => 'ServiceID',
                    Data         => $Services,
                    SelectedID   => $GetParam{ServiceID},
                    PossibleNone => 1,
                    Translation  => 0,
                    TreeView     => $TreeView,
                    Max          => 100,
                },
                {
                    Name         => 'SLAID',
                    Data         => $SLAs,
                    SelectedID   => $GetParam{SLAID},
                    PossibleNone => 1,
                    Translation  => 0,
                    Max          => 100,
                },
                {
                    Name         => 'StandardTemplateID',
                    Data         => $StandardTemplates,
                    SelectedID   => $GetParam{StandardTemplateID},
                    PossibleNone => 1,
                    Translation  => 1,
                    Max          => 100,
                },
                {
                    Name         => 'TypeID',
                    Data         => $Types,
                    SelectedID   => $GetParam{TypeID},
                    PossibleNone => 1,
                    Translation  => 0,
                    Max          => 100,
                },
                {
                    Name         => 'NewQueueID',
                    Data         => $NewQueues,
                    SelectedID   => $GetParam{NewQueueID},
                    PossibleNone => 1,
                    Translation  => 0,
                    TreeView     => $TreeView,
                    Max          => 100,
                },
                @DynamicFieldAJAX,
                @TemplateAJAX,
            ],
        );
        return $LayoutObject->Attachment(
            ContentType => 'application/json; charset=' . $LayoutObject->{Charset},
            Content     => $JSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }
    else {

        my $Body = '';

        # if ReplyToArticle is given, get this article to generate
        # the quoted article content
        if ( $Self->{ReplyToArticle} ) {

            # get article to quote
            $Body = $LayoutObject->ArticleQuote(
                TicketID          => $Self->{TicketID},
                ArticleID         => $Self->{ReplyToArticle},
                FormID            => $Self->{FormID},
                UploadCacheObject => $UploadCacheObject,
            );

            # prepare quoted body content
            $Body = $Self->_GetQuotedReplyBody(
                %{ $Self->{ReplyToArticleContent} },
                Body => $Body,
            );
        }

        # if a body content was pre defined, add this before the quoted article content
        if ( $GetParam{Body} ) {

            # make sure body is rich text
            if ( $LayoutObject->{BrowserRichText} ) {
                $GetParam{Body} = $LayoutObject->Ascii2RichText(
                    String => $GetParam{Body},
                );
            }

            $Body = $GetParam{Body} . $Body;
        }

        # fillup configured default vars
        if ( $Body eq '' && $Config->{Body} ) {
            $Body = $LayoutObject->Output(
                Template => $Config->{Body},
            );

            # make sure body is rich text
            if ( $LayoutObject->{BrowserRichText} ) {
                $Body = $LayoutObject->Ascii2RichText(
                    String => $Body,
                );
            }
        }

        # set Body var to calculated content
        $GetParam{Body} = $Body;

        my %SafetyCheckResult = $Kernel::OM->Get('Kernel::System::HTMLUtils')->Safety(
            String => $GetParam{Body},

            # Strip out external content if BlockLoadingRemoteContent is enabled.
            NoExtSrcLoad => $ConfigObject->Get('Ticket::Frontend::BlockLoadingRemoteContent'),

            # Disallow potentially unsafe content.
            NoApplet     => 1,
            NoObject     => 1,
            NoEmbed      => 1,
            NoSVG        => 1,
            NoJavaScript => 1,
        );
        $GetParam{Body} = $SafetyCheckResult{String};

        if ( $Self->{ReplyToArticle} ) {
            my $TicketSubjectRe = $ConfigObject->Get('Ticket::SubjectRe') || 'Re';
            $GetParam{Subject} = $TicketSubjectRe . ': ' . $Self->{ReplyToArticleContent}{Subject};
        }
        elsif ( !defined $GetParam{Subject} && $Config->{Subject} ) {
            $GetParam{Subject} = $LayoutObject->Output(
                Template => $Config->{Subject},
            );
        }

        my @TicketTypeDynamicFields;
        my @ArticleTypeDynamicFields;

        # cycle trough the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( @{$DynamicField} ) {
            next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);
# ---
# OTRSMasterSlave
# ---
            if (
                !$MasterSlaveAdvancedEnabled
                && $DynamicFieldConfig->{Name} eq $MasterSlaveDynamicField
                )
            {
                next DYNAMICFIELD;
            }
# ---

            my $PossibleValuesFilter;

            my $IsACLReducible = $DynamicFieldBackendObject->HasBehavior(
                DynamicFieldConfig => $DynamicFieldConfig,
                Behavior           => 'IsACLReducible',
            );

            if ($IsACLReducible) {

                # get PossibleValues
                my $PossibleValues = $DynamicFieldBackendObject->PossibleValuesGet(
                    DynamicFieldConfig => $DynamicFieldConfig,
                );

                # check if field has PossibleValues property in its configuration
                if ( IsHashRefWithData($PossibleValues) ) {

                    # convert possible values key => value to key => key for ACLs using a Hash slice
                    my %AclData = %{$PossibleValues};
                    @AclData{ keys %AclData } = keys %AclData;

                    # set possible values filter from ACLs
                    my $ACL = $TicketObject->TicketAcl(
                        %GetParam,
                        Action        => $Self->{Action},
                        TicketID      => $Self->{TicketID},
                        ReturnType    => 'Ticket',
                        ReturnSubType => 'DynamicField_' . $DynamicFieldConfig->{Name},
                        Data          => \%AclData,
                        UserID        => $Self->{UserID},
                    );
                    if ($ACL) {
                        my %Filter = $TicketObject->TicketAclData();

                        # convert Filer key => key back to key => value using map
                        %{$PossibleValuesFilter} = map { $_ => $PossibleValues->{$_} }
                            keys %Filter;
                    }
                }
            }
# ---
# OTRSMasterSlave
# ---
            if ( $DynamicFieldConfig->{Name} eq $MasterSlaveDynamicField ) {
               $PossibleValuesFilter =  $Self->_GetMasterSlaveData(
                Ticket => \%Ticket,
                MasterSlaveDynamicField => $MasterSlaveDynamicField,
               );
            }
# ---

            # to store dynamic field value from database (or undefined)
            my $Value;

            if ( $DynamicFieldConfig->{ObjectType} eq 'Ticket' ) {

                # Only get values for Ticket fields (all screens based on AgentTickeActionCommon
                # generates a new article, then article fields will be always empty at the beginning).
                # Value is stored in the database from Ticket.
                $Value = $Ticket{ 'DynamicField_' . $DynamicFieldConfig->{Name} };

                # Get field html.
                my $DynamicFieldHTML = $DynamicFieldBackendObject->EditFieldRender(
                    DynamicFieldConfig   => $DynamicFieldConfig,
                    PossibleValuesFilter => $PossibleValuesFilter,
                    Value                => $Value,
                    Mandatory            => $Config->{DynamicField}->{ $DynamicFieldConfig->{Name} } == 2,
                    LayoutObject         => $LayoutObject,
                    ParamObject          => $ParamObject,
                    AJAXUpdate           => 1,
                    UpdatableFields      => $Self->_GetFieldsToUpdate(),
                );

                push @TicketTypeDynamicFields, {
                    Name  => $DynamicFieldConfig->{Name},
                    Label => $DynamicFieldHTML->{Label},
                    Field => $DynamicFieldHTML->{Field},
                };
            }
            elsif ( $DynamicFieldConfig->{ObjectType} eq 'Article' ) {
                my $Class            = '';
                my $MandatoryTooltip = 0;

                if ( $Config->{DynamicField}->{ $DynamicFieldConfig->{Name} } == 2 ) {
                    if (
                        $Config->{NoteMandatory} ||
                        $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
                        )
                    {
                        $Class = 'Validate_Required';
                    }
                    else {
                        $Class            = 'Validate_DependingRequiredAND Validate_Depending_CreateArticle';
                        $MandatoryTooltip = 1;
                    }
                }

                # Get field html.
                my $DynamicFieldHTML = $DynamicFieldBackendObject->EditFieldRender(
                    DynamicFieldConfig   => $DynamicFieldConfig,
                    PossibleValuesFilter => $PossibleValuesFilter,
                    Value                => $Value,
                    Mandatory            => ( $Class eq 'Validate_Required' ) ? 1 : 0,
                    Class                => $Class,
                    LayoutObject         => $LayoutObject,
                    ParamObject          => $ParamObject,
                    AJAXUpdate           => 1,
                    UpdatableFields      => $Self->_GetFieldsToUpdate(),
                );

                push @ArticleTypeDynamicFields, {
                    Name             => $DynamicFieldConfig->{Name},
                    Label            => $DynamicFieldHTML->{Label},
                    Field            => $DynamicFieldHTML->{Field},
                    MandatoryTooltip => $MandatoryTooltip,
                };
            }
        }

        # print form ...
        my $Output = $LayoutObject->Header(
            Type      => 'Small',
            Value     => $Ticket{TicketNumber},
            BodyClass => 'Popup',
        );
        $Output .= $Self->_Mask(
            TimeUnitsRequired => (
                $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
                ? 'Validate_Required'
                : ''
            ),
            TicketTypeDynamicFields  => \@TicketTypeDynamicFields,
            ArticleTypeDynamicFields => \@ArticleTypeDynamicFields,
            %GetParam,
            %Ticket,
        );
        $Output .= $LayoutObject->Footer(
            Type => 'Small',
        );
        return $Output;
    }
}

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

    # get list type
    my $TreeView = 0;

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    if ( $ConfigObject->Get('Ticket::Frontend::ListType') eq 'tree' ) {
        $TreeView = 1;
    }

    # get needed objects
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    my %Ticket = $TicketObject->TicketGet( TicketID => $Self->{TicketID} );

    # get config of frontend module
    my $Config = $ConfigObject->Get("Ticket::Frontend::$Self->{Action}");

    # get layout object
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    # Define the dynamic fields to show based on the object type.
    my $ObjectType = ['Ticket'];

    # Only screens that add notes can modify Article dynamic fields.
    if ( $Config->{Note} ) {
        $ObjectType = [ 'Ticket', 'Article' ];
    }

    # Get dynamic fields for this screen.
    my $DynamicField = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        Valid       => 1,
        ObjectType  => $ObjectType,
        FieldFilter => $Config->{DynamicField} || {},
    );

    # Widget Ticket Actions
    if (
        ( $ConfigObject->Get('Ticket::Type') && $Config->{TicketType} )
        ||
        ( $ConfigObject->Get('Ticket::Service')     && $Config->{Service} )     ||
        ( $ConfigObject->Get('Ticket::Responsible') && $Config->{Responsible} ) ||
        $Config->{Title}    ||
        $Config->{Queue}    ||
        $Config->{Owner}    ||
        $Config->{State}    ||
        $Config->{Priority} ||
        scalar @{ $Param{TicketTypeDynamicFields} } > 0
        )
    {
        $LayoutObject->Block(
            Name => 'WidgetTicketActions',
        );
    }

    if ( $Config->{Title} ) {
        $LayoutObject->Block(
            Name => 'Title',
            Data => \%Param,
        );
    }

    my $DynamicFieldNames = $Self->_GetFieldsToUpdate(
        OnlyDynamicFields => 1,
    );

    # send data to JS
    $LayoutObject->AddJSData(
        Key   => 'DynamicFieldNames',
        Value => $DynamicFieldNames,
    );

    # types
    if ( $ConfigObject->Get('Ticket::Type') && $Config->{TicketType} ) {
        my %Type = $TicketObject->TicketTypeList(
            %Param,
            Action => $Self->{Action},
            UserID => $Self->{UserID},
        );
        $Param{TypeStrg} = $LayoutObject->BuildSelection(
            Class        => 'Validate_Required Modernize ' . ( $Param{Errors}->{TypeIDInvalid} || '' ),
            Data         => \%Type,
            Name         => 'TypeID',
            SelectedID   => $Param{TypeID},
            PossibleNone => 1,
            Sort         => 'AlphanumericValue',
            Translation  => 0,
        );
        $LayoutObject->Block(
            Name => 'Type',
            Data => {%Param},
        );
    }

    # services
    if ( $ConfigObject->Get('Ticket::Service') && $Config->{Service} ) {
        my $Services = $Self->_GetServices(
            %Param,
            Action         => $Self->{Action},
            CustomerUserID => $Ticket{CustomerUserID},
            UserID         => $Self->{UserID},
        );

        # reset previous ServiceID to reset SLA-List if no service is selected
        if ( !$Param{ServiceID} || !$Services->{ $Param{ServiceID} } ) {
            $Param{ServiceID} = '';
        }

        $Param{ServiceStrg} = $LayoutObject->BuildSelection(
            Data       => $Services,
            Name       => 'ServiceID',
            SelectedID => $Param{ServiceID},
            Class      => "Modernize "
                . ( $Config->{ServiceMandatory} ? 'Validate_Required ' : '' )
                . ( $Param{ServiceInvalid} || '' ),
            PossibleNone => 1,
            TreeView     => $TreeView,
            Sort         => 'TreeView',
            Translation  => 0,
            Max          => 200,
        );

        $LayoutObject->Block(
            Name => 'Service',
            Data => {
                ServiceMandatory => $Config->{ServiceMandatory} || 0,
                %Param,
            },
        );

        my %SLA = $TicketObject->TicketSLAList(
            %Param,
            Action => $Self->{Action},
            UserID => $Self->{UserID},
        );

        $Param{SLAStrg} = $LayoutObject->BuildSelection(
            Data       => \%SLA,
            Name       => 'SLAID',
            SelectedID => $Param{SLAID},
            Class      => "Modernize "
                . ( $Config->{SLAMandatory} ? 'Validate_Required ' : '' )
                . ( $Param{ServiceInvalid} || '' ),
            PossibleNone => 1,
            Sort         => 'AlphanumericValue',
            Translation  => 0,
            Max          => 200,
        );

        $LayoutObject->Block(
            Name => 'SLA',
            Data => {
                SLAMandatory => $Config->{SLAMandatory},
                %Param,
            },
        );
    }

    if ( $Config->{Queue} ) {

        # fetch all queues
        my %MoveQueues = $TicketObject->TicketMoveList(
            TicketID => $Self->{TicketID},
            UserID   => $Self->{UserID},
            Action   => $Self->{Action},
            Type     => 'move_into',
        );

        # set move queues
        $Param{QueuesStrg} = $LayoutObject->AgentQueueListOption(
            Data     => { %MoveQueues, '' => '-' },
            Multiple => 0,
            Size     => 0,
            Class    => 'NewQueueID Modernize '
                . ( $Config->{QueueMandatory} ? 'Validate_Required ' : '' )
                . ( $Param{NewQueueInvalid} || '' ),
            Name           => 'NewQueueID',
            SelectedID     => $Param{NewQueueID},
            TreeView       => $TreeView,
            CurrentQueueID => $Param{QueueID},
            OnChangeSubmit => 0,
        );

        $LayoutObject->Block(
            Name => 'Queue',
            Data => {
                QueueMandatory => $Config->{QueueMandatory} || 0,
                %Param
            },
        );
    }

    # get needed objects
    my $QueueObject = $Kernel::OM->Get('Kernel::System::Queue');
    my $UserObject  = $Kernel::OM->Get('Kernel::System::User');
    my $GroupObject = $Kernel::OM->Get('Kernel::System::Group');

    if ( $Config->{Owner} ) {

        # get user of own groups
        my %ShownUsers;
        my %AllGroupsMembers = $UserObject->UserList(
            Type  => 'Long',
            Valid => 1,
        );
        if ( $ConfigObject->Get('Ticket::ChangeOwnerToEveryone') ) {
            %ShownUsers = %AllGroupsMembers;
        }
        else {
            my $GID        = $QueueObject->GetQueueGroupID( QueueID => $Ticket{QueueID} );
            my %MemberList = $GroupObject->PermissionGroupGet(
                GroupID => $GID,
                Type    => 'owner',
            );
            for my $UserID ( sort keys %MemberList ) {
                $ShownUsers{$UserID} = $AllGroupsMembers{$UserID};
            }
        }

        my $ACL = $TicketObject->TicketAcl(
            %Ticket,
            Action        => $Self->{Action},
            ReturnType    => 'Ticket',
            ReturnSubType => 'NewOwner',
            Data          => \%ShownUsers,
            UserID        => $Self->{UserID},
        );

        if ($ACL) {
            %ShownUsers = $TicketObject->TicketAclData();
        }

        # get old owner
        my @OldUserInfo = $TicketObject->TicketOwnerList( TicketID => $Self->{TicketID} );
        my @OldOwners;
        my %OldOwnersShown;
        my %SeenOldOwner;
        if (@OldUserInfo) {
            my $Counter = 1;
            USER:
            for my $User ( reverse @OldUserInfo ) {

                # skip if old owner is already in the list
                next USER if $SeenOldOwner{ $User->{UserID} };
                $SeenOldOwner{ $User->{UserID} } = 1;
                my $Key   = $User->{UserID};
                my $Value = "$Counter: $User->{UserFullname}";
                push @OldOwners, {
                    Key   => $Key,
                    Value => $Value,
                };
                $OldOwnersShown{$Key} = $Value;
                $Counter++;
            }
        }

        my $OldOwnerSelectedID = '';
        if ( $Param{OldOwnerID} ) {
            $OldOwnerSelectedID = $Param{OldOwnerID};
        }
        elsif ( $OldUserInfo[0]->{UserID} ) {
            $OldOwnerSelectedID = $OldUserInfo[0]->{UserID} . '1';
        }

        my $OldOwnerACL = $TicketObject->TicketAcl(
            %Ticket,
            Action        => $Self->{Action},
            ReturnType    => 'Ticket',
            ReturnSubType => 'OldOwner',
            Data          => \%OldOwnersShown,
            UserID        => $Self->{UserID},
        );

        if ($OldOwnerACL) {
            %OldOwnersShown = $TicketObject->TicketAclData();
        }

        # build string
        $Param{OwnerStrg} = $LayoutObject->BuildSelection(
            Data       => \%ShownUsers,
            SelectedID => $Param{NewOwnerID},
            Name       => 'NewOwnerID',
            Class      => 'Modernize '
                . ( $Config->{OwnerMandatory} ? 'Validate_Required ' : '' )
                . ( $Param{NewOwnerInvalid} || '' ),
            Size         => 1,
            PossibleNone => 1,
            Filters      => {
                OldOwners => {
                    Name   => $LayoutObject->{LanguageObject}->Translate('Previous Owner'),
                    Values => \%OldOwnersShown,
                },
            },
        );

        $LayoutObject->Block(
            Name => 'Owner',
            Data => {
                OwnerMandatory => $Config->{OwnerMandatory} || 0,
                %Param,
            },
        );
    }

    if ( $ConfigObject->Get('Ticket::Responsible') && $Config->{Responsible} ) {

        # get user of own groups
        my %ShownUsers;
        my %AllGroupsMembers = $UserObject->UserList(
            Type  => 'Long',
            Valid => 1,
        );
        if ( $ConfigObject->Get('Ticket::ChangeOwnerToEveryone') ) {
            %ShownUsers = %AllGroupsMembers;
        }
        else {
            my $GID        = $QueueObject->GetQueueGroupID( QueueID => $Ticket{QueueID} );
            my %MemberList = $GroupObject->PermissionGroupGet(
                GroupID => $GID,
                Type    => 'responsible',
            );
            for my $UserID ( sort keys %MemberList ) {
                $ShownUsers{$UserID} = $AllGroupsMembers{$UserID};
            }
        }

        my $ACL = $TicketObject->TicketAcl(
            %Ticket,
            Action        => $Self->{Action},
            ReturnType    => 'Ticket',
            ReturnSubType => 'Responsible',
            Data          => \%ShownUsers,
            UserID        => $Self->{UserID},
        );

        if ($ACL) {
            %ShownUsers = $TicketObject->TicketAclData();
        }

        # get responsible
        $Param{ResponsibleStrg} = $LayoutObject->BuildSelection(
            Data       => \%ShownUsers,
            SelectedID => $Param{NewResponsibleID},
            Name       => 'NewResponsibleID',
            Class      => 'Modernize '
                . ( $Config->{ResponsibleMandatory} ? 'Validate_Required ' : '' )
                . ( $Param{NewResponsibleInvalid} || '' ),
            PossibleNone => 1,
            Size         => 1,
        );
        $LayoutObject->Block(
            Name => 'Responsible',
            Data => {
                ResponsibleMandatory => $Config->{ResponsibleMandatory} || 0,
                %Param,
            },
        );

    }

    if ( $Config->{State} ) {

        my %State;
        my %StateList = $TicketObject->TicketStateList(
            Action   => $Self->{Action},
            TicketID => $Self->{TicketID},
            UserID   => $Self->{UserID},
        );
        if ( !$Param{NewStateID} ) {
            if ( $Config->{StateDefault} ) {
                $State{SelectedValue} = $Config->{StateDefault};
            }
        }
        else {
            $State{SelectedID} = $Param{NewStateID};
        }

        # build next states string
        $Param{StateStrg} = $LayoutObject->BuildSelection(
            Data  => \%StateList,
            Name  => 'NewStateID',
            Class => 'Modernize '
                . ( $Config->{StateMandatory} ? 'Validate_Required ' : '' )
                . ( $Param{NewStateInvalid} || '' ),
            PossibleNone => $Config->{StateDefault} ? 0 : 1,
            %State,
        );
        $LayoutObject->Block(
            Name => 'State',
            Data => {
                StateMandatory => $Config->{StateMandatory} || 0,
                %Param,
            },
        );

        if ( IsArrayRefWithData( $Config->{StateType} ) ) {

            STATETYPE:
            for my $StateType ( @{ $Config->{StateType} } ) {

                next STATETYPE if !$StateType;
                next STATETYPE if $StateType !~ /pending/i;

                # get used calendar
                my $Calendar = $TicketObject->TicketCalendarGet(
                    %Ticket,
                );

                $Param{DateString} = $LayoutObject->BuildDateSelection(
                    %Param,
                    Format           => 'DateInputFormatLong',
                    YearPeriodPast   => 0,
                    YearPeriodFuture => 5,
                    DiffTime         => $ConfigObject->Get('Ticket::Frontend::PendingDiffTime')
                        || 0,
                    Class                => $Param{DateInvalid} || ' ',
                    Validate             => 1,
                    ValidateDateInFuture => 1,
                    Calendar             => $Calendar,
                );

                $LayoutObject->Block(
                    Name => 'StatePending',
                    Data => \%Param,
                );

                last STATETYPE;
            }
        }
    }

    # get priority
    if ( $Config->{Priority} ) {

        my %Priority;
        my %PriorityList = $TicketObject->TicketPriorityList(
            UserID   => $Self->{UserID},
            TicketID => $Self->{TicketID},
            Action   => $Self->{Action},
        );
        if ( !$Config->{PriorityDefault} ) {
            $PriorityList{''} = '-';
        }
        if ( !$Param{NewPriorityID} ) {
            if ( $Config->{PriorityDefault} ) {
                $Priority{SelectedValue} = $Config->{PriorityDefault};
            }
        }
        else {
            $Priority{SelectedID} = $Param{NewPriorityID};
        }
        $Priority{SelectedID} ||= $Param{PriorityID};
        $Param{PriorityStrg} = $LayoutObject->BuildSelection(
            Data  => \%PriorityList,
            Name  => 'NewPriorityID',
            Class => 'Modernize',
            %Priority,
        );
        $LayoutObject->Block(
            Name => 'Priority',
            Data => \%Param,
        );
    }

    # Get Ticket type dynamic fields.
    for my $TicketTypeDynamicField ( @{ $Param{TicketTypeDynamicFields} } ) {
        $LayoutObject->Block(
            Name => 'TicketTypeDynamicField',
            Data => $TicketTypeDynamicField,
        );

        # Output customization block too, if it exists.
        $LayoutObject->Block(
            Name => 'TicketTypeDynamicField_' . $TicketTypeDynamicField->{Name},
            Data => $TicketTypeDynamicField,
        );
    }

    # End Widget Ticket Actions

    # Widget Article
    if ( $Config->{Note} ) {

        $Param{WidgetStatus} = 'Collapsed';

        if (
            $Config->{NoteMandatory}
            || $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
            || $Self->{ReplyToArticle}
            || $Param{CreateArticle}
            )
        {
            $Param{WidgetStatus} = 'Expanded';
        }

        if (
            $Config->{NoteMandatory}
            || $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
            )
        {
            $Param{SubjectRequired} = 'Validate_Required';
            $Param{BodyRequired}    = 'Validate_Required';
        }
        else {
            $Param{SubjectRequired} = 'Validate_DependingRequiredAND Validate_Depending_CreateArticle';
            $Param{BodyRequired}    = 'Validate_DependingRequiredAND Validate_Depending_CreateArticle';
        }

        # set customer visibility of this note to the same value as the article for whom this is the reply
        if ( $Self->{ReplyToArticle} && !defined $Param{IsVisibleForCustomer} ) {
            $Param{IsVisibleForCustomer} = $Self->{ReplyToArticleContent}->{IsVisibleForCustomer};
        }
        elsif ( !defined $Param{IsVisibleForCustomer} ) {
            $Param{IsVisibleForCustomer} = $Config->{IsVisibleForCustomerDefault};
        }

        # show attachments
        ATTACHMENT:
        for my $Attachment ( @{ $Param{Attachments} } ) {
            if (
                $Attachment->{ContentID}
                && $LayoutObject->{BrowserRichText}
                && ( $Attachment->{ContentType} =~ /image/i )
                && ( $Attachment->{Disposition} eq 'inline' )
                )
            {
                next ATTACHMENT;
            }

            push @{ $Param{AttachmentList} }, $Attachment;
        }

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

        # get all user ids of agents, that can be shown in this dialog
        # based on queue rights
        my %ShownUsers;
        my %AllGroupsMembers = $UserObject->UserList(
            Type  => 'Long',
            Valid => 1,
        );
        my $GID        = $QueueObject->GetQueueGroupID( QueueID => $Ticket{QueueID} );
        my %MemberList = $GroupObject->PermissionGroupGet(
            GroupID => $GID,
            Type    => 'note',
        );
        for my $UserID ( sort keys %MemberList ) {
            $ShownUsers{$UserID} = $AllGroupsMembers{$UserID};
        }

        # create email parser object
        my $EmailParserObject = Kernel::System::EmailParser->new(
            Mode  => 'Standalone',
            Debug => 0,
        );

        # check and retrieve involved and informed agents of ReplyTo Note
        my @ReplyToUsers;
        my %ReplyToUsersHash;
        my %ReplyToUserIDs;
        if ( $Self->{ReplyToArticle} ) {
            my @ReplyToParts = $EmailParserObject->SplitAddressLine(
                Line => $Self->{ReplyToArticleContent}->{To} || '',
            );

            REPLYTOPART:
            for my $SingleReplyToPart (@ReplyToParts) {
                my $ReplyToAddress = $EmailParserObject->GetEmailAddress(
                    Email => $SingleReplyToPart,
                );

                next REPLYTOPART if !$ReplyToAddress;
                push @ReplyToUsers, $ReplyToAddress;
            }

            $ReplyToUsersHash{$_}++ for @ReplyToUsers;

            # get user ids of available users
            for my $UserID ( sort keys %ShownUsers ) {
                my %UserData = $UserObject->GetUserData(
                    UserID => $UserID,
                );

                my $UserEmail = $UserData{UserEmail};
                if ( $ReplyToUsersHash{$UserEmail} ) {
                    $ReplyToUserIDs{$UserID} = 1;
                }
            }

            # add original note sender to list of user ids
            for my $UserID ( sort @{ $Self->{ReplyToSenderUserID} } ) {

                # if sender replies to himself, do not include sender in list
                if ( $UserID ne $Self->{UserID} ) {
                    $ReplyToUserIDs{$UserID} = 1;
                }
            }

            # remove user id of active user
            delete $ReplyToUserIDs{ $Self->{UserID} };
        }

        if ( $Config->{InformAgent} || $Config->{InvolvedAgent} ) {
            $LayoutObject->Block(
                Name => 'InformAdditionalAgents',
            );
        }

        # get param object
        my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

        # get all agents for "involved agents"
        if ( $Config->{InvolvedAgent} ) {

            my @UserIDs = $TicketObject->TicketInvolvedAgentsList(
                TicketID => $Self->{TicketID},
            );

            my @InvolvedAgents;
            my $Counter = 1;

            my @InvolvedUserID = $ParamObject->GetArray( Param => 'InvolvedUserID' );

            my %AgentWithPermission = $GroupObject->PermissionGroupGet(
                GroupID => $GID,
                Type    => 'ro',
            );

            USER:
            for my $User ( reverse @UserIDs ) {

                next USER if !defined $AgentWithPermission{ $User->{UserID} };

                my $Value = "$Counter: $User->{UserFullname}";
                if ( $User->{OutOfOfficeMessage} ) {
                    $Value .= " $User->{OutOfOfficeMessage}";
                }

                push @InvolvedAgents, {
                    Key   => $User->{UserID},
                    Value => $Value,
                };
                $Counter++;

                # add involved user as selected entries, if available in ReplyToAddresses list
                if ( $Self->{ReplyToArticle} && $ReplyToUserIDs{ $User->{UserID} } ) {
                    push @InvolvedUserID, $User->{UserID};
                    delete $ReplyToUserIDs{ $User->{UserID} };
                }
            }

            my $InvolvedAgentSize = $ConfigObject->Get('Ticket::Frontend::InvolvedAgentMaxSize') || 3;
            $Param{InvolvedAgentStrg} = $LayoutObject->BuildSelection(
                Data       => \@InvolvedAgents,
                SelectedID => \@InvolvedUserID,
                Name       => 'InvolvedUserID',
                Class      => 'Modernize',
                Multiple   => 1,
                Size       => $InvolvedAgentSize,
            );

            # block is called below "inform agents"
        }

        # agent list
        if ( $Config->{InformAgent} ) {

            # get inform user list
            my %InformAgents;
            my @InformUserID    = $ParamObject->GetArray( Param => 'InformUserID' );
            my %InformAgentList = $GroupObject->PermissionGroupGet(
                GroupID => $GID,
                Type    => 'ro',
            );
            for my $UserID ( sort keys %InformAgentList ) {
                $InformAgents{$UserID} = $AllGroupsMembers{$UserID};
            }

            if ( $Self->{ReplyToArticle} ) {

                # get email address of all users and compare to replyto-addresses
                for my $UserID ( sort keys %InformAgents ) {
                    if ( $ReplyToUserIDs{$UserID} ) {
                        push @InformUserID, $UserID;
                        delete $ReplyToUserIDs{$UserID};
                    }
                }
            }

            my $InformAgentSize = $ConfigObject->Get('Ticket::Frontend::InformAgentMaxSize')
                || 3;
            $Param{OptionStrg} = $LayoutObject->BuildSelection(
                Data       => \%InformAgents,
                SelectedID => \@InformUserID,
                Name       => 'InformUserID',
                Class      => 'Modernize',
                Multiple   => 1,
                Size       => $InformAgentSize,
            );
            $LayoutObject->Block(
                Name => 'InformAgent',
                Data => \%Param,
            );
        }

        # get involved
        if ( $Config->{InvolvedAgent} ) {

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

        # show list of agents, that receive this note (ReplyToNote)
        # at least sender of original note and all recepients of the original note
        # that couldn't be selected with involved/inform agents
        if ( $Self->{ReplyToArticle} ) {

            my $UsersHashSize = keys %ReplyToUserIDs;
            my $Counter       = 0;
            $Param{UserListWithoutSelection} = join( ',', keys %ReplyToUserIDs );

            if ( $UsersHashSize > 0 ) {
                $LayoutObject->Block(
                    Name => 'InformAgentsWithoutSelection',
                    Data => \%Param,
                );

                for my $UserID ( sort keys %ReplyToUserIDs ) {
                    $Counter++;

                    my %UserData = $UserObject->GetUserData(
                        UserID => $UserID,
                    );

                    $LayoutObject->Block(
                        Name => 'InformAgentsWithoutSelectionSingleUser',
                        Data => \%UserData,
                    );

                    # output a separator (InformAgentsWithoutSelectionSingleUserSeparator),
                    # if not last entry
                    if ( $Counter < $UsersHashSize ) {
                        $LayoutObject->Block(
                            Name => 'InformAgentsWithoutSelectionSingleUserSeparator',
                            Data => \%UserData,
                        );
                    }
                }
            }
        }

        # add rich text editor
        if ( $LayoutObject->{BrowserRichText} ) {

            # use height/width defined for this screen
            $Param{RichTextHeight} = $Config->{RichTextHeight} || 0;
            $Param{RichTextWidth}  = $Config->{RichTextWidth}  || 0;

            # set up rich text editor
            $LayoutObject->SetRichTextParameters(
                Data => \%Param,
            );
        }

        if (
            $Config->{NoteMandatory}
            || $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
            )
        {
            $LayoutObject->Block(
                Name => 'SubjectLabelMandatory',
            );
            $LayoutObject->Block(
                Name => 'RichTextLabelMandatory',
            );
        }
        else {
            $LayoutObject->Block(
                Name => 'SubjectLabel',
            );
            $LayoutObject->Block(
                Name => 'RichTextLabel',
            );
        }

        # build text template string
        my %StandardTemplates = $Kernel::OM->Get('Kernel::System::StandardTemplate')->StandardTemplateList(
            Valid => 1,
            Type  => 'Note',
        );

        my $QueueStandardTemplates = $Self->_GetStandardTemplates(
            %Param,
            TicketID => $Self->{TicketID} || '',
        );

        if (
            IsHashRefWithData(
                $QueueStandardTemplates
                    || ( $Config->{Queue} && IsHashRefWithData( \%StandardTemplates ) )
            )
            )
        {
            $Param{StandardTemplateStrg} = $LayoutObject->BuildSelection(
                Data         => $QueueStandardTemplates || {},
                Name         => 'StandardTemplateID',
                SelectedID   => $Param{StandardTemplateID} || '',
                Class        => 'Modernize',
                PossibleNone => 1,
                Sort         => 'AlphanumericValue',
                Translation  => 1,
                Max          => 200,
            );
            $LayoutObject->Block(
                Name => 'StandardTemplate',
                Data => {%Param},
            );
        }

        # show time accounting box
        if ( $ConfigObject->Get('Ticket::Frontend::AccountTime') ) {
            if ( $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime') ) {
                $LayoutObject->Block(
                    Name => 'TimeUnitsLabelMandatory',
                    Data => \%Param,
                );
            }
            else {
                $LayoutObject->Block(
                    Name => 'TimeUnitsLabel',
                    Data => \%Param,
                );
            }
            $LayoutObject->Block(
                Name => 'TimeUnits',
                Data => \%Param,
            );
        }

        # Get Article type dynamic fields.
        for my $ArticleTypeDynamicField ( @{ $Param{ArticleTypeDynamicFields} } ) {
            $LayoutObject->Block(
                Name => 'ArticleTypeDynamicField',
                Data => $ArticleTypeDynamicField,
            );

            if ( $ArticleTypeDynamicField->{MandatoryTooltip} ) {
                $LayoutObject->Block(
                    Name => 'ArticleTypeDynamicFieldError',
                    Data => $ArticleTypeDynamicField,
                );
            }

            # Output customization block too, if it exists.
            $LayoutObject->Block(
                Name => 'ArticleTypeDynamicField_' . $ArticleTypeDynamicField->{Name},
                Data => $ArticleTypeDynamicField,
            );
        }
    }

    # End Widget Article

    # get output back
    return $LayoutObject->Output(
        TemplateFile => $Self->{Action},
        Data         => \%Param
    );
}

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

    my %NextStates = $Kernel::OM->Get('Kernel::System::Ticket')->TicketStateList(
        TicketID => $Self->{TicketID},
        Action   => $Self->{Action},
        UserID   => $Self->{UserID},
        %Param,
    );

    return \%NextStates;
}

sub _GetResponsible {
    my ( $Self, %Param ) = @_;
    my %ShownUsers;
    my %AllGroupsMembers = $Kernel::OM->Get('Kernel::System::User')->UserList(
        Type  => 'Long',
        Valid => 1,
    );

    # show all users
    if ( $Kernel::OM->Get('Kernel::Config')->Get('Ticket::ChangeOwnerToEveryone') ) {
        %ShownUsers = %AllGroupsMembers;
    }

    # show only users with responsible or rw pemissions in the queue
    elsif ( $Param{QueueID} && !$Param{AllUsers} ) {
        my $GID = $Kernel::OM->Get('Kernel::System::Queue')->GetQueueGroupID(
            QueueID => $Param{NewQueueID} || $Param{QueueID}
        );
        my %MemberList = $Kernel::OM->Get('Kernel::System::Group')->PermissionGroupGet(
            GroupID => $GID,
            Type    => 'responsible',
        );
        for my $UserID ( sort keys %MemberList ) {
            $ShownUsers{$UserID} = $AllGroupsMembers{$UserID};
        }
    }

    # get ticket object
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    # workflow
    my $ACL = $TicketObject->TicketAcl(
        %Param,
        Action        => $Self->{Action},
        ReturnType    => 'Ticket',
        ReturnSubType => 'Responsible',
        Data          => \%ShownUsers,
        UserID        => $Self->{UserID},
    );

    return { $TicketObject->TicketAclData() } if $ACL;

    return \%ShownUsers;
}

sub _GetOwners {
    my ( $Self, %Param ) = @_;
    my %ShownUsers;
    my %AllGroupsMembers = $Kernel::OM->Get('Kernel::System::User')->UserList(
        Type  => 'Long',
        Valid => 1,
    );

    # show all users
    if ( $Kernel::OM->Get('Kernel::Config')->Get('Ticket::ChangeOwnerToEveryone') ) {
        %ShownUsers = %AllGroupsMembers;
    }

    # show only users with owner or rw pemissions in the queue
    elsif ( $Param{QueueID} && !$Param{AllUsers} ) {
        my $GID = $Kernel::OM->Get('Kernel::System::Queue')->GetQueueGroupID(
            QueueID => $Param{NewQueueID} || $Param{QueueID}
        );
        my %MemberList = $Kernel::OM->Get('Kernel::System::Group')->PermissionGroupGet(
            GroupID => $GID,
            Type    => 'owner',
        );
        for my $UserID ( sort keys %MemberList ) {
            $ShownUsers{$UserID} = $AllGroupsMembers{$UserID};
        }
    }

    # get ticket object
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    # workflow
    my $ACL = $TicketObject->TicketAcl(
        %Param,
        Action        => $Self->{Action},
        ReturnType    => 'Ticket',
        ReturnSubType => 'NewOwner',
        Data          => \%ShownUsers,
        UserID        => $Self->{UserID},
    );

    return { $TicketObject->TicketAclData() } if $ACL;

    return \%ShownUsers;
}

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

    # get ticket object
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    my @OldUserInfo = $TicketObject->TicketOwnerList( TicketID => $Self->{TicketID} );
    my %UserHash;
    if (@OldUserInfo) {
        my $Counter = 1;
        USER:
        for my $User ( reverse @OldUserInfo ) {

            next USER if $UserHash{ $User->{UserID} };

            $UserHash{ $User->{UserID} } = "$Counter: $User->{UserFullname}";
            $Counter++;
        }
    }

    # workflow
    my $ACL = $TicketObject->TicketAcl(
        %Param,
        Action        => $Self->{Action},
        ReturnType    => 'Ticket',
        ReturnSubType => 'OldOwner',
        Data          => \%UserHash,
        UserID        => $Self->{UserID},
    );

    return { $TicketObject->TicketAclData() } if $ACL;

    return \%UserHash;
}

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

    # get service
    my %Service;

    # get options for default services for unknown customers
    my $DefaultServiceUnknownCustomer
        = $Kernel::OM->Get('Kernel::Config')->Get('Ticket::Service::Default::UnknownCustomer');

    # check if no CustomerUserID is selected
    # if $DefaultServiceUnknownCustomer = 0 leave CustomerUserID empty, it will not get any services
    # if $DefaultServiceUnknownCustomer = 1 set CustomerUserID to get default services
    if ( !$Param{CustomerUserID} && $DefaultServiceUnknownCustomer ) {
        $Param{CustomerUserID} = '<DEFAULT>';
    }

    # get service list
    if ( $Param{CustomerUserID} ) {
        %Service = $Kernel::OM->Get('Kernel::System::Ticket')->TicketServiceList(
            %Param,
            TicketID => $Self->{TicketID},
            Action   => $Self->{Action},
            UserID   => $Self->{UserID},
        );
    }
    return \%Service;
}

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

    # if non set customers can get default services then they should also be able to get the SLAs
    #  for those services (this works during ticket creation).
    # if no CustomerUserID is set, TicketSLAList will complain during AJAX updates as UserID is not
    #  passed. See bug 11147.

    # get options for default services for unknown customers
    my $DefaultServiceUnknownCustomer
        = $Kernel::OM->Get('Kernel::Config')->Get('Ticket::Service::Default::UnknownCustomer');

    # check if no CustomerUserID is selected
    # if $DefaultServiceUnknownCustomer = 0 leave CustomerUserID empty, it will not get any services
    # if $DefaultServiceUnknownCustomer = 1 set CustomerUserID to get default services
    if ( !$Param{CustomerUserID} && $DefaultServiceUnknownCustomer ) {
        $Param{CustomerUserID} = '<DEFAULT>';
    }

    my %SLA;
    if ( $Param{ServiceID} ) {
        %SLA = $Kernel::OM->Get('Kernel::System::Ticket')->TicketSLAList(
            %Param,
            TicketID => $Self->{TicketID},
            Action   => $Self->{Action},
            UserID   => $Self->{UserID},
        );
    }
    return \%SLA;
}

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

    my %Priorities = $Kernel::OM->Get('Kernel::System::Ticket')->TicketPriorityList(
        %Param,
        Action   => $Self->{Action},
        UserID   => $Self->{UserID},
        TicketID => $Self->{TicketID},
    );

    # get config of frontend module
    my $Config = $Kernel::OM->Get('Kernel::Config')->Get("Ticket::Frontend::$Self->{Action}");

    if ( !$Config->{PriorityDefault} ) {
        $Priorities{''} = '-';
    }
    return \%Priorities;
}

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

    my @UpdatableFields;

    # set the fields that can be updateable via AJAXUpdate
    if ( !$Param{OnlyDynamicFields} ) {
        @UpdatableFields = qw(
            TypeID ServiceID SLAID NewOwnerID NewResponsibleID NewStateID
            NewPriorityID
        );
    }

    # define the dynamic fields to show based on the object type
    my $ObjectType = ['Ticket'];

    # get config of frontend module
    my $Config = $Kernel::OM->Get('Kernel::Config')->Get("Ticket::Frontend::$Self->{Action}");
# ---
# OTRSMasterSlave
# ---
    # get master/slave dynamic field
    my $MasterSlaveDynamicField = $Kernel::OM->Get('Kernel::Config')->Get('MasterSlave::DynamicField') || '';
    my $MasterSlaveAdvancedEnabled = $Kernel::OM->Get('Kernel::Config')->Get('MasterSlave::AdvancedEnabled') || 0;

    if ($MasterSlaveAdvancedEnabled) {
        $Config->{DynamicField}->{$MasterSlaveDynamicField} = 1;
    }
# ---

    # only screens that add notes can modify Article dynamic fields
    if ( $Config->{Note} ) {
        $ObjectType = [ 'Ticket', 'Article' ];
    }

    # get the dynamic fields for this screen
    my $DynamicField = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        Valid       => 1,
        ObjectType  => $ObjectType,
        FieldFilter => $Config->{DynamicField} || {},
    );

    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{$DynamicField} ) {
        next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);

        my $IsACLReducible = $Kernel::OM->Get('Kernel::System::DynamicField::Backend')->HasBehavior(
            DynamicFieldConfig => $DynamicFieldConfig,
            Behavior           => 'IsACLReducible',
        );
        next DYNAMICFIELD if !$IsACLReducible;

        push @UpdatableFields, 'DynamicField_' . $DynamicFieldConfig->{Name};
    }

    return \@UpdatableFields;
}

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

    # get needed objects
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    if ( $LayoutObject->{BrowserRichText} ) {

        # rewrap body if exists
        if ( $Param{Body} ) {
            $Param{Body} =~ s/\t/ /g;
            my $Quote = $LayoutObject->Ascii2Html(
                Text           => $ConfigObject->Get('Ticket::Frontend::Quote') || '',
                HTMLResultMode => 1,
            );
            if ($Quote) {

                # quote text
                $Param{Body} = "<blockquote type=\"cite\">$Param{Body}</blockquote>\n";

                # cleanup not compat. tags
                $Param{Body} = $LayoutObject->RichTextDocumentCleanup(
                    String => $Param{Body},
                );

                my $ResponseFormat = $LayoutObject->{LanguageObject}
                    ->FormatTimeString( $Param{CreateTime}, 'DateFormat', 'NoSeconds' );
                $ResponseFormat .= ' - ' . $Param{From} . ' ';
                $ResponseFormat
                    .= $LayoutObject->{LanguageObject}->Translate('wrote') . ':';

                $Param{Body} = $ResponseFormat . $Param{Body};

            }
            else {
                $Param{Body} = "<br/>" . $Param{Body};

                if ( $Param{CreateTime} ) {
                    $Param{Body} = $LayoutObject->{LanguageObject}->Translate('Date') .
                        ": $Param{CreateTime}<br/>" . $Param{Body};
                }

                for (qw(Subject ReplyTo Reply-To Cc To From)) {
                    if ( $Param{$_} ) {
                        $Param{Body} = $LayoutObject->{LanguageObject}->Translate($_) .
                            ": $Param{$_}<br/>" . $Param{Body};
                    }
                }

                my $From = $LayoutObject->Ascii2RichText(
                    String => $Param{From},
                );

                my $MessageFrom = $LayoutObject->{LanguageObject}->Translate('Message from');
                my $EndMessage  = $LayoutObject->{LanguageObject}->Translate('End message');

                $Param{Body} = "<br/>---- $MessageFrom $From ---<br/><br/>" . $Param{Body};
                $Param{Body} .= "<br/>---- $EndMessage ---<br/>";
            }
        }
    }
    else {

        # prepare body, subject, ReplyTo ...
        # rewrap body if exists
        if ( $Param{Body} ) {
            $Param{Body} =~ s/\t/ /g;
            my $Quote = $ConfigObject->Get('Ticket::Frontend::Quote');
            if ($Quote) {
                $Param{Body} =~ s/\n/\n$Quote /g;
                $Param{Body} = "\n$Quote " . $Param{Body};

                my $ResponseFormat = $LayoutObject->{LanguageObject}
                    ->FormatTimeString( $Param{CreateTime}, 'DateFormat', 'NoSeconds' );
                $ResponseFormat .= ' - ' . $Param{From} . ' ';
                $ResponseFormat
                    .= $LayoutObject->{LanguageObject}->Translate('wrote') . ":\n";

                $Param{Body} = $ResponseFormat . $Param{Body};
            }
            else {
                $Param{Body} = "\n" . $Param{Body};
                if ( $Param{CreateTime} ) {
                    $Param{Body} = $LayoutObject->{LanguageObject}->Translate('Date') .
                        ": $Param{CreateTime}\n" . $Param{Body};
                }

                for (qw(Subject ReplyTo Reply-To Cc To From)) {
                    if ( $Param{$_} ) {
                        $Param{Body} = $LayoutObject->{LanguageObject}->Translate($_) .
                            ": $Param{$_}\n" . $Param{Body};
                    }
                }

                my $MessageFrom = $LayoutObject->{LanguageObject}->Translate('Message from');
                my $EndMessage  = $LayoutObject->{LanguageObject}->Translate('End message');

                $Param{Body} = "\n---- $MessageFrom $Param{From} ---\n\n" . $Param{Body};
                $Param{Body} .= "\n---- $EndMessage ---\n";
            }
        }
    }

    return $Param{Body};
}

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

    # get create templates
    my %Templates;

    # check needed
    return \%Templates if !$Param{QueueID} && !$Param{TicketID};

    my $QueueID = $Param{QueueID} || '';
    if ( !$Param{QueueID} && $Param{TicketID} ) {

        # get QueueID from the ticket
        my %Ticket = $Kernel::OM->Get('Kernel::System::Ticket')->TicketGet(
            TicketID      => $Param{TicketID},
            DynamicFields => 0,
            UserID        => $Self->{UserID},
        );
        $QueueID = $Ticket{QueueID} || '';
    }

    # fetch all std. templates
    my %StandardTemplates = $Kernel::OM->Get('Kernel::System::Queue')->QueueStandardTemplateMemberList(
        QueueID       => $QueueID,
        TemplateTypes => 1,
    );

    # return empty hash if there are no templates for this screen
    return \%Templates if !IsHashRefWithData( $StandardTemplates{Note} );

    # return just the templates for this screen
    return $StandardTemplates{Note};
}

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

    # get type
    my %Type;
    if ( $Param{QueueID} || $Param{TicketID} ) {
        %Type = $Kernel::OM->Get('Kernel::System::Ticket')->TicketTypeList(
            %Param,
            TicketID => $Self->{TicketID},
            Action   => $Self->{Action},
            UserID   => $Self->{UserID},
        );
    }
    return \%Type;
}

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

    # Get Queues.
    my %Queues = $Kernel::OM->Get('Kernel::System::Ticket')->TicketMoveList(
        %Param,
        TicketID => $Self->{TicketID},
        UserID   => $Self->{UserID},
        Action   => $Self->{Action},
        Type     => 'move_into',
    );
    return \%Queues;
}
# ---
# OTRSMasterSlave
# ---
sub _GetMasterSlaveData {
    my ($Self, %Param) = @_;

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    my $UnsetMasterSlave  = $ConfigObject->Get('MasterSlave::UnsetMasterSlave') || 0;
    my $UpdateMasterSlave = $ConfigObject->Get('MasterSlave::UpdateMasterSlave') || 0;

    my %Ticket = %{ $Param{Ticket} };
    my $MasterSlaveDynamicField = $Param{MasterSlaveDynamicField};

    my $FieldValue = $Ticket{ 'DynamicField_' . $MasterSlaveDynamicField } || '';

    # get layout object
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    my %Data = (
        '' => '-',
    );

    if ( $FieldValue ne 'Master' ) {
        $Data{Master} = $LayoutObject->{LanguageObject}->Translate('New Master Ticket');
    }
    if ( $UnsetMasterSlave && $FieldValue eq 'Master' ) {
        $Data{UnsetMaster} = $LayoutObject->{LanguageObject}->Translate('Unset Master Ticket');
    }
    if ( $UnsetMasterSlave && $FieldValue =~ m{^SlaveOf:(.*?)$}xms ) {
        $Data{UnsetSlave}  = $LayoutObject->{LanguageObject}->Translate('Unset Slave Ticket');
    }

    # get ticket object
    my $TicketObject      = $Kernel::OM->Get('Kernel::System::Ticket');
    my $TicketHook        = $ConfigObject->Get('Ticket::Hook');
    my $TicketHookDivider = $ConfigObject->Get('Ticket::HookDivider');

    if ( $UpdateMasterSlave ) {
        my @TicketIDs = $TicketObject->TicketSearch(
            Result => 'ARRAY',

            # master slave dynamic field
            'DynamicField_' . $MasterSlaveDynamicField => {
                Equals => 'Master',
            },

            StateType  => 'Open',
            Limit      => 60,
            UserID     => $Self->{UserID},
            Permission => 'ro',
        );

        TICKETID:
        for my $TicketID (@TicketIDs) {
            my %CurrentTicket = $TicketObject->TicketGet(
                TicketID => $TicketID,
            );

            next TICKETID if !%CurrentTicket;
            next TICKETID if $FieldValue eq "SlaveOf:$CurrentTicket{TicketNumber}";
            next TICKETID if $FieldValue && $Ticket{TicketID} eq $CurrentTicket{TicketID};

            $Data{"SlaveOf:$CurrentTicket{TicketNumber}"} = $LayoutObject->{LanguageObject}->Translate(
                'Slave of %s%s%s: %s',
                $TicketHook,
                $TicketHookDivider,
                $CurrentTicket{TicketNumber},
                $CurrentTicket{Title},
            );
        }
    }

    return \%Data;
}
# ---

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::TicketBulk::MasterSlave;

use strict;
use warnings;

use Kernel::Language qw(Translatable);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::Ticket',
    'Kernel::System::Web::Request',
);

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

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

    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # get master/slave dynamic field
    $Self->{MasterSlaveDynamicField}    = $ConfigObject->Get('MasterSlave::DynamicField')    || '';
    $Self->{MasterSlaveAdvancedEnabled} = $ConfigObject->Get('MasterSlave::AdvancedEnabled') || 0;

    if ( $Self->{MasterSlaveDynamicField} ) {
        $Self->{DynamicFieldConfig} = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
            Name => $Self->{MasterSlaveDynamicField},
        );
    }

    return $Self;
}

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

    # if there is no configured dynamic field or if advanced mode is not enable, there is nothing to do
    return if !$Self->{MasterSlaveDynamicField};
    return if !$Self->{MasterSlaveAdvancedEnabled};

    my $ServerError;
    my $ErrorMessage;
    if ( exists $Param{Errors}->{ $Self->{DynamicFieldConfig}->{Name} } ) {
        $ServerError  = 1;
        $ErrorMessage = $Param{Errors}->{ $Self->{DynamicFieldConfig}->{Name} };
    }

    my $PossibleValuesFilter = $Self->_GetMasterSlaveData(
        %Param,
        MasterSlaveDynamicField => $Self->{MasterSlaveDynamicField},
    );

    # get field HTML
    my $DynamicFieldHTML = $Kernel::OM->Get('Kernel::System::DynamicField::Backend')->EditFieldRender(
        DynamicFieldConfig   => $Self->{DynamicFieldConfig},
        PossibleValuesFilter => $PossibleValuesFilter,
        ServerError          => $ServerError || '',
        ErrorMessage         => $ErrorMessage || '',
        LayoutObject         => $Kernel::OM->Get('Kernel::Output::HTML::Layout'),
        ParamObject          => $Kernel::OM->Get('Kernel::System::Web::Request'),
        Mandatory            => 0,
    );

    # indentation here is on purpose so the HTML will look according to the framework
    my $HTMLString = <<"EOF";
                    $DynamicFieldHTML->{Label}
                    <div class="Field">
                        $DynamicFieldHTML->{Field}
                    </div>
                    <div class="Clear"></div>
EOF

    return $HTMLString;
}

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

    # if there is no configured dynamic field or if advanced mode is not enable, there is nothing to do
    return if !$Self->{MasterSlaveDynamicField};
    return if !$Self->{MasterSlaveAdvancedEnabled};

    my $PossibleValuesFilter = $Self->_GetMasterSlaveData(
        %Param,
        MasterSlaveDynamicField => $Self->{MasterSlaveDynamicField},
    );

    my $ValidationResult = $Kernel::OM->Get('Kernel::System::DynamicField::Backend')->EditFieldValueValidate(
        DynamicFieldConfig   => $Self->{DynamicFieldConfig},
        PossibleValuesFilter => $PossibleValuesFilter,
        ParamObject          => $Kernel::OM->Get('Kernel::System::Web::Request'),
        Mandatory            => 0,
    );

    if ( $ValidationResult->{ServerError} ) {
        return (
            {
                ErrorKey   => $Self->{DynamicFieldConfig}->{Name},
                ErrorValue => $ValidationResult->{ErrorMessage},
            }
        );
    }

    return;
}

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

    # if there is no configured dynamic field or if advanced mode is not enable, there is nothing to do
    return 1 if !$Self->{MasterSlaveDynamicField};
    return 1 if !$Self->{MasterSlaveAdvancedEnabled};

    # get needed objects
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    # extract the dynamic field value form the web request
    my $DynamicFieldValue = $DynamicFieldBackendObject->EditFieldValueGet(
        DynamicFieldConfig => $Self->{DynamicFieldConfig},
        ParamObject        => $Kernel::OM->Get('Kernel::System::Web::Request'),
        LayoutObject       => $Kernel::OM->Get('Kernel::Output::HTML::Layout'),
    );

    # set the value
    my $Success = $DynamicFieldBackendObject->ValueSet(
        DynamicFieldConfig => $Self->{DynamicFieldConfig},
        ObjectID           => $Param{TicketID},
        Value              => $DynamicFieldValue,
        UserID             => $Param{UserID},
    );

    return 1;
}

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

    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    # get master slave config
    my $UnsetMasterSlave  = $ConfigObject->Get('MasterSlave::UnsetMasterSlave')  || 0;
    my $UpdateMasterSlave = $ConfigObject->Get('MasterSlave::UpdateMasterSlave') || 0;

    my %Data = (
        ''     => '-',
        Master => $LayoutObject->{LanguageObject}->Translate('New Master Ticket'),
    );

    if ($UnsetMasterSlave) {
        $Data{UnsetMaster} = Translatable('Unset Master Tickets');
        $Data{UnsetSlave}  = Translatable('Unset Slave Tickets');
    }

    # get needed objects
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    if ($UpdateMasterSlave) {

        my @TicketIDs = $TicketObject->TicketSearch(
            Result => 'ARRAY',

            # master slave dynamic field
            'DynamicField_' . $Param{MasterSlaveDynamicField} => {
                Equals => 'Master',
            },

            StateType  => 'Open',
            Limit      => 60,
            UserID     => $Param{UserID},
            Permission => 'ro',
        );

        my $TicketHook        = $ConfigObject->Get('Ticket::Hook');
        my $TicketHookDivider = $ConfigObject->Get('Ticket::HookDivider');

        TICKETID:
        for my $TicketID (@TicketIDs) {

            # get each ticket from the search results
            my %CurrentTicket = $TicketObject->TicketGet(
                TicketID => $TicketID
            );
            next TICKETID if !%CurrentTicket;

            $Data{"SlaveOf:$CurrentTicket{TicketNumber}"} = $LayoutObject->{LanguageObject}->Translate(
                'Slave of %s%s%s: %s',
                $TicketHook,
                $TicketHookDivider,
                $CurrentTicket{TicketNumber},
                $CurrentTicket{Title},
            );
        }
    }
    return \%Data;

}
1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1 class="InvisibleText">
        [% Translate("Dynamic Fields") | html %] - [% Translate(Data.ObjectTypeName) | html %]:
        [% USE Title = String(Data.BreadcrumbText) %]
        [% Title | html %]
    </h1>

    [% BreadcrumbPath = [
            {
                Name => Translate('Dynamic Fields Management'),
                Link => 'AdminDynamicField',
            },
        ]
    %]

    [% USE BreadcrumbName = String(Translate(Data.ObjectTypeName)) %]

    [% SWITCH Data.Mode %]
        [% CASE 'Add' %]
            [% BreadcrumbPath.push({ Name => BreadcrumbName.append( ': ', Title ) }) %]
        [% CASE 'Change' %]
            [% BreadcrumbPath.push({ Name => BreadcrumbName.append( ': ', Title, ' - ', Data.Name ) }) %]
    [% END %]

    [% INCLUDE "Breadcrumb.tt" Path = BreadcrumbPath %]

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

    <div class="SidebarColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Actions") | html %]</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
                    <li>
                        <a href="[% Env("Baselink") %]Action=AdminDynamicField" class="CallForAction Fullsize Center"><span><i class="fa fa-caret-left"></i>[% Translate("Go back to overview") | html %]</span></a>
                    </li>
                </ul>
            </div>
        </div>
    </div>

    <div class="ContentColumn">
        <form action="[% Env("CGIHandle") %]" method="post" class="Validate PreventMultipleSubmits">
            <input type="hidden" name="Action" value="AdminDynamicFieldMasterSlave" />
            <input type="hidden" name="Subaction" value="[% Data.Mode | html %]Action" />
            <input type="hidden" name="ObjectType" value="[% Data.ObjectType | html %]" />
            <input type="hidden" name="FieldType" value="[% Data.FieldType | html %]" />
            <input type="hidden" name="ID" value="[% Data.ID | html %]" />

            <div class="WidgetSimple">
                <div class="Header">
                    <h2>[% Translate("General") | html %]</h2>
                </div>
                <div class="Content">
                    <div class="LayoutGrid ColumnsWithSpacing">
                        <div class="Size1of2">
                            <fieldset class="TableLike">
                                <label class="Mandatory" for="Name"><span class="Marker">*</span> [% Translate("Name") | html %]:</label>
                                <div class="Field">
                                    <input id="Name" class="W50pc [% Data.NameServerError | html %] [% Data.ShowWarning | html %]  Validate_Alphanumeric" type="text" maxlength="200" value="[% Data.Name | html %]" name="Name" [% Data.ReadonlyInternalField | html %] />
                                    <div id="NameError" class="TooltipErrorMessage"><p>[% Translate("This field is required, and the value should be alphabetic and numeric characters only.") | html %]</p></div>
                                    <div id="NameServerError" class="TooltipErrorMessage"><p>[% Translate(Data.NameServerErrorMessage) | html %]</p></div>
                                    <p class="FieldExplanation">[% Translate("Must be unique and only accept alphabetic and numeric characters.") | html %]</p>
                                    <p class="Warning Hidden">[% Translate("Changing this value will require manual changes in the system.") | html %]</p>
                                </div>
                                <div class="Clear"></div>

                                <label class="Mandatory" for="Label"><span class="Marker">*</span> [% Translate("Label") | html %]:</label>
                                <div class="Field">
                                    <input id="Label" class="W50pc [% Data.LabelServerError | html %] Validate_Required" type="text" maxlength="200" value="[% Data.Label | html %]" name="Label"/>
                                    <div id="LabelError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                                    <div id="LabelServerError" class="TooltipErrorMessage"><p>[% Translate(Data.LabelServerErrorMessage) | html %]</p></div>
                                    <p class="FieldExplanation">[% Translate("This is the name to be shown on the screens where the field is active.") | html %]</p>
                                </div>
                                <div class="Clear"></div>

                                <label class="Mandatory" for="FieldOrder"><span class="Marker">*</span> [% Translate("Field order") | html %]:</label>
                                <div class="Field">
                                    [% Data.DynamicFieldOrderStrg %]
                                    <div id="FieldOrderError" class="TooltipErrorMessage"><p>[% Translate("This field is required and must be numeric.") | html %]</p></div>
                                    <div id="FieldOrderServerError" class="TooltipErrorMessage"><p>[% Translate(Data.FieldOrderServerErrorMessage) | html %]</p></div>
                                    <p class="FieldExplanation">[% Translate("This is the order in which this field will be shown on the screens where is active.") | html %]</p>
                                </div>
                                <div class="Clear"></div>
                            </fieldset>
                        </div>
                        <div class="Size1of2">
                            <fieldset class="TableLike">
                                <label for="ValidID">[% Translate("Validity") | html %]:</label>
                                <div class="Field">
                                    [% Data.ValidityStrg %]
                                </div>
                                <div class="Clear"></div>

                                <div class="SpacingTop"></div>
                                <label for="FieldTypeName">[% Translate("Field type") | html %]:</label>
                                <div class="Field">
                                    <input id="FieldTypeName" readonly class="W50pc" type="text" maxlength="200" value="[% Translate(Data.FieldTypeName) | html %]" name="FieldTypeName"/>
                                    <div class="Clear"></div>
                                </div>

                                <div class="SpacingTop"></div>
                                <label for="ObjectTypeName">[% Translate("Object type") | html %]:</label>
                                <div class="Field">
                                    <input id="ObjectTypeName" readonly class="W50pc" type="text" maxlength="200" value="[% Translate(Data.ObjectTypeName) | html %]" name="ObjectTypeName"/>
                                    <div class="Clear"></div>
                                </div>

[% RenderBlockStart("InternalField") %]
                                <div class="SpacingTop"></div>
                                <label for="InternalField">[% Translate("Internal field") | html %]:</label>
                                <div class="Field">
                                    <input id="InternalField" readonly class="W50pc" type="text" maxlength="1" value="[% Data.InternalField | html %]" name="InternalField"/>
                                    <p class="FieldExplanation">
                                        [% Translate("This field is protected and can't be deleted.") | html %]
                                    </p>
                                    <div class="Clear"></div>
                                </div>
[% RenderBlockEnd("InternalField") %]
                            </fieldset>
                        </div>
                    </div>
                </div>
            </div>
            <fieldset class="TableLike">
                <div class="Field SpacingTop">
                    <button type="submit" class="Primary CallForAction" value="[% Translate("Submit") | html %]"><span>[% Translate("Submit") | html %]</span></button>
                    [% Translate("or") | html %]
                    <a href="[% Env("Baselink") %]Action=AdminDynamicField">[% Translate("Cancel") | html %]</a>
                </div>
                <div class="Clear"></div>
            </fieldset>
        </form>
    </div>
</div>

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# $origin: otrs - 8207d0f681adcdeb5c1b497ac547a1d9749838d5 - Kernel/Output/HTML/Templates/Standard/AgentTicketActionCommon.tt
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --
# ---
# OTRSMasterSlave
# ---

# This module uses AgentTicketActionCommon as a base, for easy update and framework compatibility
# special markers has been set along the file to easy spot the differences introduced by
# OTRSMasterSlave package
# ---

[% RenderBlockStart("Properties") %]

<form action="[% Env("CGIHandle") %]" method="post" enctype="multipart/form-data" name="compose" id="Compose" class="Validate PreventMultipleSubmits">
    <input type="hidden" name="Action" value="[% Env("Action") %]"/>
    <input type="hidden" name="Subaction" value="Store"/>
    <input type="hidden" name="TicketID" value="[% Data.TicketID | html %]"/>
    <input type="hidden" name="ReplyToArticle" value="[% Data.ReplyToArticle | html %]"/>
    <input type="hidden" name="Expand" id="Expand" value=""/>
    <input type="hidden" name="FormID" value="[% Data.FormID | html %]"/>
    <input type="hidden" name="FormDraftTitle" value="[% Data.FormDraftTitle | html %]"/>
    <input type="hidden" name="FormDraftID" value="[% Data.FormDraftID | html %]"/>
    <input type="hidden" name="FormDraftAction" id="FormDraftAction" />

    <div class="LayoutPopup ARIARoleMain">
        <div class="Header">
            <h1>
[% RenderBlockStart("HeaderAgentTicketFreeText") %]
[% Translate("Change Free Text of %s%s%s", Config('Ticket::Hook'), Config('Ticket::HookDivider'), Data.TicketNumber) | html %]
[% RenderBlockEnd("HeaderAgentTicketFreeText") %]
[% RenderBlockStart("HeaderAgentTicketOwner") %]
[% Translate("Change Owner of %s%s%s", Config('Ticket::Hook'), Config('Ticket::HookDivider'), Data.TicketNumber) | html %]
[% RenderBlockEnd("HeaderAgentTicketOwner") %]
[% RenderBlockStart("HeaderAgentTicketClose") %]
[% Translate("Close %s%s%s", Config('Ticket::Hook'), Config('Ticket::HookDivider'), Data.TicketNumber) | html %]
[% RenderBlockEnd("HeaderAgentTicketClose") %]
[% RenderBlockStart("HeaderAgentTicketNote") %]
[% Translate("Add Note to %s%s%s", Config('Ticket::Hook'), Config('Ticket::HookDivider'), Data.TicketNumber) | html %]
[% RenderBlockEnd("HeaderAgentTicketNote") %]
[% RenderBlockStart("HeaderAgentTicketPending") %]
[% Translate("Set Pending Time for %s%s%s", Config('Ticket::Hook'), Config('Ticket::HookDivider'), Data.TicketNumber) | html %]
[% RenderBlockEnd("HeaderAgentTicketPending") %]
[% RenderBlockStart("HeaderAgentTicketPriority") %]
[% Translate("Change Priority of %s%s%s", Config('Ticket::Hook'), Config('Ticket::HookDivider'), Data.TicketNumber) | html %]
[% RenderBlockEnd("HeaderAgentTicketPriority") %]
[% RenderBlockStart("HeaderAgentTicketResponsible") %]
[% Translate("Change Responsible of %s%s%s", Config('Ticket::Hook'), Config('Ticket::HookDivider'), Data.TicketNumber) | html %]
[% RenderBlockEnd("HeaderAgentTicketResponsible") %]
# ---
# OTRSMasterSlave
# ---
[% RenderBlockStart("HeaderAgentTicketMasterSlave") %]
[% Translate("Manage Master/Slave status for %s%s%s", Config('Ticket::Hook'), Config('Ticket::HookDivider'), Data.TicketNumber) | html %]
[% RenderBlockEnd("HeaderAgentTicketMasterSlave") %]
# ---
                 &mdash; [% Data.Title | html %]
            </h1>
            <p class="AsteriskExplanation">[% Translate("All fields marked with an asterisk (*) are mandatory.") | html %]</p>

[% RenderBlockStart("TicketBack") %]
            <p>
                <a class="CancelClosePopup" href="#">[% Translate("Cancel & close") | html %]</a>
            </p>
[% RenderBlockEnd("TicketBack") %]
[% RenderBlockStart("PropertiesLock") %]
            <p>
                [% Translate("The ticket has been locked") | html %].
                <a class="UndoClosePopup" href="[% Env("Baselink") %]Action=AgentTicketLock;Subaction=Unlock;TicketID=[% Data.TicketID %][% IF Data.PreviousOwner %];PreviousOwner=[% Data.PreviousOwner %][% END %];[% Env("ChallengeTokenParam") | html %]">
                    [% Translate("Undo & close") | html %]
                </a>
            </p>
[% RenderBlockEnd("PropertiesLock") %]

        </div>
        <div class="Content">
            [% INCLUDE "FormElements/DraftNotifications.tt" %]
[% RenderBlockStart("WidgetTicketActions") %]
            <div class="WidgetSimple Expanded">
                <div class="Header">
                    <div class="WidgetAction Toggle">
                        <a href="#" title="[% Translate("Toggle this widget") | html %]"><i class="fa fa-caret-right"></i><i class="fa fa-caret-down"></i></a>
                    </div>
                    <h2>[% Translate("Ticket Settings") | html %]</h2>
                </div>
                <div class="Content">
                    <fieldset class="TableLike FixedLabel">
[% RenderBlockStart("Title") %]
                        <label class="Mandatory" for="Title"><span class="Marker">*</span>[% Translate("Title") | html %]:</label>
                        <div class="Field">
                            <input type="text" id="Title" name="Title" value="[% Data.Title | html %]" class="W75pc Validate_Required [% Data.TitleInvalid | html %]" />
                            <div id="TitleError" class="TooltipErrorMessage" ><p>[% Translate("This field is required.") | html %]</p></div>
                            <div id="TitleServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("Title") %]

[% RenderBlockStart("Type") %]
                        <label class="Mandatory" for="TypeID"><span class="Marker">*</span>[% Translate("Type") | html %]:</label>
                        <div class="Field">
                        [% Data.TypeStrg %]
                            <div id="TypeIDError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                            <div id="TypeIDServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("Type") %]

[% RenderBlockStart("Queue") %]
                        [% IF Data.QueueMandatory %]
                            <label class="Mandatory" for="NewQueueID"><span class="Marker">*</span>[% Translate("Queue") | html %]:</label>
                        [% ELSE %]
                            <label for="NewQueueID">[% Translate("Queue") | html %]:</label>
                        [% END %]
                        <div class="Field">
                            [% Data.QueuesStrg %]
                            [% IF Data.QueueMandatory %]
                                <div id="NewQueueIDError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                            [% END %]
                            <div id="NewQueueIDServerError" class="TooltipErrorMessage"><p>[% Translate("Queue invalid.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("Queue") %]

[% RenderBlockStart("Service") %]
                        [% IF Data.ServiceMandatory %]
                            <label class="Mandatory" for="ServiceID"><span class="Marker">*</span>[% Translate("Service") | html %]:</label>
                        [% ELSE %]
                            <label for="ServiceID">[% Translate("Service") | html %]:</label>
                        [% END %]
                        <div class="Field">
                            [% Data.ServiceStrg %]
                            [% IF Data.ServiceMandatory %]
                                <div id="ServiceIDError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                            [% END %]
                            <div id="ServiceIDServerError" class="TooltipErrorMessage"><p>[% Translate("Service invalid.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("Service") %]

[% RenderBlockStart("ServiceMandatory") %]
                        <label class="Mandatory" for="ServiceID"><span class="Marker">*</span>[% Translate("Service") | html %]:</label>
                        <div class="Field">
                            [% Data.ServiceStrg %]
                            <div id="ServiceIDError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                            <div id="ServiceIDServerError" class="TooltipErrorMessage"><p>[% Translate("Service invalid.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("ServiceMandatory") %]

[% RenderBlockStart("SLA") %]
                        [% IF Data.SLAMandatory %]
                            <label class="Mandatory" for="SLAID"><span class="Marker">*</span>[% Translate("Service Level Agreement") | html %]:</label>
                        [% ELSE %]
                            <label for="SLAID">[% Translate("Service Level Agreement") | html %]:</label>
                        [% END %]
                        <div class="Field">
                            [% Data.SLAStrg %]
                            [% IF Data.SLAMandatory %]
                                <div id="SLAIDError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                                <div id="SLAIDServerError" class="TooltipErrorMessage"><p>[% Translate("SLA invalid.") | html %]</p></div>
                            [% END %]
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("SLA") %]

[% RenderBlockStart("SLAMandatory") %]
                        <label class="Mandatory" for="SLAID"><span class="Marker">*</span>[% Translate("Service Level Agreement") | html %]:</label>
                        <div class="Field">
                            [% Data.SLAStrg %]
                            <div id="SLAIDError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                            <div id="SLAIDServerError" class="TooltipErrorMessage"><p>[% Translate("SLA invalid.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("SLAMandatory") %]

[% RenderBlockStart("Owner") %]
                        [% IF Data.OwnerMandatory %]
                            <label class="Mandatory" for="NewOwnerID"><span class="Marker">*</span>[% Translate("New Owner") | html %]:</label>
                        [% ELSE %]
                            <label for="NewOwnerID">[% Translate("New Owner") | html %]:</label>
                        [% END %]
                        <div class="Field">
                            [% Data.OwnerStrg %]
                            [% IF Data.OwnerMandatory %]
                                <div id="NewOwnerIDError" class="TooltipErrorMessage"><p>[% Translate("Please set a new owner!") | html %]</p></div>
                            [% END %]
                                <div id="NewOwnerIDServerError" class="TooltipErrorMessage"><p>[% Translate("Owner invalid.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("Owner") %]

[% RenderBlockStart("OwnerMandatory") %]
                        <label class="Mandatory" for="NewOwnerID"><span class="Marker">*</span>[% Translate("New Owner") | html %]:</label>
                        <div class="Field">
                            [% Data.OwnerStrg %]
                            <div id="NewOwnerIDError" class="TooltipErrorMessage"><p>[% Translate("Please set a new owner!") | html %]</p></div>
                            <div id="NewOwnerIDServerError" class="TooltipErrorMessage"><p>[% Translate("Owner invalid.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("OwnerMandatory") %]

[% RenderBlockStart("Responsible") %]
                        [% IF Data.ResponsibleMandatory %]
                            <label class="Mandatory" for="NewResponsibleID">
                                <span class="Marker">*</span>[% Translate("New Responsible") | html %]:
                            </label>
                        [% ELSE %]
                            <label for="NewResponsibleID">[% Translate("New Responsible") | html %]:</label>
                        [% END %]
                        <div class="Field">
                            [% Data.ResponsibleStrg %]
                            [% IF Data.ResponsibleMandatory %]
                                <div id="NewResponsibleIDError" class="TooltipErrorMessage"><p>[% Translate("Please set a new responsible!") | html %]</p></div>
                                <div id="NewResponsibleIDServerError" class="TooltipErrorMessage"><p>[% Translate("Responsible invalid.") | html %]</p></div>
                            [% END %]
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("Responsible") %]

[% RenderBlockStart("ResponsibleMandatory") %]
                        <label class="Mandatory" for="NewResponsibleID"><span class="Marker">*</span>[% Translate("New Responsible") | html %]:</label>
                        <div class="Field">
                            [% Data.ResponsibleStrg %]
                            <div id="NewResponsibleIDError" class="TooltipErrorMessage"><p>[% Translate("Please set a new responsible!") | html %]</p></div>
                            <div id="NewResponsibleIDServerError" class="TooltipErrorMessage"><p>[% Translate("Responsible invalid.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("ResponsibleMandatory") %]

[% RenderBlockStart("State") %]
                        [% IF Data.StateMandatory %]
                            <label class="Mandatory" for="NewStateID"><span class="Marker">*</span>[% Translate("Next state") | html %]:</label>
                        [% ELSE %]
                            <label for="NewStateID">[% Translate("Next state") | html %]:</label>
                        [% END %]
                        <div class="Field">
                            [% Data.StateStrg %]
                            [% IF Data.StateMandatory %]
                                <div id="NewStateIDError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                                <div id="NewStateIDServerError" class="TooltipErrorMessage"><p>[% Translate("State invalid.") | html %]</p></div>
                            [% END %]
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("State") %]

[% RenderBlockStart("StatePending") %]
                        <label>[% Translate("Pending date") | html %]:</label>
                        <div class="Field">
                            [% Data.DateString %]
                            <div id="DayServerError" class="TooltipErrorMessage"><p>[% Translate("Date invalid!") | html %]</p></div>
                            <div id="HourServerError" class="TooltipErrorMessage"><p>[% Translate("Date invalid!") | html %]</p></div>
                            <p class="FieldExplanation">[% Translate("For all pending* states.") | html %]</p>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("StatePending") %]

[% RenderBlockStart("Priority") %]
                        <label for="NewPriorityID">[% Translate("Priority") | html %]:</label>
                        <div class="Field">
                    [% Data.PriorityStrg %]
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("Priority") %]

[% RenderBlockStart("TicketTypeDynamicField") %]
                        <div class="Row Row_DynamicField_[% Data.Name | html %]">
                        [% Data.Label %]
                            <div class="Field">
                            [% Data.Field %]
                            </div>
                            <div class="Clear"></div>
                        </div>
[% RenderBlockEnd("TicketTypeDynamicField") %]

# Example of how to use fixed dynamic field blocks for customizations. Block below is for fields of type 'Ticket'.
#   NOTE: Field1 and Field2 are the names of the dynamic fields and have to be replaced with the actual field names.
#[% RenderBlockStart("TicketTypeDynamicField_Field1") %]
#                       <div class="Row Row_DynamicField_[% Data.Name | html %]">
#                           [% Data.Label %]
#                           <div class="Field">
#                               [% Data.Field %]
#                           </div>
#                           <div class="Clear"></div>
#                       </div>
#[% RenderBlockEnd("TicketTypeDynamicField_Field2") %]
#[% RenderBlockStart("TicketTypeDynamicField_Field2") %]
#                       <div class="Row Row_DynamicField_[% Data.Name | html %]">
#                           [% Data.Label %]
#                           <div class="Field">
#                               [% Data.Field %]
#                           </div>
#                           <div class="Clear"></div>
#                       </div>
#[% RenderBlockEnd("TicketTypeDynamicField_Field2") %]
                    </fieldset>
                </div>
            </div>
[% RenderBlockEnd("WidgetTicketActions") %]

[% RenderBlockStart("WidgetArticle") %]
                <div class="WidgetSimple [% Data.WidgetStatus | html %]" id="WidgetArticle">
                    <div class="Header">
                        <div class="WidgetAction Toggle">
                            <a href="#" title="[% Translate("Toggle this widget") | html %]"><i class="fa fa-caret-right"></i><i class="fa fa-caret-down"></i></a>
                        </div>
                        <h2>[% Translate("Add Article") | html %]</h2>
                    </div>
                    <div class="Content">

                        <fieldset class="TableLike FixedLabel">
[% IF Data.WidgetStatus == 'Expanded' %]
                            <input type="hidden" id="CreateArticle" name="CreateArticle" value="1" />
[% ELSE %]
                            <label for="CreateArticle">[% Translate("Create an Article") | html %]:</label>
                            <div class="Field">
                                <input type="checkbox" id="CreateArticle" name="CreateArticle" value="1" [% IF Data.CreateArticle %] checked="checked"[% END %] />
                            </div>
                            <div class="Clear"></div>
[% END %]
                        </fieldset>
[% RenderBlockStart("InformAdditionalAgents") %]
                        <fieldset class="TableLike FixedLabel">
[% RenderBlockStart("InformAgent") %]
                            <label for="InformUserID">[% Translate("Inform agents") | html %]:</label>
                            <div class="Field">
                            [% Data.OptionStrg %]
                            </div>
                            <div class="Clear"></div>
[% RenderBlockEnd("InformAgent") %]
[% RenderBlockStart("InvolvedAgent") %]
                            <label for="InvolvedUserID">[% Translate("Inform involved agents") | html %]:</label>
                            <div class="Field">
                            [% Data.InvolvedAgentStrg %]
                            </div>
                            <div class="Clear"></div>
[% RenderBlockEnd("InvolvedAgent") %]
                            <div class="Field">
                                <p class="FieldExplanation">
                                    [% Translate("Here you can select additional agents which should receive a notification regarding the new article.") | html %]
                                </p>
                            </div>
                        </fieldset>
[% RenderBlockEnd("InformAdditionalAgents") %]
[% RenderBlockStart("InformAgentsWithoutSelection") %]
                        <fieldset class="TableLike FixedLabel">
                            <label>[% Translate("Text will also be received by") | html %]:</label>
                            <div class="Field">
                                <input type="hidden" name="UserListWithoutSelection" value="[% Data.UserListWithoutSelection  | html %]" />
[% RenderBlockStart("InformAgentsWithoutSelectionSingleUser") %]
                                <span title="[% Data.UserEmail %]">[% Data.UserFullname %]</span>[% RenderBlockStart("InformAgentsWithoutSelectionSingleUserSeparator") %],[% RenderBlockEnd("InformAgentsWithoutSelectionSingleUserSeparator") %]
[% RenderBlockEnd("InformAgentsWithoutSelectionSingleUser") %]
                            </div>
                            <div class="Clear"></div>
                        </fieldset>
[% RenderBlockEnd("InformAgentsWithoutSelection") %]

                        <fieldset class="TableLike FixedLabel">
[% RenderBlockStart("SubjectLabel") %]
                            <label for="Subject">[% Translate("Subject") | html %]:</label>
[% RenderBlockEnd("SubjectLabel") %]
[% RenderBlockStart("SubjectLabelMandatory") %]
                            <label class="Mandatory" for="Subject"><span class="Marker">*</span>[% Translate("Subject") | html %]:</label>
[% RenderBlockEnd("SubjectLabelMandatory") %]
                            <div class="Field">
                                <input type="text" id="Subject" name="Subject" value="[% Data.Subject | html %]" class="W75pc Validate [% Data.SubjectInvalid %] [% Data.SubjectRequired %]"/>
                                <div id="SubjectError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                                <div id="SubjectServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                            </div>
                            <div class="Clear"></div>
[% RenderBlockStart("TicketOptions") %]
                            <label>[% Translate("Options") | html %]:</label>
                            <div class="Field">

<!-- OutputFilterHook_TicketOptionsBegin -->
<!-- OutputFilterHook_TicketOptionsEnd -->

                            </div>
                            <div class="Clear"></div>
[% RenderBlockEnd("TicketOptions") %]

<!-- OutputFilterHook_NoTicketOptionsFallback -->

[% RenderBlockStart("StandardTemplate") %]
                            <label for="StandardTemplateID">[% Translate("Text Template") | html %]:</label>
                            <div class="Field">
                                [% Data.StandardTemplateStrg %]
                                <p class="FieldExplanation">[% Translate("Setting a template will overwrite any text or attachment.") | html %]</p>
                            </div>
                            <div class="Clear"></div>
[% RenderBlockEnd("StandardTemplate") %]

[% RenderBlockStart("RichTextLabel") %]
                            <label for="RichText">[% Translate("Text") | html %]:</label>
[% RenderBlockEnd("RichTextLabel") %]
[% RenderBlockStart("RichTextLabelMandatory") %]
                            <label class="Mandatory" for="RichText"><span class="Marker">*</span>[% Translate("Text") | html %]:</label>
[% RenderBlockEnd("RichTextLabelMandatory") %]

                            <div id="RichTextField" class="RichTextField">
                                <textarea id="RichText" class="RichText Validate [% Data.BodyInvalid %] [% Data.BodyRequired %]" name="Body" rows="15" cols="[% Config("Ticket::Frontend::TextAreaNote") %]">[% Data.Body | html %]</textarea>
                                <div id="RichTextError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                                <div id="RichTextServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                            </div>
                            <div class="Clear"></div>

                            <label>[% Translate("Attachments") | html %]:</label>
                            <div class="Field">
[% INCLUDE "FormElements/AttachmentList.tt" %]
                            </div>
                            <div class="Clear"></div>

                            <label for="IsVisibleForCustomer">[% Translate("Is visible for customer") | html %]:</label>
                            <div class="Field">
                                <input type="checkbox" name="IsVisibleForCustomer" id="IsVisibleForCustomer"
                                    [% IF Data.IsVisibleForCustomer %]checked="checked" [% END %]/>
                            </div>
                            <div class="Clear"></div>

[% RenderBlockStart("TimeUnitsLabel") %]
                            <label for="TimeUnits">[% Translate("Time units") | html %] [% Translate(Config("Ticket::Frontend::TimeUnits")) | html %]:</label>
[% RenderBlockEnd("TimeUnitsLabel") %]
[% RenderBlockStart("TimeUnitsLabelMandatory") %]
                            <label class="Mandatory" for="TimeUnits"><span class="Marker">*</span>[% Translate("Time units") | html %] [% Translate(Config("Ticket::Frontend::TimeUnits")) | html %]:</label>
[% RenderBlockEnd("TimeUnitsLabelMandatory") %]
[% RenderBlockStart("TimeUnits") %]
                            <div class="Field">
                                <input type="text" name="TimeUnits" id="TimeUnits" value="[% Data.TimeUnits | html %]" class="W50pc Validate_TimeUnits [% Data.TimeUnitsRequired | html %] [% Data.TimeUnitsInvalid | html %]"/>
                                <div id="TimeUnitsError" class="TooltipErrorMessage"><p>[% Translate("Invalid time!") | html %]</p></div>
                                <div id="TimeUnitsServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                            </div>
                            <div class="Clear"></div>
[% RenderBlockEnd("TimeUnits") %]

[% RenderBlockStart("ArticleTypeDynamicField") %]
                            <div class="Row Row_DynamicField_[% Data.Name | html %]">
                            [% Data.Label %]
                                <div class="Field">
                                [% Data.Field %]
[% RenderBlockStart("ArticleTypeDynamicFieldError") %]
                                    <div id="DynamicField_[% Data.Name %]Error" class="TooltipErrorMessage">
                                        <p>[% Translate("This field is required.") | html %]</p>
                                    </div>
[% RenderBlockEnd("ArticleTypeDynamicFieldError") %]
                                </div>
                                <div class="Clear"></div>
                            </div>
[% RenderBlockEnd("ArticleTypeDynamicField") %]

# Example of how to use fixed dynamic field blocks for customizations. Block below is for fields of type 'Article'.
#   NOTE: Field1 and Field2 are the names of the dynamic fields and have to be replaced with the actual field names.
#[% RenderBlockStart("ArticleTypeDynamicField_Field1") %]
#                            <div class="Row Row_DynamicField_[% Data.Name | html %]">
#                                [% Data.Label %]
#                                <div class="Field">
#                                [% Data.Field %]
#                                [% IF (Data.MandatoryTooltip) %]
#                                    <div id="DynamicField_[% Data.Name %]Error" class="TooltipErrorMessage">
#                                        <p>[% Translate("This field is required.") | html %]</p>
#                                    </div>
#                                [% END %]
#                                </div>
#                                <div class="Clear"></div>
#                            </div>
#[% RenderBlockEnd("ArticleTypeDynamicField_Field1") %]
#[% RenderBlockStart("ArticleTypeDynamicField_Field2") %]
#                            <div class="Row Row_DynamicField_[% Data.Name | html %]">
#                                [% Data.Label %]
#                                <div class="Field">
#                                [% Data.Field %]
#                                [% IF (Data.MandatoryTooltip) %]
#                                    <div id="DynamicField_[% Data.Name %]Error" class="TooltipErrorMessage">
#                                        <p>[% Translate("This field is required.") | html %]</p>
#                                    </div>
#                                [% END %]
#                                </div>
#                                <div class="Clear"></div>
#                            </div>
#[% RenderBlockEnd("ArticleTypeDynamicField_Field2") %]
                        </fieldset>
                    </div>
                </div>
[% RenderBlockEnd("WidgetArticle") %]

            </div>
            <div class="Footer">
                <button class="CallForAction Primary" id="submitRichText" accesskey="g" title="[% Translate("Submit") | html %] (g)" type="submit" value="[% Translate("Submit") | html %]">
                    <span><i class="fa fa-check-square-o"></i> [% Translate("Submit") | html %]</span>
                </button>
                [% INCLUDE "FormElements/DraftButtons.tt" %]
            </div>
        </div>
    </form>
[% RenderBlockEnd("Properties") %]

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::DynamicField::Driver::MasterSlave;

use strict;
use warnings;

use Kernel::System::VariableCheck qw(:all);

use parent qw(Kernel::System::DynamicField::Driver::BaseSelect);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::Language',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::DynamicFieldValue',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::LinkObject',
    'Kernel::System::Log',
    'Kernel::System::Main',
    'Kernel::System::Ticket',
);

=head1 NAME

Kernel::System::DynamicField::Driver::MasterSlave

=head1 SYNOPSIS

DynamicFields MasterSlave Driver delegate

=head1 PUBLIC INTERFACE

This module implements the public interface of L<Kernel::System::DynamicField::Backend>.
Please look there for a detailed reference of the functions.

=head2 new()

usually, you want to create an instance of this
by using Kernel::System::DynamicField::Backend->new();

=cut

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

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

    # set field behaviors
    $Self->{Behaviors} = {
        'IsACLReducible'               => 0,
        'IsNotificationEventCondition' => 1,
        'IsSortable'                   => 1,
        'IsFiltrable'                  => 1,
        'IsStatsCondition'             => 1,
        'IsCustomerInterfaceCapable'   => 0,
    };

    # get the Dynamic Field Backend custom extensions
    my $DynamicFieldDriverExtensions
        = $Kernel::OM->Get('Kernel::Config')->Get('DynamicFields::Extension::Driver::MasterSlave');

    EXTENSION:
    for my $ExtensionKey ( sort keys %{$DynamicFieldDriverExtensions} ) {

        # skip invalid extensions
        next EXTENSION if !IsHashRefWithData( $DynamicFieldDriverExtensions->{$ExtensionKey} );

        # create a extension config shortcut
        my $Extension = $DynamicFieldDriverExtensions->{$ExtensionKey};

        # check if extension has a new module
        if ( $Extension->{Module} ) {

            # check if module can be loaded
            if (
                !$Kernel::OM->Get('Kernel::System::Main')->RequireBaseClass( $Extension->{Module} )
                )
            {
                die "Can't load dynamic fields backend module"
                    . " $Extension->{Module}! $@";
            }
        }

        # check if extension contains more behaviors
        if ( IsHashRefWithData( $Extension->{Behaviors} ) ) {

            %{ $Self->{Behaviors} } = (
                %{ $Self->{Behaviors} },
                %{ $Extension->{Behaviors} }
            );
        }
    }

    return $Self;
}

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

    # Special cases where the values are different but they should be reported as equals.
    return
        if !defined $Param{Value1}
        && (
        defined $Param{Value2}
        && (
            $Param{Value2} eq 'UnsetMaster'
            || $Param{Value2} eq 'UnsetSlave'
            || $Param{Value2} eq ''
        )
        );

    # Compare the results.
    return DataIsDifferent(
        Data1 => \$Param{Value1},
        Data2 => \$Param{Value2}
    );
}

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

    my $Success = $Self->_HandleLinks(
        FieldName  => $Param{DynamicFieldConfig}->{Name},
        FieldValue => $Param{Value},
        TicketID   => $Param{ObjectID},
        UserID     => $Param{UserID},
    );

    if ( !$Success ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "There was an error handling the links for master/slave, value could not be set",
        );

        return;
    }

    my $Value = $Param{Value} !~ /^(?:UnsetMaster|UnsetSlave)$/ ? $Param{Value} : '';

    $Success = $Kernel::OM->Get('Kernel::System::DynamicFieldValue')->ValueSet(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
        Value    => [
            {
                ValueText => $Value,
            },
        ],
        UserID => $Param{UserID},
    );

    return $Success;
}

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

    # get the field value from the http request
    my $Value = $Self->EditFieldValueGet(
        DynamicFieldConfig => $Param{DynamicFieldConfig},
        ParamObject        => $Param{ParamObject},

        # not necessary for this Driver but place it for consistency reasons
        ReturnValueStructure => 1,
    );

    my $ServerError;
    my $ErrorMessage;

    # perform necessary validations
    if ( $Param{Mandatory} && !$Value ) {
        return {
            ServerError => 1,
        };
    }
    else {

        my $PossibleValues;

        # use PossibleValuesFilter if sent
        if ( defined $Param{PossibleValuesFilter} ) {
            $PossibleValues = $Param{PossibleValuesFilter};
        }
        else {

            # get possible values list
            $PossibleValues = $Self->PossibleValuesGet(
                %Param,
            );
        }

        # validate if value is in possible values list (but let pass empty values)
        if ( $Value && !$PossibleValues->{$Value} ) {
            $ServerError  = 1;
            $ErrorMessage = 'The field content is invalid';
        }
    }

    # create resulting structure
    my $Result = {
        ServerError  => $ServerError,
        ErrorMessage => $ErrorMessage,
    };

    return $Result;
}

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

    # set HTMLOutput as default if not specified
    if ( !defined $Param{HTMLOutput} ) {
        $Param{HTMLOutput} = 1;
    }

    # get raw Value strings from field value
    my $Value = defined $Param{Value} ? $Param{Value} : '';

    # get real value
    if ( $Param{DynamicFieldConfig}->{Config}->{PossibleValues}->{$Value} ) {

        # get readable value
        $Value = $Param{DynamicFieldConfig}->{Config}->{PossibleValues}->{$Value};
    }

    if ( $Value eq 'Master' ) {
        $Value = $Param{LayoutObject}->{LanguageObject}->Translate('Master');
    }
    elsif ( $Value =~ m{SlaveOf:(\d+)}msx ) {

        my $TicketNumber = $1;
        if ($TicketNumber) {
            my $ConfigObject      = $Kernel::OM->Get('Kernel::Config');
            my $TicketHook        = $ConfigObject->Get('Ticket::Hook');
            my $TicketHookDivider = $ConfigObject->Get('Ticket::HookDivider');
            $Value = $Param{LayoutObject}->{LanguageObject}->Translate(
                'Slave of %s%s%s',
                $TicketHook,
                $TicketHookDivider,
                $TicketNumber,
            );
        }
    }

    # set title as value after update and before limit
    my $Title = $Value;

    # HTMLOutput transformations
    if ( $Param{HTMLOutput} ) {
        $Value = $Param{LayoutObject}->Ascii2Html(
            Text => $Value,
            Max  => $Param{ValueMaxChars} || '',
        );

        $Title = $Param{LayoutObject}->Ascii2Html(
            Text => $Title,
            Max  => $Param{TitleMaxChars} || '',
        );
    }
    else {
        if ( $Param{ValueMaxChars} && length($Value) > $Param{ValueMaxChars} ) {
            $Value = substr( $Value, 0, $Param{ValueMaxChars} ) . '...';
        }
        if ( $Param{TitleMaxChars} && length($Title) > $Param{TitleMaxChars} ) {
            $Title = substr( $Title, 0, $Param{TitleMaxChars} ) . '...';
        }
    }

    # set field link from config
    my $Link        = $Param{DynamicFieldConfig}->{Config}->{Link}        || '';
    my $LinkPreview = $Param{DynamicFieldConfig}->{Config}->{LinkPreview} || '';

    my $Data = {
        Value       => $Value,
        Title       => $Title,
        Link        => $Link,
        LinkPreview => $LinkPreview,
    };

    return $Data;
}

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

    # to store the possible values
    my %PossibleValues = (
        '' => '-',
    );

    # get needed objects
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    # find all current open master slave tickets
    my @TicketIDs = $TicketObject->TicketSearch(
        Result => 'ARRAY',

        # master slave dynamic field
        'DynamicField_' . $Param{DynamicFieldConfig}->{Name} => {
            Equals => 'Master',
        },

        StateType  => 'Open',
        Limit      => 60,
        UserID     => $LayoutObject->{UserID},
        Permission => 'ro',
    );

    # set dynamic field possible values
    $PossibleValues{Master} = $LayoutObject->{LanguageObject}->Translate('New Master Ticket');

    my $ConfigObject      = $Kernel::OM->Get('Kernel::Config');
    my $TicketHook        = $ConfigObject->Get('Ticket::Hook');
    my $TicketHookDivider = $ConfigObject->Get('Ticket::HookDivider');

    TICKET:
    for my $TicketID (@TicketIDs) {
        my %CurrentTicket = $TicketObject->TicketGet(
            TicketID      => $TicketID,
            DynamicFields => 1,
        );

        next TICKET if !%CurrentTicket;

        # set dynamic field possible values
        $PossibleValues{"SlaveOf:$CurrentTicket{TicketNumber}"} = $LayoutObject->{LanguageObject}->Translate(
            'Slave of %s%s%s: %s',
            $TicketHook,
            $TicketHookDivider,
            $CurrentTicket{TicketNumber},
            $CurrentTicket{Title},
        );
    }

    # If config UnsetMasterSlave is enabled and we are requesting values from AdminGenericAgent,
    #   add UnsetMaster and UnsetSlave possible values. See bug#14778 (https://bugs.otrs.org/show_bug.cgi?id=14778).
    if ( $ConfigObject->Get('MasterSlave::UnsetMasterSlave') && $LayoutObject->{Action} eq 'AdminGenericAgent' )
    {
        $PossibleValues{UnsetMaster} = $LayoutObject->{LanguageObject}->Translate('Unset Master Ticket');
        $PossibleValues{UnsetSlave}  = $LayoutObject->{LanguageObject}->Translate('Unset Slave Ticket');
    }

    # return the possible values hash as a reference
    return \%PossibleValues;
}

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

    # check needed stuff
    for my $Needed (qw(FieldName FieldValue TicketID UserID)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );
            return;
        }
    }

    # get ticket object
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    my $FieldName = $Param{FieldName};

    my %Ticket = $Param{Ticket}
        ? %{ $Param{Ticket} }
        : $TicketObject->TicketGet(
        TicketID      => $Param{TicketID},
        DynamicFields => 1,
        );

    my $OldValue = $Ticket{ 'DynamicField_' . $FieldName };

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # get master slave config
    my $MasterSlaveKeepParentChildAfterUnset  = $ConfigObject->Get('MasterSlave::KeepParentChildAfterUnset')  || 0;
    my $MasterSlaveFollowUpdatedMaster        = $ConfigObject->Get('MasterSlave::FollowUpdatedMaster')        || 0;
    my $MasterSlaveKeepParentChildAfterUpdate = $ConfigObject->Get('MasterSlave::KeepParentChildAfterUpdate') || 0;

    my $NewValue = $Param{FieldValue};

    # get link object
    my $LinkObject = $Kernel::OM->Get('Kernel::System::LinkObject');

    # set a new master ticket
    # check if it is already a master ticket
    if (
        $NewValue eq 'Master'
        && ( !$OldValue || $OldValue ne $NewValue )
        )
    {

        # check if it was a slave ticket before and if we have to delete
        # the old parent child link (MasterSlaveKeepParentChildAfterUnset)
        if (
            $OldValue
            && $OldValue =~ /^SlaveOf:(.*?)$/
            && !$MasterSlaveKeepParentChildAfterUnset
            )
        {
            my $SourceKey = $TicketObject->TicketIDLookup(
                TicketNumber => $1,
                UserID       => $Param{UserID},
            );

            $LinkObject->LinkDelete(
                Object1 => 'Ticket',
                Key1    => $SourceKey,
                Object2 => 'Ticket',
                Key2    => $Param{TicketID},
                Type    => 'ParentChild',
                UserID  => $Param{UserID},
            );
        }
    }

    # set a new slave ticket
    # check if it's already the slave of the wished master ticket
    elsif (
        $NewValue =~ /^SlaveOf:(.*?)$/
        && ( !$OldValue || $OldValue ne $NewValue )
        )
    {
        my $SourceKey = $TicketObject->TicketIDLookup(
            TicketNumber => $1,
            UserID       => $Param{UserID},
        );

        $LinkObject->LinkAdd(
            SourceObject => 'Ticket',
            SourceKey    => $SourceKey,
            TargetObject => 'Ticket',
            TargetKey    => $Param{TicketID},
            Type         => 'ParentChild',
            State        => 'Valid',
            UserID       => $Param{UserID},
        );

        my %Links = $LinkObject->LinkKeyList(
            Object1   => 'Ticket',
            Key1      => $Param{TicketID},
            Object2   => 'Ticket',
            State     => 'Valid',
            Type      => 'ParentChild',      # (optional)
            Direction => 'Target',           # (optional) default Both (Source|Target|Both)
            UserID    => $Param{UserID},
        );

        my @SlaveTicketIDs;

        LINKEDTICKETID:
        for my $LinkedTicketID ( sort keys %Links ) {
            next LINKEDTICKETID if !$Links{$LinkedTicketID};

            # just take ticket with slave attributes for action
            my %LinkedTicket = $TicketObject->TicketGet(
                TicketID      => $LinkedTicketID,
                DynamicFields => 1,
            );

            my $LinkedTicketValue = $LinkedTicket{ 'DynamicField_' . $FieldName };

            next LINKEDTICKETID if !$LinkedTicketValue;
            next LINKEDTICKETID if $LinkedTicketValue !~ /^SlaveOf:(.*?)$/;

            # remember linked ticket id
            push @SlaveTicketIDs, $LinkedTicketID;
        }

        if ( $OldValue && $OldValue eq 'Master' ) {

            if ( $MasterSlaveFollowUpdatedMaster && @SlaveTicketIDs ) {
                for my $LinkedTicketID (@SlaveTicketIDs) {
                    $LinkObject->LinkAdd(
                        SourceObject => 'Ticket',
                        SourceKey    => $SourceKey,
                        TargetObject => 'Ticket',
                        TargetKey    => $LinkedTicketID,
                        Type         => 'ParentChild',
                        State        => 'Valid',
                        UserID       => $Param{UserID},
                    );
                }
            }

            if ( !$MasterSlaveKeepParentChildAfterUnset ) {
                for my $LinkedTicketID (@SlaveTicketIDs) {
                    $LinkObject->LinkDelete(
                        Object1 => 'Ticket',
                        Key1    => $Param{TicketID},
                        Object2 => 'Ticket',
                        Key2    => $LinkedTicketID,
                        Type    => 'ParentChild',
                        UserID  => $Param{UserID},
                    );
                }
            }
        }
        elsif (
            $OldValue
            && $OldValue =~ /^SlaveOf:(.*?)$/
            && !$MasterSlaveKeepParentChildAfterUpdate
            )
        {
            my $SourceKey = $TicketObject->TicketIDLookup(
                TicketNumber => $1,
                UserID       => $Param{UserID},
            );

            $LinkObject->LinkDelete(
                Object1 => 'Ticket',
                Key1    => $SourceKey,
                Object2 => 'Ticket',
                Key2    => $Param{TicketID},
                Type    => 'ParentChild',
                UserID  => $Param{UserID},
            );
        }
    }
    elsif ( $NewValue =~ /^(?:UnsetMaster|UnsetSlave)$/ && $OldValue ) {

        if ( $NewValue eq 'UnsetMaster' && !$MasterSlaveKeepParentChildAfterUnset ) {
            my %Links = $LinkObject->LinkKeyList(
                Object1   => 'Ticket',
                Key1      => $Param{TicketID},
                Object2   => 'Ticket',
                State     => 'Valid',
                Type      => 'ParentChild',      # (optional)
                Direction => 'Target',           # (optional) default Both (Source|Target|Both)
                UserID    => $Param{UserID},
            );

            my @SlaveTicketIDs;

            LINKEDTICKETID:
            for my $LinkedTicketID ( sort keys %Links ) {
                next LINKEDTICKETID if !$Links{$LinkedTicketID};

                # just take ticket with slave attributes for action
                my %LinkedTicket = $TicketObject->TicketGet(
                    TicketID      => $LinkedTicketID,
                    DynamicFields => 1,
                );

                my $LinkedTicketValue = $LinkedTicket{ 'DynamicField_' . $FieldName };
                next LINKEDTICKETID if !$LinkedTicketValue;
                next LINKEDTICKETID if $LinkedTicketValue !~ /^SlaveOf:(.*?)$/;

                # remember ticket id
                push @SlaveTicketIDs, $LinkedTicketID;
            }

            my $MasterSlaveDF = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
                Name => $FieldName,
            );
            my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

            for my $LinkedTicketID (@SlaveTicketIDs) {
                $LinkObject->LinkDelete(
                    Object1 => 'Ticket',
                    Key1    => $Param{TicketID},
                    Object2 => 'Ticket',
                    Key2    => $LinkedTicketID,
                    Type    => 'ParentChild',
                    UserID  => $Param{UserID},
                );

                # UnsetSlave DynamicField value from affected slave tickets.
                $DynamicFieldBackendObject->ValueSet(
                    DynamicFieldConfig => $MasterSlaveDF,
                    FieldID            => $MasterSlaveDF->{ID},
                    ObjectID           => $LinkedTicketID,
                    Value              => 'UnsetSlave',
                    UserID             => $Param{UserID},
                );
            }
        }
        elsif (
            $NewValue eq 'UnsetSlave'
            && !$MasterSlaveKeepParentChildAfterUnset
            && $OldValue =~ /^SlaveOf:(.*?)$/
            )
        {
            my $SourceKey = $TicketObject->TicketIDLookup(
                TicketNumber => $1,
                UserID       => $Param{UserID},
            );

            $LinkObject->LinkDelete(
                Object1 => 'Ticket',
                Key1    => $SourceKey,
                Object2 => 'Ticket',
                Key2    => $Param{TicketID},
                Type    => 'ParentChild',
                UserID  => $Param{UserID},
            );
        }
    }

    return 1;
}

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

    # take config from field config
    my $FieldConfig = $Param{DynamicFieldConfig}->{Config};
    my $FieldName   = 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name};
    my $FieldLabel  = $Param{DynamicFieldConfig}->{Label};

    my $Value;

    my @DefaultValue;

    if ( defined $Param{DefaultValue} ) {
        @DefaultValue = split /;/, $Param{DefaultValue};
    }

    # set the field value
    if (@DefaultValue) {
        $Value = \@DefaultValue;
    }

    # get the field value, this function is always called after the profile is loaded
    my $FieldValues = $Self->SearchFieldValueGet(
        %Param,
    );

    if ( defined $FieldValues ) {
        $Value = $FieldValues;
    }

    # check and set class if necessary
    my $FieldClass = 'DynamicFieldMultiSelect Modernize';

    # set TreeView class
    if ( $FieldConfig->{TreeView} ) {
        $FieldClass .= ' DynamicFieldWithTreeView';
    }

    my $LanguageObject = $Kernel::OM->Get('Kernel::Language');

    # set PossibleValues (master should be always an option)
    my $SelectionData = {
        Master => $LanguageObject->Translate('Master Ticket'),
    };

    # get historical values from database
    my $HistoricalValues = $Self->HistoricalValuesGet(%Param);

    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    if ( IsHashRefWithData($HistoricalValues) ) {

        my $ConfigObject      = $Kernel::OM->Get('Kernel::Config');
        my $TicketHook        = $ConfigObject->Get('Ticket::Hook');
        my $TicketHookDivider = $ConfigObject->Get('Ticket::HookDivider');

        # Recreate the display value from the already set tickets.
        VALUE:
        for my $ValueKey ( sort keys %{$HistoricalValues} ) {

            if ( $ValueKey =~ m{SlaveOf:(.*)}gmx ) {
                my $TicketNumber = $1;

                my $TicketID = $TicketObject->TicketIDLookup(
                    TicketNumber => $TicketNumber,
                    UserID       => 1,
                );

                my %Ticket;
                if ($TicketID) {
                    %Ticket = $TicketObject->TicketGet(
                        TicketID => $TicketID
                    );
                }

                next VALUE if !%Ticket;

                $SelectionData->{$ValueKey} = $LanguageObject->Translate(
                    'Slave of %s%s%s: %s',
                    $TicketHook,
                    $TicketHookDivider,
                    $Ticket{TicketNumber},
                    $Ticket{Title},
                );
            }
        }
    }

    # use PossibleValuesFilter if defined
    $SelectionData = $Param{PossibleValuesFilter} // $SelectionData;

    my $HTMLString = $Param{LayoutObject}->BuildSelection(
        Data         => $SelectionData,
        Name         => $FieldName,
        SelectedID   => $Value,
        Translation  => 0,
        PossibleNone => 0,
        Class        => $FieldClass,
        Multiple     => 1,
        HTMLQuote    => 1,
    );

    # call EditLabelRender on the common Driver
    my $LabelString = $Self->EditLabelRender(
        %Param,
        FieldName => $FieldName,
    );

    my $Data = {
        Field => $HTMLString,
        Label => $LabelString,
    };

    return $Data;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

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

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::Ticket::Event::MasterSlave;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::CheckItem',
    'Kernel::System::CommunicationChannel',
    'Kernel::System::CustomerUser',
    'Kernel::System::DateTime',
    'Kernel::System::LinkObject',
    'Kernel::System::Log',
    'Kernel::System::Ticket',
    'Kernel::System::Ticket::Article',
);

use Kernel::System::VariableCheck qw(:all);

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

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

    return $Self;
}

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

    # check needed stuff
    for my $Needed (qw(Data Event Config)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Needed!"
            );

            return;
        }
    }
    if ( !$Param{Data}->{TicketID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need Data->{TicketID}!"
        );

        return;
    }

    # get ticket object
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    # get ticket attributes
    my %Ticket = $TicketObject->TicketGet(
        TicketID      => $Param{Data}->{TicketID},
        DynamicFields => 1,
    );

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # get master/slave dynamic field
    my $MasterSlaveDynamicField = $ConfigObject->Get('MasterSlave::DynamicField');

    # check if it's a master/slave ticket
    return 1 if !$Ticket{ 'DynamicField_' . $MasterSlaveDynamicField };
    return 1 if $Ticket{ 'DynamicField_' . $MasterSlaveDynamicField } !~ /^(master|yes)$/i;

    # get link object
    my $LinkObject = $Kernel::OM->Get('Kernel::System::LinkObject');

    # find slaves
    my %Links = $LinkObject->LinkKeyList(
        Object1   => 'Ticket',
        Key1      => $Param{Data}->{TicketID},
        Object2   => 'Ticket',
        State     => 'Valid',
        Type      => 'ParentChild',
        Direction => 'Target',
        UserID    => $Param{UserID},
    );

    my @TicketIDs;
    TICKETID:
    for my $TicketID ( sort keys %Links ) {
        next TICKETID if !$Links{$TicketID};

        # just take ticket with slave attributes for action
        my %Ticket = $TicketObject->TicketGet(
            TicketID      => $TicketID,
            DynamicFields => 1,
        );

        my $TicketValue = $Ticket{ 'DynamicField_' . $MasterSlaveDynamicField };

        next TICKETID if !$TicketValue;
        next TICKETID if $TicketValue !~ /^SlaveOf:(.*?)$/;

        # remember ticket id
        push @TicketIDs, $TicketID;
    }

    # no slaves
    if ( !@TicketIDs ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'notice',
            Message  => "No Slaves of ticket $Ticket{TicketID}!",
        );

        return 1;
    }

    # get ticket object
    my $CustomerUserObject = $Kernel::OM->Get('Kernel::System::CustomerUser');

    # auto response action
    if ( $Param{Event} eq 'ArticleSend' ) {
        my @Articles = $Kernel::OM->Get('Kernel::System::Ticket::Article')->ArticleList(
            TicketID => $Param{Data}->{TicketID},
        );

        return 1 if !@Articles;

        my $ArticleBackendObject = $Kernel::OM->Get('Kernel::System::Ticket::Article')->BackendForArticle(
            TicketID  => $Articles[-1]->{TicketID},
            ArticleID => $Articles[-1]->{ArticleID},
        );
        my %Article = $ArticleBackendObject->ArticleGet(
            TicketID  => $Articles[-1]->{TicketID},
            ArticleID => $Articles[-1]->{ArticleID},
        );

        # check if the send mail is of type forward
        my $IsForward = $Self->_ArticleHistoryTypeGiven(
            TicketID    => $Param{Data}->{TicketID},
            ArticleID   => $Article{ArticleID},
            HistoryType => 'Forward',
            UserID      => $Param{UserID},
        );

        # if the SysConfig is disabled and the new article is an forward article
        # then we don't want to have the forward for the slave tickets
        my $ForwardSlaves = $ConfigObject->Get('MasterSlave::ForwardSlaves');

        return 1 if $IsForward && !$ForwardSlaves;

        # do not send internal communications to end customers of slave tickets
        return 1 if !$Article{IsVisibleForCustomer};

        # mark ticket to prevent a loop
        $TicketObject->HistoryAdd(
            TicketID     => $Param{Data}->{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => 'MasterTicketAction: ArticleSend',
        );

        # just a flag for know when the first slave ticket is present
        my $FirstSlaveTicket = 1;
        my $TmpArticleBody;

        # Get attachments in master for usage in slave tickets (see bug#14983).
        my %AttachmentIndex = $ArticleBackendObject->ArticleAttachmentIndex(
            ArticleID => $Articles[-1]->{ArticleID},
        );

        my @Attachments;
        ATTACHMENT:
        for my $FileID ( sort keys %AttachmentIndex ) {
            next ATTACHMENT if !$FileID;
            my %Attachment = $ArticleBackendObject->ArticleAttachment(
                ArticleID => $Articles[-1]->{ArticleID},
                FileID    => $FileID,
            );

            next ATTACHMENT if !IsHashRefWithData( \%Attachment );
            push @Attachments, {%Attachment};
        }

        # perform action on linked tickets
        TICKETID:
        for my $TicketID (@TicketIDs) {
            my $CheckSuccess = $Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: ArticleSend',
            );
            next TICKETID if !$CheckSuccess;

            my %TicketSlave = $TicketObject->TicketGet(
                TicketID => $TicketID,
            );

            # try to get the customer data of the slave ticket
            my %Customer;
            if ( $TicketSlave{CustomerUserID} ) {
                %Customer = $CustomerUserObject->CustomerUserDataGet(
                    User => $TicketSlave{CustomerUserID},
                );
            }

            my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

            # check if customer email is valid
            if (
                $Customer{UserEmail}
                && !$CheckItemObject->CheckEmail( Address => $Customer{UserEmail} )
                )
            {
                $Customer{UserEmail} = '';
            }

            # if we can't find a valid UserEmail in CustomerData
            # we have to get the last Article with SenderType 'customer'
            # and get the UserEmail
            if ( !$Customer{UserEmail} ) {
                my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

                my @Articles = $ArticleObject->ArticleList(
                    TicketID          => $TicketID,
                    ArticleSenderType => 'customer',
                );

                my $CommunicationChannelObject = $Kernel::OM->Get('Kernel::System::CommunicationChannel');

                my @MIMEBaseChannels;
                for my $Channel (qw(Email Internal Phone)) {
                    my %CommunicationChannel = $CommunicationChannelObject->ChannelGet(
                        ChannelName => $Channel,
                    );

                    push @MIMEBaseChannels, $CommunicationChannel{ChannelID};
                }

                ARTICLE:
                for my $Article ( reverse @Articles ) {
                    if ( grep { $Article->{CommunicationChannelID} == $_ } @MIMEBaseChannels ) {
                        my $ArticleBackendObject = $ArticleObject->BackendForArticle( %{$Article} );

                        my %ArticleData = $ArticleBackendObject->ArticleGet(
                            %{$Article},
                        );

                        if ( $ArticleData{From} ) {
                            $Customer{UserEmail} = $ArticleData{From};
                            last ARTICLE;
                        }
                    }
                }
            }

            # check if customer email is valid
            if (
                $Customer{UserEmail}
                && !$CheckItemObject->CheckEmail( Address => $Customer{UserEmail} )
                )
            {
                $Customer{UserEmail} = '';
            }

            # if we still have no UserEmail, drop an error
            if ( !$Customer{UserEmail} ) {
                my $Success = $TicketObject->HistoryAdd(
                    TicketID     => $TicketID,
                    CreateUserID => $Param{UserID},
                    HistoryType  => 'Misc',
                    Name =>
                        "MasterTicket: no customer email found, send no master message to customer.",
                );
                if ( !$Success ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => 'System was unable to add a new history entry (no customer email found',
                    );
                }
                next TICKETID;
            }

            # set the new To for ArticleSend
            $Article{To} = $Customer{UserEmail};

            # rebuild subject
            $ConfigObject->Set(
                Key   => 'Ticket::SubjectCleanAllNumbers',
                Value => 1,
            );
            my $Subject = $TicketObject->TicketSubjectBuild(
                TicketNumber => $TicketSlave{TicketNumber},
                Subject      => $Article{Subject} || '',
            );

            # exchange Customer from MasterTicket for the one into the SlaveTicket
            my $ReplaceOnCommunicationChannels
                = $ConfigObject->Get('ReplaceCustomerRealNameOnSlaveArticleCommunicationChannels');
            my $ChannelName = $ArticleBackendObject->ChannelNameGet();

            if (
                defined $ReplaceOnCommunicationChannels->{$ChannelName} &&
                $ReplaceOnCommunicationChannels->{$ChannelName} eq '1'
                )
            {
                if ($FirstSlaveTicket) {
                    $FirstSlaveTicket = 0;
                    $TmpArticleBody   = $Article{Body};
                }
                else {
                    # get body from temporal in oder to get it
                    # without changes from previous slave tickets
                    $Article{Body} = $TmpArticleBody;
                }

                my $Search = $CustomerUserObject->CustomerName(
                    UserLogin => $Ticket{CustomerUserID},
                ) || '';
                my $Replace = $CustomerUserObject->CustomerName(
                    UserLogin => $TicketSlave{CustomerUserID},
                ) || '';
                if ( $Search && $Replace ) {
                    $Article{Body} =~ s{ \Q$Search\E }{$Replace}xmsg;
                }
            }

            # send article again
            $ArticleBackendObject->ArticleSend(
                %Article,
                Subject        => $Subject,
                Cc             => '',
                Bcc            => '',
                HistoryType    => 'SendAnswer',
                HistoryComment => "Sent answer to '$Article{To}' based on master ticket.",
                TicketID       => $TicketID,
                UserID         => $Param{UserID},
                Attachment     => \@Attachments,
            );
        }
        return 1;
    }

    # article create
    elsif ( $Param{Event} eq 'ArticleCreate' ) {

        my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');
        my @Articles      = $ArticleObject->ArticleList(
            TicketID => $Param{Data}->{TicketID},
        );

        return 1 if !@Articles;

        my $ArticleBackendObject = $ArticleObject->BackendForArticle(
            TicketID  => $Articles[-1]->{TicketID},
            ArticleID => $Articles[-1]->{ArticleID},
        );
        my %Article = $ArticleBackendObject->ArticleGet(
            TicketID  => $Articles[-1]->{TicketID},
            ArticleID => $Articles[-1]->{ArticleID},
        );

        # mark ticket to prevent a loop
        $TicketObject->HistoryAdd(
            TicketID     => $Param{Data}->{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: ArticleCreate",
        );

        my $ChannelName = $ArticleBackendObject->ChannelNameGet();

        # do not process email articles (already done in ArticleSend event!)
        if (
            $Article{SenderType} eq 'agent'
            && $Article{IsVisibleForCustomer}
            && $ChannelName eq 'Email'
            )
        {

            return 1;
        }

        # set the same state, but only for notes
        if ( $ChannelName ne 'Internal' ) {

            return 1;
        }

        # perform action on linked tickets
        TICKETID:
        for my $TicketID (@TicketIDs) {
            my $CheckSuccess = $Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: ArticleCreate',
            );
            next TICKETID if !$CheckSuccess;

            # article create
            $ArticleBackendObject->ArticleCreate(
                %Article,
                HistoryType    => 'AddNote',
                HistoryComment => 'Added article based on master ticket.',
                TicketID       => $TicketID,
                UserID         => $Param{UserID},
            );
        }

        return 1;
    }

    # state action
    elsif ( $Param{Event} eq 'TicketStateUpdate' ) {

        # mark ticket to prevent a loop
        $TicketObject->HistoryAdd(
            TicketID     => $Param{Data}->{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketStateUpdate",
        );

        # perform action on linked tickets
        TICKETID:
        for my $TicketID (@TicketIDs) {
            my $CheckSuccess = $Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketStateUpdate',
            );
            next TICKETID if !$CheckSuccess;

            # set the same state
            $TicketObject->TicketStateSet(
                StateID  => $Ticket{StateID},
                TicketID => $TicketID,
                UserID   => $Param{UserID},
            );
        }

        return 1;
    }

    # set pending time
    elsif ( $Param{Event} eq 'TicketPendingTimeUpdate' ) {

        # mark ticket to prevent a loop
        $TicketObject->HistoryAdd(
            TicketID     => $Param{Data}->{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketPendingTimeUpdate",
        );

        # perform action on linked tickets
        TICKETID:
        for my $TicketID (@TicketIDs) {
            my $CheckSuccess = $Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketPendingTimeUpdate',
            );

            next TICKETID if !$CheckSuccess;

            # set the same pending time
            my $TimeStamp = '0000-00-00 00:00:00';
            if ( $Ticket{RealTillTimeNotUsed} ) {
                my $DateTimeObject = $Kernel::OM->Create(
                    'Kernel::System::DateTime',
                    ObjectParams => {
                        Epoch => $Ticket{RealTillTimeNotUsed},
                    }
                );

                $TimeStamp = $DateTimeObject->ToString();
            }
            $TicketObject->TicketPendingTimeSet(
                String   => $TimeStamp,
                TicketID => $TicketID,
                UserID   => $Param{UserID},
            );
        }
        return 1;
    }

    # priority action
    elsif ( $Param{Event} eq 'TicketPriorityUpdate' ) {

        # mark ticket to prevent a loop
        $TicketObject->HistoryAdd(
            TicketID     => $Param{Data}->{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketPriorityUpdate",
        );

        # perform action on linked tickets
        TICKETID:
        for my $TicketID (@TicketIDs) {
            my $CheckSuccess = $Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketPriorityUpdate',
            );
            next TICKETID if !$CheckSuccess;

            # set the same state
            $TicketObject->TicketPrioritySet(
                TicketID   => $TicketID,
                PriorityID => $Ticket{PriorityID},
                UserID     => $Param{UserID},
            );
        }
        return 1;
    }

    # owner action
    elsif ( $Param{Event} eq 'TicketOwnerUpdate' ) {

        # mark ticket to prevent a loop
        $TicketObject->HistoryAdd(
            TicketID     => $Param{Data}->{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketOwnerUpdate",
        );

        # perform action on linked tickets
        TICKETID:
        for my $TicketID (@TicketIDs) {
            my $CheckSuccess = $Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketOwnerUpdate',
            );
            next TICKETID if !$CheckSuccess;

            # set the same state
            $TicketObject->TicketOwnerSet(
                TicketID           => $TicketID,
                NewUserID          => $Ticket{OwnerID},
                SendNoNotification => 0,
                UserID             => $Param{UserID},
            );
        }
        return 1;
    }

    # responsible action
    elsif ( $Param{Event} eq 'TicketResponsibleUpdate' ) {

        # mark ticket to prevent a loop
        $TicketObject->HistoryAdd(
            TicketID     => $Param{Data}->{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketResponsibleUpdate",
        );

        # perform action on linked tickets
        TICKETID:
        for my $TicketID (@TicketIDs) {
            my $CheckSuccess = $Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketResponsibleUpdate',
            );
            next TICKETID if !$CheckSuccess;

            # set the same state
            $TicketObject->TicketResponsibleSet(
                TicketID           => $TicketID,
                NewUserID          => $Ticket{ResponsibleID},
                SendNoNotification => 0,
                UserID             => $Param{UserID},
            );
        }
        return 1;
    }

    # unlock/lock action
    elsif ( $Param{Event} eq 'TicketLockUpdate' ) {

        # mark ticket to prevent a loop
        $TicketObject->HistoryAdd(
            TicketID     => $Param{Data}->{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketLockUpdate",
        );

        # perform action on linked tickets
        TICKETID:
        for my $TicketID (@TicketIDs) {
            my $CheckSuccess = $Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketLockUpdate',
            );
            next TICKETID if !$CheckSuccess;

            # set the same state
            $TicketObject->TicketLockSet(
                Lock               => $Ticket{Lock},
                TicketID           => $TicketID,
                SendNoNotification => 1,
                UserID             => $Param{UserID},
            );
        }
        return 1;
    }
    return 1;
}

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

    # check needed stuff
    for my $Needed (qw(TicketID String UserID)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Needed!"
            );

            return;
        }
    }

    # get ticket object
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    my @Lines = $TicketObject->HistoryGet(
        TicketID => $Param{TicketID},
        UserID   => $Param{UserID},
    );

    # get time object
    my $CurrentDateTimeObject = $Kernel::OM->Create('Kernel::System::DateTime');

    for my $Data ( reverse @Lines ) {
        if ( $Data->{HistoryType} eq 'Misc' && $Data->{Name} eq $Param{String} ) {
            my $DateTimeObject = $Kernel::OM->Create(
                'Kernel::System::DateTime',
                ObjectParams => {
                    String => $Data->{CreateTime},
                }
            );
            $DateTimeObject->Add(
                Seconds => 15,
            );
            if ( $DateTimeObject > $CurrentDateTimeObject ) {
                $TicketObject->HistoryAdd(
                    TicketID     => $Param{TicketID},
                    CreateUserID => $Param{UserID},
                    HistoryType  => 'Misc',
                    Name         => "MasterTicketLoop: stopped",
                );

                return;
            }
        }
    }
    return 1;
}

=head2 _ArticleHistoryTypeGiven()

Check if history type for article is given

    my $IsHistoryType = $EventObject->_ArticleHistoryTypeGiven(
        TicketID    => $TicketID,
        ArticleID   => $ArticleID,
        HistoryType => 'Forward',
        UserID      => $UserID,
    );

Returns

   my $IsHistoryType = 1;

=cut

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

    # check needed stuff
    for my $Needed (qw(TicketID ArticleID HistoryType UserID)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Needed!"
            );
            return;
        }
    }

    # get ticket object
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    my $TicketID    = $Param{TicketID};
    my $ArticleID   = $Param{ArticleID};
    my $HistoryType = $Param{HistoryType};
    my $UserID      = $Param{UserID};

    my @Lines = $TicketObject->HistoryGet(
        TicketID => $TicketID,
        UserID   => $UserID,
    );

    my $Matched = 0;

    return $Matched if !IsArrayRefWithData( \@Lines );

    HISTORYDATA:
    for my $HistoryData ( reverse @Lines ) {
        next HISTORYDATA if $HistoryData->{ArticleID} != $ArticleID;
        next HISTORYDATA if $HistoryData->{HistoryType} ne $HistoryType;

        $Matched = 1;

        last HISTORYDATA;
    }

    return $Matched;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use Kernel::System::VariableCheck qw(:all);

# get needed objects
my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');
my $ParamObject        = $Kernel::OM->Get('Kernel::System::Web::Request');

$Kernel::OM->ObjectParamAdd(
    'Kernel::Output::HTML::Layout' => {
        UserID => 1,
    },
);
my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

# get helper object
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $HelperObject = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# get master/slave dynamic field data
my $MasterSlaveDynamicField     = $ConfigObject->Get('MasterSlave::DynamicField');
my $MasterSlaveDynamicFieldData = $DynamicFieldObject->DynamicFieldGet(
    Name => $MasterSlaveDynamicField,
);

my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

# find all current open master slave tickets
my @TicketIDs = $TicketObject->TicketSearch(
    Result => 'ARRAY',

    # master slave dynamic field
    'DynamicField_' . $MasterSlaveDynamicField => {
        Equals => 'Master',
    },

    StateType  => 'Open',
    UserID     => 1,
    Permission => 'ro',
);

# set tickets to removed so they are not fond later in the test cases
#    the tickets will be restored automatically at the end of the test
#    due to the RestoreDatabase option in helper object
for my $TicketID (@TicketIDs) {
    my $Success = $TicketObject->TicketStateSet(
        State    => 'removed',
        TicketID => $TicketID,
        UserID   => 1,
    );
    $Self->True(
        $Success,
        "Temporary set Master Ticket: $TicketID to removed",
    );
}

# define tests
my @Tests = (
    {
        Name   => 'MasterSlave - Possible Values Filter',
        Config => {
            DynamicFieldConfig   => $MasterSlaveDynamicFieldData,
            PossibleValuesFilter => {
                Master => 'New Master Ticket',
            },
            LayoutObject => $LayoutObject,
            ParamObject  => $ParamObject,
        },
        ExpectedResults => {
            Field =>
                '<select class="DynamicFieldText Modernize" id="DynamicField_MasterSlave" name="DynamicField_MasterSlave" size="1">
  <option value="Master">New Master Ticket</option>
</select>
',
            Label => '<label id="LabelDynamicField_MasterSlave" for="DynamicField_MasterSlave">
Master Ticket:
</label>
'
        },
    },
    {
        Name   => 'UnsetMasterSlave - Possible Values Filter',
        Config => {
            DynamicFieldConfig   => $MasterSlaveDynamicFieldData,
            PossibleValuesFilter => {
                Master      => 'New Master Ticket',
                UnsetMaster => 'Unset Master Tickets',
                UnsetSlave  => 'Unset Slave Tickets',
            },
            LayoutObject => $LayoutObject,
            ParamObject  => $ParamObject,
        },
        ExpectedResults => {
            Field =>
                '<select class="DynamicFieldText Modernize" id="DynamicField_MasterSlave" name="DynamicField_MasterSlave" size="1">
  <option value="Master">New Master Ticket</option>
  <option value="UnsetMaster">Unset Master Tickets</option>
  <option value="UnsetSlave">Unset Slave Tickets</option>
</select>
',
            Label => '<label id="LabelDynamicField_MasterSlave" for="DynamicField_MasterSlave">
Master Ticket:
</label>
'
        },
    },
    {
        Name   => 'MasterSlave: No value ',
        Config => {
            DynamicFieldConfig => $MasterSlaveDynamicFieldData,
            LayoutObject       => $LayoutObject,
            ParamObject        => $ParamObject,
            Class              => 'MyClass',
            UseDefaultValue    => 0,
        },
        ExpectedResults => {
            Field => <<"EOF" . '</select>',
<select class="DynamicFieldText Modernize MyClass" id="DynamicField_MasterSlave" name="DynamicField_MasterSlave" size="1">
  <option value="">-</option>
  <option value="Master">New Master Ticket</option>
EOF
            Label => '<label id="LabelDynamicField_MasterSlave" for="DynamicField_MasterSlave">
Master Ticket:
</label>
'
        },
    },
    {
        Name   => 'MasterSlave: No value / Default',
        Config => {
            DynamicFieldConfig => $MasterSlaveDynamicFieldData,
            LayoutObject       => $LayoutObject,
            ParamObject        => $ParamObject,
            Class              => 'MyClass',
        },
        ExpectedResults => {
            Field => <<"EOF" . '</select>',
<select class="DynamicFieldText Modernize MyClass" id="DynamicField_MasterSlave" name="DynamicField_MasterSlave" size="1">
  <option value="" selected="selected">-</option>
  <option value="Master">New Master Ticket</option>
EOF
            Label => '<label id="LabelDynamicField_MasterSlave" for="DynamicField_MasterSlave">
Master Ticket:
</label>
'
        },
    },
    {
        Name   => 'MasterSlave: Value direct',
        Config => {
            DynamicFieldConfig => $MasterSlaveDynamicFieldData,
            LayoutObject       => $LayoutObject,
            ParamObject        => $ParamObject,
            Value              => 'Master',
            Class              => 'MyClass',
            UseDefaultValue    => 0,
        },
        ExpectedResults => {
            Field => <<"EOF" . '</select>',
<select class="DynamicFieldText Modernize MyClass" id="DynamicField_MasterSlave" name="DynamicField_MasterSlave" size="1">
  <option value="">-</option>
  <option value="Master" selected="selected">New Master Ticket</option>
EOF
            Label => '<label id="LabelDynamicField_MasterSlave" for="DynamicField_MasterSlave">
Master Ticket:
</label>
'
        },
    },
    {
        Name   => 'MasterSlave: Mandatory',
        Config => {
            DynamicFieldConfig => $MasterSlaveDynamicFieldData,
            LayoutObject       => $LayoutObject,
            ParamObject        => $ParamObject,
            Value              => 'Master',
            Class              => 'MyClass',
            UseDefaultValue    => 0,
            Mandatory          => 1,
        },
        ExpectedResults => {
            Field =>
                '<select class="DynamicFieldText Modernize MyClass Validate_Required" id="DynamicField_MasterSlave" name="DynamicField_MasterSlave" size="1">
  <option value="">-</option>
  <option value="Master" selected="selected">New Master Ticket</option>
</select>
<div id="DynamicField_MasterSlaveError" class="TooltipErrorMessage">
    <p>
        This field is required.
    </p>
</div>
',
            Label => '<label id="LabelDynamicField_MasterSlave" for="DynamicField_MasterSlave" class="Mandatory">
    <span class="Marker">*</span>
Master Ticket:
</label>
'
        },
    },
    {
        Name   => 'MasterSlave: Server Error',
        Config => {
            DynamicFieldConfig => $MasterSlaveDynamicFieldData,
            LayoutObject       => $LayoutObject,
            ParamObject        => $ParamObject,
            Value              => 'Master',
            Class              => 'MyClass',
            UseDefaultValue    => 0,
            ServerError        => 1,
            ErrorMessage       => 'This is an error.'
        },
        ExpectedResults => {
            Field =>
                '<select class="DynamicFieldText Modernize MyClass ServerError" id="DynamicField_MasterSlave" name="DynamicField_MasterSlave" size="1">
  <option value="">-</option>
  <option value="Master" selected="selected">New Master Ticket</option>
</select>
<div id="DynamicField_MasterSlaveServerError" class="TooltipErrorMessage">
    <p>
        This is an error.
    </p>
</div>
',
            Label => '<label id="LabelDynamicField_MasterSlave" for="DynamicField_MasterSlave">
Master Ticket:
</label>
'
        },
    },
);

# ------------------------------------------------------------ #
# execute tests
# ------------------------------------------------------------ #
my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

for my $Test (@Tests) {

    my $FieldHTML = $DynamicFieldBackendObject->EditFieldRender( %{ $Test->{Config} } );

    # heredocs always have the newline, even if it is not expected
    if ( $FieldHTML->{Field} !~ m{\n$} ) {
        chomp $Test->{ExpectedResults}->{Field};
    }

    $Self->IsDeeply(
        $FieldHTML,
        $Test->{ExpectedResults},
        "$Test->{Name} | EditFieldRender()",
    );
}

# Cleanup is done by RestoreDatabase.

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));
use Kernel::System::VariableCheck qw(IsHashRefWithData);

# get needed objects
my $TicketObject              = $Kernel::OM->Get('Kernel::System::Ticket');
my $LinkObject                = $Kernel::OM->Get('Kernel::System::LinkObject');
my $DynamicFieldObject        = $Kernel::OM->Get('Kernel::System::DynamicField');
my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
my $ConfigObject              = $Kernel::OM->Get('Kernel::Config');

# get helper object
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $HelperObject = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# get master/slave dynamic field data
my $MasterSlaveDynamicField     = $ConfigObject->Get('MasterSlave::DynamicField');
my $MasterSlaveDynamicFieldData = $DynamicFieldObject->DynamicFieldGet(
    Name => $MasterSlaveDynamicField,
);

# get random ID
my $RandomID = $HelperObject->GetRandomID();

# create test tickets
my @TicketIDs;
my @TicketNumbers;
for my $Ticket ( 1 .. 3 ) {
    my $TicketNumber = $TicketObject->TicketCreateNumber();
    my $TicketID     = $TicketObject->TicketCreate(
        TN           => $TicketNumber,
        Title        => 'Unit test ticket ' . $RandomID . ' ' . $Ticket,
        Queue        => 'Raw',
        Lock         => 'unlock',
        Priority     => '3 normal',
        State        => 'new',
        CustomerNo   => '123465',
        CustomerUser => 'customer@example.com',
        OwnerID      => 1,
        UserID       => 1,
    );
    $Self->True(
        $TicketID,
        "TicketCreate() Ticket ID $TicketID - TN: $TicketNumber",
    );
    push @TicketIDs,     $TicketID;
    push @TicketNumbers, $TicketNumber;
}

# ------------------------------------------------------------ #
# test master/slave ticket value set
# ------------------------------------------------------------ #

# set first test ticket as master ticket
my $Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[0],
    Value              => 'Master',
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[0] DynamicField $MasterSlaveDynamicField updated as MasterTicket",
);

# set second test ticket as slave ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[1],
    Value              => "SlaveOf:$TicketNumbers[0]",
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[1] DynamicField $MasterSlaveDynamicField updated as SlaveOf:$TicketNumbers[0]",
);

# verify there is parent-child link between master/slave tickets
my %LinkKeyList = $LinkObject->LinkKeyList(
    Object1   => 'Ticket',
    Key1      => $TicketIDs[0],
    Object2   => 'Ticket',
    State     => 'Valid',
    Type      => 'ParentChild',
    Direction => 'Both',
    UserID    => 1,
);

$Self->True(
    IsHashRefWithData( \%LinkKeyList ) ? 1 : 0,
    "LinkKeyList() Master/Slave link found - Ticket ID $TicketIDs[0] and $TicketIDs[1]",
);

# ------------------------------------------------------------ #
# test UnsetMaster|UnsetSlave
# ------------------------------------------------------------ #

# enable the MasterSlave::KeepParentChildAfterUnset sysconfig
$Success = $ConfigObject->Set(
    Key   => 'MasterSlave::KeepParentChildAfterUnset',
    Value => 1
);
$Self->True(
    $Success,
    "Set() sysconfig KeepParentChildAfterUnset - enabled",
);

# set second ticket as slave ticket again
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[1],
    Value              => "SlaveOf:$TicketNumbers[0]",
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[1] DynamicField $MasterSlaveDynamicField updated again as SlaveOf:$TicketNumbers[0]",
);

# verify there is parent-child link between master/slave tickets
%LinkKeyList = $LinkObject->LinkKeyList(
    Object1   => 'Ticket',
    Key1      => $TicketIDs[0],
    Object2   => 'Ticket',
    State     => 'Valid',
    Type      => 'ParentChild',
    Direction => 'Target',
    UserID    => 1,
);
$Self->True(
    IsHashRefWithData( \%LinkKeyList ) ? 1 : 0,
    "LinkKeyList() Master/Slave link found - Ticket ID $TicketIDs[0] and $TicketIDs[1]",
);

# UnsetMaster value from first ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[0],
    Value              => 'UnsetMaster',
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[0] DynamicField $MasterSlaveDynamicField updated as UnsetMaster",
);

# UnsetSlave value from second ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[1],
    Value              => 'UnsetSlave',
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[1] DynamicField $MasterSlaveDynamicField updated as UnsetSlave",
);

# verify there is still parent-child link between two tickets
%LinkKeyList = $LinkObject->LinkKeyList(
    Object1   => 'Ticket',
    Key1      => $TicketIDs[0],
    Object2   => 'Ticket',
    State     => 'Valid',
    Type      => 'ParentChild',
    Direction => 'Target',
    UserID    => 1,
);
$Self->True(
    IsHashRefWithData( \%LinkKeyList ) ? 1 : 0,
    "LinkKeyList() Master/Slave link found - Ticket ID $TicketIDs[0] and $TicketIDs[1] - KeepParentChildAfterUnset sysconfig enabled",
);

# disable the MasterSlave::KeepParentChildAfterUnset sysconfig
$Success = $ConfigObject->Set(
    Key   => 'MasterSlave::KeepParentChildAfterUnset',
    Value => 0
);
$Self->True(
    $Success,
    "Set() sysconfig KeepParentChildAfterUnset - disabled",
);

# remove parent-child link between two tickets
$Success = $LinkObject->LinkDeleteAll(
    Object => 'Ticket',
    Key    => $TicketIDs[0],
    UserID => 1,
);
$Self->True(
    $Success,
    "LinkDeleteAll() parent-child link for ticket ID $TicketIDs[0] - removed",
);

# set first test ticket as master ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[0],
    Value              => 'Master',
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[0] DynamicField $MasterSlaveDynamicField updated as MasterTicket",
);

# set second test ticket as slave ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[1],
    Value              => "SlaveOf:$TicketNumbers[0]",
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[1] DynamicField $MasterSlaveDynamicField updated as SlaveOf:$TicketNumbers[0]",
);

# verify there is still parent-child link between two tickets
%LinkKeyList = $LinkObject->LinkKeyList(
    Object1   => 'Ticket',
    Key1      => $TicketIDs[0],
    Object2   => 'Ticket',
    State     => 'Valid',
    Type      => 'ParentChild',
    Direction => 'Target',
    UserID    => 1,
);
$Self->True(
    IsHashRefWithData( \%LinkKeyList ) ? 1 : 0,
    "LinkKeyList() Master/Slave link found - Ticket ID $TicketIDs[0] and $TicketIDs[1]",
);

# UnsetMaster value from first ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[0],
    Value              => 'UnsetMaster',
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[0] DynamicField $MasterSlaveDynamicField updated as UnsetMaster",
);

# UnsetSlave value from second ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[1],
    Value              => 'UnsetSlave',
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[1] DynamicField $MasterSlaveDynamicField updated as UnsetSlave",
);

# verify there is no more parent-child link between two tickets
%LinkKeyList = $LinkObject->LinkKeyList(
    Object1   => 'Ticket',
    Key1      => $TicketIDs[0],
    Object2   => 'Ticket',
    State     => 'Valid',
    Type      => 'ParentChild',
    Direction => 'Target',
    UserID    => 1,
);
$Self->True(
    !IsHashRefWithData( \%LinkKeyList ),
    "LinkKeyList() Master/Slave link removed - Ticket ID $TicketIDs[0] and $TicketIDs[1] - KeepParentChildAfterUnset sysconfig disabled",
);

# ------------------------------------------------------------ #
# test update Master|Slave field
# ------------------------------------------------------------ #

# enable the MasterSlave::KeepParentChildAfterUpdate sysconfig
$Success = $ConfigObject->Set(
    Key   => 'MasterSlave::KeepParentChildAfterUpdate',
    Value => 1
);
$Self->True(
    $Success,
    "Set() sysconfig KeepParentChildAfterUpdate - enabled",
);

# set first test ticket as master ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[0],
    Value              => 'Master',
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[0] DynamicField $MasterSlaveDynamicField updated as MasterTicket",
);

# set second test ticket as master ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[1],
    Value              => "Master",
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[1] DynamicField $MasterSlaveDynamicField updated as MasterTicket",
);

# set third test ticket as slave of second ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[2],
    Value              => "SlaveOf:$TicketNumbers[1]",
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[2] DynamicField $MasterSlaveDynamicField updated as SlaveOf:$TicketNumbers[1]",
);

# verify there is parent-child link between two tickets
%LinkKeyList = $LinkObject->LinkKeyList(
    Object1   => 'Ticket',
    Key1      => $TicketIDs[1],
    Object2   => 'Ticket',
    State     => 'Valid',
    Type      => 'ParentChild',
    Direction => 'Target',
    UserID    => 1,
);
$Self->True(
    IsHashRefWithData( \%LinkKeyList ) ? 1 : 0,
    "LinkKeyList() Master/Slave link found - Ticket ID $TicketIDs[1] and $TicketIDs[2]",
);

# set second ticket as slave of first ticket
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $MasterSlaveDynamicFieldData,
    FieldID            => $MasterSlaveDynamicFieldData->{ID},
    ObjectID           => $TicketIDs[1],
    Value              => "SlaveOf:$TicketNumbers[0]",
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $TicketIDs[1] DynamicField $MasterSlaveDynamicField updated as SlaveOf:$TicketNumbers[0]",
);

# verify there parent-child-parent link between three tickets
%LinkKeyList = $LinkObject->LinkKeyList(
    Object1   => 'Ticket',
    Key1      => $TicketIDs[1],
    Object2   => 'Ticket',
    State     => 'Valid',
    Type      => 'ParentChild',
    Direction => 'Both',
    UserID    => 1,
);
$Self->True(
    IsHashRefWithData( \%LinkKeyList ),
    "LinkKeyList() Master/Slave link found - Ticket ID $TicketIDs[0], $TicketIDs[1] and $TicketIDs[2] - KeepParentChildAfterUpdate sysconfig enabled",
);

# Cleanup is done by RestoreDatabase.

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7CgpteSAkU2VsZW5pdW0gPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OlNlbGVuaXVtJyk7CgokU2VsZW5pdW0tPlJ1blRlc3QoCiAgICBzdWIgewoKICAgICAgICBteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCiAgICAgICAgIyBFbmFibGUgdGhlIFVuc2V0TWFzdGVyU2xhdmUgY29uZmlnLgogICAgICAgICRIZWxwZXItPkNvbmZpZ1NldHRpbmdDaGFuZ2UoCiAgICAgICAgICAgIEtleSAgID0+ICdNYXN0ZXJTbGF2ZTo6VW5zZXRNYXN0ZXJTbGF2ZScsCiAgICAgICAgICAgIFZhbHVlID0+IDEsCiAgICAgICAgKTsKCiAgICAgICAgIyBDcmVhdGUgdGVzdCB1c2VyIGFuZCBsb2cgaW4uCiAgICAgICAgbXkgJFRlc3RVc2VyTG9naW4gPSAkSGVscGVyLT5UZXN0VXNlckNyZWF0ZSgKICAgICAgICAgICAgR3JvdXBzID0+IFsgJ2FkbWluJywgJ3VzZXJzJyBdLAogICAgICAgICkgfHwgZGllICJEaWQgbm90IGdldCB0ZXN0IHVzZXIiOwoKICAgICAgICAkU2VsZW5pdW0tPkxvZ2luKAogICAgICAgICAgICBUeXBlICAgICA9PiAnQWdlbnQnLAogICAgICAgICAgICBVc2VyICAgICA9PiAkVGVzdFVzZXJMb2dpbiwKICAgICAgICAgICAgUGFzc3dvcmQgPT4gJFRlc3RVc2VyTG9naW4sCiAgICAgICAgKTsKCiAgICAgICAgbXkgJFNjcmlwdEFsaWFzID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCdTY3JpcHRBbGlhcycpOwoKICAgICAgICAjIE5hdmlnYXRlIHRvIEFkbWluR2VuZXJpY0FnZW50IHNjcmVlbiBmb3IgbmV3IGpvYiBhZGRpbmcuCiAgICAgICAgJFNlbGVuaXVtLT5WZXJpZmllZEdldCgiJHtTY3JpcHRBbGlhc31pbmRleC5wbD9BY3Rpb249QWRtaW5HZW5lcmljQWdlbnQ7U3ViYWN0aW9uPVVwZGF0ZSIpOwoKICAgICAgICAkU2VsZW5pdW0tPldhaXRGb3IoIEphdmFTY3JpcHQgPT4gInJldHVybiB0eXBlb2YoXCQpID09PSAnZnVuY3Rpb24nOyIgKTsKCiAgICAgICAgIyBFeHBhbmQgYXBwcm9wcmlhdGUgd2lkZ2V0LgogICAgICAgICRTZWxlbml1bS0+ZXhlY3V0ZV9zY3JpcHQoCiAgICAgICAgICAgICJcJCgnLldpZGdldFNpbXBsZS5Db2xsYXBzZWQ6Y29udGFpbnMoXCJVcGRhdGUvQWRkIFRpY2tldCBBdHRyaWJ1dGVzXCIpIC5XaWRnZXRBY3Rpb24uVG9nZ2xlIGEnKS50cmlnZ2VyKCdjbGljaycpOyIKICAgICAgICApOwogICAgICAgICRTZWxlbml1bS0+V2FpdEZvcigKICAgICAgICAgICAgSmF2YVNjcmlwdCA9PiAicmV0dXJuIFwkKCcuV2lkZ2V0U2ltcGxlLkV4cGFuZGVkJykubGVuZ3RoOyIKICAgICAgICApOwoKICAgICAgICAjIEFkZCBhcHByb3ByaWF0ZSBkeW5hbWljIGZpZWxkLgogICAgICAgICRTZWxlbml1bS0+ZXhlY3V0ZV9zY3JpcHQoCiAgICAgICAgICAgICJcJCgnI0FkZE5ld0R5bmFtaWNGaWVsZHMnKS52YWwoJ0R5bmFtaWNGaWVsZF9NYXN0ZXJTbGF2ZScpLnRyaWdnZXIoJ3JlZHJhdy5JbnB1dEZpZWxkJykudHJpZ2dlcignY2hhbmdlJyk7IgogICAgICAgICk7CiAgICAgICAgJFNlbGVuaXVtLT5XYWl0Rm9yKAogICAgICAgICAgICBKYXZhU2NyaXB0ID0+ICJyZXR1cm4gXCQoJyNTZWxlY3RlZE5ld0R5bmFtaWNGaWVsZHMgI0R5bmFtaWNGaWVsZF9NYXN0ZXJTbGF2ZScpLmxlbmd0aDsiCiAgICAgICAgKTsKCiAgICAgICAgIyBWZXJpZnkgcG9zc2libGUgTWFzdGVyU2xhdmUgdmFsdWVzICdVbnNldE1hc3RlcicgYW5kICdVbnNldFNsYXZlJyBpbiAnVXBkYXRlL0FkZCBUaWNrZXQgQXR0cmlidXRlcycgd2lkZ2V0LgogICAgICAgICMgU2VlIGJ1ZyMxNDc3OCAoaHR0cHM6Ly9idWdzLm90cnMub3JnL3Nob3dfYnVnLmNnaT9pZD0xNDc3OCkuCiAgICAgICAgZm9yIG15ICRPcHRpb24gKHF3KFVuc2V0TWFzdGVyIFVuc2V0U2xhdmUpKSB7CiAgICAgICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAgICAgJFNlbGVuaXVtLT5leGVjdXRlX3NjcmlwdCgicmV0dXJuIFwkKCcjRHluYW1pY0ZpZWxkX01hc3RlclNsYXZlIG9wdGlvblt2YWx1ZT0kT3B0aW9uXScpLmxlbmd0aDsiKSwKICAgICAgICAgICAgICAgICJNYXN0ZXJTbGF2ZSBvcHRpb24gJyRPcHRpb24nIGlzIGF2YWlsYWJsZS4iCiAgICAgICAgICAgICk7CiAgICAgICAgfQoKICAgICAgICAjIERpc2FibGUgdGhlIFVuc2V0TWFzdGVyU2xhdmUgY29uZmlnLgogICAgICAgICRIZWxwZXItPkNvbmZpZ1NldHRpbmdDaGFuZ2UoCiAgICAgICAgICAgIEtleSAgID0+ICdNYXN0ZXJTbGF2ZTo6VW5zZXRNYXN0ZXJTbGF2ZScsCiAgICAgICAgICAgIFZhbHVlID0+IDAsCiAgICAgICAgKTsKCiAgICAgICAgIyBSZWZyZXNoIHNjcmVlbi4KICAgICAgICAkU2VsZW5pdW0tPlZlcmlmaWVkUmVmcmVzaCgpOwoKICAgICAgICAjIEV4cGFuZCBhcHByb3ByaWF0ZSB3aWRnZXQuCiAgICAgICAgJFNlbGVuaXVtLT5leGVjdXRlX3NjcmlwdCgKICAgICAgICAgICAgIlwkKCcuV2lkZ2V0U2ltcGxlLkNvbGxhcHNlZDpjb250YWlucyhcIlVwZGF0ZS9BZGQgVGlja2V0IEF0dHJpYnV0ZXNcIikgLldpZGdldEFjdGlvbi5Ub2dnbGUgYScpLnRyaWdnZXIoJ2NsaWNrJyk7IgogICAgICAgICk7CiAgICAgICAgJFNlbGVuaXVtLT5XYWl0Rm9yKAogICAgICAgICAgICBKYXZhU2NyaXB0ID0+ICJyZXR1cm4gXCQoJy5XaWRnZXRTaW1wbGUuRXhwYW5kZWQnKS5sZW5ndGg7IgogICAgICAgICk7CgogICAgICAgICMgQWRkIGFwcHJvcHJpYXRlIGR5bmFtaWMgZmllbGQuCiAgICAgICAgJFNlbGVuaXVtLT5leGVjdXRlX3NjcmlwdCgKICAgICAgICAgICAgIlwkKCcjQWRkTmV3RHluYW1pY0ZpZWxkcycpLnZhbCgnRHluYW1pY0ZpZWxkX01hc3RlclNsYXZlJykudHJpZ2dlcigncmVkcmF3LklucHV0RmllbGQnKS50cmlnZ2VyKCdjaGFuZ2UnKTsiCiAgICAgICAgKTsKICAgICAgICAkU2VsZW5pdW0tPldhaXRGb3IoCiAgICAgICAgICAgIEphdmFTY3JpcHQgPT4gInJldHVybiBcJCgnI1NlbGVjdGVkTmV3RHluYW1pY0ZpZWxkcyAjRHluYW1pY0ZpZWxkX01hc3RlclNsYXZlJykubGVuZ3RoOyIKICAgICAgICApOwoKICAgICAgICAjIFZlcmlmeSBwb3NzaWJsZSBNYXN0ZXJTbGF2ZSB2YWx1ZXMgJ1Vuc2V0TWFzdGVyJyBhbmQgJ1Vuc2V0U2xhdmUnIGFyZSBub3QgYXZhaWxhYmxlCiAgICAgICAgIyAgIGluICdVcGRhdGUvQWRkIFRpY2tldCBBdHRyaWJ1dGVzJyB3aWRnZXQuCiAgICAgICAgZm9yIG15ICRPcHRpb24gKHF3KFVuc2V0TWFzdGVyIFVuc2V0U2xhdmUpKSB7CiAgICAgICAgICAgICRTZWxmLT5GYWxzZSgKICAgICAgICAgICAgICAgICRTZWxlbml1bS0+ZXhlY3V0ZV9zY3JpcHQoInJldHVybiBcJCgnI0R5bmFtaWNGaWVsZF9NYXN0ZXJTbGF2ZSBvcHRpb25bdmFsdWU9JE9wdGlvbl0nKS5sZW5ndGg7IiksCiAgICAgICAgICAgICAgICAiTWFzdGVyU2xhdmUgb3B0aW9uICckT3B0aW9uJyBpcyBub3QgYXZhaWxhYmxlLiIKICAgICAgICAgICAgKTsKICAgICAgICB9CiAgICB9Cik7CgoxOwo=
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

## no critic (Modules::RequireExplicitPackage)
use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

# Create local function for wait on AJAX update.
my $WaitForAJAX = sub {

    Time::HiRes::sleep(0.2);

    $Selenium->WaitFor(
        JavaScript =>
            'return typeof($) === "function" && !$("span.AJAXLoader:visible").length;'
    );
};

$Selenium->RunTest(
    sub {

        my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

        # Enable the advanced MasterSlave.
        $Helper->ConfigSettingChange(
            Key   => 'MasterSlave::AdvancedEnabled',
            Value => 1,
        );

        # Do not check RichText.
        $Helper->ConfigSettingChange(
            Key   => 'Frontend::RichText',
            Value => 0,
        );

        # Do not check service and type.
        $Helper->ConfigSettingChange(
            Key   => 'Ticket::Service',
            Value => 0
        );
        $Helper->ConfigSettingChange(
            Key   => 'Ticket::Type',
            Value => 0,
        );
        $Helper->ConfigSettingChange(
            Valid => 1,
            Key   => 'CheckEmailAddresses',
            Value => 0,
        );

        # Do not send emails.
        $Helper->ConfigSettingChange(
            Key   => 'SendmailModule',
            Value => 'Kernel::System::Email::Test',
        );

        # Add test customer for testing.
        my $TestCustomerLoginPhone = $Helper->TestCustomerUserCreate();

        # Create test user and login.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'users' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $Kernel::OM->Get('Kernel::Config')->Get('ScriptAlias');

        # Navigate to AgentTicketPhone screen.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketPhone");

        $Selenium->WaitForjQueryEventBound(
            CSSSelector => '#Dest',
            Event       => 'change',
        );

        $Selenium->execute_script("\$('#Dest').val('2||Raw').trigger('redraw.InputField').trigger('change');");

        # Wait for AJAX to finish.
        $WaitForAJAX->();

        my $MasterTicketSubject = "Master Ticket";
        $Selenium->find_element( "#FromCustomer", 'css' )->send_keys($TestCustomerLoginPhone);
        $Selenium->WaitFor( JavaScript => 'return typeof($) === "function" && $("li.ui-menu-item:visible").length;' );

        $Selenium->execute_script(
            "\$('li.ui-menu-item:nth-child(1) a').trigger('click');",
        );

        # Wait for AJAX to finish.
        $WaitForAJAX->();

        $Selenium->find_element( "#Subject",  'css' )->send_keys($MasterTicketSubject);
        $Selenium->find_element( "#RichText", 'css' )->send_keys('Selenium body test');
        $Selenium->execute_script(
            "\$('#DynamicField_MasterSlave').val('Master').trigger('redraw.InputField').trigger('change');"
        );

        # Wait for AJAX to finish.
        $WaitForAJAX->();

        $Selenium->execute_script(
            "\$('#submitRichText')[0].scrollIntoView(true);",
        );
        $Self->True(
            $Selenium->execute_script(
                "return \$('#submitRichText').length;"
            ),
            "Element '#submitRichText' is found in the screen"
        );
        $Selenium->find_element( "#submitRichText", 'css' )->VerifiedClick();

        my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

        # Get master test phone ticket data.
        my ( $MasterTicketID, $MasterTicketNumber ) = $TicketObject->TicketSearch(
            Result            => 'HASH',
            Limit             => 1,
            CustomerUserLogin => $TestCustomerLoginPhone,
            UserID            => 1,
        );

        $Self->True(
            $MasterTicketID,
            "Master TicketID - $MasterTicketID",
        );
        $Self->True(
            $MasterTicketNumber,
            "Master TicketNumber - $MasterTicketNumber",
        );

        # Add new test customer for testing.
        my $TestCustomerLoginsEmail = $Helper->TestCustomerUserCreate();

        # Navigate to AgentTicketEmail screen.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketEmail");

        $Selenium->WaitForjQueryEventBound(
            CSSSelector => '#Dest',
            Event       => 'change',
        );

        # Create slave test email ticket.
        $Selenium->execute_script("\$('#Dest').val('2||Raw').trigger('redraw.InputField').trigger('change');");

        # Wait for AJAX to finish.
        $WaitForAJAX->();

        $Selenium->find_element( "#ToCustomer", 'css' )->send_keys($TestCustomerLoginsEmail);
        $Selenium->WaitFor( JavaScript => 'return typeof($) === "function" && $("li.ui-menu-item:visible").length;' );
        $Selenium->execute_script(
            "\$('li.ui-menu-item:nth-child(1) a').trigger('click');",
        );

        # Wait for AJAX to finish.
        $WaitForAJAX->();

        $Selenium->find_element( "#Subject",  'css' )->send_keys('Slave Ticket');
        $Selenium->find_element( "#RichText", 'css' )->send_keys('Selenium body test');
        $Selenium->execute_script(
            "\$('#DynamicField_MasterSlave').val('SlaveOf:$MasterTicketNumber').trigger('redraw.InputField').trigger('change');"
        );

        # Wait for AJAX to finish.
        $WaitForAJAX->();

        $Selenium->execute_script(
            "\$('#submitRichText')[0].scrollIntoView(true);",
        );
        $Self->True(
            $Selenium->execute_script(
                "return \$('#submitRichText').length;"
            ),
            "Element '#submitRichText' is found in the screen"
        );
        $Selenium->find_element( "#submitRichText", 'css' )->VerifiedClick();

        # get slave test email ticket data
        my ( $SlaveTicketID, $SlaveTicketNumber ) = $TicketObject->TicketSearch(
            Result            => 'HASH',
            Limit             => 1,
            CustomerUserLogin => $TestCustomerLoginsEmail,
            UserID            => 1,
        );

        $Self->True(
            $SlaveTicketID,
            "Slave TicketID - $SlaveTicketID",
        );
        $Self->True(
            $SlaveTicketNumber,
            "Slave TicketNumber - $SlaveTicketNumber",
        );

        # Navigate to ticket zoom page of created master test ticket.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketZoom;TicketID=$MasterTicketID");

        # Verify master-slave ticket link.
        $Self->True(
            index( $Selenium->get_page_source(), $SlaveTicketNumber ) > -1,
            "Slave ticket number: $SlaveTicketNumber - found",
        );

        # Navigate to history view of created master test ticket.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketHistory;TicketID=$MasterTicketID");

        # Verify dynamic field master ticket update.
        $Self->True(
            index( $Selenium->get_page_source(), 'Changed dynamic field MasterSlave from "" to "Master".' ) > -1,
            "Master dynamic field update value - found",
        );

        # Navigate to ticket zoom page of created slave test ticket.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketZoom;TicketID=$SlaveTicketID");

        # Verify slave-master ticket link
        $Self->True(
            index( $Selenium->get_page_source(), $MasterTicketNumber ) > -1,
            "Master ticket number: $MasterTicketNumber - found",
        );

        # Navigate to history view of created slave test ticket.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketHistory;TicketID=$SlaveTicketID");

        # Verify dynamic field slave ticket update.
        $Self->True(
            index( $Selenium->get_page_source(), 'Changed dynamic field MasterSlave from "" to "SlaveOf:' ) > -1,
            "Slave dynamic field update value - found",
        );

        # Delete test tickets.
        for my $TicketID ( $MasterTicketID, $SlaveTicketID ) {
            my $Success = $TicketObject->TicketDelete(
                TicketID => $TicketID,
                UserID   => 1,
            );

            # Ticket deletion could fail if apache still writes to ticket history. Try again in this case.
            if ( !$Success ) {
                sleep 3;
                $Success = $TicketObject->TicketDelete(
                    TicketID => $TicketID,
                    UserID   => 1,
                );
            }

            $Self->True(
                $Success,
                "Ticket ID $TicketID - deleted"
            );
        }

        # Make sure the cache is correct.
        $Kernel::OM->Get('Kernel::System::Cache')->CleanUp(
            Type => 'Ticket',
        );
    }
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

## no critic (Modules::RequireExplicitPackage)
use strict;
use warnings;
use utf8;

use Kernel::System::VariableCheck qw(:all);

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper       = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

        # Enable the advanced MasterSlave.
        $Helper->ConfigSettingChange(
            Key   => 'MasterSlave::AdvancedEnabled',
            Value => 1,
        );

        # Enable change the MasterSlave state of a ticket.
        $Helper->ConfigSettingChange(
            Key   => 'MasterSlave::UpdateMasterSlave',
            Value => 1,
        );

        # Do not check RichText.
        $Helper->ConfigSettingChange(
            Key   => 'Frontend::RichText',
            Value => 0,
        );

        # Do not send emails.
        $Helper->ConfigSettingChange(
            Key   => 'SendmailModule',
            Value => 'Kernel::System::Email::Test',
        );

        # Disable Type feature.
        $Helper->ConfigSettingChange(
            Valid => 1,
            Key   => 'Ticket::Type',
            Value => 0,
        );

        # Make sure InvovedAgent and InformAgent are disabled, otherwise it uses part of the visible
        # Screen making the submit button not visible and then not click-able.
        $Helper->ConfigSettingChange(
            Key   => 'Ticket::Frontend::AgentTicketMasterSlave###InvolvedAgent',
            Value => 0,
        );
        $Helper->ConfigSettingChange(
            Key   => 'Ticket::Frontend::AgentTicketMasterSlave###InformAgent',
            Value => 0,
        );

        # Make sure Note is enabled in AgentTicketMasterSlave screen.
        $Helper->ConfigSettingChange(
            Key   => 'Ticket::Frontend::AgentTicketMasterSlave###Note',
            Value => 1,
        );

        # Create test user.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'users' ],
        ) || die "Did not get test user";

        # Get test user ID.
        my $TestUserID = $Kernel::OM->Get('Kernel::System::User')->UserLookup(
            UserLogin => $TestUserLogin,
        );

        # Create two test tickets.
        my @TicketIDs;
        my @TicketNumbers;
        for my $TicketCreate ( 1 .. 2 ) {
            my $TicketNumber = $TicketObject->TicketCreateNumber();
            my $TicketID     = $TicketObject->TicketCreate(
                TN           => $TicketNumber,
                Title        => 'Selenium test Ticket',
                Queue        => 'Raw',
                Lock         => 'unlock',
                Priority     => '3 normal',
                StateID      => 1,
                TypeID       => 1,
                CustomerID   => 'SeleniumCustomer',
                CustomerUser => 'SeleniumCustomer@localhost.com',
                OwnerID      => $TestUserID,
                UserID       => $TestUserID,
            );
            $Self->True(
                $TicketID,
                "Ticket ID $TicketID - created",
            );

            push @TicketIDs,     $TicketID;
            push @TicketNumbers, $TicketNumber;
        }

        # Login as test user.
        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $Kernel::OM->Get('Kernel::Config')->Get('ScriptAlias');

        # Navigate to ticket zoom page of first created test ticket.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketZoom;TicketID=$TicketIDs[0]");

        # Click on MasterSlave and switch window.
        $Selenium->find_element("//a[contains(\@href, \'Action=AgentTicketMasterSlave;TicketID=$TicketIDs[0]' )]")
            ->click();

        $Selenium->WaitFor( WindowCount => 2 );
        my $Handles = $Selenium->get_window_handles();
        $Selenium->switch_to_window( $Handles->[1] );

        # Wait until page has loaded, if necessary.
        $Selenium->WaitFor(
            JavaScript =>
                'return typeof($) === "function" && $(".WidgetSimple").length;'
        );

        # Open collapsed widgets, if necessary.
        $Selenium->execute_script(
            "\$('.WidgetSimple.Collapsed .WidgetAction > a').trigger('click');"
        );

        $Selenium->WaitFor(
            JavaScript =>
                'return typeof($) === "function" && $(".WidgetSimple.Expanded").length;'
        );

        # Check page.
        for my $ID (
            qw(DynamicField_MasterSlave Subject RichText FileUpload IsVisibleForCustomer submitRichText)
            )
        {
            my $Element = $Selenium->find_element( "#$ID", 'css' );
            $Element->is_enabled();
            $Element->is_displayed();
        }

        # Set first test ticket as master ticket.
        $Selenium->execute_script(
            "\$('#DynamicField_MasterSlave').val('Master').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->find_element( "#Subject",  'css' )->send_keys('Selenium Master Ticket');
        $Selenium->find_element( "#RichText", 'css' )->send_keys('Selenium Master Ticket');
        $Selenium->find_element("//button[\@class='CallForAction Primary'][contains(.,'Submit')]")->click();

        # Switch window back to agent ticket zoom view of the first created test ticket.
        $Selenium->WaitFor( WindowCount => 1 );
        $Selenium->switch_to_window( $Handles->[0] );

        $Selenium->WaitFor(
            JavaScript =>
                'return typeof(Core) == "object" && typeof(Core.App) == "object" && Core.App.PageLoadComplete;'
        );

        # Navigate to ticket history page of first created test ticket.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketHistory;TicketID=$TicketIDs[0]");

        # Wait until page has loaded, if necessary.
        $Selenium->WaitFor(
            JavaScript =>
                'return typeof($) === "function" && $(".WidgetSimple").length;'
        );

        # Verify dynamic field master ticket update.
        $Self->True(
            index( $Selenium->get_page_source(), 'Changed dynamic field MasterSlave from "" to "Master".' ) > -1,
            "Master dynamic field update value - found",
        );

        # Navigate to ticket zoom page of second created test ticket.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketZoom;TicketID=$TicketIDs[1]");

        # Click on MasterSlave and switch window.
        $Selenium->find_element("//a[contains(\@href, \'Action=AgentTicketMasterSlave;TicketID=$TicketIDs[1]' )]")
            ->click();

        $Selenium->WaitFor( WindowCount => 2 );
        $Handles = $Selenium->get_window_handles();
        $Selenium->switch_to_window( $Handles->[1] );

        # Wait until form has loaded, if necessary.
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('#submitRichText').length"
        );

        sleep 1;
        $Selenium->VerifiedRefresh();

        $Selenium->execute_script(
            "\$('#DynamicField_MasterSlave').val('SlaveOf:$TicketNumbers[0]').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->find_element( "#Subject",  'css' )->send_keys('Selenium Slave Ticket');
        $Selenium->find_element( "#RichText", 'css' )->send_keys('Selenium Slave Ticket');
        $Selenium->find_element("//button[\@class='CallForAction Primary'][contains(.,'Submit')]")->click();

        $Selenium->switch_to_window( $Handles->[0] );
        $Selenium->WaitFor( WindowCount => 1 );

        $Selenium->WaitFor(
            JavaScript =>
                'return typeof(Core) == "object" && typeof(Core.App) == "object" && Core.App.PageLoadComplete;'
        );

        # Navigate to ticket history page of first created test ticket.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketHistory;TicketID=$TicketIDs[1]");

        # Wait until page has loaded, if necessary.
        $Selenium->WaitFor(
            JavaScript =>
                'return typeof($) === "function" && $(".WidgetSimple").length;'
        );

        # Verify dynamic field slave ticket update.
        $Self->True(
            index(
                $Selenium->get_page_source(),
                'Changed dynamic field MasterSlave from "" to "SlaveOf:' . $TicketNumbers[0]
            ) > -1,
            "Slave dynamic field update value - found",
        );

        # Disable Note in AgentTicketMasterSlave screen and
        # check if MasterSlave dynamic field is shown in the screen (see bug#14780).
        $Helper->ConfigSettingChange(
            Key   => 'Ticket::Frontend::AgentTicketMasterSlave###Note',
            Value => 0,
        );

        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketMasterSlave;TicketID=$TicketIDs[1]");

        $Selenium->WaitFor(
            JavaScript =>
                'return typeof($) === "function" && $("#DynamicField_MasterSlave").length;'
        );

        $Self->True(
            $Selenium->execute_script('return $("#DynamicField_MasterSlave").length;'),
            "MasterSlave dynamic field is found"
        );

        # Create two more test tickets.
        for my $TicketCreate ( 1 .. 2 ) {
            my $TicketNumber = $TicketObject->TicketCreateNumber();
            my $TicketID     = $TicketObject->TicketCreate(
                TN           => $TicketNumber,
                Title        => 'Selenium test Ticket',
                Queue        => 'Raw',
                Lock         => 'unlock',
                Priority     => '3 normal',
                StateID      => 1,
                TypeID       => 1,
                CustomerID   => 'SeleniumCustomer',
                CustomerUser => 'SeleniumCustomer@localhost.com',
                OwnerID      => $TestUserID,
                UserID       => $TestUserID,
            );
            $Self->True(
                $TicketID,
                "Ticket ID $TicketID is created.",
            );

            push @TicketIDs,     $TicketID;
            push @TicketNumbers, $TicketNumber;
        }

        # Set Master-Slave relation for test Tickets.
        my @TestCase = (
            {
                TicketID => $TicketIDs[2],
                Value    => "Master",
            },
            {
                TicketID => $TicketIDs[3],
                Value    => "SlaveOf:$TicketNumbers[2]",
            },
        );

        my $MasterSlaveDFName = $Kernel::OM->Get('Kernel::Config')->Get('MasterSlave::DynamicField') || '';
        my $DynamicField      = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
            Name => $MasterSlaveDFName,
        );
        my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
        my $Success;
        for my $Test (@TestCase) {
            $Success = $DynamicFieldBackendObject->ValueSet(
                DynamicFieldConfig => $DynamicField,
                FieldID            => $DynamicField->{ID},
                ObjectID           => $Test->{TicketID},
                Value              => $Test->{Value},
                UserID             => 1,
            );
            $Self->True(
                $Success,
                "Value '$Test->{Value}' is set for TicketID $Test->{TicketID}.",
            );
        }

        # Verify Master Slave dynamic field value for Slave ticket.
        my $DynamicFieldValueObject = $Kernel::OM->Get('Kernel::System::DynamicFieldValue');
        my $Value                   = $DynamicFieldValueObject->ValueGet(
            FieldID  => $DynamicField->{ID},
            ObjectID => $TicketIDs[3],
        );
        $Self->Is(
            $Value->[0]->{ValueText},
            "SlaveOf:$TicketNumbers[2]",
            "Master Slave dynamic field value for Slave ticket"
        );

        # Verify there is Master-Slave relation between two test Tickets.
        my $LinkObject = $Kernel::OM->Get('Kernel::System::LinkObject');
        my $LinkList   = $LinkObject->LinkList(
            Object => 'Ticket',
            Key    => $TicketIDs[2],
            State  => 'Valid',
            UserID => 1,
        );
        $Self->True(
            IsHashRefWithData($LinkList),
            "MasterSlave link exists."
        );

        # Enable config "MasterSlave::UnsetMasterSlave".
        $Helper->ConfigSettingChange(
            Key   => 'MasterSlave::UnsetMasterSlave',
            Value => 1,
        );

        # Navigate to AgentTicketMasterSlave screen of Master Ticket.
        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentTicketMasterSlave;TicketID=$TicketIDs[2]"
        );

        # UnsetMaster status of Master Ticket.
        $Selenium->execute_script(
            "\$('#DynamicField_MasterSlave').val('UnsetMaster').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->find_element("//button[\@class='CallForAction Primary'][contains(.,'Submit')]")->VerifiedClick();

        # Make sure the cache is correct.
        my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');
        for my $Cache (qw( DynamicFieldValue LinkObject )) {
            $CacheObject->CleanUp( Type => $Cache );
        }

        # Check if link to Slave ticket is deleted. See bug#14781.
        $LinkList = $LinkObject->LinkList(
            Object => 'Ticket',
            Key    => $TicketIDs[2],
            State  => 'Valid',
            UserID => 1,
        );
        $Self->True(
            !IsHashRefWithData($LinkList),
            "MasterSlave link deleted."
        );

        # Check if Slave value for Master Slave field is deleted from Slave ticket after UnsetMaster.
        $Value = $DynamicFieldValueObject->ValueGet(
            FieldID  => $DynamicField->{ID},
            ObjectID => $TicketIDs[3],
        );

        $Self->IsDeeply(
            $Value,
            [],
            "Master Slave dynamic field value for Slave ticket"
        );

        # Check if there is MasterSlave error log during ticket creation. See bug#14899.
        my $RandomID = $Helper->GetRandomID();

        # Navigate to AgentTicketPhone and create test ticket.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketPhone");

        $Selenium->WaitFor( JavaScript => 'return typeof($) === "function" && $("#FromCustomer").length;' );

        my $TestCustomer = 'customer' . $RandomID . '@gmail.com';
        $Selenium->find_element( "#FromCustomer", 'css' )->send_keys($TestCustomer);
        $Selenium->WaitFor( JavaScript => 'return !$(".AJAXLoader:visible").length;' );

        # Lose the focus.
        $Selenium->find_element( 'body', 'css' )->click();
        $Selenium->WaitFor(
            JavaScript => 'return $("#TicketCustomerContentFromCustomer input.CustomerTicketText").length;'
        );

        $Selenium->InputFieldValueSet(
            Element => '#Dest',
            Value   => '2||Raw',
        );
        $Selenium->WaitFor( JavaScript => 'return !$(".AJAXLoader:visible").length;' );

        $Selenium->find_element( "#Subject",  'css' )->clear();
        $Selenium->find_element( "#Subject",  'css' )->send_keys("Subject$RandomID");
        $Selenium->find_element( "#RichText", 'css' )->clear();
        $Selenium->find_element( "#RichText", 'css' )->send_keys("Text$RandomID");

        # Submit form.
        $Selenium->execute_script(
            "\$('#submitRichText')[0].scrollIntoView(true);",
        );
        $Self->True(
            $Selenium->execute_script("return \$('#submitRichText').length;"),
            "Element '#submitRichText' is found in screen"
        );
        $Selenium->find_element( '#submitRichText', 'css' )->VerifiedClick();

        $Selenium->WaitFor(
            JavaScript =>
                'return typeof($) === "function" && $(".MessageBox a[href*=\'AgentTicketZoom;TicketID=\']").length;'
        );

        my @Ticket   = split( 'TicketID=', $Selenium->get_current_url() );
        my $TicketID = $Ticket[1];

        push @TicketIDs, $TicketID;

        # Navigate to AdminLog.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AdminLog");

        # Check if MasterSlave error log is created.
        my $Message = "Could not update field MasterSlave for Ticket ID $TicketID";
        $Self->True(
            $Selenium->execute_script("return \$('#LogEntries .Error:eq(0)').text().indexOf('$Message') == -1;"),
            "Error message is not created for update field MasterSlave for Ticket ID $TicketID",
        );

        # Delete created test tickets.
        for my $TicketID (@TicketIDs) {
            my $Success = $TicketObject->TicketDelete(
                TicketID => $TicketID,
                UserID   => 1,
            );
            $Self->True(
                $Success,
                "Ticket ID $TicketID is deleted."
            );
        }

        # Make sure the cache is correct.
        $CacheObject->CleanUp(
            Type => 'Ticket',
        );
    }
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

## no critic (Modules::RequireExplicitPackage)
use strict;
use warnings;
use utf8;

use Kernel::System::VariableCheck qw(:all);

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {
        my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

        my $TicketObject       = $Kernel::OM->Get('Kernel::System::Ticket');
        my $ArticleObject      = $Kernel::OM->Get('Kernel::System::Ticket::Article');
        my $QueueObject        = $Kernel::OM->Get('Kernel::System::Queue');
        my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

        $Helper->ConfigSettingChange(
            Valid => 1,
            Key   => 'CheckMXRecord',
            Value => 0,
        );
        $Helper->ConfigSettingChange(
            Valid => 1,
            Key   => 'CheckEmailAddresses',
            Value => 0,
        );
        $Helper->ConfigSettingChange(
            Key   => 'SendmailModule',
            Value => 'Kernel::System::Email::Test',
        );
        $Helper->ConfigSettingChange(
            Valid => 1,
            Key   => 'MasterSlave::ForwardSlaves',
            Value => 1,
        );

        my $RandomID = $Helper->GetRandomID();

        # Create test salutation using rich text.
        my $SalutationText = "<strong>Test Bold ${RandomID} </strong>";
        my $SalutationID   = $Kernel::OM->Get('Kernel::System::Salutation')->SalutationAdd(
            Name        => "New Salutation $RandomID",
            Text        => $SalutationText,
            ContentType => 'text/html',
            Comment     => 'some comment',
            ValidID     => 1,
            UserID      => 1,
        );
        $Self->True(
            $SalutationID,
            "SalutationID $SalutationID is created.",
        );

        # Create test queue.
        my $QueueName = "Salutation $RandomID";
        my $QueueID   = $QueueObject->QueueAdd(
            Name            => $QueueName,
            Group           => 'admin',
            ValidID         => 1,
            GroupID         => 1,
            SystemAddressID => 1,
            SalutationID    => $SalutationID,
            SignatureID     => 1,
            Comment         => 'Some comment',
            UserID          => 1

        );
        $Self->True(
            $QueueID,
            "QueueID $QueueID is created.",
        );

        # Assign answer templates to queue.
        for my $TemplateID ( 1 .. 2 ) {
            my $Success = $QueueObject->QueueStandardTemplateMemberAdd(
                QueueID            => $QueueID,
                StandardTemplateID => $TemplateID,
                Active             => 1,
                UserID             => 1,
            );
            $Self->True(
                $Success,
                "TemplateID '$TemplateID' is assigned to QueueID '$QueueID'.",
            );
        }

        my $TestCustomerUserLogin = $Helper->TestCustomerUserCreate();
        my %TestCustomerUser      = $Kernel::OM->Get('Kernel::System::CustomerUser')->CustomerUserDataGet(
            User => $TestCustomerUserLogin,
        );

        # Create master ticket.
        my $MasterTicketNumber = $TicketObject->TicketCreateNumber();
        my $MasterTicketID     = $TicketObject->TicketCreate(
            TN           => $MasterTicketNumber,
            Title        => "Master $RandomID",
            QueueID      => $QueueID,
            Lock         => 'unlock',
            Priority     => '3 normal',
            State        => 'new',
            CustomerNo   => $TestCustomerUserLogin,
            CustomerUser => $TestCustomerUserLogin,
            OwnerID      => 1,
            UserID       => 1,
        );
        $Self->True(
            $MasterTicketID,
            "TicketID $MasterTicketID is created (master).",
        );

        my $ArticleID = $ArticleObject->BackendForChannel( ChannelName => 'Phone' )->ArticleCreate(
            TicketID             => $MasterTicketID,
            IsVisibleForCustomer => 1,
            SenderType           => 'agent',
            Subject              => 'Master Article',
            Body                 => 'Unit test MasterTicket',
            ContentType          => 'text/plain; charset=ISO-8859-15',
            HistoryType          => 'PhoneCallCustomer',
            HistoryComment       => 'Unit test article',
            UserID               => 1,
        );
        $Self->True(
            $ArticleID,
            "ArticleID $ArticleID is created.",
        );

        # Get master/slave dynamic field data.
        my $MasterSlaveDynamicField     = $Kernel::OM->Get('Kernel::Config')->Get('MasterSlave::DynamicField');
        my $MasterSlaveDynamicFieldData = $DynamicFieldObject->DynamicFieldGet(
            Name => $MasterSlaveDynamicField,
        );

        my $DynamicField = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
            ID => $MasterSlaveDynamicFieldData->{ID},
        );

        my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

        # Set test ticket as master ticket.
        my $Success = $DynamicFieldBackendObject->ValueSet(
            DynamicFieldConfig => $DynamicField,
            ObjectID           => $MasterTicketID,
            Value              => 'Master',
            UserID             => 1,
        );
        $Self->True(
            $Success,
            "TicketID $MasterTicketID, DynamicField '$MasterSlaveDynamicField' is updated as MasterTicket.",
        );

        # Create slave ticket.
        my $SlaveTicketID = $TicketObject->TicketCreate(
            Title        => "Slave $RandomID",
            QueueID      => $QueueID,
            Lock         => 'unlock',
            Priority     => '3 normal',
            State        => 'new',
            CustomerID   => $TestCustomerUserLogin,
            CustomerUser => $TestCustomerUserLogin,
            OwnerID      => 1,
            UserID       => 1,
        );
        $Self->True(
            $SlaveTicketID,
            "TicketID $SlaveTicketID is created (slave).",
        );

        # Set test ticket to slave.
        $Success = $DynamicFieldBackendObject->ValueSet(
            DynamicFieldConfig => $DynamicField,
            ObjectID           => $SlaveTicketID,
            Value              => "SlaveOf:$MasterTicketNumber",
            UserID             => 1,
        );
        $Self->True(
            $Success,
            "TicketID $SlaveTicketID, DynamicField '$MasterSlaveDynamicField' is updated as SlaveTicket.",
        );

        # Create test user.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'users' ],
        );

        # Get test user ID.
        my $TestUserID = $Kernel::OM->Get('Kernel::System::User')->UserLookup(
            UserLogin => $TestUserLogin,
        );

        # Login as test user.
        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $Kernel::OM->Get('Kernel::Config')->Get('ScriptAlias');

        # Navigate to AgentTicketCompose screen.
        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentTicketCompose;TicketID=$MasterTicketID;ArticleID=$ArticleID;ReplyAll=;ResponseID=1"
        );

        $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#submitRichText').length;" );
        $Selenium->find_element( "#ToCustomer",     'css' )->send_keys( $TestCustomerUser{UserEmail} );
        $Selenium->find_element( "#submitRichText", 'css' )->VerifiedClick();

        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentTicketZoom;TicketID=$SlaveTicketID"
        );

        # Wait for the iframe to show up.
        $Selenium->WaitFor(
            JavaScript =>
                "return typeof(\$) === 'function' && \$('.ArticleMailContent iframe').contents().length == 1;"
        );

        $Selenium->SwitchToFrame(
            FrameSelector => '.ArticleMailContent iframe',
            WaitForLoad   => 0,
        );

        # Check if slave ticket article hes salutation in rich text format. See bug#14983.
        $Self->True(
            index( $Selenium->get_page_source(), $SalutationText ) > -1,
            "Slave article contains rich text '$SalutationText'. ",
        );

        # Cleanup.
        my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

        # Delete test created tickets.
        for my $TicketID ( $MasterTicketID, $SlaveTicketID ) {
            $Success = $TicketObject->TicketDelete(
                TicketID => $TicketID,
                UserID   => 1,
            );

            # Ticket deletion could fail if apache still writes to ticket history. Try again in this case.
            if ( !$Success ) {
                sleep 3;
                $Success = $TicketObject->TicketDelete(
                    TicketID => $TicketID,
                    UserID   => 1,
                );
            }
            $Self->True(
                $Success,
                "TicketID $TicketID is deleted."
            );
        }

        $Success = $DBObject->Do(
            SQL  => "DELETE FROM queue_standard_template WHERE queue_id = ?",
            Bind => [ \$QueueID, ],
        );
        $Self->True(
            $Success,
            "Standard_template_queue relation for QueueID $QueueID is deleted."
        );

        # Delete queues.
        $Success = $DBObject->Do(
            SQL  => "DELETE FROM queue WHERE id = ?",
            Bind => [ \$QueueID, ],
        );
        $Self->True(
            $Success,
            "QueueID $QueueID is deleted.",
        );

        # Delete salutation.
        $Success = $DBObject->Do(
            SQL  => "DELETE FROM salutation WHERE id = ?",
            Bind => [ \$SalutationID, ],
        );
        $Self->True(
            $Success,
            "SalutationID $SalutationID is deleted.",
        );

    }
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

## no critic (Modules::RequireExplicitPackage)
use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper       = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

        # Enable the advanced MasterSlave.
        $Helper->ConfigSettingChange(
            Key   => 'MasterSlave::AdvancedEnabled',
            Value => 1,
        );

        # Enable change the MasterSlave state of a ticket.
        $Helper->ConfigSettingChange(
            Key   => 'MasterSlave::UpdateMasterSlave',
            Value => 1,
        );

        # Do not send emails.
        $Helper->ConfigSettingChange(
            Key   => 'SendmailModule',
            Value => 'Kernel::System::Email::Test',
        );

        # Create test user.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'users' ],
        ) || die "Did not get test user";

        # Get test user ID.
        my $TestUserID = $Kernel::OM->Get('Kernel::System::User')->UserLookup(
            UserLogin => $TestUserLogin,
        );

        # Create three test tickets.
        my @TicketIDs;
        my @TicketNumbers;
        my $TicketTitle = 'MasterSlave ' . $Helper->GetRandomID();
        for my $TicketCreate ( 1 .. 3 ) {
            my $TicketNumber = $TicketObject->TicketCreateNumber();
            my $TicketID     = $TicketObject->TicketCreate(
                TN           => $TicketNumber,
                Title        => $TicketTitle,
                Queue        => 'Raw',
                Lock         => 'unlock',
                Priority     => '3 normal',
                StateID      => 1,
                TypeID       => 1,
                CustomerID   => 'SeleniumCustomer',
                CustomerUser => 'customer@example.com',
                OwnerID      => $TestUserID,
                UserID       => $TestUserID,
            );
            $Self->True(
                $TicketID,
                "Ticket ID $TicketID - created",
            );

            push @TicketIDs,     $TicketID;
            push @TicketNumbers, $TicketNumber;
        }

        # Login as test user.
        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $Kernel::OM->Get('Kernel::Config')->Get('ScriptAlias');

        # Navigate to AgentTicketSearch.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketSearch");

        # Wait until form has loaded, if necessary.
        $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#Attribute').length;" );

        # Search test created tickets by title.
        $Selenium->execute_script("\$('#Attribute').val('Title').trigger('redraw.InputField').trigger('change');");

        $Selenium->find_element( "Title", 'name' )->send_keys($TicketTitle);
        $Selenium->find_element("//button[\@id='SearchFormSubmit'][\@value='Run search']")->VerifiedClick();

        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('.Checkbox[value=$TicketIDs[0]]').length;"
        );

        # Select first test created ticket.
        $Selenium->execute_script("\$('.Checkbox[value=$TicketIDs[0]]').click();");
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('.Checkbox[value=$TicketIDs[0]]:checked').length;"
        );

        # Click on bulk and switch screen.
        $Selenium->find_element( "Bulk", 'link_text' )->click();

        $Selenium->WaitFor( WindowCount => 2 );
        my $Handles = $Selenium->get_window_handles();
        $Selenium->switch_to_window( $Handles->[1] );

        # Wait until popup is completely loaded.
        $Selenium->WaitFor(
            JavaScript =>
                'return typeof(Core) == "object" && typeof(Core.App) == "object" && Core.App.PageLoadComplete;',
        );

        $Selenium->WaitFor(
            JavaScript =>
                "return typeof(\$) === 'function' && \$('#DynamicField_MasterSlave').length && \$('#submitRichText').length;"
        );

        # Set test ticket as master ticket.
        $Selenium->execute_script(
            "\$('#DynamicField_MasterSlave').val('Master').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->find_element( "#submitRichText", 'css' )->click();

        $Selenium->switch_to_window( $Handles->[0] );
        $Selenium->WaitFor( WindowCount => 1 );

        # Wait until popup is completely loaded.
        $Selenium->VerifiedRefresh();

        # Select second and third test created ticket.
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('.Checkbox[value=$TicketIDs[1]]').length;"
        );
        $Selenium->execute_script("\$('.Checkbox[value=$TicketIDs[1]]').click();");
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('input[value=$TicketIDs[1]]:checked').length;"
        );

        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('.Checkbox[value=$TicketIDs[2]]').length;"
        );
        $Selenium->execute_script("\$('.Checkbox[value=$TicketIDs[2]]').click();");
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('input[value=$TicketIDs[2]]:checked').length;"
        );

        # Click on bulk and switch screen.
        $Selenium->find_element( "Bulk", 'link_text' )->click();

        $Selenium->WaitFor( WindowCount => 2 );
        $Handles = $Selenium->get_window_handles();
        $Selenium->switch_to_window( $Handles->[1] );

        # Wait until popup is completely loaded.
        $Selenium->WaitFor(
            JavaScript =>
                'return typeof(Core) == "object" && typeof(Core.App) == "object" && Core.App.PageLoadComplete;',
        );

        $Selenium->WaitFor(
            JavaScript =>
                "return typeof(\$) === 'function' && \$('#DynamicField_MasterSlave').length && \$('#submitRichText').length;"
        );

        # Set test tickets as slave tickets.
        $Selenium->execute_script(
            "\$('#DynamicField_MasterSlave').val('SlaveOf:$TicketNumbers[0]').trigger('redraw.InputField').trigger('change');"
        );

        $Selenium->find_element( "#submitRichText", 'css' )->click();

        $Selenium->switch_to_window( $Handles->[0] );
        $Selenium->WaitFor( WindowCount => 1 );

        # Navigate to master ticket zoom view.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentTicketZoom;TicketID=$TicketIDs[0]");

        # Verify master-slave ticket link.
        $Self->True(
            index( $Selenium->get_page_source(), $TicketNumbers[1] ) > -1,
            "Slave ticket number: $TicketNumbers[1] - found",
        );
        $Self->True(
            index( $Selenium->get_page_source(), $TicketNumbers[2] ) > -1,
            "Slave ticket number: $TicketNumbers[2] - found",
        );

        # Delete created test tickets.
        for my $TicketID (@TicketIDs) {
            my $Success = $TicketObject->TicketDelete(
                TicketID => $TicketID,
                UserID   => 1,
            );
            $Self->True(
                $Success,
                "Ticket ID $TicketID - deleted"
            );
        }

        # Make sure the cache is correct.
        $Kernel::OM->Get('Kernel::System::Cache')->CleanUp(
            Type => 'Ticket',
        );
    }
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

## no critic (Modules::RequireExplicitPackage)
use strict;
use warnings;
use utf8;

use vars (qw($Self));

# start RestoreDatabse
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $HelperObject = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

# enable config MasterSlave::ForwardSlaves
$ConfigObject->Set(
    Key   => 'MasterSlave::ForwardSlaves',
    Value => 1,
);
$ConfigObject->Set(
    Key   => 'CheckMXRecord',
    Value => 0,
);
$ConfigObject->Set(
    Key   => 'CheckEmailAddresses',
    Value => 0,
);
$ConfigObject->Set(
    Key   => 'SendmailModule',
    Value => 'Kernel::System::Email::DoNotSendEmail',
);

my $RandomID = $HelperObject->GetRandomID();

my $DynamicFieldObject          = $Kernel::OM->Get('Kernel::System::DynamicField');
my $MasterSlaveDynamicField     = $ConfigObject->Get('MasterSlave::DynamicField');
my $MasterSlaveDynamicFieldData = $DynamicFieldObject->DynamicFieldGet(
    Name => $MasterSlaveDynamicField,
);

# Create new user.
my $TestUserLogin = $HelperObject->TestUserCreate(
    Groups   => [ 'admin', 'users' ],
    Language => 'en'
);
$Self->True(
    $TestUserLogin,
    "UserAdd() $TestUserLogin",
);

# Create new customer users.
my $TestCustomerUserLogin1 = $HelperObject->TestCustomerUserCreate(
    Language => 'en',
);
$Self->True(
    $TestCustomerUserLogin1,
    "CustomerUserAdd() $TestCustomerUserLogin1",
);

my $TestCustomerUserLogin2 = $HelperObject->TestCustomerUserCreate(
    Language => 'en',
);
$Self->True(
    $TestCustomerUserLogin2,
    "CustomerUserAdd() $TestCustomerUserLogin2",
);

my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

# Create first test ticket.
my $MasterTicketNumber = $TicketObject->TicketCreateNumber();
my $MasterTicketID     = $TicketObject->TicketCreate(
    TN           => $MasterTicketNumber,
    Title        => 'Master unit test ticket ' . $RandomID,
    Queue        => 'Raw',
    Lock         => 'unlock',
    Priority     => '3 normal',
    State        => 'new',
    CustomerNo   => $TestCustomerUserLogin1,
    CustomerUser => $TestCustomerUserLogin1,
    ,
    OwnerID => 1,
    UserID  => 1,
);
$Self->True(
    $MasterTicketID,
    "TicketCreate() Ticket ID $MasterTicketID",
);

my $EmailBackendObject = $Kernel::OM->Get('Kernel::System::Ticket::Article')->BackendForChannel(
    ChannelName => 'Email',
);

# Create article for test ticket.
my $ArticleID = $EmailBackendObject->ArticleCreate(
    TicketID             => $MasterTicketID,
    From                 => 'root@localhost',
    To                   => 'test@examples.com',
    IsVisibleForCustomer => 1,
    SenderType           => 'agent',
    Subject              => 'Master Article',
    Body                 => "Unit test MasterTicket $TestCustomerUserLogin1 $TestCustomerUserLogin1",
    ContentType          => 'text/plain; charset=ISO-8859-15',
    HistoryType          => 'EmailCustomer',
    HistoryComment       => 'Unit test article',
    UserID               => 1,
);
$Self->True(
    $ArticleID,
    "ArticleCreate() Article ID $ArticleID",
);

my $DynamicField = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
    ID => $MasterSlaveDynamicFieldData->{ID},
);

my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

# Set test ticket as master ticket.
my $Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicField,
    ObjectID           => $MasterTicketID,
    Value              => 'Master',
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $MasterTicketID DynamicField $MasterSlaveDynamicField updated as MasterTicket",
);

# Create second test ticket.
my $SlaveTicketNumber = $TicketObject->TicketCreateNumber();
my $SlaveTicketID     = $TicketObject->TicketCreate(
    TN           => $SlaveTicketNumber,
    Title        => 'Slave unit test ticket ' . $RandomID,
    Queue        => 'Raw',
    Lock         => 'unlock',
    Priority     => '3 normal',
    State        => 'new',
    CustomerID   => $TestCustomerUserLogin2,
    CustomerUser => $TestCustomerUserLogin2,
    OwnerID      => 1,
    UserID       => 1,
);
$Self->True(
    $SlaveTicketID,
    "TicketCreate() Ticket ID $SlaveTicketID",
);

# Set test ticket as slave of master ticket.
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicField,
    ObjectID           => $SlaveTicketID,
    Value              => "SlaveOf:$MasterTicketNumber",
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $SlaveTicketID DynamicField $MasterSlaveDynamicField updated as SlaveTicket",
);

my $EventObject = $Kernel::OM->Get('Kernel::System::Ticket::Event::MasterSlave');

my @Tests = (
    {
        Name           => 'Do not replace',
        EffectiveValue => {
            Email => 0,
        },
        Match => "$TestCustomerUserLogin1 $TestCustomerUserLogin1",
    },
    {
        Name           => 'Customer replace',
        EffectiveValue => {
            Email => 1,
        },
        Match => "$TestCustomerUserLogin2 $TestCustomerUserLogin2",
    },
);

my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

TEST:
for my $Test (@Tests) {
    $ConfigObject->Set(
        Key   => 'ReplaceCustomerRealNameOnSlaveArticleCommunicationChannels',
        Value => $Test->{EffectiveValue},
    );
    my $Success = $EventObject->Run(
        Event => 'ArticleSend',
        Data  => {
            TicketID => $MasterTicketID,
        },
        Config => {},
        UserID => 1,
    );
    $Self->True(
        $Success,
        "$Test->{Name} - ArticleSend event executed correctly",
    );

    my @Articles = $ArticleObject->ArticleList(
        TicketID => $SlaveTicketID,
    );

    my $ArticleBackendObject = $ArticleObject->BackendForArticle(
        TicketID  => $Articles[-1]->{TicketID},
        ArticleID => $Articles[-1]->{ArticleID},
    );

    my %Article = $ArticleBackendObject->ArticleGet(
        TicketID  => $Articles[-1]->{TicketID},
        ArticleID => $Articles[-1]->{ArticleID},
    );

    my $MatchSuccess = $Article{Body} =~ m{$Test->{Match}} // 0;

    $Self->True(
        $MatchSuccess,
        "$Test->{Name} - New article body matched customer real name: '$Test->{Match}'",
    );
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

## no critic (Modules::RequireExplicitPackage)
use strict;
use warnings;
use utf8;

use vars (qw($Self));

# get needed objects
my $TicketObject              = $Kernel::OM->Get('Kernel::System::Ticket');
my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
my $LinkObject                = $Kernel::OM->Get('Kernel::System::LinkObject');
my $DynamicFieldObject        = $Kernel::OM->Get('Kernel::System::DynamicField');
my $UserObject                = $Kernel::OM->Get('Kernel::System::User');
my $CustomerUserObject        = $Kernel::OM->Get('Kernel::System::CustomerUser');
my $ConfigObject              = $Kernel::OM->Get('Kernel::Config');

# start RestoreDatabse
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $HelperObject = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# enable config MasterSlave::ForwardSlaves
$ConfigObject->Set(
    Key   => 'MasterSlave::ForwardSlaves',
    Value => 1,
);
$ConfigObject->Set(
    Key   => 'CheckMXRecord',
    Value => 0,
);
$ConfigObject->Set(
    Key   => 'CheckEmailAddresses',
    Value => 0,
);

# get random ID
my $RandomID = $HelperObject->GetRandomID();

# get master/slave dynamic field data
my $MasterSlaveDynamicField     = $ConfigObject->Get('MasterSlave::DynamicField');
my $MasterSlaveDynamicFieldData = $DynamicFieldObject->DynamicFieldGet(
    Name => $MasterSlaveDynamicField,
);

# create new user
my $TestUserLogin = $HelperObject->TestUserCreate(
    Groups   => [ 'admin', 'users' ],
    Language => 'en'
);
$Self->True(
    $TestUserLogin,
    "UserAdd() $TestUserLogin",
);

# create new customer user
my $TestCustomerUserLogin = $HelperObject->TestCustomerUserCreate(
    Language => 'en',
);
$Self->True(
    $TestCustomerUserLogin,
    "CustomerUserAdd() $TestCustomerUserLogin",
);

# create first test ticket
my $MasterTicketNumber = $TicketObject->TicketCreateNumber();
my $MasterTicketID     = $TicketObject->TicketCreate(
    TN           => $MasterTicketNumber,
    Title        => 'Master unit test ticket ' . $RandomID,
    Queue        => 'Raw',
    Lock         => 'unlock',
    Priority     => '3 normal',
    State        => 'new',
    CustomerNo   => $TestCustomerUserLogin,
    CustomerUser => $TestCustomerUserLogin,
    ,
    OwnerID => 1,
    UserID  => 1,
);
$Self->True(
    $MasterTicketID,
    "TicketCreate() Ticket ID $MasterTicketID",
);

my $EmailBackendObject = $Kernel::OM->Get('Kernel::System::Ticket::Article')->BackendForChannel(
    ChannelName => 'Email',
);

# create article for test ticket
my $ArticleID = $EmailBackendObject->ArticleCreate(
    TicketID             => $MasterTicketID,
    IsVisibleForCustomer => 1,
    SenderType           => 'agent',
    Subject              => 'Master Article',
    Body                 => 'Unit test MasterTicket',
    ContentType          => 'text/plain; charset=ISO-8859-15',
    HistoryType          => 'EmailCustomer',
    HistoryComment       => 'Unit test article',
    UserID               => 1,
);
$Self->True(
    $ArticleID,
    "ArticleCreate() Article ID $ArticleID",
);

my $DynamicField = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
    ID => $MasterSlaveDynamicFieldData->{ID},
);

# set test ticket as master ticket
my $Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicField,
    ObjectID           => $MasterTicketID,
    Value              => 'Master',
    UserID             => 1,
);
$Self->True(
    $Success,
    "ValueSet() Ticket ID $MasterTicketID DynamicField $MasterSlaveDynamicField updated as MasterTicket",
);

# create second test ticket
my $SlaveTicketNumber = $TicketObject->TicketCreateNumber();
my $SlaveTicketID     = $TicketObject->TicketCreate(
    TN           => $SlaveTicketNumber,
    Title        => 'Slave unit test ticket ' . $RandomID,
    Queue        => 'Raw',
    Lock         => 'unlock',
    Priority     => '3 normal',
    State        => 'new',
    CustomerID   => $TestCustomerUserLogin,
    CustomerUser => $TestCustomerUserLogin . '@localunittest.com',
    OwnerID      => 1,
    UserID       => 1,
);
$Self->True(
    $SlaveTicketID,
    "TicketCreate() Ticket ID $SlaveTicketID",
);
$Success = $DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicField,
    ObjectID           => $SlaveTicketID,
    Value              => "SlaveOf:$MasterTicketNumber",
    UserID             => 1,
);

# ------------------------------------------------------------ #
# test event ArticleSend
# ------------------------------------------------------------ #

# create master ticket article and forward it
my $MasterNewSubject = $TicketObject->TicketSubjectBuild(
    TicketNumber => $MasterTicketNumber,
    Subject      => 'Master Article',
    Action       => 'Forward',
);
my $ForwardArticleID = $EmailBackendObject->ArticleSend(
    TicketID             => $MasterTicketID,
    IsVisibleForCustomer => 1,
    SenderType           => 'agent',
    From                 => 'Some Agent <email@example.com>',
    To                   => 'Some Customer A <customer-a@example.com>',
    Subject              => $MasterNewSubject,
    Body                 => 'Unit test forwarded article',
    Charset              => 'iso-8859-15',
    MimeType             => 'text/plain',
    HistoryType          => 'Forward',
    HistoryComment       => 'Forwarded article',
    NoAgentNotify        => 0,
    UserID               => 1,
);
$Self->True(
    $Success,
    "ArticleSend() Forwarded MasterTicket Article ID $ForwardArticleID",
);

# get master ticket history
my @MasterHistoryLines = $TicketObject->HistoryGet(
    TicketID => $MasterTicketID,
    UserID   => 1,
);

my $MasterLastHistoryEntry = $MasterHistoryLines[-1];

# verify master ticket article is created
$Self->IsDeeply(
    $MasterLastHistoryEntry->{Name},
    'MasterTicketAction: ArticleSend',
    "MasterTicket ArticleSend event - ",
);

# get slave ticket history
my @SlaveHistoryLines = $TicketObject->HistoryGet(
    TicketID => $SlaveTicketID,
    UserID   => 1,
);
my $SlaveLastHistoryEntry = $SlaveHistoryLines[-1];

# verify slave ticket article tried to send
$Self->IsDeeply(
    $SlaveLastHistoryEntry->{Name},
    'MasterTicket: no customer email found, send no master message to customer.',
    "SlaveTicket ArticleSend event - ",
);

# ------------------------------------------------------------ #
# test event ArticleCreate
# ------------------------------------------------------------ #
my $InternalBackendObject = $Kernel::OM->Get('Kernel::System::Ticket::Article')->BackendForChannel(
    ChannelName => 'Internal',
);

# create note article for master ticket
my $ArticleIDCreate = $InternalBackendObject->ArticleCreate(
    TicketID             => $MasterTicketID,
    IsVisibleForCustomer => 0,
    SenderType           => 'agent',
    Subject              => 'Note article',
    Body                 => 'Unit test MasterTicket',
    ContentType          => 'text/plain; charset=ISO-8859-15',
    HistoryType          => 'AddNote',
    HistoryComment       => 'Unit test article',
    UserID               => 1,
);
$Self->True(
    $ArticleIDCreate,
    "ArticleCreate() Note Article ID $ArticleIDCreate created for MasterTicket ID $MasterTicketID",
);

# get master ticket history
@MasterHistoryLines = $TicketObject->HistoryGet(
    TicketID => $MasterTicketID,
    UserID   => 1,
);
($MasterLastHistoryEntry) = grep { $_->{Name} eq 'MasterTicketAction: ArticleCreate' } @MasterHistoryLines;

# verify master ticket article is created
$Self->IsDeeply(
    $MasterLastHistoryEntry->{Name},
    'MasterTicketAction: ArticleCreate',
    "MasterTicket ArticleCreate event - ",
);

# get slave ticket history
@SlaveHistoryLines = $TicketObject->HistoryGet(
    TicketID => $SlaveTicketID,
    UserID   => 1,
);
($SlaveLastHistoryEntry) = grep { $_->{Name} eq 'Added article based on master ticket.' } @SlaveHistoryLines;

# verify slave ticket article is created
$Self->IsDeeply(
    $SlaveLastHistoryEntry->{Name},
    'Added article based on master ticket.',
    "SlaveTicket ArticleCreate event - ",
);

# ------------------------------------------------------------ #
# test event TicketStateUpdate
# ------------------------------------------------------------ #

# change master ticket state to 'open'
$Success = $TicketObject->TicketStateSet(
    State    => 'open',
    TicketID => $MasterTicketID,
    UserID   => 1,
);
$Self->True(
    $Success,
    "TicketStateSet() MasterTicket state updated - 'open'",
);

# get master ticket history
@MasterHistoryLines = $TicketObject->HistoryGet(
    TicketID => $MasterTicketID,
    UserID   => 1,
);
$MasterLastHistoryEntry = $MasterHistoryLines[-1];

# verify master ticket state is updated
$Self->IsDeeply(
    $MasterLastHistoryEntry->{Name},
    'MasterTicketAction: TicketStateUpdate',
    "MasterTicket TicketStateUpdate event - ",
);

# verify slave ticket state is updated
my %SlaveTicketData = $TicketObject->TicketGet(
    TicketID => $SlaveTicketID,
    UserID   => 1,
);

$Self->IsDeeply(
    $SlaveTicketData{State},
    'open',
    "SlaveTicket state updated - 'open' - ",
);

# ------------------------------------------------------------ #
# test event TicketPendingTimeUpdate
# ------------------------------------------------------------ #

# change pending time for master ticket
$Success = $TicketObject->TicketPendingTimeSet(
    Year     => 0000,
    Month    => 00,
    Day      => 00,
    Hour     => 00,
    Minute   => 00,
    TicketID => $MasterTicketID,
    UserID   => 1,
);
$Self->True(
    $Success,
    "TicketPendingTimeSet() MasterTicket pending time updated",
);

# get master ticket history
@MasterHistoryLines = $TicketObject->HistoryGet(
    TicketID => $MasterTicketID,
    UserID   => 1,
);
$MasterLastHistoryEntry = $MasterHistoryLines[-1];

# verify master ticket pending time is updated
$Self->IsDeeply(
    $MasterLastHistoryEntry->{Name},
    'MasterTicketAction: TicketPendingTimeUpdate',
    "MasterTicket TicketPendingTimeUpdate event - ",
);

# get slave ticket history
@SlaveHistoryLines = $TicketObject->HistoryGet(
    TicketID => $SlaveTicketID,
    UserID   => 1,
);
$SlaveLastHistoryEntry = $SlaveHistoryLines[-1];

# verify slave ticket pending time is updated
$Self->IsDeeply(
    $SlaveLastHistoryEntry->{Name},
    '%%00-00-00 00:00',
    "SlaveTicket pending time update - ",
);

# ------------------------------------------------------------ #
# test event TicketPriorityUpdate
# ------------------------------------------------------------ #

# change master ticket priority to '2 low'
$Success = $TicketObject->TicketPrioritySet(
    TicketID => $MasterTicketID,
    Priority => '2 low',
    UserID   => 1,
);
$Self->True(
    $Success,
    "TicketPrioritySet() MasterTicket priority updated - '2 low'",
);

# get master ticket history
@MasterHistoryLines = $TicketObject->HistoryGet(
    TicketID => $MasterTicketID,
    UserID   => 1,
);
$MasterLastHistoryEntry = $MasterHistoryLines[-1];

# verify master ticket priority is updated
$Self->IsDeeply(
    $MasterLastHistoryEntry->{Name},
    'MasterTicketAction: TicketPriorityUpdate',
    "MasterTicket TicketPriorityUpdate event - ",
);

# verify slave ticket priority is updated
%SlaveTicketData = $TicketObject->TicketGet(
    TicketID => $SlaveTicketID,
    UserID   => 1,
);
$Self->IsDeeply(
    $SlaveTicketData{Priority},
    '2 low',
    "SlaveTicket priority updated - '2 low' - ",
);

# ------------------------------------------------------------ #
# test event TicketOwnerUpdate
# ------------------------------------------------------------ #

# change master ticket owner
$Success = $TicketObject->TicketOwnerSet(
    TicketID => $MasterTicketID,
    NewUser  => $TestUserLogin,
    UserID   => 1,
);
$Self->True(
    $Success,
    "TicketOwnerSet() MasterTicket owner updated - $TestUserLogin",
);

# get master ticket history
@MasterHistoryLines = $TicketObject->HistoryGet(
    TicketID => $MasterTicketID,
    UserID   => 1,
);
$MasterLastHistoryEntry = $MasterHistoryLines[-1];

# verify master ticket owner is updated
$Self->IsDeeply(
    $MasterLastHistoryEntry->{Name},
    'MasterTicketAction: TicketOwnerUpdate',
    "MasterTicket TicketOwnerUpdate event - ",
);

# verify slave ticket owner is updated
%SlaveTicketData = $TicketObject->TicketGet(
    TicketID => $SlaveTicketID,
    UserID   => 1,
);
$Self->IsDeeply(
    $SlaveTicketData{Owner},
    $TestUserLogin,
    "SlaveTicket owner updated - ",
);

# ------------------------------------------------------------ #
# test event TicketResponsibleUpdate
# ------------------------------------------------------------ #

# set new responsible user for master ticket
$Success = $TicketObject->TicketResponsibleSet(
    TicketID => $MasterTicketID,
    NewUser  => $TestUserLogin,
    UserID   => 1,
);
$Self->True(
    $Success,
    "TicketResponsibleSet() MasterTicket responsible updated - $TestUserLogin",
);

# get master ticket history
@MasterHistoryLines = $TicketObject->HistoryGet(
    TicketID => $MasterTicketID,
    UserID   => 1,
);
$MasterLastHistoryEntry = $MasterHistoryLines[-1];

# verify master ticket responsible user is updated
$Self->IsDeeply(
    $MasterLastHistoryEntry->{Name},
    'MasterTicketAction: TicketResponsibleUpdate',
    "MasterTicket TicketResponsibleUpdate event - ",
);

# verify slave ticket owner is updated
%SlaveTicketData = $TicketObject->TicketGet(
    TicketID => $SlaveTicketID,
    UserID   => 1,
);
$Self->IsDeeply(
    $SlaveTicketData{Responsible},
    $TestUserLogin,
    "SlaveTicket responsible updated - ",
);

# ------------------------------------------------------------ #
# test event TicketLockUpdate
# ------------------------------------------------------------ #

# lock master ticket
$Success = $TicketObject->TicketLockSet(
    Lock     => 'lock',
    TicketID => $MasterTicketID,
    UserID   => 1,
);
$Self->True(
    $Success,
    "TicketLockSet() MasterTicket is locked",
);

# get master ticket history
@MasterHistoryLines = $TicketObject->HistoryGet(
    TicketID => $MasterTicketID,
    UserID   => 1,
);
$MasterLastHistoryEntry = $MasterHistoryLines[-1];

# verify master ticket is locked
$Self->IsDeeply(
    $MasterLastHistoryEntry->{Name},
    'MasterTicketAction: TicketLockUpdate',
    "MasterTicket TicketLockUpdate event - ",
);

# verify slave ticket is locked
%SlaveTicketData = $TicketObject->TicketGet(
    TicketID => $SlaveTicketID,
    UserID   => 1,
);
$Self->IsDeeply(
    $SlaveTicketData{Lock},
    'lock',
    "SlaveTicket lock updated - ",
);

# cleanup is done by RestoreDatabase

1;

Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgQ29yZSA9IENvcmUgfHwge307CkNvcmUuQWdlbnQgPSBDb3JlLkFnZW50IHx8IHt9OwpDb3JlLkFnZW50LkFkbWluID0gQ29yZS5BZ2VudC5BZG1pbiB8fCB7fTsKCi8qKgogKiBAbmFtZXNwYWNlIENvcmUuQWdlbnQuQWRtaW4uRHluYW1pY0ZpZWxkTWFzdGVyU2xhdmUKICogQG1lbWJlcm9mIENvcmUuQWdlbnQuQWRtaW4KICogQGF1dGhvciBPVFJTIEFHCiAqIEBkZXNjcmlwdGlvbgogKiAgICAgIFRoaXMgbmFtZXNwYWNlIGNvbnRhaW5zIHRoZSBzcGVjaWFsIGZ1bmN0aW9ucyBmb3IgU3lzdGVtQ29uZmlndXJhdGlvbiBtb2R1bGUuCiAqLwogQ29yZS5BZ2VudC5BZG1pbi5EeW5hbWljRmllbGRNYXN0ZXJTbGF2ZSA9IChmdW5jdGlvbiAoVGFyZ2V0TlMpIHsKCiAgICAkKCcuU2hvd1dhcm5pbmcnKS5iaW5kKCdjaGFuZ2Uga2V5dXAnLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgJCgncC5XYXJuaW5nJykucmVtb3ZlQ2xhc3MoJ0hpZGRlbicpOwogICAgfSk7CiAgICBDb3JlLkFnZW50LkFkbWluLkR5bmFtaWNGaWVsZC5WYWxpZGF0aW9uSW5pdCgpOwoKICAgIENvcmUuSW5pdC5SZWdpc3Rlck5hbWVzcGFjZShUYXJnZXROUywgJ0FQUF9NT0RVTEUnKTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oQ29yZS5BZ2VudC5BZG1pbi5EeW5hbWljRmllbGRNYXN0ZXJTbGF2ZSB8fCB7fSkpOwo=
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyAkb3JpZ2luOiBvdHJzIC0gODIwN2QwZjY4MWFkY2RlYjVjMWI0OTdhYzU0N2ExZDk3NDk4MzhkNSAtIHZhci9odHRwZC9odGRvY3MvanMvQ29yZS5BZ2VudC5UaWNrZXRBY3Rpb25Db21tb24uanMKLy8gLS0KLy8gVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKLy8gdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91Ci8vIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgovLyAtLQoKInVzZSBzdHJpY3QiOwoKdmFyIENvcmUgPSBDb3JlIHx8IHt9OwpDb3JlLkFnZW50ID0gQ29yZS5BZ2VudCB8fCB7fTsKCi8vIC0tLQovLyBPVFJTTWFzdGVyU2xhdmUKLy8gLS0tCi8vLyoqCi8vICogQG5hbWVzcGFjZSBDb3JlLkFnZW50LlRpY2tldEFjdGlvbkNvbW1vbgovLyAqIEBtZW1iZXJvZiBDb3JlLkFnZW50Ci8vICogQGF1dGhvciBPVFJTIEFHCi8vICogQGRlc2NyaXB0aW9uCi8vICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIEFnZW50VGlja2V0QWN0aW9uQ29tbW9uLgovLyAqLwovL0NvcmUuQWdlbnQuVGlja2V0QWN0aW9uQ29tbW9uID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewovKioKICogQG5hbWVzcGFjZSBDb3JlLkFnZW50LlRpY2tldE1hc3RlclNsYXZlCiAqIEBtZW1iZXJvZiBDb3JlLkFnZW50CiAqIEBhdXRob3IgT1RSUyBBRwogKiBAZGVzY3JpcHRpb24KICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIFRpY2tldE1hc3RlclNsYXZlLgogKi8KQ29yZS5BZ2VudC5UaWNrZXRNYXN0ZXJTbGF2ZSA9IChmdW5jdGlvbiAoVGFyZ2V0TlMpIHsKLy8gLS0tCgogICAgLyoqCiAgICAgKiBAbmFtZSBJbml0CiAgICAgKiBAbWVtYmVyb2YgQ29yZS5BZ2VudC5UaWNrZXRBY3Rpb25Db21tb24KICAgICAqIEBmdW5jdGlvbgogICAgICogQGRlc2NyaXB0aW9uCiAgICAgKiAgICAgIFRoaXMgZnVuY3Rpb24gaW5pdGlhbGl6ZXMgdGhlIG1vZHVsZSBmdW5jdGlvbmFsaXR5LgogICAgICovCiAgICBUYXJnZXROUy5Jbml0ID0gZnVuY3Rpb24gKCkgewoKICAgICAgICB2YXIgRHluYW1pY0ZpZWxkTmFtZXMgPSBDb3JlLkNvbmZpZy5HZXQoJ0R5bmFtaWNGaWVsZE5hbWVzJyksCiAgICAgICAgICAgIEZpZWxkcyA9IFsnVHlwZUlEJywgJ1NlcnZpY2VJRCcsICdTTEFJRCcsICdOZXdPd25lcklEJywgJ05ld1Jlc3BvbnNpYmxlSUQnLCAnTmV3U3RhdGVJRCcsICdOZXdQcmlvcml0eUlEJ10sCiAgICAgICAgICAgIE1vZGlmaWVkRmllbGRzOwoKICAgICAgICAvLyBCaW5kIGV2ZW50cyB0byBzcGVjaWZpYyBmaWVsZHMKICAgICAgICAkLmVhY2goRmllbGRzLCBmdW5jdGlvbihJbmRleCwgVmFsdWUpIHsKICAgICAgICAgICAgTW9kaWZpZWRGaWVsZHMgPSBDb3JlLkRhdGEuQ29weU9iamVjdChGaWVsZHMpLmNvbmNhdChEeW5hbWljRmllbGROYW1lcyk7CiAgICAgICAgICAgIE1vZGlmaWVkRmllbGRzLnNwbGljZShJbmRleCwgMSk7CgogICAgICAgICAgICBGaWVsZFVwZGF0ZShWYWx1ZSwgTW9kaWZpZWRGaWVsZHMpOwogICAgICAgIH0pOwoKICAgICAgICAvLyBCaW5kIGV2ZW50IHRvIFF1ZXVlIGZpZWxkLgogICAgICAgICQoJyNOZXdRdWV1ZUlEJykub24oJ2NoYW5nZScsIGZ1bmN0aW9uICgpIHsKICAgICAgICAgICAgQ29yZS5BSkFYLkZvcm1VcGRhdGUoJCgnI0NvbXBvc2UnKSwgJ0FKQVhVcGRhdGUnLCAnTmV3UXVldWVJRCcsIFsnVHlwZUlEJywgJ1NlcnZpY2VJRCcsICdOZXdPd25lcklEJywgJ05ld1Jlc3BvbnNpYmxlSUQnLCAnTmV3U3RhdGVJRCcsICdOZXdQcmlvcml0eUlEJywgJ1N0YW5kYXJkVGVtcGxhdGVJRCddLmNvbmNhdChEeW5hbWljRmllbGROYW1lcykpOwogICAgICAgIH0pOwoKICAgICAgICAvLyBCaW5kIGV2ZW50IHRvIFN0YW5kYXJkVGVtcGxhdGUgZmllbGQuCiAgICAgICAgJCgnI1N0YW5kYXJkVGVtcGxhdGVJRCcpLm9uKCdjaGFuZ2UnLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgIENvcmUuQWdlbnQuVGlja2V0QWN0aW9uLkNvbmZpcm1UZW1wbGF0ZU92ZXJ3cml0ZSgnUmljaFRleHQnLCAkKHRoaXMpLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgICAgICBDb3JlLkFKQVguRm9ybVVwZGF0ZSgkKCcjQ29tcG9zZScpLCAnQUpBWFVwZGF0ZScsICdTdGFuZGFyZFRlbXBsYXRlSUQnLCBbJ1JpY2hUZXh0RmllbGQnXSk7CiAgICAgICAgICAgIH0pOwogICAgICAgICAgICByZXR1cm4gZmFsc2U7CiAgICAgICAgfSk7CgogICAgICAgIC8vIEJpbmQgY2xpY2sgZXZlbnQgdG8gQ3JlYXRlQXJ0aWNsZSBjaGVja2JveCBhbmQgdG9nZ2xlIHdpZGdldC4KICAgICAgICAkKCcjQ3JlYXRlQXJ0aWNsZSwgI1dpZGdldEFydGljbGUgLldpZGdldEFjdGlvbi5Ub2dnbGUnKS5vbignY2xpY2snLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgICQoJyNXaWRnZXRBcnRpY2xlIC5WYWxpZGF0ZV9EZXBlbmRpbmdSZXF1aXJlZEFORC5WYWxpZGF0ZV9EZXBlbmRpbmdfQ3JlYXRlQXJ0aWNsZScpLmVhY2goZnVuY3Rpb24gKEluZGV4LCBFbGVtZW50KSB7CiAgICAgICAgICAgICAgICB2YXIgJEVsZW1lbnQgPSAkKEVsZW1lbnQpOwogICAgICAgICAgICAgICAgdmFyIENsb3Nlc3RDbGFzcyA9ICdGaWVsZCc7CiAgICAgICAgICAgICAgICBpZiAoJEVsZW1lbnQuYXR0cignaWQnKSA9PT0gJ1JpY2hUZXh0JykgewogICAgICAgICAgICAgICAgICAgIENsb3Nlc3RDbGFzcyA9ICdSaWNoVGV4dEZpZWxkJzsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIGlmICgkKCcjQ3JlYXRlQXJ0aWNsZScpLnByb3AoJ2NoZWNrZWQnKSAmJiAkKCcjV2lkZ2V0QXJ0aWNsZScpLmhhc0NsYXNzKCdFeHBhbmRlZCcpKSB7CiAgICAgICAgICAgICAgICAgICAgJEVsZW1lbnQuY2xvc2VzdCgnLicgKyBDbG9zZXN0Q2xhc3MpCiAgICAgICAgICAgICAgICAgICAgICAgIC5wcmV2KCdsYWJlbCcpCiAgICAgICAgICAgICAgICAgICAgICAgIC5hZGRDbGFzcygnTWFuZGF0b3J5JykKICAgICAgICAgICAgICAgICAgICAgICAgLnByZXBlbmQoJzxzcGFuIGNsYXNzPSJNYXJrZXIiPio8L3NwYW4+Jyk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICBlbHNlIHsKICAgICAgICAgICAgICAgICAgICAkRWxlbWVudC5jbG9zZXN0KCcuJyArIENsb3Nlc3RDbGFzcykKICAgICAgICAgICAgICAgICAgICAgICAgLnByZXYoJ2xhYmVsJykKICAgICAgICAgICAgICAgICAgICAgICAgLnJlbW92ZUNsYXNzKCdNYW5kYXRvcnknKQogICAgICAgICAgICAgICAgICAgICAgICAuZmluZCgnc3BhbicpCiAgICAgICAgICAgICAgICAgICAgICAgIC5yZW1vdmUoKTsKICAgICAgICAgICAgICAgICAgICBDb3JlLkZvcm0uVmFsaWRhdGUuVW5IaWdobGlnaHRFcnJvcihFbGVtZW50KTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfSk7CiAgICAgICAgfSk7CiAgICB9OwoKICAgIC8qKgogICAgICogQHByaXZhdGUKICAgICAqIEBuYW1lIEZpZWxkVXBkYXRlCiAgICAgKiBAbWVtYmVyb2YgQ29yZS5BZ2VudC5UaWNrZXRBY3Rpb25Db21tb24KICAgICAqIEBmdW5jdGlvbgogICAgICogQHBhcmFtIHtTdHJpbmd9IFZhbHVlIC0gRmllbGRJRAogICAgICogQHBhcmFtIHtBcnJheX0gTW9kaWZpZWRGaWVsZHMgLSBGaWVsZHMKICAgICAqIEBkZXNjcmlwdGlvbgogICAgICogICAgICBDcmVhdGUgb24gY2hhbmdlIGV2ZW50IGhhbmRsZXIKICAgICAqLwogICAgZnVuY3Rpb24gRmllbGRVcGRhdGUgKFZhbHVlLCBNb2RpZmllZEZpZWxkcykgewogICAgICAgICQoJyMnICsgVmFsdWUpLm9uKCdjaGFuZ2UnLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgIENvcmUuQUpBWC5Gb3JtVXBkYXRlKCQoJyNDb21wb3NlJyksICdBSkFYVXBkYXRlJywgVmFsdWUsIE1vZGlmaWVkRmllbGRzKTsKICAgICAgICB9KTsKICAgIH0KCiAgICBDb3JlLkluaXQuUmVnaXN0ZXJOYW1lc3BhY2UoVGFyZ2V0TlMsICdBUFBfTU9EVUxFJyk7CgogICAgcmV0dXJuIFRhcmdldE5TOwovLyAtLS0KLy8gT1RSU01hc3RlclNsYXZlCi8vIC0tLQovL30oQ29yZS5BZ2VudC5UaWNrZXRBY3Rpb25Db21tb24gfHwge30pKTsKfShDb3JlLkFnZW50LlRpY2tldE1hc3RlclNsYXZlIHx8IHt9KSk7Ci8vIC0tLQo=
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package var::packagesetup::OTRSMasterSlave;

use strict;
use warnings;

use Kernel::System::VariableCheck qw(:all);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Cache',
    'Kernel::System::DB',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::DynamicFieldValue',
    'Kernel::System::LinkObject',
    'Kernel::System::Log',
    'Kernel::System::SysConfig',
    'Kernel::System::Ticket',
);

=head1 NAME

var::packagesetup::OTRSMasterSlave - code to execute during package installation

=head1 SYNOPSIS

Functions for installing the OTRSMasterSlave package.

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object. Do not use it directly, instead use:

    use Kernel::System::ObjectManager;
    local $Kernel::OM = Kernel::System::ObjectManager->new();
    my $CodeObject = $Kernel::OM->Get('var::packagesetup::OTRSMasterSlave');

=cut

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

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

    # Force a reload of ZZZAuto.pm to get the fresh configuration values.
    for my $Module ( sort keys %INC ) {
        if ( $Module =~ m/ZZZAA?uto\.pm$/ ) {
            delete $INC{$Module};
        }
    }

    $Kernel::OM->ObjectsDiscard(
        Objects => ['Kernel::Config'],
    );

    # get dynamic fields list
    $Self->{DynamicFieldsList} = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldListGet(
        Valid      => 0,
        ObjectType => ['Ticket'],
    );

    if ( !IsArrayRefWithData( $Self->{DynamicFieldsList} ) ) {
        $Self->{DynamicFieldsList} = [];
    }

    # create a dynamic field lookup table (by name)
    DYNAMICFIELD:
    for my $DynamicField ( @{ $Self->{DynamicFieldsList} } ) {
        next DYNAMICFIELD if !$DynamicField;
        next DYNAMICFIELD if !IsHashRefWithData($DynamicField);
        next DYNAMICFIELD if !$DynamicField->{Name};
        $Self->{DynamicFieldLookup}->{ $DynamicField->{Name} } = $DynamicField;
    }

    return $Self;
}

=item CodeInstall()

run the code install part

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

=cut

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

    # if we got an installed version of MasterSlave, migrate the data
    # otherwise set the dynamic fields
    my $MasterSlaveDynamicFieldID = $Self->_CheckMasterSlaveData();
    if ($MasterSlaveDynamicFieldID) {

        $Self->_MigrateOTRSMasterSlave(
            DynamicFieldID => $MasterSlaveDynamicFieldID,
            MasterSlave    => 1,
        );
    }
    else {
        $Self->_SetDynamicFields();
    }

    # set dashboard config if needed
    $Self->_SetDashboardConfig();

    return 1;
}

=item CodeReinstall()

run the code reinstall part

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

=cut

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

    return 1;
}

=item CodeUpgrade()

run the code upgrade part

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

=cut

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

    $Self->_SetDynamicFields();

    # set dashboard config if needed
    $Self->_SetDashboardConfig();

    return 1;
}

=item CodeUpgrade125()

run the code upgrade part for versions prior to 1.2.5

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

=cut

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

    # upgrade/migrate only in case there is a installed
    # version of OTRSMasterSlave version < 1.2.5
    $Self->_MigrateOTRSMasterSlave();

    return 1;
}

=item CodeUpgradeFromLowerThan_4_0_91()

This function is only executed if the installed module version is smaller than 4.0.91.

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

=cut

sub CodeUpgradeFromLowerThan_4_0_91 {    ## no critic
    my ( $Self, %Param ) = @_;

    # change configurations to match the new module location.
    $Self->_MigrateConfigs();

    return 1;
}

=item CodeUninstall()

run the code uninstall part

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

=cut

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

    $Self->_RemoveDynamicFields();

    return 1;
}

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

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # get dynamic field names from SysConfig
    my $MasterSlaveDynamicField = $ConfigObject->Get('MasterSlave::DynamicField') || 'MasterSlave';

    # set attributes of new dynamic fields
    my %NewDynamicFields = (
        $MasterSlaveDynamicField => {
            Name       => $MasterSlaveDynamicField,
            Label      => 'Master Ticket',
            FieldType  => 'MasterSlave',
            ObjectType => 'Ticket',
            Config     => {
                DefaultValue       => '',
                PossibleNone       => 1,
                TranslatableValues => 1,
            },
            InternalField => 1,
        },
    );

    # set MaxFieldOrder (needed for adding new dynamic fields)
    my $MaxFieldOrder = 0;
    if ( !IsArrayRefWithData( $Self->{DynamicFieldsList} ) ) {
        $MaxFieldOrder = 1;
    }
    else {
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( @{ $Self->{DynamicFieldsList} } ) {
            if ( int $DynamicFieldConfig->{FieldOrder} > int $MaxFieldOrder ) {
                $MaxFieldOrder = $DynamicFieldConfig->{FieldOrder};
            }
        }
    }

    # get dynamic field object
    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

    for my $NewFieldName ( sort keys %NewDynamicFields ) {

        # check if dynamic field already exists
        if ( IsHashRefWithData( $Self->{DynamicFieldLookup}->{$NewFieldName} ) ) {

            # get the dynamic field configuration
            my $DynamicFieldConfig = $Self->{DynamicFieldLookup}->{$NewFieldName};

            my $Update;

            # update field configuration if it was other than MasterSlave (e.g. Dropdown)
            if ( $DynamicFieldConfig->{FieldType} ne 'MasterSlave' ) {
                my $ID = $DynamicFieldConfig->{ID};
                %{$DynamicFieldConfig} = ( %{$DynamicFieldConfig}, %{ $NewDynamicFields{$NewFieldName} } );
                $Update = 1;
            }

            # if dynamic field exists make sure is valid
            if ( $DynamicFieldConfig->{ValidID} ne '1' ) {
                $Update = 1;
            }

            if ($Update) {

                my $Success = $DynamicFieldObject->DynamicFieldUpdate(
                    %{$DynamicFieldConfig},
                    ValidID => 1,
                    Reorder => 0,
                    UserID  => 1,
                );

                if ( !$Success ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "Could not set dynamic field '$NewFieldName' to valid!",
                    );
                }
            }
            if ( $DynamicFieldConfig->{InternalField} ne '1' ) {

                # update InternalField value manually since API does not support
                # internal_field update
                my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
                    SQL => '
                        UPDATE dynamic_field
                        SET internal_field = 1
                        WHERE id = ?',
                    Bind => [ \$DynamicFieldConfig->{ID} ],
                );
                if ( !$Success ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "Could not set dynamic field '$NewFieldName' as internal!",
                    );
                }

                # clean dynamic field cache
                $Kernel::OM->Get('Kernel::System::Cache')->CleanUp(
                    Type => 'DynamicField',
                );
            }
        }

        # otherwise create it
        else {
            $MaxFieldOrder++;
            my $ID = $DynamicFieldObject->DynamicFieldAdd(
                %{ $NewDynamicFields{$NewFieldName} },
                FieldOrder => $MaxFieldOrder,
                ValidID    => 1,
                UserID     => 1,
            );

            if ( !$ID ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Could not add dynamic field '$NewFieldName'!",
                );
            }
        }
    }

    # enable dynamic field for ticket zoom
    # get old configuration
    my $WindowConfig  = $ConfigObject->Get('Ticket::Frontend::AgentTicketZoom');
    my %DynamicFields = %{ $WindowConfig->{DynamicField} || {} };

    $DynamicFields{$MasterSlaveDynamicField} =
        defined $DynamicFields{$MasterSlaveDynamicField}
        ? $DynamicFields{$MasterSlaveDynamicField}
        : 1;

    my $SysConfigObject = $Kernel::OM->Get('Kernel::System::SysConfig');
    return 0 if !$SysConfigObject->SettingsSet(
        UserID   => 1,
        Comments => 'OTRSMasterSlave - deploy AgentTicketZoom dynamic fields',
        Settings => [
            {
                Name           => 'Ticket::Frontend::AgentTicketZoom###DynamicField',
                EffectiveValue => \%DynamicFields,
                IsValid        => 1,
            },
        ],
    );

    return 1;
}

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

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # get dynamic field names from SysConfig
    my $MasterSlaveDynamicField = $ConfigObject->Get('MasterSlave::DynamicField')
        || 'MasterSlave';

    # check if there isn't already a dynamic field with the destined name
    return 1 if IsHashRefWithData( $Self->{DynamicFieldLookup}->{$MasterSlaveDynamicField} );

    my $OldMasterSlaveDynamicFieldID;

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # check if we got a DynamicFieldID
    if ( $Param{DynamicFieldID} ) {
        $OldMasterSlaveDynamicFieldID = $Param{DynamicFieldID};
    }
    else {

        # if not: get the migrated field ID by searching for possible data
        $DBObject->Prepare(
            SQL => "SELECT dfv.field_id FROM dynamic_field_value dfv "
                . "WHERE dfv.value_text LIKE 'SlaveOf:%' OR dfv.value_text = 'Master'",
            Limit => 1,
        );

        while ( my @Row = $DBObject->FetchrowArray() ) {
            $OldMasterSlaveDynamicFieldID = $Row[0];
        }
    }

    # check if we found a valid ID
    return 0 if !$OldMasterSlaveDynamicFieldID;

    if ( $Param{MasterSlave} ) {
        $Self->_MigrateMasterSlaveData(
            DynamicFieldID => $OldMasterSlaveDynamicFieldID,
        );
    }

    # get dynamic field object
    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

    # try to get the dynamic field  data (for field order etc.)
    my $OldDynamicField = $DynamicFieldObject->DynamicFieldGet(
        ID => $OldMasterSlaveDynamicFieldID,
    );

    return 0 if !IsHashRefWithData($OldDynamicField);

    # update the name of the dynamic field to MasterSlave and store it
    # and return the result of this function
    return 0 if !$DynamicFieldObject->DynamicFieldUpdate(
        %{$OldDynamicField},
        Name       => $MasterSlaveDynamicField,
        Label      => 'Master Ticket',
        FieldType  => 'MasterSlave',
        ObjectType => 'Ticket',
        Config     => {
            DefaultValue       => '',
            PossibleNone       => 1,
            TranslatableValues => 1,
        },
        InternalField => 1,
        ValidID       => 1,
        Reorder       => 0,
        UserID        => 1,
    );

    # update InternalField value manually since API does not support internal_field update
    my $Success = $DBObject->Do(
        SQL => '
            UPDATE dynamic_field
            SET internal_field = 1
            WHERE id = ?',
        Bind => [ \$OldMasterSlaveDynamicFieldID->{ID} ],
    );
    if ( !$Success ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Could not set dynamic field '$MasterSlaveDynamicField' as internal!",
        );
    }

    # clean dynamic field cache
    $Kernel::OM->Get('Kernel::System::Cache')->CleanUp(
        Type => 'DynamicField',
    );

    # activate the DynamicField in ticket details block
    my $KeyString       = "Ticket::Frontend::AgentTicketZoom";
    my $ExistingSetting = $ConfigObject->Get($KeyString) || {};
    my %ValuesToSet     = %{ $ExistingSetting->{DynamicField} || {} };
    $ValuesToSet{MasterSlave} = 1;

    return if !$Kernel::OM->Get('Kernel::System::SysConfig')->SettingsSet(
        UserID   => 1,
        Comments => 'OTRSMasterSlave - deploy dynamic fields.',
        Settings => [
            {
                Name           => $KeyString . "###DynamicField",
                EffectiveValue => \%ValuesToSet,
                IsValid        => 1,
            },
        ],
    );

    return 1;
}

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

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # if not: get the migrated field ID by searching for possible data
    $DBObject->Prepare(
        SQL   => "SELECT dfv.field_id FROM dynamic_field_value dfv WHERE dfv.value_text = 'Slave'",
        Limit => 1,
    );

    my $OldMasterSlaveDynamicFieldID;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        $OldMasterSlaveDynamicFieldID = $Row[0];
    }

    return $OldMasterSlaveDynamicFieldID;
}

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

    if ( !$Param{DynamicFieldID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need DynamicFieldID for MasterSlave data migration!",
        );
        return;
    }

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # get all the slave ticket ids we have to update
    $DBObject->Prepare(
        SQL => "
            SELECT dfv.object_id
            FROM dynamic_field_value dfv
            WHERE dfv.value_text = 'Slave'
                AND dfv.field_id = ?",
        Bind  => [ \$Param{DynamicFieldID} ],
        Limit => 50,
    );

    my @DynamicFieldData;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        push( @DynamicFieldData, $Row[0] );
    }

    # try to get the dynamic field data for setting new value
    my $DynamicFieldConfig = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldGet(
        ID => $Param{DynamicFieldID},
    );

    return if !IsHashRefWithData($DynamicFieldConfig);

    # loop over the ticket ids we have to update
    OLDSLAVESTYLE:
    for my $TicketID (@DynamicFieldData) {

        # get linked objects
        my $LinkListWithData = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkListWithData(
            Object    => 'Ticket',
            Key       => $TicketID,
            State     => 'Valid',
            Type      => 'ParentChild',
            Direction => 'Source',
            UserID    => 1,
        );

        # check what tickets might be the master
        my @ParentTicketIDs = keys %{ $LinkListWithData->{Ticket}->{ParentChild}->{Source} };

        my $TicketNumber;

        # if we got more than one possible master ticket, try to determine which
        # one is the master we are looking for
        if ($#ParentTicketIDs) {

            $DBObject->Prepare(
                SQL => "
                    SELECT dfv.object_id
                    FROM dynamic_field_value dfv
                    WHERE dfv.value_text = 'Master'
                        AND dfv.field_id = ?
                        AND dfv.object_id IN ("
                    . join( ',', map { $DBObject->Quote($_) } @ParentTicketIDs )
                    . ')',
                Bind  => [ \$Param{DynamicFieldID} ],
                Limit => 1,
            );

            my @ParentTicketIDs;
            while ( my @Row = $DBObject->FetchrowArray() ) {
                push( @ParentTicketIDs, $Row[0] );
            }

            if ($#ParentTicketIDs) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message =>
                        "Couldn't determine MasterTicket for TicketID $TicketID (Possible Masters: "
                        . join ', ', @ParentTicketIDs
                        . ")!",
                );
            }

            $TicketNumber = $Kernel::OM->Get('Kernel::System::Ticket')->TicketNumberLookup(
                TicketID => $ParentTicketIDs[0],
                UserID   => 1,
            );
        }
        else {
            $TicketNumber
                = $LinkListWithData->{Ticket}->{ParentChild}->{Source}->{ $ParentTicketIDs[0] }->{TicketNumber};
        }

        # update the dynamic field value to valid
        # data for OTRSMasterSlave
        my $Success = $Kernel::OM->Get('Kernel::System::DynamicField::Backend')->ValueSet(
            DynamicFieldConfig => $DynamicFieldConfig,
            ObjectID           => $TicketID,
            Value              => 'SlaveOf:' . $TicketNumber,
            UserID             => 1,
        );

        if ( !$Success ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Error while migrating MasterSlave DynamicField '"
                    . $Param{DynamicFieldID}
                    . "' for TicketID '"
                    . $TicketID
                    . "'!",
            );
            return;
        }
    }

    # do some recursion if we got more than 50 slave tickets in this run
    # doing this to have a better performance
    if ( 50 == scalar @DynamicFieldData ) {
        my $Success = $Self->_MigrateMasterSlaveData(
            DynamicFieldID => $Param{DynamicFieldID},
        );

        if ( !$Success ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Error while migrating MasterSlave data!",
            );
            return;
        }
    }

    return 1;
}

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

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # get dynamic field names from SysConfig
    my $MasterSlaveDynamicField = $ConfigObject->Get('MasterSlave::DynamicField') || 'MasterSlave';

    # check if dynamic field already exists
    if ( IsHashRefWithData( $Self->{DynamicFieldLookup}->{$MasterSlaveDynamicField} ) ) {

        # get the field ID
        my $DynamicFieldID = $Self->{DynamicFieldLookup}->{$MasterSlaveDynamicField}->{ID};

        # delete all field values
        my $ValuesDeleteSuccess = $Kernel::OM->Get('Kernel::System::DynamicFieldValue')->AllValuesDelete(
            FieldID => $DynamicFieldID,
            UserID  => 1,
        );

        if ($ValuesDeleteSuccess) {

            # delete field
            my $Success = $Kernel::OM->Get('Kernel::System::DynamicField')->DynamicFieldDelete(
                ID      => $DynamicFieldID,
                UserID  => 1,
                Reorder => 1,
            );

            if ( !$Success ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Could not delete dynamic field '$MasterSlaveDynamicField'!",
                );
            }
        }
        else {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Could not delete values for dynamic field '$MasterSlaveDynamicField'!",
            );
        }
    }

    my $SysConfigObject = $Kernel::OM->Get('Kernel::System::SysConfig');

    # disable SysConfig settings
    return if !$SysConfigObject->SettingsSet(
        UserID   => 1,
        Comments => 'OTRSMasterSlave - # disable SysConfig settings.',
        Settings => [
            {
                Name           => 'DynamicFields::Driver###MasterSlave',
                EffectiveValue => {
                    DisplayName  => 'Master / Slave',
                    Module       => 'Kernel::System::DynamicField::Driver::MasterSlave',
                    ConfigDialog => 'AdminDynamicFieldMasterSlave',
                    DisabledAdd  => 1,
                },
                IsValid => 0,
            },
            {
                Name           => 'PreApplicationModule###AgentPreMasterSlave',
                EffectiveValue => 'Kernel::Modules::AgentPreMasterSlave',
                IsValid        => 0,
            },
        ],
    );

    # discard config object and dynamic field backend to prevent error messages due missing driver
    $Kernel::OM->ObjectsDiscard(
        Objects => [ 'Kernel::Config', 'Kernel::System::DynamicField::Backend' ],
    );

    # disable dynamic field for ticket zoom
    # get old configuration
    my $WindowConfig  = $ConfigObject->Get('Ticket::Frontend::AgentTicketZoom');
    my %DynamicFields = %{ $WindowConfig->{DynamicField} || {} };

    if ( defined $DynamicFields{$MasterSlaveDynamicField} ) {
        $DynamicFields{$MasterSlaveDynamicField} = 0;
    }

    return if !$SysConfigObject->SettingsSet(
        UserID   => 1,
        Comments => 'OTRSMasterSlave - deploy dynamic fields.',
        Settings => [
            {
                Name           => 'Ticket::Frontend::AgentTicketZoom###DynamicField',
                EffectiveValue => \%DynamicFields,
                IsValid        => 1,
            },
        ],
    );

    return 1;
}

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

    # get dynamic field names from SysConfig
    my $MasterSlaveDynamicField = $Kernel::OM->Get('Kernel::Config')->Get('MasterSlave::DynamicField') || 'MasterSlave';

    # if MasterSlave dynamic field is 'MasterSlave' the config is already set, nothing else to do
    return 1 if ( $MasterSlaveDynamicField eq 'MasterSlave' );

    # otherwise set the new config
    # attributes common for both Master and Slave widgets
    my %CommonConfig = (
        Module        => 'Kernel::Output::HTML::DashboardTicketGeneric',
        Filter        => 'All',
        Time          => 'Age',
        Limit         => 10,
        Permission    => 'rw',
        Block         => 'ContentLarge',
        Group         => '',
        Default       => 1,
        CacheTTLLocal => 0.5,
    );

    # attributes for Master widget
    my %MasterConfig = (
        Title       => 'Master Tickets',
        Description => 'All master tickets',
        Attributes  => 'DynamicField_' . $MasterSlaveDynamicField . '_Equals=Master;',
    );

    # attributes for Slave widget
    my %SlaveConfig = (
        Title       => 'Slave Tickets',
        Description => 'All slave tickets',
        Attributes  => 'DynamicField_' . $MasterSlaveDynamicField . '_Like=Slave*;',
    );

    # get SysConfig object
    my $SysConfigObject = $Kernel::OM->Get('Kernel::System::SysConfig');

    # write configurations
    return if !$SysConfigObject->SettingsSet(
        UserID   => 1,
        Comments => 'OTRSMasterSlave - deploy dynamic fields for dashboard.',
        Settings => [
            {
                Name           => 'DashboardBackend###0900-TicketMaster',
                EffectiveValue => {
                    %CommonConfig,
                    %MasterConfig,
                },
                IsValid => 1,
            },
            {
                Name           => 'DashboardBackend###0910-TicketSlave',
                EffectiveValue => {
                    %CommonConfig,
                    %SlaveConfig,
                },
                IsValid => 1,
            },
        ],
    );

    return 1;
}

sub _MigrateConfigs {

    # create needed objects
    my $SysConfigObject = $Kernel::OM->Get('Kernel::System::SysConfig');
    my $ConfigObject    = $Kernel::OM->Get('Kernel::Config');

    # migrate master slave ticket menu SysConfig
    # get setting content for master slave SysConfig
    my $Setting = $ConfigObject->Get('Ticket::Frontend::MenuModule');

    if ( $Setting->{'480-MasterSlave'}->{Module} ) {

        # update module location
        $Setting->{'480-MasterSlave'}->{Module} = "Kernel::Output::HTML::TicketMenu::Generic";

        # set new setting
        $SysConfigObject->SettingsSet(
            UserID   => 1,
            Comments => 'OTRSMasterSlave - deploy menu module settings.',
            Settings => [
                {
                    Name           => 'Ticket::Frontend::MenuModule###480-MasterSlave',
                    EffectiveValue => $Setting->{'480-MasterSlave'},
                    IsValid        => 1,
                },
            ],
        );
    }

    # migrate master slave dashboard SysConfig
    # get setting content for master slave SysConfig
    $Setting = $ConfigObject->Get('DashboardBackend');

    BACKEND:
    for my $Backend (qw(0900-TicketMaster 0910-TicketSlave)) {
        next BACKEND if !$Setting->{$Backend}->{Module};

        # update module location
        $Setting->{$Backend}->{Module} = "Kernel::Output::HTML::Dashboard::TicketGeneric";

        # set new setting
        $SysConfigObject->SettingsSet(
            UserID   => 1,
            Comments => 'OTRSMasterSlave - deploy dashboard backend settings.',
            Settings => [
                {
                    Name           => "DashboardBackend###$Backend",
                    EffectiveValue => $Setting->{$Backend},
                    IsValid        => 1,
                },
            ],
        );
    }

    return 1;
}
1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

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

=cut

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
Get('var::packagesetup::OTRSMasterSlave')->CodeInstall();
]]>
Get('var::packagesetup::OTRSMasterSlave')->CodeUpgrade125();
]]>
Get('var::packagesetup::OTRSMasterSlave')->CodeUpgrade();
]]>
{Name}->{Content};
$Kernel::OM->Get($CodeModule)->CodeUpgradeFromLowerThan_4_0_91();
]]>
Get('var::packagesetup::OTRSMasterSlave')->CodeReinstall();
]]>
Get('var::packagesetup::OTRSMasterSlave')->CodeUninstall();
]]>
WELCOME
You are about to install the OTRS package OTRSMasterSlave.
((enjoy))
]]>
BIENVENIDO
Usted está a punto de instalar el paquete OTRSMasterSlave de OTRS.
((enjoy))
]]>
ÜDVÖZÖLJÜK
Ön az OTRSMasterSlave OTRS csomag telepítésére készül.
((enjoy))
]]>
WELCOME
You are about to upgrade the OTRS package OTRSMasterSlave.
((enjoy))
]]>
BIENVENIDO
Usted está apunto de actualizar el paquete OTRSMasterSlave de OTRS.
((enjoy))
]]>
ÜDVÖZÖLJÜK
Ön az OTRSMasterSlave OTRS csomag frissítésére készül.
((enjoy))
]]>
ATTENTION
If you uninstall this package, ParentChild ticket links will remain, but all ticket Master
Slave relations that were created during the use of this package be deleted.
All data from these relations will be irrevocably lost!
((enjoy))
]]>
ATENCIÓN
Si usted desinstala este paquete, los vínculos de tickets PadreHijo
permanecerán, pero todas relaciones de tickets Master Slave creadas durante
el uso de este paquete serán borradas.
¡Todos los datos de esas relaciones se perderán irrevocablemente!.
((enjoy))
]]>
FIGYELEM
Ha eltávolítja ezt a csomagot, akkor a szülő-gyermek jegy hivatkozások meg fognak maradni,
de az összes olyan mester-alárendelt kapcsolat törlésre kerül, amelyek a csomag használata
során lettek létrehozva.
Ezekből a kapcsolatokból az összes adat visszavonhatatlanul el fog veszni!
((enjoy))
]]>