/
Wuerth Phoenix DynamicField From DB-1.2.0.opm
31 lines (31 loc) · 283 KB
/
Wuerth Phoenix DynamicField From DB-1.2.0.opm
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
<?xml version="1.0" encoding="utf-8" ?>
<otrs_package version="1.1">
<Name>Wuerth Phoenix DynamicField From DB</Name>
<Version>1.2.0</Version>
<Vendor>Wuerth Phoenix S.R.L.</Vendor>
<URL>http://www.wuerth-phoenix.com/</URL>
<License>GNU GENERAL PUBLIC LICENSE Version 2, June 1991</License>
<ChangeLog Date="2012-04-26 18:45:21" Version="1.0.0">First Release.</ChangeLog>
<ChangeLog Date="2013-01-07 18:45:21" Version="1.1.0">New DynamicField Backend: MultiselectFromDB.</ChangeLog>
<ChangeLog Date="2013-01-08 18:45:21" Version="1.1.1">Better error handling on configuration. Local DBObject of otrs now supported.</ChangeLog>
<ChangeLog Date="2013-01-08 18:45:21" Version="1.1.2">- fixed bug in DisplayValueRender that tried to use missing DBIString when "Store Key and value" was set to "yes" - fixed bug that caused the creation/edit of a ticket to fail when an error message in the DynamicField was displayed - added a 'FromDB' class to identify DynamicFieldsFromDB in dom/css</ChangeLog>
<ChangeLog Date="2013-02-26 18:45:21" Version="1.1.3">Added Debug mode, added textarea in query field in TextAreaFromDB</ChangeLog>
<ChangeLog Date="2013-09-17 10:40:21" Version="1.2.0">Merged various fixes from projects. 1.2.0 Tag. </ChangeLog>
<Description Lang="en">Adds a new DynamicField Type that permits to load the possible values at runtime from an external DB.</Description>
<Framework>3.2.x</Framework>
<IntroInstall Lang="en" Title="Thank you!" Type="post">Thank you for choosing the WP DynamicField From DB.</IntroInstall>
<BuildDate>2013-09-17 15:02:43</BuildDate>
<BuildHost>yourhost.example.com</BuildHost>
<Filelist>
<File Location="Kernel/Config/Files/DynamicFieldDropdownFromDB.xml" Permission="644" Encode="Base64">PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiID8+CjxvdHJzX2NvbmZpZyB2ZXJzaW9uPSIxLjAiIGluaXQ9IkNoYW5nZXMiPgoKPENvbmZpZ0l0ZW0gTmFtZT0iRnJvbnRlbmQ6Ok1vZHVsZSMjI0FkbWluRHluYW1pY0ZpZWxkRHJvcGRvd25Gcm9tREIiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+IAo8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+IAo8R3JvdXA+RHluYW1pY0ZpZWxkczwvR3JvdXA+CjxTdWJHcm91cD5Gcm9udGVuZDo6QWRtaW46Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CjxTZXR0aW5nPiAKPEZyb250ZW5kTW9kdWxlUmVnPiAKPEdyb3VwPmFkbWluPC9Hcm91cD4KPERlc2NyaXB0aW9uPkFkbWluPC9EZXNjcmlwdGlvbj4KPFRpdGxlIFRyYW5zbGF0YWJsZT0iMSI+RHluYW1pYyBGaWVsZHMgRHJvcC1kb3duIEZyb20gREIgQmFja2VuZCBHVUk8L1RpdGxlPgo8TG9hZGVyPgo8Q1NTPkNvcmUuQWdlbnQuQWRtaW4uRHluYW1pY0ZpZWxkLmNzczwvQ1NTPgo8SmF2YVNjcmlwdD5Db3JlLkFnZW50LkFkbWluLkR5bmFtaWNGaWVsZC5qczwvSmF2YVNjcmlwdD4gCjxKYXZhU2NyaXB0PkNvcmUuQWdlbnQuQWRtaW4uRHluYW1pY0ZpZWxkRHJvcGRvd24uanM8L0phdmFTY3JpcHQ+IAo8L0xvYWRlcj4gCjwvRnJvbnRlbmRNb2R1bGVSZWc+CjwvU2V0dGluZz4KPC9Db25maWdJdGVtPgoKPENvbmZpZ0l0ZW0gTmFtZT0iRnJvbnRlbmQ6Ok1vZHVsZSMjI0FkbWluRHluYW1pY0ZpZWxkVGV4dEFyZWFGcm9tREIiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+IAo8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+IAo8R3JvdXA+RHluYW1pY0ZpZWxkczwvR3JvdXA+CjxTdWJHcm91cD5Gcm9udGVuZDo6QWRtaW46Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CjxTZXR0aW5nPiAKPEZyb250ZW5kTW9kdWxlUmVnPiAKPEdyb3VwPmFkbWluPC9Hcm91cD4KPERlc2NyaXB0aW9uPkFkbWluPC9EZXNjcmlwdGlvbj4KPFRpdGxlIFRyYW5zbGF0YWJsZT0iMSI+RHluYW1pYyBGaWVsZHMgRHJvcC1kb3duIEZyb20gREIgQmFja2VuZCBHVUk8L1RpdGxlPgo8TG9hZGVyPgo8Q1NTPkNvcmUuQWdlbnQuQWRtaW4uRHluYW1pY0ZpZWxkLmNzczwvQ1NTPgo8SmF2YVNjcmlwdD5Db3JlLkFnZW50LkFkbWluLkR5bmFtaWNGaWVsZC5qczwvSmF2YVNjcmlwdD4gCjxKYXZhU2NyaXB0PkNvcmUuQWdlbnQuQWRtaW4uRHluYW1pY0ZpZWxkVGV4dGFyZWEuanM8L0phdmFTY3JpcHQ+IAo8L0xvYWRlcj4gCjwvRnJvbnRlbmRNb2R1bGVSZWc+CjwvU2V0dGluZz4KPC9Db25maWdJdGVtPgoKPENvbmZpZ0l0ZW0gTmFtZT0iRnJvbnRlbmQ6Ok1vZHVsZSMjI0FkbWluRHluYW1pY0ZpZWxkTXVsdGlzZWxlY3RGcm9tREIiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+IAo8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+IAo8R3JvdXA+RHluYW1pY0ZpZWxkczwvR3JvdXA+CjxTdWJHcm91cD5Gcm9udGVuZDo6QWRtaW46Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CjxTZXR0aW5nPiAKPEZyb250ZW5kTW9kdWxlUmVnPiAKPEdyb3VwPmFkbWluPC9Hcm91cD4KPERlc2NyaXB0aW9uPkFkbWluPC9EZXNjcmlwdGlvbj4KPFRpdGxlIFRyYW5zbGF0YWJsZT0iMSI+RHluYW1pYyBGaWVsZHMgTXVsdGlzZWxlY3QgRnJvbSBEQiBCYWNrZW5kIEdVSTwvVGl0bGU+CjxMb2FkZXI+CjxDU1M+Q29yZS5BZ2VudC5BZG1pbi5EeW5hbWljRmllbGQuY3NzPC9DU1M+CjxKYXZhU2NyaXB0PkNvcmUuQWdlbnQuQWRtaW4uRHluYW1pY0ZpZWxkLmpzPC9KYXZhU2NyaXB0PiAKPEphdmFTY3JpcHQ+Q29yZS5BZ2VudC5BZG1pbi5EeW5hbWljRmllbGREcm9wZG93bi5qczwvSmF2YVNjcmlwdD4gCjwvTG9hZGVyPiAKPC9Gcm9udGVuZE1vZHVsZVJlZz4KPC9TZXR0aW5nPgo8L0NvbmZpZ0l0ZW0+Cgo8Q29uZmlnSXRlbSBOYW1lPSJEeW5hbWljRmllbGRzOjpCYWNrZW5kIyMjRHJvcGRvd25Gcm9tREIiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CjxEZXNjcmlwdGlvbiBUcmFuc2xhdGFibGU9IjEiPkR5bmFtaWNGaWVsZCBiYWNrZW5kIHJlZ2lzdHJhdGlvbi48L0Rlc2NyaXB0aW9uPgo8R3JvdXA+RHluYW1pY0ZpZWxkczwvR3JvdXA+CjxTdWJHcm91cD5EeW5hbWljRmllbGRzOjpCYWNrZW5kOjpSZWdpc3RyYXRpb248L1N1Ykdyb3VwPiAKPFNldHRpbmc+IAo8SGFzaD4KPEl0ZW0gS2V5PSJEaXNwbGF5TmFtZSIgVHJhbnNsYXRhYmxlPSIxIj5Ecm9wZG93bkZyb21EQjwvSXRlbT4KPEl0ZW0gS2V5PSJNb2R1bGUiPktlcm5lbDo6U3lzdGVtOjpEeW5hbWljRmllbGQ6OkJhY2tlbmQ6OkRyb3Bkb3duRnJvbURCPC9JdGVtPiAKPEl0ZW0gS2V5PSJDb25maWdEaWFsb2ciPkFkbWluRHluYW1pY0ZpZWxkRHJvcGRvd25Gcm9tREI8L0l0ZW0+IAo8L0hhc2g+IAo8L1NldHRpbmc+CjwvQ29uZmlnSXRlbT4KCjxDb25maWdJdGVtIE5hbWU9IkR5bmFtaWNGaWVsZHM6OkJhY2tlbmQjIyNUZXh0QXJlYUZyb21EQiIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KPERlc2NyaXB0aW9uIFRyYW5zbGF0YWJsZT0iMSI+RHluYW1pY0ZpZWxkIGJhY2tlbmQgcmVnaXN0cmF0aW9uLjwvRGVzY3JpcHRpb24+CjxHcm91cD5EeW5hbWljRmllbGRzPC9Hcm91cD4KPFN1Ykdyb3VwPkR5bmFtaWNGaWVsZHM6OkJhY2tlbmQ6OlJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+IAo8U2V0dGluZz4gCjxIYXNoPgo8SXRlbSBLZXk9IkRpc3BsYXlOYW1lIiBUcmFuc2xhdGFibGU9IjEiPlRleHRBcmVhRnJvbURCPC9JdGVtPgo8SXRlbSBLZXk9Ik1vZHVsZSI+S2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZDo6QmFja2VuZDo6VGV4dEFyZWFGcm9tREI8L0l0ZW0+IAo8SXRlbSBLZXk9IkNvbmZpZ0RpYWxvZyI+QWRtaW5EeW5hbWljRmllbGRUZXh0QXJlYUZyb21EQjwvSXRlbT4gCjwvSGFzaD4gCjwvU2V0dGluZz4KPC9Db25maWdJdGVtPgoKPENvbmZpZ0l0ZW0gTmFtZT0iRHluYW1pY0ZpZWxkczo6QmFja2VuZCMjI011bHRpc2VsZWN0RnJvbURCIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgo8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5EeW5hbWljRmllbGQgYmFja2VuZCByZWdpc3RyYXRpb24uPC9EZXNjcmlwdGlvbj4KPEdyb3VwPkR5bmFtaWNGaWVsZHM8L0dyb3VwPgo8U3ViR3JvdXA+RHluYW1pY0ZpZWxkczo6QmFja2VuZDo6UmVnaXN0cmF0aW9uPC9TdWJHcm91cD4gCjxTZXR0aW5nPiAKPEhhc2g+CjxJdGVtIEtleT0iRGlzcGxheU5hbWUiIFRyYW5zbGF0YWJsZT0iMSI+TXVsdGlzZWxlY3RGcm9tREI8L0l0ZW0+CjxJdGVtIEtleT0iTW9kdWxlIj5LZXJuZWw6OlN5c3RlbTo6RHluYW1pY0ZpZWxkOjpCYWNrZW5kOjpNdWx0aXNlbGVjdEZyb21EQjwvSXRlbT4gCjxJdGVtIEtleT0iSXRlbVNlcGFyYXRvciI+LCA8YnIvPjwvSXRlbT4gCjxJdGVtIEtleT0iQ29uZmlnRGlhbG9nIj5BZG1pbkR5bmFtaWNGaWVsZE11bHRpc2VsZWN0RnJvbURCPC9JdGVtPiAKPC9IYXNoPiAKPC9TZXR0aW5nPgo8L0NvbmZpZ0l0ZW0+CiAKPC9vdHJzX2NvbmZpZz4K</File>
<File Location="Kernel/Output/HTML/Standard/AdminDynamicFieldDropdownFromDB.dtl" Permission="644" Encode="Base64"># --
# AdminDynamicFieldDropdown.dtl - provides HTML form for AdminDynamicFieldsDropdown
# Copyright (C) 2001-2011 OTRS AG, http://otrs.org/
# --
# $Id: AdminDynamicFieldDropdown.dtl,v 1.16 2011/12/05 08:48:41 mg Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1>$Text{"Dynamic Fields"} - $Text{"$Data{"ObjectTypeName"}"}: $Text{"$Data{"Mode"}"} $Text{"$Data{"FieldTypeName"}"} $Text{"Field"}</h1>

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

    <div class="SidebarColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Actions"}</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
                    <li>
                        <a href="$Env{"Baselink"}Action=AdminDynamicField" class="CallForAction"><span>$Text{"Go back to overview"}</span></a>
                    </li>
                </ul>
            </div>

        <p class="FieldExplanation" style="margin-top: 30px;">
            $Text{"This functionality is provided by"}
            <img src="/otrs-web/skins/Agent/default/img/SDEpoweredby.png">
        </p>

        </div>
    </div>
    <div class="ContentColumn">
        <form action="$Env{"CGIHandle"}" method="post" class="Validate PreventMultipleSubmits">
            <input type="hidden" name="Action" value="AdminDynamicFieldDropdownFromDB" />
            <input type="hidden" name="Subaction" value="$QData{"Mode"}Action" />
            <input type="hidden" name="ObjectType" value="$QData{"ObjectType"}" />
            <input type="hidden" name="FieldType" value="$QData{"FieldType"}" />
            <input type="hidden" name="ID" value="$QData{"ID"}" />

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

                                <label class="Mandatory" for="Label"><span class="Marker">*</span> $Text{"Label"}:</label>
                                <div class="Field">
                                    <input id="Label" class="W50pc $QData{"LabelServerError"} Validate_Required" type="text" maxlength="200" value="$QData{"Label"}" name="Label"/>
                                    <div id="LabelError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="LabelServerError" class="TooltipErrorMessage"><p>$Text{"$Data{"LabelServerErrorMessage"}"}</p></div>
                                    <p class="FieldExplanation">$Text{"This is the name to be shown on the screens where the field is active."}</p>
                                </div>
                                <div class="Clear"></div>

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

                                <div class="SpacingTop"></div>
                                <label for="FieldTypeName">$Text{"Field type"}:</label>
                                <div class="Field">
                                    <input id="FieldTypeName" readonly="readonly" class="W50pc" type="text" maxlength="200" value="$QData{"FieldTypeName"}" name="FieldTypeName"/>
                                    <div class="Clear"></div>
                                </div>

                                <div class="SpacingTop"></div>
                                <label for="ObjectTypeName">$Text{"Object type"}:</label>
                                <div class="Field">
                                    <input id="ObjectTypeName" readonly="readonly" class="W50pc" type="text" maxlength="200" value="$QData{"ObjectTypeName"}" name="ObjectTypeName"/>
                                    <div class="Clear"></div>
                                </div>
                            </fieldset>
                        </div>
                    </div>
                </div>
            </div>
            <div class="WidgetSimple">
                <div class="Header">
                    <h2>$Text{"$Data{"FieldTypeName"}"} $Text{"Field Settings"}</h2>
                </div>
                <div class="Content">
                    <fieldset class="TableLike">
                        <div class="SpacingTop"></div>
                        <label for="DefaultValue">$Text{"Default value"}:</label>
                        <div class="Field">
                            $Data{"DefaultValueStrg"}
                            <p class="FieldExplanation">$Text{"This is the default value for this field."}</p>
                        </div>
                        <div class="Clear"></div>

                        <div class="SpacingTop"></div>
                        <label for="DefaultValue">$Text{"Add empty value"}:</label>
                        <div class="Field">
                            $Data{"PossibleNoneStrg"}
                            <p class="FieldExplanation">$Text{"Activate this option to create an empty selectable value."}</p>
                        </div>
                        <div class="Clear"></div>

                        <label for="TranslatableValues">$Text{"Translatable values"}:</label>
                        <div class="Field">
                            $Data{"TranslatableValuesStrg"}
                            <p class="FieldExplanation">$Text{"If you activate this option the values will be translated to the user defined language."}</p>
                            <p class="FieldExplanation"><strong>$Text{"Note"}: </strong>$Text{"You need to add the translations manually into the language translation files."}</p>
                        </div>
                        <div class="Clear"></div>
                        <label for="Link">$Text{"Show link"}:</label>
                        <div class="Field">
                            <input id="Link" class="W50pc" type="text" maxlength="500" value="$QData{"Link"}" name="Link"/>
                            <p class="FieldExplanation">
                                $Text{"Here you can specify an optional HTTP link for the field value in Overviews and Zoom screens."}
                                <br/>
                                $Text{"You can specify the Key instead of the Value putting the '%KEY%' placeholder."}
                                <br/>
                                $Text{"Example"}: http://some.example.com/handle?query=<span>$</span>LQData{"Field1"}?keyid=%KEY%
                            </p>
                        </div>
                        <label for="DBIstring">$Text{"DBI connect string"}:</label>
                        <div class="Field">
                            <input id="DBIstring" class="W50pc" type="text" maxlength="500" value="$QData{"DBIstring"}" name="DBIstring"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the DBI connect string. If nothing is specified, the local DBObject of otrs is used."}
                                <br/>
                                $Text{"Example"}: DBI:mysql:databasename;host=db.example.com
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <label for="DBIuser">$Text{"DB username"}:</label>
                        <div class="Field">
                            <input id="DBIuser" class="W50pc" type="text" maxlength="500" value="$QData{"DBIuser"}" name="DBIuser"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the DB username"}
                            </p>
                        </div>
                        <label for="DBIpass">$Text{"DB password"}:</label>
                        <div class="Field">
                            <input id="DBIpass" class="W50pc" type="text" maxlength="500" value="$QData{"DBIpass"}" name="DBIpass"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the DB password"}
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <div class="Clear"></div>
			<label for="Query">$Text{"Query"}:</label>
                        <div class="Field">
                            <textarea id="Query" class="W50pc" type="textarea" rows="5" maxlength="1000" name="Query">$QData{"Query"}</textarea>
                            <p class="FieldExplanation">
                                $Text{"Here you can specify the query to execute for the items in the DynamicField. Put a ? Placeholder for each Parameter."}
				<br/>
				$Text{"Keep in mind that only the first column is stored (and not visualized in the output) as key, so specifying an unique identifier is mandatory."}
                                <br/>
                                $Text{"Example with parameters"}: SELECT id, name FROM table WHERE refid = ?
				<br/>
                                $Text{"Example without parameters"}: SELECT id, name FROM table
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <label for="StoreValue">$Text{"Store Key and Value in DB"}:</label>
                        <div class="Field">
                            $Data{"StoreValue"}
                            <p class="FieldExplanation">$Text{"Activate this option to store Key and Values in DB. If activated, no more Visualization Query is needed."}
                            <br/>
                            $Text{"The data is stored in the '<key>||<value>' format in the database."}
                            </p>
                        </div>
                        <div class="Clear"></div>


                        <div class="Clear"></div>
			<label for="VisualQuery">$Text{"Visualization Query"}:</label>
                        <div class="Field">
                            <textarea id="VisualQuery" class="W50pc" type="textarea" rows="3" maxlength="500" name="VisualQuery">$QData{"VisualQuery"}</textarea>
                            <p class="FieldExplanation">
                                $Text{"Here you must specify the query to execute for retrieving back the string visualized once stored the key. This query *must* have *only one* Parameter that is susbtituted with the stored key."}
                                <br/>
                                $Text{"Example with parameters"}: SELECT name FROM table WHERE id = ?
				<br/>
                                $Text{"NOTE: This feature is only available with the OTRSTicketMaskExtension add-on on which it depends on."}
                            </p>
                        </div>
                        <div class="Clear"></div>
			<label for="Parameters">$Text{"Parameters"}:</label>
                        <div class="Field">
                            <input id="Parameters" class="W50pc" type="text" maxlength="500" value="$QData{"Parameters"}" name="Parameters"/>
                            <p class="FieldExplanation">
                                $Text{"Here you must specify the Parameters. ORDER IS RELEVANT!"}
                                <br/>
 				$Text{"IMPORTANT: don't put any spaces before/after parameters!"}
				<br/>
                                $Text{"Example"}: Customer,From
                            </p>
                        </div>
                         <div class="Clear"></div>
			<label for="Separator">$Text{"Separator"}:</label>
                        <div class="Field">
                            <input id="Separator" class="W50pc" type="text" maxlength="500" value="$QData{"Separator"}" name="Separator"/>
                            <p class="FieldExplanation">
                                $Text{"When multiple columns are selected in the query, the are concateneted in a single row. Specify the separator here."}
                                <br/>
                               $Text{"Example"}: ,
                            </p>
                        </div>
<!-- dtl:block:TicketMaskExtensionDisabled -->
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
                       <script type="text/javascript">
			  $("#Separator").attr("disabled", "disabled");
			  $("#Separator").attr("value", "(not available)");
			  $("#Separator").css({color:'#999999'});
			  $("#Parameters").attr("disabled", "disabled");
			  $("#Parameters").attr("value", "(not available)");
			  $("#Parameters").css({color:'#999999'});
                       </script>
