<?xml version="1.0" encoding="utf-8" ?>
<otrs_package version="1.1">
    <Name>DiscreteSystemAddresses</Name>
    <Version>11.0.4</Version>
    <Vendor>Rother OSS GmbH</Vendor>
    <URL>https://otobo.io/</URL>
    <License>GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007</License>
    <ChangeLog Date="2026-04-24 10:36:54" Version="11.0.4">Update to OTOBO 11.0.16.</ChangeLog>
    <ChangeLog Date="2026-02-04 16:47:44" Version="11.0.3">Update to OTOBO 11.0.15. Switch to ITSMCore compatibility greater than 11.0.5.</ChangeLog>
    <ChangeLog Date="2025-05-06 12:15:59" Version="11.0.2">Update to OTOBO 11.0.9.</ChangeLog>
    <ChangeLog Date="2024-07-09 09:04:17" Version="11.0.1">Updated package to 11.0.</ChangeLog>
    <Description Lang="en">This extension facilitates communication between system addresses within OTOBO (e.g. in cc), by linking tickets in different queues to each other and using address pools to ensure a new article is added to all relevant tickets within the system (e.g. one in the IT Queue and one in HR).</Description>
    <Description Lang="de">Diese Erweiterung erleichtert die Kommunikation zwischen Systemadressen innerhalb von OTOBO (z. B. in cc), indem sie Tickets in verschiedenen Queues miteinander verknüpft und Adress-Bereiche verwendet, um sicherzustellen, dass ein neuer Artikel zu allen relevanten Tickets innerhalb des Systems hinzugefügt wird (z. B. eines in der IT-Queue und eine in der Personalabteilung).</Description>
    <Framework Minimum="11.0.9">11.0.x</Framework>
    <IntroInstall Lang="en" Title="Install Information" Type="pre">
        &lt;br/&gt;
        &lt;strong&gt;WELCOME&lt;/strong&gt;
        &lt;br/&gt;
        &lt;br/&gt;
        You are about to install the OTOBO package DiscreteSystemAddresses.&lt;br/&gt;
        &lt;br/&gt;
        &lt;br/&gt;
        ((enjoy))&lt;br/&gt;
        &lt;br/&gt;
    </IntroInstall>
    <IntroInstall Lang="de" Title="Installation Information" Type="pre">
        &lt;br/&gt;
        &lt;strong&gt;WILLKOMMEN&lt;/strong&gt;
        &lt;br/&gt;
        &lt;br/&gt;
        Sie sind im Begriff das OTOBO-Paket DiscreteSystemAddresses zu installieren.&lt;br/&gt;
        &lt;br/&gt;
        &lt;br/&gt;
        ((enjoy))&lt;br/&gt;
        &lt;br/&gt;
    </IntroInstall>
    <BuildCommitID>fb9c87a93f420ab31e05fcc7289b0be5dda106ea</BuildCommitID>
    <BuildDate>2026-04-24 10:36:57</BuildDate>
    <BuildHost>opms.rother-oss.com</BuildHost>
    <Filelist>
        <File Location="Kernel/Autoload/TicketSubjectCleanInterdivisional.pm" Permission="660" Encode="Base64">IyAtLQojIE9UT0JPIGlzIGEgd2ViLWJhc2VkIHRpY2tldGluZyBzeXN0ZW0gZm9yIHNlcnZpY2Ugb3JnYW5pc2F0aW9ucy4KIyAtLQojIENvcHlyaWdodCAoQykgMjAxOS0yMDI2IFJvdGhlciBPU1MgR21iSCwgaHR0cHM6Ly9vdG9iby5pby8KIyAtLQojIFRoaXMgcHJvZ3JhbSBpcyBmcmVlIHNvZnR3YXJlOiB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IgbW9kaWZ5IGl0IHVuZGVyCiMgdGhlIHRlcm1zIG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkgdGhlIEZyZWUgU29mdHdhcmUKIyBGb3VuZGF0aW9uLCBlaXRoZXIgdmVyc2lvbiAzIG9mIHRoZSBMaWNlbnNlLCBvciAoYXQgeW91ciBvcHRpb24pIGFueSBsYXRlciB2ZXJzaW9uLgojIFRoaXMgcHJvZ3JhbSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLCBidXQgV0lUSE9VVAojIEFOWSBXQVJSQU5UWTsgd2l0aG91dCBldmVuIHRoZSBpbXBsaWVkIHdhcnJhbnR5IG9mIE1FUkNIQU5UQUJJTElUWSBvciBGSVRORVNTCiMgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiBTZWUgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGZvciBtb3JlIGRldGFpbHMuCiMgWW91IHNob3VsZCBoYXZlIHJlY2VpdmVkIGEgY29weSBvZiB0aGUgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UKIyBhbG9uZyB3aXRoIHRoaXMgcHJvZ3JhbS4gSWYgbm90LCBzZWUgPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvPi4KIyAtLQoKdXNlIEtlcm5lbDo6U3lzdGVtOjpUaWNrZXQ7ICAgICMjIG5vIGNyaXRpYyAoTW9kdWxlczo6UmVxdWlyZUV4cGxpY2l0UGFja2FnZSkKCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OlRpY2tldDsgICAgIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRmlsZW5hbWVNYXRjaGVzUGFja2FnZSkKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6Q29uZmlnJywKICAgICdLZXJuZWw6OlN5c3RlbTo6TGlua09iamVjdCcsCik7CgojIGRpc2FibGUgcmVkZWZpbmUgd2FybmluZ3MgaW4gdGhpcyBzY29wZQp7CiAgICBubyB3YXJuaW5ncyAncmVkZWZpbmUnOyAgICAjIyBubyBjcml0aWMgcXcoVGVzdGluZ0FuZERlYnVnZ2luZzo6UHJvaGliaXROb1dhcm5pbmdzKQoKICAgICMgYmFja3VwIG9yaWdpbmFsIHJvdXRpbmVzCiAgICBteSAkVGlja2V0U3ViamVjdENsZWFuID0gXCZLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpUaWNrZXRTdWJqZWN0Q2xlYW47CgogICAgIyBSZW1vdmUgaW50ZXJkaXZpc2lvbmFsIHRpY2tldCBudW1iZXJzCiAgICAqS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6VGlja2V0U3ViamVjdENsZWFuID0gc3ViIHsKICAgICAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICAgICAjIHN0YW5kYXJkIGNsZWFuZWQgc3ViamVjdAogICAgICAgIG15ICRTdWJqZWN0ID0gJnskVGlja2V0U3ViamVjdENsZWFufShAXyk7CgogICAgICAgIHJldHVybiBpZiAhZGVmaW5lZCAkU3ViamVjdDsKCiAgICAgICAgbXkgJENvbmZpZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CgogICAgICAgICMgZ2V0IGNvbmZpZyBvcHRpb25zCiAgICAgICAgbXkgJFRpY2tldEhvb2sgICAgICAgID0gJENvbmZpZ09iamVjdC0+R2V0KCdUaWNrZXQ6Okhvb2snKTsKICAgICAgICBteSAkVGlja2V0SG9va0RpdmlkZXIgPSAkQ29uZmlnT2JqZWN0LT5HZXQoJ1RpY2tldDo6SG9va0RpdmlkZXInKTsKCiAgICAgICAgcmV0dXJuICRTdWJqZWN0IGlmICRTdWJqZWN0ICF+IC8kVGlja2V0SG9vay87CiAgICAgICAgcmV0dXJuICRTdWJqZWN0IGlmICRDb25maWdPYmplY3QtPkdldCgnVGlja2V0OjpTdWJqZWN0Q2xlYW5BbGxOdW1iZXJzJyk7CgogICAgICAgIG15ICRUaWNrZXRJRCA9ICRTZWxmLT5UaWNrZXRJRExvb2t1cCgKICAgICAgICAgICAgVGlja2V0TnVtYmVyID0+ICRQYXJhbXtUaWNrZXROdW1iZXJ9LAogICAgICAgICAgICBVc2VySUQgICAgICAgPT4gMSwKICAgICAgICApOwoKICAgICAgICAjIGdldCBsaW5rZWQgdGlja2V0cwogICAgICAgIG15ICVMaW5rZWRUaWNrZXRzID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxpbmtPYmplY3QnKS0+TGlua0tleUxpc3RXaXRoRGF0YSgKICAgICAgICAgICAgT2JqZWN0MSA9PiAnVGlja2V0JywKICAgICAgICAgICAgS2V5MSAgICA9PiAkVGlja2V0SUQsCiAgICAgICAgICAgIE9iamVjdDIgPT4gJ1RpY2tldCcsCiAgICAgICAgICAgIFN0YXRlICAgPT4gJ1ZhbGlkJywKICAgICAgICAgICAgVHlwZSAgICA9PiAnSW50ZXJkaXZpc2lvbmFsJywKICAgICAgICAgICAgVXNlcklEICA9PiAxLAogICAgICAgICk7CgogICAgICAgIHJldHVybiAkU3ViamVjdCBpZiAhJUxpbmtlZFRpY2tldHM7CgogICAgICAgIGZvciBteSAkVGlja2V0ICggdmFsdWVzICVMaW5rZWRUaWNrZXRzICkgewoKICAgICAgICAgICAgIyByZW1vdmUgYWxsIHBvc3NpYmxlIHRpY2tldCBob29rIGZvcm1hdHMgd2l0aCBbXQogICAgICAgICAgICAkU3ViamVjdCA9fiBzL1xbXHMqXFEkVGlja2V0SG9vazogJFRpY2tldC0+e1RpY2tldE51bWJlcn1cRVxzKlxdXHMqLy9nOwogICAgICAgICAgICAkU3ViamVjdCA9fiBzL1xbXHMqXFEkVGlja2V0SG9vazokVGlja2V0LT57VGlja2V0TnVtYmVyfVxFXHMqXF1ccyovL2c7CiAgICAgICAgICAgICRTdWJqZWN0ID1+IHMvXFtccypcUSRUaWNrZXRIb29rJFRpY2tldEhvb2tEaXZpZGVyJFRpY2tldC0+e1RpY2tldE51bWJlcn1cRVxzKlxdXHMqLy9nOwoKICAgICAgICAgICAgIyByZW1vdmUgYWxsIHBvc3NpYmxlIHRpY2tldCBob29rIGZvcm1hdHMgd2l0aG91dCBbXQogICAgICAgICAgICAkU3ViamVjdCA9fiBzL1xRJFRpY2tldEhvb2s6ICRUaWNrZXQtPntUaWNrZXROdW1iZXJ9XEVccyovL2c7CiAgICAgICAgICAgICRTdWJqZWN0ID1+IHMvXFEkVGlja2V0SG9vazokVGlja2V0LT57VGlja2V0TnVtYmVyfVxFXHMqLy9nOwogICAgICAgICAgICAkU3ViamVjdCA9fiBzL1xRJFRpY2tldEhvb2skVGlja2V0SG9va0RpdmlkZXIkVGlja2V0LT57VGlja2V0TnVtYmVyfVxFXHMqLy9nOwogICAgICAgIH0KCiAgICAgICAgIyB0cmltIHdoaXRlIHNwYWNlIGF0IHRoZSBiZWdpbm5pbmcgb3IgZW5kCiAgICAgICAgJFN1YmplY3QgPX4gcy8oXlxzK3xccyskKS8vOwoKICAgICAgICByZXR1cm4gJFN1YmplY3Q7CiAgICB9Owp9CgoxOwo=</File>
        <File Location="Kernel/Config/Files/XML/AddressPool.xml" Permission="660" Encode="Base64"><?xml version="1.0" encoding="utf-8"?>
<otobo_config version="2.0" init="Application">
    <Setting Name="PostMaster::AddressPool::EmailHeaders" Required="1" Valid="1">
        <Description Translatable="1">Email headers used for address-pool assignment. Secondary headers are not evaluated if pools could be assigned via the primary headers.</Description>
        <Navigation>Core::Email::PostMaster</Navigation>
        <Value>
            <Hash>
                <Item Key="Primary">
                    <Array>
                        <Item>Resent-To</Item>
                        <Item>Envelope-To</Item>
                        <Item>To</Item>
                        <Item>Cc</Item>
                    </Array>
                </Item>
                <Item Key="Secondary">
                    <Array>
                        <Item>Delivered-To</Item>
                        <Item>X-Original-To</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool01" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool02" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool03" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool04" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool05" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool06" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool07" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool08" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool09" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool10" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool11" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool12" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool13" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool14" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool15" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool16" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool17" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool18" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool19" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue"></Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool20" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool21" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool22" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool23" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool24" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::AddressPool###Pool25" Required="0" Valid="0">
        <Description Translatable="1">All email addresses which make up one pool. Queues are assigned implicitely. A default queue can be provided.</Description>
        <Navigation>Core::Email::PostMaster::AddressPool</Navigation>
        <Value>
            <Hash>
                <Item Key="Name"></Item>
                <Item Key="DefaultQueue" ValueType="Entity" ValueEntityType="Queue" ValueRegex="">Postmaster</Item>
                <Item Key="Emails">
                    <Array>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="LinkObject::Type###Interdivisional" Required="1" Valid="1">
        <Description Translatable="1">This setting defines the link type 'Interdivisional'. If the source name and the target name contain the same value, the resulting link is a non-directional one. If the values are different, the resulting link is a directional link.</Description>
        <Navigation>Core::LinkObject</Navigation>
        <Value>
            <Hash>
                <Item Key="SourceName" Translatable="1">Interdivisional</Item>
                <Item Key="TargetName" Translatable="1">Interdivisional</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="LinkObject::PossibleLink###2200" Required="0" Valid="1">
        <Description Translatable="1">Links 2 tickets with a "Interdivisional" type link.</Description>
        <Navigation>Core::LinkObject</Navigation>
        <Value>
            <Hash>
                <Item Key="Object1">Ticket</Item>
                <Item Key="Object2">Ticket</Item>
                <Item Key="Type">Interdivisional</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="PostMaster::PreFilterModule###000-SkipViaMessageID" Required="0" Valid="1">
        <Description Translatable="1">Module to check if an article with the corresponding message id already exists in the system. (E.g. if the mail was received and processed through a different email inbox.) Multiple parsing can be allowed. Pools which already contain an article with the respective message id will still be ignored. Options are "Always", "BouncedEmail" (multiple parsing will only be done for Emails with "Resent-To"-header, e.g. PoolA bounced a mail to PoolB) and "MailAccountQueue" (if dispatching via queue is enabled for a mail account, an email can be moved to the respective inbox, to be evaluated again).</Description>
        <Navigation>Core::Email::PostMaster</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::PostMaster::Filter::SkipViaMessageID</Item>
                <Item Key="AllowMultiParse">
                    <Hash>
                        <Item Key="Always">0</Item>
                        <Item Key="MailAccountQueue">1</Item>
                        <Item Key="BouncedEmail">1</Item>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="AutoloadPerlPackages###1000-TicketSubjectCleanInterdivisionalTicketNumbers" Required="0" Valid="1">
        <Description Translatable="1">Remove ticket numbers of all interdivisionally linked tickets when subject is cleaned.</Description>
        <Navigation>Core::Autoload</Navigation>
        <Value>
            <Array>
                <Item ValueType="String">Kernel::Autoload::TicketSubjectCleanInterdivisional</Item>
            </Array>
        </Value>
    </Setting>
</otobo_config>
</File>
        <File Location="Kernel/Language/de_DiscreteSystemAddresses.pm" Permission="660" Encode="Base64">IyAtLQojIE9UT0JPIGlzIGEgd2ViLWJhc2VkIHRpY2tldGluZyBzeXN0ZW0gZm9yIHNlcnZpY2Ugb3JnYW5pc2F0aW9ucy4KIyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIwIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDE5LTIwMjYgUm90aGVyIE9TUyBHbWJILCBodHRwczovL290b2JvLmlvLwojIC0tCiMgVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU6IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkgaXQgdW5kZXIKIyB0aGUgdGVybXMgb2YgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZQojIEZvdW5kYXRpb24sIGVpdGhlciB2ZXJzaW9uIDMgb2YgdGhlIExpY2Vuc2UsIG9yIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiMgVGhpcyBwcm9ncmFtIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsIGJ1dCBXSVRIT1VUCiMgQU5ZIFdBUlJBTlRZOyB3aXRob3V0IGV2ZW4gdGhlIGltcGxpZWQgd2FycmFudHkgb2YgTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MKIyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuIFNlZSB0aGUgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KIyBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZQojIGFsb25nIHdpdGggdGhpcyBwcm9ncmFtLiBJZiBub3QsIHNlZSA8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy8+LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0Rpc2NyZXRlU3lzdGVtQWRkcmVzc2VzOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwp1c2UgdXRmODsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FsbCBlbWFpbCBhZGRyZXNzZXMgd2hpY2ggbWFrZSB1cCBvbmUgcG9vbC4gUXVldWVzIGFyZSBhc3NpZ25lZCBpbXBsaWNpdGVseS4gQSBkZWZhdWx0IHF1ZXVlIGNhbiBiZSBwcm92aWRlZC4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbWFpbCBoZWFkZXJzIHVzZWQgZm9yIGFkZHJlc3MtcG9vbCBhc3NpZ25tZW50LiBTZWNvbmRhcnkgaGVhZGVycyBhcmUgbm90IGV2YWx1YXRlZCBpZiBwb29scyBjb3VsZCBiZSBhc3NpZ25lZCB2aWEgdGhlIHByaW1hcnkgaGVhZGVycy4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbnRlcmRpdmlzaW9uYWwnfSA9ICdCZXJlaWNoc8O8YmVyZ3JlaWZlbmQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGlua3MgMiB0aWNrZXRzIHdpdGggYSAiSW50ZXJkaXZpc2lvbmFsIiB0eXBlIGxpbmsuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J01vZHVsZSB0byBjaGVjayBpZiBhbiBhcnRpY2xlIHdpdGggdGhlIGNvcnJlc3BvbmRpbmcgbWVzc2FnZSBpZCBhbHJlYWR5IGV4aXN0cyBpbiB0aGUgc3lzdGVtLiAoRS5nLiBpZiB0aGUgbWFpbCB3YXMgcmVjZWl2ZWQgYW5kIHByb2Nlc3NlZCB0aHJvdWdoIGEgZGlmZmVyZW50IGVtYWlsIGluYm94LikgTXVsdGlwbGUgcGFyc2luZyBjYW4gYmUgYWxsb3dlZC4gUG9vbHMgd2hpY2ggYWxyZWFkeSBjb250YWluIGFuIGFydGljbGUgd2l0aCB0aGUgcmVzcGVjdGl2ZSBtZXNzYWdlIGlkIHdpbGwgc3RpbGwgYmUgaWdub3JlZC4gT3B0aW9ucyBhcmUgIkFsd2F5cyIsICJCb3VuY2VkRW1haWwiIChtdWx0aXBsZSBwYXJzaW5nIHdpbGwgb25seSBiZSBkb25lIGZvciBFbWFpbHMgd2l0aCAiUmVzZW50LVRvIi1oZWFkZXIsIGUuZy4gUG9vbEEgYm91bmNlZCBhIG1haWwgdG8gUG9vbEIpIGFuZCAiTWFpbEFjY291bnRRdWV1ZSIgKGlmIGRpc3BhdGNoaW5nIHZpYSBxdWV1ZSBpcyBlbmFibGVkIGZvciBhIG1haWwgYWNjb3VudCwgYW4gZW1haWwgY2FuIGJlIG1vdmVkIHRvIHRoZSByZXNwZWN0aXZlIGluYm94LCB0byBiZSBldmFsdWF0ZWQgYWdhaW4pLid9ID0KICAgICAgICAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlbW92ZSB0aWNrZXQgbnVtYmVycyBvZiBhbGwgaW50ZXJkaXZpc2lvbmFsbHkgbGlua2VkIHRpY2tldHMgd2hlbiBzdWJqZWN0IGlzIGNsZWFuZWQuJ30gPQogICAgICAgICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGhpcyBzZXR0aW5nIGRlZmluZXMgdGhlIGxpbmsgdHlwZSBcJ0ludGVyZGl2aXNpb25hbFwnLiBJZiB0aGUgc291cmNlIG5hbWUgYW5kIHRoZSB0YXJnZXQgbmFtZSBjb250YWluIHRoZSBzYW1lIHZhbHVlLCB0aGUgcmVzdWx0aW5nIGxpbmsgaXMgYSBub24tZGlyZWN0aW9uYWwgb25lLiBJZiB0aGUgdmFsdWVzIGFyZSBkaWZmZXJlbnQsIHRoZSByZXN1bHRpbmcgbGluayBpcyBhIGRpcmVjdGlvbmFsIGxpbmsuJ30gPQogICAgICAgICcnOwoKCiAgICBwdXNoIEB7ICRTZWxmLT57SmF2YVNjcmlwdFN0cmluZ3N9IC8vIFtdIH0sICgKICAgICk7Cgp9CgoxOwo=</File>
        <File Location="Custom/Kernel/System/PostMaster.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/System/PostMaster.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::System::PostMaster;

use strict;
use warnings;

# core modules

# CPAN modules

# OTOBO modules
use Kernel::System::EmailParser           ();
# Rother OSS / DiscreteSystemAddresses
use Kernel::System::PostMaster::AddressPool;
# EO DiscreteSystemAddresses
use Kernel::System::PostMaster::DestQueue ();
use Kernel::System::PostMaster::NewTicket ();
use Kernel::System::PostMaster::FollowUp  ();
use Kernel::System::PostMaster::Reject    ();
use Kernel::System::VariableCheck         qw(IsHashRefWithData);

our %ObjectManagerFlags = (
    NonSingleton => 1,
);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::DynamicField',
# Rother OSS / DiscreteSystemAddresses
    'Kernel::System::LinkObject',
    'Kernel::System::Log',
# EO DiscreteSystemAddresses
    'Kernel::System::Main',
    'Kernel::System::Queue',
    'Kernel::System::State',
    'Kernel::System::Ticket',
    'Kernel::System::Ticket::Article',
);

=head1 NAME

Kernel::System::PostMaster - postmaster lib

=head1 DESCRIPTION

All postmaster functions. E. g. to process emails.

=head1 PUBLIC INTERFACE

=head2 new()

Don't use the constructor directly, use the ObjectManager instead:

    my $PostMasterObject = $Kernel::OM->Create(
        'Kernel::System::PostMaster',
        ObjectParams => {
            Email        => \@ArrayOfEmailContent,
            Trusted      => 1, # 1|0 ignore X-OTOBO header if false
        },
    );

=cut

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

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

    # check needed parameters
    $Self->{Email}                  = $Param{Email}                  || die "Got no Email!";
    $Self->{CommunicationLogObject} = $Param{CommunicationLogObject} || die "Got no CommunicationLogObject!";

# Rother OSS / DiscreteSystemAddresses
    $Self->{OriginCommunicationLogObject} = $Self->{CommunicationLogObject};
# EO DiscreteSystemAddresses

    # create needed objects
    $Self->{ParserObject} = Kernel::System::EmailParser->new(
        Email => $Param{Email},
    );
    $Self->{DestQueueObject} = Kernel::System::PostMaster::DestQueue->new( %{$Self} );
    $Self->{NewTicketObject} = Kernel::System::PostMaster::NewTicket->new( %{$Self} );
    $Self->{FollowUpObject}  = Kernel::System::PostMaster::FollowUp->new( %{$Self} );
    $Self->{RejectObject}    = Kernel::System::PostMaster::Reject->new( %{$Self} );

# Rother OSS / DiscreteSystemAddresses
#    $Self->{DestQueueObject} = Kernel::System::PostMaster::DestQueue->new( %{$Self} );
#    $Self->{NewTicketObject} = Kernel::System::PostMaster::NewTicket->new( %{$Self} );
#    $Self->{FollowUpObject}  = Kernel::System::PostMaster::FollowUp->new( %{$Self} );
#    $Self->{RejectObject}    = Kernel::System::PostMaster::Reject->new( %{$Self} );
    $Self->_CreateMailObjects(
        Data => $Self,
    );
# EO DiscreteSystemAddresses

    # check needed config options
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    for my $Option (qw(PostmasterUserID PostmasterX-Header)) {
        $Self->{$Option} = $ConfigObject->Get($Option)
            || die "Found no '$Option' option in configuration!";
    }

    # should I use X-OTOBO headers?
    $Self->{Trusted} = $Param{Trusted} // 1;

    if ( $Self->{Trusted} ) {

        # get dynamic field objects
        my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

        # add Dynamic Field headers
        my $DynamicFields = $DynamicFieldObject->DynamicFieldList(
            Valid      => 1,
            ObjectType => [ 'Ticket', 'Article' ],
            ResultType => 'HASH',
        );

        # create a lookup table
        my %HeaderLookup = map { $_ => 1 } @{ $Self->{'PostmasterX-Header'} };

        for my $DynamicField ( values %$DynamicFields ) {
            for my $Header (
                'X-OTOBO-DynamicField-' . $DynamicField,
                'X-OTOBO-FollowUp-DynamicField-' . $DynamicField,
                )
            {

                # only add the header if is not alreday in the config
                if ( !$HeaderLookup{$Header} ) {
                    push @{ $Self->{'PostmasterX-Header'} }, $Header;
                }
            }
        }
    }

    return $Self;
}

=head2 Run()

to execute the run process

    my ($RetCode, $TicketID) = $PostMasterObject->Run(
        Queue   => 'Junk',  # optional, specify target queue for new tickets
        QueueID => 1,       # optional, specify target queue for new tickets
    );

An empty list is returned in case of an error.

The first returned value indicates what has been done.

    0 = error (also undefined)
    1 = new ticket created
    2 = follow up / open/reopen
    3 = follow up / close -> new ticket
    4 = follow up / close -> reject
    5 = ignored (because of X-OTOBO-Ignore header)

When there is a new or followup ticket then this ticket id is returned as the second value.

=cut

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

    my @Return;

    # ConfigObject section / get params
# Rother OSS / DiscreteSystemAddresses
#   my $GetParam = $Self->GetEmailParams();
#
#    # check if follow up
#    my ( $Tn, $TicketID ) = $Self->CheckFollowUp( GetParam => $GetParam );

    my $GetParam = $Param{EmailParams} || $Self->GetEmailParams();

    my ( $Tn, $TicketID );
# EO DiscreteSystemAddresses

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

# Rother OSS / DiscreteSystemAddresses
    my $LinkObject = $Kernel::OM->Get('Kernel::System::LinkObject');

    my @TicketIDsToLink;
    if ( !$Param{AddressPool} ) {
        ( $Tn, $TicketID ) = $Self->CheckFollowUp( GetParam => $GetParam );
# EO DiscreteSystemAddresses

        # Run the PreFilterModules.
        # These filter modules may modify the email parameters in %GetParam, including
        # the body and the attachments.
        if ( ref $ConfigObject->Get('PostMaster::PreFilterModule') eq 'HASH' ) {

            my %Jobs = %{ $ConfigObject->Get('PostMaster::PreFilterModule') };

            # get main objects
            my $MainObject = $Kernel::OM->Get('Kernel::System::Main');

            JOB:
            for my $Job ( sort keys %Jobs ) {

                return unless $MainObject->Require( $Jobs{$Job}->{Module} );

                # Note that passing ParserObject to the constructor of the filter object
                # allows the filter to modify the message itself. This is used in SMIME decryption.
                my $FilterObject = $Jobs{$Job}->{Module}->new(
                    %{$Self},
                );

                if ( !$FilterObject ) {
                    $Self->{CommunicationLogObject}->ObjectLog(
                        ObjectLogType => 'Message',
                        Priority      => 'Error',
                        Key           => 'Kernel::System::PostMaster',
                        Value         => "new() of PreFilterModule $Jobs{$Job}->{Module} not successfully!",
                    );
                    next JOB;
                }

                # modify params
                my $Run = $FilterObject->Run(
                    GetParam  => $GetParam,
                    JobConfig => $Jobs{$Job},
                    TicketID  => $TicketID,
                    UserID    => $Self->{PostmasterUserID},
# Rother OSS / DiscreteSystemAddresses
                    QueueID   => $Param{QueueID},
# EO DiscreteSystemAddresses
                );
                if ( !$Run ) {
                    $Self->{CommunicationLogObject}->ObjectLog(
                        ObjectLogType => 'Message',
                        Priority      => 'Error',
                        Key           => 'Kernel::System::PostMaster',
                        Value         => "Execute Run() of PreFilterModule $Jobs{$Job}->{Module} not successfully!",
                    );
                }
            }
        }

        # should I ignore the incoming mail?
        if ( $GetParam->{'X-OTOBO-Ignore'} && $GetParam->{'X-OTOBO-Ignore'} =~ /(yes|true)/i ) {
            $Self->{CommunicationLogObject}->ObjectLog(
                ObjectLogType => 'Message',
                Priority      => 'Info',
                Key           => 'Kernel::System::PostMaster',
                Value         =>
                    "Ignored Email (From: $GetParam->{'From'}, Message-ID: $GetParam->{'Message-ID'}) "
                    . "because the X-OTOBO-Ignore is set (X-OTOBO-Ignore: $GetParam->{'X-OTOBO-Ignore'}).",
            );

            return (5);
        }

        #
        # ticket section
        #

        # check if follow up (again, with new GetParam)
        ( $Tn, $TicketID ) = $Self->CheckFollowUp( GetParam => $GetParam );

# Rother OSS / DiscreteSystemAddresses

        # build mail address list
        my @AddressedPools = $Self->{AddressPoolObject}->FilterPools(
            Params => $GetParam,
        );

        if ( @AddressedPools ) {
            # get the Pool for QueueID from MailAccount, in case of dispatching via Queue
            my $MailAccountPool;
            if ( $Param{QueueID} || $Param{Queue} ) {
                my $QueueObject = $Kernel::OM->Get('Kernel::System::Queue');

                $Param{QueueID}      ||= $QueueObject->QueueLookup(
                    Queue   => $Param{Queue},
                );
                my %MailAccountAddress = $QueueObject->GetSystemAddress(
                    QueueID => $Param{QueueID},
                );
                $MailAccountPool       = $Self->{AddressPoolObject}->PoolLookup(
                    Address => $MailAccountAddress{Email},
                );

                # add the Pool, if not addressed
                if ( $MailAccountPool && ( !grep { $_ eq $MailAccountPool } @AddressedPools ) ) {
                    unshift @AddressedPools, $MailAccountPool;
                }
            }

            # if we are reprocessing an email filter out pools which already have an article with this message id
            if ( $GetParam->{IgnoreAddressPools} ) {
                @AddressedPools = grep { !$GetParam->{IgnoreAddressPools}{$_} } @AddressedPools;

                if ( !@AddressedPools ) {
                    $Self->{CommunicationLogObject}->ObjectLog(
                        ObjectLogType => 'Message',
                        Priority      => 'Info',
                        Key           => 'Kernel::System::PostMaster',
                        Value         =>
                            "Ignored Email (From: $GetParam->{'From'}, Message-ID: $GetParam->{'Message-ID'}) "
                            . "because all addressed pools already hold an article for it.",
                    );

                    return (5);
                }
            }

            # for follow-ups
            if ( $TicketID ) {
                my $Pool = $Self->{AddressPoolObject}->PoolLookup(
                    TicketID => $TicketID,
                );

                if ( $Pool ) {
                    if ( grep { $_ eq $Pool } @AddressedPools ) {
                        $Param{AddressPool} = $Pool;
                    }

                    # if a follow up in a not addressed pool exists, look for linked tickets also in the parent run
                    else {
                        $Param{FollowUpTicketID} = $TicketID;
                    }
                }

                # if pools are addressed, non pool tickets are ignored
                else {
                    $Tn       = undef;
                    $TicketID = undef;
                }
            }

            $Param{AddressPool} //= shift @AddressedPools;

            ADDRESS:
            for my $Pool ( @AddressedPools ) {
                next ADDRESS if $Pool eq $Param{AddressPool};

                my %EmailParams  = %{$GetParam};
                my $RecuTicketID = $Self->RecursivePostMasterRun(
                    EmailParams      => \%EmailParams,
                    AddressPool      => $Pool,
                    FollowUpTicketID => $TicketID,
                    QueueID          =>
                        ( $MailAccountPool && $MailAccountPool eq $Pool ) ? $Param{QueueID} : undef,
                );

                push @TicketIDsToLink, $RecuTicketID;
            }

            # take queue by mail account only into consideration for its respective pool
            if ( !$MailAccountPool || $MailAccountPool ne $Param{AddressPool} ) {
                delete $Param{Queue};
                delete $Param{QueueID};
            }

            # set communication log to origin
            $Self->{CommunicationLogObject} = $Self->{OriginCommunicationLogObject};
            $Self->_CreateMailObjects(
                Data => $Self,
            );
        }
    }

    if ( $Param{FollowUpTicketID} ) {
        ( $Tn, $TicketID ) = $Self->{AddressPoolObject}->FindLinkedTicket(
            TicketID    => $Param{FollowUpTicketID},
            AddressPool => $Param{AddressPool},
            UserID      => $Self->{PostmasterUserID},
        );
    }

    if ( $Param{AddressPool} ) {
        $GetParam->{AddressPool} = $Param{AddressPool};

        XQUEUEHEADER:
        for my $XQueueHeader (qw(X-OTOBO-Queue X-OTOBO-FollowUp-Queue)) {

            next XQUEUEHEADER if !$GetParam->{$XQueueHeader};

            my $QueueExist = $Self->{AddressPoolObject}->QueueCheck(
                Queue       => $GetParam->{$XQueueHeader},
                AddressPool => $GetParam->{AddressPool},
            );

            if ( !$QueueExist ) {
                delete( $GetParam->{$XQueueHeader} );
            }
        }
    }

# EO DiscreteSystemAddresses

    # run all PreCreateFilterModules
    if ( ref $ConfigObject->Get('PostMaster::PreCreateFilterModule') eq 'HASH' ) {

        my %Jobs = %{ $ConfigObject->Get('PostMaster::PreCreateFilterModule') };

        my $MainObject = $Kernel::OM->Get('Kernel::System::Main');

        JOB:
        for my $Job ( sort keys %Jobs ) {

            return unless $MainObject->Require( $Jobs{$Job}->{Module} );

            my $FilterObject = $Jobs{$Job}->{Module}->new(
                %{$Self},
            );

            if ( !$FilterObject ) {
                $Self->{CommunicationLogObject}->ObjectLog(
                    ObjectLogType => 'Message',
                    Priority      => 'Error',
                    Key           => 'Kernel::System::PostMaster',
                    Value         => "new() of PreCreateFilterModule $Jobs{$Job}->{Module} not successfully!",
                );
                next JOB;
            }

            # modify params
            my $Run = $FilterObject->Run(
                GetParam  => $GetParam,
                JobConfig => $Jobs{$Job},
                TicketID  => $TicketID,
                UserID    => $Self->{PostmasterUserID},
            );
            if ( !$Run ) {
                $Self->{CommunicationLogObject}->ObjectLog(
                    ObjectLogType => 'Message',
                    Priority      => 'Error',
                    Key           => 'Kernel::System::PostMaster',
                    Value         => "Execute Run() of PreCreateFilterModule $Jobs{$Job}->{Module} not successfully!",
                );
            }
        }
    }

    # check if it's a follow up ...
    if ( $Tn && $TicketID ) {

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

        # get ticket data
        my %Ticket = $TicketObject->TicketGet(
            TicketID      => $TicketID,
            DynamicFields => 0,
        );

        # get queue object
        my $QueueObject = $Kernel::OM->Get('Kernel::System::Queue');

        # check if it is possible to do the follow up
        # get follow up option (possible or not)
        my $FollowUpPossible = $QueueObject->GetFollowUpOption(
            QueueID => $Ticket{QueueID},
        );

        # get lock option (should be the ticket locked - if closed - after the follow up)
        my $Lock = $QueueObject->GetFollowUpLockOption(
            QueueID => $Ticket{QueueID},
        );

        # get state details
        my %State = $Kernel::OM->Get('Kernel::System::State')->StateGet(
            ID => $Ticket{StateID},
        );

        # Check if we need to treat a bounce e-mail always as a normal follow-up (to reopen the ticket if needed).
        my $BounceEmailAsFollowUp = 0;
        if ( $GetParam->{'X-OTOBO-Bounce'} ) {
            $BounceEmailAsFollowUp = $ConfigObject->Get('PostmasterBounceEmailAsFollowUp');
        }

        # create a new ticket
        if ( !$BounceEmailAsFollowUp && $FollowUpPossible =~ /new ticket/i && $State{TypeName} =~ /^(removed|close)/i )
        {

            $Self->{CommunicationLogObject}->ObjectLog(
                ObjectLogType => 'Message',
                Priority      => 'Info',
                Key           => 'Kernel::System::PostMaster',
                Value         => "Follow up for [$Tn] but follow up not possible ($Ticket{State}). Create new ticket.",
            );

            # send mail && create new article
            # get queue if of From: and To:
            if ( !$Param{QueueID} ) {
                $Param{QueueID} = $Self->{DestQueueObject}->GetQueueID(
                    Params => $GetParam,
                );
            }

            # check if trusted returns a new queue id
            my $TQueueID = $Self->{DestQueueObject}->GetTrustedQueueID(
                Params => $GetParam,
            );
            if ($TQueueID) {
                $Param{QueueID} = $TQueueID;
            }

            # Clean out the old TicketNumber from the subject (see bug#9108).
            # This avoids false ticket number detection on customer replies.
            if ( $GetParam->{Subject} ) {
                $GetParam->{Subject} = $TicketObject->TicketSubjectClean(
                    TicketNumber => $Tn,
                    Subject      => $GetParam->{Subject},
                );
            }

            $TicketID = $Self->{NewTicketObject}->Run(
                InmailUserID     => $Self->{PostmasterUserID},
                GetParam         => $GetParam,
                QueueID          => $Param{QueueID},
                Comment          => "Because the old ticket [$Tn] is '$State{Name}'",
                AutoResponseType => 'auto reply/new ticket',
                LinkToTicketID   => $TicketID,
            );

            return unless $TicketID;

            @Return = ( 3, $TicketID );
        }

        # reject follow up
        elsif ( !$BounceEmailAsFollowUp && $FollowUpPossible =~ /reject/i && $State{TypeName} =~ /^(removed|close)/i ) {

            $Self->{CommunicationLogObject}->ObjectLog(
                ObjectLogType => 'Message',
                Priority      => 'Info',
                Key           => 'Kernel::System::PostMaster',
                Value         => "Follow up for [$Tn] but follow up not possible. Follow up rejected.",
            );

            # send reject mail and add article to ticket
            my $Run = $Self->{RejectObject}->Run(
                TicketID         => $TicketID,
                InmailUserID     => $Self->{PostmasterUserID},
                GetParam         => $GetParam,
                Lock             => $Lock,
                Tn               => $Tn,
                Comment          => 'Follow up rejected.',
                AutoResponseType => 'auto reject',
            );

            return unless $Run;

            @Return = ( 4, $TicketID );
        }

        # create normal follow up
        else {

            my $Run = $Self->{FollowUpObject}->Run(
                TicketID         => $TicketID,
                InmailUserID     => $Self->{PostmasterUserID},
                GetParam         => $GetParam,
                Lock             => $Lock,
                Tn               => $Tn,
                AutoResponseType => 'auto follow up',
            );

            return unless $Run;

            @Return = ( 2, $TicketID );
        }
    }

    # create new ticket
    else {

        if ( $Param{Queue} && !$Param{QueueID} ) {

            # queue lookup if queue name is given
            $Param{QueueID} = $Kernel::OM->Get('Kernel::System::Queue')->QueueLookup(
                Queue => $Param{Queue},
            );
        }

        # get queue from From: or To:
        if ( !$Param{QueueID} ) {
            $Param{QueueID} = $Self->{DestQueueObject}->GetQueueID( Params => $GetParam );
        }

        # check if trusted returns a new queue id
        my $TQueueID = $Self->{DestQueueObject}->GetTrustedQueueID(
            Params => $GetParam,
        );
        if ($TQueueID) {
            $Param{QueueID} = $TQueueID;
        }
        $TicketID = $Self->{NewTicketObject}->Run(
            InmailUserID     => $Self->{PostmasterUserID},
            GetParam         => $GetParam,
            QueueID          => $Param{QueueID},
            AutoResponseType => 'auto reply',
        );

        return unless $TicketID;

        @Return = ( 1, $TicketID );
    }

    # run all PostFilterModules (modify email params)
    if ( ref $ConfigObject->Get('PostMaster::PostFilterModule') eq 'HASH' ) {

        my %Jobs = %{ $ConfigObject->Get('PostMaster::PostFilterModule') };

        # get main objects
        my $MainObject = $Kernel::OM->Get('Kernel::System::Main');

        JOB:
        for my $Job ( sort keys %Jobs ) {

            return unless $MainObject->Require( $Jobs{$Job}->{Module} );

            my $FilterObject = $Jobs{$Job}->{Module}->new(
                %{$Self},
            );

            if ( !$FilterObject ) {
                $Self->{CommunicationLogObject}->ObjectLog(
                    ObjectLogType => 'Message',
                    Priority      => 'Error',
                    Key           => 'Kernel::System::PostMaster',
                    Value         => "new() of PostFilterModule $Jobs{$Job}->{Module} not successfully!",
                );

                next JOB;
            }

            # modify params
            my $Run = $FilterObject->Run(
                TicketID  => $TicketID,
                GetParam  => $GetParam,
                JobConfig => $Jobs{$Job},
                Return    => $Return[0],
                UserID    => $Self->{PostmasterUserID},
            );

            if ( !$Run ) {
                $Self->{CommunicationLogObject}->ObjectLog(
                    ObjectLogType => 'Message',
                    Priority      => 'Error',
                    Key           => 'Kernel::System::PostMaster',
                    Value         => "Execute Run() of PostFilterModule $Jobs{$Job}->{Module} not successfully!",
                );
            }
        }
    }

# Rother OSS / DiscreteSystemAddresses
    # create link of type 'Interdivisional' to tickets
    if ( $Param{AddressPool}
        && ( @TicketIDsToLink || $Param{FollowUpTicketID} || $GetParam->{IgnoreAddressPools} ) ) {

        # add all ticket ids to the return - mainly used for unit tests
        push @Return, @TicketIDsToLink;

        # add the ticket ID of the first ticket
        push @TicketIDsToLink, $Return[1];

        # add the original follow up ID, if it was in a pool not addressed
        if ( $Param{FollowUpTicketID} ) {
            push @TicketIDsToLink, $Param{FollowUpTicketID};
        }

        # if this is an email parsed an additional time, link all previously created tickets
        if ( $GetParam->{IgnoreAddressPools} ) {
            push @TicketIDsToLink, ( values $GetParam->{IgnoreAddressPools}->%* );
        }

        # add all tickets already linked
        my %LinkedTickets = $LinkObject->LinkKeyList(
            Object1 => 'Ticket',
            Key1    => $Param{FollowUpTicketID} || $TicketID,
            Object2 => 'Ticket',
            State   => 'Valid',
            Type    => 'Interdivisional',
            UserID  => $Self->{PostmasterUserID},
        );

        push @TicketIDsToLink, keys %LinkedTickets;

        $Self->{AddressPoolObject}->InterdivisionalTicketLinkAdd(
            TicketIDs => \@TicketIDsToLink,
            UserID    => $Self->{PostmasterUserID},
        );
    }
# EO DiscreteSystemAddresses

    return @Return;
}

=head2 CheckFollowUp()

to detect the ticket number in processing email

    my ($TicketNumber, $TicketID) = $PostMasterObject->CheckFollowUp(
        Subject => 'Re: [Ticket:#123456] Some Subject',
    );

=cut

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

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

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

    # Load CheckFollowUp Modules
    my $Jobs = $ConfigObject->Get('PostMaster::CheckFollowUpModule');

    if ( IsHashRefWithData($Jobs) ) {
        my $MainObject = $Kernel::OM->Get('Kernel::System::Main');
        JOB:
        for my $Job ( sort keys %$Jobs ) {
            my $Module = $Jobs->{$Job};

            return if !$MainObject->Require( $Jobs->{$Job}->{Module} );

            my $CheckObject = $Jobs->{$Job}->{Module}->new(
                %{$Self},
            );

            if ( !$CheckObject ) {
                $Self->{CommunicationLogObject}->ObjectLog(
                    ObjectLogType => 'Message',
                    Priority      => 'Error',
                    Key           => 'Kernel::System::PostMaster',
                    Value         => "new() of CheckFollowUp $Jobs->{$Job}->{Module} not successfully!",
                );
                next JOB;
            }
            my $TicketID = $CheckObject->Run(
                %Param,
                UserID => $Self->{PostmasterUserID},
            );
            if ($TicketID) {
                my %Ticket = $TicketObject->TicketGet(
                    TicketID      => $TicketID,
                    DynamicFields => 0,
                );
                if (%Ticket) {

                    $Self->{CommunicationLogObject}->ObjectLog(
                        ObjectLogType => 'Message',
                        Priority      => 'Debug',
                        Key           => 'Kernel::System::PostMaster',
                        Value         =>
                            "Found follow up ticket with TicketNumber '$Ticket{TicketNumber}' and TicketID '$TicketID'.",
                    );

                    return ( $Ticket{TicketNumber}, $TicketID );
                }
            }
        }
    }

    return;
}

=head2 GetEmailParams()

to get all configured PostmasterX-Header email headers

    my %Header = $PostMasterObject->GetEmailParams();

=cut

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

    my %GetParam;

    # parse section
    HEADER:
    for my $Param ( @{ $Self->{'PostmasterX-Header'} } ) {

        # do not scan x-otobo headers if mailbox is not marked as trusted
        next HEADER if ( !$Self->{Trusted} && $Param =~ /^x-otobo/i );

        $GetParam{$Param} = $Self->{ParserObject}->GetParam( WHAT => $Param );

        next HEADER if !$GetParam{$Param};

        $Self->{CommunicationLogObject}->ObjectLog(
            ObjectLogType => 'Message',
            Priority      => 'Debug',
            Key           => 'Kernel::System::PostMaster',
            Value         => "$Param: " . $GetParam{$Param},
        );
    }

    # set compat. headers
    if ( $GetParam{'Message-Id'} ) {
        $GetParam{'Message-ID'} = $GetParam{'Message-Id'};
    }
    if ( $GetParam{'Reply-To'} ) {
        $GetParam{'ReplyTo'} = $GetParam{'Reply-To'};
    }
    if (
        $GetParam{'Mailing-List'}
        || $GetParam{'Precedence'}
        || $GetParam{'X-Loop'}
        || $GetParam{'X-No-Loop'}
        || $GetParam{'X-OTOBO-Loop'}
        || (
            $GetParam{'Auto-Submitted'}
            && substr( $GetParam{'Auto-Submitted'}, 0, 5 ) eq 'auto-'
        )
        )
    {
        $GetParam{'X-OTOBO-Loop'} = 'yes';
    }
    if ( !$GetParam{'X-Sender'} ) {

        # get sender email
        my @EmailAddresses = $Self->{ParserObject}->SplitAddressLine(
            Line => $GetParam{From},
        );
        for my $Email (@EmailAddresses) {
            $GetParam{'X-Sender'} = $Self->{ParserObject}->GetEmailAddress(
                Email => $Email,
            );
        }
    }

    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

    # set sender type if not given
    for my $Key (qw(X-OTOBO-SenderType X-OTOBO-FollowUp-SenderType)) {

        if ( !$GetParam{$Key} ) {
            $GetParam{$Key} = 'customer';
        }

        # check if X-OTOBO-SenderType exists, if not, set customer
        if ( !$ArticleObject->ArticleSenderTypeLookup( SenderType => $GetParam{$Key} ) ) {
            $Self->{CommunicationLogObject}->ObjectLog(
                ObjectLogType => 'Message',
                Priority      => 'Error',
                Key           => 'Kernel::System::PostMaster',
                Value         => "Can't find sender type '$GetParam{$Key}' in db, take 'customer'",
            );
            $GetParam{$Key} = 'customer';
        }
    }

    # Set article customer visibility if not given.
    for my $Key (qw(X-OTOBO-IsVisibleForCustomer X-OTOBO-FollowUp-IsVisibleForCustomer)) {
        if ( !defined $GetParam{$Key} ) {
            $GetParam{$Key} = 1;
        }
    }

    # Get body.
    $GetParam{Body} = $Self->{ParserObject}->GetMessageBody();

    # Get content type, disposition and charset.
    $GetParam{'Content-Type'}        = $Self->{ParserObject}->GetReturnContentType();
    $GetParam{'Content-Disposition'} = $Self->{ParserObject}->GetContentDisposition();
    $GetParam{Charset}               = $Self->{ParserObject}->GetReturnCharset();

    # Get attachments.
    my @Attachments = $Self->{ParserObject}->GetAttachments();
    $GetParam{Attachment} = \@Attachments;

    return \%GetParam;
}

# Rother OSS / DiscreteSystemAddresses

=head2 RecursivePostMasterRun()

Recursive postmaster run for address pool

    my $TicketID = $PostMasterObject->RecursivePostMasterRun(
        EmailParams      => \%EmailParams,  # (optional)
        AddressPool      => 'Pool1',        # (optional)
        FollowUpTicketID => 4,              # (optional)
    );

Return:

    TicketID = 5

=cut

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

    my $TicketID;

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

    # create new communication log for every article
    my $CommunicationLogObject = $Kernel::OM->Create(
        'Kernel::System::CommunicationLog',
        ObjectParams => {
            Transport   => 'Email',
            Direction   => 'Incoming',
            AccountType => 'STDIN',
        },
    );
    $CommunicationLogObject->ObjectLogStart( ObjectLogType => 'Message' );

    # create connection details link
    my $OriginConnectionID    = $Self->{OriginCommunicationLogObject}->{Current}->{Connection};
    my $OriginCommunicationID = $Self->{OriginCommunicationLogObject}->{CommunicationID};
    if ( $OriginConnectionID && $OriginCommunicationID ) {

        my $DetailsLink = $ConfigObject->{HttpType} . "://" . $ConfigObject->{FQDN} .
            "/otobo/index.pl?Action=AdminCommunicationLog;Subaction=Zoom;CommunicationID=$OriginCommunicationID;ObjectLogID=$OriginConnectionID";

        $CommunicationLogObject->ObjectLog(
            ObjectLogType => 'Message',
            Priority      => 'Debug',
            Key           => 'Kernel::System::PostMaster::AddressPool::OriginalMail',
            Value         => "For more details see: $DetailsLink",
        );
    }

    # set new communication log
    $Self->{CommunicationLogObject} = $CommunicationLogObject;

    # create needed objects again
    $Self->_CreateMailObjects(
        Data => $Self,
    );

    # set status message
    my $MessageStatus = 'Successful';

    # run post master
    my @Success = eval {
        $Self->Run(
            EmailParams      => $Param{EmailParams},
            AddressPool      => $Param{AddressPool},
            FollowUpTicketID => $Param{FollowUpTicketID},
        );
    };
    if ( !$Success[0] ) {
        $MessageStatus = 'Failed';
    }
    else {
        $TicketID = $Success[1];
    }

    # stop object / communication log
    $CommunicationLogObject->ObjectLogStop(
        ObjectLogType => 'Message',
        Status        => $MessageStatus,
    );
    $CommunicationLogObject->CommunicationStop( Status => 'Successful' );

    return $TicketID;
}

=head2 _CreateMailObjects()

Create new mail objects (DestQueue, NewTicket, FollowUp, Reject, AddressPool)

    $PostMasterObject->_CreateMailObjects(
        Data => $Self,
    );

=cut

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

    # check needed stuff
    if ( !$Param{Data} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need Data!",
        );
        return;
    }

    # create mail objects
    $Self->{DestQueueObject}   = Kernel::System::PostMaster::DestQueue->new( $Param{Data}->%* );
    $Self->{NewTicketObject}   = Kernel::System::PostMaster::NewTicket->new( $Param{Data}->%* );
    $Self->{FollowUpObject}    = Kernel::System::PostMaster::FollowUp->new( $Param{Data}->%* );
    $Self->{RejectObject}      = Kernel::System::PostMaster::Reject->new( $Param{Data}->%* );
    $Self->{AddressPoolObject} = Kernel::System::PostMaster::AddressPool->new( $Param{Data}->%* );

    return 1;
}

# EO DiscreteSystemAddresses

1;
</File>
        <File Location="Custom/Kernel/System/SystemAddress.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - ff9e297baf287e16071d3ac6ad7f6c13f11ac7fa - Kernel/System/SystemAddress.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::System::SystemAddress;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::Cache',
    'Kernel::System::DB',
    'Kernel::System::Log',
    'Kernel::System::Valid',
# Rother OSS / DiscreteSystemAddresses
    'Kernel::System::Queue',
    'Kernel::System::Ticket',
    'Kernel::System::PostMaster::AddressPool',
# EO DiscreteSystemAddresses
);

=head1 NAME

Kernel::System::SystemAddress - all system address functions

=head1 DESCRIPTION

Global module to add/edit/update system addresses.

=head1 PUBLIC INTERFACE

=head2 new()

create an object

    my $SystemAddressObject = $Kernel::OM->Get('Kernel::System::SystemAddress');

=cut

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

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

    $Self->{CacheType} = 'SystemAddress';
    $Self->{CacheTTL}  = 60 * 60 * 24 * 20;

    return $Self;
}

=head2 SystemAddressAdd()

add system address with attributes

    my $ID = $SystemAddressObject->SystemAddressAdd(
        Name     => 'info@example.com',
        Realname => 'Hotline',
        ValidID  => 1,
        QueueID  => 123,
        Comment  => 'some comment',
        UserID   => 123,
    );

=cut

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

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

    # check if a system address with this name already exists
    if ( $Self->NameExistsCheck( Name => $Param{Name} ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "A system address with the name '$Param{Name}' already exists.",
        );
        return;
    }

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # insert new system address
    return if !$DBObject->Do(
        SQL => 'INSERT INTO system_address (value0, value1, valid_id, comments, queue_id, '
            . ' create_time, create_by, change_time, change_by)'
            . ' VALUES (?, ?, ?, ?, ?, current_timestamp, ?, current_timestamp, ?)',
        Bind => [
            \$Param{Name},    \$Param{Realname}, \$Param{ValidID}, \$Param{Comment},
            \$Param{QueueID}, \$Param{UserID},   \$Param{UserID},
        ],
    );

    # get system address id
    $DBObject->Prepare(
        SQL   => 'SELECT id FROM system_address WHERE value0 = ? AND value1 = ?',
        Bind  => [ \$Param{Name}, \$Param{Realname}, ],
        Limit => 1,
    );

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

    $Kernel::OM->Get('Kernel::System::Cache')->CleanUp(
        Type => $Self->{CacheType},
    );

    return $ID;
}

=head2 SystemAddressGet()

get system address with attributes

    my %SystemAddress = $SystemAddressObject->SystemAddressGet(
        ID => 1,
    );

returns:

    %SystemAddress = (
        ID         => 1,
        Name       => 'info@example.com'
        Realname   => 'Hotline',
        QueueID    => 123,
        Comment    => 'some comment',
        ValidID    => 1,
        CreateTime => '2010-11-29 11:04:04',
        ChangeTime => '2010-12-07 12:33:56',
    )

=cut

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

    # check needed stuff
    if ( !$Param{ID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need ID!",
        );
        return;
    }

    my $CacheKey = 'SystemAddressGet::' . $Param{ID};

    my $Cached = $Kernel::OM->Get('Kernel::System::Cache')->Get(
        Type => $Self->{CacheType},
        Key  => $CacheKey,
    );

    return %{$Cached} if ref $Cached eq 'HASH';

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # get system address
    return if !$DBObject->Prepare(
        SQL => 'SELECT value0, value1, comments, valid_id, queue_id, change_time, create_time '
            . ' FROM system_address WHERE id = ?',
        Bind  => [ \$Param{ID} ],
        Limit => 1,
    );

    # fetch the result
    my %Data;
    while ( my @Data = $DBObject->FetchrowArray() ) {
        %Data = (
            ID         => $Param{ID},
            Name       => $Data[0],
            Realname   => $Data[1],
            Comment    => $Data[2],
            ValidID    => $Data[3],
            QueueID    => $Data[4],
            ChangeTime => $Data[5],
            CreateTime => $Data[6],
        );
    }

    $Kernel::OM->Get('Kernel::System::Cache')->Set(
        Type  => $Self->{CacheType},
        TTL   => $Self->{CacheTTL},
        Key   => $CacheKey,
        Value => \%Data,
    );

    return %Data;
}

=head2 SystemAddressUpdate()

update system address with attributes

    $SystemAddressObject->SystemAddressUpdate(
        ID       => 1,
        Name     => 'info@example.com',
        Realname => 'Hotline',
        ValidID  => 1,
        QueueID  => 123,
        Comment  => 'some comment',
        UserID   => 123,
    );

=cut

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

    # Check needed stuff.
    for my $Needed (qw(ID Name ValidID Realname QueueID UserID)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );
            return;
        }
    }

    # Check if a system address with this name already exists.
    if (
        $Self->NameExistsCheck(
            ID   => $Param{ID},
            Name => $Param{Name}
        )
        )
    {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "A system address with the name '$Param{Name}' already exists.",
        );
        return;
    }

    # Check if a system address is used in some queue's or auto response's.
    if ( $Self->SystemAddressIsUsed( SystemAddressID => $Param{ID} ) && $Param{ValidID} > 1 )
    {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  =>
                "This system address '$Param{Name}' cannot be set to invalid, because it is used in one or more queue(s) or auto response(s).",
        );
        return;
    }

    # Update system address.
    return if !$Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL => 'UPDATE system_address SET value0 = ?, value1 = ?, comments = ?, valid_id = ?, '
            . ' change_time = current_timestamp, change_by = ?, queue_id = ? WHERE id = ?',
        Bind => [
            \$Param{Name},   \$Param{Realname}, \$Param{Comment}, \$Param{ValidID},
            \$Param{UserID}, \$Param{QueueID},  \$Param{ID},
        ],
    );

    $Kernel::OM->Get('Kernel::System::Cache')->CleanUp(
        Type => $Self->{CacheType},
    );

    return 1;
}

=head2 SystemAddressList()

get a list of system addresses

    my %List = $SystemAddressObject->SystemAddressList(
        Valid => 0,  # optional, defaults to 1
    );

returns:

    %List = (
        '1' => 'sales@example.com',
        '2' => 'purchasing@example.com',
        '3' => 'service@example.com',
    );

=cut

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

    my $Valid = 1;
    if ( !$Param{Valid} && defined $Param{Valid} ) {
        $Valid = 0;
    }

    my $CacheKey = 'SystemAddressList::' . $Valid;

    my $Cached = $Kernel::OM->Get('Kernel::System::Cache')->Get(
        Type => $Self->{CacheType},
        Key  => $CacheKey,
    );

    return %{$Cached} if ref $Cached eq 'HASH';

    my $ValidSQL = '';
    if ($Valid) {
        my $ValidIDs = join ',', $Kernel::OM->Get('Kernel::System::Valid')->ValidIDsGet();
        $ValidSQL = " WHERE valid_id IN ($ValidIDs)";
    }

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # get system address
    return if !$DBObject->Prepare(
        SQL => "
            SELECT id, value0
            FROM system_address
            $ValidSQL",
    );

    my %List;

    while ( my @Data = $DBObject->FetchrowArray() ) {
        $List{ $Data[0] } = $Data[1];
    }

    $Kernel::OM->Get('Kernel::System::Cache')->Set(
        Type  => $Self->{CacheType},
        TTL   => $Self->{CacheTTL},
        Key   => $CacheKey,
        Value => \%List,
    );

    return %List;
}

=head2 SystemAddressIsLocalAddress()

Checks if the given address is a local (system) address. Returns true
for local addresses.

# Rother OSS / DiscreteSystemAddresses
#    if ( $SystemAddressObject->SystemAddressIsLocalAddress( Address => 'info@example.com' ) ) {
    my $IsLocal = $SystemAddressObject->SystemAddressIsLocalAddress(
        Address  => 'info@example.com',
        QueueID  => 4,                   # (optional)
        TicketID => 1,                   # (optional)
    );

    if ( $IsLocal ) {
# EO DiscreteSystemAddresses
        # is local
    }
    else {
        # is not local
    }

=cut

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

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

# Rother OSS / DiscreteSystemAddresses
    # get objects
    my $QueueObject       = $Kernel::OM->Get('Kernel::System::Queue');
    my $TicketObject      = $Kernel::OM->Get('Kernel::System::Ticket');
    my $AddressPoolObject = $Kernel::OM->Get('Kernel::System::PostMaster::AddressPool');

    # get address pool
    my $AddressPool = $AddressPoolObject->PoolLookup(
        Address => $Param{Address},
    );

    # check if address exist in same address pool of ticket
    if (
        $AddressPool
        && ( $Param{QueueID} || $Param{TicketID} )
        )
    {

        my $QueueID = $Param{QueueID};
        if ( $Param{TicketID} ) {
            $QueueID = $TicketObject->TicketQueueID(
                TicketID => $Param{TicketID},
            );
        }
        my $Queue = $QueueObject->QueueLookup(
            QueueID => $QueueID,
        );

        my $QueueExist = $AddressPoolObject->QueueCheck(
            Queue       => $Queue,
            AddressPool => $AddressPool,
        );

        return $QueueExist;
    }
# EO DiscreteSystemAddresses

    return $Self->SystemAddressQueueID(%Param);
}

=head2 SystemAddressQueueID()

find dispatching queue id of email address

    my $QueueID = $SystemAddressObject->SystemAddressQueueID( Address => 'info@example.com' );

=cut

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

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

    # remove spaces
    $Param{Address} =~ s/\s+//g;

    my $CacheKey = 'SystemAddressQueueID::' . $Param{Address};
    my $Cached   = $Kernel::OM->Get('Kernel::System::Cache')->Get(
        Type => $Self->{CacheType},
        Key  => $CacheKey,
    );

    return ${$Cached} if ref $Cached eq 'SCALAR';

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    if ( $DBObject->GetDatabaseFunction('CaseSensitive') ) {

        return if !$DBObject->Prepare(
            SQL => "SELECT queue_id FROM system_address WHERE "
                . "valid_id IN ( ${\(join ', ', $Kernel::OM->Get('Kernel::System::Valid')->ValidIDsGet())} ) "
                . "AND LOWER(value0) = LOWER(?)",
            Bind  => [ \$Param{Address} ],
            Limit => 1,
        );
    }
    else {
        return if !$DBObject->Prepare(
            SQL => "SELECT queue_id FROM system_address WHERE "
                . "valid_id IN ( ${\(join ', ', $Kernel::OM->Get('Kernel::System::Valid')->ValidIDsGet())} ) "
                . "AND value0 = ?",
            Bind  => [ \$Param{Address} ],
            Limit => 1,
        );
    }

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

    $Kernel::OM->Get('Kernel::System::Cache')->Set(
        Type  => $Self->{CacheType},
        TTL   => $Self->{CacheTTL},
        Key   => $CacheKey,
        Value => \$QueueID,
    );

    return $QueueID;
}

=head2 SystemAddressQueueList()

get a list of the queues and their system addresses IDs

    my %List = $SystemAddressObject->SystemAddressQueueList(
        Valid => 0,  # optional, defaults to 1
    );

returns:

    %List = (
        '5' => 3,
        '7' => 1,
        '9' => 2,
    );

=cut

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

    # set default value
    my $Valid = $Param{Valid} // 1;

    # create the valid list
    my $ValidIDs = join ', ', $Kernel::OM->Get('Kernel::System::Valid')->ValidIDsGet();

    # build SQL
    my $SQL = 'SELECT queue_id, id FROM system_address';

    # add WHERE statement in case Valid param is set to '1', for valid system address
    if ($Valid) {
        $SQL .= ' WHERE valid_id IN (' . $ValidIDs . ')';
    }

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # get data from database
    return if !$DBObject->Prepare(
        SQL => $SQL,
    );

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

    return %SystemAddressQueueList;

}

=head2 NameExistsCheck()

return 1 if another system address with this name already exists

    $Exist = $SystemAddressObject->NameExistsCheck(
        Name => 'Some Address',
        ID => 1, # optional
    );

=cut

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

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');
    return if !$DBObject->Prepare(
        SQL  => 'SELECT id FROM system_address WHERE value0 = ?',
        Bind => [ \$Param{Name} ],
    );

    # fetch the result
    my $Flag;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        if ( !$Param{ID} || $Param{ID} ne $Row[0] ) {
            $Flag = 1;
        }
    }

    if ($Flag) {
        return 1;
    }

    return 0;
}

=head2 SystemAddressIsUsed()

Return 1 if system address is used in one of the queue's or auto response's.

    $SytemAddressIsUsed = $SystemAddressObject->SystemAddressIsUsed(
        SystemAddressID => 1,
    );

=cut

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

    # Check needed param.
    if ( !$Param{SystemAddressID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need SystemAddressID!"
        );
        return;
    }

    # Get database object.
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    return if !$DBObject->Prepare(
        SQL => 'SELECT DISTINCT sa.id FROM system_address sa
            LEFT JOIN queue q ON q.system_address_id = sa.id
            LEFT JOIN auto_response ar ON ar.system_address_id = sa.id
            WHERE q.system_address_id = ? OR ar.system_address_id = ?',
        Bind  => [ \$Param{SystemAddressID}, \$Param{SystemAddressID} ],
        Limit => 1,
    );

    # Fetch the result.
    my $SystemAddressIsUsed;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        $SystemAddressIsUsed = $Row[0] ? 1 : 0;
    }

    return $SystemAddressIsUsed;
}

1;
</File>
        <File Location="Custom/Kernel/System/PostMaster/DestQueue.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/System/PostMaster/DestQueue.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::System::PostMaster::DestQueue;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Log',
# Rother OSS / DiscreteSystemAddresses
    'Kernel::System::PostMaster::AddressPool',
# EO DiscreteSystemAddresses
    'Kernel::System::Queue',
    'Kernel::System::SystemAddress',
);

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

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

    # get parser object
    $Self->{ParserObject} = $Param{ParserObject} || die "Got no ParserObject!";

    # Get communication log object.
    $Self->{CommunicationLogObject} = $Param{CommunicationLogObject} || die "Got no CommunicationLogObject!";

    return $Self;
}

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

    # get email headers
    my %GetParam = %{ $Param{Params} };

    # check possible to, cc and resent-to emailaddresses
    my $Recipient = '';
    RECIPIENT:
    for my $Key (qw(Resent-To Envelope-To To Cc Delivered-To X-Original-To)) {

        next RECIPIENT if !$GetParam{$Key};

        if ($Recipient) {
            $Recipient .= ', ';
        }

        $Recipient .= $GetParam{$Key};
    }

    # get system address object
    my $SystemAddressObject = $Kernel::OM->Get('Kernel::System::SystemAddress');

# Rother OSS / DiscreteSystemAddresses
    my $AddressPoolObject = $Kernel::OM->Get('Kernel::System::PostMaster::AddressPool');
# EO DiscreteSystemAddresses

    # get addresses
    my @EmailAddresses = $Self->{ParserObject}->SplitAddressLine( Line => $Recipient );

    # check addresses
    EMAIL:
    for my $Email (@EmailAddresses) {

        next EMAIL if !$Email;

        my $Address = $Self->{ParserObject}->GetEmailAddress( Email => $Email );

        next EMAIL if !$Address;

# Rother OSS / DiscreteSystemAddresses
        # check if email exist in address pool
        if ( $GetParam{AddressPool} ) {

            next EMAIL if
                ( $AddressPoolObject->PoolLookup( Address => $Address ) // '' ) ne $GetParam{AddressPool};
        }
# EO DiscreteSystemAddresses

        # lookup queue id if recipiend address
        my $QueueID = $SystemAddressObject->SystemAddressQueueID(
            Address => $Address,
        );

        if ($QueueID) {
            $Self->{CommunicationLogObject}->ObjectLog(
                ObjectLogType => 'Message',
                Priority      => 'Debug',
                Key           => ref($Self),
                Value         => "Match email: $Email to QueueID $QueueID (MessageID:$GetParam{'Message-ID'})!",
            );

            return $QueueID;
        }

        # Address/Email not matched with any that is configured in the system
        #   or any error occurred while checking it.

        $Self->{CommunicationLogObject}->ObjectLog(
            ObjectLogType => 'Message',
            Priority      => 'Debug',
            Key           => ref($Self),
            Value         => "No match for email: $Email (MessageID:$GetParam{'Message-ID'})!",
        );
    }

    # If we get here means that none of the addresses in the message is defined as a system address
    #   or an error occurred while checking it.

# Rother OSS / DiscreteSystemAddresses
#    my $Queue   = $Kernel::OM->Get('Kernel::Config')->Get('PostmasterDefaultQueue');
    my $Queue;

    if ( $GetParam{AddressPool} ) {
        $Queue = $Kernel::OM->Get('Kernel::Config')->Get('PostMaster::AddressPool')->{ $GetParam{AddressPool} }{'DefaultQueue'};
    }

    $Queue //= $Kernel::OM->Get('Kernel::Config')->Get('PostmasterDefaultQueue');
# EO DiscreteSystemAddresses

    my $QueueID = $Kernel::OM->Get('Kernel::System::Queue')->QueueLookup(
        Queue => $Queue,
    );

    if ($QueueID) {
        $Self->{CommunicationLogObject}->ObjectLog(
            ObjectLogType => 'Message',
            Priority      => 'Debug',
            Key           => ref($Self),
            Value         => "MessageID:$GetParam{'Message-ID'} to 'PostmasterDefaultQueue' ( QueueID:${QueueID} ).",
        );

        return $QueueID;
    }

    $Self->{CommunicationLogObject}->ObjectLog(
        ObjectLogType => 'Message',
        Priority      => 'Error',
        Key           => ref($Self),
        Value         => "Couldn't get QueueID for 'PostmasterDefaultQueue' (${Queue}) !",
    );

    $Self->{CommunicationLogObject}->ObjectLog(
        ObjectLogType => 'Message',
        Priority      => 'Debug',
        Key           => ref($Self),
        Value         => "MessageID:$GetParam{'Message-ID'} to QueueID:1!",
    );

    return 1;
}

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

    # get email headers
    my %GetParam = %{ $Param{Params} };

    return if !$GetParam{'X-OTOBO-Queue'};

    $Self->{CommunicationLogObject}->ObjectLog(
        ObjectLogType => 'Message',
        Priority      => 'Debug',
        Key           => 'Kernel::System::PostMaster::DestQueue',
        Value         => "Existing X-OTOBO-Queue header: $GetParam{'X-OTOBO-Queue'} (MessageID:$GetParam{'Message-ID'})!",
    );

    # get dest queue
    return $Kernel::OM->Get('Kernel::System::Queue')->QueueLookup(
        Queue => $GetParam{'X-OTOBO-Queue'},
    );
}

1;
</File>
        <File Location="Custom/Kernel/System/Ticket/Article/Backend/Email.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/System/Ticket/Article/Backend/Email.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::System::Ticket::Article::Backend::Email;

use strict;
use warnings;

use parent 'Kernel::System::Ticket::Article::Backend::MIMEBase';

# core modules

# CPAN modules
use Mail::Address ();

# OTOBO modules
use Kernel::System::VariableCheck qw(:all);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::CustomerUser',
    'Kernel::System::DB',
    'Kernel::System::Email',
    'Kernel::System::HTMLUtils',
    'Kernel::System::Log',
    'Kernel::System::Main',
    'Kernel::System::PostMaster::LoopProtection',
    'Kernel::System::State',
    'Kernel::System::TemplateGenerator',
    'Kernel::System::Ticket',
    'Kernel::System::Ticket::Article',
    'Kernel::System::DateTime',
    'Kernel::System::MailQueue',
);

=head1 NAME

Kernel::System::Ticket::Article::Backend::Email - backend class for email based articles

=head1 DESCRIPTION

This class provides functions to manipulate email based articles in the database.

Inherits from L<Kernel::System::Ticket::Article::Backend::MIMEBase>, please have a look there for its base API,
and below for the additional functions this backend provides.

=head1 PUBLIC INTERFACE

=cut

sub ChannelNameGet {
    return 'Email';
}

=head2 ArticleGetByMessageID()

Return article data by supplied message ID.

    my %Article = $ArticleBackendObject->ArticleGetByMessageID(
        MessageID     => '<13231231.1231231.32131231@example.com>',     # (required)
        DynamicFields => 1,                                             # (optional) To include the dynamic field values for this article on the return structure.
        RealNames     => 1,                                             # (optional) To include the From/To/Cc/Bcc fields with real names.
    );

=cut

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

    if ( !$Param{MessageID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need MessageID!',
        );
        return;
    }

    my $MD5 = $Kernel::OM->Get('Kernel::System::Main')->MD5sum( String => $Param{MessageID} );

    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # Get ticket and article ID from meta article table.

# Rother OSS / DiscreteSystemAddresses
#   return if !$DBObject->Prepare(
#       SQL => '
#           SELECT sa.id, sa.ticket_id FROM article sa
#           LEFT JOIN article_data_mime sadm ON sa.id = sadm.article_id
#           WHERE sadm.a_message_id_md5 = ?
#       ',
#       Bind  => [ \$MD5 ],
#       Limit => 10,
#   );
#
#    my $Count = 0;
#    while ( my @Row = $DBObject->FetchrowArray() ) {
#        $Param{ArticleID} = $Row[0];
#        $Param{TicketID}  = $Row[1];
#        $Count++;
#    }
#
#    # No reference found.
#    return if $Count == 0;
    return if !$DBObject->Prepare(
        SQL => '
            SELECT sa.id, sa.ticket_id FROM article sa
            LEFT JOIN article_data_mime sadm ON sa.id = sadm.article_id
            WHERE sadm.a_message_id_md5 = ?
            ORDER BY sa.id DESC
        ',
        Bind  => [ \$MD5 ],
    );

    my @TicketIDs;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        $Param{ArticleID} //= $Row[0];
        $Param{TicketID}  //= $Row[1];
        push @TicketIDs, $Row[1];
    }

    # No reference found.
    return if !@TicketIDs;
# EO DiscreteSystemAddresses
    return if !$Param{TicketID} || !$Param{ArticleID};

# Rother OSS / DiscreteSystemAddresses
#   # More than one reference found! That should not happen, since 'a message_id' should be unique!
#   if ( $Count > 1 ) {
#       $Kernel::OM->Get('Kernel::System::Log')->Log(
#           Priority => 'notice',
#           Message  =>
#               "The MessageID '$Param{MessageID}' is in your database more than one time! That should not happen, since 'a message_id' should be unique!",
#       );
#       return;
#   }
#
#    return $Self->ArticleGet(
#        %Param,
#    );
    my %Article = $Self->ArticleGet(
        %Param,
    );

    if ( scalar @TicketIDs > 1 ) {
        $Article{AmbiguousTicketIDs} = \@TicketIDs;
    }

    return %Article;
# EO DiscreteSystemAddresses
}

=head2 ArticleSend()

Send article via email and create article with attachments.

    my $ArticleID = $ArticleBackendObject->ArticleSend(
        TicketID             => 123,                              # (required)
        SenderTypeID         => 1,                                # (required)
                                                                  # or
        SenderType           => 'agent',                          # (required) agent|system|customer
        IsVisibleForCustomer => 1,                                # (required) Is article visible for customer?
        UserID               => 123,                              # (required)

        From        => 'Some Agent <email@example.com>',                       # required
        To          => 'Some Customer A <customer-a@example.com>',             # required if both Cc and Bcc are not present
        Cc          => 'Some Customer B <customer-b@example.com>',             # required if both To and Bcc are not present
        Bcc         => 'Some Customer C <customer-c@example.com>',             # required if both To and Cc are not present
        ReplyTo     => 'Some Customer B <customer-b@example.com>',             # not required, is possible to use 'Reply-To' instead
        Subject     => 'some short description',                               # required
        Body        => 'the message text',                                     # required
        InReplyTo   => '<asdasdasd.12@example.com>',                           # not required but useful
        References  => '<asdasdasd.1@example.com> <asdasdasd.12@example.com>', # not required but useful
        Charset     => 'iso-8859-15'
        MimeType    => 'text/plain',
        Loop        => 0, # 1|0 used for bulk emails
        Attachment => [
            {
                Content     => $Content,
                ContentType => $ContentType,
                Filename    => 'lala.txt',
            },
            {
                Content     => $Content,
                ContentType => $ContentType,
                Filename    => 'lala1.txt',
            },
        ],
        EmailSecurity => {
            Backend     => 'PGP',                       # PGP or SMIME
            Method      => 'Detached',                  # Optional Detached or Inline (defaults to Detached)
            SignKey     => '81877F5E',                  # Optional
            EncryptKeys => [ '81877F5E', '3b630c80' ],  # Optional
        }
        HistoryType    => 'OwnerUpdate',  # Move|AddNote|PriorityUpdate|WebRequestCustomer|...
        HistoryComment => 'Some free text!',
        NoAgentNotify  => 0,            # if you don't want to send agent notifications
    );


    my $ArticleID = $ArticleBackendObject->ArticleSend(                (Backwards compatibility)
        TicketID             => 123,                              # (required)
        SenderTypeID         => 1,                                # (required)
                                                                  # or
        SenderType           => 'agent',                          # (required) agent|system|customer
        IsVisibleForCustomer => 1,                                # (required) Is article visible for customer?
        UserID               => 123,                              # (required)

        From        => 'Some Agent <email@example.com>',                       # required
        To          => 'Some Customer A <customer-a@example.com>',             # required if both Cc and Bcc are not present
        Cc          => 'Some Customer B <customer-b@example.com>',             # required if both To and Bcc are not present
        Bcc         => 'Some Customer C <customer-c@example.com>',             # required if both To and Cc are not present
        ReplyTo     => 'Some Customer B <customer-b@example.com>',             # not required, is possible to use 'Reply-To' instead
        Subject     => 'some short description',                               # required
        Body        => 'the message text',                                     # required
        InReplyTo   => '<asdasdasd.12@example.com>',                           # not required but useful
        References  => '<asdasdasd.1@example.com> <asdasdasd.12@example.com>', # not required but useful
        Charset     => 'iso-8859-15'
        MimeType    => 'text/plain',
        Loop        => 0, # 1|0 used for bulk emails
        Attachment => [
            {
                Content     => $Content,
                ContentType => $ContentType,
                Filename    => 'lala.txt',
            },
            {
                Content     => $Content,
                ContentType => $ContentType,
                Filename    => 'lala1.txt',
            },
        ],
        Sign => {
            Type    => 'PGP',
            SubType => 'Inline|Detached',
            Key     => '81877F5E',
            Type    => 'SMIME',
            Key     => '3b630c80',
        },
        Crypt => {
            Type    => 'PGP',
            SubType => 'Inline|Detached',
            Key     => '81877F5E',
            Type    => 'SMIME',
            Key     => '3b630c80',
        },
        HistoryType    => 'OwnerUpdate',  # Move|AddNote|PriorityUpdate|WebRequestCustomer|...
        HistoryComment => 'Some free text!',
        NoAgentNotify  => 0,            # if you don't want to send agent notifications
    );

Events:
    ArticleSend

=cut

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

    my $ToOrig      = $Param{To}          || '';
    my $Loop        = $Param{Loop}        || 0;
    my $HistoryType = $Param{HistoryType} || 'SendAnswer';

    my $ArticleObject  = $Kernel::OM->Get('Kernel::System::Ticket::Article');
    my $DateTimeObject = $Kernel::OM->Create('Kernel::System::DateTime');

    # Lookup if no ID is passed.
    if ( $Param{SenderType} && !$Param{SenderTypeID} ) {
        $Param{SenderTypeID} = $ArticleObject->ArticleSenderTypeLookup( SenderType => $Param{SenderType} );
    }

    for my $Needed (qw(TicketID UserID SenderTypeID From Body Charset MimeType)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );
            return;
        }
    }

    if ( !defined $Param{IsVisibleForCustomer} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need IsVisibleForCustomer!",
        );
        return;
    }

    # Map ReplyTo into Reply-To if present.
    if ( $Param{ReplyTo} ) {
        $Param{'Reply-To'} = $Param{ReplyTo};
    }

    # Clean up body string.
    $Param{Body} =~ s/(\r\n|\n\r)/\n/g;
    $Param{Body} =~ s/\r/\n/g;

    # initialize parameter for attachments, so that the content pushed into that ref from
    # EmbeddedImagesExtract will stay available
    if ( !$Param{Attachment} ) {
        $Param{Attachment} = [];
    }

    # check for base64 images in body and process them
    $Kernel::OM->Get('Kernel::System::HTMLUtils')->EmbeddedImagesExtract(
        DocumentRef    => \$Param{Body},
        AttachmentsRef => $Param{Attachment},
    );

    # create article
    my $Time      = $DateTimeObject->ToEpoch();
    my $Random    = rand 999999;
    my $FQDN      = $Kernel::OM->Get('Kernel::Config')->Get('FQDN');
    my $MessageID = "<$Time.$Random\@$FQDN>";
    my $ArticleID = $Self->ArticleCreate(
        %Param,
        MessageID => $MessageID,
    );
    return if !$ArticleID;

    # Send the mail
    my $Result = $Kernel::OM->Get('Kernel::System::Email')->Send(
        %Param,
        ArticleID    => $ArticleID,
        'Message-ID' => $MessageID,
    );

    # return if mail wasn't sent
    if ( !$Result->{Success} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Message  => "Impossible to send message to: $Param{'To'} .",
            Priority => 'error',
        );
        return;
    }

    # write article to file system
    my $Plain = $Self->ArticleWritePlain(
        ArticleID => $ArticleID,
        Email     => sprintf( "%s\n%s", $Result->{Data}->{Header}, $Result->{Data}->{Body} ),
        UserID    => $Param{UserID},
    );
    return if !$Plain;

    # log
    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'info',
        Message  => sprintf(
            "Queued email to '%s' from '%s'. HistoryType => %s, Subject => %s;",
            $Param{To},
            $Param{From},
            $HistoryType,
            $Param{Subject},
        ),
    );

    # event
    $Self->EventHandler(
        Event => 'ArticleSend',
        Data  => {
            TicketID  => $Param{TicketID},
            ArticleID => $ArticleID,
        },
        UserID => $Param{UserID},
    );

    return $ArticleID;
}

=head2 ArticleBounce()

Bounce an article.

    my $Success = $ArticleBackendObject->ArticleBounce(
        From      => 'some@example.com',
        To        => 'webmaster@example.com',
        TicketID  => 123,
        ArticleID => 123,
        UserID    => 123,
    );

Events:
    ArticleBounce

=cut

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

    for my $Item (qw(TicketID ArticleID From To UserID)) {
        if ( !$Param{$Item} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Item!",
            );
            return;
        }
    }

    my $DateTimeObject = $Kernel::OM->Create('Kernel::System::DateTime');

    # create message id
    my $Time         = $DateTimeObject->ToEpoch();
    my $Random       = rand 999999;
    my $FQDN         = $Kernel::OM->Get('Kernel::Config')->Get('FQDN');
    my $NewMessageID = "<$Time.$Random.0\@$FQDN>";
    my $Email        = $Self->ArticlePlain( ArticleID => $Param{ArticleID} );

    # check if plain email exists
    if ( !$Email ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No such plain article for ArticleID ($Param{ArticleID})!",
        );
        return;
    }

    # pipe all into sendmail
    my $BounceSent = $Kernel::OM->Get('Kernel::System::Email')->Bounce(
        'Message-ID' => $NewMessageID,
        From         => $Param{From},
        To           => $Param{To},
        Email        => $Email,
    );

    return if !$BounceSent->{Success};

    # write history
    my $HistoryType = $Param{HistoryType} || 'Bounce';
    $Kernel::OM->Get('Kernel::System::Ticket')->HistoryAdd(
        TicketID     => $Param{TicketID},
        ArticleID    => $Param{ArticleID},
        HistoryType  => $HistoryType,
        Name         => "\%\%$Param{To}",
        CreateUserID => $Param{UserID},
    );

    # event
    $Self->EventHandler(
        Event => 'ArticleBounce',
        Data  => {
            TicketID  => $Param{TicketID},
            ArticleID => $Param{ArticleID},
        },
        UserID => $Param{UserID},
    );

    return 1;
}

=head2 SendAutoResponse()

Send an auto response to a customer via email.

    my $ArticleID = $ArticleBackendObject->SendAutoResponse(
        TicketID         => 123,
        AutoResponseType => 'auto reply',
        OrigHeader       => {
            From    => 'some@example.com',
            Subject => 'For the message!',
        },
        UserID => 123,
    );

Events:
    ArticleAutoResponse

=cut

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

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

    # return if no notification is active
    return 1 if $Self->{SendNoNotification};

    # get orig email header
    my %OrigHeader = %{ $Param{OrigHeader} };

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

    # get ticket
    my %Ticket = $TicketObject->TicketGet(
        TicketID      => $Param{TicketID},
        DynamicFields => 0,                  # not needed here, TemplateGenerator will fetch the ticket on its own
    );

    # get auto default responses
    my %AutoResponse = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->AutoResponse(
        TicketID         => $Param{TicketID},
        AutoResponseType => $Param{AutoResponseType},
        OrigHeader       => $Param{OrigHeader},
        UserID           => $Param{UserID},
    );

    # return if no valid auto response exists
    return if !$AutoResponse{Text};
    return if !$AutoResponse{SenderRealname};
    return if !$AutoResponse{SenderAddress};

    # send if notification should be sent (not for closed tickets)!?
    my %State = $Kernel::OM->Get('Kernel::System::State')->StateGet( ID => $Ticket{StateID} );
    if (
        $Param{AutoResponseType} eq 'auto reply'
        && ( $State{TypeName} eq 'closed' || $State{TypeName} eq 'removed' )
        )
    {

        # add history row
        $TicketObject->HistoryAdd(
            TicketID    => $Param{TicketID},
            HistoryType => 'Misc',
            Name        => "Sent no auto response or agent notification because ticket is "
                . "state-type '$State{TypeName}'!",
            CreateUserID => $Param{UserID},
        );

        # return
        return;
    }

    # log that no auto response was sent!
    if ( $OrigHeader{'X-OTOBO-Loop'} && $OrigHeader{'X-OTOBO-Loop'} !~ /^(false|no)$/i ) {

        # add history row
        $TicketObject->HistoryAdd(
            TicketID    => $Param{TicketID},
            HistoryType => 'Misc',
            Name        => "Sent no auto-response because the sender doesn't want "
                . "an auto-response (e. g. loop or precedence header)",
            CreateUserID => $Param{UserID},
        );
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'info',
            Message  => "Sent no '$Param{AutoResponseType}' for Ticket ["
                . "$Ticket{TicketNumber}] ($OrigHeader{From}) because the "
                . "sender doesn't want an auto-response (e. g. loop or precedence header)"
        );
        return;
    }

    # check reply to for auto response recipient
    if ( $OrigHeader{ReplyTo} ) {
        $OrigHeader{From} = $OrigHeader{ReplyTo};
    }

    # get loop protection object
    my $LoopProtectionObject = $Kernel::OM->Get('Kernel::System::PostMaster::LoopProtection');

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

    my @AutoReplyAddresses;
    my @Addresses = $EmailParser->SplitAddressLine( Line => $OrigHeader{From} );
    ADDRESS:
    for my $Address (@Addresses) {
        my $Email = $EmailParser->GetEmailAddress( Email => $Address );
        if ( !$Email ) {

            # add it to ticket history
            $TicketObject->HistoryAdd(
                TicketID     => $Param{TicketID},
                CreateUserID => $Param{UserID},
                HistoryType  => 'Misc',
                Name         => "Sent no auto response to '$Address' - no valid email address.",
            );

            # log
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "Sent no auto response to '$Address' because of invalid address.",
            );
            next ADDRESS;

        }
        if ( !$LoopProtectionObject->Check( To => $Email ) ) {

            # add history row
            $TicketObject->HistoryAdd(
                TicketID     => $Param{TicketID},
                HistoryType  => 'LoopProtection',
                Name         => "\%\%$Email",
                CreateUserID => $Param{UserID},
            );

            # log
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "Sent no '$Param{AutoResponseType}' for Ticket ["
                    . "$Ticket{TicketNumber}] ($Email) because of loop protection."
            );
            next ADDRESS;
        }
        else {

            # increase loop count
            return if !$LoopProtectionObject->SendEmail( To => $Email );
        }

        # check if sender is e. g. MAILER-DAEMON or Postmaster
        my $NoAutoRegExp = $Kernel::OM->Get('Kernel::Config')->Get('SendNoAutoResponseRegExp');
        if ( $Email =~ /$NoAutoRegExp/i ) {

            # add it to ticket history
            $TicketObject->HistoryAdd(
                TicketID     => $Param{TicketID},
                CreateUserID => $Param{UserID},
                HistoryType  => 'Misc',
                Name         => "Sent no auto response to '$Email', SendNoAutoResponseRegExp matched.",
            );

            # log
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'info',
                Message  => "Sent no auto response to '$Email' because config"
                    . " option SendNoAutoResponseRegExp (/$NoAutoRegExp/i) matched.",
            );
            next ADDRESS;
        }

        push @AutoReplyAddresses, $Address;
    }

    my $AutoReplyAddresses = join( ', ', @AutoReplyAddresses );
    my $Cc;

    # also send CC to customer user if customer user id is used and addresses do not match
    if ( $Ticket{CustomerUserID} ) {

        my %CustomerUser = $Kernel::OM->Get('Kernel::System::CustomerUser')->CustomerUserDataGet(
            User => $Ticket{CustomerUserID},
        );

        if (
            $CustomerUser{UserEmail}
            && $OrigHeader{From} !~ /\Q$CustomerUser{UserEmail}\E/i
            && $Param{IsVisibleForCustomer}
            )
        {
            $Cc = $CustomerUser{UserEmail};
        }
    }

    # get history type
    my $HistoryType;
    if ( $Param{AutoResponseType} =~ /^auto follow up$/i ) {
        $HistoryType = 'SendAutoFollowUp';
    }
    elsif ( $Param{AutoResponseType} =~ /^auto reply$/i ) {
        $HistoryType = 'SendAutoReply';
    }
    elsif ( $Param{AutoResponseType} =~ /^auto reply\/new ticket$/i ) {
        $HistoryType = 'SendAutoReply';
    }
    elsif ( $Param{AutoResponseType} =~ /^auto reject$/i ) {
        $HistoryType = 'SendAutoReject';
    }
    else {
        $HistoryType = 'Misc';
    }

    if ( !@AutoReplyAddresses && !$Cc ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'info',
            Message  => "No auto response addresses for Ticket [$Ticket{TicketNumber}]"
                . " (TicketID=$Param{TicketID})."
        );
        return;
    }

    # Format sender realname and address compliant to RFC 5322. This is relevant when the real name contain commas
    # or other special symbols.
    my $From = Mail::Address->new( $AutoResponse{SenderRealname}, $AutoResponse{SenderAddress} );

    # send email
    my $ArticleID = $Self->ArticleSend(
        IsVisibleForCustomer => 1,
        SenderType           => 'system',
        TicketID             => $Param{TicketID},
        HistoryType          => $HistoryType,
        HistoryComment       => "\%\%$AutoReplyAddresses",
        From                 => $From->format(),
        To                   => $AutoReplyAddresses,
        Cc                   => $Cc,
        Charset              => 'utf-8',
        MimeType             => $AutoResponse{ContentType},
        Subject              => $AutoResponse{Subject},
        Body                 => $AutoResponse{Text},
        InReplyTo            => $OrigHeader{'Message-ID'},
        Loop                 => 1,
        UserID               => $Param{UserID},
    );

    # log
    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'info',
        Message  => "Sent auto response ($HistoryType) for Ticket [$Ticket{TicketNumber}]"
            . " (TicketID=$Param{TicketID}, ArticleID=$ArticleID) to '$AutoReplyAddresses'."
    );

    # event
    $Self->EventHandler(
        Event => 'ArticleAutoResponse',
        Data  => {
            TicketID => $Param{TicketID},
        },
        UserID => $Param{UserID},
    );

    return 1;
}

=head2 ArticleTransmissionStatus()

Get the transmission status for one article.

    my $TransmissionStatus = $ArticleBackendObject->ArticleTransmissionStatus(
        ArticleID => 123,   # required
    );

This returns something like:

    $TransmissionStatus = {
        ArticleID  => 123,
        MessageID  => 456,
        Message    => 'Descriptive message of last communication',  # only in case of failed status
        CreateTime => '2017-01-01 12:34:56',
        Status     => [Processing|Failed],
        Attempts   => 1,                                            # only in case of processing status
        DueTime    => '2017-01-02 12:34:56',                        # only in case of processing status
    }

=cut

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

    if ( !$Param{ArticleID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need ArticleID',
        );
        return;
    }

    my $Result = $Self->ArticleGetTransmissionError( %Param, );
    return $Result if $Result && %{$Result};

    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    return if !$DBObject->Prepare(
        SQL =>
            'SELECT article_id, create_time, attempts, due_time FROM mail_queue WHERE article_id = ?',
        Bind => [ \$Param{ArticleID} ],
    );

    if ( my @Row = $DBObject->FetchrowArray() ) {
        return {
            ArticleID  => $Row[0],
            CreateTime => $Row[1],
            Attempts   => $Row[2],
            DueTime    => $Row[3],
            Status     => 'Processing',
        };
    }

    return;
}

=head2 ArticleCreateTransmissionError()

Creates a Transmission Error entry for one article.

    my $Success = $ArticleBackendObject->ArticleCreateTransmissionError(
        ArticleID => 123,                   # Required
        MessageID => 456,                   # Optional
        Message   => '',                    # Optional
    );

=cut

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

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

    my $SQL = 'INSERT INTO article_data_mime_send_error(';

    my @Fields;
    my @Bind;

    my %MapDB = (
        ArticleID => 'article_id',
        MessageID => 'message_id',
        Message   => 'log_message',
    );

    my @PlaceHolder;

    for my $Field ( sort keys %MapDB ) {
        if ( IsStringWithData( $Param{$Field} ) ) {
            push @Fields,      $MapDB{$Field};
            push @PlaceHolder, '?';
            push @Bind,        \$Param{$Field};
        }
    }
    push @Fields, 'create_time';

    $SQL .= join( ', ', @Fields )
        . ') values(';

    $SQL .= join ', ', @PlaceHolder;

    $SQL .= ', current_timestamp)';

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    return if !$DBObject->Do(
        SQL  => $SQL,
        Bind => \@Bind,
    );

    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');
    my $TicketID      = $ArticleObject->TicketIDLookup(
        ArticleID => $Param{ArticleID},
    );

    # event
    $Self->EventHandler(
        Event => 'ArticleTransmissionErrorCreate',
        Data  => {
            ArticleID => $Param{ArticleID},
            TicketID  => $TicketID,
        },
        UserID => $Param{UserID} || 1,
    );

    return 1;
}

=head2 ArticleGetTransmissionError()

Get the Transmission Error entry for a given article.

    my %TransmissionError = $ArticleBackendObject->ArticleGetTransmissionError(
        ArticleID => 123,   # Required
    );

    Returns:
    {
        ArticleID  => 123,
        MessageID  => 456,
        Message    => 'Descriptive message of last communication',
        CreateTime => '2017-01-01 01:02:03',
        Status     => 'Failed',
    }
    or undef in case of failure to retrive a record from the database.

=cut

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

    # check needed stuff
    if ( !$Param{ArticleID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need ArticleID!"
        );
        return;
    }

    # prepare/filter ArticleID
    $Param{ArticleID} = quotemeta( $Param{ArticleID} );
    $Param{ArticleID} =~ s/\0//g;

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # can't open article, try database
    return if !$DBObject->Prepare(
        SQL =>
            'SELECT article_id, message_id, log_message, create_time FROM article_data_mime_send_error WHERE article_id = ?',
        Bind => [ \$Param{ArticleID} ],
    );

    my @Row = $DBObject->FetchrowArray();
    if (@Row) {
        return {
            'ArticleID'  => $Row[0],
            'MessageID'  => $Row[1],
            'Message'    => $Row[2],
            'CreateTime' => $Row[3],
            'Status'     => 'Failed',
        };
    }

    return;
}

=head2 ArticleUpdateTransmissionError()

Updates the Transmission Error.

    my $Result = $ArticleBackendObject->ArticleUpdateTransmissionError(
        ArticleID => 123,                           # Required
        MessageID => 456,                           # Optional
        Message   => 'Short descriptive message',   # Optional
    );

Returns 1 on Success, undef on failure.

=cut

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

    # check needed stuff
    if ( !$Param{ArticleID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need ArticleID!"
        );
        return;
    }

    my @FieldsToUpdate;
    my @Bind;

    if ( IsStringWithData( $Param{MessageID} ) ) {
        push @FieldsToUpdate, 'message_id = ?';
        push @Bind,           \$Param{MessageID};
    }

    if ( IsStringWithData( $Param{Message} ) ) {
        push @FieldsToUpdate, 'log_message = ?';
        push @Bind,           \$Param{Message};
    }

    return if !scalar @Bind;

    my $SQL = 'UPDATE article_data_mime_send_error SET '
        . join( ', ', @FieldsToUpdate )
        . ' WHERE article_id = ?';

    push @Bind, \$Param{ArticleID};

    # get database object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    # db update
    return if !$Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => $SQL,
        Bind => \@Bind,
    );

    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');
    my $TicketID      = $ArticleObject->TicketIDLookup(
        ArticleID => $Param{ArticleID},
    );

    # event
    $Self->EventHandler(
        Event => 'ArticleTransmissionErrorUpdate',
        Data  => {
            ArticleID => $Param{ArticleID},
            TicketID  => $TicketID,
        },
        UserID => $Param{UserID} || 1,
    );

    return 1;
}

1;
</File>
        <File Location="Kernel/System/PostMaster/AddressPool.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::System::PostMaster::AddressPool;

use strict;
use warnings;

use Kernel::System::VariableCheck qw(IsHashRefWithData IsArrayRefWithData);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Log',
    'Kernel::System::Queue',
    'Kernel::System::Ticket',
    'Kernel::System::LinkObject',
);

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

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

    # get parser object
    $Self->{ParserObject} = $Param{ParserObject} || '';

    # Get communication log object.
    $Self->{CommunicationLogObject} = $Param{CommunicationLogObject} || '';

    return $Self;
}

=head2 FilterPools()

Return pools addressed in the address list of To, Cc ...

    my @AddressedPools = $AddressPoolObject->FilterPools(
        Params => $GetParam,
    );

=cut

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

    # check needed stuff
    if ( !$Self->{ParserObject} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need ParserObject!",
        );
        return;
    }

    if ( !IsHashRefWithData( $Param{Params} ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need Params as hash ref!",
        );
        return;
    }

    # get headers
    my %GetParam        = $Param{Params}->%*;
    my $EmailHeader     = $Kernel::OM->Get('Kernel::Config')->Get('PostMaster::AddressPool::EmailHeaders');
    my @PrimaryHeader   = @{ $EmailHeader->{Primary}   || [] };
    my @SecondaryHeader = @{ $EmailHeader->{Secondary} || [] };

    # check possible address headers
    my %PoolsSeen;
    my @Pools;
    HEADER:
    for my $Header (@PrimaryHeader) {

        next HEADER if !$GetParam{$Header};

        my @Emails = $Self->{ParserObject}->SplitAddressLine( Line => $GetParam{$Header} );
        EMAIL:
        for my $Email (@Emails) {

            next EMAIL if !$Email;

            my $Address = $Self->{ParserObject}->GetEmailAddress( Email => $Email );

            next EMAIL if !$Address;

            my $AddressPool = $Self->PoolLookup(
                Address => $Address
            );

            next EMAIL if !$AddressPool;

            next EMAIL if $PoolsSeen{$AddressPool}++;

            push @Pools, $AddressPool;
        }
    }

    return @Pools if @Pools;

    # if no pools are addressed in the above set of headers, also look for secondary ones
    HEADER:
    for my $Header (@SecondaryHeader) {

        next HEADER if !$GetParam{$Header};

        my @Emails = $Self->{ParserObject}->SplitAddressLine( Line => $GetParam{$Header} );
        EMAIL:
        for my $Email (@Emails) {

            next EMAIL if !$Email;

            my $Address = $Self->{ParserObject}->GetEmailAddress( Email => $Email );

            next EMAIL if !$Address;

            my $AddressPool = $Self->PoolLookup(
                Address => $Address
            );

            next EMAIL if !$AddressPool;

            next EMAIL if $PoolsSeen{$AddressPool}++;

            push @Pools, $AddressPool;
        }
    }

    return @Pools;
}

=head2 AddressList()

=for stopwords addresspools
Get all addresses defined in addresspools

    my %AddressToPool = $AddressPoolObject->AddressList();

Return:

    %AddressToPool = (
              'test1@example.com' => 'Pool1',
              'test2@example.com' => 'Pool2',
              'test3@example.com' => 'Pool3',
              ...
            )

=cut

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

    return $Self->{AddressToPool}->%* if $Self->{AddressToPool};

    $Self->{AddressToPool} = {};
    my $PoolConfigs = $Kernel::OM->Get('Kernel::Config')->Get('PostMaster::AddressPool');

    if ( !$PoolConfigs ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'debug',
            Message  => "No pools defined in PostMaster::AddressPool.",
        );

        return ();
    }

    POOL:
    for my $Pool ( keys $PoolConfigs->%* ) {
        next POOL if !$PoolConfigs->{$Pool}{Emails};

        for my $Address ( $PoolConfigs->{$Pool}{Emails}->@* ) {
            $Self->{AddressToPool}{ lc($Address) } = $Pool;
        }
    }

    return $Self->{AddressToPool}->%*;
}

=head2 PoolLookup()

Get address pool by address or ticket id

    my $PoolName = $AddressPoolObject->PoolLookup(
        Address  => 'test1@example.com',
    );

    my $PoolName = $AddressPoolObject->PoolLookup(
        TicketID => 4,
    );

Return:

    $PoolName = "Pool1"

=cut

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

    # check needed stuff
    if ( !$Param{Address} && !$Param{TicketID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need Address or TicketID!",
        );
        return;
    }

    # get address pool name list
    my %AddressToPool = $Self->AddressList();

    return $AddressToPool{ lc( $Param{Address} ) } if $Param{Address};

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

    # get ticket queue id
    my $QueueID = $TicketObject->TicketQueueID(
        TicketID => $Param{TicketID},
    );

    # set address pool name
    my %QueueData = $QueueObject->QueueGet(
        ID => $QueueID,
    );

    return if !$QueueData{Email};

    return $AddressToPool{ lc( $QueueData{Email} ) };
}

=head2 QueueCheck()

Check if queue exists in address pool

    my $QueueExist = $AddressPoolObject->QueueCheck(
        Queue       => 'Junk',
        AddressPool => 'Pool1',
    );

Return:

    $QueueExist = 1

    Or

    $QueueExist = 0

=cut

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

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

    # get queue
    my %Queue = $Kernel::OM->Get('Kernel::System::Queue')->QueueGet(
        Name => $Param{Queue},
    );

    return if !%Queue;

    my $Pool = $Self->PoolLookup(
        Address => $Queue{Email},
    );

    return if !$Pool;
    return if $Pool ne $Param{AddressPool};

    return 1;
}

=head2 FindLinkedTicket()

Find linked ticket with 'Interdivisional' type in address pool

    my ( $LTTicketNumber, $LTTicketID ) = $AddressPoolObject->FindLinkedTicket(
        TicketID    => 4,
        AddressPool => 'Pool1',
        UserID      => 1,
    );

Return:

    ( $LTTicketNumber, $LTTicketID ) = ("2023012338000074", 5)

=cut

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

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

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

    # get linked tickets
    my %LinkedTickets = $LinkObject->LinkKeyListWithData(
        Object1 => 'Ticket',
        Key1    => $Param{TicketID},
        Object2 => 'Ticket',
        State   => 'Valid',
        Type    => 'Interdivisional',
        UserID  => $Param{UserID},
    );

    for my $LinkedTicket ( keys %LinkedTickets ) {

        # get ticket number / ticket id
        my $LTTicketID     = $LinkedTickets{$LinkedTicket}{TicketID};
        my $LTTicketNumber = $LinkedTickets{$LinkedTicket}{TicketNumber};

        my $LTPool = $Self->PoolLookup(
            TicketID => $LTTicketID,
        );

        if ( $LTPool && $LTPool eq $Param{AddressPool} ) {
            return ( $LTTicketNumber, $LTTicketID );
        }
    }

    return;
}

=head2 InterdivisionalTicketLinkAdd()

Add link for address pool tickets with new 'Interdivisional' type

    $AddressPoolObject->InterdivisionalTicketLinkAdd(
        TicketIDs => [1, 5, 8],
        UserID    => 1,
    );

=cut

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

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

    if ( !IsArrayRefWithData( $Param{TicketIDs} ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need TicketIDs as array ref!",
        );
        return;
    }

    my @TicketIDs;
    my %Seen;
    ID:
    for my $ID ( $Param{TicketIDs}->@* ) {
        next ID if $Seen{$ID}++;

        push @TicketIDs, $ID;
    }

    if ( scalar(@TicketIDs) < 2 ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need at least 2 ticket ids!",
        );
        return;
    }

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

    # link tickets
    for my $Source ( 0 .. $#TicketIDs ) {

        for my $Target ( $Source + 1 .. $#TicketIDs ) {

            my $Success = $LinkObject->LinkAdd(
                SourceObject => 'Ticket',
                SourceKey    => $TicketIDs[$Source],
                TargetObject => 'Ticket',
                TargetKey    => $TicketIDs[$Target],
                Type         => 'Interdivisional',
                State        => 'Valid',
                UserID       => $Param{UserID},
            );
        }
    }

    return;
}

1;
</File>
        <File Location="Kernel/System/PostMaster/Filter/SkipViaMessageID.pm" Permission="660" Encode="Base64">IyAtLQojIE9UT0JPIGlzIGEgd2ViLWJhc2VkIHRpY2tldGluZyBzeXN0ZW0gZm9yIHNlcnZpY2Ugb3JnYW5pc2F0aW9ucy4KIyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIwIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDE5LTIwMjYgUm90aGVyIE9TUyBHbWJILCBodHRwczovL290b2JvLmlvLwojIC0tCiMgVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU6IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkgaXQgdW5kZXIKIyB0aGUgdGVybXMgb2YgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZQojIEZvdW5kYXRpb24sIGVpdGhlciB2ZXJzaW9uIDMgb2YgdGhlIExpY2Vuc2UsIG9yIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiMgVGhpcyBwcm9ncmFtIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsIGJ1dCBXSVRIT1VUCiMgQU5ZIFdBUlJBTlRZOyB3aXRob3V0IGV2ZW4gdGhlIGltcGxpZWQgd2FycmFudHkgb2YgTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MKIyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuIFNlZSB0aGUgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KIyBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZQojIGFsb25nIHdpdGggdGhpcyBwcm9ncmFtLiBJZiBub3QsIHNlZSA8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy8+LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6U3lzdGVtOjpQb3N0TWFzdGVyOjpGaWx0ZXI6OlNraXBWaWFNZXNzYWdlSUQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgTWFpbDo6QWRkcmVzczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpQb3N0TWFzdGVyOjpBZGRyZXNzUG9vbCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OlN5c3RlbUFkZHJlc3MnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXQ6OkFydGljbGUnLAopOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGdldCBwYXJzZXIgb2JqZWN0CiAgICAkU2VsZi0+e1BhcnNlck9iamVjdH0gPSAkUGFyYW17UGFyc2VyT2JqZWN0fSB8fCBkaWUgIkdvdCBubyBQYXJzZXJPYmplY3QhIjsKCiAgICAjIEdldCBjb21tdW5pY2F0aW9uIGxvZyBvYmplY3QuCiAgICAkU2VsZi0+e0NvbW11bmljYXRpb25Mb2dPYmplY3R9ID0gJFBhcmFte0NvbW11bmljYXRpb25Mb2dPYmplY3R9IHx8IGRpZSAiR290IG5vIENvbW11bmljYXRpb25Mb2dPYmplY3QhIjsKCiAgICAjIEdldCBBcnRpY2xlIGJhY2tlbmQgb2JqZWN0LgogICAgJFNlbGYtPntBcnRpY2xlQmFja2VuZE9iamVjdH0gPQogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXQ6OkFydGljbGUnKS0+QmFja2VuZEZvckNoYW5uZWwoIENoYW5uZWxOYW1lID0+ICdFbWFpbCcgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCnN1YiBSdW4gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAkU2VsZi0+X0FkZENvbW11bmljYXRpb25Mb2coIE1lc3NhZ2UgPT4gJ1NlYXJjaGluZyBmb3IgaGVhZGVyIE1lc3NhZ2UtSUQuJyApOwoKICAgIG15ICRNZXNzYWdlSUQgPSAkUGFyYW17R2V0UGFyYW19LT57J01lc3NhZ2UtSUQnfTsKCiAgICByZXR1cm4gMSBpZiAhJE1lc3NhZ2VJRDsKCiAgICAkU2VsZi0+X0FkZENvbW11bmljYXRpb25Mb2coCiAgICAgICAgTWVzc2FnZSA9PiBzcHJpbnRmKAogICAgICAgICAgICAnU2VhcmNoaW5nIGZvciBhcnRpY2xlIHdpdGggbWVzc2FnZSBpZCAiJXMiLicsCiAgICAgICAgICAgICRNZXNzYWdlSUQsCiAgICAgICAgKSwKICAgICk7CgogICAgbXkgJUFydGljbGUgPSAkU2VsZi0+e0FydGljbGVCYWNrZW5kT2JqZWN0fS0+QXJ0aWNsZUdldEJ5TWVzc2FnZUlEKAogICAgICAgIE1lc3NhZ2VJRCA9PiAkTWVzc2FnZUlELAogICAgKTsKCiAgICByZXR1cm4gMSBpZiAhJUFydGljbGU7CgogICAgaWYgKCAhJEFydGljbGV7QW1iaWd1b3VzVGlja2V0SURzfSApIHsKICAgICAgICBteSAoJFNlbmRlckVtYWlsKSA9IE1haWw6OkFkZHJlc3MtPnBhcnNlKCAkQXJ0aWNsZXtGcm9tfSApOwoKICAgICAgICBteSAkSXNMb2NhbCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpTeXN0ZW1BZGRyZXNzJyktPlN5c3RlbUFkZHJlc3NJc0xvY2FsQWRkcmVzcygKICAgICAgICAgICAgQWRkcmVzcyA9PiAkU2VuZGVyRW1haWwtPmFkZHJlc3MoKSwKICAgICAgICApOwoKICAgICAgICAjIGlmIHdlIGFyZSByZWFkaW5nIGEgbWVzc2FnZSBzZW50IHZpYSBvdXIgT1RPQk8gaXRzZWxmIGZvciB0aGUgZmlyc3QgdGltZQogICAgICAgIGlmICgkSXNMb2NhbCkgewogICAgICAgICAgICAkU2VsZi0+X0FkZENvbW11bmljYXRpb25Mb2coCiAgICAgICAgICAgICAgICBNZXNzYWdlID0+IHNwcmludGYoCiAgICAgICAgICAgICAgICAgICAgJ0VtYWlsIHdpdGggbWVzc2FnZSBpZCAiJXMiIHdhcyBzZW50IGFzIG5ldyBhcnRpY2xlIGZyb20gYSBsb2NhbCBhZGRyZXNzLicsCiAgICAgICAgICAgICAgICAgICAgJE1lc3NhZ2VJRCwKICAgICAgICAgICAgICAgICksCiAgICAgICAgICAgICk7CgogICAgICAgICAgICByZXR1cm4gMTsKICAgICAgICB9CiAgICB9CgogICAgaWYgKCAhJFBhcmFte0pvYkNvbmZpZ30gfHwgISRQYXJhbXtKb2JDb25maWd9e0FsbG93TXVsdGlQYXJzZX0gKSB7CiAgICAgICAgJFNlbGYtPl9BZGRDb21tdW5pY2F0aW9uTG9nKAogICAgICAgICAgICBNZXNzYWdlID0+IHNwcmludGYoCiAgICAgICAgICAgICAgICAnQXJ0aWNsZSB3aXRoIG1lc3NhZ2UgaWQgIiVzIiBhbHJlYWR5IGV4aXN0cywgc2V0dGluZyBYLU9UT0JPLUlnbm9yZS4nLAogICAgICAgICAgICAgICAgJE1lc3NhZ2VJRCwKICAgICAgICAgICAgKSwKICAgICAgICApOwogICAgICAgICRQYXJhbXtHZXRQYXJhbX17J1gtT1RPQk8tSWdub3JlJ30gPSAneWVzJzsKCiAgICAgICAgcmV0dXJuIDE7CiAgICB9CgogICAgbXkgJE11bHRpUGFyc2UgPSAkUGFyYW17Sm9iQ29uZmlnfXtBbGxvd011bHRpUGFyc2V9e0Fsd2F5c30gPyAxIDogMDsKCiAgICBpZiAoICEkTXVsdGlQYXJzZSAmJiAkUGFyYW17Sm9iQ29uZmlnfXtBbGxvd011bHRpUGFyc2V9e01haWxBY2NvdW50UXVldWV9ICYmICRQYXJhbXtRdWV1ZUlEfSApIHsKICAgICAgICAkU2VsZi0+X0FkZENvbW11bmljYXRpb25Mb2coCiAgICAgICAgICAgIE1lc3NhZ2UgPT4gc3ByaW50ZigKICAgICAgICAgICAgICAgICdBcnRpY2xlIHdpdGggbWVzc2FnZSBpZCAiJXMiIGFscmVhZHkgZXhpc3RzLCBwYXJzZSBhZ2FpbiBhcyBleHBsaWNpdCBRdWV1ZUlEIGlzIHByb3ZpZGVkLicsCiAgICAgICAgICAgICAgICAkTWVzc2FnZUlELAogICAgICAgICAgICApLAogICAgICAgICk7CgogICAgICAgICRNdWx0aVBhcnNlID0gMTsKICAgIH0KCiAgICBpZiAoICEkTXVsdGlQYXJzZSAmJiAkUGFyYW17Sm9iQ29uZmlnfXtBbGxvd011bHRpUGFyc2V9e0JvdW5jZWRFbWFpbH0gJiYgJFBhcmFte0dldFBhcmFtfXsnUmVzZW50LVRvJ30gKSB7CiAgICAgICAgJFNlbGYtPl9BZGRDb21tdW5pY2F0aW9uTG9nKAogICAgICAgICAgICBNZXNzYWdlID0+IHNwcmludGYoCiAgICAgICAgICAgICAgICAnQXJ0aWNsZSB3aXRoIG1lc3NhZ2UgaWQgIiVzIiBhbHJlYWR5IGV4aXN0cywgcGFyc2UgYWdhaW4gYXMgZW1haWwgd2FzIGJvdW5jZWQuJywKICAgICAgICAgICAgICAgICRNZXNzYWdlSUQsCiAgICAgICAgICAgICksCiAgICAgICAgKTsKCiAgICAgICAgJE11bHRpUGFyc2UgPSAxOwogICAgfQoKICAgIGlmICggISRNdWx0aVBhcnNlICkgewogICAgICAgICRTZWxmLT5fQWRkQ29tbXVuaWNhdGlvbkxvZygKICAgICAgICAgICAgTWVzc2FnZSA9PiBzcHJpbnRmKAogICAgICAgICAgICAgICAgJ0FydGljbGUgd2l0aCBtZXNzYWdlIGlkICIlcyIgYWxyZWFkeSBleGlzdHMsIHNldHRpbmcgWC1PVE9CTy1JZ25vcmUuJywKICAgICAgICAgICAgICAgICRNZXNzYWdlSUQsCiAgICAgICAgICAgICksCiAgICAgICAgKTsKICAgICAgICAkUGFyYW17R2V0UGFyYW19eydYLU9UT0JPLUlnbm9yZSd9ID0gJ3llcyc7CgogICAgICAgIHJldHVybiAxOwogICAgfQoKICAgIG15ICRBZGRyZXNzUG9vbE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpQb3N0TWFzdGVyOjpBZGRyZXNzUG9vbCcpOwogICAgVElDS0VUSUQ6CiAgICBmb3IgbXkgJFRpY2tldElEICggJEFydGljbGV7QW1iaWd1b3VzVGlja2V0SURzfS0+QCogKSB7CiAgICAgICAgbXkgJFBvb2wgPSAkQWRkcmVzc1Bvb2xPYmplY3QtPlBvb2xMb29rdXAoCiAgICAgICAgICAgIFRpY2tldElEID0+ICRUaWNrZXRJRCwKICAgICAgICApOwoKICAgICAgICBuZXh0IFRJQ0tFVElEIGlmICEkUG9vbDsKCiAgICAgICAgJFBhcmFte0dldFBhcmFtfXtJZ25vcmVBZGRyZXNzUG9vbHN9eyRQb29sfSA9ICRUaWNrZXRJRDsKICAgIH0KCiAgICByZXR1cm4gMTsKfQoKc3ViIF9BZGRDb21tdW5pY2F0aW9uTG9nIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgJFNlbGYtPntDb21tdW5pY2F0aW9uTG9nT2JqZWN0fS0+T2JqZWN0TG9nKAogICAgICAgIE9iamVjdExvZ1R5cGUgPT4gJ01lc3NhZ2UnLAogICAgICAgIFByaW9yaXR5ICAgICAgPT4gJFBhcmFte1ByaW9yaXR5fSB8fCAnRGVidWcnLAogICAgICAgIEtleSAgICAgICAgICAgPT4gcmVmKCRTZWxmKSwKICAgICAgICBWYWx1ZSAgICAgICAgID0+ICRQYXJhbXtNZXNzYWdlfSwKICAgICk7CgogICAgcmV0dXJuOwp9CgoxOwo=</File>
        <File Location="Custom/Kernel/Modules/AgentTicketBounce.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/Modules/AgentTicketBounce.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::Modules::AgentTicketBounce;

use strict;
use warnings;

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

our $ObjectManagerDisabled = 1;

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

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

    # get article ID
    $Self->{ArticleID} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'ArticleID' ) || '';

    return $Self;
}

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

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

    # check needed stuff
    for my $Needed (qw(ArticleID TicketID QueueID)) {
        if ( !defined $Self->{$Needed} ) {
            return $LayoutObject->ErrorScreen(
                Message => $LayoutObject->{LanguageObject}->Translate( '%s is needed!', $Needed ),
                Comment => Translatable('Please contact the administrator.'),
            );
        }
    }

    # get ticket data
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');
    my %Ticket       = $TicketObject->TicketGet( TicketID => $Self->{TicketID} );

    # get config object
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    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( 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) {

        my %AclActionLookup = reverse %AclAction;

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

    # get lock state && write (lock) permissions
    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(
                    Value     => $Ticket{Number},
                    Type      => 'Small',
                    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;
            }
            else {
                $LayoutObject->Block(
                    Name => 'TicketBack',
                    Data => {
                        %Param,
                        TicketID => $Self->{TicketID},
                    },
                );
            }
        }
    }
    else {
        $LayoutObject->Block(
            Name => 'TicketBack',
            Data => {
                %Param,
                TicketID => $Self->{TicketID},
            },
        );
    }

    # ------------------------------------------------------------ #
    # show screen
    # ------------------------------------------------------------ #
    if ( !$Self->{Subaction} ) {

        my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

        my $ArticleBackendObject = $ArticleObject->BackendForArticle(
            TicketID  => $Self->{TicketID},
            ArticleID => $Self->{ArticleID},
        );

        # check if plain article exists
        if ( !$ArticleBackendObject->ArticlePlain( ArticleID => $Self->{ArticleID} ) ) {
            return $LayoutObject->ErrorScreen(
                Message => $LayoutObject->{LanguageObject}->Translate(
                    'Plain article not found for article %s!',
                    $Self->{ArticleID}
                ),
            );
        }

        # get article data
        my %Article = $ArticleBackendObject->ArticleGet(
            TicketID      => $Self->{TicketID},
            ArticleID     => $Self->{ArticleID},
            DynamicFields => 0,
        );

        # Check if article is from the same TicketID as we checked permissions for.
        if ( $Article{TicketID} ne $Self->{TicketID} ) {
            return $LayoutObject->ErrorScreen(
                Message => $LayoutObject->{LanguageObject}->Translate(
                    'Article does not belong to ticket %s!',
                    $Self->{TicketID}
                ),
            );
        }

        # prepare to (ReplyTo!) ...
        if ( $Article{ReplyTo} ) {
            $Article{To} = $Article{ReplyTo};
        }
        else {
            $Article{To} = $Article{From};
        }

        # get template generator object
        $Kernel::OM->ObjectParamAdd(
            'Kernel::System::TemplateGenerator' => { %{$Self} }
        );
        my $TemplateGenerator = $Kernel::OM->Get('Kernel::System::TemplateGenerator');

        # prepare salutation
        $Param{Salutation} = $TemplateGenerator->Salutation(
            TicketID  => $Self->{TicketID},
            ArticleID => $Self->{ArticleID},
            Data      => \%Param,
            UserID    => $Self->{UserID},
        );

        # prepare signature
        $Param{Signature} = $TemplateGenerator->Signature(
            TicketID  => $Self->{TicketID},
            ArticleID => $Self->{ArticleID},
            Data      => \%Param,
            UserID    => $Self->{UserID},
        );

        # prepare bounce text
        $Param{BounceText} = $ConfigObject->Get('Ticket::Frontend::BounceText') || '';

        # make sure body is rich text
        if ( $LayoutObject->{BrowserRichText} ) {

            # prepare bounce tags
            $Param{BounceText} =~ s/<OTOBO_TICKET>/&lt;OTOBO_TICKET&gt;/g;
            $Param{BounceText} =~ s/<OTOBO_BOUNCE_TO>/&lt;OTOBO_BOUNCE_TO&gt;/g;

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

        # build InformationFormat
        if ( $LayoutObject->{BrowserRichText} ) {
            $Param{InformationFormat} = "$Param{Salutation}<br/>
<br/>
$Param{BounceText}<br/>
<br/>
$Param{Signature}";
        }
        else {
            $Param{InformationFormat} = "$Param{Salutation}

$Param{BounceText}

$Param{Signature}";
        }

        # prepare sender of bounce email
        my %Address = $Kernel::OM->Get('Kernel::System::Queue')->GetSystemAddress(
            QueueID => $Ticket{QueueID},
        );
        $Article{From} = "$Address{RealName} <$Address{Email}>";

        # get next states
        my %NextStates = $TicketObject->TicketStateList(
            Action   => $Self->{Action},
            TicketID => $Self->{TicketID},
            UserID   => $Self->{UserID},
        );

        # build next states string
        if ( !$Config->{StateDefault} ) {
            $NextStates{''} = '-';
        }
        $Param{NextStatesStrg} = $LayoutObject->BuildSelection(
            Data          => \%NextStates,
            Name          => 'BounceStateID',
            SelectedValue => $Config->{StateDefault},
            Class         => 'Modernize',
            Translation   => 1,
        );

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

        # explanatory message about asterisk
        if ( $ConfigObject->Get('Ticket::Frontend::AsteriskExplanation') ) {
            $LayoutObject->Block(
                Name => 'AsteriskExplanation',
            );
        }

        # print form ...
        my $Output = $LayoutObject->Header(
            Value     => $Ticket{TicketNumber},
            Type      => 'Small',
            BodyClass => 'Popup',
        );
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AgentTicketBounce',
            Data         => {
                %Param,
                %Article,
                TicketID     => $Self->{TicketID},
                ArticleID    => $Self->{ArticleID},
                TicketNumber => $Ticket{TicketNumber},
            },
        );
        $Output .= $LayoutObject->Footer(
            Type => 'Small',
        );
        return $Output;
    }

    # ------------------------------------------------------------ #
    # bounce
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Store' ) {

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

        # get params
        for my $Parameter (qw(BounceTo To Subject Body InformSender BounceStateID)) {
            $Param{$Parameter} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Parameter )
                || '';
        }

        # Make sure sender is correct one.
        $Param{From} = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->Sender(
            QueueID => $Ticket{QueueID},
            UserID  => $Self->{UserID},
        );

        my %Error;

        # check forward email address
        if ( !$Param{BounceTo} ) {
            $Error{'BounceToInvalid'} = 'ServerError';
            $LayoutObject->Block( Name => 'BounceToCustomerGenericServerErrorMsg' );
        }

        # get check item object
        my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

        for my $Email ( Mail::Address->parse( $Param{BounceTo} ) ) {
            my $Address = $Email->address();

# Rother OSS / DiscreteSystemAddresses
#            if ( $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress( Address => $Address ) )
            my $IsLocal = $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress(
                Address  => $Address,
                TicketID => $Self->{TicketID},
            );
            if ($IsLocal)
# EO DiscreteSystemAddresses
            {
                $LayoutObject->Block( Name => 'BounceToCustomerGenericServerErrorMsg' );
                $Error{'BounceToInvalid'} = 'ServerError';
            }

            # check email address
            elsif ( !$CheckItemObject->CheckEmail( Address => $Address ) ) {
                my $BounceToErrorMsg =
                    'BounceTo'
                    . $CheckItemObject->CheckErrorType()
                    . 'ServerErrorMsg';
                $LayoutObject->Block( Name => $BounceToErrorMsg );
                $Error{'BounceToInvalid'} = 'ServerError';
            }
        }

        if ( $Param{InformSender} ) {
            if ( !$Param{To} ) {
                $Error{'ToInvalid'} = 'ServerError';
                $LayoutObject->Block( Name => 'ToCustomerGenericServerErrorMsg' );
            }
            else {

                # check email address(es)
                for my $Email ( Mail::Address->parse( $Param{To} ) ) {
                    if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                        my $ToErrorMsg =
                            'To'
                            . $CheckItemObject->CheckErrorType()
                            . 'ServerErrorMsg';
                        $LayoutObject->Block( Name => $ToErrorMsg );
                        $Error{'ToInvalid'} = 'ServerError';
                    }
                }
            }

            if ( !$Param{Subject} ) {
                $Error{'SubjectInvalid'} = 'ServerError';
            }
            if ( !$Param{Body} ) {
                $Error{'BodyInvalid'} = 'ServerError';
            }
        }

        #check for error
        if (%Error) {

            # get next states
            my %NextStates = $TicketObject->TicketStateList(
                Action   => $Self->{Action},
                TicketID => $Self->{TicketID},
                UserID   => $Self->{UserID},
            );

            # build next states string
            if ( !$Config->{StateDefault} ) {
                $NextStates{''} = '-';
            }
            $Param{NextStatesStrg} = $LayoutObject->BuildSelection(
                Data        => \%NextStates,
                Name        => 'BounceStateID',
                SelectedID  => $Param{BounceStateID},
                Class       => 'Modernize',
                Translation => 1,
            );

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

            # prepare bounce tags if body is rich text
            if ( $LayoutObject->{BrowserRichText} ) {

                # prepare bounce tags
                $Param{Body} =~ s/&lt;OTOBO_TICKET&gt;/&amp;lt;OTOBO_TICKET&amp;gt;/gi;
                $Param{Body} =~ s/&lt;OTOBO_BOUNCE_TO&gt;/&amp;lt;OTOBO_BOUNCE_TO&amp;gt;/gi;
            }

            $Param{InformationFormat}   = $Param{Body};
            $Param{InformSenderChecked} = $Param{InformSender} ? 'checked ' : '';

            # explanatory message about asterisk
            if ( $ConfigObject->Get('Ticket::Frontend::AsteriskExplanation') ) {
                $LayoutObject->Block(
                    Name => 'AsteriskExplanation',
                );
            }

            my $Output = $LayoutObject->Header(
                Type      => 'Small',
                BodyClass => 'Popup',
            );
            $Output .= $LayoutObject->Output(
                TemplateFile => 'AgentTicketBounce',
                Data         => {
                    %Param,
                    %Error,
                    TicketID  => $Self->{TicketID},
                    ArticleID => $Self->{ArticleID},

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

        my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

        my $ArticleBackendObject = $ArticleObject->BackendForArticle(
            TicketID  => $Self->{TicketID},
            ArticleID => $Self->{ArticleID},
        );

        my $Bounce = $ArticleBackendObject->ArticleBounce(
            TicketID    => $Self->{TicketID},
            ArticleID   => $Self->{ArticleID},
            UserID      => $Self->{UserID},
            To          => $Param{BounceTo},
            From        => $Param{From},
            HistoryType => 'Bounce',
        );

        # error page
        if ( !$Bounce ) {
            return $LayoutObject->ErrorScreen(
                Message => Translatable('Can\'t bounce email!'),
                Comment => Translatable('Please contact the administrator.'),
            );
        }

        # send customer info?
        if ( $Param{InformSender} ) {

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

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

            # replace placeholders
            $Param{Body} =~ s/(&lt;|<)OTOBO_TICKET(&gt;|>)/$Ticket{TicketNumber}/g;
            $Param{Body} =~ s/(&lt;|<)OTOBO_BOUNCE_TO(&gt;|>)/$Param{BounceTo}/g;

            # send
            my $ArticleID = $ArticleBackendObject->ArticleSend(
                TicketID             => $Self->{TicketID},
                SenderType           => 'agent',
                IsVisibleForCustomer => 1,
                HistoryType          => 'Bounce',
                HistoryComment       => "Bounced info to '$Param{To}'.",
                From                 => $Param{From},
                Email                => $Param{Email},
                To                   => $Param{To},
                Subject              => $Param{Subject},
                UserID               => $Self->{UserID},
                Body                 => $Param{Body},
                Charset              => $LayoutObject->{UserCharset},
                MimeType             => $MimeType,
            );

            # error page
            if ( !$ArticleID ) {
                return $LayoutObject->ErrorScreen(
                    Message => Translatable('Can\'t send email!'),
                    Comment => Translatable('Please contact the administrator.'),
                );
            }
        }

        # check if there is a chosen bounce state id
        if ( $Param{BounceStateID} ) {

            # set state
            my %StateData = $Kernel::OM->Get('Kernel::System::State')->StateGet(
                ID => $Param{BounceStateID},
            );
            $TicketObject->TicketStateSet(
                TicketID  => $Self->{TicketID},
                ArticleID => $Self->{ArticleID},
                StateID   => $Param{BounceStateID},
                UserID    => $Self->{UserID},
            );

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

            # redirect
            if (
                $StateData{TypeName} =~ /^close/i
                && !$ConfigObject->Get('Ticket::Frontend::RedirectAfterCloseDisabled')
                )
            {
                return $LayoutObject->PopupClose(
                    URL => ( $Self->{LastScreenOverview} || 'Action=AgentDashboard' ),
                );
            }
        }
        return $LayoutObject->PopupClose(
            URL => "Action=AgentTicketZoom;TicketID=$Self->{TicketID};ArticleID=$Self->{ArticleID}",
        );
    }
    return $LayoutObject->ErrorScreen(
        Message => Translatable('Wrong Subaction!'),
        Comment => Translatable('Please contact the administrator.'),
    );
}

1;
</File>
        <File Location="Custom/Kernel/Modules/AgentTicketCompose.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/Modules/AgentTicketCompose.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::Modules::AgentTicketCompose;

use v5.24;
use strict;
use warnings;

# core modules

# CPAN modules

# OTOBO modules
use Kernel::System::VariableCheck qw(:all);
use Kernel::Language              qw(Translatable);
use Mail::Address                 ();

our $ObjectManagerDisabled = 1;

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

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

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

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

    $Self->{Debug} = $Param{Debug} || 0;

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

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

    # get the dynamic fields for this screen
    my $DynamicFieldList = $DynamicFieldObject->DynamicFieldListGet(
        Valid       => 1,
        ObjectType  => [ 'Ticket', 'Article' ],
        FieldFilter => $Config->{DynamicField} || {},
    );

    my $Definition = $Kernel::OM->Get('Kernel::System::Ticket::Mask')->DefinitionGet(
        Mask => $Self->{Action},
    ) || {};

    $Self->{DynamicField}   = [];
    $Self->{MaskDefinition} = $Definition->{Mask};

    # align sysconfig and ticket mask data I
    for my $DynamicField ( @{ $DynamicFieldList // [] } ) {
        if ( exists $Definition->{DynamicFields}{ $DynamicField->{Name} } ) {
            my $Parameters = delete $Definition->{DynamicFields}{ $DynamicField->{Name} } // {};

            for my $Attribute ( keys $Parameters->%* ) {
                $DynamicField->{$Attribute} = $Parameters->{$Attribute};
            }
        }
        else {
            push $Self->{MaskDefinition}->@*, {
                DF        => $DynamicField->{Name},
                Mandatory => $Config->{DynamicField}{ $DynamicField->{Name} } == 2 ? 1 : 0,
            };

            if ( $Config->{DynamicField}{ $DynamicField->{Name} } == 2 ) {
                $DynamicField->{Mandatory} = 1;
            }
        }

        push $Self->{DynamicField}->@*, $DynamicField;
    }

    # align sysconfig and ticket mask data II
    for my $DynamicFieldName ( keys $Definition->{DynamicFields}->%* ) {
        push $Self->{DynamicField}->@*, $DynamicFieldObject->DynamicFieldGet(
            Name => $DynamicFieldName,
        );

        my $Parameters = $Definition->{DynamicFields}{$DynamicFieldName} // {};

        for my $Attribute ( keys $Parameters->%* ) {
            $Self->{DynamicField}[-1]{$Attribute} = $Parameters->{$Attribute};
        }
    }

    # get form id
    $Self->{FormID} = $Kernel::OM->Get('Kernel::System::Web::FormCache')->PrepareFormID(
        ParamObject  => $Kernel::OM->Get('Kernel::System::Web::Request'),
        LayoutObject => $Kernel::OM->Get('Kernel::Output::HTML::Layout'),
    );

    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');
    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');
    my $MainObject    = $Kernel::OM->Get('Kernel::System::Main');

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

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

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

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

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

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

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

                # Show lock state.
                if ( !$Success ) {
                    return $LayoutObject->FatalError();
                }
            }

            $TicketBackType .= 'Undo';
        }
        else {
            my $AccessOk = $TicketObject->OwnerCheck(
                TicketID => $Self->{TicketID},
                OwnerID  => $Self->{UserID},
            );
            if ( !$AccessOk ) {
                my $Output = $LayoutObject->Header(
                    Value     => $Ticket{Number},
                    Type      => 'Small',
                    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;
            }
        }
    }

    my $ArticleBackendObject = $ArticleObject->BackendForChannel( ChannelName => 'Email' );

    # get params
    my %GetParam;
    for (
        qw(
            To Cc Bcc Subject Body InReplyTo References ResponseID ReplyArticleID StateID ArticleID
            IsVisibleForCustomerPresent IsVisibleForCustomer TimeUnits Year Month Day Hour Minute FormID ReplyAll
            FormDraftID Title
        )
        )
    {
        $GetParam{$_} = $ParamObject->GetParam( Param => $_ );
    }

    # Make sure sender is correct one. See bug#14872 ( https://bugs.otrs.org/show_bug.cgi?id=14872 ).
    $GetParam{From} = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->Sender(
        QueueID => $Ticket{QueueID},
        UserID  => $Self->{UserID},
    );

    # hash for check duplicated entries
    my %AddressesList;

    my @MultipleCustomer;
    my $CustomersNumberTo = $ParamObject->GetParam( Param => 'CustomerTicketCounterToCustomer' ) || 0;
    my $Selected          = $ParamObject->GetParam( Param => 'CustomerSelected' )                || '';

    # get check item object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    if ($CustomersNumberTo) {
        my $CustomerCounter = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberTo ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElement = $ParamObject->GetParam( Param => 'CustomerTicketText_' . $Count );

            next COUNT unless $CustomerElement;

            my $CustomerSelected = ( $Selected eq $Count ? 'checked ' : '' );
            my $CustomerKey      = $ParamObject->GetParam( Param => 'CustomerKey_' . $Count )   || '';
            my $CustomerQueue    = $ParamObject->GetParam( Param => 'CustomerQueue_' . $Count ) || '';

            if ( $GetParam{To} ) {
                $GetParam{To} .= ', ' . $CustomerElement;
            }
            else {
                $GetParam{To} = $CustomerElement;
            }

            # check email address
            my $CustomerErrorMsg = 'CustomerGenericServerErrorMsg';
            my $CustomerError    = '';
            for my $Email ( Mail::Address->parse($CustomerElement) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsg = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerError = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElement} && $CustomerError eq '' ) {
                $CustomerErrorMsg = 'IsDuplicatedServerErrorMsg';
                $CustomerError    = 'ServerError';
            }

            my $CustomerDisabled = '';
            my $CountAux         = $CustomerCounter++;
            if ( $CustomerError ne '' ) {
                $CustomerDisabled = 'disabled';
                $CountAux         = $Count . 'Error';
            }

            if ( $CustomerQueue ne '' ) {
                $CustomerQueue = $Count;
            }

            push @MultipleCustomer, {
                Count            => $CountAux,
                CustomerElement  => $CustomerElement,
                CustomerSelected => $CustomerSelected,
                CustomerKey      => $CustomerKey,
                CustomerError    => $CustomerError,
                CustomerErrorMsg => $CustomerErrorMsg,
                CustomerDisabled => $CustomerDisabled,
                CustomerQueue    => $CustomerQueue,
            };
            $AddressesList{$CustomerElement} = 1;
        }
    }

    my @MultipleCustomerCc;
    my $CustomersNumberCc = $ParamObject->GetParam( Param => 'CustomerTicketCounterCcCustomer' ) || 0;

    if ($CustomersNumberCc) {
        my $CustomerCounterCc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberCc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementCc = $ParamObject->GetParam( Param => 'CcCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementCc;

            my $CustomerKeyCc   = $ParamObject->GetParam( Param => 'CcCustomerKey_' . $Count )   || '';
            my $CustomerQueueCc = $ParamObject->GetParam( Param => 'CcCustomerQueue_' . $Count ) || '';

            if ( $GetParam{Cc} ) {
                $GetParam{Cc} .= ', ' . $CustomerElementCc;
            }
            else {
                $GetParam{Cc} = $CustomerElementCc;
            }

            # check email address
            my $CustomerErrorMsgCc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorCc    = '';
            for my $Email ( Mail::Address->parse($CustomerElementCc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsgCc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorCc = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElementCc} && $CustomerErrorCc eq '' ) {
                $CustomerErrorMsgCc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorCc    = 'ServerError';
            }

            my $CustomerDisabledCc = '';
            my $CountAuxCc         = $CustomerCounterCc++;
            if ( $CustomerErrorCc ne '' ) {
                $CustomerDisabledCc = 'disabled';
                $CountAuxCc         = $Count . 'Error';
            }

            if ( $CustomerQueueCc ne '' ) {
                $CustomerQueueCc = $Count;
            }

            push @MultipleCustomerCc, {
                Count            => $CountAuxCc,
                CustomerElement  => $CustomerElementCc,
                CustomerKey      => $CustomerKeyCc,
                CustomerError    => $CustomerErrorCc,
                CustomerErrorMsg => $CustomerErrorMsgCc,
                CustomerDisabled => $CustomerDisabledCc,
                CustomerQueue    => $CustomerQueueCc,
            };
            $AddressesList{$CustomerElementCc} = 1;
        }
    }

    my @MultipleCustomerBcc;
    my $CustomersNumberBcc = $ParamObject->GetParam( Param => 'CustomerTicketCounterBccCustomer' ) || 0;

    if ($CustomersNumberBcc) {
        my $CustomerCounterBcc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberBcc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementBcc = $ParamObject->GetParam( Param => 'BccCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementBcc;

            my $CustomerKeyBcc   = $ParamObject->GetParam( Param => 'BccCustomerKey_' . $Count )   || '';
            my $CustomerQueueBcc = $ParamObject->GetParam( Param => 'BccCustomerQueue_' . $Count ) || '';

            if ( $GetParam{Bcc} ) {
                $GetParam{Bcc} .= ', ' . $CustomerElementBcc;
            }
            else {
                $GetParam{Bcc} = $CustomerElementBcc;
            }

            # check email address
            my $CustomerErrorMsgBcc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorBcc    = '';
            for my $Email ( Mail::Address->parse($CustomerElementBcc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsgBcc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorBcc = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElementBcc} && $CustomerErrorBcc eq '' ) {
                $CustomerErrorMsgBcc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorBcc    = 'ServerError';
            }

            my $CustomerDisabledBcc = '';
            my $CountAuxBcc         = $CustomerCounterBcc++;
            if ( $CustomerErrorBcc ne '' ) {
                $CustomerDisabledBcc = 'disabled';
                $CountAuxBcc         = $Count . 'Error';
            }

            if ( $CustomerQueueBcc ne '' ) {
                $CustomerQueueBcc = $Count;
            }

            push @MultipleCustomerBcc, {
                Count            => $CountAuxBcc,
                CustomerElement  => $CustomerElementBcc,
                CustomerKey      => $CustomerKeyBcc,
                CustomerError    => $CustomerErrorBcc,
                CustomerErrorMsg => $CustomerErrorMsgBcc,
                CustomerDisabled => $CustomerDisabledBcc,
                CustomerQueue    => $CustomerQueueBcc,
            };
            $AddressesList{$CustomerElementBcc} = 1;
        }
    }

    # get Dynamic fields form ParamObject
    my %DynamicFieldValues;

    # 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 ( @{ $Self->{DynamicField} } ) {
        next DYNAMICFIELD unless 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 unless $DynamicFieldItem;
            next DYNAMICFIELD unless 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,
        );
    }

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

    # send email
    if ( $Self->{Subaction} eq 'SendEmail' || $Self->{LoadedFormDraftID} ) {

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

        # get valid state id
        if ( !$GetParam{StateID} ) {
            my %Ticket = $TicketObject->TicketGet(
                TicketID => $Self->{TicketID},
                UserID   => 1,
            );
            $GetParam{StateID} = $Ticket{StateID};
        }

        my %StateData = $Kernel::OM->Get('Kernel::System::State')->StateGet( ID => $GetParam{StateID} );

        my %Error;

        # get check item object
        my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

        # check some values
        LINE:
        for my $Line (qw(To Cc Bcc)) {
            next LINE if !$GetParam{$Line};
            for my $Email ( Mail::Address->parse( $GetParam{$Line} ) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $Error{ $Line . 'ErrorType' } = $Line . $CheckItemObject->CheckErrorType() . 'ServerErrorMsg';
                    $Error{ $Line . 'Invalid' }   = 'ServerError';
                }
                my $IsLocal = $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress(
# Rother OSS / DiscreteSystemAddresses
                    TicketID => $Self->{TicketID},
# EO DiscreteSystemAddresses
                    Address => $Email->address()
                );
                if ($IsLocal) {
                    $Error{ $Line . 'IsLocalAddress' } = 'ServerError';
                }
            }
        }

        if ( $Error{ToIsLocalAddress} ) {
            $LayoutObject->Block(
                Name => 'ToIsLocalAddressServerErrorMsg',
                Data => \%GetParam,
            );
        }

        if ( $Error{CcIsLocalAddress} ) {
            $LayoutObject->Block(
                Name => 'CcIsLocalAddressServerErrorMsg',
                Data => \%GetParam,
            );
        }

        if ( $Error{BccIsLocalAddress} ) {
            $LayoutObject->Block(
                Name => 'BccIsLocalAddressServerErrorMsg',
                Data => \%GetParam,
            );
        }

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

                # meddling with the innards of Kernel::System::Web::Request
                $ParamObject->SetArray(
                    Param  => 'FormDraftID',
                    Values => ['']
                );
            }

            my $FormDraftActionOk;
            if (
                $FormDraftAction eq 'Add'
                ||
                ( $FormDraftAction eq 'Update' && $GetParam{FormDraftID} )
                )
            {
                $FormDraftActionOk = $ParamObject->SaveFormDraft(
                    UserID     => $Self->{UserID},
                    ObjectType => 'Ticket',
                    ObjectID   => $Self->{TicketID},
                );
            }
            elsif ( $FormDraftAction eq 'Delete' && $GetParam{FormDraftID} ) {
                $FormDraftActionOk = $Kernel::OM->Get('Kernel::System::FormDraft')->FormDraftDelete(
                    FormDraftID => $GetParam{FormDraftID},
                    UserID      => $Self->{UserID},
                );
            }

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

        # Return JSON when there already is a response
        if (%FormDraftResponse) {

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

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

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

            # convert pending date to a 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');

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

        # check if at least one recipient has been chosen
        if ( !$GetParam{To} ) {
            $Error{'ToInvalid'} = 'ServerError';
        }

        # check some values
        LINE:
        for my $Line (qw(To Cc Bcc)) {
            next LINE if !$GetParam{$Line};
            for my $Email ( Mail::Address->parse( $GetParam{$Line} ) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $Error{ $Line . 'Invalid' } = 'ServerError';
                }
            }
        }

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

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

        # check time units
        if (
            $ConfigObject->Get('Ticket::Frontend::AccountTime')
            && $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
            && $GetParam{TimeUnits} eq ''
            )
        {
            $Error{TimeUnitsInvalid} = 'ServerError';
        }

        # prepare subject
        my $Tn = $TicketObject->TicketNumberLookup( TicketID => $Self->{TicketID} );
        $GetParam{Subject} = $TicketObject->TicketSubjectBuild(
            TicketNumber => $Tn,
            Subject      => $GetParam{Subject} || '',
        );

        my %ArticleParam;

        # run compose modules
        if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {

            # use ticket QueueID in compose modules
            $GetParam{QueueID} = $Ticket{QueueID};

            my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
            for my $Job ( sort keys %Jobs ) {

                # load module
                if ( !$MainObject->Require( $Jobs{$Job}->{Module} ) ) {
                    return $LayoutObject->FatalError();
                }
                my $Object = $Jobs{$Job}->{Module}->new( %{$Self}, Debug => $Self->{Debug} );

                my $Multiple;

                # get params
                PARAMETER:
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                    if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                        @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                        $Multiple = 1;
                        next PARAMETER;
                    }

                    $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
                }

                # run module
                $Object->Run(
                    %GetParam,
                    StoreNew => 1,
                    Config   => $Jobs{$Job}
                );

                # get options that have been removed from the selection
                # and add them back to the selection so that the submit
                # will contain options that were hidden from the agent
                my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );

                if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                    my @RemovedOptions = $Object->GetOptionsToRemoveAJAX(%GetParam);
                    if (@RemovedOptions) {
                        if ($Multiple) {
                            for my $RemovedOption (@RemovedOptions) {
                                push @{ $GetParam{$Key} }, $RemovedOption;
                            }
                        }
                        else {
                            $GetParam{$Key} = shift @RemovedOptions;
                        }
                    }
                }

                # ticket params
                %ArticleParam = (
                    %ArticleParam,
                    $Object->ArticleOption( %GetParam, %ArticleParam, Config => $Jobs{$Job} ),
                );

                # get errors
                %Error = (
                    %Error,
                    $Object->Error( %GetParam, Config => $Jobs{$Job} ),
                );
            }
        }

        # remember dynamic field validation results if erroneous
        my %DynamicFieldValidationResult;
        my %DynamicFieldPossibleValues;

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

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

            $DynamicFieldPossibleValues{ 'DynamicField_' . $DynamicFieldConfig->{Name} } = $PossibleValuesFilter;

            my $ValidationResult = $DynamicFieldBackendObject->EditFieldValueValidate(
                DynamicFieldConfig   => $DynamicFieldConfig,
                PossibleValuesFilter => $PossibleValuesFilter,
                ParamObject          => $ParamObject,

                # Mandatory is added to the configs by $Self->new
                Mandatory => $DynamicFieldConfig->{Mandatory},
            );

            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';
                $DynamicFieldValidationResult{ $DynamicFieldConfig->{Name} } = $ValidationResult;
            }
        }

        # 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(
                Value     => $Ticket{TicketNumber},
                Type      => 'Small',
                BodyClass => 'Popup',
            );

            # When a draft is loaded, inform a user that article subject will be empty
            # if contains only the ticket hook (if nothing is modified).
            if ( $Error{LoadedFormDraft} ) {
                $Output .= $LayoutObject->Notify(
                    Data => $LayoutObject->{LanguageObject}->Translate(
                        'Article subject will be empty if the subject contains only the ticket hook!'
                    ),
                );
            }

            $GetParam{StandardResponse} = $GetParam{Body};
            $Output .= $Self->_Mask(
                TicketID   => $Self->{TicketID},
                NextStates => $Self->_GetNextStates(
                    %GetParam,
                ),
                ResponseFormat      => $LayoutObject->Ascii2Html( Text => $GetParam{Body} ),
                Errors              => \%Error,
                MultipleCustomer    => \@MultipleCustomer,
                MultipleCustomerCc  => \@MultipleCustomerCc,
                MultipleCustomerBcc => \@MultipleCustomerBcc,
                Attachments         => \@Attachments,
                GetParam            => \%GetParam,
                TicketBackType      => $TicketBackType,
                %Ticket,
                %GetParam,
                DFPossibleValues => \%DynamicFieldPossibleValues,
                DFErrors         => \%DynamicFieldValidationResult,
            );
            $Output .= $LayoutObject->Footer(
                Type => 'Small',
            );
            return $Output;
        }

        # replace <OTOBO_TICKET_STATE> with next ticket state name
        if ( $StateData{Name} ) {
            $GetParam{Body} =~ s/<OTOBO_TICKET_STATE>/$StateData{Name}/g;
            $GetParam{Body} =~ s/&lt;OTOBO_TICKET_STATE&gt;/$StateData{Name}/g;
        }

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

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

        # get recipients
        my $Recipients = '';
        LINE:
        for my $Line (qw(To Cc Bcc)) {

            next LINE if !$GetParam{$Line};

            if ($Recipients) {
                $Recipients .= ', ';
            }
            $Recipients .= $GetParam{$Line};
        }

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

            # remove unused inline images
            my @NewAttachmentData;
            ATTACHMENT:
            for my $Attachment (@AttachmentData) {
                my $ContentID = $Attachment->{ContentID};
                if ( $ContentID && ( $Attachment->{ContentType} =~ /image/i ) ) {
                    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;

                    # IF the image is referenced in the body set it as inline.
                    if ( $GetParam{Body} =~ /(\Q$ContentIDHTMLQuote\E|\Q$ContentID\E)/i ) {
                        $Attachment->{Disposition} = 'inline';
                    }
                    elsif ( $Attachment->{Disposition} eq 'inline' ) {

                        # Ignore attachment if not linked in body.
                        next ATTACHMENT;
                    }
                }

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

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

        my $IsVisibleForCustomer = $Config->{IsVisibleForCustomerDefault};
        if ( $GetParam{IsVisibleForCustomerPresent} ) {
            $IsVisibleForCustomer = $GetParam{IsVisibleForCustomer} ? 1 : 0;
        }

        # Get attributes like sender address
        my %Data = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->Attributes(
            TicketID => $Self->{TicketID},
            Data     => {},
            UserID   => $Self->{UserID},
        );

        # send email
        my $ArticleID = $ArticleBackendObject->ArticleSend(
            IsVisibleForCustomer => $IsVisibleForCustomer,
            SenderType           => 'agent',
            TicketID             => $Self->{TicketID},
            HistoryType          => 'SendAnswer',
            HistoryComment       => "\%\%$Recipients",
            From                 => $Data{From},
            To                   => $GetParam{To},
            Cc                   => $GetParam{Cc},
            Bcc                  => $GetParam{Bcc},
            Subject              => $GetParam{Subject},
            UserID               => $Self->{UserID},
            Body                 => $GetParam{Body},
            InReplyTo            => $GetParam{InReplyTo},
            References           => $GetParam{References},
            Charset              => $LayoutObject->{UserCharset},
            MimeType             => $MimeType,
            Attachment           => \@AttachmentData,
            %ArticleParam,
        );

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

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

        # set dynamic fields
        # cycle through the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( @{ $Self->{DynamicField} } ) {
            next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);
            next DYNAMICFIELD if $DynamicFieldConfig->{Readonly};

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

            # set the value which was taken from web request
            # TODO: for Reference and Lens, the order is relevant
            my $Success = $DynamicFieldBackendObject->ValueSet(
                DynamicFieldConfig => $DynamicFieldConfig,
                ObjectID           => $ObjectID,
                Value              => $DynamicFieldValues{ $DynamicFieldConfig->{Name} },
                UserID             => $Self->{UserID},
            );
        }

        # set state
        $TicketObject->TicketStateSet(
            TicketID  => $Self->{TicketID},
            ArticleID => $ArticleID,
            StateID   => $GetParam{StateID},
            UserID    => $Self->{UserID},
        );

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

        # set pending time
        elsif ( $StateData{TypeName} =~ /^pending/i ) {
            $TicketObject->TicketPendingTimeSet(
                UserID   => $Self->{UserID},
                TicketID => $Self->{TicketID},
                Year     => $GetParam{Year},
                Month    => $GetParam{Month},
                Day      => $GetParam{Day},
                Hour     => $GetParam{Hour},
                Minute   => $GetParam{Minute},
            );
        }

        # log use response id and reply article id (useful for response diagnostics)
        $TicketObject->HistoryAdd(
            Name         => "ResponseTemplate ($GetParam{ResponseID}/$GetParam{ReplyArticleID}/$ArticleID)",
            HistoryType  => 'Misc',
            TicketID     => $Self->{TicketID},
            CreateUserID => $Self->{UserID},
        );

        # remove all form data
        $Kernel::OM->Get('Kernel::System::Web::FormCache')->FormIDRemove( FormID => $Self->{FormID} );

        # 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
        if (
            $StateData{TypeName} =~ /^close/i
            && !$ConfigObject->Get('Ticket::Frontend::RedirectAfterCloseDisabled')
            )
        {
            return $LayoutObject->PopupClose(
                URL => ( $Self->{LastScreenOverview} || 'Action=AgentDashboard' ),
            );
        }

        return $LayoutObject->PopupClose(
            URL => "Action=AgentTicketZoom;TicketID=$Self->{TicketID};ArticleID=$ArticleID",
        );
    }

    # check for SMIME / PGP if customer has changed
    elsif ( $Self->{Subaction} eq 'AJAXUpdate' ) {

        my @ExtendedData;

        # run compose modules
        if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {

            # use ticket QueueID in compose modules
            $GetParam{QueueID} = $Ticket{QueueID};

            my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
            JOB:
            for my $Job ( sort keys %Jobs ) {

                # load module
                next JOB if !$MainObject->Require( $Jobs{$Job}->{Module} );

                my $Object = $Jobs{$Job}->{Module}->new(
                    %{$Self},
                    Debug => $Self->{Debug},
                );

                my $Multiple;

                # get params
                PARAMETER:
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                    if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                        @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                        $Multiple = 1;
                        next PARAMETER;
                    }

                    $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
                }

                # run module
                my %Data = $Object->Data( %GetParam, Config => $Jobs{$Job} );

                # get AJAX param values
                if ( $Object->can('GetParamAJAX') ) {
                    %GetParam = ( %GetParam, $Object->GetParamAJAX(%GetParam) );
                }

                # get options that have to be removed from the selection visible
                # to the agent. These options will be added again on submit.
                if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                    my @OptionsToRemove = $Object->GetOptionsToRemoveAJAX(%GetParam);

                    for my $OptionToRemove (@OptionsToRemove) {
                        delete $Data{$OptionToRemove};
                    }
                }

                my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );
                if ($Key) {
                    push @ExtendedData, {
                        Name         => $Key,
                        Data         => \%Data,
                        SelectedID   => $GetParam{$Key},
                        Translation  => 1,
                        PossibleNone => 1,
                        Multiple     => $Multiple,
                        Max          => 150,
                    };
                }
            }
        }

        my $NextStates = $Self->_GetNextStates(
            %GetParam,
        );

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

        # cycle trough the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( @{ $Self->{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       => $Self->{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 $JSON = $LayoutObject->BuildSelectionJSON(
            [
                @ExtendedData,
                {
                    Name         => 'StateID',
                    Data         => $NextStates,
                    SelectedID   => $GetParam{StateID},
                    Translation  => 1,
                    PossibleNone => 1,
                    Max          => 100,
                },
                @DynamicFieldAJAX,
            ],
        );

        # can't use JSONReply here, as we already have JSON
        return $LayoutObject->Attachment(
            ContentType => 'application/json',
            Content     => $JSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }
    else {
        my $Output = $LayoutObject->Header(
            Value     => $Ticket{TicketNumber},
            Type      => 'Small',
            BodyClass => 'Popup',
        );

        # get std attachment object
        my $StdAttachmentObject = $Kernel::OM->Get('Kernel::System::StdAttachment');

        # handle response template attachments and state
        if ( $GetParam{ResponseID} ) {

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

            # set response preselected state
            my %Response = $Kernel::OM->Get('Kernel::System::ResponseTemplatesStatePreselection')->StandardTemplateGet(
                ID => $GetParam{ResponseID},
            );
            $GetParam{StateID} ||= $Response{PreSelectedTicketStateID};
        }

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

        # get last customer article or selected article
        my @ArticleList;

        if ( $GetParam{ArticleID} ) {

            @ArticleList = $ArticleObject->ArticleList(
                TicketID  => $Self->{TicketID},
                ArticleID => $GetParam{ArticleID},
            );
        }
        else {

            @ArticleList = $ArticleObject->ArticleList(
                TicketID             => $Self->{TicketID},
                IsVisibleForCustomer => 1,
                OnlyLast             => 1,
            );
        }

        my %Data;

        ARTICLEMETADATA:
        for my $ArticleMetaData (@ArticleList) {

            next ARTICLEMETADATA if !$ArticleMetaData;
            next ARTICLEMETADATA if !IsHashRefWithData($ArticleMetaData);

            my $CurrentArticleBackendObject = $ArticleObject->BackendForArticle( %{$ArticleMetaData} );

            %Data = $CurrentArticleBackendObject->ArticleGet(
                TicketID  => $Self->{TicketID},
                ArticleID => $ArticleMetaData->{ArticleID},
            );

            $Data{CommunicationChannelName} = $CurrentArticleBackendObject->ChannelNameGet();

            last ARTICLEMETADATA;
        }

        # Merge ticket data with article data, see bug#13995 (https://bugs.otrs.org/show_bug.cgi?id=13995).
        %Data = ( %Ticket, %Data );

        # If article is not a MIMEBase article, get customer recipients from the backend.
        if ( !$Data{To} && !$Data{From} ) {
            my @CustomerUserIDs = $LayoutObject->ArticleCustomerRecipientsGet(
                TicketID  => $Self->{TicketID},
                ArticleID => $Data{ArticleID},
                UserID    => $Self->{UserID},
            );

            my $CustomerUserObject = $Kernel::OM->Get('Kernel::System::CustomerUser');

            my @CustomerRecipients;

            CUSTOMER_USER_ID:
            for my $CustomerUserID (@CustomerUserIDs) {
                my %Customer = $CustomerUserObject->CustomerUserDataGet(
                    User => $CustomerUserID,
                );
                next CUSTOMER_USER_ID if !%Customer;

                push @CustomerRecipients, "\"$Customer{UserFirstname} $Customer{UserLastname}\" <$Customer{UserEmail}>";
            }

            $Data{To} = join( ',', @CustomerRecipients ) // '';

            # Include sender name in 'From' field for correct quoting.
            my %ArticleFields = $LayoutObject->ArticleFields(
                TicketID  => $Self->{TicketID},
                ArticleID => $Data{ArticleID},
            );
            $Data{From} = $ArticleFields{Sender}->{Value} // '';
        }

        # set OrigFrom for correct email quoting (xxxx wrote)
        $Data{OrigFrom} = $Data{From};

        # check article type and replace To with From (in case)
        if ( $Data{SenderType} !~ /customer/ ) {

            # replace From/To, To/From because sender is agent
            my $To = $Data{To};
            $Data{To}   = $Data{From};
            $Data{From} = $To;

            $Data{ReplyTo} = '';
        }

        # build OrigFromName (to only use the realname)
        $Data{OrigFromName} = $Data{OrigFrom};
        $Data{OrigFromName} =~ s/<.*>|\(.*\)|\"|;|,//g;
        $Data{OrigFromName} =~ s/( $)|(  $)//g;

        # Fallback to OrigFrom if realname part is empty.
        if ( !$Data{OrigFromName} ) {
            $Data{OrigFromName} = $Data{OrigFrom};
        }

        # get customer data
        my %Customer;
        if ( $Ticket{CustomerUserID} ) {
            %Customer = $Kernel::OM->Get('Kernel::System::CustomerUser')->CustomerUserDataGet(
                User => $Ticket{CustomerUserID}
            );
        }

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

        my %SafetyCheckResult = $Kernel::OM->Get('Kernel::System::HTMLUtils')->Safety(
            String => $Data{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,
        );
        $Data{Body} = $SafetyCheckResult{String};

        # restrict number of body lines if configured
        if (
            $Data{Body}
            && $ConfigObject->Get('Ticket::Frontend::ResponseQuoteMaxLines')
            )
        {
            my $MaxLines = $ConfigObject->Get('Ticket::Frontend::ResponseQuoteMaxLines');

            # split body - one element per line
            my @Body = split /\n/, $Data{Body};

            # only modify if body is longer than allowed
            if ( scalar @Body > $MaxLines ) {

                # splice to max. allowed lines and reassemble
                @Body       = @Body[ 0 .. ( $MaxLines - 1 ) ];
                $Data{Body} = join "\n", @Body;
            }
        }

        if ( $LayoutObject->{BrowserRichText} ) {

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

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

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

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

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

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

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

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

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

            # prepare body, subject, ReplyTo
            # re-wrap body if exists
            if ( $Data{Body} ) {
                $Data{Body} =~ s/\t/ /g;
                my $Quote = $ConfigObject->Get('Ticket::Frontend::Quote');
                if ($Quote) {
                    $Data{Body} =~ s/\n/\n$Quote /g;
                    $Data{Body} = "\n$Quote " . $Data{Body};
                }
                else {
                    $Data{Body} = "\n" . $Data{Body};
                    if ( $Data{CreateTime} ) {
                        $Data{Body} = $LayoutObject->{LanguageObject}->Translate('Date') .
                            ": $Data{CreateTime}\n" . $Data{Body};
                    }

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

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

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

        # get system address object
        my $SystemAddress = $Kernel::OM->Get('Kernel::System::SystemAddress');

        # add not local To addresses to Cc
        for my $Email ( Mail::Address->parse( $Data{To} ) ) {
            my $IsLocal = $SystemAddress->SystemAddressIsLocalAddress(
                Address => $Email->address(),
# Rother OSS / DiscreteSystemAddresses
                TicketID => $Self->{TicketID},
# EO DiscreteSystemAddresses
            );
            if ( !$IsLocal ) {
                if ( $Data{Cc} ) {
                    $Data{Cc} .= ', ';
                }
                $Data{Cc} .= $Email->format();
            }
        }

        # check ReplyTo
        if ( $Data{ReplyTo} ) {
            $Data{To} = $Data{ReplyTo};
        }
        else {
            $Data{To} = $Data{From};

            # try to remove some wrong text to from line (by way of ...)
            # added by some strange mail programs on bounce
            $Data{To} =~ s/(.+?\<.+?\@.+?\>)\s+\(by\s+way\s+of\s+.+?\)/$1/ig;
        }

        # get to email (just "some@example.com")
        for my $Email ( Mail::Address->parse( $Data{To} ) ) {
            $Data{ToEmail} = $Email->address();
        }

        # only reply to sender
        if ( !$GetParam{ReplyAll} ) {
            $Data{Cc}  = '';
            $Data{Bcc} = '';
        }

        # use customer database email
        # do not add customer email to cc, if article is visible for customer
        if (
            $ConfigObject->Get('Ticket::Frontend::ComposeAddCustomerAddress')
            && $Data{IsVisibleForCustomer}
            )
        {
            # check if customer is in recipient list
            if ( $Customer{UserEmail} && $Data{ToEmail} !~ /^\Q$Customer{UserEmail}\E$/i ) {

                if ( $Data{SenderType} eq 'agent' && !$Data{IsVisibleForCustomer} ) {
                    if ( $Data{To} ) {
                        $Data{To} .= ', ' . $Customer{UserEmail};
                    }
                    else {
                        $Data{To} = $Customer{UserEmail};
                    }
                }

                # replace To with customers database address
                elsif ( $ConfigObject->Get('Ticket::Frontend::ComposeReplaceSenderAddress') ) {

                    $Output .= $LayoutObject->Notify(
                        Data => $LayoutObject->{LanguageObject}->Translate(
                            'Address %s replaced with registered customer address.',
                            $Data{ToEmail},
                        ),

                    );
                    $Data{To} = $Customer{UserEmail};
                }

                # add customers database address to Cc
                else {
                    $Output .= $LayoutObject->Notify(
                        Info => Translatable("Customer user automatically added in Cc."),
                    );
                    if ( $Data{Cc} ) {
                        $Data{Cc} .= ', ' . $Customer{UserEmail};
                    }
                    else {
                        $Data{Cc} = $Customer{UserEmail};
                    }
                }
            }
        }

        # find duplicate addresses
        my %Recipient;
        for my $Type (qw(To Cc Bcc)) {
            if ( $Data{$Type} ) {
                my $NewLine = '';
                for my $Email ( Mail::Address->parse( $Data{$Type} ) ) {
                    my $Address = lc $Email->address();

                    # only use email addresses with @ inside
                    if ( $Address && $Address =~ /@/ && !$Recipient{$Address} ) {
                        $Recipient{$Address} = 1;
                        my $IsLocal = $SystemAddress->SystemAddressIsLocalAddress(
                            Address => $Address,
# Rother OSS / DiscreteSystemAddresses
                            TicketID => $Self->{TicketID},
# EO DiscreteSystemAddresses
                        );
                        if ( !$IsLocal ) {
                            if ($NewLine) {
                                $NewLine .= ', ';
                            }
                            $NewLine .= $Email->format();
                        }
                    }
                }
                $Data{$Type} = $NewLine;
            }
        }

        # get template
        my $TemplateGenerator = $Kernel::OM->Get('Kernel::System::TemplateGenerator');

        # use key StdResponse to pass the data to the template for legacy reasons,
        #   because existing systems may have it in their configuration as that was
        #   the key used before the internal switch to StandardResponse And StandardTemplate

        $Data{StdResponse} = $TemplateGenerator->Template(
            TicketID   => $Self->{TicketID},
            ArticleID  => $GetParam{ArticleID},
            TemplateID => $GetParam{ResponseID},
            Data       => \%Data,
            UserID     => $Self->{UserID},
        );

        # get salutation
        $Data{Salutation} = $TemplateGenerator->Salutation(
            TicketID => $Self->{TicketID},
            Data     => \%Data,
            UserID   => $Self->{UserID},
        );

        # get signature
        $Data{Signature} = $TemplateGenerator->Signature(
            TicketID => $Self->{TicketID},
            Data     => \%Data,
            UserID   => $Self->{UserID},
        );

        # $TemplateGenerator->Attributes() does not overwrite %Data, but it adds more keys
        %Data = $TemplateGenerator->Attributes(
            TicketID  => $Self->{TicketID},
            ArticleID => $GetParam{ArticleID},
            Data      => \%Data,
            UserID    => $Self->{UserID},
        );

        my $ResponseFormat = $ConfigObject->Get('Ticket::Frontend::ResponseFormat');

        # make sure body is rich text
        my %DataHTML = %Data;
        if ( $LayoutObject->{BrowserRichText} ) {
            if ($ResponseFormat) {
                $ResponseFormat = $LayoutObject->Ascii2RichText(
                    String => $ResponseFormat,
                );

                # restore qdata formatting for Output replacement
                $ResponseFormat =~ s/&quot;/"/gi;

                # html quote to have it correct in edit area
                $ResponseFormat = $LayoutObject->Ascii2Html(
                    Text => $ResponseFormat,
                );

                # restore qdata formatting for Output replacement
                $ResponseFormat =~ s/&quot;/"/gi;
            }

            # quote all non html content to have it correct in edit area
            KEY:
            for my $Key ( sort keys %DataHTML ) {
                next KEY if !$DataHTML{$Key};
                next KEY if $Key eq 'Salutation';
                next KEY if $Key eq 'Body';
                next KEY if $Key eq 'StdResponse';
                next KEY if $Key eq 'Signature';
                $DataHTML{$Key} = $LayoutObject->Ascii2RichText(
                    String => $DataHTML{$Key},
                );
            }
        }

        # build new response format based on template
        $Data{ResponseFormat} = $LayoutObject->Output(
            Template => $ResponseFormat,
            Data     => { %Param, %DataHTML },
        );

        # check some values
        my %Error;
        LINE:
        for my $Line (qw(To Cc Bcc)) {
            next LINE if !$Data{$Line};
            for my $Email ( Mail::Address->parse( $Data{$Line} ) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $Error{ $Line . "Invalid" } = " ServerError";
                }
            }
        }
        if ( $Data{From} ) {
            for my $Email ( Mail::Address->parse( $Data{From} ) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $Error{"FromInvalid"} .= $CheckItemObject->CheckError();
                }
            }
        }

        # remember dynamic field validation results if erroneous
        my %DynamicFieldPossibleValues;

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

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

            $DynamicFieldPossibleValues{ 'DynamicField_' . $DynamicFieldConfig->{Name} } = $PossibleValuesFilter;

        }

        # build references if exist
        my $References = ( $Data{MessageID} || '' ) . ( $Data{References} || '' );

        # run compose modules
        if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {

            # use ticket QueueID in compose modules
            $GetParam{QueueID} = $Ticket{QueueID};

            my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
            for my $Job ( sort keys %Jobs ) {

                # load module
                if ( !$MainObject->Require( $Jobs{$Job}->{Module} ) ) {
                    return $LayoutObject->FatalError();
                }
                my $Object = $Jobs{$Job}->{Module}->new( %{$Self}, Debug => $Self->{Debug} );

                # get params
                PARAMETER:
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                    if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                        @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                        next PARAMETER;
                    }

                    $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
                }

                # run module
                my $NewParams = $Object->Run( %GetParam, Config => $Jobs{$Job} );

                if ($NewParams) {
                    for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                        $GetParam{$Parameter} = $NewParams;
                    }
                }

                # get errors
                %Error = (
                    %Error,
                    $Object->Error( %GetParam, Config => $Jobs{$Job} ),
                );
            }
        }

        # build view
        $Output .= $Self->_Mask(
            TicketID   => $Self->{TicketID},
            NextStates => $Self->_GetNextStates(
                %GetParam,
            ),
            Attachments         => \@Attachments,
            Errors              => \%Error,
            MultipleCustomer    => \@MultipleCustomer,
            MultipleCustomerCc  => \@MultipleCustomerCc,
            MultipleCustomerBcc => \@MultipleCustomerBcc,
            GetParam            => \%GetParam,
            ResponseID          => $GetParam{ResponseID},
            ReplyArticleID      => $GetParam{ArticleID},
            %Ticket,
            %Data,
            InReplyTo        => $Data{MessageID},
            References       => "$References",
            TicketBackType   => $TicketBackType,
            DFPossibleValues => \%DynamicFieldPossibleValues,
        );
        $Output .= $LayoutObject->Footer(
            Type => 'Small',
        );
        return $Output;
    }
}

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

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

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

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

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

    my %State;
    if ( $Param{GetParam}->{StateID} ) {
        $State{SelectedID} = $Param{GetParam}->{StateID};
    }
    else {
        $State{SelectedValue} //= $Config->{StateDefault};
    }
    $Param{NextStatesStrg} = $LayoutObject->BuildSelection(
        Data         => $Param{NextStates},
        Name         => 'StateID',
        PossibleNone => 1,
        %State,
        %Param,
        Class       => 'Modernize FormUpdate',
        Translation => 1,
    );

    my $IsVisibleForCustomer = $Config->{IsVisibleForCustomerDefault};
    if ( $Param{GetParam}->{IsVisibleForCustomerPresent} ) {
        $IsVisibleForCustomer = $Param{GetParam}->{IsVisibleForCustomer} ? 1 : 0;
    }

    $LayoutObject->Block(
        Name => 'IsVisibleForCustomer',
        Data => {
            IsVisibleForCustomer => $IsVisibleForCustomer,
        },
    );

    # prepare errors!
    if ( $Param{Errors} ) {
        for my $Error ( sort keys %{ $Param{Errors} } ) {
            $Param{$Error} = $LayoutObject->Ascii2Html(
                Text => $Param{Errors}->{$Error},
            );
        }
    }

    # get used calendar
    my $Calendar = $Kernel::OM->Get('Kernel::System::Ticket')->TicketCalendarGet(
        QueueID => $Param{QueueID},
        SLAID   => $Param{SLAID},
    );

    my $QuickDateButtons = $Config->{QuickDateButtons} // $ConfigObject->Get('Ticket::Frontend::DefaultQuickDateButtons');

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

    # Multiple-Autocomplete
    $Param{To} = ( scalar @{ $Param{MultipleCustomer} } ? '' : $Param{To} );
    if ( defined $Param{To} && $Param{To} ne '' ) {
        $Param{ToInvalid} = '';
    }

    $Param{Cc} = ( scalar @{ $Param{MultipleCustomerCc} } ? '' : $Param{Cc} );
    if ( defined $Param{Cc} && $Param{Cc} ne '' ) {
        $Param{CcInvalid} = '';
    }

    # Cc
    my $CustomerCounterCc = 0;
    if ( $Param{MultipleCustomerCc} ) {
        for my $Item ( @{ $Param{MultipleCustomerCc} } ) {
            $LayoutObject->Block(
                Name => 'CcMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Cc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CcCustomerErrorExplantion',
                );
            }
            $CustomerCounterCc++;
        }
    }

    if ( !$CustomerCounterCc ) {
        $Param{CcCustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'CcMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterCc,
        },
    );

    # Bcc
    my $CustomerCounterBcc = 0;
    if ( $Param{MultipleCustomerBcc} ) {
        for my $Item ( @{ $Param{MultipleCustomerBcc} } ) {
            $LayoutObject->Block(
                Name => 'BccMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Bcc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'BccCustomerErrorExplantion',
                );
            }
            $CustomerCounterBcc++;
        }
    }

    if ( !$CustomerCounterBcc ) {
        $Param{BccCustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'BccMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterBcc++,
        },
    );

    # To
    my $CustomerCounter = 0;
    if ( $Param{MultipleCustomer} ) {
        for my $Item ( @{ $Param{MultipleCustomer} } ) {
            $LayoutObject->Block(
                Name => 'MultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CustomerErrorExplantion',
                );
            }
            $CustomerCounter++;
        }
    }

    if ( !$CustomerCounter ) {
        $Param{CustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'MultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounter,
        },
    );

    if ( $Param{ToInvalid} && $Param{Errors} ) {
        $LayoutObject->Block(
            Name => 'ToServerErrorMsg',
        );
    }

    my $CustomerUserObject = $Kernel::OM->Get('Kernel::System::CustomerUser');

    # set preselected values for Cc field
    if ( $Param{Cc} && $Param{Cc} ne '' && !$CustomerCounterCc ) {

        # split Cc values
        my @EmailAddressesCc;
        for my $Email ( Mail::Address->parse( $Param{Cc} ) ) {

            my %CustomerSearch = $CustomerUserObject->CustomerSearch(
                PostMasterSearch => $Email->address(),
                Limit            => 1,
            );

            # CustomerSearch hash could have one item from each backend, so insert just the first one.
            if (%CustomerSearch) {
                CUSTOMERUSERID:
                for my $CustomerUserID ( sort keys %CustomerSearch ) {
                    push @EmailAddressesCc, {
                        CustomerKey        => $CustomerUserID,
                        CustomerTicketText => $CustomerSearch{$CustomerUserID},
                    };
                    last CUSTOMERUSERID;
                }
            }
            else {
                push @EmailAddressesCc, {
                    CustomerKey        => '',
                    CustomerTicketText => $Email->[0] ? "$Email->[0] <$Email->[1]>" : "$Email->[1]",
                };
            }
        }

        $LayoutObject->AddJSData(
            Key   => 'EmailAddressesCc',
            Value => \@EmailAddressesCc,
        );

        $Param{Cc} = '';
    }

    # set preselected values for To field
    if ( defined $Param{To} && $Param{To} ne '' && !$CustomerCounter ) {

        # split To values
        my @EmailAddressesTo;
        for my $Email ( Mail::Address->parse( $Param{To} ) ) {

            my %CustomerSearch = $CustomerUserObject->CustomerSearch(
                PostMasterSearch => $Email->address(),
                Limit            => 1,
            );

            # CustomerSearch hash could have one item from each backend, so insert just the first one.
            if (%CustomerSearch) {
                CUSTOMERUSERID:
                for my $CustomerUserID ( sort keys %CustomerSearch ) {
                    push @EmailAddressesTo, {
                        CustomerKey        => $CustomerUserID,
                        CustomerTicketText => $CustomerSearch{$CustomerUserID},
                    };
                    last CUSTOMERUSERID;
                }
            }
            else {
                push @EmailAddressesTo, {
                    CustomerKey        => '',
                    CustomerTicketText => $Email->[0] ? "$Email->[0] <$Email->[1]>" : "$Email->[1]",
                };
            }
        }

        $LayoutObject->AddJSData(
            Key   => 'EmailAddressesTo',
            Value => \@EmailAddressesTo,
        );

        $Param{To} = '';
    }

    $LayoutObject->Block(
        Name => $Param{TicketBackType},
        Data => {

            #            FormID => $Self->{FormID},
            %Param,
        },
    );

    # render dynamic fields
    {
        my %DynamicFieldConfigs = map { $_->{Name} => $_ } $Self->{DynamicField}->@*;

        # grep dynamic field values from ticket data
        my %DynamicFieldValues
            = map { 'DynamicField_' . $_ => $Param{ 'DynamicField_' . $_ } } grep { $DynamicFieldConfigs{$_}->{ObjectType} eq 'Ticket' } keys %DynamicFieldConfigs;

        $Param{DynamicFieldHTML} = $Kernel::OM->Get('Kernel::Output::HTML::DynamicField::Mask')->EditSectionRender(
            Content              => $Self->{MaskDefinition},
            DynamicFields        => \%DynamicFieldConfigs,
            LayoutObject         => $LayoutObject,
            ParamObject          => $Kernel::OM->Get('Kernel::System::Web::Request'),
            DynamicFieldValues   => \%DynamicFieldValues,
            PossibleValuesFilter => $Param{DFPossibleValues},
            Errors               => $Param{DFErrors},
            Object               => {
                CustomerID     => $Param{CustomerID},
                CustomerUserID => $Param{CustomerUserID},
                UserID         => $Self->{UserID},
                %DynamicFieldValues,
            },
        );

    }

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

    # Show the customer user address book if the module is registered.
    if ( $ConfigObject->Get('Frontend::Module')->{AgentCustomerUserAddressBook} ) {
        $Param{OptionCustomerUserAddressBook} = 1;
    }

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

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

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

    # explanatory message about asterisk
    if ( $ConfigObject->Get('Ticket::Frontend::AsteriskExplanation') ) {
        $LayoutObject->Block(
            Name => 'AsteriskExplanation',
        );
    }

    # create & return output
    return $LayoutObject->Output(
        TemplateFile => 'AgentTicketCompose',
        Data         => {
            FormID         => $Self->{FormID},
            FormDraft      => $Config->{FormDraft},
            FormDraftID    => $Self->{LoadedFormDraftID},
            FormDraftTitle => $LoadedFormDraft ? $LoadedFormDraft->{Title} : '',
            FormDraftMeta  => $LoadedFormDraft,
            %Param,
        },
    );
}

1;
</File>
        <File Location="Custom/Kernel/Modules/AgentTicketEmail.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 9b816e1169e8d7a897d5b78d49ccf36bf5c6daaa - Kernel/Modules/AgentTicketEmail.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::Modules::AgentTicketEmail;

use strict;
use warnings;

use Mail::Address ();

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

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

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

    # get the dynamic fields for this screen
    my $DynamicFieldList = $DynamicFieldObject->DynamicFieldListGet(
        Valid       => 1,
        ObjectType  => [ 'Ticket', 'Article' ],
        FieldFilter => $Config->{DynamicField} || {},
    );

    my $Definition = $Kernel::OM->Get('Kernel::System::Ticket::Mask')->DefinitionGet(
        Mask => $Self->{Action},
    ) || {};

    $Self->{MaskDefinition} = $Definition->{Mask};
    $Self->{DynamicField}   = {};

    # align sysconfig and ticket mask data I
    for my $DynamicField ( @{ $DynamicFieldList // [] } ) {
        if ( exists $Definition->{DynamicFields}{ $DynamicField->{Name} } ) {
            my $Parameters = delete $Definition->{DynamicFields}{ $DynamicField->{Name} } // {};

            for my $Attribute ( keys $Parameters->%* ) {
                $DynamicField->{$Attribute} = $Parameters->{$Attribute};
            }
        }
        else {
            push $Self->{MaskDefinition}->@*, {
                DF        => $DynamicField->{Name},
                Mandatory => $Config->{DynamicField}{ $DynamicField->{Name} } == 2 ? 1 : 0,
            };

            if ( $Config->{DynamicField}{ $DynamicField->{Name} } == 2 ) {
                $DynamicField->{Mandatory} = 1;
            }
        }

        $Self->{DynamicField}{ $DynamicField->{Name} } = $DynamicField;
    }

    # align sysconfig and ticket mask data II
    for my $DynamicFieldName ( keys $Definition->{DynamicFields}->%* ) {
        $Self->{DynamicField}{$DynamicFieldName} = $DynamicFieldObject->DynamicFieldGet(
            Name => $DynamicFieldName,
        );

        my $Parameters = $Definition->{DynamicFields}{$DynamicFieldName} // {};

        for my $Attribute ( keys $Parameters->%* ) {
            $Self->{DynamicField}{$DynamicFieldName}{$Attribute} = $Parameters->{$Attribute};
        }
    }

    # get form id
    $Self->{FormID} = $Kernel::OM->Get('Kernel::System::Web::FormCache')->PrepareFormID(
        ParamObject  => $Kernel::OM->Get('Kernel::System::Web::Request'),
        LayoutObject => $Kernel::OM->Get('Kernel::Output::HTML::Layout'),
    );

    # methods which are used to determine the possible values of the standard fields
    $Self->{FieldMethods} = [
        {
            FieldID => 'Dest',
            Method  => \&_GetTos
        },
        {
            FieldID => 'NewUserID',
            Method  => \&_GetUsers
        },
        {
            FieldID => 'NewResponsibleID',
            Method  => \&_GetResponsibles
        },
        {
            FieldID => 'NextStateID',
            Method  => \&_GetNextStates
        },
        {
            FieldID => 'PriorityID',
            Method  => \&_GetPriorities
        },
        {
            FieldID => 'ServiceID',
            Method  => \&_GetServices
        },
        {
            FieldID => 'SLAID',
            Method  => \&_GetSLAs
        },
        {
            FieldID => 'StandardTemplateID',
            Method  => \&_GetStandardTemplates
        },
        {
            FieldID => 'TypeID',
            Method  => \&_GetTypes
        },
    ];

    # dependencies of standard fields which are not defined via ACLs
    $Self->{InternalDependancy} = {
        Dest => {
            NewUserID          => 1,
            NewResponsibleID   => 1,
            StandardTemplateID => 1,
        },
        ServiceID => {
            SLAID     => 1,
            ServiceID => 1,    #CustomerUser updates can be submitted as ElementChanged: ServiceID
        },
        CustomerUser => {
            ServiceID => 1,
        },
        OwnerAll => {
            NewUserID => 1,
        },
        ResponsibleAll => {
            NewResponsibleID => 1,
        },
    };

    return $Self;
}

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

    # store last queue screen
    if ( $Self->{LastScreenOverview} && $Self->{LastScreenOverview} !~ /Action=AgentTicketEmail/ ) {
        $Kernel::OM->Get('Kernel::System::AuthSession')->UpdateSessionID(
            SessionID => $Self->{SessionID},
            Key       => 'LastScreenOverview',
            Value     => $Self->{RequestedURL},
        );
    }

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

    my $Debug = $Param{Debug} || 0;

    # get params
    my %GetParam;
    for my $Key (
        qw(Year Month Day Hour Minute To Cc Bcc TimeUnits PriorityID Subject Body
        TypeID ServiceID SLAID OwnerAll ResponsibleAll NewResponsibleID NewUserID
        NextStateID StandardTemplateID Dest ArticleID LinkTicketID
        )
        )
    {
        $GetParam{$Key} = $ParamObject->GetParam( Param => $Key );
    }

    # ACL compatibility translation
    my %ACLCompatGetParam;
    $ACLCompatGetParam{OwnerID} = $GetParam{NewUserID};

    # hash for check duplicated entries
    my %AddressesList;

    # MultipleCustomer To-field
    my @MultipleCustomer;
    my $CustomersNumberTo = $ParamObject->GetParam( Param => 'CustomerTicketCounterToCustomer' ) || 0;
    my $Selected          = $ParamObject->GetParam( Param => 'CustomerSelected' )                || '';

    # get check item object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    if ($CustomersNumberTo) {
        my $CustomerCounter = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberTo ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElement = $ParamObject->GetParam( Param => 'CustomerTicketText_' . $Count );

            next COUNT unless $CustomerElement;

            my $CustomerSelected = ( $Selected eq $Count ? 'checked ' : '' );
            my $CustomerKey      = $ParamObject->GetParam( Param => 'CustomerKey_' . $Count ) || '';

            if ( $GetParam{To} ) {
                $GetParam{To} .= ', ' . $CustomerElement;
            }
            else {
                $GetParam{To} = $CustomerElement;
            }

            my $CustomerErrorMsg = 'CustomerGenericServerErrorMsg';
            my $CustomerError    = '';
            my $CustomerDisabled = '';
            my $CountAux         = $CustomerCounter++;

            # check email address
            for my $Email ( Mail::Address->parse($CustomerElement) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) )
                {
                    $CustomerErrorMsg = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerError = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElement} && $CustomerError eq '' ) {
                $CustomerErrorMsg = 'IsDuplicatedServerErrorMsg';
                $CustomerError    = 'ServerError';
            }

            if ( $CustomerError ne '' ) {
                $CustomerDisabled = 'disabled="disabled"';
                $CountAux         = $Count . 'Error';
            }

            push @MultipleCustomer, {
                Count            => $CountAux,
                CustomerElement  => $CustomerElement,
                CustomerSelected => $CustomerSelected,
                CustomerKey      => $CustomerKey,
                CustomerError    => $CustomerError,
                CustomerErrorMsg => $CustomerErrorMsg,
                CustomerDisabled => $CustomerDisabled,
            };
            $AddressesList{$CustomerElement} = 1;
        }
    }

    # MultipleCustomer Cc-field
    my @MultipleCustomerCc;
    my $CustomersNumberCc = $ParamObject->GetParam( Param => 'CustomerTicketCounterCcCustomer' ) || 0;

    if ($CustomersNumberCc) {
        my $CustomerCounterCc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberCc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementCc = $ParamObject->GetParam( Param => 'CcCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementCc;

            my $CustomerKeyCc = $ParamObject->GetParam( Param => 'CcCustomerKey_' . $Count ) || '';

            my $CustomerErrorMsgCc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorCc    = '';
            my $CustomerDisabledCc = '';
            my $CountAuxCc         = $CustomerCounterCc++;

            if ( $GetParam{Cc} ) {
                $GetParam{Cc} .= ', ' . $CustomerElementCc;
            }
            else {
                $GetParam{Cc} = $CustomerElementCc;
            }

            # check email address
            for my $Email ( Mail::Address->parse($CustomerElementCc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) )
                {
                    $CustomerErrorMsgCc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorCc = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElementCc} && $CustomerErrorCc eq '' ) {
                $CustomerErrorMsgCc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorCc    = 'ServerError';
            }

            if ( $CustomerErrorCc ne '' ) {
                $CustomerDisabledCc = 'disabled="disabled"';
                $CountAuxCc         = $Count . 'Error';
            }

            push @MultipleCustomerCc, {
                Count            => $CountAuxCc,
                CustomerElement  => $CustomerElementCc,
                CustomerKey      => $CustomerKeyCc,
                CustomerError    => $CustomerErrorCc,
                CustomerErrorMsg => $CustomerErrorMsgCc,
                CustomerDisabled => $CustomerDisabledCc,
            };
            $AddressesList{$CustomerElementCc} = 1;
        }
    }

    # MultipleCustomer Bcc-field
    my @MultipleCustomerBcc;
    my $CustomersNumberBcc = $ParamObject->GetParam( Param => 'CustomerTicketCounterBccCustomer' ) || 0;

    if ($CustomersNumberBcc) {
        my $CustomerCounterBcc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberBcc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementBcc = $ParamObject->GetParam( Param => 'BccCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementBcc;

            my $CustomerKeyBcc = $ParamObject->GetParam( Param => 'BccCustomerKey_' . $Count ) || '';

            my $CustomerDisabledBcc = '';
            my $CountAuxBcc         = $CustomerCounterBcc++;
            my $CustomerErrorMsgBcc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorBcc    = '';

            if ( $GetParam{Bcc} ) {
                $GetParam{Bcc} .= ', ' . $CustomerElementBcc;
            }
            else {
                $GetParam{Bcc} = $CustomerElementBcc;
            }

            # check email address
            for my $Email ( Mail::Address->parse($CustomerElementBcc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) )
                {
                    $CustomerErrorMsgBcc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorBcc = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElementBcc} && $CustomerErrorBcc eq '' ) {
                $CustomerErrorMsgBcc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorBcc    = 'ServerError';
            }

            if ( $CustomerErrorBcc ne '' ) {
                $CustomerDisabledBcc = 'disabled="disabled"';
                $CountAuxBcc         = $Count . 'Error';
            }

            push @MultipleCustomerBcc, {
                Count            => $CountAuxBcc,
                CustomerElement  => $CustomerElementBcc,
                CustomerKey      => $CustomerKeyBcc,
                CustomerError    => $CustomerErrorBcc,
                CustomerErrorMsg => $CustomerErrorMsgBcc,
                CustomerDisabled => $CustomerDisabledBcc,
            };
            $AddressesList{$CustomerElementBcc} = 1;
        }
    }

    # set an empty value if not defined
    $GetParam{Cc}  = '' if !defined $GetParam{Cc};
    $GetParam{Bcc} = '' if !defined $GetParam{Bcc};

    # get Dynamic fields form ParamObject
    my %DynamicFieldValues;

    # get needed objects
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
    my $LayoutObject              = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ConfigObject              = $Kernel::OM->Get('Kernel::Config');
    my $CustomerUserObject        = $Kernel::OM->Get('Kernel::System::CustomerUser');
    my $UploadCacheObject         = $Kernel::OM->Get('Kernel::System::Web::UploadCache');
    my $TicketObject              = $Kernel::OM->Get('Kernel::System::Ticket');
    my $QueueObject               = $Kernel::OM->Get('Kernel::System::Queue');
    my $FieldRestrictionsObject   = $Kernel::OM->Get('Kernel::System::Ticket::FieldRestrictions');
    my $MainObject                = $Kernel::OM->Get('Kernel::System::Main');

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

# Rother OSS / ITSMCore - calculate Priority via CIP matrix
    my $CIPCalculate = $Config->{PriorityByCIP} // $ConfigObject->Get('ITSM::Frontend::CIPAllocationDefault');
    my $ServiceObject;
    my $CIPAllocateObject;
    if ( $CIPCalculate ) {
        $ServiceObject     = $Kernel::OM->Get('Kernel::System::Service');
        $CIPAllocateObject = $Kernel::OM->Get('Kernel::System::ITSMCIPAllocate');

        $Self->{InternalDependancy}{ServiceID}{PriorityID}                    = 1;
        $Self->{InternalDependancy}{DynamicField_ITSMImpact}{PriorityID}      = 1;
        $Self->{InternalDependancy}{DynamicField_ITSMCriticality}{PriorityID} = 1;
    }
# EO ITSMCore

    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( values $Self->{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 $DynamicField ( sort keys %DynamicFieldValues ) {
        next DYNAMICFIELD if !$DynamicField;
        next DYNAMICFIELD if !$DynamicFieldValues{$DynamicField};

        $DynamicFieldACLParameters{ 'DynamicField_' . $DynamicField } = $DynamicFieldValues{$DynamicField};
    }
    $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,
        );
    }

    if ( !$Self->{Subaction} || $Self->{Subaction} eq 'Created' ) {
        my %Ticket;
        if ( $Self->{TicketID} ) {
            %Ticket = $TicketObject->TicketGet( TicketID => $Self->{TicketID} );
        }

        # header and navigation bar
        my $Output = join '',
            $LayoutObject->Header(),
            $LayoutObject->NavigationBar();

        # if there is no ticket id!
        if ( $Self->{TicketID} && $Self->{Subaction} eq 'Created' ) {

            # notify info
            $Output .= $LayoutObject->Notify(
                Info => $LayoutObject->{LanguageObject}->Translate(
                    'Ticket "%s" created!',
                    $Ticket{TicketNumber},
                ),
                Link => $LayoutObject->{Baselink}
                    . 'Action=AgentTicketZoom;TicketID='
                    . $Ticket{TicketID},
            );
        }

        # store last queue screen
        if (
            $Self->{LastScreenOverview}
            && $Self->{LastScreenOverview} !~ /Action=AgentTicketEmail/
            && $Self->{RequestedURL}       !~ /Action=AgentTicketEmail.*LinkTicketID=/
            )
        {
            $Kernel::OM->Get('Kernel::System::AuthSession')->UpdateSessionID(
                SessionID => $Self->{SessionID},
                Key       => 'LastScreenOverview',
                Value     => $Self->{RequestedURL},
            );
        }

        # get split article if given
        # get ArticleID
        my %Article;
        my %CustomerData;
        my $ArticleFrom = '';
        my %SplitTicketData;
        if ( $GetParam{ArticleID} ) {

            my $Access = $TicketObject->TicketPermission(
                Type     => 'ro',
                TicketID => $Self->{TicketID},
                UserID   => $Self->{UserID}
            );

            if ( !$Access ) {
                return $LayoutObject->NoPermission(
                    Message    => Translatable('You need ro permission!'),
                    WithHeader => 'yes',
                );
            }

            # Get information from original ticket (SplitTicket).
            %SplitTicketData = $TicketObject->TicketGet(
                TicketID      => $Self->{TicketID},
                DynamicFields => 1,
                UserID        => $Self->{UserID},
            );

            my $ArticleBackendObject = $Kernel::OM->Get('Kernel::System::Ticket::Article')->BackendForArticle(
                TicketID  => $Self->{TicketID},
                ArticleID => $GetParam{ArticleID},
            );

            %Article = $ArticleBackendObject->ArticleGet(
                TicketID  => $Self->{TicketID},
                ArticleID => $GetParam{ArticleID},
            );

            # check if article is from the same TicketID as we checked permissions for.
            if ( $Article{TicketID} ne $Self->{TicketID} ) {
                return $LayoutObject->ErrorScreen(
                    Message => $LayoutObject->{LanguageObject}->Translate(
                        'Article does not belong to ticket %s!', $Self->{TicketID}
                    ),
                );
            }

            $Article{Subject} = $TicketObject->TicketSubjectClean(
                TicketNumber => $Ticket{TicketNumber},
                Subject      => $Article{Subject} || '',
            );

            # save article from for addresses list
            $ArticleFrom = $Article{From};

            # if To is present
            # and is no a queue
            # and also is no a system address
            # set To as article from
            if ( IsStringWithData( $Article{To} ) ) {
                my %Queues = $QueueObject->QueueList();

                if ( $ConfigObject->{CustomerPanelOwnSelection} ) {
                    for my $Queue ( sort keys %{ $ConfigObject->{CustomerPanelOwnSelection} } ) {
                        my $Value = $ConfigObject->{CustomerPanelOwnSelection}->{$Queue};
                        $Queues{$Queue} = $Value;
                    }
                }

                my %QueueLookup         = reverse %Queues;
                my %SystemAddressLookup = reverse $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressList();
                my @ArticleFromAddress;
                my $SystemAddressEmail;

                if ($ArticleFrom) {
                    @ArticleFromAddress = Mail::Address->parse($ArticleFrom);
                    $SystemAddressEmail = $ArticleFromAddress[0]->address();
                }

                if ( !defined $QueueLookup{ $Article{To} } && defined $SystemAddressLookup{$SystemAddressEmail} ) {
                    $ArticleFrom = $Article{To};
                }
            }

            # body preparation for plain text processing
            $Article{Body} = $LayoutObject->ArticleQuote(
                TicketID           => $Article{TicketID},
                ArticleID          => $GetParam{ArticleID},
                FormID             => $Self->{FormID},
                UploadCacheObject  => $UploadCacheObject,
                AttachmentsInclude => 1,
            );
            if ( $LayoutObject->{BrowserRichText} ) {
                $Article{ContentType} = 'text/html';
            }
            else {
                $Article{ContentType} = 'text/plain';
            }

            my %SafetyCheckResult = $Kernel::OM->Get('Kernel::System::HTMLUtils')->Safety(
                String => $Article{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,
            );
            $Article{Body} = $SafetyCheckResult{String};

            # show customer info
            if ( $ConfigObject->Get('Ticket::Frontend::CustomerInfoCompose') ) {
                if ( $Article{CustomerUserID} ) {
                    %CustomerData = $CustomerUserObject->CustomerUserDataGet(
                        User => $Article{CustomerUserID},
                    );
                }
                elsif ( $Article{CustomerID} ) {
                    %CustomerData = $CustomerUserObject->CustomerUserDataGet(
                        CustomerID => $Article{CustomerID},
                    );
                }
                elsif ( $SplitTicketData{CustomerUserID} ) {
                    %CustomerData = $CustomerUserObject->CustomerUserDataGet(
                        User => $SplitTicketData{CustomerUserID},
                    );
                }
                elsif ( $SplitTicketData{CustomerID} ) {
                    %CustomerData = $CustomerUserObject->CustomerUserDataGet(
                        CustomerID => $SplitTicketData{CustomerID},
                    );
                }
            }
            if ( $Article{CustomerUserID} ) {
                my %CustomerUserList = $CustomerUserObject->CustomerSearch(
                    UserLogin => $Article{CustomerUserID},
                );
                for my $KeyCustomerUserList ( sort keys %CustomerUserList ) {
                    $Article{From} = $CustomerUserList{$KeyCustomerUserList};
                }
            }
        }

        # multiple addresses list
        # check email address
        my $CountFrom = scalar @MultipleCustomer || 1;
        my %CustomerDataFrom;
        if ( $Article{CustomerUserID} ) {
            %CustomerDataFrom = $CustomerUserObject->CustomerUserDataGet(
                User => $Article{CustomerUserID},
            );
        }

        for my $Email ( Mail::Address->parse($ArticleFrom) ) {

            my $CountAux         = $CountFrom;
            my $CustomerError    = '';
            my $CustomerErrorMsg = 'CustomerGenericServerErrorMsg';
            my $CustomerDisabled = '';
            my $CustomerSelected = $CountFrom eq '1' ? 'checked ' : '';
            my $EmailAddress     = $Email->address();
            if ( !$CheckItemObject->CheckEmail( Address => $EmailAddress ) )
            {
                $CustomerErrorMsg = $CheckItemObject->CheckErrorType()
                    . 'ServerErrorMsg';
                $CustomerError = 'ServerError';
            }

            # check for duplicated entries
            if ( defined $AddressesList{$Email} && $CustomerError eq '' ) {
                $CustomerErrorMsg = 'IsDuplicatedServerErrorMsg';
                $CustomerError    = 'ServerError';
            }

            if ( $CustomerError ne '' ) {
                $CustomerDisabled = 'disabled="disabled"';
                $CountAux         = $CountFrom . 'Error';
            }

            my $Phrase = '';
            if ( $Email->phrase() ) {
                $Phrase = $Email->phrase();
            }

            my $CustomerKey = '';
            if (
                defined $CustomerDataFrom{UserEmail}
                && $CustomerDataFrom{UserEmail} eq $EmailAddress
                )
            {
                $CustomerKey = $Article{CustomerUserID};
            }
            elsif ($EmailAddress) {
                my %List = $CustomerUserObject->CustomerSearch(
                    PostMasterSearch => $EmailAddress,
                );

                for my $UserLogin ( sort keys %List ) {

                    # Set right one if there is more than one customer user with the same email address.
                    if ( $Phrase && $List{$UserLogin} =~ /$Phrase/ ) {
                        $CustomerKey = $UserLogin;
                    }
                }
            }

            my $CustomerElement = $EmailAddress;
            if ($Phrase) {
                $CustomerElement = $Phrase . " <$EmailAddress>";
            }

            if ( $CustomerSelected && $CustomerKey ) {
                %CustomerData = $CustomerUserObject->CustomerUserDataGet(
                    User => $CustomerKey,
                );
            }

            push @MultipleCustomer, {
                Count            => $CountAux,
                CustomerElement  => $CustomerElement,
                CustomerSelected => $CustomerSelected,
                CustomerKey      => $CustomerKey,
                CustomerError    => $CustomerError,
                CustomerErrorMsg => $CustomerErrorMsg,
                CustomerDisabled => $CustomerDisabled,
            };
            $AddressesList{$EmailAddress} = 1;
            $CountFrom++;
        }

        # get user preferences
        my %UserPreferences = $Kernel::OM->Get('Kernel::System::User')->GetUserData(
            UserID => $Self->{UserID},
        );

        my %SplitTicketParam;

        # in case of split a TicketID and ArticleID are always given, send the TicketID to calculate
        # ACLs based on parent information
        if ( $Self->{TicketID} && $Article{ArticleID} ) {
            $SplitTicketParam{TicketID} = $Self->{TicketID};
        }

        # fix to bug# 8068 Field & DynamicField preselection on TicketSplit
        # when splitting a ticket the selected attributes must remain in the new ticket screen
        # this information will be available in the SplitTicketParam hash
        if ( $SplitTicketParam{TicketID} ) {

            # get information from original ticket (SplitTicket)
            my %SplitTicketData = $TicketObject->TicketGet(
                TicketID      => $SplitTicketParam{TicketID},
                DynamicFields => 1,
                UserID        => $Self->{UserID},
            );

            # set simple IDs to pass them to the mask
            for my $SplitParam (qw(TypeID ServiceID SLAID PriorityID)) {
                $SplitTicketParam{$SplitParam} = $SplitTicketData{$SplitParam};
            }

            # set StateID as NextStateID
            $SplitTicketParam{NextStateID} = $SplitTicketData{StateID};

            # set Owner and Responsible
            $SplitTicketParam{UserSelected}            = $SplitTicketData{OwnerID};
            $SplitTicketParam{ResponsibleUserSelected} = $SplitTicketData{ResponsibleID};

            # set additional information needed for Owner and Responsible
            if ( $SplitTicketData{QueueID} ) {
                $SplitTicketParam{QueueID} = $SplitTicketData{QueueID};
            }

            $GetParam{OwnerAll}       = 1;
            $GetParam{ResponsibleAll} = 1;

            # set the selected queue in format ID||Name
            $SplitTicketParam{FromSelected} = $SplitTicketData{QueueID} . '||' . $SplitTicketData{Queue};

            for my $Key ( sort keys %SplitTicketData ) {
                if ( $Key =~ /DynamicField\_(.*)/ ) {
                    $SplitTicketParam{DynamicField}{$1} = $SplitTicketData{$Key};
                    delete $SplitTicketParam{$Key};
                }
            }
        }

        # define selected values at the start

        # in case of ticket split set $Self->{QueueID} as the QueueID of the original ticket,
        # in order to set correct ACLs on page load (initial). See bug 8687.
        if (
            IsHashRefWithData( \%SplitTicketParam )
            && $SplitTicketParam{QueueID}
            && !$Self->{QueueID}
            )
        {
            $GetParam{QueueID} = $SplitTicketParam{QueueID};
            $GetParam{Dest}    = $SplitTicketParam{FromSelected};
        }

        # Get predefined QueueID (if no queue from split ticket is set).
        elsif ( !$Self->{QueueID} && $GetParam{Dest} ) {
            my @QueueParts = split( /\|\|/, $GetParam{Dest} );
            $GetParam{QueueID} = $QueueParts[0];
        }

        else {
            my $UserDefaultQueue = $ConfigObject->Get('Ticket::Frontend::UserDefaultQueue') || '';

            if ($UserDefaultQueue) {
                $GetParam{QueueID} = $Kernel::OM->Get('Kernel::System::Queue')->QueueLookup( Queue => $UserDefaultQueue );
                if ( $GetParam{QueueID} ) {
                    $GetParam{Dest} = "$GetParam{QueueID}||$UserDefaultQueue";
                }
            }
        }

        # don't use the split ticket state, just the default
        if ( $Config->{StateDefault} ) {
            my %NextStates;

            %NextStates = $Kernel::OM->Get('Kernel::System::Ticket')->TicketStateList(
                %GetParam,
                QueueID => $GetParam{QueueID} || 1,
                Action  => $Self->{Action},
                UserID  => $Self->{UserID},
            );

            $GetParam{NextStateID} = { reverse %NextStates }->{ $Config->{StateDefault} } // '';
        }

        # split ticket info for the rest
        if ( $SplitTicketParam{UserSelected} ) {
            $GetParam{NewUserID} = $SplitTicketData{OwnerID};
        }
        if ( $SplitTicketParam{ResponsibleUserSelected} ) {
            $GetParam{NewResponsibleID} = $SplitTicketData{ResponsibleUserSelected};
        }
        for my $SplitParam (qw(TypeID ServiceID SLAID PriorityID)) {
            $SplitTicketParam{$SplitParam} = $SplitTicketData{$SplitParam};
        }

        # cycle trough the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( values $Self->{DynamicField}->%* ) {
            next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);
            next DYNAMICFIELD if !IsHashRefWithData( $DynamicFieldConfig->{Config} );
            next DYNAMICFIELD if !$DynamicFieldConfig->{Name};

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

            # in case of split a TicketID and ArticleID are always given, Get the value
            # from DB this cases
            if ( $Self->{TicketID} && $Article{ArticleID} ) {

                # select TicketID or ArticleID to get the value depending on dynamic field configuration
                my $ObjectID = $DynamicFieldConfig->{ObjectType} eq 'Ticket'
                    ? $Self->{TicketID}
                    : $Article{ArticleID};

                # get value stored on the database (split)
                $Value = $DynamicFieldBackendObject->ValueGet(
                    DynamicFieldConfig => $DynamicFieldConfig,
                    ObjectID           => $ObjectID,
                );
            }

            # otherwise (on a new ticket). Check if the user has a user specific default value for
            # the dynamic field, otherwise will use Dynamic Field default value
            elsif ( !$DynamicFieldConfig->{Readonly} ) {

                # get default value from dynamic field config (if any)
                $Value = $DynamicFieldConfig->{Config}->{DefaultValue} || '';

                # override the value from user preferences if is set
                if ( $UserPreferences{ 'UserDynamicField_' . $DynamicFieldConfig->{Name} } ) {
                    $Value = $UserPreferences{ 'UserDynamicField_' . $DynamicFieldConfig->{Name} };
                }
            }

            $GetParam{DynamicField}{ 'DynamicField_' . $DynamicFieldConfig->{Name} } = $Value;
        }

        my $Autoselect = $ConfigObject->Get('TicketACL::Autoselect') || undef;

        # gather fields which are supposed to be hidden when autoselected
        my $HideAutoselectedJSON;
        if ($Autoselect) {
            my @HideAutoselected = grep { !ref( $Autoselect->{$_} ) && $Autoselect->{$_} == 2 } keys %{$Autoselect};
            if ( $Autoselect->{DynamicField} ) {
                push @HideAutoselected,
                    map { "DynamicField_" . $_ }
                    ( grep { $Autoselect->{DynamicField}{$_} == 2 } keys %{ $Autoselect->{DynamicField} } );
            }

            if (@HideAutoselected) {
                my $JSONObject = $Kernel::OM->Get('Kernel::System::JSON');
                $HideAutoselectedJSON = $JSONObject->Encode(
                    Data => \@HideAutoselected,
                );
            }
        }

        # track changing standard fields
        my $ACLPreselection;
        if ( $ConfigObject->Get('TicketACL::ACLPreselection') ) {

            # get cached preselection rules
            my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');
            $ACLPreselection = $CacheObject->Get(
                Type => 'TicketACL',
                Key  => 'Preselection',
            );
            if ( !$ACLPreselection ) {
                $ACLPreselection = $FieldRestrictionsObject->SetACLPreselectionCache();
            }
        }

        my %Convergence = (
            StdFields => 0,
            Fields    => 0,
        );
        my %ChangedElements;
        my %ChangedElementsDFStart;
        my %ChangedStdFields;

# Rother OSS / ITSMCore - calculate Priority via CIP matrix
        if ( $CIPCalculate && $GetParam{DynamicField}{DynamicField_ITSMImpact}
            && ( $GetParam{DynamicField}{DynamicField_ITSMCriticality} || $GetParam{ServiceID} ) ) {

            # if we have an initial criticality and impact, trigger priority calculation
            $ChangedElements{DynamicField_ITSMImpact} = 1;
        }
# EO ITSMCore

        my $LoopProtection = 100;
        my %StdFieldValues;
        my %DynFieldStates = (
            Visibility => {},
            Fields     => {},
        );

        my $InitialRun = 1;

        until ( $Convergence{Fields} ) {

            # determine standard field input
            until ( $Convergence{StdFields} ) {

                my %NewChangedElements;

                # which standard fields to check - FieldID => GetParamValue (necessary for Dest)
                my %Check = (
                    Dest               => 'QueueID',
                    NewUserID          => 'NewUserID',
                    NewResponsibleID   => 'NewResponsibleID',
                    NextStateID        => 'NextStateID',
                    PriorityID         => 'PriorityID',
                    ServiceID          => 'ServiceID',
                    SLAID              => 'SLAID',
                    StandardTemplateID => 'StandardTemplateID',
                    TypeID             => 'TypeID',
                );
                if ( $ACLPreselection && !$InitialRun ) {
                    FIELD:
                    for my $FieldID ( sort keys %Check ) {
                        if ( !$ACLPreselection->{Fields}{$FieldID} ) {
                            $Kernel::OM->Get('Kernel::System::Log')->Log(
                                Priority => 'debug',
                                Message  => "$FieldID not defined in TicketACL preselection rules!"
                            );
                            next FIELD;
                        }
                        if ( $Autoselect && $Autoselect->{$FieldID} && $ChangedElements{$FieldID} ) {
                            next FIELD;
                        }
                        for my $Element ( sort keys %ChangedElements ) {
                            if (
                                $ACLPreselection->{Rules}{Ticket}{$Element}{$FieldID}
                                || $Self->{InternalDependancy}{$Element}{$FieldID}
                                )
                            {
                                next FIELD;
                            }
                            if ( !$ACLPreselection->{Fields}{$Element} ) {
                                $Kernel::OM->Get('Kernel::System::Log')->Log(
                                    Priority => 'debug',
                                    Message  => "$Element not defined in TicketACL preselection rules!"
                                );
                                next FIELD;
                            }
                        }

                        # delete unaffected fields
                        delete $Check{$FieldID};
                    }
                }

                # for each standard field which has to be checked, run the defined method
                METHOD:
                for my $Field ( @{ $Self->{FieldMethods} } ) {
                    next METHOD if !$Check{ $Field->{FieldID} };

                    # use $Check{ $Field->{FieldID} } for Dest=>QueueID
                    $StdFieldValues{ $Check{ $Field->{FieldID} } } = $Field->{Method}->(
                        $Self,
                        %GetParam,
                        OwnerID        => $GetParam{NewUserID},
                        CustomerUserID => $SplitTicketData{CustomerUserID} || '',
                        QueueID        => $GetParam{QueueID},
                        Services       => $StdFieldValues{ServiceID} || undef,      # needed for SLAID
                    );

                    # special stuff for QueueID/Dest: Dest is "QueueID||QueueName" => "QueueName";
                    if ( $Field->{FieldID} eq 'Dest' ) {
                        TOs:
                        for my $QueueID ( sort keys %{ $StdFieldValues{QueueID} } ) {
                            next TOs if ( $StdFieldValues{QueueID}{$QueueID} eq '-' );
                            $StdFieldValues{Dest}{"$QueueID||$StdFieldValues{QueueID}{ $QueueID }"} = $StdFieldValues{QueueID}{$QueueID};
                        }

                        # check current selection of QueueID (Dest will be done together with the other fields)
                        if ( $GetParam{QueueID} && !$StdFieldValues{Dest}{ $GetParam{Dest} } ) {
                            $GetParam{QueueID} = '';
                        }

                        # autoselect
                        if ( !$GetParam{QueueID} && $Autoselect && $Autoselect->{Dest} ) {
                            $GetParam{QueueID} = $FieldRestrictionsObject->Autoselect(
                                PossibleValues => $StdFieldValues{QueueID},
                            ) || '';
                        }
                    }

# Rother OSS / ITSMCore - calculate Priority via CIP matrix
                    elsif ( $Field->{FieldID} eq 'PriorityID' && $CIPCalculate && $GetParam{DynamicField}{DynamicField_ITSMImpact} ) {

                        # get the criticality either from the manually set dynamic field, or the service
                        my $Criticality = $GetParam{DynamicField_ITSMCriticality};

                        if ( !$Criticality && $GetParam{ServiceID} ) {
                            my %Service = $ServiceObject->ServiceGet(
                                ServiceID => $GetParam{ServiceID},
                                UserID    => 1,
                            );

                            $Criticality = $Service{Criticality};
                        }

                        if ( $Criticality ) {

                            # recalculate the priority if any of the impacting elements changed
                            if ( $ChangedElements{DynamicField_ITSMImpact} || $ChangedElements{ServiceID} || $ChangedElements{DynamicField_ITSMCriticality} ) {
                                my $PriorityID = $CIPAllocateObject->PriorityAllocationGet(
                                    Criticality => $Criticality,
                                    Impact      => $GetParam{DynamicField}{DynamicField_ITSMImpact},
                                );

                                if ( $StdFieldValues{PriorityID}{ $PriorityID } && $PriorityID ne $GetParam{PriorityID} ) {
                                    $GetParam{PriorityID}           = $PriorityID;
                                    $NewChangedElements{PriorityID} = 1;
                                    $ChangedStdFields{PriorityID}   = 1;
                                }
                            }

                            # if we enforce CIPAllocation in any case delete all other priority options
                            if ( $GetParam{PriorityID} && $CIPCalculate == 2 ) {
                                $StdFieldValues{PriorityID} = {
                                    $GetParam{PriorityID} => $StdFieldValues{PriorityID}{ $GetParam{PriorityID} },
                                };
                            }
                        }
                    }
# EO ITSMCore

                    # check whether current selected value is still valid for the field
                    if (
                        $GetParam{ $Field->{FieldID} }
                        && !$StdFieldValues{ $Field->{FieldID} }{ $GetParam{ $Field->{FieldID} } }
                        )
                    {
                        # if not empty the field
                        $GetParam{ $Field->{FieldID} }           = '';
                        $NewChangedElements{ $Field->{FieldID} } = 1;
                        $ChangedStdFields{ $Field->{FieldID} }   = 1;
                    }

                    # autoselect
                    if ( !$GetParam{ $Field->{FieldID} } && $Autoselect && $Autoselect->{ $Field->{FieldID} } ) {
                        $GetParam{ $Field->{FieldID} } = $FieldRestrictionsObject->Autoselect(
                            PossibleValues => $StdFieldValues{ $Field->{FieldID} },
                        ) || '';
                        if ( $GetParam{ $Field->{FieldID} } ) {
                            $NewChangedElements{ $Field->{FieldID} } = 1;
                            $ChangedStdFields{ $Field->{FieldID} }   = 1;
                        }
                    }
                }

                if ( !%NewChangedElements ) {
                    $Convergence{StdFields} = 1;
                }
                else {
                    %ChangedElements = %NewChangedElements;
                }

                %ChangedElementsDFStart = (
                    %ChangedElementsDFStart,
                    %NewChangedElements,
                );

                if ( $LoopProtection-- < 1 ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "Ran into unresolvable loop!",
                    );
                    return;
                }

            }

            %ChangedElements        = %ChangedElementsDFStart;
            %ChangedElementsDFStart = ();

            # check dynamic fields
            my %CurFieldStates;
            if ( %ChangedElements || $InitialRun ) {

                # get values and visibility of dynamic fields
                %CurFieldStates = $FieldRestrictionsObject->GetFieldStates(
                    TicketObject              => $TicketObject,
                    DynamicFields             => $Self->{DynamicField},
                    DynamicFieldBackendObject => $DynamicFieldBackendObject,
                    ChangedElements           => \%ChangedElements,                        # optional to reduce ACL evaluation
                    Action                    => $Self->{Action},
                    UserID                    => $Self->{UserID},
                    FormID                    => $Self->{FormID},
                    CustomerUser              => $SplitTicketData{CustomerUserID} || '',
                    GetParam                  => {
                        %GetParam,
                        OwnerID => $GetParam{NewUserID},
                    },
                    Autoselect      => $Autoselect,
                    ACLPreselection => $ACLPreselection,
                    LoopProtection  => \$LoopProtection,
                );

                # combine FieldStates
                $DynFieldStates{Fields} = {
                    %{ $DynFieldStates{Fields} },
                    %{ $CurFieldStates{Fields} },
                };
                $DynFieldStates{Visibility} = {
                    %{ $DynFieldStates{Visibility} },
                    %{ $CurFieldStates{Visibility} },
                };

                # store new values
                $GetParam{DynamicField} = {
                    %{ $GetParam{DynamicField} },
                    %{ $CurFieldStates{NewValues} },
                };
            }

            # if dynamic fields changed, check standard fields again
            if ( %CurFieldStates && IsHashRefWithData( $CurFieldStates{NewValues} ) ) {
                $Convergence{StdFields} = 0;
                %ChangedElements = map { $_ => 1 } keys %{ $CurFieldStates{NewValues} };
            }
            else {
                $Convergence{Fields} = 1;
            }

            $InitialRun = 0;
        }

        my %DynamicFieldPossibleValues = map {
            'DynamicField_' . $_ => defined $DynFieldStates{Fields}{$_}
                ? $DynFieldStates{Fields}{$_}{PossibleValues}
                : undef
        } ( keys $Self->{DynamicField}->%* );

        # run compose modules
        if (
            ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq
            'HASH'
            )
        {

            if ( $GetParam{Dest} && $GetParam{Dest} =~ /^(\d{1,100})\|\|.+?$/ ) {
                $GetParam{QueueID} = $1;
                my %Queue = $QueueObject->GetSystemAddress( QueueID => $GetParam{QueueID} );
                $GetParam{From} = $Queue{Email};
            }

            my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
            for my $Job ( sort keys %Jobs ) {

                # load module
                if ( !$MainObject->Require( $Jobs{$Job}->{Module} ) ) {
                    return $LayoutObject->FatalError();
                }

                my $Object = $Jobs{$Job}->{Module}->new(
                    %{$Self},
                    Debug => $Debug,
                );

                # get params
                PARAMETER:
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                    if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                        @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                        next PARAMETER;
                    }

                    $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
                }

                # run module
                my $NewParams = $Object->Run( %GetParam, Config => $Jobs{$Job} );

                if ($NewParams) {
                    for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                        $GetParam{$Parameter} = $NewParams;
                    }
                }
            }
        }

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

        # get and format default subject and body
        my $Subject = $Article{Subject};
        if ( !$Subject ) {
            $Subject = $LayoutObject->Output(
                Template => $Config->{Subject} || '',
            );
        }
        my $Body = $Article{Body} || '';
        if ( !$Body ) {
            $Body = $LayoutObject->Output(
                Template => $Config->{Body} || '',
            );
        }

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

        $Output .= $Self->_MaskEmailNew(
            %GetParam,
            NextState               => $GetParam{NextStateID} ? $StdFieldValues{NextStateID}{ $GetParam{NextStateID} } : '',
            FromSelected            => $GetParam{Dest},
            UserSelected            => $GetParam{NewUserID},
            ResponsibleUserSelected => $GetParam{NewResponsibleID},
            NextStates              => $StdFieldValues{NextStateID},
            Priorities              => $StdFieldValues{PriorityID},
            Types                   => $StdFieldValues{TypeID},
            Services                => $StdFieldValues{ServiceID},
            SLAs                    => $StdFieldValues{SLAID},
            StandardTemplates       => $StdFieldValues{StandardTemplateID},
            Users                   => $StdFieldValues{NewUserID},
            ResponsibleUsers        => $StdFieldValues{NewResponsibleID},
            FromList                => $StdFieldValues{QueueID},
            To                      => $Article{From} // '',
            Subject                 => $Subject,
            Body                    => $Body,
            CustomerUser            => $SplitTicketData{CustomerUserID},
            CustomerID              => $SplitTicketData{CustomerID},
            CustomerData            => \%CustomerData,
            Attachments             => \@Attachments,
            LinkTicketID            => $GetParam{LinkTicketID} || '',
            HideAutoselected        => $HideAutoselectedJSON,
            Visibility              => $DynFieldStates{Visibility},
            DFPossibleValues        => \%DynamicFieldPossibleValues,
            TimeUnits               => $Self->_GetTimeUnits(
                %GetParam,
                %ACLCompatGetParam,
                %SplitTicketParam,
                ArticleID => $Article{ArticleID},
            ),
            TimeUnitsRequired => (
                $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
                ? 'Validate_Required'
                : ''
            ),
            MultipleCustomer    => \@MultipleCustomer,
            MultipleCustomerCc  => \@MultipleCustomerCc,
            MultipleCustomerBcc => \@MultipleCustomerBcc,
        );

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

        return $Output;
    }

    # deliver signature
    elsif ( $Self->{Subaction} eq 'Signature' ) {
        my $CustomerUser = $ParamObject->GetParam( Param => 'SelectedCustomerUser' ) || '';
        my $QueueID      = $ParamObject->GetParam( Param => 'QueueID' );
        if ( !$QueueID ) {
            my $Dest = $ParamObject->GetParam( Param => 'Dest' ) || '';
            ($QueueID) = split( /\|\|/, $Dest );
        }

        # start with empty signature (no queue selected) - if we have a queue, get the sig.
        my $Signature = '';
        if ($QueueID) {
            $Signature = $Self->_GetSignature(
                QueueID        => $QueueID,
                CustomerUserID => $CustomerUser,
            );
        }
        my $MimeType = 'text/plain';
        if ( $LayoutObject->{BrowserRichText} ) {
            $MimeType  = 'text/html';
            $Signature = $LayoutObject->RichTextDocumentComplete(
                String => $Signature,
            );
        }

        return $LayoutObject->Attachment(
            ContentType => $MimeType . '; charset=' . $LayoutObject->{Charset},
            Content     => $Signature,
            Type        => 'inline',
            NoCache     => 1,
        );
    }

    # create new ticket and article
    elsif ( $Self->{Subaction} eq 'StoreNew' ) {

        my %Error;
        my $NextStateID = $ParamObject->GetParam( Param => 'NextStateID' ) || '';
        my %StateData;
        if ($NextStateID) {
            %StateData = $Kernel::OM->Get('Kernel::System::State')->StateGet(
                ID => $NextStateID,
            );
        }
        my $NextState        = $StateData{Name};
        my $NewResponsibleID = $ParamObject->GetParam( Param => 'NewResponsibleID' ) || '';
        my $NewUserID        = $ParamObject->GetParam( Param => 'NewUserID' )        || '';
        my $Dest             = $ParamObject->GetParam( Param => 'Dest' )             || '';

        # see if only a name has been passed
        if ( $Dest && $Dest !~ m{ \A (\d+)? \| \| .+ \z }xms ) {

            # see if we can get an ID for this queue name
            my $DestID = $QueueObject->QueueLookup(
                Queue => $Dest,
            );

            if ($DestID) {
                $Dest = $DestID . '||' . $Dest;
            }
            else {
                $Dest = '';
            }
        }

        my ($NewQueueID) = split( /\|\|/, $Dest );
        if ( !$NewQueueID ) {
            $GetParam{OwnerAll} = 1;
        }
        else {
            my %Queue = $QueueObject->GetSystemAddress( QueueID => $NewQueueID );
            $GetParam{From} = $Queue{Email};
        }

        my $CustomerUser = $ParamObject->GetParam( Param => 'CustomerUser' )
            || $ParamObject->GetParam( Param => 'PreSelectedCustomerUser' )
            || $ParamObject->GetParam( Param => 'SelectedCustomerUser' )
            || '';
        my $CustomerID           = $ParamObject->GetParam( Param => 'CustomerID' ) || '';
        my $SelectedCustomerUser = $ParamObject->GetParam( Param => 'SelectedCustomerUser' )
            || '';
        my $ExpandCustomerName = $ParamObject->GetParam( Param => 'ExpandCustomerName' )
            || 0;
        my %FromExternalCustomer;
        $FromExternalCustomer{Customer} = $ParamObject->GetParam( Param => 'PreSelectedCustomerUser' )
            || $ParamObject->GetParam( Param => 'CustomerUser' )
            || '';
        $GetParam{QueueID}            = $NewQueueID;
        $GetParam{ExpandCustomerName} = $ExpandCustomerName;

        # get sender queue from
        my $Signature = '';
        if ($NewQueueID) {
            $Signature = $Self->_GetSignature(
                QueueID        => $NewQueueID,
                CustomerUserID => $CustomerUser
            );
        }

        if ( $ParamObject->GetParam( Param => 'OwnerAllRefresh' ) ) {
            $GetParam{OwnerAll} = 1;
            $ExpandCustomerName = 3;
        }
        if ( $ParamObject->GetParam( Param => 'ResponsibleAllRefresh' ) ) {
            $GetParam{ResponsibleAll} = 1;
            $ExpandCustomerName = 3;
        }
        if ( $ParamObject->GetParam( Param => 'ClearTo' ) ) {
            $GetParam{To} = '';
            $ExpandCustomerName = 3;
        }
        for my $Count ( 1 .. 2 ) {
            my $Item = $ParamObject->GetParam( Param => "ExpandCustomerName$Count" ) || 0;
            if ( $Count == 1 && $Item ) {
                $ExpandCustomerName = 1;
            }
            elsif ( $Count == 2 && $Item ) {
                $ExpandCustomerName = 2;
            }
        }

# Rother OSS / ITSMCore - calculate Priority via CIP matrix
        if ( $CIPCalculate && $CIPCalculate == 2 && $GetParam{DynamicField}{DynamicField_ITSMImpact} ) {

            # get the criticality either from the manually set dynamic field, or the service
            my $Criticality = $GetParam{DynamicField_ITSMCriticality};

            if ( !$Criticality && $GetParam{ServiceID} ) {
                my %Service = $ServiceObject->ServiceGet(
                    ServiceID => $GetParam{ServiceID},
                    UserID    => 1,
                );

                $Criticality = $Service{Criticality};
            }

            if ( $Criticality ) {
                my $PriorityID = $CIPAllocateObject->PriorityAllocationGet(
                    Criticality => $Criticality,
                    Impact      => $GetParam{DynamicField}{DynamicField_ITSMImpact},
                );

                if ( $PriorityID ne $GetParam{PriorityID} ) {

                    # this should never happen; we just enforce the prio and write an error to the log file here
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "Got PriorityID '$GetParam{PriorityID}', but CIP enforces '$PriorityID' - overriding the frontend value.",
                    );

                    $GetParam{PriorityID} = $PriorityID;
                }
            }
        }
# EO ITSMCore

        # skip validation of hidden fields
        my %Visibility;

        # transform dynamic field data into DFName => DFName pair
        my %DynamicFieldAcl = map { $_ => $_ } keys $Self->{DynamicField}->%*;

        # call ticket ACLs for DynamicFields to check field visibility
        my $ACLResult = $TicketObject->TicketAcl(
            %GetParam,
            CustomerUserID => $CustomerUser || '',
            Action         => $Self->{Action},
            ReturnType     => 'Form',
            ReturnSubType  => '-',
            Data           => \%DynamicFieldAcl,
            UserID         => $Self->{UserID},
        );
        if ($ACLResult) {
            %Visibility = map { 'DynamicField_' . $_ => 0 } keys $Self->{DynamicField}->%*;
            my %AclData = $TicketObject->TicketAclData();
            for my $Field ( sort keys %AclData ) {
                $Visibility{ 'DynamicField_' . $Field } = 1;
            }
        }
        else {
            %Visibility = map { 'DynamicField_' . $_ => 1 } keys $Self->{DynamicField}->%*;
        }

        # remember dynamic field validation results if erroneous
        my %DynamicFieldValidationResult;
        my %DynamicFieldPossibleValues;

        # cycle through the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( values $Self->{DynamicField}->%* ) {
            next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);

            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,
                        %ACLCompatGetParam,
                        CustomerUserID => $CustomerUser || '',
                        Action         => $Self->{Action},
                        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;
                    }
                }
            }

            $DynamicFieldPossibleValues{ 'DynamicField_' . $DynamicFieldConfig->{Name} } = $PossibleValuesFilter;

            # do not validate on invisible fields
            if ( !$ExpandCustomerName && $Visibility{ 'DynamicField_' . $DynamicFieldConfig->{Name} } ) {

                my $ValidationResult = $DynamicFieldBackendObject->EditFieldValueValidate(
                    DynamicFieldConfig   => $DynamicFieldConfig,
                    PossibleValuesFilter => $PossibleValuesFilter,
                    ParamObject          => $ParamObject,

                    # Mandatory is added to the configs by $Self->new
                    Mandatory => $DynamicFieldConfig->{Mandatory},
                );

                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';
                    $DynamicFieldValidationResult{ $DynamicFieldConfig->{Name} } = $ValidationResult;
                }
            }
        }

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

        # expand customer name
        my %CustomerUserData;
        if ( $ExpandCustomerName == 1 ) {

            # search customer
            my %CustomerUserList;
            %CustomerUserList = $CustomerUserObject->CustomerSearch(
                Search => $GetParam{To},
            );

            # check if just one customer user exists
            # if just one, fillup CustomerUserID and CustomerID
            $Param{CustomerUserListCount} = 0;
            for my $KeyCustomerUser ( sort keys %CustomerUserList ) {
                $Param{CustomerUserListCount}++;
                $Param{CustomerUserListLast}     = $CustomerUserList{$KeyCustomerUser};
                $Param{CustomerUserListLastUser} = $KeyCustomerUser;
            }
            if ( $Param{CustomerUserListCount} == 1 ) {
                $GetParam{To}              = $Param{CustomerUserListLast};
                $Error{ExpandCustomerName} = 1;
                my %CustomerUserData = $CustomerUserObject->CustomerUserDataGet(
                    User => $Param{CustomerUserListLastUser},
                );
                if ( $CustomerUserData{UserCustomerID} ) {
                    $CustomerID = $CustomerUserData{UserCustomerID};
                }
                if ( $CustomerUserData{UserLogin} ) {
                    $CustomerUser = $CustomerUserData{UserLogin};
                }
            }

            # if more the one customer user exists, show list
            # and clean CustomerUserID and CustomerID
            else {

                # don't check email syntax on multi customer select
                $ConfigObject->Set(
                    Key   => 'CheckEmailAddresses',
                    Value => 0
                );
                $CustomerID = '';

                # clear to if there is no customer found
                if ( !%CustomerUserList ) {
                    $GetParam{To} = '';
                }
                $Error{ExpandCustomerName} = 1;
            }
        }

        # get from and customer id if customer user is given
        # This is used by Kernel::Modules::AdminCustomerUser
        elsif ( $ExpandCustomerName == 2 ) {
            %CustomerUserData = $CustomerUserObject->CustomerUserDataGet(
                User => $CustomerUser,
            );
            my %CustomerUserList = $CustomerUserObject->CustomerSearch(
                UserLogin => $CustomerUser,
            );
            for my $KeyCustomerUser ( sort keys %CustomerUserList ) {
                $GetParam{To} = $CustomerUserList{$KeyCustomerUser};
            }
            if ( $CustomerUserData{UserCustomerID} ) {
                $CustomerID = $CustomerUserData{UserCustomerID};
            }
            if ( $CustomerUserData{UserLogin} ) {
                $CustomerUser = $CustomerUserData{UserLogin};
            }
            if ( $FromExternalCustomer{Customer} ) {
                my %ExternalCustomerUserData = $CustomerUserObject->CustomerUserDataGet(
                    User => $FromExternalCustomer{Customer},
                );
                $FromExternalCustomer{Email} = $ExternalCustomerUserData{UserMailString};
            }
            $Error{ExpandCustomerName} = 1;
        }

        # if a new destination queue is selected
        elsif ( $ExpandCustomerName == 3 ) {
            $Error{NoSubmit} = 1;
            $CustomerUser = $SelectedCustomerUser;
        }

        # show customer info
        my %CustomerData;
        if ( $ConfigObject->Get('Ticket::Frontend::CustomerInfoCompose') ) {
            if ( $CustomerUser || $SelectedCustomerUser ) {
                %CustomerData = $CustomerUserObject->CustomerUserDataGet(
                    User => $CustomerUser || $SelectedCustomerUser,
                );
            }
            elsif ($CustomerID) {
                %CustomerData = $CustomerUserObject->CustomerUserDataGet(
                    CustomerID => $CustomerID,
                );
            }
        }

        # check email address
        PARAMETER:
        for my $Parameter (qw(To Cc Bcc)) {
            next PARAMETER if !$GetParam{$Parameter};
            for my $Email ( Mail::Address->parse( $GetParam{$Parameter} ) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $Error{ $Parameter . 'ErrorType' } = $Parameter . $CheckItemObject->CheckErrorType() . 'ServerErrorMsg';
                    $Error{ $Parameter . 'Invalid' }   = 'ServerError';
                }

                my $IsLocal = $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress(
# Rother OSS / DiscreteSystemAddresses
                    QueueID => $GetParam{QueueID},
# EO DiscreteSystemAddresses
                    Address => $Email->address()
                );
                if ($IsLocal) {
                    $Error{ $Parameter . 'IsLocalAddress' } = 'ServerError';
                }
            }
        }

        # if it is not a subaction about attachments, check for server errors
        if ( !$ExpandCustomerName ) {
            if ( !$GetParam{To} ) {
                $Error{'ToInvalid'} = 'ServerError';
            }
            if ( !$GetParam{Subject} ) {
                $Error{'SubjectInvalid'} = 'ServerError';
            }
            if ( !$NewQueueID ) {
                $Error{'DestinationInvalid'} = 'ServerError';
            }
            if ( !$GetParam{Body} ) {
                $Error{'BodyInvalid'} = 'ServerError';
            }

            # check if date is valid
            if (
                !$ExpandCustomerName
                && $StateData{TypeName}
                && $StateData{TypeName} =~ /^pending/i
                )
            {

                # convert pending date to a 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');

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

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

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

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

            if ( $ConfigObject->Get('Ticket::Type') && !$GetParam{TypeID} ) {
                $Error{'TypeInvalid'} = 'ServerError';
            }
            if (
                $ConfigObject->Get('Ticket::Frontend::AccountTime')
                && $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
                && $GetParam{TimeUnits} eq ''
                )
            {
                $Error{'TimeUnitsInvalid'} = 'ServerError';
            }
        }

        # run compose modules
        my %ArticleParam;
        if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {
            my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
            for my $Job ( sort keys %Jobs ) {

                # load module
                if ( !$MainObject->Require( $Jobs{$Job}->{Module} ) ) {
                    return $LayoutObject->FatalError();
                }

                my $Object = $Jobs{$Job}->{Module}->new(
                    %{$Self},
                    Debug => $Debug,
                );

                # get params
                my $Multiple;
                PARAMETER:
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {

                    if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                        @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                        $Multiple = 1;
                        next PARAMETER;
                    }

                    $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
                }

                # run module
                $Object->Run(
                    %GetParam,
                    StoreNew => 1,
                    Config   => $Jobs{$Job}
                );

                # get options that have been removed from the selection
                # and add them back to the selection so that the submit
                # will contain options that were hidden from the agent
                my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );

                if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                    my @RemovedOptions = $Object->GetOptionsToRemoveAJAX(%GetParam);
                    if (@RemovedOptions) {
                        if ($Multiple) {
                            for my $RemovedOption (@RemovedOptions) {
                                push @{ $GetParam{$Key} }, $RemovedOption;
                            }
                        }
                        else {
                            $GetParam{$Key} = shift @RemovedOptions;
                        }
                    }
                }

                # ticket params
                %ArticleParam = (
                    %ArticleParam,
                    $Object->ArticleOption( %GetParam, %ArticleParam, Config => $Jobs{$Job} ),
                );

                # get errors
                %Error = (
                    %Error,
                    $Object->Error( %GetParam, Config => $Jobs{$Job} ),
                );
            }
        }

        if (%Error) {

            if ( $Error{ToIsLocalAddress} ) {
                $LayoutObject->Block(
                    Name => 'ToIsLocalAddressServerErrorMsg',
                    Data => \%GetParam,
                );
            }

            if ( $Error{CcIsLocalAddress} ) {
                $LayoutObject->Block(
                    Name => 'CcIsLocalAddressServerErrorMsg',
                    Data => \%GetParam,
                );
            }

            if ( $Error{BccIsLocalAddress} ) {
                $LayoutObject->Block(
                    Name => 'BccIsLocalAddressServerErrorMsg',
                    Data => \%GetParam,
                );
            }

            # get and format default subject and body
            my $Subject = $LayoutObject->Output(
                Template => $Config->{Subject} || '',
            );

            my $Body = $LayoutObject->Output(
                Template => $Config->{Body} || '',
            );

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

            # set Body and Subject parameters for Output
            if ( !$GetParam{Subject} ) {
                $GetParam{Subject} = $Subject;
            }

            if ( !$GetParam{Body} ) {
                $GetParam{Body} = $Body;
            }

            # get services
            my $Services = $Self->_GetServices(
                %GetParam,
                %ACLCompatGetParam,
                CustomerUserID => $CustomerUser || '',
                QueueID        => $NewQueueID   || 1,
            );

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

            my $SLAs = $Self->_GetSLAs(
                %GetParam,
                %ACLCompatGetParam,
                QueueID  => $NewQueueID || 1,
                Services => $Services,
            );

            # header and navigation bar
            my $Output = join '',
                $LayoutObject->Header(),
                $LayoutObject->NavigationBar();

            # html output
            $Output .= $Self->_MaskEmailNew(
                QueueID => $Self->{QueueID},
                Users   => $Self->_GetUsers(
                    %GetParam,
                    %ACLCompatGetParam,
                    QueueID  => $NewQueueID,
                    AllUsers => $GetParam{OwnerAll}
                ),
                UserSelected     => $NewUserID,
                ResponsibleUsers => $Self->_GetResponsibles(
                    %GetParam,
                    %ACLCompatGetParam,
                    QueueID  => $NewQueueID,
                    AllUsers => $GetParam{ResponsibleAll}
                ),
                ResponsibleUserSelected => $NewResponsibleID,
                NextStates              => $Self->_GetNextStates(
                    %GetParam,
                    %ACLCompatGetParam,
                    CustomerUserID => $CustomerUser || '',
                    QueueID        => $NewQueueID   || 1,
                ),
                NextState  => $NextState,
                Priorities => $Self->_GetPriorities(
                    %GetParam,
                    %ACLCompatGetParam,
                    CustomerUserID => $CustomerUser || '',
                    QueueID        => $NewQueueID   || 1,
                ),
                Types => $Self->_GetTypes(
                    %GetParam,
                    %ACLCompatGetParam,
                    CustomerUserID => $CustomerUser || '',
                    QueueID        => $NewQueueID   || 1,
                ),
                Services          => $Services,
                SLAs              => $SLAs,
                StandardTemplates => $Self->_GetStandardTemplates(
                    %GetParam,
                    %ACLCompatGetParam,
                    QueueID => $NewQueueID || '',
                ),
                CustomerID        => $LayoutObject->Ascii2Html( Text => $CustomerID ),
                CustomerUser      => $CustomerUser,
                CustomerData      => \%CustomerData,
                TimeUnitsRequired => (
                    $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
                    ? 'Validate_Required'
                    : ''
                ),
                FromList     => $Self->_GetTos(),
                FromSelected => $Dest,
                Subject      => $LayoutObject->Ascii2Html( Text => $GetParam{Subject} ),
                Body         => $LayoutObject->Ascii2Html( Text => $GetParam{Body} ),
                Errors       => \%Error,
                Attachments  => \@Attachments,
                Signature    => $Signature,
                %GetParam,
                MultipleCustomer     => \@MultipleCustomer,
                MultipleCustomerCc   => \@MultipleCustomerCc,
                MultipleCustomerBcc  => \@MultipleCustomerBcc,
                FromExternalCustomer => \%FromExternalCustomer,
                Visibility           => \%Visibility,
                DFPossibleValues     => \%DynamicFieldPossibleValues,
                DFErrors             => \%DynamicFieldValidationResult,
            );

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

            return $Output;
        }

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

        # create new ticket, do db insert
        my $TicketID = $TicketObject->TicketCreate(
            Title        => $GetParam{Subject},
            QueueID      => $NewQueueID,
            Subject      => $GetParam{Subject},
            Lock         => 'unlock',
            TypeID       => $GetParam{TypeID},
            ServiceID    => $GetParam{ServiceID},
            SLAID        => $GetParam{SLAID},
            StateID      => $NextStateID,
            PriorityID   => $GetParam{PriorityID},
            OwnerID      => 1,
            CustomerID   => $CustomerID,
            CustomerUser => $SelectedCustomerUser,
            UserID       => $Self->{UserID},
        );

        # set ticket dynamic fields
        # cycle through the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( values $Self->{DynamicField}->%* ) {
            next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);
            next DYNAMICFIELD if $DynamicFieldConfig->{ObjectType} ne 'Ticket';
            next DYNAMICFIELD if !$Visibility{"DynamicField_$DynamicFieldConfig->{Name}"};
            next DYNAMICFIELD if $DynamicFieldConfig->{Readonly};

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

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

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

        # prepare subject
        my $Tn = $TicketObject->TicketNumberLookup( TicketID => $TicketID );
        $GetParam{Subject} = $TicketObject->TicketSubjectBuild(
            TicketNumber => $Tn,
            Subject      => $GetParam{Subject} || '',
            Type         => 'New',
        );

        # check if new owner is given (then send no agent notify)
        my $NoAgentNotify = 0;
        if ($NewUserID) {
            $NoAgentNotify = 1;
        }

        my $MimeType = 'text/plain';
        if ( $LayoutObject->{BrowserRichText} ) {
            $MimeType = 'text/html';
            $GetParam{Body} .= '<br/><br/>' . $Signature;

            # remove unused inline images
            my @NewAttachmentData;
            ATTACHMENT:
            for my $Attachment (@AttachmentData) {
                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};
            }
            @AttachmentData = @NewAttachmentData;

            # verify html document
            $GetParam{Body} = $LayoutObject->RichTextDocumentComplete(
                String => $GetParam{Body},
            );
        }
        else {
            $GetParam{Body} .= "\n\n" . $Signature;
        }

        # lookup sender
        my $TemplateGenerator = $Kernel::OM->Get('Kernel::System::TemplateGenerator');
        my $Sender            = $TemplateGenerator->Sender(
            QueueID => $NewQueueID,
            UserID  => $Self->{UserID},
        );

        my $ArticleObject        = $Kernel::OM->Get('Kernel::System::Ticket::Article');
        my $ArticleBackendObject = $ArticleObject->BackendForChannel( ChannelName => 'Email' );

        # send email
        my $ArticleID = $ArticleBackendObject->ArticleSend(
            NoAgentNotify        => $NoAgentNotify,
            Attachment           => \@AttachmentData,
            TicketID             => $TicketID,
            SenderType           => $Config->{SenderType},
            IsVisibleForCustomer => $Config->{IsVisibleForCustomer},
            From                 => $Sender,
            To                   => $GetParam{To},
            Cc                   => $GetParam{Cc},
            Bcc                  => $GetParam{Bcc},
            Subject              => $GetParam{Subject},
            Body                 => $GetParam{Body},
            Charset              => $LayoutObject->{UserCharset},
            MimeType             => $MimeType,
            UserID               => $Self->{UserID},
            HistoryType          => $Config->{HistoryType},
            HistoryComment       => $Config->{HistoryComment}
                || "\%\%$GetParam{To}, $GetParam{Cc}, $GetParam{Bcc}",
            %ArticleParam,
        );
        if ( !$ArticleID ) {
            return $LayoutObject->ErrorScreen();
        }

        # set article dynamic fields
        # cycle through the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $DynamicFieldConfig ( values $Self->{DynamicField}->%* ) {
            next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);
            next DYNAMICFIELD if $DynamicFieldConfig->{ObjectType} ne 'Article';
            next DYNAMICFIELD if !$Visibility{"DynamicField_$DynamicFieldConfig->{Name}"};
            next DYNAMICFIELD if $DynamicFieldConfig->{Readonly};

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

        # remove all form data
        $Kernel::OM->Get('Kernel::System::Web::FormCache')->FormIDRemove( FormID => $Self->{FormID} );

        # delete hidden fields cache
        $Kernel::OM->Get('Kernel::System::Cache')->Delete(
            Type => 'HiddenFields',
            Key  => $Self->{FormID},
        );

        # link tickets
        if (
            $GetParam{LinkTicketID}
            && $Config->{SplitLinkType}
            && $Config->{SplitLinkType}->{LinkType}
            && $Config->{SplitLinkType}->{Direction}
            )
        {
            my $Access = $TicketObject->TicketPermission(
                Type     => 'ro',
                TicketID => $GetParam{LinkTicketID},
                UserID   => $Self->{UserID}
            );

            if ( !$Access ) {
                return $LayoutObject->NoPermission(
                    Message    => "You need ro permission!",
                    WithHeader => 'yes',
                );
            }

            my $SourceKey = $GetParam{LinkTicketID};
            my $TargetKey = $TicketID;

            if ( $Config->{SplitLinkType}->{Direction} eq 'Source' ) {
                $SourceKey = $TicketID;
                $TargetKey = $GetParam{LinkTicketID};
            }

            # link the tickets
            $Kernel::OM->Get('Kernel::System::LinkObject')->LinkAdd(
                SourceObject => 'Ticket',
                SourceKey    => $SourceKey,
                TargetObject => 'Ticket',
                TargetKey    => $TargetKey,
                Type         => $Config->{SplitLinkType}->{LinkType} || 'Normal',
                State        => 'Valid',
                UserID       => $Self->{UserID},
            );
        }

        # set owner (if new user id is given)
        if ($NewUserID) {
            $TicketObject->TicketOwnerSet(
                TicketID  => $TicketID,
                NewUserID => $NewUserID,
                UserID    => $Self->{UserID},
            );

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

        # else set owner to current agent but do not lock it
        else {
            $TicketObject->TicketOwnerSet(
                TicketID           => $TicketID,
                NewUserID          => $Self->{UserID},
                SendNoNotification => 1,
                UserID             => $Self->{UserID},
            );
        }

        # set responsible (if new user id is given)
        if ($NewResponsibleID) {
            $TicketObject->TicketResponsibleSet(
                TicketID  => $TicketID,
                NewUserID => $NewResponsibleID,
                UserID    => $Self->{UserID},
            );
        }

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

        # closed tickets get unlocked
        if ( $StateData{TypeName} =~ /^close/i ) {

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

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

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

        # get redirect screen
        my $NextScreen = $Self->{Session}{UserCreateNextMask} || 'AgentTicketEmail';

        # redirect
        return $LayoutObject->Redirect(
            OP => "Action=$NextScreen;Subaction=Created;TicketID=$TicketID",
        );
    }
    elsif ( $Self->{Subaction} eq 'AJAXUpdate' ) {
        my $Dest           = $ParamObject->GetParam( Param => 'Dest' ) || '';
        my $CustomerUser   = $ParamObject->GetParam( Param => 'SelectedCustomerUser' );
        my $ElementChanged = $ParamObject->GetParam( Param => 'ElementChanged' ) || '';

        # get From based on selected queue
        my $QueueID = '';
        if ( $Dest =~ /^(\d{1,100})\|\|.+?$/ ) {
            $QueueID = $1;
            my %Queue = $QueueObject->GetSystemAddress( QueueID => $QueueID );
            $GetParam{From} = $Queue{Email};
        }
        $GetParam{Dest}    = $Dest;
        $GetParam{QueueID} = $QueueID;

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

        my $Autoselect = $ConfigObject->Get('TicketACL::Autoselect') || undef;
        my $ACLPreselection;
        if ( $ConfigObject->Get('TicketACL::ACLPreselection') ) {

            # get cached preselection rules
            my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');
            $ACLPreselection = $CacheObject->Get(
                Type => 'TicketACL',
                Key  => 'Preselection',
            );
            if ( !$ACLPreselection ) {
                $ACLPreselection = $FieldRestrictionsObject->SetACLPreselectionCache();
            }
        }

        my %Convergence = (
            StdFields => 0,
            Fields    => 0,
        );
        my %ChangedElements = $ElementChanged ? ( $ElementChanged => 1 ) : ();
        if ( $ChangedElements{ServiceID} ) {
            $ChangedElements{CustomerUserID} = 1;
            $ChangedElements{CustomerID}     = 1;
        }
        my %ChangedElementsDFStart = %ChangedElements;
        my %ChangedStdFields       = $ElementChanged && $ElementChanged !~ /^DynamicField_/ ? %ChangedElements : ();

        my $LoopProtection = 100;
        my %StdFieldValues;
        my %DynFieldStates = (
            Visibility => {},
            Fields     => {},
            Sets       => {},
        );

        until ( $Convergence{Fields} ) {

            # determine standard field input
            until ( $Convergence{StdFields} ) {

                my %NewChangedElements;

                # which standard fields to check - FieldID => GetParamValue (necessary for Dest)
                my %Check = (
                    Dest               => 'QueueID',
                    NewUserID          => 'NewUserID',
                    NewResponsibleID   => 'NewResponsibleID',
                    NextStateID        => 'NextStateID',
                    PriorityID         => 'PriorityID',
                    ServiceID          => 'ServiceID',
                    SLAID              => 'SLAID',
                    StandardTemplateID => 'StandardTemplateID',
                    TypeID             => 'TypeID',
                );
                if ($ACLPreselection) {
                    FIELD:
                    for my $FieldID ( sort keys %Check ) {
                        if ( !$ACLPreselection->{Fields}{$FieldID} ) {
                            $Kernel::OM->Get('Kernel::System::Log')->Log(
                                Priority => 'debug',
                                Message  => "$FieldID not defined in TicketACL preselection rules!"
                            );
                            next FIELD;
                        }
                        if ( $Autoselect && $Autoselect->{$FieldID} && $ChangedElements{$FieldID} ) {
                            next FIELD;
                        }
                        for my $Element ( sort keys %ChangedElements ) {
                            if (
                                $ACLPreselection->{Rules}{Ticket}{$Element}{$FieldID}
                                || $Self->{InternalDependancy}{$Element}{$FieldID}
                                )
                            {
                                next FIELD;
                            }
                            if ( !$ACLPreselection->{Fields}{$Element} ) {
                                $Kernel::OM->Get('Kernel::System::Log')->Log(
                                    Priority => 'debug',
                                    Message  => "$Element not defined in TicketACL preselection rules!"
                                );
                                next FIELD;
                            }
                        }

                        # delete unaffected fields
                        delete $Check{$FieldID};
                    }
                }

                # for each standard field which has to be checked, run the defined method
                METHOD:
                for my $Field ( @{ $Self->{FieldMethods} } ) {
                    next METHOD if !$Check{ $Field->{FieldID} };

                    # use $Check{ $Field->{FieldID} } for Dest=>QueueID
                    $StdFieldValues{ $Check{ $Field->{FieldID} } } = $Field->{Method}->(
                        $Self,
                        %GetParam,
                        OwnerID        => $GetParam{NewUserID},
                        CustomerUserID => $CustomerUser || '',
                        QueueID        => $GetParam{QueueID},
                        Services       => $StdFieldValues{ServiceID} || undef,    # needed for SLAID
                    );

                    # special stuff for QueueID/Dest: Dest is "QueueID||QueueName" => "QueueName";
                    if ( $Field->{FieldID} eq 'Dest' ) {
                        TOs:
                        for my $QueueID ( sort keys %{ $StdFieldValues{QueueID} } ) {
                            next TOs if ( $StdFieldValues{QueueID}{$QueueID} eq '-' );
                            $StdFieldValues{Dest}{"$QueueID||$StdFieldValues{QueueID}{ $QueueID }"} = $StdFieldValues{QueueID}{$QueueID};
                        }

                        # check current selection of QueueID (Dest will be done together with the other fields)
                        if ( $GetParam{QueueID} && !$StdFieldValues{Dest}{ $GetParam{Dest} } ) {
                            $GetParam{QueueID} = '';
                        }

                        # autoselect
                        if ( !$GetParam{QueueID} && $Autoselect && $Autoselect->{Dest} ) {
                            $GetParam{QueueID} = $FieldRestrictionsObject->Autoselect(
                                PossibleValues => $StdFieldValues{QueueID},
                            ) || '';
                        }
                    }

# Rother OSS / ITSMCore - calculate Priority via CIP matrix
                    elsif ( $Field->{FieldID} eq 'PriorityID' && $CIPCalculate && $GetParam{DynamicField}{DynamicField_ITSMImpact} ) {

                        # get the criticality either from the manually set dynamic field, or the service
                        my $Criticality = $GetParam{DynamicField_ITSMCriticality};

                        if ( !$Criticality && $GetParam{ServiceID} ) {
                            my %Service = $ServiceObject->ServiceGet(
                                ServiceID => $GetParam{ServiceID},
                                UserID    => 1,
                            );

                            $Criticality = $Service{Criticality};
                        }

                        if ( $Criticality ) {

                            # recalculate the priority if any of the impacting elements changed
                            if ( $ChangedElements{DynamicField_ITSMImpact} || $ChangedElements{ServiceID} || $ChangedElements{DynamicField_ITSMCriticality} ) {
                                my $PriorityID = $CIPAllocateObject->PriorityAllocationGet(
                                    Criticality => $Criticality,
                                    Impact      => $GetParam{DynamicField}{DynamicField_ITSMImpact},
                                );

                                if ( $StdFieldValues{PriorityID}{ $PriorityID } && $PriorityID ne $GetParam{PriorityID} ) {
                                    $GetParam{PriorityID}           = $PriorityID;
                                    $NewChangedElements{PriorityID} = 1;
                                    $ChangedStdFields{PriorityID}   = 1;
                                }
                            }

                            # if we enforce CIPAllocation in any case delete all other priority options
                            if ( $GetParam{PriorityID} && $CIPCalculate == 2 ) {
                                $StdFieldValues{PriorityID} = {
                                    $GetParam{PriorityID} => $StdFieldValues{PriorityID}{ $GetParam{PriorityID} },
                                };
                            }
                        }
                    }
# EO ITSMCore

                    # check whether current selected value is still valid for the field
                    if (
                        $GetParam{ $Field->{FieldID} }
                        && !$StdFieldValues{ $Field->{FieldID} }{ $GetParam{ $Field->{FieldID} } }
                        )
                    {
                        # if not empty the field
                        $GetParam{ $Field->{FieldID} }           = '';
                        $NewChangedElements{ $Field->{FieldID} } = 1;
                        $ChangedStdFields{ $Field->{FieldID} }   = 1;
                    }

                    # autoselect
                    if ( !$GetParam{ $Field->{FieldID} } && $Autoselect && $Autoselect->{ $Field->{FieldID} } ) {
                        $GetParam{ $Field->{FieldID} } = $FieldRestrictionsObject->Autoselect(
                            PossibleValues => $StdFieldValues{ $Field->{FieldID} },
                        ) || '';
                        if ( $GetParam{ $Field->{FieldID} } ) {
                            $NewChangedElements{ $Field->{FieldID} } = 1;
                            $ChangedStdFields{ $Field->{FieldID} }   = 1;
                        }
                    }
                }

                if ( !%NewChangedElements ) {
                    $Convergence{StdFields} = 1;
                }
                else {
                    %ChangedElements = %NewChangedElements;
                }

                %ChangedElementsDFStart = (
                    %ChangedElementsDFStart,
                    %NewChangedElements,
                );

                if ( $LoopProtection-- < 1 ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "Ran into unresolvable loop!",
                    );
                    return;
                }

            }

            %ChangedElements        = %ChangedElementsDFStart;
            %ChangedElementsDFStart = ();

            # check dynamic fields
            my %CurFieldStates;
            if (%ChangedElements) {

                # get values and visibility of dynamic fields
                %CurFieldStates = $FieldRestrictionsObject->GetFieldStates(
                    TicketObject              => $TicketObject,
                    DynamicFields             => $Self->{DynamicField},
                    DynamicFieldBackendObject => $DynamicFieldBackendObject,
                    ChangedElements           => \%ChangedElements,            # optional to reduce ACL evaluation
                    Action                    => $Self->{Action},
                    UserID                    => $Self->{UserID},
                    FormID                    => $Self->{FormID},
                    CustomerUser              => $CustomerUser || '',
                    GetParam                  => {
                        %GetParam,
                        OwnerID => $GetParam{NewUserID},
                    },
                    Autoselect      => $Autoselect,
                    ACLPreselection => $ACLPreselection,
                    LoopProtection  => \$LoopProtection,
                );

                # combine FieldStates
                $DynFieldStates{Fields} = {
                    %{ $DynFieldStates{Fields} },
                    %{ $CurFieldStates{Fields} },
                };
                $DynFieldStates{Visibility} = {
                    %{ $DynFieldStates{Visibility} },
                    %{ $CurFieldStates{Visibility} },
                };
                $DynFieldStates{Sets} = {
                    %{ $DynFieldStates{Sets} },
                    %{ $CurFieldStates{Sets} },
                };

                # store new values
                $GetParam{DynamicField} = {
                    %{ $GetParam{DynamicField} },
                    %{ $CurFieldStates{NewValues} },
                };
            }

            # if dynamic fields changed, check standard fields again
            if ( %CurFieldStates && IsHashRefWithData( $CurFieldStates{NewValues} ) ) {
                $Convergence{StdFields} = 0;
                %ChangedElements = map { $_ => 1 } keys %{ $CurFieldStates{NewValues} };
            }
            else {
                $Convergence{Fields} = 1;
            }

        }

        my $Signature = '';
        if ( $GetParam{QueueID} ) {
            $Signature = $Self->_GetSignature(
                QueueID        => $GetParam{QueueID},
                CustomerUserID => $CustomerUser,
            );
        }

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

        # cycle through the activated Dynamic Fields for this screen
        DYNAMICFIELD:
        for my $Name ( sort keys $DynFieldStates{Fields}->%* ) {
            my $DynamicFieldConfig = $Self->{DynamicField}{$Name};

            if ( $DynamicFieldConfig->{Config}{MultiValue} && ref $GetParam{DynamicField}{"DynamicField_$DynamicFieldConfig->{Name}"} eq 'ARRAY' ) {
                for my $i ( 0 .. $#{ $GetParam{DynamicField}{"DynamicField_$DynamicFieldConfig->{Name}"} } ) {
                    my $DataValues = $DynFieldStates{Fields}{$Name}{NotACLReducible}
                        ? ( $GetParam{DynamicField}{"DynamicField_$DynamicFieldConfig->{Name}"}[$i] // '' )
                        :
                        (
                            $DynamicFieldBackendObject->BuildSelectionDataGet(
                                DynamicFieldConfig => $DynamicFieldConfig,
                                PossibleValues     => $DynFieldStates{Fields}{$Name}{PossibleValues},
                                Value              => [ $GetParam{DynamicField}{"DynamicField_$DynamicFieldConfig->{Name}"}[$i] ],
                            )
                            || $DynFieldStates{Fields}{$Name}{PossibleValues}
                        );

                    # add dynamic field to the list of fields to update
                    push @DynamicFieldAJAX, {
                        Name        => 'DynamicField_' . $DynamicFieldConfig->{Name} . "_$i",
                        Data        => $DataValues,
                        SelectedID  => $GetParam{DynamicField}{"DynamicField_$DynamicFieldConfig->{Name}"}[$i],
                        Translation => $DynamicFieldConfig->{Config}{TranslatableValues} || 0,
                        Max         => 100,
                    };
                }

                # add template value for keeping templates in line with ACLs
                if ( !$DynFieldStates{Fields}{$Name}{NotACLReducible} ) {
                    my $DataValues = (
                        $DynamicFieldBackendObject->BuildSelectionDataGet(
                            DynamicFieldConfig => $DynamicFieldConfig,
                            PossibleValues     => $DynFieldStates{Fields}{$Name}{PossibleValues},
                            Value              => [ $DynamicFieldConfig->{Config}{DefaultValue} // '' ],
                            )
                            || $DynFieldStates{Fields}{$Name}{PossibleValues}
                    );

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

                next DYNAMICFIELD;
            }

            my $DataValues = $DynFieldStates{Fields}{$Name}{NotACLReducible}
                ? ( $GetParam{DynamicField}{"DynamicField_$DynamicFieldConfig->{Name}"} // '' )
                :
                (
                    $DynamicFieldBackendObject->BuildSelectionDataGet(
                        DynamicFieldConfig => $DynamicFieldConfig,
                        PossibleValues     => $DynFieldStates{Fields}{$Name}{PossibleValues},
                        Value              => $GetParam{DynamicField}{"DynamicField_$DynamicFieldConfig->{Name}"},
                    )
                    || $DynFieldStates{Fields}{$Name}{PossibleValues}
                );

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

        for my $SetField ( values $DynFieldStates{Sets}->%* ) {
            my $DynamicFieldConfig = $SetField->{DynamicFieldConfig};

            # the frontend name is the name of the inner field including its index or the '_Template' suffix
            DYNAMICFIELD:
            for my $FrontendName ( keys $SetField->{FieldStates}->%* ) {

                if ( $DynamicFieldConfig->{Config}{MultiValue} && ref $SetField->{Values}{$FrontendName} eq 'ARRAY' ) {
                    for my $i ( 0 .. $#{ $SetField->{Values}{$FrontendName} } ) {
                        my $DataValues = $SetField->{FieldStates}{$FrontendName}{NotACLReducible}
                            ? ( $SetField->{Values}{$FrontendName}[$i] // '' )
                            :
                            (
                                $DynamicFieldBackendObject->BuildSelectionDataGet(
                                    DynamicFieldConfig => $DynamicFieldConfig,
                                    PossibleValues     => $SetField->{FieldStates}{$FrontendName}{PossibleValues},
                                    Value              => [ $SetField->{Values}{$FrontendName}[$i] ],
                                )
                                || $SetField->{FieldStates}{$FrontendName}{PossibleValues}
                            );

                        # add dynamic field to the list of fields to update
                        push @DynamicFieldAJAX, {
                            Name        => 'DynamicField_' . $FrontendName . "_$i",
                            Data        => $DataValues,
                            SelectedID  => $SetField->{Values}{$FrontendName}[$i],
                            Translation => $DynamicFieldConfig->{Config}{TranslatableValues} || 0,
                            Max         => 100,
                        };
                    }

                    # add template value for keeping templates in line with ACLs
                    if ( !$SetField->{FieldStates}{$FrontendName}{NotACLReducible} ) {
                        my $DataValues = (
                            $DynamicFieldBackendObject->BuildSelectionDataGet(
                                DynamicFieldConfig => $DynamicFieldConfig,
                                PossibleValues     => $SetField->{FieldStates}{$FrontendName}{PossibleValues},
                                Value              => [ $DynamicFieldConfig->{Config}{DefaultValue} // '' ],
                                )
                                || $SetField->{FieldStates}{$FrontendName}{PossibleValues}
                        );

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

                    next DYNAMICFIELD;
                }

                my $DataValues = $SetField->{FieldStates}{$FrontendName}{NotACLReducible}
                    ? ( $SetField->{Values}{$FrontendName} // '' )
                    :
                    (
                        $DynamicFieldBackendObject->BuildSelectionDataGet(
                            DynamicFieldConfig => $DynamicFieldConfig,
                            PossibleValues     => $SetField->{FieldStates}{$FrontendName}{PossibleValues},
                            Value              => $SetField->{Values}{$FrontendName},
                        )
                        || $SetField->{FieldStates}{$FrontendName}{PossibleValues}
                    );

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

        if ( IsHashRefWithData( $DynFieldStates{Visibility} ) ) {
            push @DynamicFieldAJAX, {
                Name => 'Restrictions_Visibility',
                Data => $DynFieldStates{Visibility},
            };
        }

        # build AJAX return for the standard fields
        my @StdFieldAJAX;
        my %Attributes = (
            Dest => {
                Translation  => $TreeView,
                PossibleNone => 1,
                TreeView     => $TreeView,
                Max          => 100,
            },
            NewUserID => {
                Translation  => 0,
                PossibleNone => 1,
                Max          => 100,
            },
            NewResponsibleID => {
                Translation  => 0,
                PossibleNone => 1,
                Max          => 100,
            },
            NextStateID => {
                Translation => 1,
                Max         => 100,
            },
            PriorityID => {
                Translation => 1,
                Max         => 100,
            },
            ServiceID => {
                PossibleNone => 1,
                Translation  => $TreeView,
                TreeView     => $TreeView,
                Max          => 100,
            },
            SLAID => {
                PossibleNone => 1,
                Translation  => 1,
                Max          => 100,
            },
            StandardTemplateID => {
                PossibleNone => 1,
                Translation  => 1,
                Max          => 100,
            },
            TypeID => {
                PossibleNone => 1,
                Translation  => 1,
                Max          => 100,
            }
        );
        delete $StdFieldValues{QueueID};
        for my $Field ( sort keys %StdFieldValues ) {
            push @StdFieldAJAX, {
                Name       => $Field,
                Data       => $StdFieldValues{$Field},
                SelectedID => $GetParam{$Field},
                %{ $Attributes{$Field} },
            };
        }

        my @TemplateAJAX;

        # update ticket body and attachments if needed.
        if ( $ChangedStdFields{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},
                    UserID         => $Self->{UserID},
                    CustomerUserID => $CustomerUser,
                );

                # 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 => 'UseTemplateCreate',
                    Data => '0',
                },
                {
                    Name => 'RichText',
                    Data => $TemplateText || '',
                },
                {
                    Name     => 'TicketAttachments',
                    Data     => \@TicketAttachments,
                    KeepData => 1,
                },
            );
        }

        my @ExtendedData;

        # run compose modules
        if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {

            my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
            JOB:
            for my $Job ( sort keys %Jobs ) {

                # load module
                next JOB if !$MainObject->Require( $Jobs{$Job}->{Module} );

                my $Object = $Jobs{$Job}->{Module}->new(
                    %{$Self},
                    Debug => $Debug,
                );

                my $Multiple;

                # get params
                PARAMETER:
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {

                    if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                        @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                        $Multiple = 1;
                        next PARAMETER;
                    }

                    $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
                }

                # run module
                my %Data = $Object->Data( %GetParam, Config => $Jobs{$Job} );

                # get AJAX param values
                if ( $Object->can('GetParamAJAX') ) {
                    %GetParam = ( %GetParam, $Object->GetParamAJAX(%GetParam) );
                }

                # get options that have to be removed from the selection visible
                # to the agent. These options will be added again on submit.
                if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                    my @OptionsToRemove = $Object->GetOptionsToRemoveAJAX(%GetParam);

                    for my $OptionToRemove (@OptionsToRemove) {
                        delete $Data{$OptionToRemove};
                    }
                }

                my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );

                if ($Key) {
                    push @ExtendedData, {
                        Name         => $Key,
                        Data         => \%Data,
                        SelectedID   => $GetParam{$Key},
                        Translation  => 1,
                        PossibleNone => 1,
                        Multiple     => $Multiple,
                        Max          => 100,
                    };
                }
            }
        }

        my $JSON = $LayoutObject->BuildSelectionJSON(
            [
                {
                    Name         => 'Signature',
                    Data         => $Signature,
                    Translation  => 1,
                    PossibleNone => 1,
                    Max          => 100,
                },
                @StdFieldAJAX,
                @DynamicFieldAJAX,
                @TemplateAJAX,
                @ExtendedData,
            ],
        );

        return $LayoutObject->Attachment(
            ContentType => 'application/json',
            Content     => $JSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }
    else {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('No Subaction!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }
}

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

    # use default Queue if none is provided
    $Param{QueueID} = $Param{QueueID} || 1;

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

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

    # get users
    my %ShownUsers;
    my %AllGroupsMembers = $Kernel::OM->Get('Kernel::System::User')->UserList(
        Type  => 'Long',
        Valid => 1,
    );

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

    # just show only users with selected custom queue
    if ( $Param{QueueID} && !$Param{OwnerAll} ) {
        my @UserIDs = $TicketObject->GetSubscribedUserIDsByQueueID(%Param);
        for my $KeyGroupMember ( sort keys %AllGroupsMembers ) {
            my $Hit = 0;
            for my $UID (@UserIDs) {
                if ( $UID eq $KeyGroupMember ) {
                    $Hit = 1;
                }
            }
            if ( !$Hit ) {
                delete $AllGroupsMembers{$KeyGroupMember};
            }
        }
    }

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

    # show all users who are owner or rw in the queue group
    elsif ( $Param{QueueID} ) {
        my $GID        = $Kernel::OM->Get('Kernel::System::Queue')->GetQueueGroupID( QueueID => $Param{QueueID} );
        my %MemberList = $Kernel::OM->Get('Kernel::System::Group')->PermissionGroupGet(
            GroupID => $GID,
            Type    => 'owner',
        );
        for my $KeyMember ( sort keys %MemberList ) {
            if ( $AllGroupsMembers{$KeyMember} ) {
                $ShownUsers{$KeyMember} = $AllGroupsMembers{$KeyMember};
            }
        }
    }

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

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

    return \%ShownUsers;
}

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

    # get users
    my %ShownUsers;
    my %AllGroupsMembers = $Kernel::OM->Get('Kernel::System::User')->UserList(
        Type  => 'Long',
        Valid => 1,
    );

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

    # just show only users with selected custom queue
    if ( $Param{QueueID} && !$Param{ResponsibleAll} ) {
        my @UserIDs = $TicketObject->GetSubscribedUserIDsByQueueID(%Param);
        for my $KeyGroupMember ( sort keys %AllGroupsMembers ) {
            my $Hit = 0;
            for my $UID (@UserIDs) {
                if ( $UID eq $KeyGroupMember ) {
                    $Hit = 1;
                }
            }
            if ( !$Hit ) {
                delete $AllGroupsMembers{$KeyGroupMember};
            }
        }
    }

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

    # show all users who are responsible or rw in the queue group
    elsif ( $Param{QueueID} ) {
        my $GID        = $Kernel::OM->Get('Kernel::System::Queue')->GetQueueGroupID( QueueID => $Param{QueueID} );
        my %MemberList = $Kernel::OM->Get('Kernel::System::Group')->PermissionGroupGet(
            GroupID => $GID,
            Type    => 'responsible',
        );
        for my $KeyMember ( sort keys %MemberList ) {
            if ( $AllGroupsMembers{$KeyMember} ) {
                $ShownUsers{$KeyMember} = $AllGroupsMembers{$KeyMember};
            }
        }
    }

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

    # use default Queue if none is provided
    $Param{QueueID} = $Param{QueueID} || 1;

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

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

    # use default Queue if none is provided
    $Param{QueueID} = $Param{QueueID} || 1;

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

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

    # get service
    my %Service;

    # use default Queue if none is provided
    $Param{QueueID} = $Param{QueueID} || 1;

    # 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,
            Action => $Self->{Action},
            UserID => $Self->{UserID},
        );
    }
    return \%Service;
}

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

    # use default Queue if none is provided
    $Param{QueueID} = $Param{QueueID} || 1;

    # get services if they were not determined in an AJAX call
    if ( !defined $Param{Services} ) {
        $Param{Services} = $Self->_GetServices(%Param);
    }

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

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

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

    # check own selection
    my %NewTos;
    if ( $ConfigObject->Get('Ticket::Frontend::NewQueueOwnSelection') ) {
        %NewTos = %{ $ConfigObject->Get('Ticket::Frontend::NewQueueOwnSelection') };
    }
    else {

        # SelectionType Queue or SystemAddress?
        my %Tos;
        if ( $ConfigObject->Get('Ticket::Frontend::NewQueueSelectionType') eq 'Queue' ) {
            %Tos = $Kernel::OM->Get('Kernel::System::Ticket')->MoveList(
                %Param,
                Type   => 'create',
                Action => $Self->{Action},
                UserID => $Self->{UserID},
            );
        }
        else {
            %Tos = $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressQueueList();
        }

        # get create permission queues
        my %UserGroups = $Kernel::OM->Get('Kernel::System::Group')->PermissionUserGet(
            UserID => $Self->{UserID},
            Type   => 'create',
        );

        my $SystemAddressObject = $Kernel::OM->Get('Kernel::System::SystemAddress');
        my $QueueObject         = $Kernel::OM->Get('Kernel::System::Queue');

        # build selection string
        QUEUEID:
        for my $QueueID ( sort keys %Tos ) {

            my %QueueData = $QueueObject->QueueGet( ID => $QueueID );

            # permission check, can we create new tickets in queue
            next QUEUEID if !$UserGroups{ $QueueData{GroupID} };

            my $String = $ConfigObject->Get('Ticket::Frontend::NewQueueSelectionString')
                || '<Realname> <<Email>> - Queue: <Queue>';
            $String =~ s/<Queue>/$QueueData{Name}/g;
            $String =~ s/<QueueComment>/$QueueData{Comment}/g;

            # remove trailing spaces
            if ( !$QueueData{Comment} ) {
                $String =~ s{\s+\z}{};
            }

            if ( $ConfigObject->Get('Ticket::Frontend::NewQueueSelectionType') ne 'Queue' ) {
                my %SystemAddressData = $SystemAddressObject->SystemAddressGet(
                    ID => $Tos{$QueueID},
                );
                $String =~ s/<Realname>/$SystemAddressData{Realname}/g;
                $String =~ s/<Email>/$SystemAddressData{Name}/g;
            }
            $NewTos{$QueueID} = $String;
        }
    }

    # add empty selection
    $NewTos{''} = '-';

    return \%NewTos;
}

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

    # prepare signature
    my $TemplateGenerator = $Kernel::OM->Get('Kernel::System::TemplateGenerator');
    my $Signature         = $TemplateGenerator->Signature(
        QueueID => $Param{QueueID},
        Data    => \%Param,
        UserID  => $Self->{UserID},
    );

    return $Signature;
}

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

    my $AccountedTime = '';

    # Get accounted time if AccountTime config item is enabled.
    if ( $Kernel::OM->Get('Kernel::Config')->Get('Ticket::Frontend::AccountTime') && defined $Param{ArticleID} ) {
        $AccountedTime = $Kernel::OM->Get('Kernel::System::Ticket::Article')->ArticleAccountedTimeGet(
            ArticleID => $Param{ArticleID},
        );
    }

    return $AccountedTime ? $AccountedTime : '';
}

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

    my %Templates;
    my $QueueID = $Param{QueueID} || '';

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

    if ( !$QueueID ) {
        my $UserDefaultQueue = $ConfigObject->Get('Ticket::Frontend::UserDefaultQueue') || '';

        if ($UserDefaultQueue) {
            $QueueID = $QueueObject->QueueLookup( Queue => $UserDefaultQueue );
        }
    }

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

    if ( !$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 = $QueueObject->QueueStandardTemplateMemberList(
        QueueID       => $QueueID,
        TemplateTypes => 1,
    );

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

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

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

    $Param{FormID} = $Self->{FormID};

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

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

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

    # set JS data
    $LayoutObject->AddJSData(
        Key   => 'CustomerSearch',
        Value => {
            ShowCustomerTickets => $ConfigObject->Get('Ticket::Frontend::ShowCustomerTickets'),
        },
    );

    if ( $Param{HideAutoselected} ) {

        # add Autoselect JS
        $LayoutObject->AddJSOnDocumentComplete(
            Code => "Core.Form.InitHideAutoselected({ FieldIDs: $Param{HideAutoselected} });",
        );
    }

    # build string
    $Param{OptionStrg} = $LayoutObject->BuildSelection(
        Data         => $Param{Users},
        SelectedID   => $Param{UserSelected},
        Class        => 'Modernize FormUpdate',
        Translation  => 0,
        Name         => 'NewUserID',
        PossibleNone => 1,
    );

    my $Config = $ConfigObject->Get("Ticket::Frontend::$Self->{Action}");

    # build next states string
    $Param{NextStatesStrg} = $LayoutObject->BuildSelection(
        Data          => $Param{NextStates},
        Name          => 'NextStateID',
        Class         => 'Modernize FormUpdate',
        Translation   => 1,
        SelectedValue => $Param{NextState} || $Config->{StateDefault},
    );

    # build Destination string
    my %NewTo;
    if ( $Param{FromList} ) {
        for my $KeyFrom ( sort keys %{ $Param{FromList} } ) {
            $NewTo{"$KeyFrom||$Param{FromList}->{$KeyFrom}"} = $Param{FromList}->{$KeyFrom};
        }
    }
    if ( !$Param{FromSelected} ) {
        my $UserDefaultQueue = $ConfigObject->Get('Ticket::Frontend::UserDefaultQueue') || '';

        if ($UserDefaultQueue) {
            my $QueueID = $Kernel::OM->Get('Kernel::System::Queue')->QueueLookup( Queue => $UserDefaultQueue );
            if ($QueueID) {
                $Param{FromSelected} = "$QueueID||$UserDefaultQueue";
            }
        }
    }

    if ( $ConfigObject->Get('Ticket::Frontend::NewQueueSelectionType') eq 'Queue' ) {
        $Param{FromStrg} = $LayoutObject->AgentQueueListOption(
            Class          => 'Validate_Required Modernize FormUpdate ' . ( $Param{Errors}->{DestinationInvalid} || ' ' ),
            Data           => \%NewTo,
            Multiple       => 0,
            Size           => 0,
            Name           => 'Dest',
            TreeView       => $TreeView,
            SelectedID     => $Param{FromSelected},
            OnChangeSubmit => 0,
        );
    }
    else {
        $Param{FromStrg} = $LayoutObject->BuildSelection(
            Data        => \%NewTo,
            Class       => 'Validate_Required Modernize FormUpdate ' . ( $Param{Errors}->{DestinationInvalid} || ' ' ),
            Name        => 'Dest',
            Translation => $TreeView,
            TreeView    => $TreeView,
            SelectedID  => $Param{FromSelected},
        );
    }

    # customer info string
    if ( $ConfigObject->Get('Ticket::Frontend::CustomerInfoCompose') ) {
        $Param{CustomerTable} = $LayoutObject->AgentCustomerViewTable(
            Data => $Param{CustomerData},
            Max  => $ConfigObject->Get('Ticket::Frontend::CustomerInfoComposeMaxSize'),
        );
        $LayoutObject->Block(
            Name => 'CustomerTable',
            Data => \%Param,
        );
    }

    # prepare errors!
    if ( $Param{Errors} ) {
        for my $KeyError ( sort keys %{ $Param{Errors} } ) {
            $Param{$KeyError} = $LayoutObject->Ascii2Html( Text => $Param{Errors}->{$KeyError} );
        }
    }

    # From external
    my $ShowErrors = 1;
    if (
        defined $Param{FromExternalCustomer} &&
        defined $Param{FromExternalCustomer}->{Email} &&
        defined $Param{FromExternalCustomer}->{Customer}
        )
    {
        $ShowErrors = 0;
        $LayoutObject->Block(
            Name => 'FromExternalCustomer',
            Data => $Param{FromExternalCustomer},
        );

        $LayoutObject->AddJSData(
            Key   => 'DataEmail',
            Value => $Param{FromExternalCustomer}->{Email},
        );
        $LayoutObject->AddJSData(
            Key   => 'DataCustomer',
            Value => $Param{FromExternalCustomer}->{Customer},
        );
    }

    # Cc
    my $CustomerCounterCc = 0;
    if ( $Param{MultipleCustomerCc} ) {
        for my $Item ( @{ $Param{MultipleCustomerCc} } ) {
            if ( !$ShowErrors ) {

                # set empty values for errors
                $Item->{CustomerError}    = '';
                $Item->{CustomerDisabled} = '';
                $Item->{CustomerErrorMsg} = 'CustomerGenericServerErrorMsg';
            }
            $LayoutObject->Block(
                Name => 'CcMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Cc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CcCustomerErrorExplantion',
                );
            }
            $CustomerCounterCc++;
        }
    }

    if ( !$CustomerCounterCc ) {
        $Param{CcCustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'CcMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterCc++,
        },
    );

    # Bcc
    my $CustomerCounterBcc = 0;
    if ( $Param{MultipleCustomerBcc} ) {
        for my $Item ( @{ $Param{MultipleCustomerBcc} } ) {
            if ( !$ShowErrors ) {

                # set empty values for errors
                $Item->{CustomerError}    = '';
                $Item->{CustomerDisabled} = '';
                $Item->{CustomerErrorMsg} = 'CustomerGenericServerErrorMsg';
            }
            $LayoutObject->Block(
                Name => 'BccMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Bcc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'BccCustomerErrorExplantion',
                );
            }
            $CustomerCounterBcc++;
        }
    }

    if ( !$CustomerCounterBcc ) {
        $Param{BccCustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'BccMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterBcc++,
        },
    );

    # To
    my $CustomerCounter = 0;
    if ( $Param{MultipleCustomer} ) {
        for my $Item ( @{ $Param{MultipleCustomer} } ) {
            if ( !$ShowErrors ) {

                # set empty values for errors
                $Item->{CustomerError}    = '';
                $Item->{CustomerDisabled} = '';
                $Item->{CustomerErrorMsg} = 'CustomerGenericServerErrorMsg';
            }
            $LayoutObject->Block(
                Name => 'MultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CustomerErrorExplantion',
                );
            }
            $CustomerCounter++;
        }
    }

    if ( !$CustomerCounter ) {
        $Param{CustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'MultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounter++,
        },
    );

    if ( $Param{ToInvalid} && $Param{Errors} && !$Param{Errors}->{ToErrorType} ) {
        $LayoutObject->Block( Name => 'ToServerErrorMsg' );
    }
    if ( $Param{Errors}->{ToErrorType} || !$ShowErrors ) {
        $Param{ToInvalid} = '';
    }

    if ( $Param{CcInvalid} && $Param{Errors} && !$Param{Errors}->{CcErrorType} ) {
        $LayoutObject->Block(
            Name => 'CcServerErrorMsg',
        );
    }
    if ( $Param{Errors}->{CcErrorType} || !$ShowErrors ) {
        $Param{CcInvalid} = '';
    }

    if ( $Param{BccInvalid} && $Param{Errors} && !$Param{Errors}->{BccErrorType} ) {
        $LayoutObject->Block(
            Name => 'BccServerErrorMsg',
        );
    }
    if ( $Param{Errors}->{BccErrorType} || !$ShowErrors ) {
        $Param{BccInvalid} = '';
    }

    # build type string
    if ( $ConfigObject->Get('Ticket::Type') ) {
        $Param{TypeStrg} = $LayoutObject->BuildSelection(
            Class        => 'Validate_Required Modernize FormUpdate ' . ( $Param{Errors}->{TypeInvalid} || ' ' ),
            Data         => $Param{Types},
            Name         => 'TypeID',
            SelectedID   => $Param{TypeID},
            PossibleNone => 1,
            Sort         => 'AlphanumericValue',
            Translation  => 1,
        );
        $LayoutObject->Block(
            Name => 'TicketType',
            Data => {%Param},
        );
    }

    # build service string
    if ( $ConfigObject->Get('Ticket::Service') ) {

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

        $Param{SLAStrg} = $LayoutObject->BuildSelection(
            Data       => $Param{SLAs},
            Name       => 'SLAID',
            SelectedID => $Param{SLAID},
            Class      => 'Modernize FormUpdate '
                . ( $Config->{SLAMandatory} ? 'Validate_Required ' : '' )
                . ( $Param{Errors}->{SLAInvalid} || '' ),
            PossibleNone => 1,
            Sort         => 'AlphanumericValue',
            Translation  => 1,
            Max          => 200,
        );
        $LayoutObject->Block(
            Name => 'TicketSLA',
            Data => {
                SLAMandatory => $Config->{SLAMandatory} || 0,
                %Param,
            },
        );
    }

    # check if exists create templates regardless the queue
    my %StandardTemplates = $Kernel::OM->Get('Kernel::System::StandardTemplate')->StandardTemplateList(
        Valid => 1,
        Type  => 'Create',
    );

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

    # build priority string
    if ( !$Param{PriorityID} ) {
        $Param{Priority} = $Config->{Priority};
    }
    $Param{PriorityStrg} = $LayoutObject->BuildSelection(
        Class         => 'Modernize FormUpdate',
        Data          => $Param{Priorities},
        Name          => 'PriorityID',
        SelectedID    => $Param{PriorityID},
        SelectedValue => $Param{Priority},
        Translation   => 1,
    );

    my $QuickDateButtons = $Config->{QuickDateButtons} // $ConfigObject->Get('Ticket::Frontend::DefaultQuickDateButtons');

    # pending data string
    $Param{PendingDateString} = $LayoutObject->BuildDateSelection(
        %Param,
        Format               => 'DateInputFormatLong',
        YearPeriodPast       => 0,
        YearPeriodFuture     => 5,
        DiffTime             => $ConfigObject->Get('Ticket::Frontend::PendingDiffTime') || 0,
        Class                => $Param{Errors}->{DateInvalid}                           || ' ',
        Validate             => 1,
        ValidateDateInFuture => 1,
        QuickDateButtons     => $QuickDateButtons,
    );

    # show owner selection
    if ( $ConfigObject->Get('Ticket::Frontend::NewOwnerSelection') ) {
        $LayoutObject->Block(
            Name => 'OwnerSelection',
            Data => \%Param,
        );
    }

    # show responsible selection
    if (
        $ConfigObject->Get('Ticket::Responsible')
        && $ConfigObject->Get('Ticket::Frontend::NewResponsibleSelection')
        )
    {
        $Param{ResponsibleUsers}->{''} = '-';
        $Param{ResponsibleOptionStrg} = $LayoutObject->BuildSelection(
            Data       => $Param{ResponsibleUsers},
            SelectedID => $Param{ResponsibleUserSelected},
            Name       => 'NewResponsibleID',
            Class      => 'Modernize FormUpdate',
        );
        $LayoutObject->Block(
            Name => 'ResponsibleSelection',
            Data => \%Param,
        );
    }

    # render dynamic fields
    $Param{DynamicFieldHTML} = $Kernel::OM->Get('Kernel::Output::HTML::DynamicField::Mask')->EditSectionRender(
        Content              => $Self->{MaskDefinition},
        DynamicFields        => $Self->{DynamicField},
        LayoutObject         => $LayoutObject,
        ParamObject          => $Kernel::OM->Get('Kernel::System::Web::Request'),
        DynamicFieldValues   => $Param{DynamicField},
        PossibleValuesFilter => $Param{DFPossibleValues},
        Errors               => $Param{DFErrors},
        Visibility           => $Param{Visibility},
        Object               => {
            CustomerID     => $Param{CustomerID},
            CustomerUserID => $Param{CustomerUser},
            UserID         => $Self->{UserID},
            $Param{DynamicField}->%*,
        },
    );

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

    # Show the customer user address book if the module is registered.
    if ( $ConfigObject->Get('Frontend::Module')->{AgentCustomerUserAddressBook} ) {
        $Param{OptionCustomerUserAddressBook} = 1;
    }

    # show customer edit link
    my $OptionCustomer = $LayoutObject->Permission(
        Action => 'AdminCustomerUser',
        Type   => 'rw',
    );

    my $ShownOptionsBlock;

    if ($OptionCustomer) {

        # check if need to call Options block
        if ( !$ShownOptionsBlock ) {
            $LayoutObject->Block(
                Name => 'TicketOptions',
                Data => {
                    %Param,
                },
            );

            # set flag to "true" in order to prevent calling the Options block again
            $ShownOptionsBlock = 1;
        }

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

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

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

    # explanatory message about asterisk
    if ( $ConfigObject->Get('Ticket::Frontend::AsteriskExplanation') ) {
        $LayoutObject->Block(
            Name => 'AsteriskExplanation',
        );
    }

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

1;
</File>
        <File Location="Custom/Kernel/Modules/AgentTicketEmailOutbound.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/Modules/AgentTicketEmailOutbound.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::Modules::AgentTicketEmailOutbound;

use strict;
use warnings;

# core modules

# CPAN modules

# OTOBO modules
use Kernel::System::VariableCheck qw(:all);
use Kernel::Language              qw(Translatable);
use Mail::Address                 ();

our $ObjectManagerDisabled = 1;

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

    # allocate new hash for object
    my $Self = bless {%Param}, $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 config for frontend module
    my $Config = $Kernel::OM->Get('Kernel::Config')->Get("Ticket::Frontend::$Self->{Action}");

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

    # get the dynamic fields for this screen
    my $DynamicFieldList = $DynamicFieldObject->DynamicFieldListGet(
        Valid       => 1,
        ObjectType  => [ 'Ticket', 'Article' ],
        FieldFilter => $Config->{DynamicField} || {},
    );

    my $Definition = $Kernel::OM->Get('Kernel::System::Ticket::Mask')->DefinitionGet(
        Mask => $Self->{Action},
    ) || {};

    $Self->{DynamicField}   = [];
    $Self->{MaskDefinition} = $Definition->{Mask};

    # align sysconfig and ticket mask data I
    for my $DynamicField ( @{ $DynamicFieldList // [] } ) {
        if ( exists $Definition->{DynamicFields}{ $DynamicField->{Name} } ) {
            my $Parameters = delete $Definition->{DynamicFields}{ $DynamicField->{Name} } // {};

            for my $Attribute ( keys $Parameters->%* ) {
                $DynamicField->{$Attribute} = $Parameters->{$Attribute};
            }
        }
        else {
            push $Self->{MaskDefinition}->@*, {
                DF        => $DynamicField->{Name},
                Mandatory => $Config->{DynamicField}{ $DynamicField->{Name} } == 2 ? 1 : 0,
            };

            if ( $Config->{DynamicField}{ $DynamicField->{Name} } == 2 ) {
                $DynamicField->{Mandatory} = 1;
            }
        }

        push $Self->{DynamicField}->@*, $DynamicField;
    }

    # align sysconfig and ticket mask data II
    for my $DynamicFieldName ( keys $Definition->{DynamicFields}->%* ) {
        push $Self->{DynamicField}->@*, $DynamicFieldObject->DynamicFieldGet(
            Name => $DynamicFieldName,
        );

        my $Parameters = $Definition->{DynamicFields}{$DynamicFieldName} // {};

        for my $Attribute ( keys $Parameters->%* ) {
            $Self->{DynamicField}[-1]{$Attribute} = $Parameters->{$Attribute};
        }
    }

    # get form id
    $Self->{FormID} = $Kernel::OM->Get('Kernel::System::Web::FormCache')->PrepareFormID(
        ParamObject  => $Kernel::OM->Get('Kernel::System::Web::Request'),
        LayoutObject => $Kernel::OM->Get('Kernel::Output::HTML::Layout'),
    );

    $Self->{Debug} = $Param{Debug} || 0;

    return $Self;
}

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

    my $Output;

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

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

    # fetch the actual acl action data
    my %AclAction = $TicketObject->TicketAclActionData();

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

    # check if ACL restrictions exist
    if ($ACL) {

        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 (
        $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'LoadFormDraft' )
        && !$Self->{LoadedFormDraftID}
        )
    {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Loading draft failed!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    if ( $Self->{Subaction} eq 'SendEmail' ) {

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

        $Output = $Self->SendEmail();
    }
    elsif ( $Self->{Subaction} eq 'AJAXUpdateTemplate' ) {

        my %GetParam;
        for my $Key (
            qw(
                NewStateID NewPriorityID TimeUnits IsVisibleForCustomer Title Body Subject NewQueueID
                Year Month Day Hour Minute NewOwnerID NewOwnerType OldOwnerID NewResponsibleID
                TypeID ServiceID SLAID Expand ReplyToArticle StandardTemplateID CreateArticle
                ElementChanged
            )
            )
        {
            $GetParam{$Key} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Key );
        }

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

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

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

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

        my $NextStates = $Self->_GetNextStates(
            %GetParam,
            CustomerUserID => $CustomerUser || '',
            QueueID        => $QueueID,
        );

        # 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 ( @{ $Self->{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 $StandardTemplates = $Self->_GetStandardTemplates(
            %GetParam,
            QueueID => $QueueID || '',
        );

        my @TemplateAJAX;

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

        # update ticket body and attachments if needed.
        if ( $GetParam{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
            my $TemplateGenerator = $Kernel::OM->Get('Kernel::System::TemplateGenerator');

            if ( IsPositiveInteger( $GetParam{StandardTemplateID} ) ) {

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

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

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

                # send a list of attachments in the upload cache back to the client side 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},
                    );
                }
            }

            # Get the first article of the ticket.
            my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');
            my @MetaArticles  = $ArticleObject->ArticleList(
                TicketID  => $Self->{TicketID},
                UserID    => $Self->{UserID},
                OnlyFirst => 1,
            );
            my %Article = $ArticleObject->BackendForArticle( %{ $MetaArticles[0] } )->ArticleGet(
                %{ $MetaArticles[0] },
                DynamicFields => 0,
            );

            # get the matching signature for the current user
            my $Signature = $TemplateGenerator->Signature(
                TicketID  => $Self->{TicketID},
                ArticleID => $Article{ArticleID},
                Data      => \%Article,
                UserID    => $Self->{UserID},
            );

            # append the signature to the body text
            if ( $LayoutObject->{BrowserRichText} ) {
                $TemplateText = $TemplateText . '<br><br>' . $Signature;
            }
            else {
                $TemplateText = $TemplateText . "\n\n" . $Signature;
            }

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

        my $JSON = $LayoutObject->BuildSelectionJSON(
            [
                {
                    Name         => 'NewStateID',
                    Data         => $NextStates,
                    SelectedID   => $GetParam{NewStateID},
                    Translation  => 1,
                    PossibleNone => $Config->{StateDefault} ? 0 : 1,
                    Max          => 100,
                },
                {
                    Name         => 'StandardTemplateID',
                    Data         => $StandardTemplates,
                    SelectedID   => $GetParam{StandardTemplateID},
                    PossibleNone => 1,
                    Translation  => 1,
                    Max          => 100,
                },
                @TemplateAJAX,
            ],
        );

        return $LayoutObject->Attachment(
            ContentType => 'application/json',
            Content     => $JSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }
    elsif ( $Self->{Subaction} eq 'AJAXUpdate' ) {
        $Output = $Self->AjaxUpdate();
    }
    elsif ( $Self->{LoadedFormDraftID} ) {
        $Output = $Self->SendEmail();
    }
    else {
        $Output = $Self->Form();
    }

    return $Output;
}

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

    my %Error;
    my %ACLCompatGetParam;

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

    # ACL compatibility translation
    $ACLCompatGetParam{NextStateID} = $ParamObject->GetParam( Param => 'NextStateID' );

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

    # check needed stuff
    if ( !$Self->{TicketID} ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Got no TicketID!'),
            Comment => Translatable('System Error!'),
        );
    }

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

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

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

    # get config for 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( WithHeader => 'yes' );
    }

    my %GetParamExtended = $Self->_GetExtendedParams();

    my %GetParam            = %{ $GetParamExtended{GetParam} };
    my @MultipleCustomer    = @{ $GetParamExtended{MultipleCustomer} };
    my @MultipleCustomerCc  = @{ $GetParamExtended{MultipleCustomerCc} };
    my @MultipleCustomerBcc = @{ $GetParamExtended{MultipleCustomerBcc} };

    # 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 ) {
                return join '',
                    $LayoutObject->Header(
                        Type      => 'Small',
                        BodyClass => 'Popup',
                    ),
                    $LayoutObject->Warning(
                        Message => Translatable('Sorry, you need to be the ticket owner to perform this action.'),
                        Comment => Translatable('Please change the owner first.'),
                    ),
                    $LayoutObject->Footer(
                        Type => 'Small',
                    );
            }
            else {
                $LayoutObject->Block(
                    Name => 'TicketBack',
                    Data => {
                        %Param,
                        TicketID => $Self->{TicketID},
                    },
                );
            }
        }
    }
    else {
        $LayoutObject->Block(
            Name => 'TicketBack',
            Data => {
                %Param,
                TicketID => $Self->{TicketID},
            },
        );
    }

    # Get selected or last customer article.
    my %Data;
    if ( $GetParam{ArticleID} ) {
        my $ArticleBackendObject = $ArticleObject->BackendForArticle(
            TicketID  => $Self->{TicketID},
            ArticleID => $GetParam{ArticleID},
        );
        %Data = $ArticleBackendObject->ArticleGet(
            TicketID      => $Self->{TicketID},
            ArticleID     => $GetParam{ArticleID},
            DynamicFields => 1,
        );

        # Check if article is from the same TicketID as we checked permissions for.
        if ( $Data{TicketID} ne $Self->{TicketID} ) {

            return $LayoutObject->ErrorScreen(
                Message => $LayoutObject->{LanguageObject}->Translate( 'Article does not belong to ticket %s!', $Self->{TicketID} ),
            );
        }
    }
    else {

        # Get last customer article.
        my @MetaArticles = $ArticleObject->ArticleList(
            TicketID   => $Self->{TicketID},
            SenderType => 'customer',
            OnlyLast   => 1,
            UserID     => $Self->{UserID},
        );
        if (@MetaArticles) {
            %Data = $ArticleObject->BackendForArticle( %{ $MetaArticles[0] } )->ArticleGet(
                TicketID => $Self->{TicketID},
                %{ $MetaArticles[0] },
                DynamicFields => 0,
            );
        }
    }

    # prepare signature
    my $TemplateGenerator = $Kernel::OM->Get('Kernel::System::TemplateGenerator');
    $Data{Signature} = $TemplateGenerator->Signature(
        TicketID  => $Self->{TicketID},
        ArticleID => $Data{ArticleID},
        Data      => \%Data,
        UserID    => $Self->{UserID},
    );

    if ( $GetParam{EmailTemplateID} ) {

        # get template
        $Data{StdTemplate} = $TemplateGenerator->Template(
            TicketID   => $Self->{TicketID},
            ArticleID  => $Data{ArticleID},
            TemplateID => $GetParam{EmailTemplateID},
            Data       => \%Data,
            UserID     => $Self->{UserID},
        );

        # get signature
        $Data{Signature} = $TemplateGenerator->Signature(
            TicketID => $Self->{TicketID},
            Data     => \%Data,
            UserID   => $Self->{UserID},
        );
    }

    # empty the body for preparation
    $Data{Body} = '';

    # prepare body for richtext or plaintext content
    if ( $LayoutObject->{BrowserRichText} ) {

        # add the temp
        if ( $GetParam{EmailTemplateID} ) {
            $Data{Body} = $Data{StdTemplate} . '<br/>' . $Data{Body};
        }

        $Data{Body} = $Data{Body} . $Data{Signature};
    }
    else {

        # add the temp
        if ( $GetParam{EmailTemplateID} ) {
            $Data{Body} = $Data{StdTemplate} . "\n" . $Data{Body};
        }

        $Data{Body} = $Data{Body} . $Data{Signature};

        # prepare body for plain text
        $Data{Body} =~ s/\t/ /g;
    }

    # get needed objects
    my $StdAttachmentObject = $Kernel::OM->Get('Kernel::System::StdAttachment');
    my $UploadCacheObject   = $Kernel::OM->Get('Kernel::System::Web::UploadCache');

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

    # get all attachments meta data
    my @Attachments = $Kernel::OM->Get('Kernel::System::Web::UploadCache')->FormIDGetAllFilesMeta(
        FormID => $Self->{FormID},
    );

    # check some values
    for my $Recipient (qw(To Cc Bcc Subject)) {
        if ( $Data{$Recipient} ) {
            delete $Data{$Recipient};
        }
    }

    # put & get attributes like sender address
    %Data = $TemplateGenerator->Attributes(
        TicketID  => $Self->{TicketID},
        ArticleID => $GetParam{ArticleID},
        Data      => \%Data,
        UserID    => $Self->{UserID},
    );

    # prepare the subject
    $Data{Subject} =~ s{ \A (?: .*? : \s*) }{}xms;

    # run compose modules
    if ( ref( $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') ) eq 'HASH' ) {

        # use ticket QueueID in compose modules
        $GetParam{QueueID} = $Ticket{QueueID};

        my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
        for my $Job ( sort keys %Jobs ) {

            # load module
            if ( !$Kernel::OM->Get('Kernel::System::Main')->Require( $Jobs{$Job}->{Module} ) ) {
                return $LayoutObject->FatalError();
            }
            my $Object = $Jobs{$Job}->{Module}->new(
                %{$Self},
                Debug => $Self->{Debug},
            );

            # get params
            PARAMETER:
            for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                    @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                    next PARAMETER;
                }

                $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
            }

            # run module
            my $NewParams = $Object->Run( %Data, %GetParam, Config => $Jobs{$Job} );

            if ($NewParams) {
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                    $GetParam{$Parameter} = $NewParams;
                }
            }

            # get errors
            %Error = ( %Error, $Object->Error( %GetParam, Config => $Jobs{$Job} ) );
        }
    }

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

    # remember dynamic field validation results if erroneous
    my %DynamicFieldPossibleValues;

    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( $Self->{DynamicField}->@* ) {
        next DYNAMICFIELD unless IsHashRefWithData($DynamicFieldConfig);

        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,
                    %ACLCompatGetParam,
                    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;
                }
            }
        }

        $DynamicFieldPossibleValues{ 'DynamicField_' . $DynamicFieldConfig->{Name} } = $PossibleValuesFilter;
    }

    # extract dynamic field values from ticket data
    my %TicketDFValues =
        map  { 'DynamicField_' . $_->{Name} => $Ticket{ 'DynamicField_' . $_->{Name} } }
        grep { $_->{ObjectType} eq 'Ticket' }
        $Self->{DynamicField}->@*;

    # build view
    # start with page
    my $Output = $LayoutObject->Header(
        Value     => $Ticket{TicketNumber},
        Type      => 'Small',
        BodyClass => 'Popup',
    );

    # Inform a user that article subject will be empty if contains only the ticket hook (if nothing is modified).
    $Output .= $LayoutObject->Notify(
        Data => $LayoutObject->{LanguageObject}->Translate('Article subject will be empty if the subject contains only the ticket hook!'),
    );

    $Output .= $Self->_Mask(
        TicketNumber   => $Ticket{TicketNumber},
        TicketID       => $Self->{TicketID},
        Title          => $Ticket{Title},
        CustomerID     => $Ticket{CustomerID},
        CustomerUserID => $Ticket{CustomerUserID},
        QueueID        => $Ticket{QueueID},
        NextStates     => $Self->_GetNextStates(
            %GetParam,
            %ACLCompatGetParam,
        ),
        TimeUnitsRequired => (
            $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
            ? 'Validate_Required'
            : ''
        ),
        Errors              => \%Error,
        MultipleCustomer    => \@MultipleCustomer,
        MultipleCustomerCc  => \@MultipleCustomerCc,
        MultipleCustomerBcc => \@MultipleCustomerBcc,
        Attachments         => \@Attachments,
        %Data,
        %GetParam,
        DFPossibleValues => \%DynamicFieldPossibleValues,
        DFValues         => \%TicketDFValues,
    );
    $Output .= $LayoutObject->Footer(
        Type => 'Small',
    );

    return $Output;
}

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

    my %Error;
    my %ACLCompatGetParam;

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

    # ACL compatibility translation
    $ACLCompatGetParam{NextStateID} = $ParamObject->GetParam( Param => 'NextStateID' );

    my %GetParamExtended = $Self->_GetExtendedParams();

    my %GetParam            = %{ $GetParamExtended{GetParam} };
    my @MultipleCustomer    = @{ $GetParamExtended{MultipleCustomer} };
    my @MultipleCustomerCc  = @{ $GetParamExtended{MultipleCustomerCc} };
    my @MultipleCustomerBcc = @{ $GetParamExtended{MultipleCustomerBcc} };

    my %DynamicFieldValues;

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

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

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

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

            # meddling with the innards of Kernel::System::Web::Request
            $ParamObject->SetArray(
                Param  => 'FormDraftID',
                Values => ['']
            );
        }

        my $FormDraftActionOk;
        if (
            $FormDraftAction eq 'Add'
            ||
            ( $FormDraftAction eq 'Update' && $GetParam{FormDraftID} )
            )
        {
            $FormDraftActionOk = $ParamObject->SaveFormDraft(
                UserID     => $Self->{UserID},
                ObjectType => 'Ticket',
                ObjectID   => $Self->{TicketID},
            );
        }
        elsif ( $FormDraftAction eq 'Delete' && $GetParam{FormDraftID} ) {
            $FormDraftActionOk = $Kernel::OM->Get('Kernel::System::FormDraft')->FormDraftDelete(
                FormDraftID => $GetParam{FormDraftID},
                UserID      => $Self->{UserID},
            );
        }

        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',
            Content     => $JSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }

    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{ $Self->{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;

    my $QueueID = $Self->{QueueID};
    my %StateData;

    if ( $GetParam{ComposeStateID} ) {
        %StateData = $Kernel::OM->Get('Kernel::System::State')->StateGet(
            ID => $GetParam{ComposeStateID},
        );
    }

    my $NextState = $StateData{Name};

    # check pending date
    if ( defined $StateData{TypeName} && $StateData{TypeName} =~ /^pending/i ) {

        # convert pending date to a 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');

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

    # check To
    if ( !$GetParam{To} ) {
        $Error{'ToInvalid'} = 'ServerError';
    }

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

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

    if (
        $ConfigObject->Get('Ticket::Frontend::AccountTime')
        && $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
        && $GetParam{TimeUnits} eq ''
        )
    {
        $Error{'TimeUnitsInvalid'} = 'ServerError';
    }

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

    # get information on the current ticket
    my %Ticket = $TicketObject->TicketGet(
        TicketID      => $Self->{TicketID},
        DynamicFields => 1,
    );

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

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

    # remember dynamic field validation results if erroneous
    my %DynamicFieldValidationResult;
    my %DynamicFieldPossibleValues;

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

        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,
                    %ACLCompatGetParam,
                    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;
                }
            }
        }

        $DynamicFieldPossibleValues{ 'DynamicField_' . $DynamicFieldConfig->{Name} } = $PossibleValuesFilter;

        my $ValidationResult = $DynamicFieldBackendObject->EditFieldValueValidate(
            DynamicFieldConfig   => $DynamicFieldConfig,
            PossibleValuesFilter => $PossibleValuesFilter,
            ParamObject          => $ParamObject,

            # Mandatory is added to the configs by $Self->new
            Mandatory => $DynamicFieldConfig->{Mandatory},
        );

        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';
            $DynamicFieldValidationResult{ $DynamicFieldConfig->{Name} } = $ValidationResult;
        }

    }

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

    # get check item object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    # check some values
    LINE:
    for my $Line (qw(To Cc Bcc)) {
        next LINE if !$GetParam{$Line};
        for my $Email ( Mail::Address->parse( $GetParam{$Line} ) ) {
            if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                $Error{ $Line . 'ErrorType' } = $Line . $CheckItemObject->CheckErrorType() . 'ServerErrorMsg';
                $Error{ $Line . 'Invalid' }   = 'ServerError';
            }
            my $IsLocal = $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress(
# Rother OSS / DiscreteSystemAddresses
                TicketID => $Self->{TicketID},
# EO DiscreteSystemAddresses
                Address => $Email->address(),
            );
            if ($IsLocal) {
                $Error{ $Line . 'IsLocalAddress' } = 'ServerError';
            }
        }
    }

    # Make sure sender is correct one. See bug#14872 ( https://bugs.otrs.org/show_bug.cgi?id=14872 ).
    $GetParam{From} = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->Sender(
        QueueID => $Ticket{QueueID},
        UserID  => $Self->{UserID},
    );

    if ( $Self->{LoadedFormDraftID} ) {

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

    # run compose modules
    my %ArticleParam;
    if ( ref( $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') ) eq 'HASH' ) {
        my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
        for my $Job ( sort keys %Jobs ) {

            # load module
            if ( !$Kernel::OM->Get('Kernel::System::Main')->Require( $Jobs{$Job}->{Module} ) ) {
                return $LayoutObject->FatalError();
            }
            my $Object = $Jobs{$Job}->{Module}->new(
                %{$Self},
                Debug => $Self->{Debug},
            );

            my $Multiple;

            # get params
            PARAMETER:
            for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                    @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                    $Multiple = 1;
                    next PARAMETER;
                }

                $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
            }

            # run module
            $Object->Run(
                %GetParam,
                StoreNew => 1,
                Config   => $Jobs{$Job},
            );

            # get options that have been removed from the selection
            # and add them back to the selection so that the submit
            # will contain options that were hidden from the agent
            my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );

            if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                my @RemovedOptions = $Object->GetOptionsToRemoveAJAX(%GetParam);
                if (@RemovedOptions) {
                    if ($Multiple) {
                        for my $RemovedOption (@RemovedOptions) {
                            push @{ $GetParam{$Key} }, $RemovedOption;
                        }
                    }
                    else {
                        $GetParam{$Key} = shift @RemovedOptions;
                    }
                }
            }

            # ticket params
            %ArticleParam = (
                %ArticleParam,
                $Object->ArticleOption(
                    %GetParam,
                    %ArticleParam,
                    Config => $Jobs{$Job},
                ),
            );

            # get errors
            %Error = (
                %Error,
                $Object->Error(
                    %GetParam,
                    Config => $Jobs{$Job},
                ),
            );
        }
    }

    # Check whether there is an error.
    # An error is not necessarily bad, often it is just that a form draft has been loaded.
    if (%Error) {
        my $QueueID = $TicketObject->TicketQueueID( TicketID => $Self->{TicketID} );

        # extract dynamic field values from ticket data
        my %TicketDFValues =
            map  { 'DynamicField_' . $_->{Name} => $Ticket{ 'DynamicField_' . $_->{Name} } }
            grep { $_->{ObjectType} eq 'Ticket' }
            $Self->{DynamicField}->@*;

        my $Output = $LayoutObject->Header(
            Type      => 'Small',
            BodyClass => 'Popup',
        );

        # When a draft is loaded, inform a user that article subject will be empty
        # if contains only the ticket hook (if nothing is modified).
        if ( $Error{LoadedFormDraft} ) {
            $Output .= $LayoutObject->Notify(
                Data => $LayoutObject->{LanguageObject}->Translate(
                    'Article subject will be empty if the subject contains only the ticket hook!'
                ),
            );
        }

        $Output .= $Self->_Mask(
            TicketNumber   => $Ticket{TicketNumber},
            Title          => $Ticket{Title},
            CustomerID     => $Ticket{CustomerID},
            CustomerUserID => $Ticket{CustomerUserID},
            TicketID       => $Self->{TicketID},
            QueueID        => $QueueID,
            NextStates     => $Self->_GetNextStates(
                %GetParam,
                %ACLCompatGetParam,
            ),
            Errors              => \%Error,
            MultipleCustomer    => \@MultipleCustomer,
            MultipleCustomerCc  => \@MultipleCustomerCc,
            MultipleCustomerBcc => \@MultipleCustomerBcc,
            Attachments         => \@Attachments,
            %GetParam,
            DFPossibleValues => \%DynamicFieldPossibleValues,
            DFErrors         => \%DynamicFieldValidationResult,
            DFValues         => \%TicketDFValues,
        );
        $Output .= $LayoutObject->Footer(
            Type => 'Small',
        );

        return $Output;
    }

    # replace <OTOBO_TICKET_STATE> with next ticket state name
    if ($NextState) {
        $GetParam{Body} =~ s/(&lt;|<)OTOBO_TICKET_STATE(&gt;|>)/$NextState/g;
    }

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

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

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

        # remove unused inline images
        my @NewAttachmentData;
        ATTACHMENT:
        for my $Attachment (@AttachmentData) {
            my $ContentID = $Attachment->{ContentID};
            if ( $ContentID && ( $Attachment->{ContentType} =~ /image/i ) ) {
                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};
        }
        @AttachmentData = @NewAttachmentData;

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

    # send email
    my $To = '';
    KEY:
    for my $Key (qw(To Cc Bcc)) {
        next KEY if !$GetParam{$Key};
        if ($To) {
            $To .= ', ';
        }
        $To .= $GetParam{$Key};
    }

    # Get attributes like sender address.
    my %Data = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->Attributes(
        TicketID => $Self->{TicketID},
        Data     => {},
        UserID   => $Self->{UserID},
    );

    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');
    my $ArticleID     = $ArticleObject->BackendForChannel( ChannelName => 'Email' )->ArticleSend(
        SenderType           => 'agent',
        IsVisibleForCustomer => $GetParam{IsVisibleForCustomer} // 0,
        TicketID             => $Self->{TicketID},
        HistoryType          => 'EmailAgent',
        HistoryComment       => "\%\%$To",
        From                 => $Data{From},
        To                   => $GetParam{To},
        Cc                   => $GetParam{Cc},
        Bcc                  => $GetParam{Bcc},
        Subject              => $GetParam{Subject},
        UserID               => $Self->{UserID},
        Body                 => $GetParam{Body},

        # We start a new communication here, so don't send any references.
        #   This might lead to information disclosure (domain names; see bug#11246).
        InReplyTo  => '',
        References => '',
        Charset    => $LayoutObject->{UserCharset},
        MimeType   => $MimeType,
        Attachment => \@AttachmentData,
        %ArticleParam,
    );

    # error page
    if ( !$ArticleID ) {
        return $LayoutObject->ErrorScreen(
            Comment => Translatable('Please contact the administrator.'),
        );
    }

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

    # set dynamic fields
    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{ $Self->{DynamicField} } ) {
        next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);
        next DYNAMICFIELD if $DynamicFieldConfig->{Readonly};

        # set the object ID (TicketID or ArticleID) depending on the field configuration
        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},
        );
    }

    # set state
    if ($NextState) {
        $TicketObject->TicketStateSet(
            TicketID  => $Self->{TicketID},
            ArticleID => $ArticleID,
            State     => $NextState,
            UserID    => $Self->{UserID},
        );

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

        # set pending time
        elsif ( $StateData{TypeName} =~ /^pending/i ) {
            $TicketObject->TicketPendingTimeSet(
                UserID   => $Self->{UserID},
                TicketID => $Self->{TicketID},
                %GetParam,
            );
        }
    }

    # remove all form data
    $Kernel::OM->Get('Kernel::System::Web::FormCache')->FormIDRemove( FormID => $Self->{FormID} );

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

    # redirect
    if (
        defined $StateData{TypeName}
        && $StateData{TypeName} =~ /^close/i
        && !$ConfigObject->Get('Ticket::Frontend::RedirectAfterCloseDisabled')
        )
    {
        return $LayoutObject->PopupClose(
            URL => ( $Self->{LastScreenOverview} || 'Action=AgentDashboard' ),
        );
    }

    return $LayoutObject->PopupClose(
        URL => "Action=AgentTicketZoom;TicketID=$Self->{TicketID};ArticleID=$ArticleID",
    );
}

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

    my %Error;
    my %ACLCompatGetParam;

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

    # ACL compatibility translation
    $ACLCompatGetParam{NextStateID} = $ParamObject->GetParam( Param => 'NextStateID' );

    my %GetParamExtended = $Self->_GetExtendedParams();

    my %GetParam            = %{ $GetParamExtended{GetParam} };
    my @MultipleCustomer    = @{ $GetParamExtended{MultipleCustomer} };
    my @MultipleCustomerCc  = @{ $GetParamExtended{MultipleCustomerCc} };
    my @MultipleCustomerBcc = @{ $GetParamExtended{MultipleCustomerBcc} };

    my %Ticket = $Kernel::OM->Get('Kernel::System::Ticket')->TicketGet( TicketID => $Self->{TicketID} );

    # Make sure sender is correct one. See bug#14872 ( https://bugs.otrs.org/show_bug.cgi?id=14872 ).
    $GetParam{From} = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->Sender(
        QueueID => $Ticket{QueueID},
        UserID  => $Self->{UserID},
    );

    my @ExtendedData;

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

    # run compose modules
    if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {
        my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
        JOB:
        for my $Job ( sort keys %Jobs ) {

            # load module
            next JOB if !$Kernel::OM->Get('Kernel::System::Main')->Require( $Jobs{$Job}->{Module} );

            my $Object = $Jobs{$Job}->{Module}->new(
                %{$Self},
                Debug => $Self->{Debug},
            );

            my $Multiple;

            # get params
            PARAMETER:
            for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                    @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                    $Multiple = 1;
                    next PARAMETER;
                }

                $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
            }

            # run module
            my %Data = $Object->Data( %GetParam, Config => $Jobs{$Job} );

            # get AJAX param values
            if ( $Object->can('GetParamAJAX') ) {
                %GetParam = ( %GetParam, $Object->GetParamAJAX(%GetParam) );
            }

            # get options that have to be removed from the selection visible
            # to the agent. These options will be added again on submit.
            if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                my @OptionsToRemove = $Object->GetOptionsToRemoveAJAX(%GetParam);

                for my $OptionToRemove (@OptionsToRemove) {
                    delete $Data{$OptionToRemove};
                }
            }

            my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );
            if ($Key) {
                push @ExtendedData, {
                    Name         => $Key,
                    Data         => \%Data,
                    SelectedID   => $GetParam{$Key},
                    Translation  => 1,
                    PossibleNone => 1,
                    Multiple     => $Multiple,
                    Max          => 100,
                };
            }
        }
    }

    my %DynamicFieldValues;

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

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

    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{ $Self->{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;

    my $NextStates = $Self->_GetNextStates(
        %GetParam,
        %ACLCompatGetParam,
    );

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

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

    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{ $Self->{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,
            %ACLCompatGetParam,
            Action        => $Self->{Action},
            TicketID      => $Self->{TicketID},
            QueueID       => $Self->{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 $JSON = $LayoutObject->BuildSelectionJSON(
        [
            {
                Name         => 'ComposeStateID',
                Data         => $NextStates,
                SelectedID   => $GetParam{ComposeStateID},
                Translation  => 1,
                PossibleNone => 1,
                Max          => 100,
            },
            @ExtendedData,
            @DynamicFieldAJAX,
        ],
    );

    return $LayoutObject->Attachment(
        ContentType => 'application/json',
        Content     => $JSON,
        Type        => 'inline',
        NoCache     => 1,
    );
}

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

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

    return \%NextStates;
}

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

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

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

    # build next states string
    my %State;
    if ( !$Param{ComposeStateID} ) {
        $State{SelectedValue} = $Config->{StateDefault};
    }
    else {
        $State{SelectedID} = $Param{ComposeStateID};
    }

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

    $Param{NextStatesStrg} = $LayoutObject->BuildSelection(
        Data         => $Param{NextStates},
        Name         => 'ComposeStateID',
        PossibleNone => 1,
        Translation  => 1,
        Class        => 'Modernize FormUpdate',
        %State,
    );

    if ( !$Param{IsVisibleForCustomerPresent} ) {
        $Param{IsVisibleForCustomer} = $Config->{IsVisibleForCustomerDefault};
    }

    # prepare errors!
    if ( $Param{Errors} ) {
        for my $Error ( sort keys %{ $Param{Errors} } ) {
            $Param{$Error} = $LayoutObject->Ascii2Html(
                Text => $Param{Errors}->{$Error},
            );
        }
    }

    my $QuickDateButtons = $Config->{QuickDateButtons} // $ConfigObject->Get('Ticket::Frontend::DefaultQuickDateButtons');

    # pending data string
    $Param{PendingDateString} = $LayoutObject->BuildDateSelection(
        %Param,
        YearPeriodPast       => 0,
        YearPeriodFuture     => 5,
        Format               => 'DateInputFormatLong',
        DiffTime             => $ConfigObject->Get('Ticket::Frontend::PendingDiffTime') || 0,
        Class                => $Param{Errors}->{DateInvalid}                           || ' ',
        Validate             => 1,
        ValidateDateInFuture => 1,
        QuickDateButtons     => $QuickDateButtons,
    );

    # Multiple-Autocomplete
    # Cc
    my $CustomerCounterCc = 0;
    if ( $Param{MultipleCustomerCc} ) {
        for my $Item ( @{ $Param{MultipleCustomerCc} } ) {
            $LayoutObject->Block(
                Name => 'CcMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Cc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CcCustomerErrorExplantion',
                );
            }
            $CustomerCounterCc++;
        }
    }

    if ( !$CustomerCounterCc ) {
        $Param{CcCustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'CcMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterCc++,
        },
    );

    # Bcc
    my $CustomerCounterBcc = 0;
    if ( $Param{MultipleCustomerBcc} ) {
        for my $Item ( @{ $Param{MultipleCustomerBcc} } ) {
            $LayoutObject->Block(
                Name => 'BccMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Bcc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'BccCustomerErrorExplantion',
                );
            }
            $CustomerCounterBcc++;
        }
    }

    if ( !$CustomerCounterBcc ) {
        $Param{BccCustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'BccMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterBcc++,
        },
    );

    # To
    my $CustomerCounter = 0;
    if ( $Param{MultipleCustomer} ) {
        for my $Item ( @{ $Param{MultipleCustomer} } ) {
            $LayoutObject->Block(
                Name => 'MultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CustomerErrorExplantion',
                );
            }
            $CustomerCounter++;
        }
    }

    if ( !$CustomerCounter ) {
        $Param{CustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'MultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounter++,
        },
    );

    if ( $Param{ToInvalid} && $Param{Errors} && !$Param{Errors}->{ToErrorType} ) {
        $LayoutObject->Block(
            Name => 'ToServerErrorMsg',
        );
    }

    if ( $Param{ToIsLocalAddress} && $Param{Errors} && !$Param{Errors}->{ToErrorType} ) {
        $LayoutObject->Block(
            Name => 'ToIsLocalAddressServerErrorMsg',
            Data => \%Param,
        );
    }

    if ( $Param{CcInvalid} && $Param{Errors} && !$Param{Errors}->{CcErrorType} ) {
        $LayoutObject->Block(
            Name => 'CcServerErrorMsg',
        );
    }

    if ( $Param{CcIsLocalAddress} && $Param{Errors} && !$Param{Errors}->{CcErrorType} ) {
        $LayoutObject->Block(
            Name => 'CcIsLocalAddressServerErrorMsg',
            Data => \%Param,
        );
    }

    if ( $Param{BccInvalid} && $Param{Errors} && !$Param{Errors}->{BccErrorType} ) {
        $LayoutObject->Block(
            Name => 'BccServerErrorMsg',
        );
    }

    if ( $Param{BccIsLocalAddress} && $Param{Errors} && !$Param{Errors}->{BccErrorType} ) {
        $LayoutObject->Block(
            Name => 'BccIsLocalAddressServerErrorMsg',
            Data => \%Param,
        );
    }

    # render dynamic fields
    {
        my %DynamicFieldConfigs = map { $_->{Name} => $_ } $Self->{DynamicField}->@*;

        $Param{DynamicFieldHTML} = $Kernel::OM->Get('Kernel::Output::HTML::DynamicField::Mask')->EditSectionRender(
            Content              => $Self->{MaskDefinition},
            DynamicFields        => \%DynamicFieldConfigs,
            LayoutObject         => $LayoutObject,
            ParamObject          => $Kernel::OM->Get('Kernel::System::Web::Request'),
            DynamicFieldValues   => $Param{DFValues},
            PossibleValuesFilter => $Param{DFPossibleValues},
            Errors               => $Param{DFErrors},
            Object               => {
                CustomerID     => $Param{CustomerID},
                CustomerUserID => $Param{CustomerUserID},
                UserID         => $Self->{UserID},
                $Param{DFValues}->%*,
            },
        );
    }

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

    # Show the customer user address book if the module is registered.
    if ( $ConfigObject->Get('Frontend::Module')->{AgentCustomerUserAddressBook} ) {
        $Param{OptionCustomerUserAddressBook} = 1;
    }

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

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

    if (
        IsHashRefWithData(
            $QueueStandardTemplates
                || ( $Param{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 attachments
    ATTACHMENT:
    for my $Attachment ( @{ $Param{Attachments} } ) {
        if (
            $Attachment->{ContentID}
            && $LayoutObject->{BrowserRichText}
            && ( $Attachment->{ContentType} =~ /image/i )
            )
        {
            next ATTACHMENT;
        }

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

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

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

    # explanatory message about asterisk
    if ( $ConfigObject->Get('Ticket::Frontend::AsteriskExplanation') ) {
        $LayoutObject->Block(
            Name => 'AsteriskExplanation',
        );
    }

    # create & return output
    return $LayoutObject->Output(
        TemplateFile => 'AgentTicketEmailOutbound',
        Data         => {
            %Param,
            FormID         => $Self->{FormID},
            FormDraft      => $Config->{FormDraft},
            FormDraftID    => $Self->{LoadedFormDraftID},
            FormDraftTitle => $LoadedFormDraft ? $LoadedFormDraft->{Title} : '',
            FormDraftMeta  => $LoadedFormDraft,
        },
    );
}

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

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

    # get params
    my %GetParam;
    for my $Key (
        qw(To Cc Bcc Subject Body ComposeStateID IsVisibleForCustomer IsVisibleForCustomerPresent
        ArticleID TimeUnits Year Month Day Hour Minute FormDraftID Title)
        )
    {
        my $Value = $ParamObject->GetParam( Param => $Key );
        if ( defined $Value ) {
            $GetParam{$Key} = $Value;
        }
    }

    $GetParam{EmailTemplateID} = $ParamObject->GetParam( Param => 'EmailTemplateID' ) || '';

    # hash for check duplicated entries
    my %AddressesList;
    my @MultipleCustomer;
    my $CustomersNumberTo = $ParamObject->GetParam( Param => 'CustomerTicketCounterToCustomer' ) || 0;
    my $Selected          = $ParamObject->GetParam( Param => 'CustomerSelected' )                || '';

    # get check item object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    if ($CustomersNumberTo) {
        my $CustomerCounter = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberTo ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElement = $ParamObject->GetParam( Param => 'CustomerTicketText_' . $Count );

            next COUNT unless $CustomerElement;

            my $CustomerSelected = ( $Selected eq $Count ? 'checked ' : '' );
            my $CustomerKey      = $ParamObject->GetParam( Param => 'CustomerKey_' . $Count )   || '';
            my $CustomerQueue    = $ParamObject->GetParam( Param => 'CustomerQueue_' . $Count ) || '';

            if ( $GetParam{To} ) {
                $GetParam{To} .= ', ' . $CustomerElement;
            }
            else {
                $GetParam{To} = $CustomerElement;
            }

            # check email address
            my $CustomerErrorMsg = 'CustomerGenericServerErrorMsg';
            my $CustomerError    = '';
            for my $Email ( Mail::Address->parse($CustomerElement) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsg = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerError = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElement} && $CustomerError eq '' ) {
                $CustomerErrorMsg = 'IsDuplicatedServerErrorMsg';
                $CustomerError    = 'ServerError';
            }

            my $CustomerDisabled = '';
            my $CountAux         = $CustomerCounter++;
            if ( $CustomerError ne '' ) {
                $CustomerDisabled = 'disabled="disabled"';
                $CountAux         = $Count . 'Error';
            }

            if ( $CustomerQueue ne '' ) {
                $CustomerQueue = $Count;
            }

            push @MultipleCustomer, {
                Count            => $CountAux,
                CustomerElement  => $CustomerElement,
                CustomerSelected => $CustomerSelected,
                CustomerKey      => $CustomerKey,
                CustomerError    => $CustomerError,
                CustomerErrorMsg => $CustomerErrorMsg,
                CustomerDisabled => $CustomerDisabled,
                CustomerQueue    => $CustomerQueue,
            };
            $AddressesList{$CustomerElement} = 1;
        }
    }

    my @MultipleCustomerCc;
    my $CustomersNumberCc = $ParamObject->GetParam( Param => 'CustomerTicketCounterCcCustomer' ) || 0;

    if ($CustomersNumberCc) {
        my $CustomerCounterCc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberCc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementCc = $ParamObject->GetParam( Param => 'CcCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementCc;

            my $CustomerKeyCc   = $ParamObject->GetParam( Param => 'CcCustomerKey_' . $Count )   || '';
            my $CustomerQueueCc = $ParamObject->GetParam( Param => 'CcCustomerQueue_' . $Count ) || '';

            if ( $GetParam{Cc} ) {
                $GetParam{Cc} .= ', ' . $CustomerElementCc;
            }
            else {
                $GetParam{Cc} = $CustomerElementCc;
            }

            # check email address
            my $CustomerErrorMsgCc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorCc    = '';
            for my $Email ( Mail::Address->parse($CustomerElementCc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsgCc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorCc = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElementCc} && $CustomerErrorCc eq '' ) {
                $CustomerErrorMsgCc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorCc    = 'ServerError';
            }

            my $CustomerDisabledCc = '';
            my $CountAuxCc         = $CustomerCounterCc++;
            if ( $CustomerErrorCc ne '' ) {
                $CustomerDisabledCc = 'disabled="disabled"';
                $CountAuxCc         = $Count . 'Error';
            }

            if ( $CustomerQueueCc ne '' ) {
                $CustomerQueueCc = $Count;
            }

            push @MultipleCustomerCc, {
                Count            => $CountAuxCc,
                CustomerElement  => $CustomerElementCc,
                CustomerKey      => $CustomerKeyCc,
                CustomerError    => $CustomerErrorCc,
                CustomerErrorMsg => $CustomerErrorMsgCc,
                CustomerDisabled => $CustomerDisabledCc,
                CustomerQueue    => $CustomerQueueCc,
            };
            $AddressesList{$CustomerElementCc} = 1;
        }
    }

    my @MultipleCustomerBcc;
    my $CustomersNumberBcc = $ParamObject->GetParam( Param => 'CustomerTicketCounterBccCustomer' ) || 0;

    if ($CustomersNumberBcc) {
        my $CustomerCounterBcc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberBcc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementBcc = $ParamObject->GetParam( Param => 'BccCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementBcc;

            my $CustomerKeyBcc   = $ParamObject->GetParam( Param => 'BccCustomerKey_' . $Count )   || '';
            my $CustomerQueueBcc = $ParamObject->GetParam( Param => 'BccCustomerQueue_' . $Count ) || '';

            if ( $GetParam{Bcc} ) {
                $GetParam{Bcc} .= ', ' . $CustomerElementBcc;
            }
            else {
                $GetParam{Bcc} = $CustomerElementBcc;
            }

            # check email address
            my $CustomerErrorMsgBcc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorBcc    = '';
            for my $Email ( Mail::Address->parse($CustomerElementBcc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsgBcc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorBcc = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElementBcc} && $CustomerErrorBcc eq '' ) {
                $CustomerErrorMsgBcc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorBcc    = 'ServerError';
            }

            my $CustomerDisabledBcc = '';
            my $CountAuxBcc         = $CustomerCounterBcc++;
            if ( $CustomerErrorBcc ne '' ) {
                $CustomerDisabledBcc = 'disabled="disabled"';
                $CountAuxBcc         = $Count . 'Error';
            }

            if ( $CustomerQueueBcc ne '' ) {
                $CustomerQueueBcc = $Count;
            }

            push @MultipleCustomerBcc, {
                Count            => $CountAuxBcc,
                CustomerElement  => $CustomerElementBcc,
                CustomerKey      => $CustomerKeyBcc,
                CustomerError    => $CustomerErrorBcc,
                CustomerErrorMsg => $CustomerErrorMsgBcc,
                CustomerDisabled => $CustomerDisabledBcc,
                CustomerQueue    => $CustomerQueueBcc,
            };
            $AddressesList{$CustomerElementBcc} = 1;
        }
    }

    return (
        GetParam            => \%GetParam,
        MultipleCustomer    => \@MultipleCustomer,
        MultipleCustomerCc  => \@MultipleCustomerCc,
        MultipleCustomerBcc => \@MultipleCustomerBcc,
    );
}

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

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

1;
</File>
        <File Location="Custom/Kernel/Modules/AgentTicketEmailResend.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/Modules/AgentTicketEmailResend.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::Modules::AgentTicketEmailResend;

use strict;
use warnings;

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

our $ObjectManagerDisabled = 1;

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

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

    $Self->{Debug} = $Param{Debug} || 0;

    # Get form ID.
    $Self->{FormID} = $Kernel::OM->Get('Kernel::System::Web::FormCache')->PrepareFormID(
        ParamObject  => $Kernel::OM->Get('Kernel::System::Web::Request'),
        LayoutObject => $Kernel::OM->Get('Kernel::Output::HTML::Layout'),
    );

    return $Self;
}

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

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

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

    my $ConfigObject  = $Kernel::OM->Get('Kernel::Config');
    my $TicketObject  = $Kernel::OM->Get('Kernel::System::Ticket');
    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

    my $ArticleBackendObject = $ArticleObject->BackendForChannel( ChannelName => 'Email' );

    # Get config for 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 article action.
    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) {

        my %AclActionLookup = reverse %AclAction;

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

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

    # Get lock state.
    my $TicketBackType = 'TicketBack';
    if ( $Config->{RequiredLock} ) {
        if ( !$TicketObject->TicketLockGet( TicketID => $Self->{TicketID} ) ) {

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

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

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

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

                # Show lock state.
                if ( !$Success ) {
                    return $LayoutObject->FatalError();
                }
            }

            $TicketBackType .= 'Undo';
        }
        else {
            my $AccessOk = $TicketObject->OwnerCheck(
                TicketID => $Self->{TicketID},
                OwnerID  => $Self->{UserID},
            );
            if ( !$AccessOk ) {
                my $Output = $LayoutObject->Header(
                    Value     => $Ticket{Number},
                    Type      => 'Small',
                    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;
            }
        }
    }

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

    my %GetParam;
    for (
        qw(
            From To Cc Bcc Subject Body InReplyTo References ArticleID
            IsVisibleForCustomerPresent IsVisibleForCustomer TimeUnits FormID
        )
        )
    {
        $GetParam{$_} = $ParamObject->GetParam( Param => $_ );
    }

    if ( !$GetParam{ArticleID} ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('No ArticleID is given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # Hash for lookup of duplicated entries.
    my %AddressesList;

    my @MultipleCustomer;
    my $CustomersNumberTo = $ParamObject->GetParam( Param => 'CustomerTicketCounterToCustomer' ) || 0;
    my $Selected          = $ParamObject->GetParam( Param => 'CustomerSelected' )                || '';

    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    if ($CustomersNumberTo) {
        my $CustomerCounter = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberTo ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElement = $ParamObject->GetParam( Param => 'CustomerTicketText_' . $Count );

            next COUNT unless $CustomerElement;

            my $CustomerSelected = ( $Selected eq $Count ? 'checked ' : '' );
            my $CustomerKey      = $ParamObject->GetParam( Param => 'CustomerKey_' . $Count )   || '';
            my $CustomerQueue    = $ParamObject->GetParam( Param => 'CustomerQueue_' . $Count ) || '';

            if ( $GetParam{To} ) {
                $GetParam{To} .= ', ' . $CustomerElement;
            }
            else {
                $GetParam{To} = $CustomerElement;
            }

            # Check email address.
            my $CustomerErrorMsg = 'CustomerGenericServerErrorMsg';
            my $CustomerError    = '';
            for my $Email ( Mail::Address->parse($CustomerElement) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsg = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerError = 'ServerError';
                }
            }

            # Check for duplicated entries.
            if ( defined $AddressesList{$CustomerElement} && $CustomerError eq '' ) {
                $CustomerErrorMsg = 'IsDuplicatedServerErrorMsg';
                $CustomerError    = 'ServerError';
            }

            my $CustomerDisabled = '';
            my $CountAux         = $CustomerCounter++;
            if ( $CustomerError ne '' ) {
                $CustomerDisabled = 'disabled="disabled"';
                $CountAux         = $Count . 'Error';
            }

            if ( $CustomerQueue ne '' ) {
                $CustomerQueue = $Count;
            }

            push @MultipleCustomer, {
                Count            => $CountAux,
                CustomerElement  => $CustomerElement,
                CustomerSelected => $CustomerSelected,
                CustomerKey      => $CustomerKey,
                CustomerError    => $CustomerError,
                CustomerErrorMsg => $CustomerErrorMsg,
                CustomerDisabled => $CustomerDisabled,
                CustomerQueue    => $CustomerQueue,
            };
            $AddressesList{$CustomerElement} = 1;
        }
    }

    my @MultipleCustomerCc;
    my $CustomersNumberCc = $ParamObject->GetParam( Param => 'CustomerTicketCounterCcCustomer' ) || 0;

    if ($CustomersNumberCc) {
        my $CustomerCounterCc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberCc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementCc = $ParamObject->GetParam( Param => 'CcCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementCc;

            my $CustomerKeyCc   = $ParamObject->GetParam( Param => 'CcCustomerKey_' . $Count )   || '';
            my $CustomerQueueCc = $ParamObject->GetParam( Param => 'CcCustomerQueue_' . $Count ) || '';

            if ( $GetParam{Cc} ) {
                $GetParam{Cc} .= ', ' . $CustomerElementCc;
            }
            else {
                $GetParam{Cc} = $CustomerElementCc;
            }

            # check email address
            my $CustomerErrorMsgCc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorCc    = '';
            for my $Email ( Mail::Address->parse($CustomerElementCc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsgCc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorCc = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElementCc} && $CustomerErrorCc eq '' ) {
                $CustomerErrorMsgCc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorCc    = 'ServerError';
            }

            my $CustomerDisabledCc = '';
            my $CountAuxCc         = $CustomerCounterCc++;
            if ( $CustomerErrorCc ne '' ) {
                $CustomerDisabledCc = 'disabled="disabled"';
                $CountAuxCc         = $Count . 'Error';
            }

            if ( $CustomerQueueCc ne '' ) {
                $CustomerQueueCc = $Count;
            }

            push @MultipleCustomerCc, {
                Count            => $CountAuxCc,
                CustomerElement  => $CustomerElementCc,
                CustomerKey      => $CustomerKeyCc,
                CustomerError    => $CustomerErrorCc,
                CustomerErrorMsg => $CustomerErrorMsgCc,
                CustomerDisabled => $CustomerDisabledCc,
                CustomerQueue    => $CustomerQueueCc,
            };
            $AddressesList{$CustomerElementCc} = 1;
        }
    }

    my @MultipleCustomerBcc;
    my $CustomersNumberBcc = $ParamObject->GetParam( Param => 'CustomerTicketCounterBccCustomer' ) || 0;

    if ($CustomersNumberBcc) {
        my $CustomerCounterBcc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberBcc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementBcc = $ParamObject->GetParam( Param => 'BccCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementBcc;

            my $CustomerKeyBcc   = $ParamObject->GetParam( Param => 'BccCustomerKey_' . $Count )   || '';
            my $CustomerQueueBcc = $ParamObject->GetParam( Param => 'BccCustomerQueue_' . $Count ) || '';

            if ( $GetParam{Bcc} ) {
                $GetParam{Bcc} .= ', ' . $CustomerElementBcc;
            }
            else {
                $GetParam{Bcc} = $CustomerElementBcc;
            }

            # Check email address.
            my $CustomerErrorMsgBcc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorBcc    = '';
            for my $Email ( Mail::Address->parse($CustomerElementBcc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsgBcc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorBcc = 'ServerError';
                }
            }

            # Check for duplicated entries.
            if ( defined $AddressesList{$CustomerElementBcc} && $CustomerErrorBcc eq '' ) {
                $CustomerErrorMsgBcc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorBcc    = 'ServerError';
            }

            my $CustomerDisabledBcc = '';
            my $CountAuxBcc         = $CustomerCounterBcc++;
            if ( $CustomerErrorBcc ne '' ) {
                $CustomerDisabledBcc = 'disabled="disabled"';
                $CountAuxBcc         = $Count . 'Error';
            }

            if ( $CustomerQueueBcc ne '' ) {
                $CustomerQueueBcc = $Count;
            }

            push @MultipleCustomerBcc, {
                Count            => $CountAuxBcc,
                CustomerElement  => $CustomerElementBcc,
                CustomerKey      => $CustomerKeyBcc,
                CustomerError    => $CustomerErrorBcc,
                CustomerErrorMsg => $CustomerErrorMsgBcc,
                CustomerDisabled => $CustomerDisabledBcc,
                CustomerQueue    => $CustomerQueueBcc,
            };
            $AddressesList{$CustomerElementBcc} = 1;
        }
    }

    my $UploadCacheObject = $Kernel::OM->Get('Kernel::System::Web::UploadCache');
    my $MainObject        = $Kernel::OM->Get('Kernel::System::Main');

    # Send email.
    if ( $Self->{Subaction} eq 'SendEmail' ) {

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

        my %Error;

        my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

        # Check some values.
        LINE:
        for my $Line (qw(To Cc Bcc)) {
            next LINE if !$GetParam{$Line};
            for my $Email ( Mail::Address->parse( $GetParam{$Line} ) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $Error{ $Line . 'ErrorType' } = $Line . $CheckItemObject->CheckErrorType() . 'ServerErrorMsg';
                    $Error{ $Line . 'Invalid' }   = 'ServerError';
                }
                my $IsLocal = $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress(
# Rother OSS / DiscreteSystemAddresses
                    TicketID => $Self->{TicketID},
# EO DiscreteSystemAddresses
                    Address => $Email->address()
                );
                if ($IsLocal) {
                    $Error{ $Line . 'IsLocalAddress' } = 'ServerError';
                }
            }
        }

        if ( $Error{ToIsLocalAddress} ) {
            $LayoutObject->Block(
                Name => 'ToIsLocalAddressServerErrorMsg',
                Data => \%GetParam,
            );
        }

        if ( $Error{CcIsLocalAddress} ) {
            $LayoutObject->Block(
                Name => 'CcIsLocalAddressServerErrorMsg',
                Data => \%GetParam,
            );
        }

        if ( $Error{BccIsLocalAddress} ) {
            $LayoutObject->Block(
                Name => 'BccIsLocalAddressServerErrorMsg',
                Data => \%GetParam,
            );
        }

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

        # Check some values.
        LINE:
        for my $Line (qw(To Cc Bcc)) {
            next LINE if !$GetParam{$Line};
            for my $Email ( Mail::Address->parse( $GetParam{$Line} ) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $Error{ $Line . 'Invalid' } = 'ServerError';
                }
            }
        }

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

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

        # Check time units.
        if (
            $ConfigObject->Get('Ticket::Frontend::AccountTime')
            && $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
            && $GetParam{TimeUnits} eq ''
            )
        {
            $Error{TimeUnitsInvalid} = 'ServerError';
        }

        # Prepare subject.
        my $Tn = $TicketObject->TicketNumberLookup( TicketID => $Self->{TicketID} );
        $GetParam{Subject} = $TicketObject->TicketSubjectBuild(
            TicketNumber => $Tn,
            Subject      => $GetParam{Subject} || '',
        );

        my %ArticleParam;

        # Run compose modules.
        if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {

            # Use ticket QueueID in compose modules.
            $GetParam{QueueID} = $Ticket{QueueID};

            my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
            for my $Job ( sort keys %Jobs ) {

                # load module
                if ( !$MainObject->Require( $Jobs{$Job}->{Module} ) ) {
                    return $LayoutObject->FatalError();
                }
                my $Object = $Jobs{$Job}->{Module}->new( %{$Self}, Debug => $Self->{Debug} );

                my $Multiple;

                # get params
                PARAMETER:
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                    if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                        @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                        $Multiple = 1;
                        next PARAMETER;
                    }

                    $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
                }

                # Run module.
                $Object->Run(
                    %GetParam,
                    StoreNew => 1,
                    Config   => $Jobs{$Job},
                );

                # Get options that have been removed from the selection and add them back to the selection so that the
                #   submit will contain options that were hidden from the agent.
                my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );

                if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                    my @RemovedOptions = $Object->GetOptionsToRemoveAJAX(%GetParam);
                    if (@RemovedOptions) {
                        if ($Multiple) {
                            for my $RemovedOption (@RemovedOptions) {
                                push @{ $GetParam{$Key} }, $RemovedOption;
                            }
                        }
                        else {
                            $GetParam{$Key} = shift @RemovedOptions;
                        }
                    }
                }

                # Ticket params.
                %ArticleParam = (
                    %ArticleParam,
                    $Object->ArticleOption( %GetParam, %ArticleParam, Config => $Jobs{$Job} ),
                );

                # Get errors.
                %Error = (
                    %Error,
                    $Object->Error( %GetParam, Config => $Jobs{$Job} ),
                );
            }
        }

        # Check if there is an error.
        if (%Error) {

            my $Output = $LayoutObject->Header(
                Value     => $Ticket{TicketNumber},
                Type      => 'Small',
                BodyClass => 'Popup',
            );
            $GetParam{StandardResponse} = $GetParam{Body};
            $Output .= $Self->_Mask(
                TicketID            => $Self->{TicketID},
                Errors              => \%Error,
                MultipleCustomer    => \@MultipleCustomer,
                MultipleCustomerCc  => \@MultipleCustomerCc,
                MultipleCustomerBcc => \@MultipleCustomerBcc,
                Attachments         => \@Attachments,
                GetParam            => \%GetParam,
                TicketBackType      => $TicketBackType,
                %Ticket,
                %GetParam,
            );
            $Output .= $LayoutObject->Footer(
                Type => 'Small',
            );
            return $Output;
        }

        # Get pre-loaded attachments.
        my @AttachmentData = $UploadCacheObject->FormIDGetAllFilesData(
            FormID => $Self->{FormID},
        );

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

        # Get recipients.
        my $Recipients = '';
        LINE:
        for my $Line (qw(To Cc Bcc)) {

            next LINE if !$GetParam{$Line};

            if ($Recipients) {
                $Recipients .= ', ';
            }
            $Recipients .= $GetParam{$Line};
        }

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

            # Remove unused inline images.
            my @NewAttachmentData;
            ATTACHMENT:
            for my $Attachment (@AttachmentData) {
                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};
            }
            @AttachmentData = @NewAttachmentData;

            # Verify HTML document.
            $GetParam{Body} = $LayoutObject->RichTextDocumentComplete(
                String => $GetParam{Body},
            );
        }

        my $IsVisibleForCustomer = $Config->{IsVisibleForCustomerDefault};
        if ( $GetParam{IsVisibleForCustomerPresent} ) {
            $IsVisibleForCustomer = $GetParam{IsVisibleForCustomer} ? 1 : 0;
        }

        # Send email.
        my $ArticleID = $ArticleBackendObject->ArticleSend(
            IsVisibleForCustomer => $IsVisibleForCustomer,
            SenderType           => 'agent',
            TicketID             => $Self->{TicketID},
            HistoryType          => 'EmailResend',
            HistoryComment       => "\%\%$Recipients",
            From                 => $GetParam{From},
            To                   => $GetParam{To},
            Cc                   => $GetParam{Cc},
            Bcc                  => $GetParam{Bcc},
            Subject              => $GetParam{Subject},
            UserID               => $Self->{UserID},
            Body                 => $GetParam{Body},
            InReplyTo            => $GetParam{InReplyTo},
            References           => $GetParam{References},
            Charset              => $LayoutObject->{UserCharset},
            MimeType             => $MimeType,
            Attachment           => \@AttachmentData,
            %ArticleParam,
        );

        # Error page.
        if ( !$ArticleID ) {
            return $LayoutObject->ErrorScreen();
        }

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

        # remove all form data
        $Kernel::OM->Get('Kernel::System::Web::FormCache')->FormIDRemove( FormID => $Self->{FormID} );

        # Load new URL in parent window and close popup.
        return $LayoutObject->PopupClose(
            URL => "Action=AgentTicketZoom;TicketID=$Self->{TicketID};ArticleID=$ArticleID",
        );
    }

    # Check for SMIME / PGP if customer has changed.
    elsif ( $Self->{Subaction} eq 'AJAXUpdate' ) {

        my @ExtendedData;

        # Run compose modules.
        if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {

            # Use ticket QueueID in compose modules.
            $GetParam{QueueID} = $Ticket{QueueID};

            my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
            JOB:
            for my $Job ( sort keys %Jobs ) {

                # Load module.
                next JOB if !$MainObject->Require( $Jobs{$Job}->{Module} );

                my $Object = $Jobs{$Job}->{Module}->new(
                    %{$Self},
                    Debug => $Self->{Debug},
                );

                my $Multiple;

                PARAMETER:
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                    if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                        @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                        $Multiple = 1;
                        next PARAMETER;
                    }

                    $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
                }

                # Run module.
                my %Data = $Object->Data( %GetParam, Config => $Jobs{$Job} );

                # Get AJAX param values.
                if ( $Object->can('GetParamAJAX') ) {
                    %GetParam = ( %GetParam, $Object->GetParamAJAX(%GetParam) );
                }

                # Get options that have to be removed from the selection visible to the agent. These options will be
                #   added again on submit.
                if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                    my @OptionsToRemove = $Object->GetOptionsToRemoveAJAX(%GetParam);

                    for my $OptionToRemove (@OptionsToRemove) {
                        delete $Data{$OptionToRemove};
                    }
                }

                my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );
                if ($Key) {
                    push @ExtendedData, {
                        Name         => $Key,
                        Data         => \%Data,
                        SelectedID   => $GetParam{$Key},
                        Translation  => 1,
                        PossibleNone => 1,
                        Multiple     => $Multiple,
                        Max          => 150,
                    };
                }
            }
        }

        my $JSON = $LayoutObject->BuildSelectionJSON(
            [
                @ExtendedData,
            ],
        );
        return $LayoutObject->Attachment(
            ContentType => 'application/json',
            Content     => $JSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }
    else {
        my $Output = $LayoutObject->Header(
            Value     => $Ticket{TicketNumber},
            Type      => 'Small',
            BodyClass => 'Popup',
        );

        # Get article data.
        my $ArticleBackendObject = $ArticleObject->BackendForArticle(
            TicketID  => $Self->{TicketID},
            ArticleID => $GetParam{ArticleID},
        );
        my %Data = $ArticleBackendObject->ArticleGet(
            TicketID  => $Self->{TicketID},
            ArticleID => $GetParam{ArticleID},
        );

        # Get article to quote.
        $Data{Body} = $LayoutObject->ArticleQuote(
            TicketID           => $Self->{TicketID},
            ArticleID          => $Data{ArticleID},
            FormID             => $Self->{FormID},
            UploadCacheObject  => $UploadCacheObject,
            AttachmentsInclude => 1,
        );

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

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

        # Get only email address in 'To' (just 'some@example.com').
        for my $Email ( Mail::Address->parse( $Data{To} ) ) {
            $Data{ToEmail} = $Email->address();
        }

        # Find duplicate addresses.
        my %Recipient;
        for my $Type (qw(To Cc Bcc)) {
            if ( $Data{$Type} ) {
                my $NewLine = '';
                for my $Email ( Mail::Address->parse( $Data{$Type} ) ) {
                    my $Address = lc $Email->address();

                    # Only use email addresses with '@' inside.
                    if ( $Address && $Address =~ /@/ && !$Recipient{$Address} ) {
                        $Recipient{$Address} = 1;
                        my $IsLocal = $SystemAddress->SystemAddressIsLocalAddress(
                            Address => $Address,
# Rother OSS / DiscreteSystemAddresses
                            TicketID => $Self->{TicketID},
# EO DiscreteSystemAddresses
                        );
                        if ( !$IsLocal ) {
                            if ($NewLine) {
                                $NewLine .= ', ';
                            }
                            $NewLine .= $Email->format();
                        }
                    }
                }
                $Data{$Type} = $NewLine;
            }
        }

        # Check some values.
        my %Error;
        LINE:
        for my $Line (qw(To Cc Bcc)) {
            next LINE if !$Data{$Line};
            for my $Email ( Mail::Address->parse( $Data{$Line} ) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $Error{ $Line . "Invalid" } = " ServerError";
                }
            }
        }
        if ( $Data{From} ) {
            for my $Email ( Mail::Address->parse( $Data{From} ) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $Error{"FromInvalid"} .= $CheckItemObject->CheckError();
                }
            }
        }

        # Run compose modules.
        if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {

            # use ticket QueueID in compose modules
            $GetParam{QueueID} = $Ticket{QueueID};

            my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
            for my $Job ( sort keys %Jobs ) {

                # Load module.
                if ( !$MainObject->Require( $Jobs{$Job}->{Module} ) ) {
                    return $LayoutObject->FatalError();
                }
                my $Object = $Jobs{$Job}->{Module}->new( %{$Self}, Debug => $Self->{Debug} );

                PARAMETER:
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                    if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                        @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                        next PARAMETER;
                    }

                    $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
                }

                # Run module.
                my $NewParams = $Object->Run( %GetParam, Config => $Jobs{$Job} );

                if ($NewParams) {
                    for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                        $GetParam{$Parameter} = $NewParams;
                    }
                }

                # Get errors.
                %Error = (
                    %Error,
                    $Object->Error( %GetParam, Config => $Jobs{$Job} ),
                );
            }
        }

        # Build view.
        $Output .= $Self->_Mask(
            TicketID            => $Self->{TicketID},
            Attachments         => \@Attachments,
            Errors              => \%Error,
            MultipleCustomer    => \@MultipleCustomer,
            MultipleCustomerCc  => \@MultipleCustomerCc,
            MultipleCustomerBcc => \@MultipleCustomerBcc,
            GetParam            => \%GetParam,
            %Ticket,
            %Data,
            TicketBackType => $TicketBackType,
        );
        $Output .= $LayoutObject->Footer(
            Type => 'Small',
        );
        return $Output;
    }
}

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

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

    my $Config = $ConfigObject->Get("Ticket::Frontend::$Self->{Action}");

    my $IsVisibleForCustomer = $Param{IsVisibleForCustomer};
    if ( $Param{GetParam}->{IsVisibleForCustomerPresent} ) {
        $IsVisibleForCustomer = $Param{GetParam}->{IsVisibleForCustomer} ? 1 : 0;
    }

    $LayoutObject->Block(
        Name => 'IsVisibleForCustomer',
        Data => {
            IsVisibleForCustomer => $IsVisibleForCustomer,
        },
    );

    # Prepare errors for the output.
    if ( $Param{Errors} ) {
        for my $Error ( sort keys %{ $Param{Errors} } ) {
            $Param{$Error} = $LayoutObject->Ascii2Html(
                Text => $Param{Errors}->{$Error},
            );
        }
    }

    # Handle multiple autocomplete in recipient fields.
    $Param{To} = ( scalar @{ $Param{MultipleCustomer} } ? '' : $Param{To} );
    if ( defined $Param{To} && $Param{To} ne '' ) {
        $Param{ToInvalid} = '';
    }
    $Param{Cc} = ( scalar @{ $Param{MultipleCustomerCc} } ? '' : $Param{Cc} );
    if ( defined $Param{Cc} && $Param{Cc} ne '' ) {
        $Param{CcInvalid} = '';
    }
    $Param{Bcc} = ( scalar @{ $Param{MultipleCustomerBcc} } ? '' : $Param{Bcc} );
    if ( defined $Param{Bcc} && $Param{Bcc} ne '' ) {
        $Param{BccInvalid} = '';
    }

    # Process the 'Cc' field.
    my $CustomerCounterCc = 0;
    if ( $Param{MultipleCustomerCc} ) {
        for my $Item ( @{ $Param{MultipleCustomerCc} } ) {
            $LayoutObject->Block(
                Name => 'CcMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Cc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CcCustomerErrorExplantion',
                );
            }
            $CustomerCounterCc++;
        }
    }

    if ( !$CustomerCounterCc ) {
        $Param{CcCustomerHiddenContainer} = 'Hidden';
    }

    # Set customer counter.
    $LayoutObject->Block(
        Name => 'CcMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterCc,
        },
    );

    # Process the 'Bcc' field.
    my $CustomerCounterBcc = 0;
    if ( $Param{MultipleCustomerBcc} ) {
        for my $Item ( @{ $Param{MultipleCustomerBcc} } ) {
            $LayoutObject->Block(
                Name => 'BccMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Bcc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'BccCustomerErrorExplantion',
                );
            }
            $CustomerCounterBcc++;
        }
    }

    if ( !$CustomerCounterBcc ) {
        $Param{BccCustomerHiddenContainer} = 'Hidden';
    }

    # Set customer counter.
    $LayoutObject->Block(
        Name => 'BccMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterBcc,
        },
    );

    # Process the 'To' field.
    my $CustomerCounter = 0;
    if ( $Param{MultipleCustomer} ) {
        for my $Item ( @{ $Param{MultipleCustomer} } ) {
            $LayoutObject->Block(
                Name => 'MultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CustomerErrorExplantion',
                );
            }
            $CustomerCounter++;
        }
    }

    if ( !$CustomerCounter ) {
        $Param{CustomerHiddenContainer} = 'Hidden';
    }

    # Set customer counter.
    $LayoutObject->Block(
        Name => 'MultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounter,
        },
    );

    if ( $Param{ToInvalid} && $Param{Errors} ) {
        $LayoutObject->Block(
            Name => 'ToServerErrorMsg',
        );
    }

    my $CustomerUserObject = $Kernel::OM->Get('Kernel::System::CustomerUser');

    # Set preselected values for 'Bcc' field.
    if ( $Param{Bcc} && $Param{Bcc} ne '' && !$CustomerCounterBcc ) {

        # split Cc values
        my @EmailAddressesBcc;
        for my $Email ( Mail::Address->parse( $Param{Bcc} ) ) {

            my %CustomerSearch = $CustomerUserObject->CustomerSearch(
                PostMasterSearch => $Email->address(),
                Limit            => 1,
            );

            if (%CustomerSearch) {
                for my $CustomerUserID ( sort keys %CustomerSearch ) {
                    push @EmailAddressesBcc, {
                        CustomerKey        => $CustomerUserID,
                        CustomerTicketText => $CustomerSearch{$CustomerUserID},
                    };
                }
            }
            else {
                push @EmailAddressesBcc, {
                    CustomerKey        => '',
                    CustomerTicketText => $Email->[0] ? "$Email->[0] <$Email->[1]>" : "$Email->[1]",
                };
            }
        }

        $LayoutObject->AddJSData(
            Key   => 'EmailAddressesBcc',
            Value => \@EmailAddressesBcc,
        );

        $Param{Bcc} = '';
    }

    # set preselected values for Cc field
    if ( $Param{Cc} && $Param{Cc} ne '' && !$CustomerCounterCc ) {

        # split Cc values
        my @EmailAddressesCc;
        for my $Email ( Mail::Address->parse( $Param{Cc} ) ) {

            my %CustomerSearch = $CustomerUserObject->CustomerSearch(
                PostMasterSearch => $Email->address(),
                Limit            => 1,
            );

            if (%CustomerSearch) {
                for my $CustomerUserID ( sort keys %CustomerSearch ) {
                    push @EmailAddressesCc, {
                        CustomerKey        => $CustomerUserID,
                        CustomerTicketText => $CustomerSearch{$CustomerUserID},
                    };
                }
            }
            else {
                push @EmailAddressesCc, {
                    CustomerKey        => '',
                    CustomerTicketText => $Email->[0] ? "$Email->[0] <$Email->[1]>" : "$Email->[1]",
                };
            }
        }

        $LayoutObject->AddJSData(
            Key   => 'EmailAddressesCc',
            Value => \@EmailAddressesCc,
        );

        $Param{Cc} = '';
    }

    # set preselected values for To field
    if ( defined $Param{To} && $Param{To} ne '' && !$CustomerCounter ) {

        # split To values
        my @EmailAddressesTo;
        for my $Email ( Mail::Address->parse( $Param{To} ) ) {

            my %CustomerSearch = $CustomerUserObject->CustomerSearch(
                PostMasterSearch => $Email->address(),
                Limit            => 1,
            );

            if (%CustomerSearch) {
                for my $CustomerUserID ( sort keys %CustomerSearch ) {
                    push @EmailAddressesTo, {
                        CustomerKey        => $CustomerUserID,
                        CustomerTicketText => $CustomerSearch{$CustomerUserID},
                    };
                }
            }
            else {
                push @EmailAddressesTo, {
                    CustomerKey        => '',
                    CustomerTicketText => $Email->[0] ? "$Email->[0] <$Email->[1]>" : "$Email->[1]",
                };
            }
        }

        $LayoutObject->AddJSData(
            Key   => 'EmailAddressesTo',
            Value => \@EmailAddressesTo,
        );

        $Param{To} = '';
    }

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

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

    # Show the customer user address book if the module is registered.
    if ( $ConfigObject->Get('Frontend::Module')->{AgentCustomerUserAddressBook} ) {
        $Param{OptionCustomerUserAddressBook} = 1;
    }

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

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

    # explanatory message about asterisk
    if ( $ConfigObject->Get('Ticket::Frontend::AsteriskExplanation') ) {
        $LayoutObject->Block(
            Name => 'AsteriskExplanation',
        );
    }

    # Create & return output.
    return $LayoutObject->Output(
        TemplateFile => 'AgentTicketEmailResend',
        Data         => {
            FormID => $Self->{FormID},
            %Param,
        },
    );
}

1;
</File>
        <File Location="Custom/Kernel/Modules/AgentTicketForward.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/Modules/AgentTicketForward.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::Modules::AgentTicketForward;

use strict;
use warnings;

# core modules

# CPAN modules

# OTOBO modules
use Kernel::System::VariableCheck qw(:all);
use Kernel::Language              qw(Translatable);
use Mail::Address                 ();

our $ObjectManagerDisabled = 1;

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

    # allocate new hash for object
    my $Self = bless {%Param}, $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 config for frontend module
    my $Config = $Kernel::OM->Get('Kernel::Config')->Get("Ticket::Frontend::$Self->{Action}");

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

    # get the dynamic fields for this screen
    my $DynamicFieldList = $DynamicFieldObject->DynamicFieldListGet(
        Valid       => 1,
        ObjectType  => [ 'Ticket', 'Article' ],
        FieldFilter => $Config->{DynamicField} || {},
    );

    my $Definition = $Kernel::OM->Get('Kernel::System::Ticket::Mask')->DefinitionGet(
        Mask => $Self->{Action},
    ) || {};

    $Self->{DynamicField}   = [];
    $Self->{MaskDefinition} = $Definition->{Mask};

    # align sysconfig and ticket mask data I
    for my $DynamicField ( @{ $DynamicFieldList // [] } ) {
        if ( exists $Definition->{DynamicFields}{ $DynamicField->{Name} } ) {
            my $Parameters = delete $Definition->{DynamicFields}{ $DynamicField->{Name} } // {};

            for my $Attribute ( keys $Parameters->%* ) {
                $DynamicField->{$Attribute} = $Parameters->{$Attribute};
            }
        }
        else {
            push $Self->{MaskDefinition}->@*, {
                DF        => $DynamicField->{Name},
                Mandatory => $Config->{DynamicField}{ $DynamicField->{Name} } == 2 ? 1 : 0,
            };

            if ( $Config->{DynamicField}{ $DynamicField->{Name} } == 2 ) {
                $DynamicField->{Mandatory} = 1;
            }
        }

        push $Self->{DynamicField}->@*, $DynamicField;
    }

    # align sysconfig and ticket mask data II
    for my $DynamicFieldName ( keys $Definition->{DynamicFields}->%* ) {
        push $Self->{DynamicField}->@*, $DynamicFieldObject->DynamicFieldGet(
            Name => $DynamicFieldName,
        );

        my $Parameters = $Definition->{DynamicFields}{$DynamicFieldName} // {};

        for my $Attribute ( keys $Parameters->%* ) {
            $Self->{DynamicField}[-1]{$Attribute} = $Parameters->{$Attribute};
        }
    }

    $Self->{Debug} = $Param{Debug} || 0;

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

    for (
        qw(To Cc Bcc Subject Body InReplyTo References ComposeStateID IsVisibleForCustomerPresent
        IsVisibleForCustomer ArticleID TimeUnits Year Month Day Hour Minute FormDraftID Title)
        )
    {
        my $Value = $ParamObject->GetParam( Param => $_ );
        if ( defined $Value ) {
            $Self->{GetParam}->{$_} = $Value;
        }
    }

    $Self->{GetParam}->{ForwardTemplateID} = $ParamObject->GetParam( Param => 'ForwardTemplateID' ) || '';

    # ACL compatibility translation
    $Self->{ACLCompatGetParam}->{NextStateID} = $Self->{GetParam}->{ComposeStateID};

    # Get form ID.
    $Self->{GetParam}->{FormID} = $Kernel::OM->Get('Kernel::System::Web::FormCache')->PrepareFormID(
        ParamObject  => $ParamObject,
        LayoutObject => $Kernel::OM->Get('Kernel::Output::HTML::Layout'),
    );

    return $Self;
}

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

    my $Output;

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

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

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

    $Self->{GetParam}->{From} = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->Sender(
        QueueID => $Ticket{QueueID},
        UserID  => $Self->{UserID},
    );

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

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

    # check if ACL restrictions exist
    if ($ACL) {

        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 (
        $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'LoadFormDraft' )
        && !$Self->{LoadedFormDraftID}
        )
    {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Loading draft failed!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    if ( $Self->{Subaction} eq 'SendEmail' ) {

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

        $Output = $Self->SendEmail();
    }
    elsif ( $Self->{Subaction} eq 'AJAXUpdate' ) {
        $Output = $Self->AjaxUpdate();
    }
    elsif ( $Self->{LoadedFormDraftID} ) {
        $Output = $Self->SendEmail();
    }
    else {
        $Output = $Self->Form();
    }

    return $Output;
}

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

    my %Error;
    my %ACLCompatGetParam = %{ $Self->{ACLCompatGetParam} };

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

    # check needed stuff
    if ( !$Self->{TicketID} ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Got no TicketID!'),
            Comment => Translatable('System Error!'),
        );
    }

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

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

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

    # get config for 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( WithHeader => 'yes' );
    }

    my %GetParamExtended = $Self->_GetExtendedParams();

    my %GetParam            = %{ $GetParamExtended{GetParam} };
    my @MultipleCustomer    = @{ $GetParamExtended{MultipleCustomer} };
    my @MultipleCustomerCc  = @{ $GetParamExtended{MultipleCustomerCc} };
    my @MultipleCustomerBcc = @{ $GetParamExtended{MultipleCustomerBcc} };

    # 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 ) {
                return join '',
                    $LayoutObject->Header(
                        Type      => 'Small',
                        BodyClass => 'Popup',
                    ),
                    $LayoutObject->Warning(
                        Message => Translatable('Sorry, you need to be the ticket owner to perform this action.'),
                        Comment => Translatable('Please change the owner first.'),
                    ),
                    $LayoutObject->Footer(
                        Type => 'Small',
                    );
            }
            else {
                $LayoutObject->Block(
                    Name => 'TicketBack',
                    Data => {
                        %Param,
                        TicketID => $Self->{TicketID},
                    },
                );
            }
        }
    }
    else {
        $LayoutObject->Block(
            Name => 'TicketBack',
            Data => {
                %Param,
                TicketID => $Self->{TicketID},
            },
        );
    }

    # Get selected or last customer article.
    my %Data;
    if ( $GetParam{ArticleID} ) {
        my $ArticleBackendObject = $ArticleObject->BackendForArticle(
            TicketID  => $Self->{TicketID},
            ArticleID => $GetParam{ArticleID},
        );
        %Data = $ArticleBackendObject->ArticleGet(
            TicketID      => $Self->{TicketID},
            ArticleID     => $GetParam{ArticleID},
            DynamicFields => 1,
        );

        # Check if article exists.
        if ( !%Data ) {
            return $LayoutObject->ErrorScreen(
                Message => $LayoutObject->{LanguageObject}->Translate( 'Article %s could not be found!', $GetParam{ArticleID} ),
            );
        }
    }
    else {

        # Get last customer article.
        my @Articles = $ArticleObject->ArticleList(
            TicketID   => $Self->{TicketID},
            SenderType => 'customer',
            OnlyLast   => 1,
        );

        # If the ticket has no customer article, get the last agent article.
        if ( !@Articles ) {
            @Articles = $ArticleObject->ArticleList(
                TicketID   => $Self->{TicketID},
                SenderType => 'agent',
                OnlyLast   => 1,
            );
        }

        # Finally, if everything failed, get latest article.
        if ( !@Articles ) {
            @Articles = $ArticleObject->ArticleList(
                TicketID => $Self->{TicketID},
                OnlyLast => 1,
            );
        }

        for my $Article (@Articles) {
            %Data = $ArticleObject->BackendForArticle( %{$Article} )->ArticleGet(
                %{$Article},
                DynamicFields => 1,
            );
        }
    }

    # prepare signature
    my $TemplateGenerator = $Kernel::OM->Get('Kernel::System::TemplateGenerator');
    $Data{Signature} = $TemplateGenerator->Signature(
        TicketID  => $Self->{TicketID},
        ArticleID => $Data{ArticleID},
        Data      => \%Data,
        UserID    => $Self->{UserID},
    );

    if ( $GetParam{ForwardTemplateID} ) {

        # get template
        $Data{StdTemplate} = $TemplateGenerator->Template(
            TicketID   => $Self->{TicketID},
            ArticleID  => $Data{ArticleID},
            TemplateID => $GetParam{ForwardTemplateID},
            Data       => \%Data,
            UserID     => $Self->{UserID},
        );

        # get signature
        $Data{Signature} = $TemplateGenerator->Signature(
            TicketID => $Self->{TicketID},
            Data     => \%Data,
            UserID   => $Self->{UserID},
        );
    }

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

    # body preparation for plain text processing
    $Data{Body} = $LayoutObject->ArticleQuote(
        TicketID           => $Data{TicketID},
        ArticleID          => $Data{ArticleID},
        FormID             => $Self->{GetParam}->{FormID},
        UploadCacheObject  => $UploadCacheObject,
        AttachmentsInclude => 1,
    );

    my %SafetyCheckResult = $Kernel::OM->Get('Kernel::System::HTMLUtils')->Safety(
        String => $Data{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,
    );
    $Data{Body} = $SafetyCheckResult{String};

    # If article is not a MIMEBase article, include sender name for correct quoting.
    if ( !$Data{From} ) {
        my %ArticleFields = $LayoutObject->ArticleFields(
            TicketID  => $Self->{TicketID},
            ArticleID => $Data{ArticleID},
        );
        $Data{Sender} = $ArticleFields{Sender}->{Value} // '';
    }

    if ( $LayoutObject->{BrowserRichText} ) {

        # prepare body, subject, ReplyTo, ...
        $Data{Body} = '<br/>' . $Data{Body};
        if ( $Data{CreateTime} ) {
            $Data{CreateTime} = $LayoutObject->{LanguageObject}->FormatTimeString( $Data{CreateTime} );
            $Data{Body}       = $LayoutObject->{LanguageObject}->Translate('Date') .
                ": $Data{CreateTime}<br/>" . $Data{Body};
        }
        for my $Key (qw(Subject ReplyTo Reply-To Cc To From Sender)) {
            if ( $Data{$Key} ) {
                my $KeyText = $LayoutObject->{LanguageObject}->Translate($Key);

                my $Value = $LayoutObject->Ascii2RichText(
                    String => $Data{$Key},
                );
                $Data{Body} = "$KeyText: $Value<br/>" . $Data{Body};
            }
        }

        my $Quote = $LayoutObject->Ascii2RichText(
            String => $ConfigObject->Get('Ticket::Frontend::Quote') || '',
        );
        if ($Quote) {

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

            # cleanup not compat. tags
            $Data{Body} = $LayoutObject->RichTextDocumentCleanup(
                String => $Data{Body},
            );
        }
        else {
            $Data{Body} = "<br/>" . $Data{Body};
        }
        my $From = $LayoutObject->Ascii2RichText(
            String => $Data{From} || $Data{Sender},
        );

        my $ForwardedMessageFrom = $LayoutObject->{LanguageObject}->Translate('Forwarded message from');
        my $EndForwardedMessage  = $LayoutObject->{LanguageObject}->Translate('End forwarded message');

        $Data{Body} = "<br/>---- $ForwardedMessageFrom $From ---<br/><br/>" . $Data{Body};
        $Data{Body} .= "<br/>---- $EndForwardedMessage ---<br/>";
        $Data{Body} = $Data{Signature} . $Data{Body};

        if ( $GetParam{ForwardTemplateID} ) {
            $Data{Body} = $Data{StdTemplate} . '<br/>' . $Data{Body};
        }

        $Data{ContentType} = 'text/html';
    }
    else {

        # prepare body, subject, ReplyTo, ...
        $Data{Body} =~ s/\t/ /g;
        my $Quote = $ConfigObject->Get('Ticket::Frontend::Quote');
        if ($Quote) {
            $Data{Body} =~ s/\n/\n$Quote /g;
            $Data{Body} = "\n$Quote " . $Data{Body};
        }
        else {
            $Data{Body} = "\n" . $Data{Body};
        }
        if ( $Data{CreateTime} ) {
            $Data{CreateTime} = $LayoutObject->{LanguageObject}->FormatTimeString( $Data{CreateTime} );
            $Data{Body}       = $LayoutObject->{LanguageObject}->Translate('Date') .
                ": $Data{CreateTime}\n" . $Data{Body};
        }
        for (qw(Subject ReplyTo Reply-To Cc To From Sender)) {
            if ( $Data{$_} ) {
                $Data{Body} = $LayoutObject->{LanguageObject}->Translate($_) .
                    ": $Data{$_}\n" . $Data{Body};
            }
        }

        my $ForwardedMessageFrom = $LayoutObject->{LanguageObject}->Translate('Forwarded message from');
        my $EndForwardedMessage  = $LayoutObject->{LanguageObject}->Translate('End forwarded message');

        my $From = $Data{From} || $Data{Sender};

        $Data{Body} = "\n---- $ForwardedMessageFrom $From ---\n\n" . $Data{Body};
        $Data{Body} .= "\n---- $EndForwardedMessage ---\n";
        $Data{Body} = $Data{Signature} . $Data{Body};

        if ( $GetParam{ForwardTemplateID} ) {
            $Data{Body} = $Data{StdTemplate} . "\n" . $Data{Body};
        }
    }

    # get std. attachment object
    my $StdAttachmentObject = $Kernel::OM->Get('Kernel::System::StdAttachment');

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

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

    # check some values
    for my $Recipient (qw(To Cc Bcc)) {
        if ( $Data{$Recipient} ) {
            delete $Data{$Recipient};
        }
    }

    # put & get attributes like sender address
    %Data = $TemplateGenerator->Attributes(
        TicketID  => $Self->{TicketID},
        ArticleID => $GetParam{ArticleID},
        Data      => \%Data,
        UserID    => $Self->{UserID},
        Action    => 'Forward',
    );

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

    # run compose modules
    if ( ref( $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') ) eq 'HASH' ) {

        # use ticket QueueID in compose modules
        $GetParam{QueueID} = $Ticket{QueueID};

        my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
        for my $Job ( sort keys %Jobs ) {

            # load module
            if ( !$Kernel::OM->Get('Kernel::System::Main')->Require( $Jobs{$Job}->{Module} ) ) {
                return $LayoutObject->FatalError();
            }
            my $Object = $Jobs{$Job}->{Module}->new(
                %{$Self},
                Debug => $Self->{Debug},
            );

            # get params
            PARAMETER:
            for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                    @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                    next PARAMETER;
                }

                $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
            }

            # run module
            my $NewParams = $Object->Run( %Data, %GetParam, Config => $Jobs{$Job} );

            if ($NewParams) {
                for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                    $GetParam{$Parameter} = $NewParams;
                }
            }

            # get errors
            %Error = ( %Error, $Object->Error( %GetParam, Config => $Jobs{$Job} ) );
        }
    }

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

    # remember dynamic field validation results if erroneous
    my %DynamicFieldPossibleValues;

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

        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,
                    %ACLCompatGetParam,
                    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;
                }
            }
        }

        $DynamicFieldPossibleValues{ 'DynamicField_' . $DynamicFieldConfig->{Name} } = $PossibleValuesFilter;
    }

    # extract dynamic field values from ticket data
    my %TicketDFValues =
        map  { 'DynamicField_' . $_->{Name} => $Ticket{ 'DynamicField_' . $_->{Name} } }
        grep { $_->{ObjectType} eq 'Ticket' }
        $Self->{DynamicField}->@*;

    # build view
    # start with page
    my $Output = $LayoutObject->Header(
        Value     => $Ticket{TicketNumber},
        Type      => 'Small',
        BodyClass => 'Popup',
    );

    # build references string
    my $References = defined $Data{References} ? $Data{References} . ' ' : '';
    $References .= defined $Data{MessageID} ? $Data{MessageID} : '';

    $Output .= $Self->_Mask(
        TicketNumber   => $Ticket{TicketNumber},
        TicketID       => $Self->{TicketID},
        Title          => $Ticket{Title},
        CustomerID     => $Ticket{CustomerID},
        CustomerUserID => $Ticket{CustomerUserID},
        QueueID        => $Ticket{QueueID},
        SLAID          => $Ticket{SLAID},
        NextStates     => $Self->_GetNextStates(
            %GetParam,
            %ACLCompatGetParam,
        ),
        TimeUnitsRequired => (
            $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
            ? 'Validate_Required'
            : ''
        ),
        Errors              => \%Error,
        MultipleCustomer    => \@MultipleCustomer,
        MultipleCustomerCc  => \@MultipleCustomerCc,
        MultipleCustomerBcc => \@MultipleCustomerBcc,
        Attachments         => \@Attachments,
        %Data,
        %GetParam,
        InReplyTo        => $Data{MessageID},
        References       => $References,
        DFPossibleValues => \%DynamicFieldPossibleValues,
        DFValues         => \%TicketDFValues,
    );
    $Output .= $LayoutObject->Footer(
        Type => 'Small',
    );

    return $Output;
}

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

    my %Error;
    my %ACLCompatGetParam = %{ $Self->{ACLCompatGetParam} };

    my %GetParamExtended = $Self->_GetExtendedParams();

    my %GetParam            = %{ $GetParamExtended{GetParam} };
    my @MultipleCustomer    = @{ $GetParamExtended{MultipleCustomer} };
    my @MultipleCustomerCc  = @{ $GetParamExtended{MultipleCustomerCc} };
    my @MultipleCustomerBcc = @{ $GetParamExtended{MultipleCustomerBcc} };

    my %DynamicFieldValues;

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

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

    # get needed objects
    my $DynamicFieldBackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
    my $LayoutObject              = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ParamObject               = $Kernel::OM->Get('Kernel::System::Web::Request');

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

            # meddling with the innards of Kernel::System::Web::Request
            $ParamObject->SetArray(
                Param  => 'FormDraftID',
                Values => ['']
            );
        }

        my $FormDraftActionOk;
        if (
            $FormDraftAction eq 'Add'
            ||
            ( $FormDraftAction eq 'Update' && $GetParam{FormDraftID} )
            )
        {
            $FormDraftActionOk = $ParamObject->SaveFormDraft(
                UserID     => $Self->{UserID},
                ObjectType => 'Ticket',
                ObjectID   => $Self->{TicketID},
            );
        }
        elsif ( $FormDraftAction eq 'Delete' && $GetParam{FormDraftID} ) {
            $FormDraftActionOk = $Kernel::OM->Get('Kernel::System::FormDraft')->FormDraftDelete(
                FormDraftID => $GetParam{FormDraftID},
                UserID      => $Self->{UserID},
            );
        }

        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',
            Content     => $JSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }

    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{ $Self->{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;

    my $QueueID = $Self->{QueueID};
    my %StateData;

    if ( $GetParam{ComposeStateID} ) {
        %StateData = $Kernel::OM->Get('Kernel::System::State')->StateGet(
            ID => $GetParam{ComposeStateID},
        );
    }

    my $NextState = $StateData{Name};

    # check pending date
    if ( defined $StateData{TypeName} && $StateData{TypeName} =~ /^pending/i ) {

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

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

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

    # check To
    if ( !$GetParam{To} ) {
        $Error{'ToInvalid'} = 'ServerError';
    }

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

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

    if (
        $ConfigObject->Get('Ticket::Frontend::AccountTime')
        && $ConfigObject->Get('Ticket::Frontend::NeedAccountedTime')
        && $GetParam{TimeUnits} eq ''
        )
    {
        $Error{'TimeUnitsInvalid'} = 'ServerError';
    }

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

    # get information on the current ticket
    my %Ticket = $TicketObject->TicketGet(
        TicketID      => $Self->{TicketID},
        DynamicFields => 1,
    );

    # prepare the subject
    $GetParam{Subject} = $TicketObject->TicketSubjectBuild(
        TicketNumber => $Ticket{TicketNumber},
        Action       => 'Forward',
        Subject      => $GetParam{Subject} || '',
    );

    # remember dynamic field validation results if erroneous
    my %DynamicFieldValidationResult;
    my %DynamicFieldPossibleValues;

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

        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,
                    %ACLCompatGetParam,
                    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;
                }
            }
        }

        $DynamicFieldPossibleValues{ 'DynamicField_' . $DynamicFieldConfig->{Name} } = $PossibleValuesFilter;

        my $ValidationResult = $DynamicFieldBackendObject->EditFieldValueValidate(
            DynamicFieldConfig   => $DynamicFieldConfig,
            PossibleValuesFilter => $PossibleValuesFilter,
            ParamObject          => $ParamObject,

            # Mandatory is added to the configs by $Self->new
            Mandatory => $DynamicFieldConfig->{Mandatory},
        );

        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';
            $DynamicFieldValidationResult{ $DynamicFieldConfig->{Name} } = $ValidationResult;
        }

    }

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

    # get check item object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    # check some values
    LINE:
    for my $Line (qw(To Cc Bcc)) {
        next LINE if !$GetParam{$Line};
        for my $Email ( Mail::Address->parse( $GetParam{$Line} ) ) {
            if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                $Error{ $Line . 'ErrorType' } = $Line . $CheckItemObject->CheckErrorType() . 'ServerErrorMsg';
                $Error{ $Line . 'Invalid' }   = 'ServerError';
            }
            my $IsLocal = $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress(
                Address => $Email->address(),
# Rother OSS / DiscreteSystemAddresses
                TicketID => $Self->{TicketID},
# EO DiscreteSystemAddresses
            );
            if ($IsLocal) {
                $Error{ $Line . 'IsLocalAddress' } = 'ServerError';
            }
        }
    }

    # Make sure sender is correct one. See bug#14872 ( https://bugs.otrs.org/show_bug.cgi?id=14872 ).
    $GetParam{From} = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->Sender(
        QueueID => $Ticket{QueueID},
        UserID  => $Self->{UserID},
    );

    if ( $Self->{LoadedFormDraftID} ) {

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

    # run compose modules
    my %ArticleParam;
    if ( ref( $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') ) eq 'HASH' ) {
        my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
        for my $Job ( sort keys %Jobs ) {

            # load module
            if ( !$Kernel::OM->Get('Kernel::System::Main')->Require( $Jobs{$Job}->{Module} ) ) {
                return $LayoutObject->FatalError();
            }
            my $Object = $Jobs{$Job}->{Module}->new(
                %{$Self},
                Debug => $Self->{Debug},
            );

            my $Multiple;

            # get params
            PARAMETER:
            for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                    @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                    $Multiple = 1;
                    next PARAMETER;
                }

                $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
            }

            # run module
            $Object->Run(
                %GetParam,
                StoreNew => 1,
                Config   => $Jobs{$Job},
            );

            # get options that have been removed from the selection
            # and add them back to the selection so that the submit
            # will contain options that were hidden from the agent
            my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );

            if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                my @RemovedOptions = $Object->GetOptionsToRemoveAJAX(%GetParam);
                if (@RemovedOptions) {
                    if ($Multiple) {
                        for my $RemovedOption (@RemovedOptions) {
                            push @{ $GetParam{$Key} }, $RemovedOption;
                        }
                    }
                    else {
                        $GetParam{$Key} = shift @RemovedOptions;
                    }
                }
            }

            # ticket params
            %ArticleParam = (
                %ArticleParam,
                $Object->ArticleOption(
                    %GetParam,
                    %ArticleParam,
                    Config => $Jobs{$Job},
                ),
            );

            # get errors
            %Error = (
                %Error,
                $Object->Error(
                    %GetParam,
                    Config => $Jobs{$Job},
                ),
            );
        }
    }

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

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

    # check if there is an error
    if (%Error) {
        my $QueueID = $TicketObject->TicketQueueID( TicketID => $Self->{TicketID} );

        # extract dynamic field values from ticket data
        my %TicketDFValues =
            map  { 'DynamicField_' . $_->{Name} => $Ticket{ 'DynamicField_' . $_->{Name} } }
            grep { $_->{ObjectType} eq 'Ticket' }
            $Self->{DynamicField}->@*;

        my $Output = $LayoutObject->Header(
            Type      => 'Small',
            BodyClass => 'Popup',
        );

        # TODD: Notification about FormDraft

        $Output .= $Self->_Mask(
            TicketNumber   => $Ticket{TicketNumber},
            Title          => $Ticket{Title},
            CustomerID     => $Ticket{CustomerID},
            CustomerUserID => $Ticket{CustomerUserID},
            TicketID       => $Self->{TicketID},
            QueueID        => $QueueID,
            SLAID          => $Ticket{SLAID},
            NextStates     => $Self->_GetNextStates(
                %GetParam,
                %ACLCompatGetParam,
            ),
            Errors              => \%Error,
            MultipleCustomer    => \@MultipleCustomer,
            MultipleCustomerCc  => \@MultipleCustomerCc,
            MultipleCustomerBcc => \@MultipleCustomerBcc,
            Attachments         => \@Attachments,
            %GetParam,
            DFPossibleValues => \%DynamicFieldPossibleValues,
            DFErrors         => \%DynamicFieldValidationResult,
            DFValues         => \%TicketDFValues,
        );
        $Output .= $LayoutObject->Footer(
            Type => 'Small',
        );

        return $Output;
    }

    # replace <OTOBO_TICKET_STATE> with next ticket state name
    if ($NextState) {
        $GetParam{Body} =~ s/(&lt;|<)OTOBO_TICKET_STATE(&gt;|>)/$NextState/g;
    }

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

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

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

        # remove unused inline images
        my @NewAttachmentData;
        ATTACHMENT:
        for my $Attachment (@AttachmentData) {
            my $ContentID = $Attachment->{ContentID};
            if ( $ContentID && ( $Attachment->{ContentType} =~ /image/i ) ) {
                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;

                # IF the image is referenced in the body set it as inline.
                if ( $GetParam{Body} =~ /(\Q$ContentIDHTMLQuote\E|\Q$ContentID\E)/i ) {
                    $Attachment->{Disposition} = 'inline';
                }
                elsif ( $Attachment->{Disposition} eq 'inline' ) {

                    # Ignore attachment if not linked in body.
                    next ATTACHMENT;
                }
            }

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

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

    # send email
    my $To = '';
    KEY:
    for my $Key (qw(To Cc Bcc)) {
        next KEY if !$GetParam{$Key};
        if ($To) {
            $To .= ', ';
        }
        $To .= $GetParam{$Key};
    }

    my $IsVisibleForCustomer = $Config->{IsVisibleForCustomerDefault};
    if ( $GetParam{IsVisibleForCustomerPresent} ) {
        $IsVisibleForCustomer = $GetParam{IsVisibleForCustomer} ? 1 : 0;
    }

    my $EmailArticleBackendObject = $Kernel::OM->Get('Kernel::System::Ticket::Article')->BackendForChannel(
        ChannelName => 'Email',
    );

    # Get attributes like sender address.
    my %Data = $Kernel::OM->Get('Kernel::System::TemplateGenerator')->Attributes(
        TicketID => $Self->{TicketID},
        Data     => {},
        UserID   => $Self->{UserID},
    );

    my $ArticleID = $EmailArticleBackendObject->ArticleSend(
        TicketID             => $Self->{TicketID},
        SenderType           => 'agent',
        IsVisibleForCustomer => $IsVisibleForCustomer,
        HistoryType          => 'Forward',
        HistoryComment       => "\%\%$To",
        From                 => $Data{From},
        To                   => $GetParam{To},
        Cc                   => $GetParam{Cc},
        Bcc                  => $GetParam{Bcc},
        Subject              => $GetParam{Subject},
        UserID               => $Self->{UserID},
        Body                 => $GetParam{Body},
        InReplyTo            => $GetParam{InReplyTo},
        References           => $GetParam{References},
        Charset              => $LayoutObject->{UserCharset},
        MimeType             => $MimeType,
        Attachment           => \@AttachmentData,
        %ArticleParam,
    );

    # error page
    if ( !$ArticleID ) {
        return $LayoutObject->ErrorScreen(
            Comment => Translatable('Please contact the administrator.'),
        );
    }

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

    # set dynamic fields
    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{ $Self->{DynamicField} } ) {
        next DYNAMICFIELD if !IsHashRefWithData($DynamicFieldConfig);
        next DYNAMICFIELD if $DynamicFieldConfig->{Readonly};

        # set the object ID (TicketID or ArticleID) depending on the field configuration
        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},
        );
    }

    # set state
    if ($NextState) {
        $TicketObject->TicketStateSet(
            TicketID  => $Self->{TicketID},
            ArticleID => $ArticleID,
            State     => $NextState,
            UserID    => $Self->{UserID},
        );

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

        # set pending time
        elsif ( $StateData{TypeName} =~ /^pending/i ) {
            $TicketObject->TicketPendingTimeSet(
                UserID   => $Self->{UserID},
                TicketID => $Self->{TicketID},
                %GetParam,
            );
        }
    }

    # remove all form data
    $Kernel::OM->Get('Kernel::System::Web::FormCache')->FormIDRemove( FormID => $Self->{GetParam}->{FormID} );

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

    # redirect
    if (
        defined $StateData{TypeName}
        && $StateData{TypeName} =~ /^close/i
        && !$ConfigObject->Get('Ticket::Frontend::RedirectAfterCloseDisabled')
        )
    {
        return $LayoutObject->PopupClose(
            URL => ( $Self->{LastScreenOverview} || 'Action=AgentDashboard' ),
        );
    }

    return $LayoutObject->PopupClose(
        URL => "Action=AgentTicketZoom;TicketID=$Self->{TicketID};ArticleID=$ArticleID",
    );
}

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

    my %Error;
    my %ACLCompatGetParam = %{ $Self->{ACLCompatGetParam} };

    my %GetParamExtended = $Self->_GetExtendedParams();

    my %GetParam            = %{ $GetParamExtended{GetParam} };
    my @MultipleCustomer    = @{ $GetParamExtended{MultipleCustomer} };
    my @MultipleCustomerCc  = @{ $GetParamExtended{MultipleCustomerCc} };
    my @MultipleCustomerBcc = @{ $GetParamExtended{MultipleCustomerBcc} };

    my @ExtendedData;

    # get needed objects
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    my $ParamObject  = $Kernel::OM->Get('Kernel::System::Web::Request');

    # run compose modules
    if ( ref $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') eq 'HASH' ) {
        my %Jobs = %{ $ConfigObject->Get('Ticket::Frontend::ArticleComposeModule') };
        JOB:
        for my $Job ( sort keys %Jobs ) {

            # load module
            next JOB if !$Kernel::OM->Get('Kernel::System::Main')->Require( $Jobs{$Job}->{Module} );

            my $Object = $Jobs{$Job}->{Module}->new(
                %{$Self},
                Debug => $Self->{Debug},
            );

            my $Multiple;

            # get params
            PARAMETER:
            for my $Parameter ( $Object->Option( %GetParam, Config => $Jobs{$Job} ) ) {
                if ( $Jobs{$Job}->{ParamType} && $Jobs{$Job}->{ParamType} ne 'Single' ) {
                    @{ $GetParam{$Parameter} } = $ParamObject->GetArray( Param => $Parameter );
                    $Multiple = 1;
                    next PARAMETER;
                }

                $GetParam{$Parameter} = $ParamObject->GetParam( Param => $Parameter );
            }

            # run module
            my %Data = $Object->Data( %GetParam, Config => $Jobs{$Job} );

            # get AJAX param values
            if ( $Object->can('GetParamAJAX') ) {
                %GetParam = ( %GetParam, $Object->GetParamAJAX(%GetParam) );
            }

            # get options that have to be removed from the selection visible
            # to the agent. These options will be added again on submit.
            if ( $Object->can('GetOptionsToRemoveAJAX') ) {
                my @OptionsToRemove = $Object->GetOptionsToRemoveAJAX(%GetParam);

                for my $OptionToRemove (@OptionsToRemove) {
                    delete $Data{$OptionToRemove};
                }
            }

            my $Key = $Object->Option( %GetParam, Config => $Jobs{$Job} );
            if ($Key) {
                push @ExtendedData, {
                    Name         => $Key,
                    Data         => \%Data,
                    SelectedID   => $GetParam{$Key},
                    Translation  => 1,
                    PossibleNone => 1,
                    Multiple     => $Multiple,
                    Max          => 100,
                };
            }
        }
    }

    my %DynamicFieldValues;

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

    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{ $Self->{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;

    my $NextStates = $Self->_GetNextStates(
        %GetParam,
        %ACLCompatGetParam,
    );

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

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

    # cycle through the activated Dynamic Fields for this screen
    DYNAMICFIELD:
    for my $DynamicFieldConfig ( @{ $Self->{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,
            %ACLCompatGetParam,
            Action        => $Self->{Action},
            TicketID      => $Self->{TicketID},
            QueueID       => $Self->{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 $JSON = $LayoutObject->BuildSelectionJSON(
        [
            {
                Name         => 'ComposeStateID',
                Data         => $NextStates,
                SelectedID   => $GetParam{ComposeStateID},
                Translation  => 1,
                PossibleNone => 1,
                Max          => 100,
            },
            @ExtendedData,
            @DynamicFieldAJAX,
        ],
    );

    return $LayoutObject->Attachment(
        ContentType => 'application/json',
        Content     => $JSON,
        Type        => 'inline',
        NoCache     => 1,
    );
}

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

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

    return \%NextStates;
}

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

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

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

    # build next states string
    my %State;
    if ( !$Param{ComposeStateID} ) {
        $State{SelectedValue} = $Config->{StateDefault};
    }
    else {
        $State{SelectedID} = $Param{ComposeStateID};
    }

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

    $Param{NextStatesStrg} = $LayoutObject->BuildSelection(
        Data         => $Param{NextStates},
        Name         => 'ComposeStateID',
        PossibleNone => 1,
        Translation  => 1,
        Class        => 'Modernize FormUpdate',
        %State,
    );

    $Param{IsVisibleForCustomer} = $Config->{IsVisibleForCustomerDefault};
    if ( $Self->{GetParam}->{IsVisibleForCustomerPresent} ) {
        $Param{IsVisibleForCustomer} = $Self->{GetParam}->{IsVisibleForCustomer} ? 1 : 0;
    }

    # prepare errors!
    if ( $Param{Errors} ) {
        for my $Error ( sort keys %{ $Param{Errors} } ) {
            $Param{$Error} = $LayoutObject->Ascii2Html(
                Text => $Param{Errors}->{$Error},
            );
        }
    }

    # get used calendar
    my $Calendar = $Kernel::OM->Get('Kernel::System::Ticket')->TicketCalendarGet(
        QueueID => $Param{QueueID},
        SLAID   => $Param{SLAID},
    );

    my $QuickDateButtons = $Config->{QuickDateButtons} // $ConfigObject->Get('Ticket::Frontend::DefaultQuickDateButtons');

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

    # Multiple-Autocomplete
    # Cc
    my $CustomerCounterCc = 0;
    if ( $Param{MultipleCustomerCc} ) {
        for my $Item ( @{ $Param{MultipleCustomerCc} } ) {
            $LayoutObject->Block(
                Name => 'CcMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Cc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CcCustomerErrorExplantion',
                );
            }
            $CustomerCounterCc++;
        }
    }

    if ( !$CustomerCounterCc ) {
        $Param{CcCustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'CcMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterCc++,
        },
    );

    # Bcc
    my $CustomerCounterBcc = 0;
    if ( $Param{MultipleCustomerBcc} ) {
        for my $Item ( @{ $Param{MultipleCustomerBcc} } ) {
            $LayoutObject->Block(
                Name => 'BccMultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => 'Bcc' . $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'BccCustomerErrorExplantion',
                );
            }
            $CustomerCounterBcc++;
        }
    }

    if ( !$CustomerCounterBcc ) {
        $Param{BccCustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'BccMultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounterBcc++,
        },
    );

    # To
    my $CustomerCounter = 0;
    if ( $Param{MultipleCustomer} ) {
        for my $Item ( @{ $Param{MultipleCustomer} } ) {
            $LayoutObject->Block(
                Name => 'MultipleCustomer',
                Data => $Item,
            );
            $LayoutObject->Block(
                Name => $Item->{CustomerErrorMsg},
                Data => $Item,
            );
            if ( $Item->{CustomerError} ) {
                $LayoutObject->Block(
                    Name => 'CustomerErrorExplantion',
                );
            }
            $CustomerCounter++;
        }
    }

    if ( !$CustomerCounter ) {
        $Param{CustomerHiddenContainer} = 'Hidden';
    }

    # set customer counter
    $LayoutObject->Block(
        Name => 'MultipleCustomerCounter',
        Data => {
            CustomerCounter => $CustomerCounter++,
        },
    );

    if ( $Param{ToInvalid} && $Param{Errors} && !$Param{Errors}->{ToErrorType} ) {
        $LayoutObject->Block(
            Name => 'ToServerErrorMsg',
        );
    }

    if ( $Param{ToIsLocalAddress} && $Param{Errors} && !$Param{Errors}->{ToErrorType} ) {
        $LayoutObject->Block(
            Name => 'ToIsLocalAddressServerErrorMsg',
            Data => \%Param,
        );
    }

    if ( $Param{CcInvalid} && $Param{Errors} && !$Param{Errors}->{CcErrorType} ) {
        $LayoutObject->Block(
            Name => 'CcServerErrorMsg',
        );
    }

    if ( $Param{CcIsLocalAddress} && $Param{Errors} && !$Param{Errors}->{CcErrorType} ) {
        $LayoutObject->Block(
            Name => 'CcIsLocalAddressServerErrorMsg',
            Data => \%Param,
        );
    }

    if ( $Param{BccInvalid} && $Param{Errors} && !$Param{Errors}->{BccErrorType} ) {
        $LayoutObject->Block(
            Name => 'BccServerErrorMsg',
        );
    }

    if ( $Param{BccIsLocalAddress} && $Param{Errors} && !$Param{Errors}->{BccErrorType} ) {
        $LayoutObject->Block(
            Name => 'BccIsLocalAddressServerErrorMsg',
            Data => \%Param,
        );
    }

    # render dynamic fields
    {
        my %DynamicFieldConfigs = map { $_->{Name} => $_ } $Self->{DynamicField}->@*;

        $Param{DynamicFieldHTML} = $Kernel::OM->Get('Kernel::Output::HTML::DynamicField::Mask')->EditSectionRender(
            Content              => $Self->{MaskDefinition},
            DynamicFields        => \%DynamicFieldConfigs,
            LayoutObject         => $LayoutObject,
            ParamObject          => $Kernel::OM->Get('Kernel::System::Web::Request'),
            DynamicFieldValues   => $Param{DFValues},
            PossibleValuesFilter => $Param{DFPossibleValues},
            Errors               => $Param{DFErrors},
            Object               => {
                CustomerID     => $Param{CustomerID},
                CustomerUserID => $Param{CustomerUserID},
                UserID         => $Self->{UserID},
                $Param{DFValues}->%*,
            },
        );
    }

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

    # Show the customer user address book if the module is registered.
    if ( $ConfigObject->Get('Frontend::Module')->{AgentCustomerUserAddressBook} ) {
        $Param{OptionCustomerUserAddressBook} = 1;
    }

    # show attachments
    ATTACHMENT:
    for my $Attachment ( @{ $Param{Attachments} } ) {
        if (
            $Attachment->{ContentID}
            && $LayoutObject->{BrowserRichText}
            && ( $Attachment->{ContentType} =~ /image/i )
            )
        {
            my $ContentIDLinkEncode = $LayoutObject->LinkEncode( $Attachment->{ContentID} );
            if ( $Param{Body} =~ /ContentID=\Q$ContentIDLinkEncode\E/i ) {
                next ATTACHMENT;
            }
        }

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

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

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

    # explanatory message about asterisk
    if ( $ConfigObject->Get('Ticket::Frontend::AsteriskExplanation') ) {
        $LayoutObject->Block(
            Name => 'AsteriskExplanation',
        );
    }

    # create & return output
    return $LayoutObject->Output(
        TemplateFile => 'AgentTicketForward',
        Data         => {
            %Param,
            FormDraft      => $Config->{FormDraft},
            FormDraftID    => $Self->{LoadedFormDraftID},
            FormDraftTitle => $LoadedFormDraft ? $LoadedFormDraft->{Title} : '',
            FormDraftMeta  => $LoadedFormDraft,
        },
    );
}

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

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

    # get params
    my %GetParam = %{ $Self->{GetParam} };

    # hash for check duplicated entries
    my %AddressesList;
    my @MultipleCustomer;
    my $CustomersNumberTo = $ParamObject->GetParam( Param => 'CustomerTicketCounterToCustomer' ) || 0;
    my $Selected          = $ParamObject->GetParam( Param => 'CustomerSelected' )                || '';

    # get check item object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

    if ($CustomersNumberTo) {
        my $CustomerCounter = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberTo ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElement = $ParamObject->GetParam( Param => 'CustomerTicketText_' . $Count );

            next COUNT unless $CustomerElement;

            my $CustomerSelected = ( $Selected eq $Count ? 'checked ' : '' );
            my $CustomerKey      = $ParamObject->GetParam( Param => 'CustomerKey_' . $Count )   || '';
            my $CustomerQueue    = $ParamObject->GetParam( Param => 'CustomerQueue_' . $Count ) || '';

            if ( $GetParam{To} ) {
                $GetParam{To} .= ', ' . $CustomerElement;
            }
            else {
                $GetParam{To} = $CustomerElement;
            }

            # check email address
            my $CustomerErrorMsg = 'CustomerGenericServerErrorMsg';
            my $CustomerError    = '';
            for my $Email ( Mail::Address->parse($CustomerElement) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsg = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerError = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElement} && $CustomerError eq '' ) {
                $CustomerErrorMsg = 'IsDuplicatedServerErrorMsg';
                $CustomerError    = 'ServerError';
            }

            my $CustomerDisabled = '';
            my $CountAux         = $CustomerCounter++;
            if ( $CustomerError ne '' ) {
                $CustomerDisabled = 'disabled="disabled"';
                $CountAux         = $Count . 'Error';
            }

            if ( $CustomerQueue ne '' ) {
                $CustomerQueue = $Count;
            }

            push @MultipleCustomer, {
                Count            => $CountAux,
                CustomerElement  => $CustomerElement,
                CustomerSelected => $CustomerSelected,
                CustomerKey      => $CustomerKey,
                CustomerError    => $CustomerError,
                CustomerErrorMsg => $CustomerErrorMsg,
                CustomerDisabled => $CustomerDisabled,
                CustomerQueue    => $CustomerQueue,
            };
            $AddressesList{$CustomerElement} = 1;
        }
    }

    my @MultipleCustomerCc;
    my $CustomersNumberCc = $ParamObject->GetParam( Param => 'CustomerTicketCounterCcCustomer' ) || 0;

    if ($CustomersNumberCc) {
        my $CustomerCounterCc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberCc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementCc = $ParamObject->GetParam( Param => 'CcCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementCc;

            my $CustomerKeyCc   = $ParamObject->GetParam( Param => 'CcCustomerKey_' . $Count )   || '';
            my $CustomerQueueCc = $ParamObject->GetParam( Param => 'CcCustomerQueue_' . $Count ) || '';

            if ( $GetParam{Cc} ) {
                $GetParam{Cc} .= ', ' . $CustomerElementCc;
            }
            else {
                $GetParam{Cc} = $CustomerElementCc;
            }

            # check email address
            my $CustomerErrorMsgCc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorCc    = '';
            for my $Email ( Mail::Address->parse($CustomerElementCc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsgCc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorCc = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElementCc} && $CustomerErrorCc eq '' ) {
                $CustomerErrorMsgCc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorCc    = 'ServerError';
            }

            my $CustomerDisabledCc = '';
            my $CountAuxCc         = $CustomerCounterCc++;
            if ( $CustomerErrorCc ne '' ) {
                $CustomerDisabledCc = 'disabled="disabled"';
                $CountAuxCc         = $Count . 'Error';
            }

            if ( $CustomerQueueCc ne '' ) {
                $CustomerQueueCc = $Count;
            }

            push @MultipleCustomerCc, {
                Count            => $CountAuxCc,
                CustomerElement  => $CustomerElementCc,
                CustomerKey      => $CustomerKeyCc,
                CustomerError    => $CustomerErrorCc,
                CustomerErrorMsg => $CustomerErrorMsgCc,
                CustomerDisabled => $CustomerDisabledCc,
                CustomerQueue    => $CustomerQueueCc,
            };
            $AddressesList{$CustomerElementCc} = 1;
        }
    }

    my @MultipleCustomerBcc;
    my $CustomersNumberBcc = $ParamObject->GetParam( Param => 'CustomerTicketCounterBccCustomer' ) || 0;

    if ($CustomersNumberBcc) {
        my $CustomerCounterBcc = 1;

        COUNT:
        for my $Count ( 1 .. $CustomersNumberBcc ) {
            last COUNT if $Count > 1_000;    # bail out when the number of customers is abnormally high

            my $CustomerElementBcc = $ParamObject->GetParam( Param => 'BccCustomerTicketText_' . $Count );

            next COUNT unless $CustomerElementBcc;

            my $CustomerKeyBcc   = $ParamObject->GetParam( Param => 'BccCustomerKey_' . $Count )   || '';
            my $CustomerQueueBcc = $ParamObject->GetParam( Param => 'BccCustomerQueue_' . $Count ) || '';

            if ( $GetParam{Bcc} ) {
                $GetParam{Bcc} .= ', ' . $CustomerElementBcc;
            }
            else {
                $GetParam{Bcc} = $CustomerElementBcc;
            }

            # check email address
            my $CustomerErrorMsgBcc = 'CustomerGenericServerErrorMsg';
            my $CustomerErrorBcc    = '';
            for my $Email ( Mail::Address->parse($CustomerElementBcc) ) {
                if ( !$CheckItemObject->CheckEmail( Address => $Email->address() ) ) {
                    $CustomerErrorMsgBcc = $CheckItemObject->CheckErrorType()
                        . 'ServerErrorMsg';
                    $CustomerErrorBcc = 'ServerError';
                }
            }

            # check for duplicated entries
            if ( defined $AddressesList{$CustomerElementBcc} && $CustomerErrorBcc eq '' ) {
                $CustomerErrorMsgBcc = 'IsDuplicatedServerErrorMsg';
                $CustomerErrorBcc    = 'ServerError';
            }

            my $CustomerDisabledBcc = '';
            my $CountAuxBcc         = $CustomerCounterBcc++;
            if ( $CustomerErrorBcc ne '' ) {
                $CustomerDisabledBcc = 'disabled="disabled"';
                $CountAuxBcc         = $Count . 'Error';
            }

            if ( $CustomerQueueBcc ne '' ) {
                $CustomerQueueBcc = $Count;
            }

            push @MultipleCustomerBcc, {
                Count            => $CountAuxBcc,
                CustomerElement  => $CustomerElementBcc,
                CustomerKey      => $CustomerKeyBcc,
                CustomerError    => $CustomerErrorBcc,
                CustomerErrorMsg => $CustomerErrorMsgBcc,
                CustomerDisabled => $CustomerDisabledBcc,
                CustomerQueue    => $CustomerQueueBcc,
            };
            $AddressesList{$CustomerElementBcc} = 1;
        }
    }

    return (
        GetParam            => \%GetParam,
        MultipleCustomer    => \@MultipleCustomer,
        MultipleCustomerCc  => \@MultipleCustomerCc,
        MultipleCustomerBcc => \@MultipleCustomerBcc,
    );
}

1;
</File>
        <File Location="Custom/Kernel/Modules/AgentTicketMerge.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/Modules/AgentTicketMerge.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::Modules::AgentTicketMerge;

use strict;
use warnings;

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

our $ObjectManagerDisabled = 1;

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

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

    return $Self;
}

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

    my %Error;
    my %GetParam;

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

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

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

    # get config param
    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( 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) {

        my %AclActionLookup = reverse %AclAction;

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

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

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

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

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

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

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

                # Show lock state.
                if ($Success) {
                    $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(
                    Value     => $Ticket{Number},
                    Type      => 'Small',
                    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 {

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

    # merge action
    if ( $Self->{Subaction} eq 'Merge' ) {

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

        # get all parameters
        for my $Parameter (qw( From To Subject Body InformSender MainTicketNumber )) {
            $GetParam{$Parameter} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Parameter )
                || '';
        }

        # 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 check item object
        my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

        # removing blank spaces from the ticket number
        $CheckItemObject->StringClean(
            StringRef => \$GetParam{'MainTicketNumber'},
            TrimLeft  => 1,
            TrimRight => 1,
        );

        # check some stuff
        my $MainTicketID = $TicketObject->TicketIDLookup(
            TicketNumber => $GetParam{'MainTicketNumber'},
        );

        # check if source and target TicketID are the same (bug#8667)
        if ( $MainTicketID && $MainTicketID == $Self->{TicketID} ) {
            $LayoutObject->FatalError(
                Message => Translatable('Can\'t merge ticket with itself!'),
            );
        }

        # check for errors
        if ( !$MainTicketID ) {
            $Error{'MainTicketNumberInvalid'} = 'ServerError';
        }

        if ( $GetParam{InformSender} ) {
            for my $Parameter (qw( To Subject Body )) {
                if ( !$GetParam{$Parameter} ) {
                    $Error{ $Parameter . 'Invalid' } = 'ServerError';
                }
            }

            # check forward email address(es)
            if ( $GetParam{To} ) {
                for my $Email ( Mail::Address->parse( $GetParam{To} ) ) {
                    my $Address = $Email->address();

# Rother OSS / DiscreteSystemAddresses
#                    if (
#                        $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress( Address => $Address )
#                        )
                    my $IsLocal = $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress(
                        Address  => $Address,
                        TicketID => $Self->{TicketID},
                    );
                    if ($IsLocal)
# EO DiscreteSystemAddresses
                    {
                        $LayoutObject->Block( Name => 'ToCustomerGenericServerErrorMsg' );
                        $Error{'ToInvalid'} = 'ServerError';
                    }

                    # check email address
                    elsif ( !$CheckItemObject->CheckEmail( Address => $Address ) ) {
                        my $ToErrorMsg =
                            'To'
                            . $CheckItemObject->CheckErrorType()
                            . 'ServerErrorMsg';
                        $LayoutObject->Block( Name => $ToErrorMsg );
                        $Error{'ToInvalid'} = 'ServerError';
                    }
                }
            }
            else {
                $LayoutObject->Block( Name => 'ToCustomerGenericServerErrorMsg' );
            }
        }

        if (%Error) {
            my $Output = $LayoutObject->Header(
                Type      => 'Small',
                BodyClass => 'Popup',
            );

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

            }

            $Param{InformSenderChecked} = $GetParam{InformSender} ? 'checked ' : '';

            $Output .= $LayoutObject->Output(
                TemplateFile => 'AgentTicketMerge',
                Data         => { %Param, %GetParam, %Ticket, %Error },
            );
            $Output .= $LayoutObject->Footer(
                Type => 'Small',
            );
            return $Output;
        }

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

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

        # error screen, don't show ticket
        if ( !$Access ) {
            return $LayoutObject->NoPermission( WithHeader => 'yes' );
        }

        my $TicketMergeResult = $TicketObject->TicketMerge(
            MainTicketID  => $MainTicketID,
            MergeTicketID => $Self->{TicketID},
            UserID        => $Self->{UserID},
        );

        # check errors
        if (
            $Self->{TicketID} == $MainTicketID
            || !$TicketMergeResult
            || $TicketMergeResult eq 'NoValidMergeStates'
            )
        {
            my $Output = $LayoutObject->Header(
                Type      => 'Small',
                BodyClass => 'Popup',
            );

            if ( $TicketMergeResult eq 'NoValidMergeStates' ) {
                $Output .= $LayoutObject->Notify(
                    Priority => 'Error',
                    Info     => 'No merge state found! Please add a valid merge state before merge action.',
                );
            }

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

            $Output .= $LayoutObject->Output(
                TemplateFile => 'AgentTicketMerge',
                Data         => { %Param, %Ticket },
            );
            $Output .= $LayoutObject->Footer(
                Type => 'Small',
            );
            return $Output;
        }
        else {

            # send customer info?
            if ( $GetParam{InformSender} ) {
                my $MimeType = 'text/plain';
                if ( $LayoutObject->{BrowserRichText} ) {
                    $MimeType = 'text/html';

                    # verify html document
                    $GetParam{Body} = $LayoutObject->RichTextDocumentComplete(
                        String => $GetParam{Body},
                    );
                }
                my %Ticket = $TicketObject->TicketGet( TicketID => $Self->{TicketID} );
                $GetParam{Body} =~ s/(&lt;|<)OTOBO_TICKET(&gt;|>)/$Ticket{TicketNumber}/g;
                $GetParam{Body}
                    =~ s/(&lt;|<)OTOBO_MERGE_TO_TICKET(&gt;|>)/$GetParam{'MainTicketNumber'}/g;

                my $EmailArticleBackendObject = $Kernel::OM->Get('Kernel::System::Ticket::Article')->BackendForChannel(
                    ChannelName => 'Email',
                );

                my $ArticleID = $EmailArticleBackendObject->ArticleSend(
                    TicketID             => $Self->{TicketID},
                    SenderType           => 'agent',
                    IsVisibleForCustomer => 1,
                    HistoryType          => 'SendAnswer',
                    HistoryComment       => "Merge info to '$GetParam{To}'.",
                    From                 => $GetParam{From},
                    Email                => $GetParam{Email},
                    To                   => $GetParam{To},
                    Subject              => $GetParam{Subject},
                    UserID               => $Self->{UserID},
                    Body                 => $GetParam{Body},
                    Charset              => $LayoutObject->{UserCharset},
                    MimeType             => $MimeType,
                );
                if ( !$ArticleID ) {

                    # error page
                    return $LayoutObject->ErrorScreen();
                }
            }

            # redirect to merged ticket
            return $LayoutObject->PopupClose(
                URL => "Action=AgentTicketZoom;TicketID=$MainTicketID",
            );
        }
    }
    else {
        my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

        # Get last customer article.
        my @Articles = $ArticleObject->ArticleList(
            TicketID   => $Self->{TicketID},
            SenderType => 'customer',
            OnlyLast   => 1,
        );

        # If the ticket has no customer article, get the last agent article.
        if ( !@Articles ) {
            @Articles = $ArticleObject->ArticleList(
                TicketID   => $Self->{TicketID},
                SenderType => 'agent',
                OnlyLast   => 1,
            );
        }

        # Finally, if everything failed, get latest article.
        if ( !@Articles ) {
            @Articles = $ArticleObject->ArticleList(
                TicketID => $Self->{TicketID},
                OnlyLast => 1,
            );
        }

        my %Article;
        for my $Article (@Articles) {
            %Article = $ArticleObject->BackendForArticle( %{$Article} )->ArticleGet(
                %{$Article},
                DynamicFields => 1,
            );
        }

        # merge box
        my $Output = $LayoutObject->Header(
            Value     => $Ticket{TicketNumber},
            Type      => 'Small',
            BodyClass => 'Popup',
        );

        # prepare salutation
        my $TemplateGenerator = $Kernel::OM->Get('Kernel::System::TemplateGenerator');
        my $Salutation        = $TemplateGenerator->Salutation(
            TicketID  => $Self->{TicketID},
            ArticleID => $Article{ArticleID},
            Data      => {
                %Article,

                # NOTE in case that we answer to a customer, the former sender (the customer)
                #   now becomes the new recipient and the former recipient (the OTOBO system)
                #   now becomes the new sender
                From => $Article{SenderType} eq 'customer' ? $Article{To}   : $Article{From},
                To   => $Article{SenderType} eq 'customer' ? $Article{From} : $Article{To},
            },
            UserID => $Self->{UserID},
        );

        # prepare signature
        my $Signature = $TemplateGenerator->Signature(
            TicketID  => $Self->{TicketID},
            ArticleID => $Article{ArticleID},
            Data      => {%Article},
            UserID    => $Self->{UserID},
        );

        # prepare subject ...
        $Article{Subject} = $TicketObject->TicketSubjectBuild(
            TicketNumber => $Ticket{TicketNumber},
            Subject      => $Article{Subject} || '',
        );

        # prepare from ...
        if ( $Article{SenderType} eq 'customer' ) {
            $Article{To} = $Article{From};
        }

        my %Address = $Kernel::OM->Get('Kernel::System::Queue')->GetSystemAddress( QueueID => $Ticket{QueueID} );
        $Article{From} = "$Address{RealName} <$Address{Email}>";

        # add salutation and signature to body
        if ( $LayoutObject->{BrowserRichText} ) {
            my $Body = $LayoutObject->Ascii2RichText(
                String => $ConfigObject->Get('Ticket::Frontend::MergeText'),
            );
            $Article{Body} = $Salutation
                . '<br/><br/>'
                . $Body
                . '<br/><br/>'
                . $Signature;
        }
        else {
            $Article{Body} = $Salutation
                . "\n\n"
                . $ConfigObject->Get('Ticket::Frontend::MergeText')
                . "\n\n"
                . $Signature;
        }

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

        # explanatory message about asterisk
        if ( $ConfigObject->Get('Ticket::Frontend::AsteriskExplanation') ) {
            $LayoutObject->Block(
                Name => 'AsteriskExplanation',
            );
        }

        $Output .= $LayoutObject->Output(
            TemplateFile => 'AgentTicketMerge',
            Data         => { %Param, %Ticket, %Article, }
        );
        $Output .= $LayoutObject->Footer(
            Type => 'Small',
        );
        return $Output;
    }
}

1;
</File>
        <File Location="Custom/Kernel/Output/HTML/ArticleAction/AgentTicketCompose.pm" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# $origin: otobo - 6efdc7bf2a3325277cd79a60f0f2407f8ad59e87 - Kernel/Output/HTML/ArticleAction/AgentTicketCompose.pm
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

package Kernel::Output::HTML::ArticleAction::AgentTicketCompose;

use strict;
use warnings;

use Kernel::Language qw(Translatable);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Log',
    'Kernel::System::Queue',
    'Kernel::System::SystemAddress',
    'Kernel::System::Ticket',
);

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

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

    return $Self;
}

# optional AclActionLookup
sub CheckAccess {
    my ( $Self, %Param ) = @_;

    # Check needed stuff.
    for my $Needed (qw(Ticket Article ChannelName UserID)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );
            return;
        }
    }

    if ( $Param{ChannelName} eq 'Email' && $Param{Article}->{SenderType} eq 'system' ) {

        # skip email notifications
        return;
    }

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

    # check if module is registered
    return if !$ConfigObject->Get('Frontend::Module')->{AgentTicketCompose};

    # check Acl
    return if !$Param{AclActionLookup}->{AgentTicketCompose};

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

    my $Config = $ConfigObject->Get('Ticket::Frontend::AgentTicketCompose');
    if ( $Config->{Permission} ) {
        my $Ok = $TicketObject->TicketPermission(
            Type     => $Config->{Permission},
            TicketID => $Param{Ticket}->{TicketID},
            UserID   => $Param{UserID},
            LogNo    => 1,
        );
        return if !$Ok;
    }
    if ( $Config->{RequiredLock} ) {
        my $Locked = $TicketObject->TicketLockGet(
            TicketID => $Param{Ticket}->{TicketID}
        );
        if ($Locked) {
            my $AccessOk = $TicketObject->OwnerCheck(
                TicketID => $Param{Ticket}->{TicketID},
                OwnerID  => $Param{UserID},
            );
            return if !$AccessOk;
        }
    }

    return 1;
}

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

    # Check needed stuff.
    for my $Needed (qw(Ticket Article StandardTemplates Type UserID)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );
            return;
        }
    }

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

    # get StandardResponsesStrg
    # my %StandardResponseHash = %{ $Param{StandardResponses} || {} };
    my %StandardResponseHash = %{ $Param{StandardTemplates}->{Answer} || {} };

    # get revers StandardResponseHash because we need to sort by Values
    # from %ReverseStandardResponseHash we get value of Key by %StandardResponseHash Value
    # and @StandardResponseArray is created as array of hashes with elements Key and Value

    my %ReverseStandardResponseHash = reverse %StandardResponseHash;
    my @StandardResponseArray       = map {
        {
            Key   => $ReverseStandardResponseHash{$_},
            Value => $_
        }
    } sort values %StandardResponseHash;

    # use this array twice (also for Reply All), so copy it first
    my @StandardResponseArrayReplyAll = @StandardResponseArray;

    my @MenuItems;
    if (@StandardResponseArray) {

        # build HTML string
        my $StandardResponsesStrg = $LayoutObject->BuildSelection(
            Name         => 'ResponseID',
            ID           => 'ResponseID' . $Param{Article}->{ArticleID},
            Class        => 'Modernize Small',
            Data         => \@StandardResponseArray,
            PossibleNone => 1,
            Translation  => 1,
        );

        push @MenuItems, {
            ItemType              => 'Dropdown',
            DropdownType          => 'Reply',
            StandardResponsesStrg => $StandardResponsesStrg,
            Name                  => Translatable('Reply'),
            Class                 => 'AsPopup PopupType_TicketAction',
            Action                => 'AgentTicketCompose',
            FormID                => 'Reply' . $Param{Article}->{ArticleID},
            ResponseElementID     => 'ResponseID' . $Param{Article}->{ArticleID},
            Type                  => $Param{Type},
        };
    }

    # check if reply all is needed
    my $Recipients = '';
    KEY:
    for my $Key (qw(From To Cc)) {
        next KEY if !$Param{Article}->{$Key};
        if ($Recipients) {
            $Recipients .= ', ';
        }
        $Recipients .= $Param{Article}->{$Key};
    }
    my $RecipientCount = 0;
    if ($Recipients) {
        my $EmailParser = Kernel::System::EmailParser->new(
            %{$Self},
            Mode => 'Standalone',
        );
        my @Addresses = $EmailParser->SplitAddressLine( Line => $Recipients );
        ADDRESS:
        for my $Address (@Addresses) {
            my $Email = $EmailParser->GetEmailAddress( Email => $Address );
            next ADDRESS if !$Email;
            my $IsLocal = $Kernel::OM->Get('Kernel::System::SystemAddress')->SystemAddressIsLocalAddress(
                Address => $Email,
# Rother OSS / DiscreteSystemAddresses
                TicketID => $Param{Ticket}->{TicketID},
# EO DiscreteSystemAddresses
            );
            next ADDRESS if $IsLocal;
            $RecipientCount++;
        }
    }
    if ( @StandardResponseArrayReplyAll && $RecipientCount > 1 ) {

        my $StandardResponsesStrg = $LayoutObject->BuildSelection(
            Name         => 'ResponseID',
            ID           => 'ResponseIDAll' . $Param{Article}->{ArticleID},
            Class        => 'Modernize Small',
            Data         => \@StandardResponseArrayReplyAll,
            PossibleNone => 1,
            Translation  => 1,
        );

        push @MenuItems, {
            ItemType              => 'Dropdown',
            DropdownType          => 'Reply',
            StandardResponsesStrg => $StandardResponsesStrg,
            Name                  => Translatable('Reply All'),
            Class                 => 'AsPopup PopupType_TicketAction',
            Action                => 'AgentTicketCompose',
            FormID                => 'ReplyAll' . $Param{Article}->{ArticleID},
            ReplyAll              => 1,
            ResponseElementID     => 'ResponseIDAll' . $Param{Article}->{ArticleID},
            Type                  => $Param{Type},
        };
    }

    return @MenuItems;
}

1;
</File>
        <File Location="scripts/test/PostMaster/AddressPool.t" Permission="660" Encode="Base64"># --
# OTOBO is a web-based ticketing system for service organisations.
# --
# Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
# Copyright (C) 2019-2026 Rother OSS GmbH, https://otobo.io/
# --
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
# --

use strict;
use warnings;
use utf8;

# Set up the test driver $Self when we are running as a standalone script.
use Kernel::System::UnitTest::MockTime qw(:all);
use Kernel::System::UnitTest::RegisterDriver;

our $Self;

use Kernel::System::PostMaster;

# get needed objects
my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
$ConfigObject->Set(
    Key   => 'CheckEmailAddresses',
    Value => 0,
);

my $QueueObject         = $Kernel::OM->Get('Kernel::System::Queue');
my $TicketObject        = $Kernel::OM->Get('Kernel::System::Ticket');
my $ArticleObject       = $Kernel::OM->Get('Kernel::System::Ticket::Article');
my $SystemAddressObject = $Kernel::OM->Get('Kernel::System::SystemAddress');

# get helper object
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase  => 1,
        UseTmpArticleDir => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
FixedTimeSet();

# Pool |        Address | Queue
#   P1   a1p1@otobo.org      q1
#   P1   a2p1@otobo.org
#   P2   a1p2@otobo.org      q2
#   P2   a2p2@otobo.org      q5
#   P3   a1p3@otobo.org      q3
#   P3   a2p3@otobo.org
#          ax@otobo.org

# Pool | DefQueue
#   P1         q4
#   P2         q5
#   P3         q3
#              q5

# add system addresses
my %SystemAddressIDs;
for my $Address (qw/a1P1@otobo.org a1p2@otoBo.org a1p3@otobo.org a2p2@otobo.org unused@otobo.org/) {
    my $SystemAddressID = $SystemAddressObject->SystemAddressAdd(
        Name     => $Address,
        Realname => 'APTest',
        ValidID  => 1,
        QueueID  => 1,
        UserID   => 1,
    );

    $SystemAddressIDs{$Address} = $SystemAddressID;
}

my %QueueAddresses = (
    q1 => 'a1P1@otobo.org',
    q2 => 'a1p2@otoBo.org',
    q3 => 'a1p3@otobo.org',
    q4 => 'unused@otobo.org',
    q5 => 'a2p2@otobo.org',
);

# add queues
my %QueueIDs;
for my $Queue ( keys %QueueAddresses ) {
    my $QueueID = $QueueObject->QueueAdd(
        Name            => $Queue,
        ValidID         => 1,
        GroupID         => 1,
        SystemAddressID => $SystemAddressIDs{ $QueueAddresses{$Queue} },
        SalutationID    => 1,
        SignatureID     => 1,
        UserID          => 1,
    );

    $QueueIDs{$Queue} = $QueueID;

    # update system address
    my %SystemAddressData = $SystemAddressObject->SystemAddressGet(
        ID => $SystemAddressIDs{ $QueueAddresses{$Queue} },
    );

    $SystemAddressData{QueueID} = $QueueID;

    $SystemAddressObject->SystemAddressUpdate(
        %SystemAddressData,
        UserID => 1,
    );
}

# set address pools
{
    my %AddressPoolData = (
        Pool01 => {
            Name   => 'P1',
            Emails => [
                'a1p1@otobo.org',
                'a2p1@otobo.org',
            ],
            DefaultQueue => 'q4',
        },
        Pool02 => {
            Name   => 'P2',
            Emails => [
                'a1p2@otobo.org',
                'a2p2@otobo.org',
            ],
            DefaultQueue => 'q5',
        },
        Pool03 => {
            Name   => 'P3',
            Emails => [
                'A1p3@otobo.org',
                'A2p3@otobo.org',
            ],
            DefaultQueue => 'q3',
        },
    );

    $Helper->ConfigSettingChange(
        Valid => 1,
        Key   => 'PostMaster::AddressPool',
        Value => \%AddressPoolData,
    );
}

# Start tests

# Test: Ticket#x in q1 (ignore defqueue of a2p1), Ticket#y in q2, nothing in q5
my $Email = GenerateEmail(
    To        => 'a2p1@otobo.orG, a1p1@otobo.orG, a1p2@otobo.orG, not@ours.com',
    Subject   => 'Initial',
    MessageID => '<20230214002814.AddressPools1@test>',
);

my ( $Return, @TicketIDs ) = ReadEmail( $Email, 1 );

# two tickets should be created...
$Self->Is(
    scalar @TicketIDs,
    2,
    "Mail1 - create two tickets.",
);

my @TestTickets;
for my $ID (@TicketIDs) {
    push @TestTickets, {
        $TicketObject->TicketGet( TicketID => $ID )
    };
}

# ticket 1 should be in q1
$Self->Is(
    $TestTickets[0]{Queue} // '',
    'q1',
    "Mail1 - Ticket1 is in q1.",
);

# ticket 2 should be in q2
$Self->Is(
    $TestTickets[1]{Queue} // '',
    'q2',
    "Mail1 - Ticket2 is in q2.",
);

my @Articles = $ArticleObject->ArticleList(
    TicketID => $TestTickets[0]{TicketID},
);

# ticket 1 should have exactly 1 article
$Self->Is(
    scalar @Articles,
    1,
    "Mail1 - Ticket1 has 1 article.",
);

# Test: Ticket1#Number -> FollowUp in q2, New in q3 (defqueue)
my $NewSubject = $TicketObject->TicketSubjectBuild(
    TicketNumber => $TestTickets[0]{TicketNumber},
    Subject      => 'Initial',
    Action       => 'Reply',
);

$Email = GenerateEmail(
    To        => 'a1p2@oTobo.org, a2p3@otobo.orG',
    Subject   => $NewSubject,
    MessageID => '<20230214002814.AddressPools2@test>',
);

( $Return, @TicketIDs ) = ReadEmail($Email);

# two articles should be created...
$Self->Is(
    scalar @TicketIDs,
    2,
    "Mail2 - create two articles.",
);

# ticket 1 should be ticket 2 of last email
$Self->Is(
    $TicketIDs[0],
    $TestTickets[1]{TicketID},
    "Mail2 - Ticket1 is old Ticket2.",
);

push @TestTickets, {
    $TicketObject->TicketGet( TicketID => $TicketIDs[1] ),
};

# ticket 3 should be in q3
$Self->Is(
    $TestTickets[2]{Queue} // '',
    'q3',
    "Mail2 - Ticket2 is in q3.",
);

# keep for later
my $Email2 = $Email;

# Test: Re Ticket3#Number creates article in Ticket1
$NewSubject = $TicketObject->TicketSubjectBuild(
    TicketNumber => $TestTickets[2]{TicketNumber},
    Subject      => 'Initial',
    Action       => 'Reply',
);

$Email = GenerateEmail(
    To        => 'a1p1@otobo.org',
    Subject   => $NewSubject,
    MessageID => '<20230214002814.AddressPools3@test>',
);

( $Return, @TicketIDs ) = ReadEmail($Email);

# ticket 1 should be ticket 1 of the first email
$Self->Is(
    $TicketIDs[0],
    $TestTickets[0]{TicketID},
    "Mail3 - Ticket1 is old Ticket1.",
);

@Articles = $ArticleObject->ArticleList(
    TicketID => $TestTickets[0]{TicketID},
);

# ticket 1 should have 2 articles
$Self->Is(
    scalar @Articles,
    2,
    "Mail3 - Ticket1 has 2 articles.",
);

# clean all Ticketnumbers
my $CleanedSubject = $TicketObject->TicketSubjectClean(
    TicketNumber => $TestTickets[0]{TicketNumber},
    Subject      => ''
        . '[' . $ConfigObject->Get('Ticket::Hook') . $ConfigObject->Get('Ticket::HookDivider') . $TestTickets[0]{TicketNumber} . ']'
        . '[' . $ConfigObject->Get('Ticket::Hook') . $ConfigObject->Get('Ticket::HookDivider') . $TestTickets[1]{TicketNumber} . ']'
        . '[' . $ConfigObject->Get('Ticket::Hook') . $ConfigObject->Get('Ticket::HookDivider') . $TestTickets[2]{TicketNumber} . ']'
        . '[' . $ConfigObject->Get('Ticket::Hook') . $ConfigObject->Get('Ticket::HookDivider') . '20030301123412340001] Test',
    Size => 0,
);

# all linked ticket numbers are cleaned
$Self->Is(
    $CleanedSubject,
    '[' . $ConfigObject->Get('Ticket::Hook') . $ConfigObject->Get('Ticket::HookDivider') . '20030301123412340001] Test',
    "TicketSubjectClean includes interdiv ticket numbers.",
);

# Test: standard case
$Email = GenerateEmail(
    To        => 'ax@otobo.org',
    Subject   => 'Kein Pool',
    MessageID => '<20230214002814.AddressPools4@test>',
);

( $Return, @TicketIDs ) = ReadEmail($Email);

# one ticket should be created
$Self->Is(
    scalar @TicketIDs,
    1,
    "Mail4 - a ticket is created.",
);

my %Ticket = $TicketObject->TicketGet( TicketID => $TicketIDs[0] );

# ticket is in default queue
$Self->Is(
    $Ticket{Queue},
    $ConfigObject->Get('PostmasterDefaultQueue'),
    "Mail4 - ticket is created in def queue.",
);

# Test: ignore already received mails
( $Return, @TicketIDs ) = ReadEmail($Email2);

# no ticket should be created
$Self->Is(
    $Return,
    5,
    "Mail5 - mail ignored.",
);

# no ticket should be created
$Self->Is(
    scalar @TicketIDs,
    0,
    "Mail5 - no article.",
);

# Test: Re Ticket3#Number creates article in Ticket1
$NewSubject = $TicketObject->TicketSubjectBuild(
    TicketNumber => $TestTickets[2]{TicketNumber},
    Subject      => 'Initial',
    Action       => 'Reply',
);

$Email = GenerateEmail(
    To        => 'a1p1@otobo.org, a1p2@otobo.org',
    Subject   => $NewSubject,
    MessageID => '<20230214002814.AddressPools6@test>',
    XHeader   => "\nX-OTOBO-FollowUp-Queue: q5",
);

( $Return, @TicketIDs ) = ReadEmail($Email);

# ticket 1 should be ticket 1 of the first email
$Self->Is(
    $TicketIDs[0],
    $TestTickets[0]{TicketID},
    "Mail6 - Ticket1 is old Ticket1.",
);

# ticket 2 should be ticket 2 of the first email
$Self->Is(
    $TicketIDs[1],
    $TestTickets[1]{TicketID},
    "Mail6 - Ticket2 is old Ticket2.",
);

for my $i ( 0, 1 ) {
    $TestTickets[$i] = { $TicketObject->TicketGet( TicketID => $TicketIDs[$i] ) };
}

# ticketqueue 1 should still be q1
$Self->Is(
    $TestTickets[0]->{Queue},
    'q1',
    "Mail6 - Queue of Ticket 1 is q1.",
);

# ticketqueue 2 should now be q5 of the X-OTOBO-FollowUp-Queue header
$Self->Is(
    $TestTickets[1]->{Queue},
    'q5',
    "Mail6 - Queue of Ticket 2 is q5.",
);

# Test: Do not ignore mails sent from the system
my $ArticleBackendObject = $Kernel::OM->Get('Kernel::System::Ticket::Article')->BackendForChannel( ChannelName => 'Email' );
$ArticleBackendObject->ArticleCreate(
    TicketID             => $TestTickets[0]{TicketID},
    SenderType           => 'agent',
    IsVisibleForCustomer => 1,
    UserID               => 1,
    From                 => '"P1" <a1p1@otobo.org>',
    To                   => '"P2" <a1p2@otobo.org>',
    Subject              => 'some short description',
    Body                 => 'the message text',
    MessageID            => '<20230214002814.AddressPools7@test>',
    ContentType          => 'text/plain; charset=ISO-8859-15',
    HistoryType          => 'AddNote',
    HistoryComment       => 'Some free text!',
    NoAgentNotify        => 1,
);

$NewSubject = $TicketObject->TicketSubjectBuild(
    TicketNumber => $TestTickets[0]{TicketNumber},
    Subject      => 'some short description',
    Action       => 'Reply',
);

$Email = GenerateEmail(
    To        => 'a1p2@otobo.org',
    Subject   => $NewSubject,
    MessageID => '<20230214002814.AddressPools7@test>',
);

( $Return, @TicketIDs ) = ReadEmail($Email);

# one article should be created
$Self->Is(
    scalar @TicketIDs,
    1,
    "Mail7 - one article.",
);

# ticket 1 should be ticket 2 of the first email
$Self->Is(
    $TicketIDs[0],
    $TestTickets[1]{TicketID},
    "Mail7 - Ticket1 is old Ticket2.",
);

( $Return, @TicketIDs ) = ReadEmail($Email);

# on second round though, mail should be ignored
$Self->Is(
    $Return,
    5,
    "Mail7.5 - mail ignored.",
);

# no ticket should be created
$Self->Is(
    scalar @TicketIDs,
    0,
    "Mail7.5 - no article.",
);

# Test: Dispatching via Queue
$Email = GenerateEmail(
    To        => 'a1p1@otobo.org',
    Subject   => 'New',
    MessageID => '<20230214002814.AddressPools8@test>',
);

( $Return, @TicketIDs ) = ReadEmail( $Email, $QueueIDs{q2} );

# two tickets should be created...
$Self->Is(
    scalar @TicketIDs,
    2,
    "Mail8 - create two tickets.",
);

my @Test8Tickets;
for my $ID (@TicketIDs) {
    push @Test8Tickets, {
        $TicketObject->TicketGet( TicketID => $ID )
    };
}

# ticket 1 should be in q2
$Self->Is(
    $Test8Tickets[0]{Queue} // '',
    'q2',
    "Mail8 - Ticket1 is in q2.",
);

# ticket 2 should be in q1
$Self->Is(
    $Test8Tickets[1]{Queue} // '',
    'q1',
    "Mail8 - Ticket2 is in q1.",
);

# Test: Dispatching via Queue second round
( $Return, @TicketIDs ) = ReadEmail( $Email, $QueueIDs{q3} );

# one additional ticket should be created...
$Self->Is(
    scalar @TicketIDs,
    1,
    "Mail8.5 - one ticket.",
);

my ( $LinkedTicketNumber, $LinkedTicketID ) = $Kernel::OM->Get('Kernel::System::PostMaster::AddressPool')->FindLinkedTicket(
    TicketID    => $TicketIDs[0],
    AddressPool => 'Pool02',
    UserID      => 1,
);

# ...and be linked to the former ones
$Self->Is(
    $LinkedTicketID,
    $Test8Tickets[0]{TicketID},
    "Mail8.5 - correctly linked.",
);

# Test: Dispatching via Queue third time unlucky
( $Return, @TicketIDs ) = ReadEmail( $Email, $QueueIDs{q3} );

# on third round though, mail should be ignored, as all pools are full
$Self->Is(
    $Return,
    5,
    "Mail8.6 - mail ignored.",
);

# no ticket should be created
$Self->Is(
    scalar @TicketIDs,
    0,
    "Mail8.6 - no article.",
);

# cleanup is done by RestoreDatabase.
$Self->DoneTesting();

sub GenerateEmail {
    my %Param = @_;

    $Param{XHeader} //= '';

    return <<"END";
From skywalker\@otobo.org Fri Dec 21 23:59:24 2001
Return-Path: <skywalker\@otobo.org>
Received: (from skywalker\@localhost)
    by avro.de (8.11.3/8.11.3/SuSE Linux 8.11.1-0.5) id f3MMSE303694
    for martin\@localhost; Fri, 21 Dec 2001 23:59:24 +0200
Date: Fri, 21 Dec 2001 23:59:24 +0200
From: Skywalker Attachment <skywalker\@otobo.org>
To: $Param{To}
Subject: $Param{Subject}
Message-ID: $Param{MessageID}
Mime-Version: 1.0
Content-Type: text/plain; charset=us-ascii
Content-Disposition: inline$Param{XHeader}
X-Operating-System: Linux 2.4.10-4GB i686
X-Uptime: 12:23am  up  5:19,  6 users,  load average: 0.11, 0.13, 0.18
Content-Length: 139
Lines: 11

This is the first test.

Adios ...

  Little "Skywalker"

--
System Tester I - <skywalker\@otobo.org>
--
Old programmers never die. They just branch to a new address.
END
}

## no critic (Perl::Critic::Policy::Subroutines::RequireArgUnpacking)

sub ReadEmail {

    # start a new incoming communication
    my $CommunicationLogObject = $Kernel::OM->Create(
        'Kernel::System::CommunicationLog',
        ObjectParams => {
            Transport   => 'Email',
            Direction   => 'Incoming',
            AccountType => 'STDIN',
        },
    );

    # start object log for the incoming connection
    $CommunicationLogObject->ObjectLogStart( ObjectLogType => 'Connection' );

    $CommunicationLogObject->ObjectLog(
        ObjectLogType => 'Connection',
        Priority      => 'Debug',
        Key           => 'Kernel::System::Console::Command::Maint::PostMaster::Read',
        Value         => 'Read Email from AddressPoolTest.',
    );

    # start object log for the email processing
    $CommunicationLogObject->ObjectLogStart( ObjectLogType => 'Message' );

    # remember the return code to stop the communictaion later with a proper status
    my $PostMasterReturnCode = 0;
    my @Return;

    # Wrap the main part of the script in an "eval" block so that any
    # unexpected (but probably transient) fatal errors (such as the
    # database being unavailable) can be trapped without causing a
    # bounce
    eval {

        $CommunicationLogObject->ObjectLog(
            ObjectLogType => 'Message',
            Priority      => 'Debug',
            Key           => 'Kernel::System::Console::Command::Maint::PostMaster::Read',
            Value         => 'Processing email with PostMaster module.',
        );

        my $PostMasterObject = $Kernel::OM->Create(
            'Kernel::System::PostMaster',
            ObjectParams => {
                CommunicationLogObject => $CommunicationLogObject,
                Email                  => [ split( /\n/, $_[0] ) ],
                Trusted                => 1,
            },
        );

        @Return = $PostMasterObject->Run( QueueID => $_[1] );

        if ( !$Return[0] ) {

            $CommunicationLogObject->ObjectLog(
                ObjectLogType => 'Message',
                Priority      => 'Error',
                Key           => 'Kernel::System::Console::Command::Maint::PostMaster::Read',
                Value         => 'PostMaster module exited with errors, could not process email. Please refer to the log!',
            );
            $CommunicationLogObject->CommunicationStop( Status => 'Failed' );

            die "Could not process email. Please refer to the log!\n";
        }

        my $Dump = $Kernel::OM->Get('Kernel::System::Main')->Dump( \@Return );
        $CommunicationLogObject->ObjectLog(
            ObjectLogType => 'Message',
            Priority      => 'Debug',
            Key           => 'Kernel::System::Console::Command::Maint::PostMaster::Read',
            Value         => "Email processing with PostMaster module completed, return data: $Dump",
        );

        $PostMasterReturnCode = $Return[0];
    };

    if ($@) {

        # An unexpected problem occurred (for example, the database was
        # unavailable). Return an EX_TEMPFAIL error to cause the mail
        # program to requeue the message instead of immediately bouncing
        # it; see sysexits.h. Most mail programs will retry an
        # EX_TEMPFAIL delivery for about four days, then bounce the
        # message.)
        my $Message = $@;

        $CommunicationLogObject->ObjectLog(
            ObjectLogType => 'Message',
            Priority      => 'Error',
            Key           => 'Kernel::System::Console::Command::Maint::PostMaster::Read',
            Value         => "An unexpected error occurred, message: $Message",
        );

        $CommunicationLogObject->ObjectLogStop(
            ObjectLogType => 'Message',
            Status        => 'Failed',
        );
        $CommunicationLogObject->ObjectLogStop(
            ObjectLogType => 'Connection',
            Status        => 'Failed',
        );
        $CommunicationLogObject->CommunicationStop( Status => 'Failed' );

        return;
    }

    $CommunicationLogObject->ObjectLog(
        ObjectLogType => 'Connection',
        Priority      => 'Debug',
        Key           => 'Kernel::System::Console::Command::Maint::PostMaster::Read',
        Value         => 'Closing connection from STDIN.',
    );

    $CommunicationLogObject->ObjectLogStop(
        ObjectLogType => 'Message',
        Status        => 'Successful',
    );
    $CommunicationLogObject->ObjectLogStop(
        ObjectLogType => 'Connection',
        Status        => 'Successful',
    );

    my %ReturnCodeMap = (
        0 => 'Failed',        # error (also false)
        1 => 'Successful',    # new ticket created
        2 => 'Successful',    # follow up / open/reopen
        3 => 'Successful',    # follow up / close -> new ticket
        4 => 'Failed',        # follow up / close -> reject
        5 => 'Successful',    # ignored (because of X-OTOBO-Ignore header)
    );

    $CommunicationLogObject->CommunicationStop(
        Status => $ReturnCodeMap{$PostMasterReturnCode} // 'Failed',
    );

    return @Return;
}
</File>
        <File Location="doc/en/DiscreteSystemAddresses.pdf" Permission="644" Encode="Base64">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</File>
    </Filelist>
</otrs_package>