//]]></script>
<!--dtl:js_on_document_complete-->
<!-- dtl:block:TicketMaskExtensionDisabled -->

                        <div class="Clear"></div>
			<label for="CacheTTL">$Text{"Cache Time To Live"}:</label>
                        <div class="Field">
                            <input id="CacheTTL" class="W50pc" type="text" maxlength="500" value="$QData{"CacheTTL"}" name="CacheTTL"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the cache validity in seconds. Consider that only queries with identical parameters are cached."}
                                <br/>
                               $Text{"Example"}: 360
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <div class="Clear"></div>
                        <label for="DisplayErrors">$Text{"DEBUG: Display Errors in Form"}:</label>
                        <div class="Field">
                            $Data{"DisplayErrorsStrg"}
                            <p class="FieldExplanation">$Text{"Activate this option to enable live debug output in fields."}
                            </p>
                        </div>
                        <div class="Clear"></div>
 
                       <div class="Clear"></div>

                    </fieldset>
                </div>
            </div>
            <fieldset class="TableLike">
                <div class="Field SpacingTop">
                    <button type="submit" class="Primary" value="$Text{"Save"}">$Text{"Save"}</button>
                    $Text{"or"}
                    <a href="$Env{"Baselink"}Action=AdminDynamicField">$Text{"Cancel"}</a>
                </div>
                <div class="Clear"></div>
            </fieldset>
        </form>
    </div>
</div>
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
$('.ShowWarning').bind('change keyup', function (Event) {
    $('p.Warning').removeClass('Hidden');
});

//bind click function to add button
$('#AddValue').bind('click', function () {
    Core.Agent.Admin.DynamicFieldDropdown.AddValue(
        $(this).closest('fieldset').find('.ValueInsert')
    );
    return false;
});

//bind click function to remove button
$('.ValueRemove').bind('click', function () {
    Core.Agent.Admin.DynamicFieldDropdown.RemoveValue($(this).attr('id'));
    return false;
});

$('.DefaultValueKeyItem,.DefaultValueItem').bind('keyup', function () {
    Core.Agent.Admin.DynamicFieldDropdown.RecreateDefaultValueList();
});

function toggleVisualQuery() {
        if ($('#StoreValue').val() == "0") {                                                                                                                                                  
                $('#VisualQuery').attr('readonly',null);                                                                                                                                      
        } else {                                                                                                                                                                              
                $('#VisualQuery').attr('readonly','readonly');                                                                                                                                
        }
}

$('#StoreValue').bind('change', function () {
	toggleVisualQuery();
});

toggleVisualQuery();

Core.Agent.Admin.DynamicField.ValidationInit();
//]]></script>
<!--dtl:js_on_document_complete-->
</File>
<File Location="Kernel/Output/HTML/Standard/AdminDynamicFieldTextAreaFromDB.dtl" Permission="644" Encode="Base64"># --
# AdminDynamicFieldDropdown.dtl - provides HTML form for AdminDynamicFieldsDropdown
# Copyright (C) 2001-2011 OTRS AG, http://otrs.org/
# --
# $Id: AdminDynamicFieldDropdown.dtl,v 1.16 2011/12/05 08:48:41 mg Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1>$Text{"Dynamic Fields"} - $Text{"$Data{"ObjectTypeName"}"}: $Text{"$Data{"Mode"}"} $Text{"$Data{"FieldTypeName"}"} $Text{"Field"}</h1>

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

    <div class="SidebarColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Actions"}</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
                    <li>
                        <a href="$Env{"Baselink"}Action=AdminDynamicField" class="CallForAction"><span>$Text{"Go back to overview"}</span></a>
                    </li>
                </ul>
            </div>
	 <p class="FieldExplanation" style="margin-top: 30px;">
            $Text{"This functionality is provided by"}
            <img src="/otrs-web/skins/Agent/default/img/SDEpoweredby.png">
        </p>

        </div>
    </div>

    <div class="ContentColumn">
        <form action="$Env{"CGIHandle"}" method="post" class="Validate PreventMultipleSubmits">
            <input type="hidden" name="Action" value="AdminDynamicFieldTextAreaFromDB" />
            <input type="hidden" name="Subaction" value="$QData{"Mode"}Action" />
            <input type="hidden" name="ObjectType" value="$QData{"ObjectType"}" />
            <input type="hidden" name="FieldType" value="$QData{"FieldType"}" />
            <input type="hidden" name="ID" value="$QData{"ID"}" />

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

                                <label class="Mandatory" for="Label"><span class="Marker">*</span> $Text{"Label"}:</label>
                                <div class="Field">
                                    <input id="Label" class="W50pc $QData{"LabelServerError"} Validate_Required" type="text" maxlength="200" value="$QData{"Label"}" name="Label"/>
                                    <div id="LabelError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="LabelServerError" class="TooltipErrorMessage"><p>$Text{"$Data{"LabelServerErrorMessage"}"}</p></div>
                                    <p class="FieldExplanation">$Text{"This is the name to be shown on the screens where the field is active."}</p>
                                </div>
                                <div class="Clear"></div>

                                <label class="Mandatory" for="FieldOrder"><span class="Marker">*</span> $Text{"Field order"}:</label>
                                <div class="Field">
                                    $Data{"DynamicFieldOrderSrtg"}
                                    <div id="FieldOrderError" class="TooltipErrorMessage"><p>$Text{"This field is required and must be numeric."}</p></div>
                                    <div id="FieldOrderServerError" class="TooltipErrorMessage"><p>$Text{"$Data{"FieldOrderServerErrorMessage"}"}</p></div>
                                    <p class="FieldExplanation">$Text{"This is the order in which this field will be shown on the screens where is active."}</p>
                                </div>
                                <div class="Clear"></div>
                            </fieldset>
                        </div>
                        <div class="Size1of2">
                            <fieldset class="TableLike">
                                <label for="ValidID">$Text{"Validity"}:</label>
                                <div class="Field">
                                    $Data{"ValidityStrg"}
                                </div>
                                <div class="Clear"></div>

                                <div class="SpacingTop"></div>
                                <label for="FieldTypeName">$Text{"Field type"}:</label>
                                <div class="Field">
                                    <input id="FieldTypeName" readonly="readonly" class="W50pc" type="text" maxlength="200" value="$QData{"FieldTypeName"}" name="FieldTypeName"/>
                                    <div class="Clear"></div>
                                </div>

                                <div class="SpacingTop"></div>
                                <label for="ObjectTypeName">$Text{"Object type"}:</label>
                                <div class="Field">
                                    <input id="ObjectTypeName" readonly="readonly" class="W50pc" type="text" maxlength="200" value="$QData{"ObjectTypeName"}" name="ObjectTypeName"/>
                                    <div class="Clear"></div>
                                </div>
                            </fieldset>
                        </div>
                    </div>
                </div>
            </div>
            <div class="WidgetSimple">
                <div class="Header">
                    <h2>$Text{"$Data{"FieldTypeName"}"} $Text{"Field Settings"}</h2>
                </div>
                <div class="Content">
                    <fieldset class="TableLike">
                        <div class="SpacingTop"></div>
                        <label for="DefaultValue">$Text{"Default value"}:</label>
                        <div class="Field">
                            $Data{"DefaultValueStrg"}
                            <p class="FieldExplanation">$Text{"This is the default value for this field."}</p>
                        </div>
                        <div class="Clear"></div>

                        <div class="SpacingTop"></div>
                        <label for="DefaultValue">$Text{"Add empty value"}:</label>
                        <div class="Field">
                            $Data{"PossibleNoneStrg"}
                            <p class="FieldExplanation">$Text{"Activate this option to create an empty selectable value."}</p>
                        </div>
                        <div class="Clear"></div>

                        <label for="TranslatableValues">$Text{"Translatable values"}:</label>
                        <div class="Field">
                            $Data{"TranslatableValuesStrg"}
                            <p class="FieldExplanation">$Text{"If you activate this option the values will be translated to the user defined language."}</p>
                            <p class="FieldExplanation"><strong>$Text{"Note"}: </strong>$Text{"You need to add the translations manually into the language translation files."}</p>
                        </div>
                        <div class="Clear"></div>
                       <label for="DBIstring">$Text{"DBI connect string"}:</label>
                        <div class="Field">
                            <input id="DBIstring" class="W50pc" type="text" maxlength="500" value="$QData{"DBIstring"}" name="DBIstring"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the DBI connect string"}
                                <br/>
                                $Text{"Example"}: DBI:mysql:databasename;host=db.example.com
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <label for="DBIuser">$Text{"DB username"}:</label>
                        <div class="Field">
                            <input id="DBIuser" class="W50pc" type="text" maxlength="500" value="$QData{"DBIuser"}" name="DBIuser"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the DB username"}
                            </p>
                        </div>
                        <label for="DBIpass">$Text{"DB password"}:</label>
                        <div class="Field">
                            <input id="DBIpass" class="W50pc" type="text" maxlength="500" value="$QData{"DBIpass"}" name="DBIpass"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the DB password"}
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <div class="Clear"></div>
			<label for="Query">$Text{"Query"}:</label>
                        <div class="Field">
                            <textarea id="Query" class="W50pc" type="textarea" rows="5" maxlength="1000" name="Query">$QData{"Query"}</textarea>
                            <p class="FieldExplanation">
                                $Text{"Here you can specify the query to execute for the items in the DynamicField. Put a ? Placeholder for each Parameter."}
				<br/>
				$Text{"Each column will then visualized like 'COLUMNNAME: VALUE'"}
                                <br/>
                                $Text{"Example with parameters"}: SELECT name FROM table WHERE id = ?
				<br/>
                                $Text{"Example without parameters"}: SELECT name FROM table
                            </p>
                        </div>
                        <div class="Clear"></div>
			<label for="Parameters">$Text{"Parameters"}:</label>
                        <div class="Field">
                            <input id="Query" class="W50pc" type="text" maxlength="500" value="$QData{"Parameters"}" name="Parameters"/>
                            <p class="FieldExplanation">
                                $Text{"Here you must specify the Parameters. ORDER IS RELEVANT!"}
                                <br/>
 				$Text{"IMPORTANT: don't put any spaces before/after parameters!"}
				<br/>
                                $Text{"Example"}: Customer,From
                            </p>
                        </div>
                        <div class="Clear"></div>
			<label for="CacheTTL">$Text{"Cache Time To Live"}:</label>
                        <div class="Field">
                            <input id="CacheTTL" class="W50pc" type="text" maxlength="500" value="$QData{"CacheTTL"}" name="CacheTTL"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the cache validity in seconds. Consider that only queries with identical parameters are cached."}
                                <br/>
                               $Text{"Example"}: 360
                            </p>
                        </div>
 
                      <div class="Clear"></div>
                      <div class="Clear"></div>
                        <label for="DisplayErrors">$Text{"DEBUG: Display Errors in Form"}:</label>
                        <div class="Field">
                            $Data{"DisplayErrors"}
                            <p class="FieldExplanation">$Text{"Activate this option to enable live debug output in fields."}
                            </p>
                        </div>
                        <div class="Clear"></div>

                    </fieldset>
                </div>
            </div>
            <fieldset class="TableLike">
                <div class="Field SpacingTop">
                    <button type="submit" class="Primary" value="$Text{"Save"}">$Text{"Save"}</button>
                    $Text{"or"}
                    <a href="$Env{"Baselink"}Action=AdminDynamicField">$Text{"Cancel"}</a>
                </div>
                <div class="Clear"></div>
            </fieldset>
        </form>
    </div>
</div>
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
$('.ShowWarning').bind('change keyup', function (Event) {
    $('p.Warning').removeClass('Hidden');
});

//bind click function to add button
$('#AddValue').bind('click', function () {
    Core.Agent.Admin.DynamicFieldDropdown.AddValue(
        $(this).closest('fieldset').find('.ValueInsert')
    );
    return false;
});

//bind click function to remove button
$('.ValueRemove').bind('click', function () {
    Core.Agent.Admin.DynamicFieldDropdown.RemoveValue($(this).attr('id'));
    return false;
});

$('.DefaultValueKeyItem,.DefaultValueItem').bind('keyup', function () {
    Core.Agent.Admin.DynamicFieldDropdown.RecreateDefaultValueList();
});

Core.Agent.Admin.DynamicField.ValidationInit();
//]]></script>
<!--dtl:js_on_document_complete-->
</File>
<File Location="Kernel/Output/HTML/Standard/AdminDynamicFieldMultiselectFromDB.dtl" Permission="644" Encode="Base64"># --
# AdminDynamicFieldDropdown.dtl - provides HTML form for AdminDynamicFieldsDropdown
# Copyright (C) 2001-2011 OTRS AG, http://otrs.org/
# --
# $Id: AdminDynamicFieldDropdown.dtl,v 1.16 2011/12/05 08:48:41 mg Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1>$Text{"Dynamic Fields"} - $Text{"$Data{"ObjectTypeName"}"}: $Text{"$Data{"Mode"}"} $Text{"$Data{"FieldTypeName"}"} $Text{"Field"}</h1>

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

    <div class="SidebarColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Actions"}</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
                    <li>
                        <a href="$Env{"Baselink"}Action=AdminDynamicField" class="CallForAction"><span>$Text{"Go back to overview"}</span></a>
                    </li>
                </ul>
            </div>
        </div>
    </div>

    <div class="ContentColumn">
        <form action="$Env{"CGIHandle"}" method="post" class="Validate PreventMultipleSubmits">
            <input type="hidden" name="Action" value="AdminDynamicFieldDropdownFromDB" />
            <input type="hidden" name="Subaction" value="$QData{"Mode"}Action" />
            <input type="hidden" name="ObjectType" value="$QData{"ObjectType"}" />
            <input type="hidden" name="FieldType" value="$QData{"FieldType"}" />
            <input type="hidden" name="ID" value="$QData{"ID"}" />

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

                                <label class="Mandatory" for="Label"><span class="Marker">*</span> $Text{"Label"}:</label>
                                <div class="Field">
                                    <input id="Label" class="W50pc $QData{"LabelServerError"} Validate_Required" type="text" maxlength="200" value="$QData{"Label"}" name="Label"/>
                                    <div id="LabelError" class="TooltipErrorMessage"><p>$Text{"This field is required."}</p></div>
                                    <div id="LabelServerError" class="TooltipErrorMessage"><p>$Text{"$Data{"LabelServerErrorMessage"}"}</p></div>
                                    <p class="FieldExplanation">$Text{"This is the name to be shown on the screens where the field is active."}</p>
                                </div>
                                <div class="Clear"></div>

                                <label class="Mandatory" for="FieldOrder"><span class="Marker">*</span> $Text{"Field order"}:</label>
                                <div class="Field">
                                    $Data{"DynamicFieldOrderSrtg"}
                                    <div id="FieldOrderError" class="TooltipErrorMessage"><p>$Text{"This field is required and must be numeric."}</p></div>
                                    <div id="FieldOrderServerError" class="TooltipErrorMessage"><p>$Text{"$Data{"FieldOrderServerErrorMessage"}"}</p></div>
                                    <p class="FieldExplanation">$Text{"This is the order in which this field will be shown on the screens where is active."}</p>
                                </div>
                                <div class="Clear"></div>
                            </fieldset>
                        </div>
                        <div class="Size1of2">
                            <fieldset class="TableLike">
                                <label for="ValidID">$Text{"Validity"}:</label>
                                <div class="Field">
                                    $Data{"ValidityStrg"}
                                </div>
                                <div class="Clear"></div>

                                <div class="SpacingTop"></div>
                                <label for="FieldTypeName">$Text{"Field type"}:</label>
                                <div class="Field">
                                    <input id="FieldTypeName" readonly="readonly" class="W50pc" type="text" maxlength="200" value="$QData{"FieldTypeName"}" name="FieldTypeName"/>
                                    <div class="Clear"></div>
                                </div>

                                <div class="SpacingTop"></div>
                                <label for="ObjectTypeName">$Text{"Object type"}:</label>
                                <div class="Field">
                                    <input id="ObjectTypeName" readonly="readonly" class="W50pc" type="text" maxlength="200" value="$QData{"ObjectTypeName"}" name="ObjectTypeName"/>
                                    <div class="Clear"></div>
                                </div>
                            </fieldset>
                        </div>
                    </div>
                </div>
            </div>
            <div class="WidgetSimple">
                <div class="Header">
                    <h2>$Text{"$Data{"FieldTypeName"}"} $Text{"Field Settings"}</h2>
                </div>
                <div class="Content">
                    <fieldset class="TableLike">
                        <div class="SpacingTop"></div>
                        <label for="DefaultValue">$Text{"Default value"}:</label>
                        <div class="Field">
                            $Data{"DefaultValueStrg"}
                            <p class="FieldExplanation">$Text{"This is the default value for this field."}</p>
                        </div>
                        <div class="Clear"></div>

                        <div class="SpacingTop"></div>
                        <label for="DefaultValue">$Text{"Add empty value"}:</label>
                        <div class="Field">
                            $Data{"PossibleNoneStrg"}
                            <p class="FieldExplanation">$Text{"Activate this option to create an empty selectable value."}</p>
                        </div>
                        <div class="Clear"></div>

                        <label for="TranslatableValues">$Text{"Translatable values"}:</label>
                        <div class="Field">
                            $Data{"TranslatableValuesStrg"}
                            <p class="FieldExplanation">$Text{"If you activate this option the values will be translated to the user defined language."}</p>
                            <p class="FieldExplanation"><strong>$Text{"Note"}: </strong>$Text{"You need to add the translations manually into the language translation files."}</p>
                        </div>
                        <div class="Clear"></div>
                        <label for="Link">$Text{"Show link"}:</label>
                        <div class="Field">
                            <input id="Link" class="W50pc" type="text" maxlength="500" value="$QData{"Link"}" name="Link"/>
                            <p class="FieldExplanation">
                                $Text{"Here you can specify an optional HTTP link for the field value in Overviews and Zoom screens."}
                                <br/>
                                $Text{"You can specify the Key instead of the Value putting the '%KEY%' placeholder."}
                                <br/>
                                $Text{"Example"}: http://some.example.com/handle?query=<span>$</span>LQData{"Field1"}?keyid=%KEY%
                            </p>
                        </div>
                        <label for="DBIstring">$Text{"DBI connect string"}:</label>
                        <div class="Field">
                            <input id="DBIstring" class="W50pc" type="text" maxlength="500" value="$QData{"DBIstring"}" name="DBIstring"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the DBI connect string. If nothing is specified, the local DBObject of otrs is used."}
                                <br/>
                                $Text{"Example"}: DBI:mysql:databasename;host=db.example.com
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <label for="DBIuser">$Text{"DB username"}:</label>
                        <div class="Field">
                            <input id="DBIuser" class="W50pc" type="text" maxlength="500" value="$QData{"DBIuser"}" name="DBIuser"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the DB username"}
                            </p>
                        </div>
                        <label for="DBIpass">$Text{"DB password"}:</label>
                        <div class="Field">
                            <input id="DBIpass" class="W50pc" type="text" maxlength="500" value="$QData{"DBIpass"}" name="DBIpass"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the DB password"}
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <div class="Clear"></div>
			<label for="Query">$Text{"Query"}:</label>
                        <div class="Field">
                            <textarea id="Query" class="W50pc" type="textarea" rows="5" maxlength="500" name="Query">$QData{"Query"}</textarea>
                            <p class="FieldExplanation">
                                $Text{"Here you can specify the query to execute for the items in the DynamicField. Put a ? Placeholder for each Parameter."}
				<br/>
				$Text{"Keep in mind that only the first column is stored (and not visualized in the output) as key, so specifying an unique identifier is mandatory."}
                                <br/>
                                $Text{"Example with parameters"}: SELECT id, name FROM table WHERE refid = ?
				<br/>
                                $Text{"Example without parameters"}: SELECT id, name FROM table
                            </p>
                        </div>

                        <div class="Clear"></div>
			<label for="Parameters">$Text{"Parameters"}:</label>
                        <div class="Field">
                            <input id="Parameters" class="W50pc" type="text" maxlength="500" value="$QData{"Parameters"}" name="Parameters"/>
                            <p class="FieldExplanation">
                                $Text{"Here you must specify the Parameters. ORDER IS RELEVANT!"}
                                <br/>
 				$Text{"IMPORTANT: don't put any spaces before/after parameters!"}
				<br/>
                                $Text{"Example"}: Customer,From
                            </p>
                        </div>
                         <div class="Clear"></div>
			<label for="Separator">$Text{"Separator"}:</label>
                        <div class="Field">
                            <input id="Separator" class="W50pc" type="text" maxlength="500" value="$QData{"Separator"}" name="Separator"/>
                            <p class="FieldExplanation">
                                $Text{"When multiple columns are selected in the query, the are concateneted in a single row. Specify the separator here."}
                                <br/>
                               $Text{"Example"}: ,
                            </p>
                        </div>
<!-- dtl:block:TicketMaskExtensionDisabled -->
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
                       <script type="text/javascript">
			  $("#Separator").attr("disabled", "disabled");
			  $("#Separator").attr("value", "(not available)");
			  $("#Separator").css({color:'#999999'});
			  $("#Parameters").attr("disabled", "disabled");
			  $("#Parameters").attr("value", "(not available)");
			  $("#Parameters").css({color:'#999999'});
                       </script>
//]]></script>
<!--dtl:js_on_document_complete-->
<!-- dtl:block:TicketMaskExtensionDisabled -->

                        <div class="Clear"></div>
			<label for="CacheTTL">$Text{"Cache Time To Live"}:</label>
                        <div class="Field">
                            <input id="CacheTTL" class="W50pc" type="text" maxlength="500" value="$QData{"CacheTTL"}" name="CacheTTL"/>
                            <p class="FieldExplanation">
                                $Text{"Specify the cache validity in seconds. Consider that only queries with identical parameters are cached."}
                                <br/>
                               $Text{"Example"}: 360
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <div class="Clear"></div>
                        <label for="DisplayErrors">$Text{"DEBUG: Display Errors in Form"}:</label>
                        <div class="Field">
                            $Data{"DisplayErrors"}
                            <p class="FieldExplanation">$Text{"Activate this option to enable live debug output in fields."}
                            </p>
                        </div>
                        <div class="Clear"></div>
                        <div class="Clear"></div>

                    </fieldset>
                </div>
            </div>
            <fieldset class="TableLike">
                <div class="Field SpacingTop">
                    <button type="submit" class="Primary" value="$Text{"Save"}">$Text{"Save"}</button>
                    $Text{"or"}
                    <a href="$Env{"Baselink"}Action=AdminDynamicField">$Text{"Cancel"}</a>
                </div>
                <div class="Clear"></div>
            </fieldset>
        </form>
    </div>
</div>
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
$('.ShowWarning').bind('change keyup', function (Event) {
    $('p.Warning').removeClass('Hidden');
});

//bind click function to add button
$('#AddValue').bind('click', function () {
    Core.Agent.Admin.DynamicFieldDropdown.AddValue(
        $(this).closest('fieldset').find('.ValueInsert')
    );
    return false;
});

//bind click function to remove button
$('.ValueRemove').bind('click', function () {
    Core.Agent.Admin.DynamicFieldDropdown.RemoveValue($(this).attr('id'));
    return false;
});

$('.DefaultValueKeyItem,.DefaultValueItem').bind('keyup', function () {
    Core.Agent.Admin.DynamicFieldDropdown.RecreateDefaultValueList();
});

Core.Agent.Admin.DynamicField.ValidationInit();
//]]></script>
<!--dtl:js_on_document_complete-->
</File>
<File Location="Custom/Kernel/Modules/AdminDynamicFieldDropdownFromDB.pm" Permission="644" Encode="Base64"># --
# Kernel/Modules/AdminDynamicFieldDropdown.pm - provides a dynamic fields text config view for admins
# Copyright (C) 2001-2012 WuerthPhoenix SRL, http://www.wuerthphoenix.it/
# --
# $Id: AdminDynamicFieldDropdownFromDB.pm,v 1.00 2012/04/18 19:38:01 cr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Modules::AdminDynamicFieldDropdownFromDB;

use strict;
use warnings;

use Kernel::System::VariableCheck qw(:all);
use Kernel::System::Valid;
use Kernel::System::CheckItem;
use Kernel::System::DynamicField;

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

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

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

    for (qw(ParamObject LayoutObject LogObject ConfigObject)) {
        if ( !$Self->{$_} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $_!" );
        }
    }

    # create additional objects
    $Self->{ValidObject} = Kernel::System::Valid->new( %{$Self} );

    $Self->{DynamicFieldObject} = Kernel::System::DynamicField->new( %{$Self} );

    # get configured object types
    $Self->{ObjectTypeConfig} = $Self->{ConfigObject}->Get('DynamicFields::ObjectType');

    # get the fields config
    $Self->{FieldTypeConfig} = $Self->{ConfigObject}->Get('DynamicFields::Backend') || {};

    # set possible values handling strings
    $Self->{EmptyString}     = '_DynamicFields_EmptyString_Dont_Use_It_String_Please';
    $Self->{DuplicateString} = '_DynamicFields_DuplicatedString_Dont_Use_It_String_Please';
    $Self->{DeletedString}   = '_DynamicFields_DeletedString_Dont_Use_It_String_Please';

    return $Self;
}

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

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

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

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

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

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

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

    my %GetParam;
    for my $Needed (qw(ObjectType FieldType FieldOrder)) {
        $GetParam{$Needed} = $Self->{ParamObject}->GetParam( Param => $Needed );
        if ( !$Needed ) {
            return $Self->{LayoutObject}->ErrorScreen(
                Message => "Need $Needed",
            );
        }
    }

    # get the object type and field type display name
    my $ObjectTypeName = $Self->{ObjectTypeConfig}->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName  = $Self->{FieldTypeConfig}->{ $GetParam{FieldType} }->{DisplayName}   || '';

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        Mode           => 'Add',
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my %Errors;
    my %GetParam;

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

    if ( $GetParam{Name} ) {

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

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

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

        %DynamicFieldsList = reverse %DynamicFieldsList;

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

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

    if ( $GetParam{FieldOrder} ) {

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

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

    for my $ConfigParam (
        qw(
        ObjectType ObjectTypeName FieldType FieldTypeName DefaultValue PossibleNone
        TranslatableValues ValidID Link Query VisualQuery Parameters DBIstring DBIuser DBIpass Separator CacheTTL DisplayErrors
        )
        )
    {
        $GetParam{$ConfigParam} = $Self->{ParamObject}->GetParam( Param => $ConfigParam );
    }

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

    my $PossibleValues = $Self->_GetPossibleValues();

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

    # set specific config
    my $FieldConfig = {
#        PossibleValues     => $PossibleValues,
        DefaultValue       => $GetParam{DefaultValue},
        PossibleNone       => $GetParam{PossibleNone},
        TranslatableValues => $GetParam{TranslatableValues},
        Link               => $GetParam{Link},
        Query		   => $GetParam{Query},
        StoreValue	   => $GetParam{StoreValue},
        VisualQuery	   => $GetParam{VisualQuery},
        Parameters	   => $GetParam{Parameters},
        DBIstring	   => $GetParam{DBIstring},
        DBIuser		   => $GetParam{DBIuser},
        DBIpass		   => $GetParam{DBIpass},
        Separator	   => $GetParam{Separator},
        CacheTTL	   => $GetParam{CacheTTL},
        DisplayErrors  => $GetParam{DisplayErrors},
    };

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

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

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

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

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

    # get the object type and field type display name
    my $ObjectTypeName = $Self->{ObjectTypeConfig}->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName  = $Self->{FieldTypeConfig}->{ $GetParam{FieldType} }->{DisplayName}   || '';

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

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

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

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

    my %Config = ();

    # extract configuration
    if ( IsHashRefWithData( $DynamicFieldData->{Config} ) ) {

#        # set PossibleValues
#        $Config{PossibleValues} = {};
#        if ( IsHashRefWithData( $DynamicFieldData->{Config}->{PossibleValues} ) ) {
#            $Config{PossibleValues} = $DynamicFieldData->{Config}->{PossibleValues};
#        }

        # set DefaultValue
        $Config{DefaultValue} = $DynamicFieldData->{Config}->{DefaultValue};

        # set PossibleNone
        $Config{PossibleNone} = $DynamicFieldData->{Config}->{PossibleNone};

        # set TranslatalbeValues
        $Config{TranslatableValues} = $DynamicFieldData->{Config}->{TranslatableValues};

        # set Link
        $Config{Link} = $DynamicFieldData->{Config}->{Link};
        $Config{Query} = $DynamicFieldData->{Config}->{Query};
        $Config{StoreValue} = $DynamicFieldData->{Config}->{StoreValue};
        $Config{VisualQuery} = $DynamicFieldData->{Config}->{VisualQuery};
        $Config{Parameters} = $DynamicFieldData->{Config}->{Parameters};
        $Config{DBIstring} = $DynamicFieldData->{Config}->{DBIstring};
        $Config{DBIuser} = $DynamicFieldData->{Config}->{DBIuser};
        $Config{DBIpass} = $DynamicFieldData->{Config}->{DBIpass};
        $Config{Separator} = $DynamicFieldData->{Config}->{Separator};
        $Config{CacheTTL} = $DynamicFieldData->{Config}->{CacheTTL};
        $Config{DisplayErrors} = $DynamicFieldData->{Config}->{DisplayErrors};
    }

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        %${DynamicFieldData},
        %Config,
        ID             => $FieldID,
        Mode           => 'Change',
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my %Errors;
    my %GetParam;

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

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

    if ( $GetParam{Name} ) {

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

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

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

        %DynamicFieldsList = reverse %DynamicFieldsList;

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

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

    if ( $GetParam{FieldOrder} ) {

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

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

    for my $ConfigParam (
        qw(
        ObjectType ObjectTypeName FieldType FieldTypeName DefaultValue PossibleNone
        TranslatableValues ValidID Link Query StoreValue VisualQuery Parameters DBIstring DBIuser DBIpass Separator CacheTTL DisplayErrors
        )
        )
    {
        $GetParam{$ConfigParam} = $Self->{ParamObject}->GetParam( Param => $ConfigParam );
    }

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

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

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

    my $PossibleValues = $Self->_GetPossibleValues();

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

    # set specific config
    my $FieldConfig = {
        PossibleValues     => $PossibleValues,
        DefaultValue       => $GetParam{DefaultValue},
        PossibleNone       => $GetParam{PossibleNone},
        TranslatableValues => $GetParam{TranslatableValues},
        Link               => $GetParam{Link},
        Query				=> $GetParam{Query},
        StoreValue	   		=> $GetParam{StoreValue},
        VisualQuery	   		=> $GetParam{VisualQuery},
        Parameters	   		=> $GetParam{Parameters},
        DBIstring			=> $GetParam{DBIstring},
        DBIuser				=> $GetParam{DBIuser},
        DBIpass				=> $GetParam{DBIpass},
        Separator			=> $GetParam{Separator},
        CacheTTL			=> $GetParam{CacheTTL},
        DisplayErrors		=> $GetParam{DisplayErrors},
    };

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

    if ( !$UpdateSuccess ) {
        return $Self->{LayoutObject}->ErrorScreen(
            Message => "Could not update the field $GetParam{Name}",
        );
    }

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

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

    $Param{DisplayFieldName} = 'New';

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

    $Param{DeletedString} = $Self->{DeletedString};

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

    # get all fields
    my $DynamicFieldList = $Self->{DynamicFieldObject}->DynamicFieldListGet(
        Valid => 0,
    );

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

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

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

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

    my $DynamicFieldOrderStrg = $Self->{LayoutObject}->BuildSelection(
        Data          => \@DynamicfieldOrderList,
        Name          => 'FieldOrder',
        SelectedValue => $Param{FieldOrder} || 1,
        PossibleNone  => 0,
        Class         => 'W50pc Validate_Number',
    );

    my %ValidList = $Self->{ValidObject}->ValidList();

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

    # define as 0 to get the real value in the HTML
    my $ValueCounter = 0;

    # set PossibleValues
    my %PossibleValues;
    if ( IsHashRefWithData( $Param{PossibleValues} ) ) {
        %PossibleValues = %{ $Param{PossibleValues} };
    }


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

   #Kernel/System/Ticket/OTRSTicketMaskExtensions.pm 
eval
{
  require Kernel::System::Ticket::OTRSTicketMaskExtensions;
};

unless(! $@)
{
    $Self->{LayoutObject}->Block(
        Name => 'TicketMaskExtensionDisabled',
        Data => {
            %Param,
        },
    );

}
 
#if (! eval "use Kernel::System::Ticket::OTRSTicketMaskExtensions") {
#use Kernel::System::Ticket::OTRSTicketMaskExtensions;
#}


    # check and build the Default Value list based on Possible Values
#    my %DefaultValuesList;
#    POSSIBLEVALUE:
#    for my $ValueItem ( keys %PossibleValues ) {
#        next POSSIBLEVALUE if !defined $ValueItem;
#        next POSSIBLEVALUE if !defined $PossibleValues{$ValueItem};
#        $DefaultValuesList{$ValueItem} = $PossibleValues{$ValueItem}
#    }
#
#    my $DefaultValue = ( defined $Param{DefaultValue} ? $Param{DefaultValue} : '' );
#
#    # create the default value select
#    my $DefaultValueStrg = $Self->{LayoutObject}->BuildSelection(
#        Data         => \%DefaultValuesList,
#        Name         => 'DefaultValue',
#        SelectedID   => $DefaultValue,
#        PossibleNone => 1,
#
#        # Don't make is translatable because this will confuse the user (also current JS
#        # is not prepared)
#        Translation => 0,
#
#        # Multiple selections are currently not supported
#        Multiple => 0,
#        Class    => 'W50pc',
#    );

    my $PossibleNone = $Param{PossibleNone} || '0';

    # create translatable values option list
    my $PossibleNoneStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'PossibleNone',
        SelectedID => $PossibleNone,
        Class      => 'W50pc',
    );

    my $TranslatableValues = $Param{TranslatableValues} || '0';

    # create translatable values option list
    my $TranslatableValuesStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'TranslatableValues',
        SelectedID => $TranslatableValues,
        Class      => 'W50pc',
    );

    my $StoreValue = $Param{StoreValue} || '0';

    # create translatable values option list
    my $StoreValueStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'StoreValue',
        SelectedID => $StoreValue,
        Class      => 'W50pc',
    );

    my $DisplayErrors = $Param{DisplayErrors} || '0';

    # create translatable values option list
    my $DisplayErrorsStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'DisplayErrors',
        SelectedID => $DisplayErrors,
        Class      => 'W50pc',
    );


    my $Link = $Param{Link} || '';
    my $Query = $Param{Query} || '';
    my $VisualQuery = $Param{VisualQuery} || '';
    my $Parameters = $Param{Parameters} || '';
    my $DBIstring = $Param{DBIstring} || '';
    my $DBIuser = $Param{DBIuser} || '';
    my $DBIpass = $Param{DBIpass} || '';
    my $Separator = $Param{Separator} || ',';
    my $CacheTTL = $Param{CacheTTL} || '360';


    # generate output
    $Output .= $Self->{LayoutObject}->Output(
        TemplateFile => 'AdminDynamicFieldDropdownFromDB',
        Data         => {
            %Param,
            ValidityStrg           => $ValidityStrg,
            DynamicFieldOrderStrg  => $DynamicFieldOrderStrg,
#            ValueCounter           => $ValueCounter,
#            DefaultValueStrg       => $DefaultValueStrg,
            PossibleNoneStrg       => $PossibleNoneStrg,
            TranslatableValuesStrg => $TranslatableValuesStrg,
            Link                   => $Link,
            Query		   => $Query,
            StoreValue		   => $StoreValueStrg,
            VisualQuery		   => $VisualQuery,
            Parameters		   => $Parameters,
            DBIstring		   => $DBIstring,
            DBIuser		   => $DBIuser,
            DBIpass		   => $DBIpass,
            Separator		   => $Separator,
            CacheTTL		   => $CacheTTL,
            DisplayErrorsStrg	=> $DisplayErrorsStrg,
        }
    );

    $Output .= $Self->{LayoutObject}->Footer();

    return $Output;
}

sub _GetPossibleValues {
    my ( $Self, %Param ) = @_;
#
    my $PossibleValueConfig;
#
#    # get parameters from web browser
#    # get ValueCounters
#    my $ValueCounter          = $Self->{ParamObject}->GetParam( Param => 'ValueCounter' ) || 0;
#    my $EmptyValueCounter     = 0;
#    my $DuplicateValueCounter = 0;
#
#    # get possible values
#    my $Values;
#    VALUEINDEX:
#    for my $ValueIndex ( 1 .. $ValueCounter ) {
#        my $Key = $Self->{ParamObject}->GetParam( Param => 'Key' . '_' . $ValueIndex );
#        $Key = ( defined $Key ? $Key : '' );
#
#        # check if key was deleted by the user and skip it
#        next VALUEINDEX if $Key eq $Self->{DeletedString};
#
#        # check if the original value is empty
#        if ( $Key eq '' ) {
#
#            # change the empty value to a predefined string
#            $Key = $Self->{EmptyString} . int $EmptyValueCounter;
#            $EmptyValueCounter++;
#        }
#
#        # otherwise check for duplicate
#        elsif ( exists $PossibleValueConfig->{$Key} ) {
#
#            # append a predefined unique string to make this value unique
#            $Key .= '-' . $Self->{DuplicateString} . $DuplicateValueCounter;
#            $DuplicateValueCounter++;
#        }
#
#        my $Value = $Self->{ParamObject}->GetParam( Param => 'Value' . '_' . $ValueIndex );
#        $Value = ( defined $Value ? $Value : '' );
#        $PossibleValueConfig->{$Key} = $Value;
#    }

#    $PossibleValueConfig->{'dummykey'} = 'dummyvalue';

    return $PossibleValueConfig;
}

1;
</File>
<File Location="Custom/Kernel/Modules/AdminDynamicFieldTextAreaFromDB.pm" Permission="644" Encode="Base64"># --
# Kernel/Modules/AdminDynamicFieldDropdown.pm - provides a dynamic fields text config view for admins
# Copyright (C) 2001-2012 OTRS AG, http://otrs.org/
# --
# $Id: AdminDynamicFieldDropdown.pm,v 1.15 2012/04/18 19:38:01 cr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Modules::AdminDynamicFieldTextAreaFromDB;

use strict;
use warnings;

use Kernel::System::VariableCheck qw(:all);
use Kernel::System::Valid;
use Kernel::System::CheckItem;
use Kernel::System::DynamicField;

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

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

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

    for (qw(ParamObject LayoutObject LogObject ConfigObject)) {
        if ( !$Self->{$_} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $_!" );
        }
    }

    # create additional objects
    $Self->{ValidObject} = Kernel::System::Valid->new( %{$Self} );

    $Self->{DynamicFieldObject} = Kernel::System::DynamicField->new( %{$Self} );

    # get configured object types
    $Self->{ObjectTypeConfig} = $Self->{ConfigObject}->Get('DynamicFields::ObjectType');

    # get the fields config
    $Self->{FieldTypeConfig} = $Self->{ConfigObject}->Get('DynamicFields::Backend') || {};

    # set possible values handling strings
    $Self->{EmptyString}     = '_DynamicFields_EmptyString_Dont_Use_It_String_Please';
    $Self->{DuplicateString} = '_DynamicFields_DuplicatedString_Dont_Use_It_String_Please';
    $Self->{DeletedString}   = '_DynamicFields_DeletedString_Dont_Use_It_String_Please';

    return $Self;
}

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

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

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

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

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

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

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

    my %GetParam;
    for my $Needed (qw(ObjectType FieldType FieldOrder)) {
        $GetParam{$Needed} = $Self->{ParamObject}->GetParam( Param => $Needed );
        if ( !$Needed ) {
            return $Self->{LayoutObject}->ErrorScreen(
                Message => "Need $Needed",
            );
        }
    }

    # get the object type and field type display name
    my $ObjectTypeName = $Self->{ObjectTypeConfig}->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName  = $Self->{FieldTypeConfig}->{ $GetParam{FieldType} }->{DisplayName}   || '';

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        Mode           => 'Add',
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my %Errors;
    my %GetParam;

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

    if ( $GetParam{Name} ) {

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

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

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

        %DynamicFieldsList = reverse %DynamicFieldsList;

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

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

    if ( $GetParam{FieldOrder} ) {

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

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

    for my $ConfigParam (
        qw(
        ObjectType ObjectTypeName FieldType FieldTypeName DefaultValue PossibleNone
        TranslatableValues ValidID Link Query Parameters DBIstring DBIuser DBIpass CacheTTL
        )
        )
    {
        $GetParam{$ConfigParam} = $Self->{ParamObject}->GetParam( Param => $ConfigParam );
    }

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

#    my $PossibleValues = $Self->_GetPossibleValues();
#
#    # set errors for possible values entries
#    KEY:
#    for my $Key ( keys %{$PossibleValues} ) {
#
#        # check for empty original values
#        if ( $Key =~ m{\A $Self->{EmptyString} (?: \d+)}smx ) {
#
#            # set a true entry in KeyEmptyError
#            $Errors{'PossibleValueErrors'}->{'KeyEmptyError'}->{$Key} = 1;
#        }
#
#        # otherwise check for duplicate original values
#        elsif ( $Key =~ m{\A (.+) - $Self->{DuplicateString} (?: \d+)}smx ) {
#
#            # set an entry in OrigValueDuplicateError with the duplicate key as value
#            $Errors{'PossibleValueErrors'}->{'KeyDuplicateError'}->{$Key} = $1;
#        }
#
#        # check for empty new values
#        if ( !defined $PossibleValues->{$Key} ) {
#
#            # set a true entry in NewValueEmptyError
#            $Errors{'PossibleValueErrors'}->{'ValueEmptyError'}->{$Key} = 1;
#        }
#    }

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

    # set specific config
    my $FieldConfig = {
#        PossibleValues     => $PossibleValues,
        DefaultValue       => $GetParam{DefaultValue},
        PossibleNone       => $GetParam{PossibleNone},
        TranslatableValues => $GetParam{TranslatableValues},
        Link               => $GetParam{Link},
        Query				=> $GetParam{Query},
        Parameters			=> $GetParam{Parameters},
        DBIstring			=> $GetParam{DBIstring},
        DBIuser				=> $GetParam{DBIuser},
        DBIpass				=> $GetParam{DBIpass},
        CacheTTL			=> $GetParam{CacheTTL},
        DisplayErrors		=> $GetParam{DisplayErrors},
    };

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

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

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

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

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

    # get the object type and field type display name
    my $ObjectTypeName = $Self->{ObjectTypeConfig}->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName  = $Self->{FieldTypeConfig}->{ $GetParam{FieldType} }->{DisplayName}   || '';

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

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

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

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

    my %Config = ();

    # extract configuration
    if ( IsHashRefWithData( $DynamicFieldData->{Config} ) ) {

#        # set PossibleValues
#        $Config{PossibleValues} = {};
#        if ( IsHashRefWithData( $DynamicFieldData->{Config}->{PossibleValues} ) ) {
#            $Config{PossibleValues} = $DynamicFieldData->{Config}->{PossibleValues};
#        }

        # set DefaultValue
        $Config{DefaultValue} = $DynamicFieldData->{Config}->{DefaultValue};

        # set PossibleNone
        $Config{PossibleNone} = $DynamicFieldData->{Config}->{PossibleNone};

        # set TranslatalbeValues
        $Config{TranslatableValues} = $DynamicFieldData->{Config}->{TranslatableValues};

        # set Link
        $Config{Link} = $DynamicFieldData->{Config}->{Link};
        $Config{Query} = $DynamicFieldData->{Config}->{Query} || 'insert query here';
        $Config{Parameters} = $DynamicFieldData->{Config}->{Parameters};
        $Config{DBIstring} = $DynamicFieldData->{Config}->{DBIstring};
        $Config{DBIuser} = $DynamicFieldData->{Config}->{DBIuser};
        $Config{DBIpass} = $DynamicFieldData->{Config}->{DBIpass};
        $Config{CacheTTL} = $DynamicFieldData->{Config}->{CacheTTL};
        $Config{DisplayErrors} = $DynamicFieldData->{Config}->{DisplayErrors};
    }

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        %${DynamicFieldData},
        %Config,
        ID             => $FieldID,
        Mode           => 'Change',
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my %Errors;
    my %GetParam;

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

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

    if ( $GetParam{Name} ) {

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

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

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

        %DynamicFieldsList = reverse %DynamicFieldsList;

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

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

    if ( $GetParam{FieldOrder} ) {

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

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

    for my $ConfigParam (
        qw(
        ObjectType ObjectTypeName FieldType FieldTypeName DefaultValue PossibleNone
        TranslatableValues ValidID Link Query Parameters DBIstring DBIuser DBIpass CacheTTL
        )
        )
    {
        $GetParam{$ConfigParam} = $Self->{ParamObject}->GetParam( Param => $ConfigParam );
    }

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

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

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

#    my $PossibleValues = $Self->_GetPossibleValues();
#
#    # set errors for possible values entries
#    KEY:
#    for my $Key ( keys %{$PossibleValues} ) {
#
#        # check for empty original values
#        if ( $Key =~ m{\A $Self->{EmptyString} (?: \d+)}smx ) {
#
#            # set a true entry in KeyEmptyError
#            $Errors{'PossibleValueErrors'}->{'KeyEmptyError'}->{$Key} = 1;
#        }
#
#        # otherwise check for duplicate original values
#        elsif ( $Key =~ m{\A (.+) - $Self->{DuplicateString} (?: \d+)}smx ) {
#
#            # set an entry in OrigValueDuplicateError with the duplicate key as value
#            $Errors{'PossibleValueErrors'}->{'KeyDuplicateError'}->{$Key} = $1;
#        }
#
#        # check for empty new values
#        if ( !defined $PossibleValues->{$Key} ) {
#
#            # set a true entry in NewValueEmptyError
#            $Errors{'PossibleValueErrors'}->{'ValueEmptyError'}->{$Key} = 1;
#        }
#    }

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

    # set specific config
    my $FieldConfig = {
#        PossibleValues     => $PossibleValues,
        DefaultValue       => $GetParam{DefaultValue},
        PossibleNone       => $GetParam{PossibleNone},
        TranslatableValues => $GetParam{TranslatableValues},
        Link               => $GetParam{Link},
        Query				=> $GetParam{Query},
        Parameters			=> $GetParam{Parameters},
        DBIstring			=> $GetParam{DBIstring},
        DBIuser				=> $GetParam{DBIuser},
        DBIpass				=> $GetParam{DBIpass},
        CacheTTL			=> $GetParam{CacheTTL},
        DisplayErrors		=> $GetParam{DisplayErrors},
    };

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

    if ( !$UpdateSuccess ) {
        return $Self->{LayoutObject}->ErrorScreen(
            Message => "Could not update the field $GetParam{Name}",
        );
    }

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

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

    $Param{DisplayFieldName} = 'New';

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

    $Param{DeletedString} = $Self->{DeletedString};

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

    # get all fields
    my $DynamicFieldList = $Self->{DynamicFieldObject}->DynamicFieldListGet(
        Valid => 0,
    );

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

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

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

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

    my $DynamicFieldOrderSrtg = $Self->{LayoutObject}->BuildSelection(
        Data          => \@DynamicfieldOrderList,
        Name          => 'FieldOrder',
        SelectedValue => $Param{FieldOrder} || 1,
        PossibleNone  => 0,
        Class         => 'W50pc Validate_Number',
    );

    my %ValidList = $Self->{ValidObject}->ValidList();

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

    # define as 0 to get the real value in the HTML
    my $ValueCounter = 0;

    # set PossibleValues
#    my %PossibleValues;
#    if ( IsHashRefWithData( $Param{PossibleValues} ) ) {
#        %PossibleValues = %{ $Param{PossibleValues} };
#    }

#    # output the possible values and errors within (if any)
#    for my $Key ( sort keys %PossibleValues ) {
#
#        $ValueCounter++;
#
#        # needed for server side validation
#        my $KeyError;
#        my $KeyErrorStrg;
#        my $ValueError;
#
#        # to set the correct original value
#        my $KeyClone = $Key;
#
#        # check for errors
#        if ( $Param{'PossibleValueErrors'} ) {
#
#            # check for errors on original value (empty)
#            if ( $Param{'PossibleValueErrors'}->{'KeyEmptyError'}->{$Key} ) {
#
#                # if the original value was empty it has been changed in _GetParams to a predefined
#                # string and need to be set to empty again
#                $KeyClone = '';
#
#                # set the error class
#                $KeyError     = 'ServerError';
#                $KeyErrorStrg = 'This field is required.'
#            }
#
#            # check for errors on original value (duplicate)
#            elsif ( $Param{'PossibleValueErrors'}->{'KeyDuplicateError'}->{$Key} ) {
#
#                # if the original value was empty it has been changed in _GetParams to a predefined
#                # string and need to be set to the original value again
#                $KeyClone
#                    = $Param{'PossibleValueErrors'}->{'KeyDuplicateError'}->{$Key};
#
#                # set the error class
#                $KeyError     = 'ServerError';
#                $KeyErrorStrg = 'This field value is duplicated.'
#            }
#
#            # check for error on value
#            if ( $Param{'PossibleValueErrors'}->{'ValueEmptyError'}->{$Key} ) {
#
#                # set the error class
#                $ValueError = 'ServerError';
#            }
#        }
#
#        # create a value map row
#        $Self->{LayoutObject}->Block(
#            Name => 'ValueRow',
#            Data => {
#                KeyError     => $KeyError,
#                KeyErrorStrg => $KeyErrorStrg || 'This field is required.',
#                Key          => $KeyClone,
#                ValueCounter => $ValueCounter,
#                Value        => $PossibleValues{$Key},
#                ValueError   => $ValueError,
#            },
#        );
#    }

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

    # check and build the Default Value list based on Possible Values
#    my %DefaultValuesList;
#    POSSIBLEVALUE:
#    for my $ValueItem ( keys %PossibleValues ) {
#        next POSSIBLEVALUE if !defined $ValueItem;
#        next POSSIBLEVALUE if !defined $PossibleValues{$ValueItem};
#        $DefaultValuesList{$ValueItem} = $PossibleValues{$ValueItem}
#    }
#
#    my $DefaultValue = ( defined $Param{DefaultValue} ? $Param{DefaultValue} : '' );
#
#    # create the default value select
#    my $DefaultValueStrg = $Self->{LayoutObject}->BuildSelection(
#        Data         => \%DefaultValuesList,
#        Name         => 'DefaultValue',
#        SelectedID   => $DefaultValue,
#        PossibleNone => 1,
#
#        # Don't make is translatable because this will confuse the user (also current JS
#        # is not prepared)
#        Translation => 0,
#
#        # Multiple selections are currently not supported
#        Multiple => 0,
#        Class    => 'W50pc',
#    );

    my $PossibleNone = $Param{PossibleNone} || '0';

    # create translatable values option list
    my $PossibleNoneStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'PossibleNone',
        SelectedID => $PossibleNone,
        Class      => 'W50pc',
    );

    my $TranslatableValues = $Param{TranslatableValues} || '0';

    # create translatable values option list
    my $TranslatableValuesStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'TranslatableValues',
        SelectedID => $TranslatableValues,
        Class      => 'W50pc',
    );

    my $DisplayErrors = $Param{DisplayErrors} || '0';

    # create translatable values option list
    my $DisplayErrorsStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'DisplayErrors',
        SelectedID => $DisplayErrors,
        Class      => 'W50pc',
    );

    my $Link = $Param{Link} || '';
    my $Query = $Param{Query} || '';
    my $Parameters = $Param{Parameters} || '';
    my $DBIstring = $Param{DBIstring} || '';
    my $DBIuser = $Param{DBIuser} || '';
    my $DBIpass = $Param{DBIpass} || '';
    my $CacheTTL = $Param{CacheTTL} || '';

    # generate output
    $Output .= $Self->{LayoutObject}->Output(
        TemplateFile => 'AdminDynamicFieldTextAreaFromDB',
        Data         => {
            %Param,
            ValidityStrg           => $ValidityStrg,
            DynamicFieldOrderSrtg  => $DynamicFieldOrderSrtg,
#            ValueCounter           => $ValueCounter,
#            DefaultValueStrg       => $DefaultValueStrg,
            PossibleNoneStrg       => $PossibleNoneStrg,
            TranslatableValuesStrg => $TranslatableValuesStrg,
            Link                   => $Link,
            Query					=> $Query,
            Parameters				=> $Parameters,
            DBIstring				=> $DBIstring,
            DBIuser					=> $DBIuser,
            DBIpass					=> $DBIpass,
            CacheTTL				=> $CacheTTL,
            DisplayErrors			=> $DisplayErrorsStrg,
            }
    );

    $Output .= $Self->{LayoutObject}->Footer();

    return $Output;
}

#sub _GetPossibleValues {
#    my ( $Self, %Param ) = @_;
#
#    my $PossibleValueConfig;
#
#    # get parameters from web browser
#    # get ValueCounters
#    my $ValueCounter          = $Self->{ParamObject}->GetParam( Param => 'ValueCounter' ) || 0;
#    my $EmptyValueCounter     = 0;
#    my $DuplicateValueCounter = 0;
#
#    # get possible values
#    my $Values;
#    VALUEINDEX:
#    for my $ValueIndex ( 1 .. $ValueCounter ) {
#        my $Key = $Self->{ParamObject}->GetParam( Param => 'Key' . '_' . $ValueIndex );
#        $Key = ( defined $Key ? $Key : '' );
#
#        # check if key was deleted by the user and skip it
#        next VALUEINDEX if $Key eq $Self->{DeletedString};
#
#        # check if the original value is empty
#        if ( $Key eq '' ) {
#
#            # change the empty value to a predefined string
#            $Key = $Self->{EmptyString} . int $EmptyValueCounter;
#            $EmptyValueCounter++;
#        }
#
#        # otherwise check for duplicate
#        elsif ( exists $PossibleValueConfig->{$Key} ) {
#
#            # append a predefined unique string to make this value unique
#            $Key .= '-' . $Self->{DuplicateString} . $DuplicateValueCounter;
#            $DuplicateValueCounter++;
#        }
#
#        my $Value = $Self->{ParamObject}->GetParam( Param => 'Value' . '_' . $ValueIndex );
#        $Value = ( defined $Value ? $Value : '' );
#        $PossibleValueConfig->{$Key} = $Value;
#    }
#    return $PossibleValueConfig;
#}

1;
</File>
<File Location="Custom/Kernel/Modules/AdminDynamicFieldMultiselectFromDB.pm" Permission="644" Encode="Base64"># --
# Kernel/Modules/AdminDynamicFieldDropdown.pm - provides a dynamic fields text config view for admins
# Copyright (C) 2001-2012 WuerthPhoenix SRL, http://www.wuerthphoenix.it/
# --
# $Id: AdminDynamicFieldDropdownFromDB.pm,v 1.00 2012/04/18 19:38:01 cr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Modules::AdminDynamicFieldMultiselectFromDB;

use strict;
use warnings;

use Kernel::System::VariableCheck qw(:all);
use Kernel::System::Valid;
use Kernel::System::CheckItem;
use Kernel::System::DynamicField;

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

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

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

    for (qw(ParamObject LayoutObject LogObject ConfigObject)) {
        if ( !$Self->{$_} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $_!" );
        }
    }

    # create additional objects
    $Self->{ValidObject} = Kernel::System::Valid->new( %{$Self} );

    $Self->{DynamicFieldObject} = Kernel::System::DynamicField->new( %{$Self} );

    # get configured object types
    $Self->{ObjectTypeConfig} = $Self->{ConfigObject}->Get('DynamicFields::ObjectType');

    # get the fields config
    $Self->{FieldTypeConfig} = $Self->{ConfigObject}->Get('DynamicFields::Backend') || {};

    # set possible values handling strings
    $Self->{EmptyString}     = '_DynamicFields_EmptyString_Dont_Use_It_String_Please';
    $Self->{DuplicateString} = '_DynamicFields_DuplicatedString_Dont_Use_It_String_Please';
    $Self->{DeletedString}   = '_DynamicFields_DeletedString_Dont_Use_It_String_Please';

    return $Self;
}

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

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

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

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

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

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

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

    my %GetParam;
    for my $Needed (qw(ObjectType FieldType FieldOrder)) {
        $GetParam{$Needed} = $Self->{ParamObject}->GetParam( Param => $Needed );
        if ( !$Needed ) {
            return $Self->{LayoutObject}->ErrorScreen(
                Message => "Need $Needed",
            );
        }
    }

    # get the object type and field type display name
    my $ObjectTypeName = $Self->{ObjectTypeConfig}->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName  = $Self->{FieldTypeConfig}->{ $GetParam{FieldType} }->{DisplayName}   || '';

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        Mode           => 'Add',
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my %Errors;
    my %GetParam;

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

    if ( $GetParam{Name} ) {

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

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

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

        %DynamicFieldsList = reverse %DynamicFieldsList;

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

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

    if ( $GetParam{FieldOrder} ) {

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

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

    for my $ConfigParam (
        qw(
        ObjectType ObjectTypeName FieldType FieldTypeName DefaultValue PossibleNone
        TranslatableValues ValidID Link Query Parameters DBIstring DBIuser DBIpass Separator CacheTTL
        )
        )
    {
        $GetParam{$ConfigParam} = $Self->{ParamObject}->GetParam( Param => $ConfigParam );
    }

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

    my $PossibleValues = $Self->_GetPossibleValues();

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

    # set specific config
    my $FieldConfig = {
#        PossibleValues     => $PossibleValues,
        DefaultValue       => $GetParam{DefaultValue},
        PossibleNone       => $GetParam{PossibleNone},
        TranslatableValues => $GetParam{TranslatableValues},
        Link               => $GetParam{Link},
        Query		   => $GetParam{Query},
        Parameters	   => $GetParam{Parameters},
        DBIstring	   => $GetParam{DBIstring},
        DBIuser		   => $GetParam{DBIuser},
        DBIpass		   => $GetParam{DBIpass},
        Separator	   => $GetParam{Separator},
        CacheTTL	   => $GetParam{CacheTTL},
        DisplayErrors  => $GetParam{DisplayErrors},
    };

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

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

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

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

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

    # get the object type and field type display name
    my $ObjectTypeName = $Self->{ObjectTypeConfig}->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName  = $Self->{FieldTypeConfig}->{ $GetParam{FieldType} }->{DisplayName}   || '';

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

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

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

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

    my %Config = ();

    # extract configuration
    if ( IsHashRefWithData( $DynamicFieldData->{Config} ) ) {

#        # set PossibleValues
#        $Config{PossibleValues} = {};
#        if ( IsHashRefWithData( $DynamicFieldData->{Config}->{PossibleValues} ) ) {
#            $Config{PossibleValues} = $DynamicFieldData->{Config}->{PossibleValues};
#        }

        # set DefaultValue
        $Config{DefaultValue} = $DynamicFieldData->{Config}->{DefaultValue};

        # set PossibleNone
        $Config{PossibleNone} = $DynamicFieldData->{Config}->{PossibleNone};

        # set TranslatalbeValues
        $Config{TranslatableValues} = $DynamicFieldData->{Config}->{TranslatableValues};

        # set Link
        $Config{Link} = $DynamicFieldData->{Config}->{Link};
        $Config{Query} = $DynamicFieldData->{Config}->{Query};
        $Config{Parameters} = $DynamicFieldData->{Config}->{Parameters};
        $Config{DBIstring} = $DynamicFieldData->{Config}->{DBIstring};
        $Config{DBIuser} = $DynamicFieldData->{Config}->{DBIuser};
        $Config{DBIpass} = $DynamicFieldData->{Config}->{DBIpass};
        $Config{Separator} = $DynamicFieldData->{Config}->{Separator};
        $Config{CacheTTL} = $DynamicFieldData->{Config}->{CacheTTL};
        $Config{DisplayErrors} = $DynamicFieldData->{Config}->{DisplayErrors};
    }

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        %${DynamicFieldData},
        %Config,
        ID             => $FieldID,
        Mode           => 'Change',
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my %Errors;
    my %GetParam;

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

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

    if ( $GetParam{Name} ) {

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

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

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

        %DynamicFieldsList = reverse %DynamicFieldsList;

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

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

    if ( $GetParam{FieldOrder} ) {

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

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

    for my $ConfigParam (
        qw(
        ObjectType ObjectTypeName FieldType FieldTypeName DefaultValue PossibleNone
        TranslatableValues ValidID Link Query Parameters DBIstring DBIuser DBIpass Separator CacheTTL
        )
        )
    {
        $GetParam{$ConfigParam} = $Self->{ParamObject}->GetParam( Param => $ConfigParam );
    }

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

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

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

    my $PossibleValues = $Self->_GetPossibleValues();

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

    # set specific config
    my $FieldConfig = {
        PossibleValues     => $PossibleValues,
        DefaultValue       => $GetParam{DefaultValue},
        PossibleNone       => $GetParam{PossibleNone},
        TranslatableValues => $GetParam{TranslatableValues},
        Link               => $GetParam{Link},
        Query				=> $GetParam{Query},
        Parameters			=> $GetParam{Parameters},
        DBIstring			=> $GetParam{DBIstring},
        DBIuser				=> $GetParam{DBIuser},
        DBIpass				=> $GetParam{DBIpass},
        Separator			=> $GetParam{Separator},
        CacheTTL			=> $GetParam{CacheTTL},
        DisplayErrors		=> $GetParam{DisplayErrors},
    };

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

    if ( !$UpdateSuccess ) {
        return $Self->{LayoutObject}->ErrorScreen(
            Message => "Could not update the field $GetParam{Name}",
        );
    }

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

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

    $Param{DisplayFieldName} = 'New';

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

    $Param{DeletedString} = $Self->{DeletedString};

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

    # get all fields
    my $DynamicFieldList = $Self->{DynamicFieldObject}->DynamicFieldListGet(
        Valid => 0,
    );

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

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

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

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

    my $DynamicFieldOrderSrtg = $Self->{LayoutObject}->BuildSelection(
        Data          => \@DynamicfieldOrderList,
        Name          => 'FieldOrder',
        SelectedValue => $Param{FieldOrder} || 1,
        PossibleNone  => 0,
        Class         => 'W50pc Validate_Number',
    );

    my %ValidList = $Self->{ValidObject}->ValidList();

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

    # define as 0 to get the real value in the HTML
    my $ValueCounter = 0;

    # set PossibleValues
    my %PossibleValues;
    if ( IsHashRefWithData( $Param{PossibleValues} ) ) {
        %PossibleValues = %{ $Param{PossibleValues} };
    }


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

   #Kernel/System/Ticket/OTRSTicketMaskExtensions.pm 
eval
{
  require Kernel::System::Ticket::OTRSTicketMaskExtensions;
};

unless(! $@)
{
    $Self->{LayoutObject}->Block(
        Name => 'TicketMaskExtensionDisabled',
        Data => {
            %Param,
        },
    );

}
 
#if (! eval "use Kernel::System::Ticket::OTRSTicketMaskExtensions") {
#use Kernel::System::Ticket::OTRSTicketMaskExtensions;
#}


    # check and build the Default Value list based on Possible Values
#    my %DefaultValuesList;
#    POSSIBLEVALUE:
#    for my $ValueItem ( keys %PossibleValues ) {
#        next POSSIBLEVALUE if !defined $ValueItem;
#        next POSSIBLEVALUE if !defined $PossibleValues{$ValueItem};
#        $DefaultValuesList{$ValueItem} = $PossibleValues{$ValueItem}
#    }
#
#    my $DefaultValue = ( defined $Param{DefaultValue} ? $Param{DefaultValue} : '' );
#
#    # create the default value select
#    my $DefaultValueStrg = $Self->{LayoutObject}->BuildSelection(
#        Data         => \%DefaultValuesList,
#        Name         => 'DefaultValue',
#        SelectedID   => $DefaultValue,
#        PossibleNone => 1,
#
#        # Don't make is translatable because this will confuse the user (also current JS
#        # is not prepared)
#        Translation => 0,
#
#        # Multiple selections are currently not supported
#        Multiple => 0,
#        Class    => 'W50pc',
#    );

    my $PossibleNone = $Param{PossibleNone} || '0';

    # create translatable values option list
    my $PossibleNoneStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'PossibleNone',
        SelectedID => $PossibleNone,
        Class      => 'W50pc',
    );

    my $TranslatableValues = $Param{TranslatableValues} || '0';

    # create translatable values option list
    my $TranslatableValuesStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'TranslatableValues',
        SelectedID => $TranslatableValues,
        Class      => 'W50pc',
    );

    my $DisplayErrors = $Param{DisplayErrors} || '0';

    # create translatable values option list
    my $DisplayErrorsStrg = $Self->{LayoutObject}->BuildSelection(
        Data => {
            0 => 'No',
            1 => 'Yes',
        },
        Name       => 'DisplayErrors',
        SelectedID => $DisplayErrors,
        Class      => 'W50pc',
    );

    my $Link = $Param{Link} || '';
    my $Query = $Param{Query} || '';
    my $Parameters = $Param{Parameters} || '';
    my $DBIstring = $Param{DBIstring} || '';
    my $DBIuser = $Param{DBIuser} || '';
    my $DBIpass = $Param{DBIpass} || '';
    my $Separator = $Param{Separator} || ',';
    my $CacheTTL = $Param{CacheTTL} || '360';

    # generate output
    $Output .= $Self->{LayoutObject}->Output(
        TemplateFile => 'AdminDynamicFieldMultiselectFromDB',
        Data         => {
            %Param,
            ValidityStrg           => $ValidityStrg,
            DynamicFieldOrderSrtg  => $DynamicFieldOrderSrtg,
#            ValueCounter           => $ValueCounter,
#            DefaultValueStrg       => $DefaultValueStrg,
            PossibleNoneStrg       => $PossibleNoneStrg,
            TranslatableValuesStrg => $TranslatableValuesStrg,
            Link                   => $Link,
            Query				=> $Query,
            Parameters			=> $Parameters,
            DBIstring			=> $DBIstring,
            DBIuser				=> $DBIuser,
            DBIpass				=> $DBIpass,
            Separator			=> $Separator,
            CacheTTL			=> $CacheTTL,
            DisplayErrors		=> $CacheTTL,
            }
    );

    $Output .= $Self->{LayoutObject}->Footer();

    return $Output;
}

sub _GetPossibleValues {
    my ( $Self, %Param ) = @_;
#
    my $PossibleValueConfig;
#
#    # get parameters from web browser
#    # get ValueCounters
#    my $ValueCounter          = $Self->{ParamObject}->GetParam( Param => 'ValueCounter' ) || 0;
#    my $EmptyValueCounter     = 0;
#    my $DuplicateValueCounter = 0;
#
#    # get possible values
#    my $Values;
#    VALUEINDEX:
#    for my $ValueIndex ( 1 .. $ValueCounter ) {
#        my $Key = $Self->{ParamObject}->GetParam( Param => 'Key' . '_' . $ValueIndex );
#        $Key = ( defined $Key ? $Key : '' );
#
#        # check if key was deleted by the user and skip it
#        next VALUEINDEX if $Key eq $Self->{DeletedString};
#
#        # check if the original value is empty
#        if ( $Key eq '' ) {
#
#            # change the empty value to a predefined string
#            $Key = $Self->{EmptyString} . int $EmptyValueCounter;
#            $EmptyValueCounter++;
#        }
#
#        # otherwise check for duplicate
#        elsif ( exists $PossibleValueConfig->{$Key} ) {
#
#            # append a predefined unique string to make this value unique
#            $Key .= '-' . $Self->{DuplicateString} . $DuplicateValueCounter;
#            $DuplicateValueCounter++;
#        }
#
#        my $Value = $Self->{ParamObject}->GetParam( Param => 'Value' . '_' . $ValueIndex );
#        $Value = ( defined $Value ? $Value : '' );
#        $PossibleValueConfig->{$Key} = $Value;
#    }

#    $PossibleValueConfig->{'dummykey'} = 'dummyvalue';

    return $PossibleValueConfig;
}

1;
</File>
<File Location="Custom/Kernel/System/DynamicField/Backend/DropdownFromDB.pm" Permission="644" Encode="Base64"># --
# Kernel/System/DynamicField/Backend/Dropdown.pm - Delegate for DynamicField Dropdown backend
# Copyright (C) 2001-2012 OTRS AG, http://otrs.org/
# --
# $Id: Dropdown.pm,v 1.63 2012/04/02 11:46:35 mg Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::DynamicField::Backend::DropdownFromDB;

use strict;
use warnings;
use DBI;

use Kernel::System::VariableCheck qw(:all);
use Kernel::System::DynamicFieldValue;
use Kernel::System::DynamicField::Backend::BackendCommon;
use Kernel::System::Cache;
use Kernel::System::Ticket;
#use Kernel::System::CacheInternal;

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

=head1 NAME

Kernel::System::DynamicField::Backend::Dropdown

=head1 SYNOPSIS

DynamicFields Dropdown backend delegate

=head1 PUBLIC INTERFACE

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

=over 4

=item new()

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

=cut

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

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

    # get needed objects
    for my $Needed (qw(ConfigObject EncodeObject LogObject MainObject DBObject)) {
        die "Got no $Needed!" if !$Param{$Needed};

        $Self->{$Needed} = $Param{$Needed};
    }

    # create additional objects
    $Self->{DynamicFieldValueObject} = Kernel::System::DynamicFieldValue->new( %{$Self} );
    $Self->{BackendCommonObject}
        = Kernel::System::DynamicField::Backend::BackendCommon->new( %{$Self} );

    $Self->{CacheObject} = Kernel::System::Cache->new(%Param);
    $Self->{DBObject} = Kernel::System::DB->new(%Param);

    return $Self;
}

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

    my $DFValue = $Self->{DynamicFieldValueObject}->ValueGet(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
    );

    return if !$DFValue;
    return if !IsArrayRefWithData($DFValue);
    return if !IsHashRefWithData( $DFValue->[0] );

    return $DFValue->[0]->{ValueText};
}

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

#    # check for valid possible values list
#    if ( !$Param{DynamicFieldConfig}->{Config}->{PossibleValues} ) {
#        $Self->{LogObject}->Log(
#            Priority => 'error',
#            Message  => "Need PossibleValues in DynamicFieldConfig!",
#        );
#        return;
#    }

    my $Success = $Self->{DynamicFieldValueObject}->ValueSet(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
        Value    => [
            {
                ValueText => $Param{Value},
            },
        ],
        UserID => $Param{UserID},
    );

    return $Success;
}

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

    my $Success = $Self->{DynamicFieldValueObject}->ValueDelete(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
        UserID   => $Param{UserID},
    );

    return $Success;
}

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

    my $Success = $Self->{DynamicFieldValueObject}->AllValuesDelete(
        FieldID => $Param{DynamicFieldConfig}->{ID},
        UserID  => $Param{UserID},
    );

    return $Success;
}

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

    my $Success = $Self->{DynamicFieldValueObject}->ValueValidate(
        Value => {
            ValueText => $Param{Value},
        },
        UserID => $Param{UserID}
    );

    return $Success;
}

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

    my %Operators = (
        Equals            => '=',
        GreaterThan       => '>',
        GreaterThanEquals => '>=',
        SmallerThan       => '<',
        SmallerThanEquals => '<=',
    );

    if ( $Operators{ $Param{Operator} } ) {
        my $SQL = " $Param{TableAlias}.value_text $Operators{$Param{Operator}} '";
        $SQL .= $Self->{DBObject}->Quote( $Param{SearchTerm} ) . "' ";
        return $SQL;
    }

    if ( $Param{Operator} eq 'Like' ) {

        my $SQL = $Self->{DBObject}->QueryCondition(
            Key   => "$Param{TableAlias}.value_text",
            Value => $Param{SearchTerm},
        );

        return $SQL;
    }

    $Self->{'LogObject'}->Log(
        'Priority' => 'error',
        'Message'  => "Unsupported Operator $Param{Operator}",
    );

    return;
}

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

    return "$Param{TableAlias}.value_text";
}

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

    # take config from field config
    my $FieldConfig = $Param{DynamicFieldConfig}->{Config};
    my $FieldName   = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};
    my $FieldLabel  = $Param{DynamicFieldConfig}->{Label};

    my $Value = '';

    # set the field value or default
    if ( $Param{UseDefaultValue} ) {
        $Value = ( defined $FieldConfig->{DefaultValue} ? $FieldConfig->{DefaultValue} : '' );
    }
    $Value = $Param{Value} if defined $Param{Value};

    # extract the dynamic field value form the web request
    my $FieldValue = $Self->EditFieldValueGet(
        %Param,
    );

    # set values from ParamObject if present
    if ( defined $FieldValue ) {
        $Value = $FieldValue;
    }
    # check and set class if necessary
    my $FieldClass = 'DynamicFieldText';
    if ( defined $Param{Class} && $Param{Class} ne '' ) {
        $FieldClass .= ' ' . $Param{Class};
    }
    $FieldClass .= ' FromDB';

    # set field as mandatory
    $FieldClass .= ' Validate_Required' if $Param{Mandatory};

    # set error css class
    $FieldClass .= ' ServerError' if $Param{ServerError};

    # set PossibleValues
    my $SelectionData = $FieldConfig->{PossibleValues};

    ### FOTH preload data at creation
    $SelectionData = $Self->AJAXPossibleValuesGet(%Param, ForceQuery => 1);
    ### END FOTH

    # use PossibleValuesFilter if defined
    $SelectionData = $Param{PossibleValuesFilter}
        if defined $Param{PossibleValuesFilter};

    # set PossibleNone attribute
    my $FieldPossibleNone;
    if ( defined $Param{OverridePossibleNone} ) {
        $FieldPossibleNone = $Param{OverridePossibleNone};
    }
    else {
        $FieldPossibleNone = $FieldConfig->{PossibleNone} || 0;
    }

    my $HTMLString = $Param{LayoutObject}->BuildSelection(
        Data => $SelectionData || {},
        Name => $FieldName,
        SelectedID   => $Value,
        Translation  => $FieldConfig->{TranslatableValues} || 0,
        PossibleNone => 0, # we do not need to set possibleNone value because we do it already in the AJAXPossibibleValuesGet
        Class        => $FieldClass,
        HTMLQuote    => 1,
    );
    
    if ($FieldConfig->{DisplayErrors}) {
    	my $DivID = $FieldName . 'Warning';
    	
    	# for client side validation
        $HTMLString .= <<"EOF";
    <div style="color:red">\$Text{"[Debug mode]"}</div>
EOF
    }    

    if ( $Param{Mandatory} ) {
        my $DivID = $FieldName . 'Error';

        # for client side validation
        $HTMLString .= <<"EOF";

    <div id="$DivID" class="TooltipErrorMessage">
        <p>
            \$Text{"This field is required."}
        </p>
    </div>
EOF
    }

    if ( $Param{ServerError} ) {

        my $ErrorMessage = $Param{ErrorMessage} || 'This field is required.';
        my $DivID = $FieldName . 'ServerError';

        # for server side validation
        $HTMLString .= <<"EOF";
    <div id="$DivID" class="TooltipErrorMessage">
        <p>
            \$Text{"$ErrorMessage"}
        </p>
    </div>
EOF
    }

    if ( $Param{AJAXUpdate} or 1 ) {

        my $FieldSelector = '#' . $FieldName;

        my $FieldsToUpdate = '';
        if ( IsArrayRefWithData( $Param{UpdatableFields} ) ) {
            my $FirstItem = 1;
            FIELD:
            for my $Field ( @{ $Param{UpdatableFields} } ) {
                next FIELD if $Field eq $FieldName;
                if ($FirstItem) {
                    $FirstItem = 0;
                }
                else {
                    $FieldsToUpdate .= ', ';
                }
                $FieldsToUpdate .= "'" . $Field . "'";
            }
        }

        #add js to call FormUpdate()
        $HTMLString .= <<"EOF";
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
    \$('$FieldSelector').bind('change', function (Event) {
        Core.AJAX.FormUpdate(\$(this).parents('form'), 'AJAXUpdate', '$FieldName', [ $FieldsToUpdate ]);
    });
//]]></script>
<!--dtl:js_on_document_complete-->
EOF
    }


    if ( $Param{SubmitOnChange} ) {

        my $FieldSelector = '#' . $FieldName;

        #add js to disable validation and do submit()
        $HTMLString .= <<"EOF";
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
    \$('$FieldSelector').bind('change', function (Event) {
        // make sure the ticket is not yet created on queue change
        \$('input#Expand').val(1);
        Core.Form.Validate.DisableValidation(\$(this).closest('form'));
        \$(this).closest('form').submit();
    });
//]]></script>
<!--dtl:js_on_document_complete-->
EOF
    }

#	/DynamicFieldConfig
#	$Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->AJAXPossibleValuesGet(%Param);

    # call EditLabelRender on the common backend
    my $LabelString = $Self->{BackendCommonObject}->EditLabelRender(
        DynamicFieldConfig => $Param{DynamicFieldConfig},
        Mandatory          => $Param{Mandatory} || '0',
        FieldName          => $FieldName,
    );

    my $Data = {
        Field => $HTMLString,
        Label => $LabelString,
    };


    return $Data;
}

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

    my $FieldName = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};

    my $Value;

    # check if there is a Template and retreive the dinalic field value from there
    if ( IsHashRefWithData( $Param{Template} ) ) {
        $Value = $Param{Template}->{$FieldName};
    }

    # otherwise get dynamic field value form param
    else {
        $Value = $Param{ParamObject}->GetParam( Param => $FieldName );
    }

    if ( defined $Param{ReturnTemplateStructure} && $Param{ReturnTemplateStructure} eq 1 ) {
        return {
            $FieldName => $Value,
        };
    }

    # for this field the normal return an the ReturnValueStructure are the same
    return $Value;
}

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

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

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

    my $ServerError;
    my $ErrorMessage;

    # perform necessary validations
    if ( $Param{Mandatory} && !$Value ) {
        return {
            ServerError => 1,
        };
    }
#    else {
#
#        # get possible values list
#        my $PossibleValues = $Param{DynamicFieldConfig}->{Config}->{PossibleValues};
#
#        # overwrite possible values if PossibleValuesFilter
#        if ( defined $Param{PossibleValuesFilter} ) {
#            $PossibleValues = $Param{PossibleValuesFilter}
#        }
#
#        # validate if value is in possible values list (but let pass empty values)
#        if ( $Value && !$PossibleValues->{$Value} ) {
#            $ServerError  = 1;
#            $ErrorMessage = 'The field content is invalid';
#        }
#    }

    # Validate anything (for now)
    my $Result = {
        ServerError  => $ServerError,
        ErrorMessage => $ErrorMessage,
    };

    return $Result;
}

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

    if ( !defined $Param{HTMLOutput} ) {
        $Param{HTMLOutput} = 1;
    }

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


    # get value from stored value in the DB if option is set
    if ($Param{DynamicFieldConfig}->{Config}->{StoreValue} eq "1") {
        $Value =~ s/^([^|]+)\|\|(.+)$/$2/;
    }
    else {
        my %PossibleValues;
    #	Type => 'Hash',
    #	Key => $Param{DynamicFieldConfig}->{Config}->{Name},
    #    );
    
    #    if ( !defined %PossibleValues ) {
        my @row;
        my $dbh;
        my $sth;

        if ( !$Param{DynamicFieldConfig}->{Config}->{DBIstring} ) {
            $Self->{DBObject}->Prepare(
                SQL => $Param{DynamicFieldConfig}->{Config}->{VisualQuery},
        		Bind => $Key,
            );

            #fetch first row
            @row = $Self->{DBObject}->FetchrowArray();

        } else {
            $dbh = DBI->connect($Param{DynamicFieldConfig}->{Config}->{DBIstring}, $Param{DynamicFieldConfig}->{Config}->{DBIuser}, $Param{DynamicFieldConfig}->{Config}->{DBIpass},
                              { PrintError => 0, AutoCommit => 0, HandleError => \&PossibleValuesError }) or return $Self->PossibleValuesError("could not connect to DB: $DBI::errstr");
        
            $dbh->{'mysql_enable_utf8'} = 1;
    
            $sth = $dbh->prepare($Param{DynamicFieldConfig}->{Config}->{VisualQuery});
            $sth->execute( $Key );
    
            # fetch first row
            @row = $sth->fetchrow_array;        
        }
    
    	my $line = '';
    	for my $col (@row) {
    	    $line .= $col.$Param{DynamicFieldConfig}->{Config}->{Separator};
    	}
    	$line = substr($line, 0, -1 * length($Param{DynamicFieldConfig}->{Config}->{Separator}));
    
        if ( $Param{DynamicFieldConfig}->{Config}->{DBIstring} ) {
            $dbh->disconnect;
        }
    
        #$Value = $line; WP - DENI 20130825 

        # get real value
        if ( $PossibleValues{$Value} ) {
                
            # get readeable value
            $Value = $PossibleValues{$Value};
        }
    }
   
    # check is needed to translate values
    if ( $Param{DynamicFieldConfig}->{Config}->{TranslatableValues} ) {

        # translate value
        $Value = $Param{LayoutObject}->{LanguageObject}->Get($Value);
    }

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

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

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

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

    
    if ($Param{DynamicFieldConfig}->{Config}->{StoreValue} && $Param{DynamicFieldConfig}->{Config}->{StoreValue} eq "1") {
    	$Key =~ s/^([^|]+)\|\|(.+)$/$1/;
    }
    $Link =~ s/%KEY%/$Key/g;

    my $Query = $Param{DynamicFieldConfig}->{Config}->{Query} || '';

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

    return $Data;
}

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

    return 1;
}

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

    # take config from field config
    my $FieldConfig = $Param{DynamicFieldConfig}->{Config};
    my $FieldName   = 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name};
    my $FieldLabel  = $Param{DynamicFieldConfig}->{Label};

    my $Value;

    my @DefaultValue;

    if ( defined $Param{DefaultValue} ) {
        my @DefaultValue = split /;/, $Param{DefaultValue};
    }

    # set the field value
    if (@DefaultValue) {
        $Value = \@DefaultValue;
    }

    # get the field value, this fuction is always called after the profile is loaded
    my $FieldValues = $Self->SearchFieldValueGet(
        %Param,
    );

    if ( defined $FieldValues ) {
        $Value = $FieldValues;
    }

    # check and set class if necessary
    my $FieldClass = 'DynamicFieldMultiSelect FromDB';

    # set PossibleValues
    my $SelectionData = $FieldConfig->{PossibleValues};

    # get historical values from database
    my $HistoricalValues = $Self->HistoricalValuesGet(%Param);

    # add historic values to current values (if they don't exist anymore)
    if ( IsHashRefWithData($HistoricalValues) ) {
        for my $Key ( keys %{$HistoricalValues} ) {
            if ( !$SelectionData->{$Key} ) {
                $SelectionData->{$Key} = $HistoricalValues->{$Key}
            }
        }
    }

    # use PossibleValuesFilter if defined
    $SelectionData = $Param{PossibleValuesFilter}
        if defined $Param{PossibleValuesFilter};

    my $HTMLString = $Param{LayoutObject}->BuildSelection(
        Data         => $SelectionData,
        Name         => $FieldName,
        SelectedID   => $Value,
        Translation  => $FieldConfig->{TranslatableValues} || 0,
        PossibleNone => 0,
        Class        => $FieldClass,
        Multiple     => 1,
        HTMLQuote    => 1,
    );

    # call EditLabelRender on the common backend
    my $LabelString = $Self->{BackendCommonObject}->EditLabelRender(
        DynamicFieldConfig => $Param{DynamicFieldConfig},
        FieldName          => $FieldName,
    );

    my $Data = {
        Field => $HTMLString,
        Label => $LabelString,
    };

    return $Data;
}

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

    my $Value;

    # get dynamic field value form param object
    if ( defined $Param{ParamObject} ) {
        my @FieldValues = $Param{ParamObject}
            ->GetArray( Param => 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name} );

        $Value = \@FieldValues;
    }

    # otherwise get the value from the profile
    elsif ( defined $Param{Profile} ) {
        $Value = $Param{Profile}->{ 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name} };
    }
    else {
        return;
    }

    if ( defined $Param{ReturnProfileStructure} && $Param{ReturnProfileStructure} eq 1 ) {
        return {
            'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name} => $Value,
        };
    }

    return $Value;

}

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

    # get field value
    my $Value = $Self->SearchFieldValueGet(%Param);

    my $DisplayValue;

    if ($Value) {
        if ( ref $Value eq 'ARRAY' ) {

            my @DisplayItemList;
            for my $Item ( @{$Value} ) {

                # set the display value
                my $DisplayItem = $Param{DynamicFieldConfig}->{Config}->{PossibleValues}->{$Item}
                    || $Item;

		$DisplayItem = 'yyy';

                if ( $Param{DynamicFieldConfig}->{Config}->{TranslatableValues} ) {

                    # translate the value
                    $DisplayItem = $Param{LayoutObject}->{LanguageObject}->Get($DisplayValue);
                }

                push @DisplayItemList, $DisplayItem;
            }

            # combine different values into one string
            $DisplayValue = join ' + ', @DisplayItemList;
        }
        else {

            # set the display value
            $DisplayValue = $Param{DynamicFieldConfig}->{PossibleValues}->{$Value};
            
            $DisplayValue = 'xxx';

            if ( $Param{DynamicFieldConfig}->{Config}->{TranslatableValues} ) {

                # translate the value
                $DisplayValue = $Param{LayoutObject}->{LanguageObject}->Get($DisplayValue);
            }
        }
    }

    # return search parameter structure
    return {
        Parameter => {
            Equals => $Value,
        },
        Display => $DisplayValue,
    };
}

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

    # set PossibleValues
    my $Values = $Param{DynamicFieldConfig}->{Config}->{PossibleValues};

    # get historical values from database
    my $HistoricalValues = $Self->{DynamicFieldValueObject}->HistoricalValueGet(
        FieldID   => $Param{DynamicFieldConfig}->{ID},
        ValueType => 'Text,',
    );

    # add historic values to current values (if they don't exist anymore)
    for my $Key ( keys %{$HistoricalValues} ) {
        if ( !$Values->{$Key} ) {
            $Values->{$Key} = $HistoricalValues->{$Key}
        }
    }

    # use PossibleValuesFilter if defined
    $Values = $Param{PossibleValuesFilter}
        if defined $Param{PossibleValuesFilter};

    return {
        Values             => $Values,
        Name               => $Param{DynamicFieldConfig}->{Label},
        Element            => 'DynamicField_' . $Param{DynamicFieldConfig}->{Name},
        TranslatableValues => $Param{DynamicFieldconfig}->{Config}->{TranslatableValues},
    };
}

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

    my $Operator = 'Equals';
    my $Value    = $Param{Value};

    return {
        $Operator => $Value,
    };
}

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

    my $Value = defined $Param{Value} ? $Param{Value} : '';

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

    # cut strings if needed
    if ( $Param{ValueMaxChars} && length($Value) > $Param{ValueMaxChars} ) {
        $Value = substr( $Value, 0, $Param{ValueMaxChars} ) . '...';
    }
    if ( $Param{TitleMaxChars} && length($Title) > $Param{TitleMaxChars} ) {
        $Title = substr( $Title, 0, $Param{TitleMaxChars} ) . '...';
    }

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

    return $Data;
}

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

    my $FieldName = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};

    # set the field types
    my $EditValueType   = 'SCALAR';
    my $SearchValueType = 'ARRAY';

    # return the correct structure
    if ( $Param{FieldType} eq 'Edit' ) {
        return {
            $FieldName => $EditValueType,
            }
    }
    elsif ( $Param{FieldType} eq 'Search' ) {
        return {
            'Search_' . $FieldName => $SearchValueType,
            }
    }
    else {
        return {
            $FieldName             => $EditValueType,
            'Search_' . $FieldName => $SearchValueType,
            }
    }
}

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

    return 1;
}

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

    my $Value = int( rand(500) );

    my $Success = $Self->ValueSet(
        %Param,
        Value => $Value,
    );

    if ( !$Success ) {
        return {
            Success => 0,
        };
    }
    return {
        Success => 1,
        Value   => $Value,
    };
}

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

    return 1;
}

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

    my $FieldName = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};

    # return false if not match
    if ( $Param{ObjectAttributes}->{$FieldName} ne $Param{Value} ) {
        return 0;
    }

    return 1;
}

sub PossibleValuesError {
    my ( $Self, $message, $displayerror) = @_;
    my %PossibleValuesError;
    my $ErrorMessage = '';
    $ErrorMessage = 'ERROR: '.$message if $displayerror;
    %PossibleValuesError = ( '-' => $ErrorMessage );
    return \%PossibleValuesError;
}

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

    ## ENABLE FOR DEBUG: ##
    my $DEBUG = 9;
    
    ## set displayerror
    my $DISPLAYERRORS = $Param{DynamicFieldConfig}->{Config}->{DisplayErrors} || 0;

    use Data::Dumper;
    open ERRLOG, '>>/tmp/DF_'.$Param{DynamicFieldConfig}->{Name}.'.log' if $DEBUG;
    print ERRLOG "[AJAXPossibleValuesGet START]---------------------------------\n" if $DEBUG;

    my $query_needed = 0;

    my %PossibleValues;

    my @SQLParameters_values;
    my @SQLParameters_values_refs;

    # split Parameters from DynamicFieldConfig in an array:
    my @SQLParameters_keys = split(',', $Param{DynamicFieldConfig}->{Config}->{Parameters});

    # We create a mapping for the parameters:
    # %SQLParameters_hash = {
    #    Param1 => 'value1',
    #    Param2 => undef,
    # }
    my %SQLParameters_hash;
    for my $key (@SQLParameters_keys) {
        $SQLParameters_hash{$key} = undef;
    }


    #### Distinguish between real ajax request and 'first visualization'
   
=comment

considerations:
- we can distinguish if it is an ajax request or a 'first load' with the presence of $Param{ParamObject}->{Query}->{param}->{ElementChanged}.
- we can distinguish if it is a new Ticket or a Ticket editing with the presence of $Param{ParamObject}->GetParam( Param => 'TicketID')

on ticket editing,
	on 'first load', load take all @SQLParameters_values from the stored ticket information,
	on 'ajax request', take value from http arguments, if not present, take from ticket information.
on ticket creation,
	on 'first load', take values from http arguments,
	on 'ajax request' take values from http arguments,

so we can solve each requirement with following code:
	if ( there is stored ticket information ) fill sqlparameter_values with stored information
	if ( there are http argument values ) fill sqlparameter_values with http argument values overwriting stored information if present
		if ( a http argument was changed that the query depends on ) force_query = 1

	if (!force_query) take from cache if ( cache present )

=cut

    if ( scalar(@SQLParameters_keys) && $Param{ParamObject} && $Param{ParamObject}->GetParam( Param => 'TicketID') ) {
    	my %TicketInfo;
    	# get Ticket from cache:
    	if ($Param{LayoutObject}->{TicketObject}->{'Cache::GetTicket'.$Param{ParamObject}->GetParam( Param => 'TicketID')}{''}{1}) {
    		%TicketInfo = %{$Param{LayoutObject}->{TicketObject}->{'Cache::GetTicket'.$Param{ParamObject}->GetParam( Param => 'TicketID')}{''}{1}};
    	}
    	else {
                my $EncodeObject = Kernel::System::Encode->new(
                    ConfigObject => $Param{ParamObject}->{ConfigObject},
                );
                my $TimeObject = Kernel::System::Time->new(
                    ConfigObject => $Param{ParamObject}->{ConfigObject},
                    LogObject    => $Param{ParamObject}->{LogObject},
                );
                my $DBObject = Kernel::System::DB->new(
                    ConfigObject => $Param{ParamObject}->{ConfigObject},
                    EncodeObject => $EncodeObject,
                    LogObject    => $Param{ParamObject}->{LogObject},
                    MainObject   => $Param{ParamObject}->{MainObject},
                );
                my $TicketObject = Kernel::System::Ticket->new(
                    ConfigObject       => $Param{ParamObject}->{ConfigObject},
                    LogObject          => $Param{ParamObject}->{LogObject},
                    DBObject           => $DBObject,
                    MainObject         => $Param{ParamObject}->{MainObject},
                    TimeObject         => $TimeObject,
                    EncodeObject       => $EncodeObject,
                );
                %TicketInfo = $TicketObject->TicketGet(
                    TicketID      => $Param{ParamObject}->GetParam( Param => 'TicketID'),
                    DynamicFields => 1,         # Optional, default 0. To include the dynamic field values for this ticket on the return structure.
                    UserID        => 0,
                );
    	}

    	for my $key (@SQLParameters_keys) {
    		if ($key eq 'SelectedCustomerUser') {
    			$SQLParameters_hash{$key} = $TicketInfo{CustomerUserID};
    		}
    		else {
    			$SQLParameters_hash{$key} = $TicketInfo{$key};
    		}
    	}
    } 
    
    if ( $Param{ParamObject} && $Param{ParamObject}->{Query}->{param} ) {
    	# REAL AJAX REQUEST, TAKE PARAMS FROM AJAX REQUEST
            # for each parameter extract value from the ParamObject
            for my $key (@SQLParameters_keys) {
                print ERRLOG Dumper($Param{ParamObject}->{Query}->{param}) if $DEBUG;
                if ($Param{ParamObject}->{Query}->{param}->{$key} && $Param{ParamObject}->{Query}->{param}->{$key}[0]) {
                    $SQLParameters_hash{$key} = $Param{ParamObject}->{Query}->{param}->{$key}[0];
                }
                # if the changed Element is in the parameter list, update data
                if ($Param{ParamObject}->{Query}->{param}->{ElementChanged} && $key eq $Param{ParamObject}->{Query}->{param}->{ElementChanged}[0]) {
                    print ERRLOG '$Param{ParamObject}->{Query}->{param}->{ElementChanged}[0]' if $DEBUG;
                    print ERRLOG Dumper(\$Param{ParamObject}->{Query}->{param}->{ElementChanged}[0]) if $DEBUG;
                    $query_needed = 1;
	            my @val = ( '+' );
                    $Param{ParamObject}->{Query}->{param}->{'DynamicField_'.$Param{DynamicFieldConfig}->{Name}} = \@val;
 	        }
                else {
                }
            }
    } else {
        print ERRLOG "ParamObject DOES NOT EXIST!!\n" if $DEBUG;
    }
    

    # finally build the original array with only values:
    for my $key (@SQLParameters_keys) {
        # sometimes we got '<key>||<value>' string, so in that case extract key only:

        if ( ! $SQLParameters_hash{$key} ) {
            # if one parameter is undef, return empty Possible values;
            return $Self->PossibleValuesError('wrong number of parameters, please check settings. ('.$key.' is missing)', $DISPLAYERRORS);
        }

        $SQLParameters_hash{$key} =~ s/^([^|]+)\|\|(.+)$/$1/;

        print ERRLOG $SQLParameters_hash{$key}."\n\n" if $DEBUG;

        push(@SQLParameters_values, $SQLParameters_hash{$key});
        push(@SQLParameters_values_refs, \$SQLParameters_hash{$key});

    }

    if (!(($Param{DynamicFieldConfig}->{Config}->{Query} =~ tr/?//) eq scalar(@SQLParameters_values))) {
        return $Self->PossibleValuesError('wrong number of parameters, please check settings.', $DISPLAYERRORS);
    }

    my $PossibleValues_ref = $Self->{CacheObject}->Get(
        Type	=> 'Hash',
        Key	=> $Param{DynamicFieldConfig}->{Name} . join('', @SQLParameters_values),
    );

    if ($PossibleValues_ref) { # we got the values in cache
        %PossibleValues = %{ $PossibleValues_ref };
    }
    else {
        my $selected_parameter = $Param{ParamObject}->{Query}->{param}->{$Param{DynamicFieldConfig}->{Config}->{Parameters}}[0];
    
        # set none value if defined on field config
        if ( $Param{DynamicFieldConfig}->{Config}->{PossibleNone} ) {
            %PossibleValues = ( '' => '-' );
        }


    	# if no query specified quit:
        if ( !$Param{DynamicFieldConfig}->{Config}->{Query} || $Param{DynamicFieldConfig}->{Config}->{Query} eq '' ) {
            return $Self->PossibleValuesError('no query specified, please check settings.', $DISPLAYERRORS);
        }

        ### SET DEFAULT SETTINGS
        # set a default Separator
        if ( !$Param{DynamicFieldConfig}->{Config}->{Separator} ) {
            $Param{DynamicFieldConfig}->{Config}->{Separator} = ', ';
        }
        ### END SET DEFAULT SETTINGS

        my @row;
        my $sth;
        my $dbh;

        # use local DB Object if no DBI string is specified.
        if ( !$Param{DynamicFieldConfig}->{Config}->{DBIstring} ) {
            $Self->{DBObject}->Prepare(
                SQL => $Param{DynamicFieldConfig}->{Config}->{Query},
                Bind => \@SQLParameters_values_refs,
            );

            #fetch first row
            @row = $Self->{DBObject}->FetchrowArray();

        } else {
            $dbh = DBI->connect($Param{DynamicFieldConfig}->{Config}->{DBIstring}, $Param{DynamicFieldConfig}->{Config}->{DBIuser}, $Param{DynamicFieldConfig}->{Config}->{DBIpass},
                              { PrintError => 0, AutoCommit => 0, HandleError => \&PossibleValuesError }) or return $Self->PossibleValuesError("could not connect to DB: $DBI::errstr", $DISPLAYERRORS);
        
            $dbh->{'mysql_enable_utf8'} = 1;
    
            $sth = $dbh->prepare($Param{DynamicFieldConfig}->{Config}->{Query});
            $sth->execute( @SQLParameters_values );
    
            # fetch first row
            @row = $sth->fetchrow_array;        
        }

        print ERRLOG ":::Extracted from DB:::\n" if $DEBUG;
        print ERRLOG Dumper(@row) if $DEBUG;
        print ERRLOG "::::::::::::::::::::::::::\n" if $DEBUG;
        close ERRLOG if $DEBUG;

        # cicle fetched rows from DB
        while (@row) {
            my $line = '';
            my $firstRow = 0;
            for my $col (@row) {
                if (!utf8::is_utf8($col)) {
                    utf8::decode( $col );
                }
                if (!$firstRow) { # skip first row 
                    $firstRow = 1;
            	next;
                }
                $line .= $col.$Param{DynamicFieldConfig}->{Config}->{Separator};
            }

            $line = substr($line, 0, -1 * length($Param{DynamicFieldConfig}->{Config}->{Separator}));
            if ($Param{DynamicFieldConfig}->{Config}->{StoreValue} && $Param{DynamicFieldConfig}->{Config}->{StoreValue} eq "1") {
                %PossibleValues = ( %PossibleValues, $row[0]."||".$line => $line);
            }
            else {
                %PossibleValues = ( %PossibleValues, $row[0] => $line );
            }

            # fetch new row depending on DB connection type
            if ( !$Param{DynamicFieldConfig}->{Config}->{DBIstring} ) {
                @row = $Self->{DBObject}->FetchrowArray();
            }
            else {
                @row = $sth->fetchrow_array;
		if (!@row) {
                    $dbh->disconnect;
                }
            }
        }

        # put all in the cache:
	
        $Self->{CacheObject}->Set(
            Type	=> 'Hash',
            Key		=> $Param{DynamicFieldConfig}->{Name} . join('', @SQLParameters_values),
            Value	=> \%PossibleValues,
            TTL		=> $Param{DynamicFieldConfig}->{CacheTTL} || 360,
        );
    }

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = \%PossibleValues;
    # retrun the possible values hash as a reference
    return \%PossibleValues;
}

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

    # get historical values from database
    my $HistoricalValues = $Self->{DynamicFieldValueObject}->HistoricalValueGet(
        FieldID   => $Param{DynamicFieldConfig}->{ID},
        ValueType => 'Text',
    );

    # BINI - 27.03.2013
    #
    # Remove "<val>||" from historical values used in search form
    #
    if ($Param{DynamicFieldConfig}->{Config}->{StoreValue} eq "1") {
      if ( IsHashRefWithData($HistoricalValues) ) {
        for my $Key ( sort keys %{$HistoricalValues} ) {
            $HistoricalValues->{$Key} =~ s/^([^|]+)\|\|(.+)$/$2/;
        }
      }
    }
    # BINI
    #

    # retrun the historical values from database
    return $HistoricalValues;
}

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

    my $Value = defined $Param{Key} ? $Param{Key} : '';

    # get real values
    my $PossibleValues = $Param{DynamicFieldConfig}->{Config}->{PossibleValues};

    if ($Value) {

        # check if there is a real value for this key (otherwise keep the key)
        if ( $Param{DynamicFieldConfig}->{Config}->{PossibleValues}->{$Value} ) {

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

            # check if translation is possible
            if (
                defined $Param{LanguageObject}
                && $Param{DynamicFieldConfig}->{Config}->{TranslatableValues}
                )
            {

                # translate value
                $Value = $Param{LanguageObject}->Get($Value);
            }
        }
    }

    return $Value;
}

1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<http://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (AGPL). If you
did not receive this file, see L<http://www.gnu.org/licenses/agpl.txt>.

=cut

=head1 VERSION

$$

=cut
</File>
<File Location="Custom/Kernel/System/DynamicField/Backend/TextAreaFromDB.pm" Permission="644" Encode="Base64"># --
# Kernel/System/DynamicField/Backend/TextArea.pm - Delegate for DynamicField TextArea backend
# Copyright (C) 2001-2012 OTRS AG, http://otrs.org/
# --
# $Id: TextArea.pm,v 1.48.2.4 2012/05/07 21:43:10 cr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::DynamicField::Backend::TextAreaFromDB;

use strict;
use warnings;

use Kernel::System::VariableCheck qw(:all);
use Kernel::System::DynamicFieldValue;
use Kernel::System::DynamicField::Backend::BackendCommon;

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

=head1 NAME

Kernel::System::DynamicField::Backend::TextArea

=head1 SYNOPSIS

DynamicFields TextArea backend delegate

=head1 PUBLIC INTERFACE

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

=over 4

=item new()

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

=cut

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

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

    # get needed objects
    for my $Needed (qw(ConfigObject EncodeObject LogObject MainObject DBObject)) {
        die "Got no $Needed!" if !$Param{$Needed};

        $Self->{$Needed} = $Param{$Needed};
    }

    # create additional objects
    $Self->{DynamicFieldValueObject} = Kernel::System::DynamicFieldValue->new( %{$Self} );
    $Self->{BackendCommonObject}
        = Kernel::System::DynamicField::Backend::BackendCommon->new( %{$Self} );

    $Self->{CacheObject} = Kernel::System::Cache->new(%Param);

    # set the maximum lenght for the textarea fields to still be a searchable field in some
    # databases
    $Self->{MaxLength} = 3800;

    return $Self;
}

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

    my $DFValue = $Self->{DynamicFieldValueObject}->ValueGet(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
    );

    return if !$DFValue;
    return if !IsArrayRefWithData($DFValue);
    return if !IsHashRefWithData( $DFValue->[0] );

    return $DFValue->[0]->{ValueText};
}

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

    my $Success = $Self->{DynamicFieldValueObject}->ValueSet(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
        Value    => [
            {
                ValueText => $Param{Value},
            },
        ],
        UserID => $Param{UserID},
    );

    return $Success;
}

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

    my $Success = $Self->{DynamicFieldValueObject}->ValueDelete(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
        UserID   => $Param{UserID},
    );

    return $Success;
}

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

    my $Success = $Self->{DynamicFieldValueObject}->AllValuesDelete(
        FieldID => $Param{DynamicFieldConfig}->{ID},
        UserID  => $Param{UserID},
    );

    return $Success;
}

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

    my $Success = $Self->{DynamicFieldValueObject}->ValueValidate(
        Value => {
            ValueText => $Param{Value},
        },
        UserID => $Param{UserID}
    );

    return $Success;
}

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

    my %Operators = (
        Equals            => '=',
        GreaterThan       => '>',
        GreaterThanEquals => '>=',
        SmallerThan       => '<',
        SmallerThanEquals => '<=',
    );

    if ( $Operators{ $Param{Operator} } ) {
        my $SQL = " $Param{TableAlias}.value_text $Operators{$Param{Operator}} '";
        $SQL .= $Self->{DBObject}->Quote( $Param{SearchTerm} ) . "' "; 
        return $SQL;
    }

    if ( $Param{Operator} eq 'Like' ) {

        my $SQL = $Self->{DBObject}->QueryCondition(
            Key   => "$Param{TableAlias}.value_text",
            Value => $Param{SearchTerm},
        );

        return $SQL;
    }

    $Self->{'LogObject'}->Log(
        'Priority' => 'error',
        'Message'  => "Unsupported Operator $Param{Operator}",
    );

    return;
}

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

    return "$Param{TableAlias}.value_text";
}

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

    # take config from field config
    my $FieldConfig = $Param{DynamicFieldConfig}->{Config};
    my $FieldName   = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};
    my $FieldLabel  = $Param{DynamicFieldConfig}->{Label};

    my $Value = '';

    # set the field value or default
    if ( $Param{UseDefaultValue} ) {
        $Value = ( defined $FieldConfig->{DefaultValue} ? $FieldConfig->{DefaultValue} : '' );
    }
    $Value = $Param{Value} if defined $Param{Value};

    # extract the dynamic field value form the web request
    my $FieldValue = $Self->EditFieldValueGet(
        %Param,
    );

    # set values from ParamObject if present
    if ( defined $FieldValue ) {
        $Value = $FieldValue;
    }

    # set the rows number
    my $RowsNumber
        = defined $FieldConfig->{Rows} && $FieldConfig->{Rows} ? $FieldConfig->{Rows} : '7';

    # set the cols number
    my $ColsNumber
        = defined $FieldConfig->{Cols} && $FieldConfig->{Cols} ? $FieldConfig->{Cols} : '42';

    # check and set class if necessary
    my $FieldClass = 'DynamicFieldTextArea';
    if ( defined $Param{Class} && $Param{Class} ne '' ) {
        $FieldClass .= ' ' . $Param{Class};
    }

    # set field as mandatory
    $FieldClass .= ' Validate_Required' if $Param{Mandatory};

    # set error css class
    $FieldClass .= ' ServerError' if $Param{ServerError};

    # set validation class for maximum characters
    $FieldClass .= ' Validate_MaxLength';

    # create field HTML
    # the XHTML definition does not support maxlenght attribute for a textarea field, therefore
    # is nedded to be set by JS code (otherwise wc3 validator will complaint about it)
    # notice that some browsers count new lines \n\r as only 1 character in this cases the
    # validation framework might rise an error while the user is still capable to enter text in the
    # textarea, otherwise the maxlenght property will prevent to enter more text than the maximum
    my $HTMLString = <<"EOF";
<textarea class="$FieldClass" id="$FieldName" name="$FieldName" title="$FieldLabel" rows="$RowsNumber" cols="$ColsNumber" >$Value</textarea>
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
  \$('#$FieldName').attr('maxlength','$Self->{MaxLength}');
//]]></script>
<!--dtl:js_on_document_complete-->
EOF

    # for client side validation
    my $DivID = $FieldName . 'Error';

    if ( $Param{Mandatory} ) {
        $HTMLString .= <<"EOF";
    <div id="$DivID" class="TooltipErrorMessage">
        <p>
            \$Text{"This field is required or The field content is too long! Maximum size is $Self->{MaxLength} characters."}
        </p>
    </div>
EOF
    }
    else {
        $HTMLString .= <<"EOF";
    <div id="$DivID" class="TooltipErrorMessage">
        <p>
            \$Text{"The field content is too long! Maximum size is $Self->{MaxLength} characters."}
        </p>
    </div>
EOF
    }

    if ($FieldConfig->{DisplayErrors}) {
    	my $DivID = $FieldName . 'Warning';
    	
    	# for client side validation
        $HTMLString .= <<"EOF";

    <div id="$DivID" class="TooltipErrorMessage">
        <p>
            \$Text{"[Debug mode]"}
        </p>
    </div>
EOF
    }

    if ( $Param{ServerError} ) {

        my $ErrorMessage = $Param{ErrorMessage} || 'This field is required.';
        my $DivID = $FieldName . 'ServerError';

        # for server side validation
        $HTMLString .= <<"EOF";
    <div id="$DivID" class="TooltipErrorMessage">
        <p>
            \$Text{"$ErrorMessage"}
        </p>
    </div>
EOF
    }

    # call EditLabelRender on the common backend
    my $LabelString = $Self->{BackendCommonObject}->EditLabelRender(
        DynamicFieldConfig => $Param{DynamicFieldConfig},
        Mandatory          => $Param{Mandatory} || '0',
        FieldName          => $FieldName,
    );

    my $Data = {
        Field => $HTMLString,
        Label => $LabelString,
    };

    return $Data;
}

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

    my $FieldName = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};

    my $Value;

#    # check if there is a Template and retreive the dinalic field value from there
#    if ( IsHashRefWithData( $Param{Template} ) ) {
#        $Value = $Param{Template}->{$FieldName};
#    }
#
#    # otherwise get dynamic field value form param
#    else {
#        $Value = $Param{ParamObject}->GetParam( Param => $FieldName );
#    }

#    if ( defined $Param{ReturnTemplateStructure} && $Param{ReturnTemplateStructure} eq '1' ) {
#        return {
#            $FieldName => $Value,
#        };
#    }

    # USE FOR DEBUGGING PURPOSES
my $DEBUG = 0;
    #
 use Data::Dumper;
open ERRLOG, '>>/tmp/TA_'.$Param{DynamicFieldConfig}->{Name}.'.log' if $DEBUG;
#print ERRLOG Dumper($Param{ParamObject}->{Query}->{param}) if $DEBUG;

print ERRLOG $Param{ParamObject} if $DEBUG;
#close ERRLOG;

    my $query_needed = 0;

    my @SQLParameters_values;
    my @SQLParameters_values_refs;

    my @SQLParameters_keys = split(',', $Param{DynamicFieldConfig}->{Config}->{Parameters});

    my %SQLParameters_hash;
    for my $key (@SQLParameters_keys) {
        $SQLParameters_hash{$key} = undef;
    }

    if ( scalar(@SQLParameters_keys) && $Param{ParamObject} && defined $Param{ParamObject} && $Param{ParamObject}->GetParam( Param => 'TicketID') ) {
    	my %TicketInfo;
    	# get Ticket from cache:
    	if ($Param{LayoutObject}->{TicketObject}->{'Cache::GetTicket'.$Param{ParamObject}->GetParam( Param => 'TicketID')}{''}{1}) {
    		%TicketInfo = %{$Param{LayoutObject}->{TicketObject}->{'Cache::GetTicket'.$Param{ParamObject}->GetParam( Param => 'TicketID')}{''}{1}};
    	}
    	else {
                my $EncodeObject = Kernel::System::Encode->new(
                    ConfigObject => $Param{ParamObject}->{ConfigObject},
                );
                my $TimeObject = Kernel::System::Time->new(
                    ConfigObject => $Param{ParamObject}->{ConfigObject},
                    LogObject    => $Param{ParamObject}->{LogObject},
                );
                my $DBObject = Kernel::System::DB->new(
                    ConfigObject => $Param{ParamObject}->{ConfigObject},
                    EncodeObject => $EncodeObject,
                    LogObject    => $Param{ParamObject}->{LogObject},
                    MainObject   => $Param{ParamObject}->{MainObject},
                );
                my $TicketObject = Kernel::System::Ticket->new(
                    ConfigObject       => $Param{ParamObject}->{ConfigObject},
                    LogObject          => $Param{ParamObject}->{LogObject},
                    DBObject           => $DBObject,
                    MainObject         => $Param{ParamObject}->{MainObject},
                    TimeObject         => $TimeObject,
                    EncodeObject       => $EncodeObject,
                );
                %TicketInfo = $TicketObject->TicketGet(
                    TicketID      => $Param{ParamObject}->GetParam( Param => 'TicketID'),
                    DynamicFields => 1,         # Optional, default 0. To include the dynamic field values for this ticket on the return structure.
                    UserID        => 0,
                );
    	}

    	for my $key (@SQLParameters_keys) {
    		if ($key eq 'SelectedCustomerUser') {
    			$SQLParameters_hash{$key} = $TicketInfo{CustomerUserID};
    		}
    		else {
    			$SQLParameters_hash{$key} = $TicketInfo{$key};
    		}
    	}
    	print ERRLOG "[Got Ticket info:]\n" if $DEBUG;
    	print ERRLOG Dumper(\%TicketInfo) if $DEBUG;
    } 
 


    for my $key (@SQLParameters_keys) {
        # WP - BINI
        # Fix per estrarre i valori puliti dai parametri (es. 2||DATO)
        # 


	if ( $Param{ParamObject}->{Query}->{param}->{$key}[0] ) {
        	$Param{ParamObject}->{Query}->{param}->{$key}[0] =~ s/^([^|]+)\|\|(.+)$/$1/;
        	push(@SQLParameters_values, $Param{ParamObject}->{Query}->{param}->{$key}[0]);
        	push(@SQLParameters_values_refs, \$Param{ParamObject}->{Query}->{param}->{$key}[0]);
	} elsif ( $SQLParameters_hash{$key} ) {
		push(@SQLParameters_values,  $SQLParameters_hash{$key});
		push(@SQLParameters_values_refs, \$SQLParameters_hash{$key});
	}


        
        # WP - BINI - ex: push(@SQLParameters_values, $Param{ParamObject}->{Query}->{param}->{$key}[0]);
        
        # if the changed Element is in the parameter list, update data
        if ($Param{ParamObject}->{Query}->{param}->{ElementChanged} and $key eq $Param{ParamObject}->{Query}->{param}->{ElementChanged}[0]) {
                $query_needed = 1;
#                print ERRLOG $key." eq ".$Param{ParamObject}->{Query}->{param}->{ElementChanged}[0]."\n";
        }
        else {
#                print ERRLOG $key." neq ".$Param{ParamObject}->{Query}->{param}->{ElementChanged}[0]."\n";
        }
    }

    

#    print ERRLOG Dumper($Param{ParamObject});
#    print ERRLOG UNIVERSAL::isa($Param{ParamObject}, 'HASH')." FieldName: $FieldName\n";
#    close ERRLOG;

    print ERRLOG "Building value\n" if $DEBUG;

    if ($Param{ParamObject}->{Query}->{param}->{$FieldName} && $Param{ParamObject}->{Query}->{param}->{$FieldName}[0]) {
        $Value = $Param{ParamObject}->{Query}->{param}->{$FieldName}[0];
    }

#    if ($query_needed and 1) {


        $Value = $Self->{CacheObject}->Get(
            Type    => 'String',
            Key     => scalar @SQLParameters_values > 0 ? $Param{DynamicFieldConfig}->{Name} . join('', @SQLParameters_values) : $Param{DynamicFieldConfig}->{Name},
        );

        if (!$Value) {

	    print ERRLOG "Doing query\n" if $DEBUG;

	    my @row;
            my $sth;
            my $dbh;


	    my $completeline;

            # use local DB Object if no DBI string is specified.
            if ( !$Param{DynamicFieldConfig}->{Config}->{DBIstring} ) {
                $Self->{DBObject}->Prepare(
                    SQL => $Param{DynamicFieldConfig}->{Config}->{Query},
                    Bind => \@SQLParameters_values_refs,
                );
    
                #fetch first row
                @row = $Self->{DBObject}->FetchrowArray();
    
            } else {
                $dbh = DBI->connect($Param{DynamicFieldConfig}->{Config}->{DBIstring}, $Param{DynamicFieldConfig}->{Config}->{DBIuser}, $Param{DynamicFieldConfig}->{Config}->{DBIpass},
                                  { PrintError => 0, AutoCommit => 0 }) or die;
            
                $dbh->{'mysql_enable_utf8'} = 1;
        
                $sth = $dbh->prepare($Param{DynamicFieldConfig}->{Config}->{Query});
                $sth->execute( @SQLParameters_values );
        
                # fetch first row
                @row = $sth->fetchrow_array;        
            }
    
            print ERRLOG ":::Extracted from DB:::\n" if $DEBUG;
            print ERRLOG Dumper(@row) if $DEBUG;
            print ERRLOG "::::::::::::::::::::::::::\n" if $DEBUG;
            close ERRLOG if $DEBUG;
    
            # cicle fetched rows from DB
            my $line = '';
            while (@row) {

		my $counter = 0;
                my @names = $sth->{NAME};

                for my $col (@row) {
                    if (!utf8::is_utf8($col)) {
                        utf8::decode( $col );
                    }
#                    if (!$firstRow) { # skip first row 
#                        $firstRow = 1;
#                	next;
#                    }

		    if ( !$Param{DynamicFieldConfig}->{Config}->{DBIstring} ) {
                        $line .= $col."\n";
                    } else {
                    	$line .= $names[0][$counter].": ".$col."\n";
                    }
                    $counter += 1;
                }
    
#                if ($Param{DynamicFieldConfig}->{Config}->{StoreValue} && $Param{DynamicFieldConfig}->{Config}->{StoreValue} eq "1") {
#                    %PossibleValues = ( %PossibleValues, $row[0]."||".$line => $line);
#                }
#                else {
#                    %PossibleValues = ( %PossibleValues, $row[0] => $line );
#                }

#	 	$completeline .= $row[0] . ": ". $line;
   
                # fetch new row depending on DB connection type
                if ( !$Param{DynamicFieldConfig}->{Config}->{DBIstring} ) {
                    @row = $Self->{DBObject}->FetchrowArray();
                }
                else {
                    @row = $sth->fetchrow_array;
    		if (!@row) {
                        $dbh->disconnect;
                    }
                }
            }
    
	    $Value = $line;    
    





























#            my @row;
#	    my $sth;
#	    my $dbh;
#
#            my $line = '';
#
#	    if ( !$Param{DynamicFieldConfig}->{Config}->{DBIstring} ) {
#                $Self->{DBObject}->Prepare(
#                	SQL => $Param{DynamicFieldConfig}->{Config}->{Query},
#	                Bind => \@SQLParameters_values_refs,
#            	);
#
#                #fetch first row
#	        @row = $Self->{DBObject}->FetchrowArray();
#
#                while ( @row = $Self->{DBObject}->FetchrowArray()) {
#    
#                    my @names = $sth->{NAME};
#                    for my $col (@row) {
#                        $line .= $names[0][$counter].": ".$col."\n";
#                        $counter += 1;
#                    }
#                }
#
#            }
#            else {
#                $dbh = DBI->connect($Param{DynamicFieldConfig}->{Config}->{DBIstring}, $Param{DynamicFieldConfig}->{Config}->{DBIuser}, $Param{DynamicFieldConfig}->{Config}->{DBIpass},
#                    { RaiseError => 1, AutoCommit => 0 });
#                $sth = $dbh->prepare($Param{DynamicFieldConfig}->{Config}->{Query});
#                $sth->execute( @SQLParameters_values );
#    #    STAMPA LA QUERY
#    #    $Self->{'LogObject'}->Log(
#    #        'Priority' => 'notice',
#    #        'Message'  => "$Param{DynamicFieldConfig}->{Config}->{Query}",
#    #    );
#    #	    my $found = $sth->fetch();
#    #	    print ERRLOG "found $found rows\n" if $DEBUG;
#    
#                my $counter = 0;
#    
#
#    
#                while ( @row = $sth->fetchrow_array ) {
#    
#                    my @names = $sth->{NAME};
#                    for my $col (@row) {
#                        $line .= $names[0][$counter].": ".$col."\n";
#                        $counter += 1;
#                    }
#                }
#	    }
    
#            $Value = $line;
	
            $Self->{CacheObject}->Set(
                Type        => 'String',
            	Key	    => scalar @SQLParameters_values > 0 ? $Param{DynamicFieldConfig}->{Name} . join('', @SQLParameters_values) : $Param{DynamicFieldConfig}->{Name},
                Value       => $Value,
                TTL         => 360,
            );
        }
	else {
		print ERRLOG "Query not needed, using cache\n" if $DEBUG;
	}
#    }
    close ERRLOG if $DEBUG;

    return $Value;
}

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

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

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

    my $ServerError;
    my $ErrorMessage;

    # perform necessary validations
    if ( $Param{Mandatory} && $Value eq '' ) {
        $ServerError = 1;
    }

    if ( length $Value > $Self->{MaxLength} ) {
        $ServerError = 1;
        $ErrorMessage
            = "The field content is too long! Maximum size is $Self->{MaxLength} characters.";
    }

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

    return $Result;
}

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

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

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

    # HTMLOuput transformations
    if ( $Param{HTMLOutput} ) {

        $Value = $Param{LayoutObject}->Ascii2Html(
            Text           => $Value,
            HTMLResultMode => 1,
            Max            => $Param{ValueMaxChars} || '',
        );

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

    # this field type does not support the Link Feature
    my $Link;

    # create return structure
    my $Data = {
        Value => $Value,
        Title => $Title,
        Link  => $Link,
    };

    return $Data;
}

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

    return 0;
}

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

    # take config from field config
    my $FieldConfig = $Param{DynamicFieldConfig}->{Config};
    my $FieldName   = 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name};
    my $FieldLabel  = $Param{DynamicFieldConfig}->{Label};

    # set the field value
    my $Value = ( defined $Param{DefaultValue} ? $Param{DefaultValue} : '' );

    # get the field value, this fuction is always called after the profile is loaded
    my $FieldValue = $Self->SearchFieldValueGet(%Param);

    # set values from profile if present
    if ( defined $FieldValue ) {
        $Value = $FieldValue;
    }

    # check and set class if necessary
    my $FieldClass = 'DynamicFieldText';

    my $HTMLString = <<"EOF";
<input type="text" class="$FieldClass" id="$FieldName" name="$FieldName" title="$FieldLabel" value="$Value" />
EOF

    # call EditLabelRender on the common backend
    my $LabelString = $Self->{BackendCommonObject}->EditLabelRender(
        DynamicFieldConfig => $Param{DynamicFieldConfig},
        FieldName          => $FieldName,
    );

    my $Data = {
        Field => $HTMLString,
        Label => $LabelString,
    };

    return $Data;
}

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

    my $Value;

    # get dynamic field value form param object
    if ( defined $Param{ParamObject} ) {
        $Value = $Param{ParamObject}
            ->GetParam( Param => 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name} );
    }

    # otherwise get the value from the profile
    elsif ( defined $Param{Profile} ) {
        $Value = $Param{Profile}->{ 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name} };
    }
    else {
        return;
    }

    if ( defined $Param{ReturnProfileStructure} && $Param{ReturnProfileStructure} eq 1 ) {
        return {
            'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name} => $Value,
        };
    }

    return $Value;

}

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

    # get field value
    my $Value = $Self->SearchFieldValueGet(%Param);

    if ( !$Value ) {
        return {
            Parameter => {
                'Like' => '',
            },
            Display => '',
            }
    }

    # return search parameter structure
    return {
        Parameter => {
            'Like' => '*' . $Value . '*',
        },
        Display => $Value,
    };
}

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

    return {
        Name    => $Param{DynamicFieldConfig}->{Label},
        Element => 'DynamicField_' . $Param{DynamicFieldConfig}->{Name},
    };
}

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

    my $Operator = 'Equals';
    my $Value    = $Param{Value};

    return {
        $Operator => $Value,
    };
}

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

    my $Value = defined $Param{Value} ? $Param{Value} : '';
    my $Title = $Value;

    # cut strings if needed
    if ( $Param{ValueMaxChars} && length($Value) > $Param{ValueMaxChars} ) {
        $Value = substr( $Value, 0, $Param{ValueMaxChars} ) . '...';
    }
    if ( $Param{TitleMaxChars} && length($Title) > $Param{TitleMaxChars} ) {
        $Title = substr( $Title, 0, $Param{TitleMaxChars} ) . '...';
    }

    # create return structure
    my $Data = {
        Value => $Value,
        Title => $Title,
    };

    return $Data;
}

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

    my $FieldName = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};

    # set the field types
    my $EditValueType   = 'SCALAR';
    my $SearchValueType = 'SCALAR';

    # return the correct structure
    if ( $Param{FieldType} eq 'Edit' ) {
        return {
            $FieldName => $EditValueType,
            }
    }
    elsif ( $Param{FieldType} eq 'Search' ) {
        return {
            'Search_' . $FieldName => $SearchValueType,
            }
    }
    else {
        return {
            $FieldName             => $EditValueType,
            'Search_' . $FieldName => $SearchValueType,
            }
    }
}

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

    return 0;
}

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

    my $Value = int( rand(500) );

    my $Success = $Self->ValueSet(
        %Param,
        Value => $Value,
    );

    if ( !$Success ) {
        return {
            Success => 0,
        };
    }
    return {
        Success => 1,
        Value   => $Value,
    };
}

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

    return 1;
}

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

    my $FieldName = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};

    # return false if not match
    if ( $Param{ObjectAttributes}->{$FieldName} ne $Param{Value} ) {
        return 0;
    }

    return 1;
}

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

    # not supported
    return;
}

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

    # get historical values from database
    my $HistoricalValues = $Self->{DynamicFieldValueObject}->HistoricalValueGet(
        FieldID   => $Param{DynamicFieldConfig}->{ID},
        ValueType => 'Text',
    );

    # retrun the historical values from database
    return $HistoricalValues;
}


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

    my $Value = defined $Param{Key} ? $Param{Key} : '';

    # get real values
    my $PossibleValues = $Param{DynamicFieldConfig}->{Config}->{PossibleValues};

    if ($Value) {

        # check if there is a real value for this key (otherwise keep the key)
        if ( $Param{DynamicFieldConfig}->{Config}->{PossibleValues}->{$Value} ) {

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

            # check if translation is possible
            if (
                defined $Param{LanguageObject}
                && $Param{DynamicFieldConfig}->{Config}->{TranslatableValues}
                )
            {

                # translate value
                $Value = $Param{LanguageObject}->Get($Value);
            }
        }
    }

    return $Value;
}


1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<http://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (AGPL). If you
did not receive this file, see L<http://www.gnu.org/licenses/agpl.txt>.

=cut

=head1 VERSION

$$

=cut
</File>
<File Location="Custom/Kernel/System/DynamicField/Backend/MultiselectFromDB.pm" Permission="644" Encode="Base64"># --
# Kernel/System/DynamicField/Backend/Multiselect.pm - Delegate for DynamicField Multiselect backend
# Copyright (C) 2001-2012 OTRS AG, http://otrs.org/
# --
# $Id: Multiselect.pm,v 1.51.2.1 2012/05/28 22:43:40 cr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::DynamicField::Backend::MultiselectFromDB;

use strict;
use warnings;

use Kernel::System::VariableCheck qw(:all);
use Kernel::System::DynamicFieldValue;
use Kernel::System::DynamicField::Backend::BackendCommon;
use Kernel::System::DynamicField::Backend::DropdownFromDB;

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

=head1 NAME

Kernel::System::DynamicField::Backend::Multiselect

=head1 SYNOPSIS

DynamicFields Multiselect backend delegate

=head1 PUBLIC INTERFACE

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

=over 4

=item new()

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

=cut

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

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

    # get needed objects
    for my $Needed (qw(ConfigObject EncodeObject LogObject MainObject DBObject)) {
        die "Got no $Needed!" if !$Param{$Needed};

        $Self->{$Needed} = $Param{$Needed};
    }

    # create additional objects
    $Self->{DynamicFieldValueObject} = Kernel::System::DynamicFieldValue->new( %{$Self} );
    $Self->{BackendCommonObject}
        = Kernel::System::DynamicField::Backend::BackendCommon->new( %{$Self} );
    $Self->{BackendDropdownFromDB}
        = Kernel::System::DynamicField::Backend::DropdownFromDB->new( %{$Self}, %Param );


    return $Self;
}

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

    my $DFValue = $Self->{DynamicFieldValueObject}->ValueGet(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
    );

    return if !$DFValue;
    return if !IsArrayRefWithData($DFValue);
    return if !IsHashRefWithData( $DFValue->[0] );

    # extract real values
    my @ReturnData;
    for my $Item ( @{$DFValue} ) {
        push @ReturnData, $Item->{ValueText}
    }

    return \@ReturnData;
}

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

    # check for valid possible values list
#    if ( !$Param{DynamicFieldConfig}->{Config}->{PossibleValues} ) {
#        $Self->{LogObject}->Log(
#            Priority => 'error',
#            Message  => "Need PossibleValues in DynamicFieldConfig!",
#        );
#        return;
#    }

    # check value
    my @Values;
    if ( ref $Param{Value} eq 'ARRAY' ) {
        @Values = @{ $Param{Value} };
    }
    else {
        @Values = ( $Param{Value} );
    }

    my @ValueText;
    for my $Item (@Values) {
        push @ValueText, { ValueText => $Item };
    }

    my $Success = $Self->{DynamicFieldValueObject}->ValueSet(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
        Value    => \@ValueText,
        UserID   => $Param{UserID},
    );

    return $Success;
}

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

    my $Success = $Self->{DynamicFieldValueObject}->ValueDelete(
        FieldID  => $Param{DynamicFieldConfig}->{ID},
        ObjectID => $Param{ObjectID},
        UserID   => $Param{UserID},
    );

    return $Success;
}

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

    my $Success = $Self->{DynamicFieldValueObject}->AllValuesDelete(
        FieldID => $Param{DynamicFieldConfig}->{ID},
        UserID  => $Param{UserID},
    );

    return $Success;
}

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

#    # check value
#    my @Values;
#    if ( IsArrayhRefWithData( $Param{Value} ) ) {
#        @Values = @{ $Param{Value} };
#    }
#    else {
#        @Values = ( $Param{Value} );
#    }
#
#    my $Success;
#    for my $Item (@Values) {
#
#        $Success = $Self->{DynamicFieldValueObject}->ValueValidate(
#            Value => {
#                ValueText => $Item,
#            },
#            UserID => $Param{UserID}
#        );
#        return if !$Success
#    }
#    return $Success;
    return 1;
}

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

    my %Operators = (
        Equals            => '=',
        GreaterThan       => '>',
        GreaterThanEquals => '>=',
        SmallerThan       => '<',
        SmallerThanEquals => '<=',
    );

    if ( $Operators{ $Param{Operator} } ) {
        my $SQL = " $Param{TableAlias}.value_text $Operators{$Param{Operator}} '";
        $SQL .= $Self->{DBObject}->Quote( $Param{SearchTerm} ) . "' ";
        return $SQL;
    }

    if ( $Param{Operator} eq 'Like' ) {

        my $SQL = $Self->{DBObject}->QueryCondition(
            Key   => "$Param{TableAlias}.value_text",
            Value => $Param{SearchTerm},
        );

        return $SQL;
    }

    $Self->{'LogObject'}->Log(
        'Priority' => 'error',
        'Message'  => "Unsupported Operator $Param{Operator}",
    );

    return;
}

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

    return "$Param{TableAlias}.value_text";
}

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

    # take config from field config
    my $FieldConfig = $Param{DynamicFieldConfig}->{Config};
    my $FieldName   = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};
    my $FieldLabel  = $Param{DynamicFieldConfig}->{Label};

    my $Value = '';

    # set the field value or default
    if ( $Param{UseDefaultValue} ) {
        $Value = ( defined $FieldConfig->{DefaultValue} ? $FieldConfig->{DefaultValue} : '' );
    }
    $Value = $Param{Value} if defined $Param{Value};

    #d extract the dynamic field value form the web request
    my $FieldValue = $Self->EditFieldValueGet(
        %Param,
    );

    # set values from ParamObject if present
    if ( IsArrayRefWithData($FieldValue) ) {
        $Value = $FieldValue;
    }

    # check and set class if necessary
    my $FieldClass = 'DynamicFieldText';
    if ( defined $Param{Class} && $Param{Class} ne '' ) {
        $FieldClass .= ' ' . $Param{Class};
    }

    # set field as mandatory
    $FieldClass .= ' Validate_Required' if $Param{Mandatory};

    # set error css class
    $FieldClass .= ' ServerError' if $Param{ServerError};

    # set PossibleValues
    my $SelectionData = $FieldConfig->{PossibleValues};

    # use PossibleValuesFilter if defined
    $SelectionData = $Param{PossibleValuesFilter}
        if defined $Param{PossibleValuesFilter};

    # check value
    my @Values;
    if ( ref $Value eq 'ARRAY' ) {
        @Values = @{$Value};
    }
    else {
        @Values = ($Value);
    }

    # set PossibleNone attribute
    my $FieldPossibleNone;
    if ( defined $Param{OverridePossibleNone} ) {
        $FieldPossibleNone = $Param{OverridePossibleNone};
    }
    else {
        $FieldPossibleNone = $FieldConfig->{PossibleNone} || 0;
    }

    my $HTMLString = $Param{LayoutObject}->BuildSelection(
        Data => $SelectionData || {},
        Name => $FieldName,
        SelectedID   => \@Values,
        Translation  => $FieldConfig->{TranslatableValues} || 0,
        PossibleNone => $FieldPossibleNone,
        Class        => $FieldClass,
        HTMLQuote    => 1,
        Multiple     => 1,
    );
    
    if ($FieldConfig->{DisplayErrors}) {
    	my $DivID = $FieldName . 'Warning';
    	
    	# for client side validation
        $HTMLString .= <<"EOF";

    <div id="$DivID" class="TooltipErrorMessage">
        <p>
            \$Text{"[Debug mode]"}
        </p>
    </div>
EOF
    }

    if ( $Param{Mandatory} ) {
        my $DivID = $FieldName . 'Error';

        # for client side validation
        $HTMLString .= <<"EOF";

    <div id="$DivID" class="TooltipErrorMessage">
        <p>
            \$Text{"This field is required."}
        </p>
    </div>
EOF
    }

    if ( $Param{ServerError} ) {

        my $ErrorMessage = $Param{ErrorMessage} || 'This field is required.';
        my $DivID = $FieldName . 'ServerError';

        # for server side validation
        $HTMLString .= <<"EOF";
    <div id="$DivID" class="TooltipErrorMessage">
        <p>
            \$Text{"$ErrorMessage"}
        </p>
    </div>
EOF
    }

    if ( $Param{AJAXUpdate} ) {

        my $FieldSelector = '#' . $FieldName;

        my $FieldsToUpdate;
        if ( IsArrayRefWithData( $Param{UpdatableFields} ) ) {
            my $FirstItem = 1;
            FIELD:
            for my $Field ( @{ $Param{UpdatableFields} } ) {
                next FIELD if $Field eq $FieldName;
                if ($FirstItem) {
                    $FirstItem = 0;
                }
                else {
                    $FieldsToUpdate .= ', ';
                }
                $FieldsToUpdate .= "'" . $Field . "'";
            }
        }

        #add js to call FormUpdate()
        $HTMLString .= <<"EOF";
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
    \$('$FieldSelector').bind('change', function (Event) {
        Core.AJAX.FormUpdate(\$(this).parents('form'), 'AJAXUpdate', '$FieldName', [ $FieldsToUpdate ]);
    });
//]]></script>
<!--dtl:js_on_document_complete-->
EOF
    }

    if ( $Param{SubmitOnChange} ) {

        my $FieldSelector = '#' . $FieldName;

        #add js to disable validation and do submit()
        $HTMLString .= <<"EOF";
<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[
    \$('$FieldSelector').bind('change', function (Event) {
        // make sure the ticket is not yet created on queue change
        \$('input#Expand').val(1);
        Core.Form.Validate.DisableValidation(\$(this).closest('form'));
        \$(this).closest('form').submit();
    });
//]]></script>
<!--dtl:js_on_document_complete-->
EOF
    }

    # call EditLabelRender on the common backend
    my $LabelString = $Self->{BackendCommonObject}->EditLabelRender(
        DynamicFieldConfig => $Param{DynamicFieldConfig},
        Mandatory          => $Param{Mandatory} || '0',
        FieldName          => $FieldName,
    );

    my $Data = {
        Field => $HTMLString,
        Label => $LabelString,
    };

    return $Data;
}

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

    my $FieldName = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};

    my $Value;

    # check if there is a Template and retreive the dinalic field value from there
    if ( IsHashRefWithData( $Param{Template} ) ) {
        $Value = $Param{Template}->{$FieldName};
    }

    # otherwise get dynamic field value form param
    else {
        my @Data = $Param{ParamObject}->GetArray( Param => $FieldName );

        $Value = \@Data;
    }

    if ( defined $Param{ReturnTemplateStructure} && $Param{ReturnTemplateStructure} eq 1 ) {
        return {
            $FieldName => $Value,
        };
    }

    # for this field the normal return an the ReturnValueStructure are the same
    return $Value;
}

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

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

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

    my $ServerError;
    my $ErrorMessage;

#    # perform necessary validations
#    if ( $Param{Mandatory} && !IsArrayRefWithData($Values) ) {
#        return {
#            ServerError => 1,
#        };
#    }
#    else {
#
#        # get possible values list
#        my $PossibleValues = $Param{DynamicFieldConfig}->{Config}->{PossibleValues};
#
#        # overwrite possible values if PossibleValuesFilter
#        if ( defined $Param{PossibleValuesFilter} ) {
#            $PossibleValues = $Param{PossibleValuesFilter}
#        }
#
#        # validate if value is in possible values list (but let pass empty values)
#        for my $Item ( @{$Values} ) {
#            if ( !$PossibleValues->{$Item} ) {
#                $ServerError  = 1;
#                $ErrorMessage = 'The field content is invalid';
#            }
#        }
#    }

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

    return $Result;
}

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

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

    # set Value and Title variables
    my $Value         = '';
    my $Title         = '';
    my $ValueMaxChars = $Param{ValueMaxChars} || '';
    my $TitleMaxChars = $Param{TitleMaxChars} || '';

    # check value
    my @Values;
    if ( ref $Param{Value} eq 'ARRAY' ) {
        @Values = @{ $Param{Value} };
    }
    else {
        @Values = ( $Param{Value} );
    }

    # get real values
    my $PossibleValues     = $Param{DynamicFieldConfig}->{Config}->{PossibleValues};
    my $TranslatableValues = $Param{DynamicFieldConfig}->{Config}->{TranslatableValues};

    my @ReadableValues;
    my @ReadableTitles;

    my $ShowValueEllipsis;
    my $ShowTitleEllipsis;

    VALUEITEM:
    for my $Item (@Values) {
        next VALUEITEM if !$Item;

        my $ReadableValue = $Item;

	$ReadableValue =~ s/^([^|]+)\|\|(.+)$/$2/;

        if ( $PossibleValues->{$Item} ) {
            $ReadableValue = $PossibleValues->{$Item};
            if ($TranslatableValues) {
                $ReadableValue = $Param{LayoutObject}->{LanguageObject}->Get($ReadableValue);
            }
        }

        my $ReadableLength = length $ReadableValue;

        # set title equal value
        my $ReadableTitle = $ReadableValue;

        # cut strings if needed
        if ( $ValueMaxChars ne '' ) {

            $ShowValueEllipsis = 1 if ( length $ReadableValue > $ValueMaxChars );
            $ReadableValue = substr $ReadableValue, 0, $ValueMaxChars;

            # decrease the max parameter
            $ValueMaxChars = $ValueMaxChars - $ReadableLength;
            $ValueMaxChars = 0 if $ValueMaxChars < 0;

        }

        if ( $TitleMaxChars ne '' ) {

            $ShowTitleEllipsis = 1 if ( length $ReadableTitle > $ValueMaxChars );
            $ReadableTitle = substr $ReadableTitle, 0, $TitleMaxChars;

            # decrease the max parameter
            $TitleMaxChars = $TitleMaxChars - $ReadableLength;
            $TitleMaxChars = 0 if $TitleMaxChars < 0;
        }

        # HTMLOuput transformations
        if ( $Param{HTMLOutput} ) {

            $ReadableValue = $Param{LayoutObject}->Ascii2Html(
                Text => $ReadableValue,
            );

            $ReadableTitle = $Param{LayoutObject}->Ascii2Html(
                Text => $ReadableTitle,
            );
        }

        push @ReadableValues, $ReadableValue if length $ReadableValue;
        push @ReadableTitles, $ReadableTitle if length $ReadableTitle;
    }

    # get specific field settings
    my $FieldConfig = $Self->{ConfigObject}->Get('DynamicFields::Backend')->{MultiselectFromDB} || {};

    # set new line separator
    my $ItemSeparator = $FieldConfig->{ItemSeparator} || ', ';

    $Value = join( $ItemSeparator, @ReadableValues );
    $Title = join( $ItemSeparator, @ReadableTitles );

    $Value .= '...' if $ShowValueEllipsis;
    $Title .= '...' if $ShowTitleEllipsis;

    # this field type does not support the Link Feature
    my $Link;

    # create return structure
    my $Data = {
        Value => $Value,
        Title => $Title,
        Link  => $Link,
    };

    return $Data;

}

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

    return 0;
}

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

    # take config from field config
    my $FieldConfig = $Param{DynamicFieldConfig}->{Config};
    my $FieldName   = 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name};
    my $FieldLabel  = $Param{DynamicFieldConfig}->{Label};

    my $Value;

    my @DefaultValue;

    if ( defined $Param{DefaultValue} ) {
        my @DefaultValue = split /;/, $Param{DefaultValue};
    }

    # set the field value
    if (@DefaultValue) {
        $Value = \@DefaultValue;
    }

    # get the field value, this fuction is always called after the profile is loaded
    my $FieldValues = $Self->SearchFieldValueGet(
        %Param,
    );

    if ( defined $FieldValues )
    {
        $Value = $FieldValues;
    }

    # check and set class if necessary
    my $FieldClass = 'DynamicFieldMultiSelect';

    # set PossibleValues
    my $SelectionData = $FieldConfig->{PossibleValues};

    # get historical values from database
    my $HistoricalValues = $Self->HistoricalValuesGet(%Param);

    # add historic values to current values (if they don't exist anymore)
    if ( IsHashRefWithData($HistoricalValues) ) {
        for my $Key ( keys %{$HistoricalValues} ) {
            if ( !$SelectionData->{$Key} ) {
                $SelectionData->{$Key} = $HistoricalValues->{$Key}
            }
        }
    }

    # use PossibleValuesFilter if defined
    $SelectionData = $Param{PossibleValuesFilter}
        if defined $Param{PossibleValuesFilter};

    my $HTMLString = $Param{LayoutObject}->BuildSelection(
        Data         => $SelectionData,
        Name         => $FieldName,
        SelectedID   => $Value,
        Translation  => $FieldConfig->{TranslatableValues} || 0,
        PossibleNone => 0,
        Class        => $FieldClass,
        Multiple     => 1,
        HTMLQuote    => 1,
    );

    # call EditLabelRender on the common backend
    my $LabelString = $Self->{BackendCommonObject}->EditLabelRender(
        DynamicFieldConfig => $Param{DynamicFieldConfig},
        FieldName          => $FieldName,
    );

    my $Data = {
        Field => $HTMLString,
        Label => $LabelString,
    };

    return $Data;
}

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

    my $Value;

    # get dynamic field value form param object
    if ( defined $Param{ParamObject} ) {
        my @FieldValues = $Param{ParamObject}
            ->GetArray( Param => 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name} );

        $Value = \@FieldValues;
    }

    # otherwise get the value from the profile
    elsif ( defined $Param{Profile} ) {
        $Value = $Param{Profile}->{ 'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name} };
    }
    else {
        return;
    }

    if ( defined $Param{ReturnProfileStructure} && $Param{ReturnProfileStructure} eq 1 ) {
        return {
            'Search_DynamicField_' . $Param{DynamicFieldConfig}->{Name} => $Value,
        };
    }

    return $Value;

}

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

    # get field value
    my $Value = $Self->SearchFieldValueGet(%Param);

    my $DisplayValue;

    if ($Value) {
        if ( ref $Value eq 'ARRAY' ) {

            my @DisplayItemList;
            for my $Item ( @{$Value} ) {

                # set the display value
                my $DisplayItem = $Param{DynamicFieldConfig}->{Config}->{PossibleValues}->{$Item};
                if ( $Param{DynamicFieldConfig}->{Config}->{TranslatableValues} ) {

                    # translate the value
                    $DisplayItem = $Param{LayoutObject}->{LanguageObject}->Get($DisplayValue);
                }

                push @DisplayItemList, $DisplayItem;
            }

            # combine different values into one string
            $DisplayValue = join ' + ', @DisplayItemList;
        }
        else {

            # set the display value
            $DisplayValue = $Param{DynamicFieldConfig}->{PossibleValues}->{$Value};

            if ( $Param{DynamicFieldConfig}->{Config}->{TranslatableValues} ) {

                # translate the value
                $DisplayValue = $Param{LayoutObject}->{LanguageObject}->Get($DisplayValue);
            }
        }
    }

    # return search parameter structure
    return {
        Parameter => {
            Equals => $Value,
        },
        Display => $DisplayValue,
    };
}

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

    # set PossibleValues
    my $Values = $Param{DynamicFieldConfig}->{Config}->{PossibleValues};

    # get historical values from database
    my $HistoricalValues = $Self->{DynamicFieldValueObject}->HistoricalValueGet(
        FieldID   => $Param{DynamicFieldConfig}->{ID},
        ValueType => 'Text,',
    );

    # add historic values to current values (if they don't exist anymore)
    for my $Key ( keys %{$HistoricalValues} ) {
        if ( !$Values->{$Key} ) {
            $Values->{$Key} = $HistoricalValues->{$Key}
        }
    }

    # use PossibleValuesFilter if defined
    $Values = $Param{PossibleValuesFilter}
        if defined $Param{PossibleValuesFilter};

    return {
        Values             => $Values,
        Name               => $Param{DynamicFieldConfig}->{Label},
        Element            => 'DynamicField_' . $Param{DynamicFieldConfig}->{Name},
        TranslatableValues => $Param{DynamicFieldconfig}->{Config}->{TranslatableValues},
    };
}

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

    my $Operator = 'Equals';
    my $Value    = $Param{Value};

    return {
        $Operator => $Value,
    };
}

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

    # set Value and Title variables
    my $Value = '';
    my $Title = '';

    # check value
    my @Values;
    if ( ref $Param{Value} eq 'ARRAY' ) {
        @Values = @{ $Param{Value} };
    }
    else {
        @Values = ( $Param{Value} );
    }

    my @ReadableValues;

    VALUEITEM:
    for my $Item (@Values) {
        next VALUEITEM if !$Item;

        push @ReadableValues, $Item;
    }

    # set new line separator
    my $ItemSeparator = ', ';

    # Ouput transformations
    $Value = join( $ItemSeparator, @ReadableValues );
    $Title = $Value;

    # cut strings if needed
    if ( $Param{ValueMaxChars} && length($Value) > $Param{ValueMaxChars} ) {
        $Value = substr( $Value, 0, $Param{ValueMaxChars} ) . '...';
    }
    if ( $Param{TitleMaxChars} && length($Title) > $Param{TitleMaxChars} ) {
        $Title = substr( $Title, 0, $Param{TitleMaxChars} ) . '...';
    }

    # create return structure
    my $Data = {
        Value => $Value,
        Title => $Title,
    };

    return $Data;
}

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

    my $FieldName = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};

    # set the field types
    my $EditValueType   = 'ARRAY';
    my $SearchValueType = 'ARRAY';

    # return the correct structure
    if ( $Param{FieldType} eq 'Edit' ) {
        return {
            $FieldName => $EditValueType,
            }
    }
    elsif ( $Param{FieldType} eq 'Search' ) {
        return {
            'Search_' . $FieldName => $SearchValueType,
            }
    }
    else {
        return {
            $FieldName             => $EditValueType,
            'Search_' . $FieldName => $SearchValueType,
            }
    }
}

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

    return 1;
}

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

    my $Value = int( rand(500) );

    my $Success = $Self->ValueSet(
        %Param,
        Value => $Value,
    );

    if ( !$Success ) {
        return {
            Success => 0,
        };
    }
    return {
        Success => 1,
        Value   => $Value,
    };
}

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

    return 1;
}

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

    my $FieldName = 'DynamicField_' . $Param{DynamicFieldConfig}->{Name};

    # the attribute must be an array
    return 0 if !IsArrayRefWithData( $Param{ObjectAttributes}->{$FieldName} );

    my $Match;

    # search in all values for this attribute
    VALUE:
    for my $AttributeValue ( @{ $Param{ObjectAttributes}->{$FieldName} } ) {

        # only need to match one
        if ( $Param{Value} eq $AttributeValue ) {
            $Match = 1;
            last VALUE;
        }
    }

    return $Match;
}

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

    # to store the possible values
    my %PossibleValues;

    if ( $Param{DynamicFieldConfig}->{Config} ) {
	$Param{DynamicFieldConfig}->{Config}->{StoreValue} = "1";
        %PossibleValues = %{ $Self->{BackendDropdownFromDB}->AJAXPossibleValuesGet(%Param) };
    }

#    # set none value if defined on field config
#    if ( $Param{DynamicFieldConfig}->{Config}->{PossibleNone} ) {
#        %PossibleValues = ( '' => '-' );
#    }
#
#    # set all other possible values if defined on field config
#    if ( IsHashRefWithData( $Param{DynamicFieldConfig}->{Config}->{PossibleValues} ) ) {
#        %PossibleValues = (
#            %PossibleValues,
#            %{ $Param{DynamicFieldConfig}->{Config}->{PossibleValues} },
#        );
#    }

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

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

    # get historical values from database
    my $HistoricalValues = $Self->{DynamicFieldValueObject}->HistoricalValueGet(
        FieldID   => $Param{DynamicFieldConfig}->{ID},
        ValueType => 'Text',
    );

    # retrun the historical values from database
    return $HistoricalValues;
}

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

    my $Value = defined $Param{Key} ? $Param{Key} : '';

    return $Value;

}

1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<http://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (AGPL). If you
did not receive this file, see L<http://www.gnu.org/licenses/agpl.txt>.

=cut

=head1 VERSION

$$

=cut
</File>
</Filelist>
</otrs_package>