FAQ
1.2.9
OTRS AG
http://otrs.org/
GNU GENERAL PUBLIC LICENSE Version 2, June 1991
Improved multi database support (mysql,mssql,postgrsql,oracle,db2), added unit tests, added install intro, fixed html design problems and added insert on courser feature.
Fixed Bug 2682, SQL Error on MS SQL Server.
Fixed SQL bug, reserved word 'result'.
Fixed bug in SQL query.
Fixed bug in permissioncheck by attachment download.
Fixed utf8 bug in cattachment get function.
Fixed quicksearch bug #2010 / #2012.
Fixed sql insert bug.
First version for otrs framework 2.2.x.
A FAQ/knowledge base.
Eine FAQ-/Wissensdatenbank.
2.2.x
The FAQ/knowledge base will be found at the navigation bar and required "faq" group permissions to use it. To create new FAQ article you need to create categories and define the permission who can use categories.
Die FAQ/Knowledge Base ist in der Navigationsleiste zu finden. Es werden "faq" Gruppen-Rechte benötig um die FAQ benutzen zu können. Um FAQ Artikel erstellen zu können müssen zuerst Kategorien erstellt und die Berechtigungs-Gruppen dieser Kategorien definiert werden.
# add faq group
require Kernel::System::Group;
$Self->{GroupObject} = Kernel::System::Group->new(%{$Self});
$Self->{GroupObject}->GroupAdd(
Name => 'faq',
Comment => 'faq database users',
ValidID => 1,
UserID => 1,
);
$Self->{GroupObject}->GroupAdd(
Name => 'faq_admin',
Comment => 'faq admin users',
ValidID => 1,
UserID => 1,
);
2008-03-19 13:01:40
opms.otrs.com
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSI/Pgo8b3Ryc19jb25maWcgdmVyc2lvbj0iMS4wIiBpbml0PSJBcHBsaWNhdGlvbiI+CiAgICA8Q1ZTPiRJZDogRkFRLnhtbCx2IDEuOCAyMDA4LzAzLzE4IDIzOjUzOjQxIG1hcnRpbiBFeHAgJDwvQ1ZTPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGlja2V0OjpGcm9udGVuZDo6Tm90ZVR5cGVzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+U3BlY2lmeSB0aGUgZGlmZmVyZW50IG5vdGUgdHlwZXMsIHRoYXQgeW91IHdhbnQgdG8gdXNlIGluIHlvdXIgc3lzdGVtLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5IaWVyIGv2bm5lbiBkaWUgdmVyc2NoaWVkZW5lbiBOb3Rpei1UeXBlbiBmZXN0Z2VsZWd0IHdlcmRlbiwgZGllIGlubmVyaGFsYiBkZXMgU3lzdGVtcyB2ZXJ3ZW5kZXQgd2VyZGVuIHNvbGxlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaWNrZXQ8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ6OlRpY2tldDo6Vmlld05vdGU8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0ibm90ZS1pbnRlcm5hbCI+MTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0ibm90ZS1leHRlcm5hbCI+MDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0ibm90ZS1yZXBvcnQiPjA8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGQVE6OkRlZmF1bHQ6OlJvb3RDYXRlZ29yeU5hbWUiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZhdWx0IGNhdGVnb3J5IG5hbWUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPlJvb3QgS2F0ZWdvcmllIE5hbWUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6RGVmYXVsdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmc+RkFRPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpEZWZhdWx0OjpSb290Q2F0ZWdvcnlDb21tZW50IiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBjYXRlZ29yeSBuYW1lLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5Sb290IEthdGVnb3JpZSBOYW1lLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkRlZmF1bHQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nPmRlZmF1bHQgY29tbWVudDwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6SXRlbTo6Vm90aW5nOjpSYXRlcyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlJhdGVzIGZvciB2b3RpbmcuIEtleSBtdXN0IGJlIGluIHByb2NlbnQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkdld2ljaHR1bmcgZvxyIGRpZSBCZXdlcnR1bmcuIERlciBLZXkgbXVzcyBpbiBQcm96ZW50IGFuZ2VnZWJlbiB3ZXJkZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6SXRlbTo6Vm90aW5nPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjEwMCI+MTAwICU8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ijc1Ij43NSAlPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSI1MCI+NTAgJTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMjUiPjI1ICU8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjAiPjAgJTwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6SXRlbTo6Vm90aW5nOjpTaG93IiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+U2hvdyB2b3RpbmcgaW4gZGVmaW5lZCBpbnRlcmZhY2VzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5JbnRlcmZhY2VzIGluIGRlbmVuIGRhcyBWb3RpbmcgRmVhdHVyZSBhbmdlemVpZ3Qgd2VyZGVuIHNvbGwuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6SXRlbTo6Vm90aW5nPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9ImludGVybmFsIj5pbnRlcm5hbCAoYWdlbnQpPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJleHRlcm5hbCI+ZXh0ZXJuYWwgKGN1c3RvbWVyKTwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6SXRlbTo6Vm90aW5nOjpRdWVzdGlvbkxhbmdLZXkiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5MYW5ndWFnZWtleSB3aGljaCBpcyBkZWZpbmVkIGluIHRoZSBsYW5ndWFnZSBmaWxlICpfRkFRLnBtLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5MYW5ndWFnZUtleSBm/HIgZGllIEZyYWdlIGJlaSBkZXIgQXJ0aWtlbGJld2VydHVuZy4gV2lyZCBpbiBkZW4gU3ByYWNoZmlsZXMgZGVmaW5pZXJ0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6Okl0ZW06OlZvdGluZzwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmc+QXJ0aWNsZVZvdGluZ1F1ZXN0aW9uPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpFeHBsb3Jlcjo6UGF0aDo6U2hvdyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlNob3cgRkFRIHBhdGggeWVzL25vLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5GQVEgUGZhZCBhbnplaWdlbiBqYS9uZWluLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkV4cGxvcmVyPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPE9wdGlvbiBTZWxlY3RlZElEPSIxIj4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMCI+Tm88L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjEiPlllczwvSXRlbT4KICAgICAgICAgICAgPC9PcHRpb24+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpFeHBsb3Jlcjo6SXRlbUxpc3Q6OlZvdGluZ1Jlc3VsdERlY2ltYWxQbGFjZXMiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWNpbWFsIHBsYWNlcyBvZiB0aGUgdm90aW5nIHJlc3VsdC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGV6aW1hbHN0ZWxsZW4gZGVzIEVyZ2VibmlzZXMgYmVpIGRlciBBcnRpa2VsYmV3ZXJ0dW5nLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkV4cGxvcmVyPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZz4yPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpFeHBsb3Jlcjo6SXRlbUxpc3Q6OlZvdGluZ1Jlc3VsdENvbG9ycyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkNTUyBjb2xvciBmb3IgdGhlIHZvdGluZyByZXN1bHQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkZhcmJlbiBm/HIgZGllIEFuemVpZ2UgZGVzIEVyZ2VibmlzZXMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6RXhwbG9yZXI8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMzAiPnJlZDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iNzAiPm9yYW5nZTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMTAwIj5ncmVlbjwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6RXhwbG9yZXI6OlBhdGg6OlNlcGVyYXRvciIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkZBUSBwYXRoIHNlcGVyYXRvci48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+VHJlbm56ZWljaGVuIGltIEZBUSBQZmFkLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkV4cGxvcmVyPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZz67PC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpDYXRlZ29yeTo6SWNvbiIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkNhdGVnb3J5IGljb24uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkthdGVnb3JpZSBJY29uIGluIGRlciBGQVEuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6Q2F0ZWdvcnk8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nPmZhcS1jYXRlZ29yeS1zbWFsbC5wbmc8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGQVE6Okl0ZW06Okljb24iIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5JdGVtIGljb24uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkFydGlrZWwgSWNvbiBpbiBkZXIgRkFRLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6Okl0ZW08L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nPmZhcS1pdGVtLXNtYWxsLnBuZzwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6RXhwbG9yZXI6OlF1aWNrU2VhcmNoOjpTaG93IiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+SW50ZXJmYWNlcyB3aGVyZSB0aGUgcXVpY2tzZWFyY2ggc2hvdWxkIGJlIHNob3duLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5JbnRlcmZhY2VzIGluIGRlbmVuIGRhcyBRdWlja1NlYXJjaCBGZWF0dXJlIGFuZ2V6ZWlndCB3aXJkLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkV4cGxvcmVyPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9ImludGVybmFsIj5pbnRlcm5hbCAoYWdlbnQpPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJleHRlcm5hbCI+ZXh0ZXJuYWwgKGN1c3RvbWVyKTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0icHVibGljIj5wdWJsaWMgKHB1YmxpYyk8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGQVE6OkV4cGxvcmVyOjpRdWlja1NlYXJjaDo6U2hvd1N1YkNhdGVnb3J5SXRlbXMiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5TaG93IGl0ZW1zIG9mIHN1YmNhdGVnb3JpZXMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkFydGlrZWwgYXVzIFN1YmthdGVnb3JpZW4gYW56ZWlnZW4gamEvbmVpbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GQVE8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpFeHBsb3JlcjwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxPcHRpb24gU2VsZWN0ZWRJRD0iMSI+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjAiPk5vPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIxIj5ZZXM8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6RXhwbG9yZXI6Okxhc3RDaGFuZ2U6OlNob3ciIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5TaG93IGxhc3QgY2hhbmdlIGl0ZW1zLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5JbnRlcmZhY2VzIGluIGRlbmVuIGRhcyBMYXN0Q2hhbmdlIEZlYXR1cmUgYW5nZXplaWd0IHdlcmRlbiBzb2xsLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkV4cGxvcmVyPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9ImludGVybmFsIj5pbnRlcm5hbCAoYWdlbnQpPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJleHRlcm5hbCI+ZXh0ZXJuYWwgKGN1c3RvbWVyKTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0icHVibGljIj5wdWJsaWMgKHB1YmxpYyk8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGQVE6OkV4cGxvcmVyOjpMYXN0Q2hhbmdlOjpTaG93U3ViQ2F0ZWdvcnlJdGVtcyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlNob3cgaXRlbXMgb2Ygc3ViY2F0ZWdvcmllcy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+QXJ0aWtlbCBhdXMgU3Via2F0ZWdvcmllbiBhbnplaWdlbiBqYS9uZWluLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkV4cGxvcmVyPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPE9wdGlvbiBTZWxlY3RlZElEPSIxIj4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMCI+Tm88L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjEiPlllczwvSXRlbT4KICAgICAgICAgICAgPC9PcHRpb24+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpFeHBsb3Jlcjo6TGFzdENoYW5nZTo6TGltaXQiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5OdW1iZXIgb2Ygc2hvd24gaXRlbXMgaW4gbGFzdCBjaGFuZ2VzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5BbnphaGwgZGVyIHp1IGFuemVpZ2VuZGVuIEFydGlrZWwgaW4gbGV0enRlbiDEbmRlcnVuZ2VuLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkV4cGxvcmVyPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZz4zPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpFeHBsb3Jlcjo6TGFzdENyZWF0ZTo6U2hvdyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPk51bWJlciBvZiBzaG93biBpdGVtcyBpbiBsYXN0IGNyZWF0ZWQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkludGVyZmFjZXMgaW4gZGVuZW4gZGFzIExhc3RDcmVhdGUgRmVhdHVyZSBhbmdlemVpZ3Qgd2VyZGVuIHNvbGwuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6RXhwbG9yZXI8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iaW50ZXJuYWwiPmludGVybmFsIChhZ2VudCk8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9ImV4dGVybmFsIj5leHRlcm5hbCAoY3VzdG9tZXIpPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJwdWJsaWMiPnB1YmxpYyAocHVibGljKTwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6RXhwbG9yZXI6Okxhc3RDcmVhdGU6OlNob3dTdWJDYXRlZ29yeUl0ZW1zIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+U2hvdyBpdGVtcyBvZiBzdWJjYXRlZ29yaWVzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5BcnRpa2VsIGF1cyBTdWJrYXRlZ29yaWVuIGFuemVpZ2VuIGphL25laW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6RXhwbG9yZXI8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8T3B0aW9uIFNlbGVjdGVkSUQ9IjEiPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIwIj5ObzwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMSI+WWVzPC9JdGVtPgogICAgICAgICAgICA8L09wdGlvbj4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGQVE6OkV4cGxvcmVyOjpMYXN0Q3JlYXRlOjpMaW1pdCIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlNob3duIGl0ZW1zIGluIGxhc3QgY3JlYXRlZC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+QW56YWhsIGRlciB6dSBhbnplaWdlbmRlbiBBcnRpa2VsIGluIHp1bGV0enQgZXJzdGVsbHRlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkV4cGxvcmVyPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZz4zPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpGQVFIb29rIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBob29rIGZvciBGQVEuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPlN0YW5kYXJkIEJlemVpY2huZXIgZvxyIGVpbmUgRkFRLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nPkZBUSM8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGQVE6OkRlZmF1bHQ6OlN0YXRlIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmYXVsdCBzdGF0ZSBmb3IgRkFRIGVudHJ5LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5TdGFuZGFyZCBTdGF0dXMgZWluZXMgIEZBUS1FaW50cmFncy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GQVE8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPFN0cmluZz5pbnRlcm5hbCAoYWdlbnQpPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpJdGVtOjpGaWVsZDEiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZpbml0aW9uIG9mIEZBUSBpdGVtIGZyZWUgdGV4dCBmaWVsZC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pdGlvbiBkZXIgZnJlaWVuIFRleHRmZWxkZXIuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6SXRlbTwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJTaG93Ij5wdWJsaWM8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkNhcHRpb24iPlN5bXB0b208L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlByaW8iPjEwMDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6SXRlbTo6RmllbGQyIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmaW5pdGlvbiBvZiBGQVEgaXRlbSBmcmVlIHRleHQgZmllbGQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaXRpb24gZGVyIGZyZWllbiBUZXh0ZmVsZGVyLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6Okl0ZW08L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iU2hvdyI+cHVibGljPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJDYXB0aW9uIj5Qcm9ibGVtPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJQcmlvIj4yMDA8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGQVE6Okl0ZW06OkZpZWxkMyIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmluaXRpb24gb2YgRkFRIGl0ZW0gZnJlZSB0ZXh0IGZpZWxkLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbml0aW9uIGRlciBmcmVpZW4gVGV4dGZlbGRlci48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GQVE8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpJdGVtPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlNob3ciPnB1YmxpYzwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iQ2FwdGlvbiI+U29sdXRpb248L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlByaW8iPjMwMDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6SXRlbTo6RmllbGQ0IiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmaW5pdGlvbiBvZiBGQVEgaXRlbSBmcmVlIHRleHQgZmllbGQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaXRpb24gZGVyIGZyZWllbiBUZXh0ZmVsZGVyLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6Okl0ZW08L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iU2hvdyI+PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJDYXB0aW9uIj5GaWVsZDQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlByaW8iPjQwMDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6SXRlbTo6RmllbGQ1IiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmaW5pdGlvbiBvZiBGQVEgaXRlbSBmcmVlIHRleHQgZmllbGQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaXRpb24gZGVyIGZyZWllbiBUZXh0ZmVsZGVyLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6Okl0ZW08L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iU2hvdyI+PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJDYXB0aW9uIj5GaWVsZDU8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlByaW8iPjUwMDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZBUTo6SXRlbTo6RmllbGQ2IiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmaW5pdGlvbiBvZiBGQVEgaXRlbSBmcmVlIHRleHQgZmllbGQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaXRpb24gZGVyIGZyZWllbiBUZXh0ZmVsZGVyLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6Okl0ZW08L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iU2hvdyI+aW50ZXJuYWw8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkNhcHRpb24iPkNvbW1lbnQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlByaW8iPjYwMDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3QjIyNGQVEiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5MaW5rIG9iamVjdCBzZXR0aW5ncyBmb3IgRkFRLiBPYmplY3RzIHRoYXQgYXJlIGtub3duIGJ5IHRoZSBzeXN0ZW0uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkxpbmsgT2JqZWt0IEVpbnN0ZWxsdW5nZW4gZvxyIGRpZSBGQVEuIE9iamVrdGUsIGRpZSBkZW0gU3lzdGVtIGJla2FubnQgc2luZC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GQVE8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpMaW5rT2JqZWN0PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik5hbWUiPkZBUSBPYmplY3Q8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPk9iamVjdDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTGlua09iamVjdHMiPgogICAgICAgICAgICAgICAgICAgIDxBcnJheT4KICAgICAgICAgICAgICAgICAgICAgICAgPEl0ZW0+VGlja2V0PC9JdGVtPgogICAgICAgICAgICAgICAgICAgICAgICA8SXRlbT5GQVE8L0l0ZW0+CiAgICAgICAgICAgICAgICAgICAgPC9BcnJheT4KICAgICAgICAgICAgICAgIDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZyb250ZW5kOjpNb2R1bGUjIyNBZ2VudEZBUSIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkZyb250ZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBGQVEgb2JqZWN0IGluIHRoZSBhZ2VudCBpbnRlcmZhY2UuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkZyb250ZW5kbW9kdWwtUmVnaXN0cmF0aW9uIGb8ciBkYXMgRkFRLU9iamVrdCBpbSBBZ2VudC1JbnRlcmZhY2UuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50OjpNb2R1bGVSZWdpc3RyYXRpb248L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8RnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgICAgICAgICA8R3JvdXA+ZmFxPC9Hcm91cD4KICAgICAgICAgICAgICAgIDxHcm91cFJvPmZhcTwvR3JvdXBSbz4KICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5GQVEtQXJlYTwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICA8TmF2QmFyTmFtZT5GQVE8L05hdkJhck5hbWU+CiAgICAgICAgICAgICAgICA8VGl0bGU+RkFRPC9UaXRsZT4KICAgICAgICAgICAgICAgIDxOYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEdyb3VwUm8+ZmFxPC9Hcm91cFJvPgogICAgICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5GQVEtQXJlYTwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICAgICAgPFR5cGU+TWVudTwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+SXRlbUFyZWE8L0Jsb2NrPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPkZBUTwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+aGVscC5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudEZBUSZhbXA7TmF2PU5vcm1hbDwvTGluaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPkZBUTwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+cTwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjgzMDA8L1ByaW8+CiAgICAgICAgICAgICAgICA8L05hdkJhcj4KICAgICAgICAgICAgICAgIDxOYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEdyb3VwUm8+ZmFxPC9Hcm91cFJvPgogICAgICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5FeHBsb3JlcjwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+RXhwbG9yZXI8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPmxpc3QucG5nPC9JbWFnZT4KICAgICAgICAgICAgICAgICAgICA8TGluaz5BY3Rpb249QWdlbnRGQVEmYW1wO1N1YmFjdGlvbj1FeHBsb3JlcjwvTGluaz4KICAgICAgICAgICAgICAgICAgICA8VHlwZT48L1R5cGU+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPjwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPE5hdkJhcj5GQVE8L05hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8QWNjZXNzS2V5PmU8L0FjY2Vzc0tleT4KICAgICAgICAgICAgICAgICAgICA8UHJpbz4xMDA8L1ByaW8+CiAgICAgICAgICAgICAgICA8L05hdkJhcj4KICAgICAgICAgICAgICAgIDxOYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEdyb3VwPmZhcTwvR3JvdXA+CiAgICAgICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPk5ldyBBcnRpY2xlPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgICAgICA8TmFtZT5OZXcgQXJ0aWNsZTwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+bmV3LnBuZzwvSW1hZ2U+CiAgICAgICAgICAgICAgICAgICAgPExpbms+QWN0aW9uPUFnZW50RkFRJmFtcDtTdWJhY3Rpb249QWRkPC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPkZBUTwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+bjwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjIwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8R3JvdXBSbz5mYXE8L0dyb3VwUm8+CiAgICAgICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPkZBUS1TZWFyY2g8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPlNlYXJjaDwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+c2VhcmNoLnBuZzwvSW1hZ2U+CiAgICAgICAgICAgICAgICAgICAgPExpbms+QWN0aW9uPUFnZW50RkFRJmFtcDtTdWJhY3Rpb249U2VhcmNoPC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPkZBUTwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+czwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjMwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8R3JvdXBSbz5mYXE8L0dyb3VwUm8+CiAgICAgICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPkhpc3Rvcnk8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPkhpc3Rvcnk8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPmhpc3RvcnkucG5nPC9JbWFnZT4KICAgICAgICAgICAgICAgICAgICA8TGluaz5BY3Rpb249QWdlbnRGQVEmYW1wO1N1YmFjdGlvbj1TeXN0ZW1IaXN0b3J5PC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPkZBUTwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+bzwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjMxMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8R3JvdXA+ZmFxX2FkbWluPC9Hcm91cD4KICAgICAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+TGFuZ3VhZ2U8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPkxhbmd1YWdlPC9OYW1lPgogICAgICAgICAgICAgICAgICAgIDxJbWFnZT5sYW5ndWFnZS5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudEZBUSZhbXA7U3ViYWN0aW9uPUxhbmd1YWdlPC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPkZBUTwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+dTwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjkxMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8R3JvdXA+ZmFxX2FkbWluPC9Hcm91cD4KICAgICAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+Q2F0ZWdvcnk8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPkNhdGVnb3J5PC9OYW1lPgogICAgICAgICAgICAgICAgICAgIDxJbWFnZT5jYXRlZ29yeS5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudEZBUSZhbXA7U3ViYWN0aW9uPUNhdGVnb3J5PC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPkZBUTwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+YzwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjkyMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICA8L0Zyb250ZW5kTW9kdWxlUmVnPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkN1c3RvbWVyRnJvbnRlbmQ6Ok1vZHVsZSMjI0N1c3RvbWVyRkFRIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RnJvbnRlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIEN1c3RvbWVyRkFRIG9iamVjdCBpbiB0aGUgY3VzdG9tZXIgaW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5Gcm9udGVuZG1vZHVsLVJlZ2lzdHJhdGlvbiBm/HIgZGFzIEN1c3RvbWVyRkFRLU9iamVrdCBpbSBDdXN0b21lci1JbnRlcmZhY2UuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RkFRPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkN1c3RvbWVyOjpNb2R1bGVSZWdpc3RyYXRpb248L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8RnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+Q3VzdG9tZXIgZmFxPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgIDxOYXZCYXJOYW1lPkZBUTwvTmF2QmFyTmFtZT4KICAgICAgICAgICAgICAgIDxUaXRsZT48L1RpdGxlPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+RkFRLUFyZWE8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPkZBUS1BcmVhPC9OYW1lPgogICAgICAgICAgICAgICAgICAgIDxJbWFnZT5oZWxwLnBuZzwvSW1hZ2U+CiAgICAgICAgICAgICAgICAgICAgPExpbms+QWN0aW9uPUN1c3RvbWVyRkFRPC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPjwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+ZjwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjQwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICA8L0Zyb250ZW5kTW9kdWxlUmVnPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlB1YmxpY0Zyb250ZW5kOjpDb21tb25QYXJhbSMjI0FjdGlvbiIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmF1bHQgdmFsdWUgZm9yIHRoZSBBY3Rpb24gcGFyYW1ldGVyIGZvciB0aGUgcHVibGljIGZyb250ZW5kLiBUaGUgQWN0aW9uIHBhcmFtZXRlciBpcyB1c2VkIGluIHRoZSBzY3JpcHRzIG9mIHRoZSBzeXN0ZW0uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPlN0YW5kYXJkd2VydCBkZXMgQWN0aW9uLVBhcmFtZXRlcnMgZvxyIGRlbiD2ZmZlbnRsaWNoZW4gRkFRLUJlcmVpY2guIERlciBBY3Rpb24tUGFyYW1ldGVyIHdpcmQgdm9uIGRlbiBTa3JpcHRlbiBkZXMgU3lzdGVtcyBiZW51dHp0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpQdWJsaWM8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nPlB1YmxpY0ZBUTwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlB1YmxpY0Zyb250ZW5kOjpNb2R1bGUjIyNQdWJsaWNGQVEiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgUHVibGljRkFRIG9iamVjdCBpbiB0aGUgcHVibGljIEZBUSBhcmVhLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5Gcm9udGVuZG1vZHVsLVJlZ2lzdHJhdGlvbiBm/HIgZGFzIFB1YmxpY0ZBUS1PYmpla3QgaW0g9mZmZW50bGljaGVuIEZBUS1CZXJlaWNoLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpQdWJsaWM6Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxGcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5DdXN0b21lciBmYXE8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgPE5hdkJhck5hbWU+RkFRPC9OYXZCYXJOYW1lPgogICAgICAgICAgICAgICAgPFRpdGxlPjwvVGl0bGU+CiAgICAgICAgICAgICAgICA8TmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5GQVEtQXJlYTwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+RkFRLUFyZWE8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPmhlbHAucG5nPC9JbWFnZT4KICAgICAgICAgICAgICAgICAgICA8TGluaz5BY3Rpb249Q3VzdG9tZXJGQVE8L0xpbms+CiAgICAgICAgICAgICAgICAgICAgPFR5cGU+PC9UeXBlPgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz48L0Jsb2NrPgogICAgICAgICAgICAgICAgICAgIDxOYXZCYXI+PC9OYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEFjY2Vzc0tleT5mPC9BY2Nlc3NLZXk+CiAgICAgICAgICAgICAgICAgICAgPFByaW8+NDAwPC9QcmlvPgogICAgICAgICAgICAgICAgPC9OYXZCYXI+CiAgICAgICAgICAgIDwvRnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRkFROjpJdGVtOjpIVE1MIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+U2hvdyBGQVEgQXJ0aWNsZSB3aXRoIEhUTUwuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkhUTUwgRGFyc3RlbGx1bmcgZGVyIEZBUSBBcnRpa2VsIGVpbnNjaGFsdGVuLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZBUTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6Okl0ZW08L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8T3B0aW9uIFNlbGVjdGVkSUQ9IjEiPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIwIj5ObzwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMSI+WWVzPC9JdGVtPgogICAgICAgICAgICA8L09wdGlvbj4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+Cjwvb3Ryc19jb25maWc+Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9GQVEucG0gLSBwcm92aWRlcyBkZSBsYW5ndWFnZSB0cmFuc2xhdGlvbgojIENvcHlyaWdodCAoQykgMjAwMS0yMDA4IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogZGVfRkFRLnBtLHYgMS4zIDIwMDgvMDMvMTkgMTA6NTU6NDMgbWFydGluIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTIuMC50eHQuCiMgLS0KcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpkZV9GQVE7Cgp1c2Ugc3RyaWN0OwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4zICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CiAgICBteSAkVHJhbnNsYXRpb24gPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRUcmFuc2xhdGlvbiBuZSAnSEFTSCc7CgogICAgIyAkJFNUQVJUJCQKCiAgICAjIG93biB0cmFuc2xhdGlvbnMKICAgICRUcmFuc2xhdGlvbi0+eydZb3UgaGF2ZSBhbHJlYWR5IHZvdGVkISd9ID0gJ1NpZSBoYWJlbiBiZXJlaXRzIGJld2VydGV0ISc7CiAgICAkVHJhbnNsYXRpb24tPnsnTm8gcmF0ZSBzZWxlY3RlZCEnfSA9ICdLZWluZSBCZXdlcnR1bmcgYXVzd+RobHQhJzsKICAgICRUcmFuc2xhdGlvbi0+eydUaGFua3MgZm9yIHZvdGUhJ30gPSAnRGFua2UgZvxyIElocmUgQmV3ZXJ0dW5nZW4hJzsKICAgICRUcmFuc2xhdGlvbi0+e1ZvdGVzfSA9ICdCZXdlcnR1bmdlbic7CiAgICAkVHJhbnNsYXRpb24tPntMYXRlc3RDaGFuZ2VkSXRlbXN9ID0gJ3p1bGV0enQgZ2XkbmRlcnRlIEFydGlrZWwnOwogICAgJFRyYW5zbGF0aW9uLT57TGF0ZXN0Q3JlYXRlZEl0ZW1zfSA9ICd6dWxldHp0IGVyc3RlbGx0ZSBBcnRpa2VsJzsKICAgICRUcmFuc2xhdGlvbi0+e0FydGljbGVWb3RpbmdRdWVzdGlvbn0gPSAnV2llIGd1dCB3dXJkZSBtaXQgZGllc2VtIEFydGlrZWwgSWhyZSBGcmFnZSBiZWFudHdvcnRldD8nOwogICAgJFRyYW5zbGF0aW9uLT57U3ViQ2F0ZWdvcnlPZn0gPSAnVW50ZXJrYXRlZ29yaWUgdm9uJzsKICAgICRUcmFuc2xhdGlvbi0+e1F1aWNrU2VhcmNofSA9ICdTY2huZWxsc3VjaGUnOwogICAgJFRyYW5zbGF0aW9uLT57RGV0YWlsU2VhcmNofSA9ICdEZXRhaWxzdWNoZSc7CiAgICAkVHJhbnNsYXRpb24tPntDYXRlZ29yaWVzfSA9ICdLYXRlZ29yaWVuJzsKICAgICRUcmFuc2xhdGlvbi0+e1N1YkNhdGVnb3JpZXN9ID0gJ1N1YmthdGVnb3JpZW4nOwogICAgJFRyYW5zbGF0aW9uLT57J0EgY2F0ZWdvcnkgc2hvdWxkIGhhdmUgYSBuYW1lISd9ID0gJ0VpbmUgS2F0ZWdvcmllbiBzb2xsdGUgZWluZW4gTmFtZW4gaGFiZW4hJzsKICAgICRUcmFuc2xhdGlvbi0+eydBIGNhdGVnb3J5IHNob3VsZCBoYXZlIGEgY29tbWVudCEnfSA9ICdFaW5lIEthdGVnb3JpZW4gc29sbHRlIGVpbmVuIEtvbW1lbnRhciBoYWJlbiEnOwoKICAgICMgJCRTVE9QJCQKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9lbl9GQVEucG0gLSBwcm92aWRlcyBlbiBsYW5ndWFnZSB0cmFuc2xhdGlvbgojIENvcHlyaWdodCAoQykgMjAwMS0yMDA4IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogZW5fRkFRLnBtLHYgMS4zIDIwMDgvMDMvMTkgMTA6NTU6NDMgbWFydGluIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTIuMC50eHQuCiMgLS0KcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjplbl9GQVE7Cgp1c2Ugc3RyaWN0OwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4zICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CiAgICBteSAkVHJhbnNsYXRpb24gPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRUcmFuc2xhdGlvbiBuZSAnSEFTSCc7CgogICAgIyAkJFNUQVJUJCQKCiAgICAjIG93biB0cmFuc2xhdGlvbnMKICAgICRUcmFuc2xhdGlvbi0+e1ZvdGVzfSA9ICdWb3Rlcyc7CiAgICAkVHJhbnNsYXRpb24tPntMYXRlc3RDaGFuZ2VkSXRlbXN9ID0gJ2xhdGVzdCBjaGFuZ2VkIGFydGljbGUnOwogICAgJFRyYW5zbGF0aW9uLT57TGF0ZXN0Q3JlYXRlZEl0ZW1zfSA9ICdsYXRlc3QgY3JlYXRlZCBhcnRpY2xlJzsKICAgICRUcmFuc2xhdGlvbi0+e0FydGljbGVWb3RpbmdRdWVzdGlvbn0gPSAnRGlkIHRoaXMgYXJ0aWNsZSBoZWxwPyc7CiAgICAkVHJhbnNsYXRpb24tPntRdWlja1NlYXJjaH0gPSAnUXVpY2sgU2VhcmNoJzsKICAgICRUcmFuc2xhdGlvbi0+e0RldGFpbFNlYXJjaH0gPSAnRGV0YWlsIFNlYXJjaCc7CiAgICAkVHJhbnNsYXRpb24tPnsnWW91IGhhdmUgYWxyZWFkeSB2b3RlZCEnfSA9ICdZb3UgaGF2ZSBhbHJlYWR5IHZvdGUhJzsKICAgICRUcmFuc2xhdGlvbi0+eydObyByYXRlIHNlbGVjdGVkISd9ID0gJ05vIHJhdGUgc2VsZWN0ZWQhJzsKICAgICRUcmFuc2xhdGlvbi0+eydUaGFua3MgZm9yIHZvdGUhJ30gPSAnVGhhbmtzIGZvciB2b3RlISc7CiAgICAkVHJhbnNsYXRpb24tPntDYXRlZ29yaWVzfSA9ICdDYXRlZ29yaWVzJzsKICAgICRUcmFuc2xhdGlvbi0+e1N1YkNhdGVnb3JpZXN9ID0gJ1N1YmNhdGVnb3JpZXMnOwoKICAgICMgJCRTVE9QJCQKICAgIHJldHVybiAxOwp9CgoxOwo=
# --
# Kernel/Modules/FAQ.pm - faq module
# Copyright (C) 2001-2008 OTRS AG, http://otrs.org/
# --
# $Id: FAQ.pm,v 1.12 2008/03/19 10:52:50 martin Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --

package Kernel::Modules::FAQ;

use strict;
use Kernel::System::User;
use Kernel::System::FAQ;
use Kernel::System::LinkObject;

use vars qw($VERSION);
$VERSION = '$Revision: 1.12 $';
$VERSION =~ s/^\$.*:\W(.*)\W.+?$/$1/;

sub new {
    my $Type = shift;
    my %Param = @_;

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

    foreach (keys %Param) {
        $Self->{$_} = $Param{$_};
    }

    # check needed Opjects
    foreach (qw(ParamObject DBObject LayoutObject LogObject ConfigObject UserObject)) {
        $Self->{LayoutObject}->FatalError(Message => "Got no $_!") if (!$Self->{$_});
    }

    # faq object
    $Self->{FAQObject} = Kernel::System::FAQ->new(%Param);
    # link object
    $Self->{LinkObject} = Kernel::System::LinkObject->new(%Param);
    # agent user
    $Self->{AgentUserObject} = Kernel::System::User->new(%Param);

    # interface settings

    $Self->{Interface} = $Self->{FAQObject}->StateTypeGet(
        Name => 'public'
    );
    $Self->{InterfaceStates} = $Self->{FAQObject}->StateTypeList(
        Types => ['public']
    );

    # global output vars
    $Self->{Notify} = [];

    # store last screen
    if($Self->{SessionObject}) {
        if($Self->{Subaction} eq 'Explorer') {
            $Self->{SessionObject}->UpdateSessionID(
                SessionID => $Self->{SessionID},
                Key => 'LastScreenOverview',
                Value => $Self->{RequestedURL},
            );
        }
        else {
            $Self->{SessionObject}->UpdateSessionID(
                SessionID => $Self->{SessionID},
                Key => 'LastScreenView',
                Value => $Self->{RequestedURL},
            );
        }
    }
    return $Self;
}

sub GetExplorer {
    my $Self = shift;
    my %Param = @_;
    my %Frontend = ();
    my %GetParam = ();
    my @Params = qw(Order Sort);

    # manage parameters
    $GetParam{CategoryID} = $Self->{ParamObject}->GetParam(Param => 'CategoryID') || 0;
    $GetParam{Order} = $Self->{ParamObject}->GetParam(Param => 'Order') || 'Title';
    $GetParam{Sort} = $Self->{ParamObject}->GetParam(Param => 'Sort') || 'up';
    foreach (@Params) {
        if(!$GetParam{$_} && !($GetParam{$_} = $Self->{ParamObject}->GetParam(Param => $_))) {
            return $Self->{LayoutObject}->FatalError(Message => "Need parameter $_");
        }
    }
    # store back link
    if($Self->{SessionObject}) {
        $Self->{SessionObject}->UpdateSessionID(
            SessionID => $Self->{SessionID},
            Key => 'LastScreenView',
            Value => $Self->{RequestedURL},
        );
    }
    # db action
    my %CategoryData = ();
    if($GetParam{CategoryID}) {
        %CategoryData = $Self->{FAQObject}->CategoryGet(CategoryID => $GetParam{CategoryID}, UserID => $Self->{UserID});
        if (!%CategoryData) {
            return $Self->{LayoutObject}->ErrorScreen();
        }
    }
    # dtl block
    $Frontend{LastScreenOverview} = $Self->{RequestedURL};
    $Self->{LayoutObject}->Block(
        Name => 'Explorer',
        Data => { %CategoryData, %Frontend },
    );
    # FAQ path
    $Self->_GetFAQPath(
        CategoryID => $GetParam{CategoryID},
    );
    # explorer title
    if($GetParam{CategoryID}) {
        $Self->{LayoutObject}->Block(
            Name => 'ExplorerTitle',
            Data => {
                        'Name' => $CategoryData{Name},
                        'Comment' => $CategoryData{Comment}
                    },
        );
    }
    else {
        $Self->{LayoutObject}->Block(
            Name => 'ExplorerTitle',
            Data => {
                        'Name' => $Self->{ConfigObject}->Get('FAQ::Default::RootCategoryName'),
                        'Comment' => $Self->{ConfigObject}->Get('FAQ::Default::RootCategoryComment')
                    },
        );
    }
    # explorer category list
    if (!$Param{Mode}) {
        $Param{Mode} = 'Public';
    }
    if (!defined($Param{CustomerUser})) {
        $Param{CustomerUser} = '';
    }
    if ($Param{Mode} && $Param{Mode} eq 'Customer') {
        $Self->_GetExplorerCategoryList(
            CategoryID => $GetParam{CategoryID},
            Order => 'Name',
            Sort => 'up',
            Mode => $Param{Mode},
            CustomerUser => $Param{CustomerUser},
        );
    }
    else {
        $Self->_GetExplorerCategoryList(
            CategoryID => $GetParam{CategoryID},
            Order => 'Name',
            Sort => 'up',
            Mode => $Param{Mode},
        );
    }
    # explorer item list
    $Self->_GetExplorerItemList(
        CategoryID => $GetParam{CategoryID},
        Order => $GetParam{Order} || 'Title',
        Sort => $GetParam{Sort} || 'up'
    );
    # quicksearch
    my %ShowQuickSearch = %{$Self->{ConfigObject}->Get('FAQ::Explorer::QuickSearch::Show')};
    if (exists($ShowQuickSearch{$Self->{Interface}{Name}})) {
        $Self->_GetExplorerQuickSearch(
            CategoryID => $GetParam{CategoryID},
        );
    }
    # latest change faq items
    my %ShowLastChange = %{$Self->{ConfigObject}->Get('FAQ::Explorer::LastChange::Show')};
    if (exists($ShowLastChange{$Self->{Interface}{Name}})) {
        $Self->_GetExplorerLastChangeItems(
            CategoryID => $GetParam{CategoryID},
            Mode => $Param{Mode},
            CustomerUser => $Param{CustomerUser},
        );
    }
    # latest create faq items
    my %ShowLastCreate = %{$Self->{ConfigObject}->Get('FAQ::Explorer::LastCreate::Show')};
    if (exists($ShowLastCreate{$Self->{Interface}{Name}})) {
        $Self->_GetExplorerLastCreateItems(
            CategoryID => $GetParam{CategoryID},
            Mode => $Param{Mode},
            CustomerUser => $Param{CustomerUser},
        );
    }
}

sub _GetFAQPath {
    my $Self = shift;
    my %Param = @_;

    $Self->{LayoutObject}->Block(
        Name => 'FAQPathCategoryElement',
        Data => { 'Name' => $Self->{ConfigObject}->Get('FAQ::Default::RootCategoryName'),
            'CategoryID' => '0'
        },
    );

    if($Self->{ConfigObject}->Get('FAQ::Explorer::Path::Show')) {
        my @CategoryList = @{$Self->{FAQObject}->FAQPathListGet(CategoryID => $Param{CategoryID})};
        foreach my $Data (@CategoryList) {
            $Self->{LayoutObject}->Block(
                Name => 'FAQPathCategoryElement',
                Data => { %{$Data} },
            );
        }
        return 1;
    }
    return 0;
}

sub _GetExplorerCategoryList {
    my $Self = shift;
    my %Param = @_;
    my %Frontend = ();
    $Frontend{CssRow} = '';

    # check needed parameters
    foreach (qw(Order Sort)) {
        if (!$Param{$_}) {
            $Self->{LayoutObject}->FatalError(Message => "Need parameter $_!")
        }
    }
    my @CategoryIDs = ();
    if ($Param{Mode} && $Param{Mode} eq 'Agent') {
        @CategoryIDs = @{$Self->{FAQObject}->AgentCategorySearch(
            ParentID => $Param{CategoryID},
            UserID => $Self->{UserID},
        )};
    }
    elsif ($Param{Mode} && $Param{Mode} eq 'Customer') {
        @CategoryIDs = @{$Self->{FAQObject}->CustomerCategorySearch(
            ParentID => $Param{CategoryID},
            CustomerUser => $Param{CustomerUser},
        )};
    }
    else {
        @CategoryIDs = @{$Self->{FAQObject}->PublicCategorySearch(
            ParentID => $Param{CategoryID},
        )};
    }

    if(@CategoryIDs) {
        $Self->{LayoutObject}->Block(
            Name => 'ExplorerCategoryList',
        );
        foreach (@CategoryIDs) {
            my %Data = $Self->{FAQObject}->CategoryGet(CategoryID => $_);
            $Data{CategoryNumber} = $Self->{FAQObject}->CategoryCount(
                ParentIDs => [$_]
            );
            $Data{ArticleNumber} = $Self->{FAQObject}->FAQCount(
                CategoryIDs => [$_],
                ItemStates => $Self->{InterfaceStates}
            );

            # css configuration
            if($Frontend{CssRow} eq 'searchpassive') {
                $Frontend{CssRow} = 'searchactive';
            }
            else {
                $Frontend{CssRow} = 'searchpassive';
            }
            $Self->{LayoutObject}->Block(
                Name => 'ExplorerCategoryRow',
                Data => { %Data, %Frontend },
            );
        }
    }
    return;
}

sub _GetExplorerItemList {
    my $Self = shift;
    my %Param = @_;

    # check needed parameters
    foreach (qw(Order Sort)) {
        if (!$Param{$_}) {
            $Self->{LayoutObject}->FatalError(Message => "Need parameter $_!")
        }
    }

    my $CssRow = '';
    my @ItemIDs = $Self->{FAQObject}->FAQSearch(
        CategoryIDs => [$Param{CategoryID}],
        States => $Self->{InterfaceStates},
        Order => $Param{Order},
        Sort => $Param{Sort},
        Limit => 300,
    );

    if (@ItemIDs) {
        $Self->{LayoutObject}->Block(
            Name => 'ExplorerFAQItemList',
            Data => {%Param}
        );
        foreach (@ItemIDs) {
            my %Frontend = ();
            my %Data = $Self->{FAQObject}->FAQGet(ItemID => $_);

            # css configuration
            if($CssRow eq 'searchpassive') {
                $CssRow = 'searchactive';
            }
            else {
                $CssRow = 'searchpassive';
            }
            $Frontend{CssRow} = $CssRow;
            $Frontend{CssColumnVotingResult} = 'color:'.$Self->{LayoutObject}->GetFAQItemVotingRateColor(Rate => $Data{Result}).';';

            $Self->{LayoutObject}->Block(
                Name => 'ExplorerFAQItemRow',
                Data => { %Data, %Frontend },
            );
        }
    }
    return;
}

sub _GetExplorerLastChangeItems {
    my $Self = shift;
    my %Param = @_;
    if($Self->{ConfigObject}->Get('FAQ::Explorer::LastChange::Show')) {
        # check needed parameters
        foreach (qw(CategoryID)) {
            if (!defined($Param{$_})) {
                $Self->{LayoutObject}->FatalError(Message => "Need parameter $_!")
            }
        }
        my @ItemIDs = ();
        if(defined($Param{CategoryID})) {
            # add current categoryid
            my @CategoryIDs = ();
            if($Param{CategoryID}) {
                push(@CategoryIDs, ($Param{CategoryID}));
            }
            if (!defined($Param{Mode})) {
                $Param{Mode} = '';
            }
            if (!defined($Param{CustomerUser})) {
                $Param{CustomerUser} = '';
            }
            # add subcategoryids
            if($Self->{ConfigObject}->Get('FAQ::Explorer::LastChange::ShowSubCategoryItems')) {
                my @SubCategoryIDs = @{$Self->{FAQObject}->CategorySubCategoryIDList(
                    ParentID => $Param{CategoryID},
                    ItemStates => $Self->{InterfaceStates},
                    Mode => $Param{Mode},
                    CustomerUser => $Param{CustomerUser},
                    UserID => $Self->{UserID},
                )};
                push(@CategoryIDs, @SubCategoryIDs);
            }
            $Self->{LayoutObject}->Block(
                Name => 'ExplorerLatestChange'
            );
            if (@CategoryIDs) {
                @ItemIDs = $Self->{FAQObject}->FAQSearch(
                    CategoryIDs => \@CategoryIDs,
                    States => $Self->{InterfaceStates},
                    Order => 'Changed',
                    Sort => 'down',
                    Limit => $Self->{ConfigObject}->Get('FAQ::Explorer::LastChange::Limit')
                );
            }
        }
        else {
            $Self->{LayoutObject}->Block(
                Name => 'ExplorerLatestChange'
            );
            @ItemIDs = $Self->{FAQObject}->FAQSearch(
                States => $Self->{InterfaceStates},
                Order => 'Changed',
                Sort => 'down',
                Limit => $Self->{ConfigObject}->Get('FAQ::Explorer::LastChange::Limit')
            );
        }
        foreach (@ItemIDs) {
            my %Data = $Self->{FAQObject}->FAQGet(ItemID => $_);
            $Self->{LayoutObject}->Block(
                Name => 'ExplorerLatestChangeFAQItemRow',
                Data => { %Data },
            );
        }
        return 1;
    }
    return 0;
}

sub _GetExplorerLastCreateItems {
    my $Self = shift;
    my %Param = @_;

    if($Self->{ConfigObject}->Get('FAQ::Explorer::LastCreate::Show')) {
        # check needed parameters
        foreach (qw(CategoryID)) {
            if (!defined($Param{$_})) {
                $Self->{LayoutObject}->FatalError(Message => "Need parameter $_!")
            }
        }
        my @ItemIDs = ();
        if(defined($Param{CategoryID})) {
            # add current categoryid
            my @CategoryIDs = ();
            if($Param{CategoryID}) {
                push(@CategoryIDs, ($Param{CategoryID}));
            }
            if (!defined($Param{Mode})) {
                $Param{Mode} = '';
            }
            if (!defined($Param{CustomerUser})) {
                $Param{CustomerUser} = '';
            }
            # add subcategoryids
            if($Self->{ConfigObject}->Get('FAQ::Explorer::LastCreate::ShowSubCategoryItems')) {
                my @SubCategoryIDs = @{$Self->{FAQObject}->CategorySubCategoryIDList(
                    ParentID => $Param{CategoryID},
                    ItemStates => $Self->{InterfaceStates},
                    Mode => $Param{Mode},
                    CustomerUser => $Param{CustomerUser},
                    UserID => $Self->{UserID},
                )};
                push(@CategoryIDs, @SubCategoryIDs);
            }
            $Self->{LayoutObject}->Block(
                Name => 'ExplorerLatestCreate'
            );
            if (@CategoryIDs) {
                @ItemIDs = $Self->{FAQObject}->FAQSearch(
                    CategoryIDs => \@CategoryIDs,
                    States => $Self->{InterfaceStates},
                    Order => 'Created',
                    Sort => 'down',
                    Limit => $Self->{ConfigObject}->Get('FAQ::Explorer::LastCreate::Limit')
                );
            }
        }
        else {
            $Self->{LayoutObject}->Block(
                Name => 'ExplorerLatestCreate'
            );
            @ItemIDs = $Self->{FAQObject}->FAQSearch(
                States => $Self->{InterfaceStates},
                Order => 'Created',
                Sort => 'down',
                Limit => $Self->{ConfigObject}->Get('FAQ::Explorer::LastCreate::Limit')
            );
        }
        # dtl block
        foreach (@ItemIDs) {
            my %Data = $Self->{FAQObject}->FAQGet(ItemID => $_);
            $Self->{LayoutObject}->Block(
                Name => 'ExplorerLatestCreateFAQItemRow',
                Data => { %Data },
            );
        }
        return 1;
    }
    return 0;
}

sub _GetExplorerQuickSearch {
    my $Self = shift;
    my %Param = @_;

    if($Self->{ConfigObject}->Get('FAQ::Explorer::QuickSearch::Show')) {
        # check needed parameters
        foreach (qw()) {
            if (!$Param{$_}) {
                $Self->{LayoutObject}->FatalError(Message => "Need parameter $_!")
            }
        }
        # dtl block
        $Self->{LayoutObject}->Block(
            Name => 'ExplorerQuickSearch',
            Data => { CategoryID => $Param{CategoryID} }
        );
        return 1;
    }
    return 0;
}

sub GetItemView {
    my $Self = shift;
    my %Param = @_;
    my %Frontend = ();
    my %GetParam = ();
    my @Params = qw(ItemID);

    # manage parameters
    foreach (@Params) {
        if (!($GetParam{$_} = $Self->{ParamObject}->GetParam(Param => $_))) {
            $Self->{LayoutObject}->FatalError(Message => "Need parameter $_");
        }
    }
    # db action
    my %ItemData = $Self->{FAQObject}->FAQGet(ItemID => $GetParam{ItemID}, UserID => $Self->{UserID});
    # html quoting
    foreach my $Key (qw (Field1 Field2 Field3 Field4 Field5 Field6)) {
        if ($Self->{ConfigObject}->Get('FAQ::Item::HTML')) {
            my @Array = split /pre>/, $ItemData{$Key};
            my $Text = '';
            foreach (@Array) {
                if ($_ =~ /(.*)\<\/$/) {
                    $Text .= 'pre>'.$_.'pre>';
                }
                else {
                    $_ =~ s/\n/\<br\>/g;
                    $Text .= $_;
                }
            }
            $ItemData{$Key} = $Text;
        }
        else {
            $ItemData{$Key} = $Self->{LayoutObject}->Ascii2Html(
                NewLine => 0,
                Text => $ItemData{$Key},
                VMax => 5000,
                HTMLResultMode => 1,
                LinkFeature => 1,
            );
        }
    }
    if (!%ItemData) {
        return $Self->{LayoutObject}->ErrorScreen();
    }
    # permission check
    if (!exists($Self->{InterfaceStates}{$ItemData{StateTypeID}})) {
        return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
    }
    # user info
    my %UserInfo = $Self->{AgentUserObject}->GetUserData(
        UserID => $ItemData{CreatedBy},
    );
    $Frontend{CreatedByLogin} = $UserInfo{UserLogin};
    %UserInfo = $Self->{AgentUserObject}->GetUserData(
        UserID => $ItemData{ChangedBy},
    );
    $Frontend{ChangedByLogin} = $UserInfo{UserLogin};
    # item view
    $Frontend{CssColumnVotingResult} = 'color:'.$Self->{LayoutObject}->GetFAQItemVotingRateColor(Rate => $ItemData{Result}).';';
    $Self->{LayoutObject}->Block(
        Name => 'View',
        Data => { %Param, %ItemData, %Frontend },
    );
    if ($Param{Permission} && $Param{Permission} eq 'rw') {
        $Self->{LayoutObject}->Block(
            Name => 'FAQItemViewLinkUpdate',
            Data => { %Param, %ItemData, %Frontend },
        );
        $Self->{LayoutObject}->Block(
            Name => 'FAQItemViewLinkDelete',
            Data => { %Param, %ItemData, %Frontend },
        );
    }
    # FAQ path
    if ($Self->_GetFAQPath(CategoryID => $ItemData{CategoryID})) {
        $Self->{LayoutObject}->Block(
            Name => 'FAQPathItemElement',
            Data => \%ItemData,
        );
    }
    # item attachment
    if (defined($ItemData{Filename})) {
        $Self->{LayoutObject}->Block(
            Name => 'FAQItemViewAttachment',
            Data => { %Param, %ItemData },
        );
    }
    # item fields
    $Self->_GetItemFields(
        ItemData => \%ItemData
    );
    # item voting
    my %ShowItemVoting = %{$Self->{ConfigObject}->Get('FAQ::Item::Voting::Show')};
    if (exists($ShowItemVoting{$Self->{Interface}{Name}})) {
        $Self->_GetItemVoting(
            ItemData => \%ItemData
        );
    }
    if ($Param{Links} && $Param{Links} == 1) {
        # get linked objects
        my %Links = $Self->{LinkObject}->AllLinkedObjects(
            Object => 'FAQ',
            ObjectID => $ItemData{ItemID},
            UserID => $Self->{UserID},
        );
        $Self->{LayoutObject}->Block(
            Name => "Links",
            Data => {},
        );
        my %LinkTypeBox = ();
        for my $LinkType ( qw(Normal Parent Child) ) {
            if ( ! $Links{$LinkType} ) {
                next;
            }
            my %ObjectType = %{$Links{$LinkType}};
            foreach my $Object (sort keys %ObjectType) {
                my %Data = %{$ObjectType{$Object}};
                foreach my $Item (sort keys %Data) {
                    if ( !$LinkTypeBox{$LinkType} ) {
                        $Self->{LayoutObject}->Block(
                            Name => 'Link',
                            Data => {
                                %Param,
                                LinkType => $LinkType,
                            },
                        );
                        $LinkTypeBox{$LinkType} = 1;
                    }
                    $Self->{LayoutObject}->Block(
                        Name => 'LinkItem',
                        Data => {
                            %{ $Data{$Item} },
                            LinkType => $LinkType,
                        },
                    );
                }
            }
        }
    }
    return;
}

sub GetItemSmallView {
    my $Self = shift;
    my %Param = @_;
    my %Frontend = ();
    my %GetParam = ();
    my @Params = qw(ItemID);

    # manage parameters
    foreach (@Params) {
        if (!($GetParam{$_} = $Self->{ParamObject}->GetParam(Param => $_))) {
            $Self->{LayoutObject}->FatalError(Message => "Need parameter $_");
        }
    }
    # db action
    my %ItemData = $Self->{FAQObject}->FAQGet(ItemID => $GetParam{ItemID}, UserID => $Self->{UserID});
    if (!%ItemData) {
        return $Self->{LayoutObject}->ErrorScreen();
    }
    # permission check
    if(!exists($Self->{InterfaceStates}{$ItemData{StateTypeID}})) {
        return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
    }
    # user info
    my %UserInfo = $Self->{AgentUserObject}->GetUserData(
        UserID => $ItemData{CreatedBy},
    );
    $Frontend{CreatedByLogin} = $UserInfo{UserLogin};
    %UserInfo = $Self->{AgentUserObject}->GetUserData(
        UserID => $ItemData{ChangedBy},
    );
    $Frontend{ChangedByLogin} = $UserInfo{UserLogin};
    # item view
    $Frontend{CssColumnVotingResult} = 'color:'.$Self->{LayoutObject}->GetFAQItemVotingRateColor(Rate => $ItemData{Result}).';';
    $Frontend{ItemFieldValues} = $Self->_GetItemFieldValues(ItemData => \%ItemData);
    $Self->{LayoutObject}->Block(
        Name => 'ViewSmall',
        Data => { %Param, %ItemData, %Frontend },
    );
    # FAQ path
    if ($Self->_GetFAQPath(CategoryID => $ItemData{CategoryID})) {
        $Self->{LayoutObject}->Block(
            Name => 'FAQPathItemElement',
            Data => \%ItemData,
        );
    }
    # item attachment
    if (defined($ItemData{Filename})) {
        $Self->{LayoutObject}->Block(
            Name => 'FAQItemViewAttachment',
            Data => { %Param, %ItemData },
        );
    }
    # get linked objects
    my %Links = $Self->{LinkObject}->AllLinkedObjects(
        Object => 'FAQ',
        ObjectID => $ItemData{ItemID},
        UserID => $Self->{UserID},
    );
    foreach my $LinkType (sort keys %Links) {
        my %ObjectType = %{$Links{$LinkType}};
        foreach my $Object (sort keys %ObjectType) {
            my %Data = %{$ObjectType{$Object}};
            foreach my $Item (sort keys %Data) {
                $Self->{LayoutObject}->Block(
                    Name => "Link$LinkType",
                    Data => $Data{$Item},
                );
            }
        }
    }
    return;
}

sub GetItemPrint {
    my $Self = shift;
    my %Param = @_;
    my %GetParam = ();
    my @Params = qw(ItemID);

    # manage parameters
    foreach (@Params) {
        if (!($GetParam{$_} = $Self->{ParamObject}->GetParam(Param => $_))) {
            $Self->{LayoutObject}->FatalError(Message => "Need parameter $_");
        }
    }
    # db action
    my %ItemData = $Self->{FAQObject}->FAQGet(ItemID => $GetParam{ItemID}, UserID => $Self->{UserID});
    if (!%ItemData) {
        return $Self->{LayoutObject}->ErrorScreen();
    }
    # html quoting
    foreach my $Key (qw (Field1 Field2 Field3 Field4 Field5 Field6)) {
        if ($Self->{ConfigObject}->Get('FAQ::Item::HTML')) {
            $ItemData{$Key} =~ s/\n/\<br\>/g;
        }
        else {
            $ItemData{$Key} = $Self->{LayoutObject}->Ascii2Html(
                NewLine => 0,
                Text => $ItemData{$Key},
                VMax => 5000,
                HTMLResultMode => 1,
                LinkFeature => 1,
            );
        }
    }
    # permission check
    if (!exists($Self->{InterfaceStates}{$ItemData{StateTypeID}})) {
        return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
    }
    # add article
    $Self->{LayoutObject}->Block(
        Name => 'Print',
        Data => \%ItemData,
    );
    # fields
    $Self->_GetItemFields(
        ItemData => \%ItemData
    );
    # links
    if ($Param{Links} && $Param{Links} == 1) {
        $Self->{LayoutObject}->Block(
            Name => 'Links',
            Data => {},
        );
        # get linked objects
        my %Links = $Self->{LinkObject}->AllLinkedObjects(
            Object => 'FAQ',
            ObjectID => $ItemData{ItemID},
            UserID => $Self->{UserID},
        );
        # links
        foreach my $LinkType (sort keys %Links) {
            my %ObjectType = %{$Links{$LinkType}};
            foreach my $Object (sort keys %ObjectType) {
                my %Data = %{$ObjectType{$Object}};
                foreach my $Item (sort keys %Data) {
                    $Self->{LayoutObject}->Block(
                        Name => "Link$LinkType",
                        Data => $Data{$Item},
                    );
                }
            }
        }
    }
    return;
}

sub _GetItemFields {
    my $Self = shift;
    my %Param = @_;
    my %GetParam = ();
    my @Params = qw(ItemData);

    # manage parameters
    foreach (@Params) {
        if(!exists($Param{$_})) {
            return $Self->{LayoutObject}->FatalError(Message => "Need parameter $_");
        }
    }
    # config values
    my %ItemFields = ();
    for(my $i=1;$i<7;$i++) {
        my %ItemConfig = %{$Self->{ConfigObject}->Get('FAQ::Item::Field'.$i)};
        if($ItemConfig{Show}) {
            $ItemFields{"Field".$i} = \%ItemConfig;
        }
    }
    foreach my $Key (sort( { $ItemFields{$a}{Prio} <=> $ItemFields{$b}{Prio} } keys(%ItemFields))) {
        my %StateTypeData = %{$Self->{FAQObject}->StateTypeGet(
            Name => $ItemFields{$Key}{Show}
        )};
        # show yes /no
        if (exists($Self->{InterfaceStates}{$StateTypeData{ID}})) {
            $Self->{LayoutObject}->Block(
                Name => 'FAQItemField',
                Data => {
                    %{$ItemFields{$Key}},
                    'StateName' => $StateTypeData{Name},
                    'Key' => $Key,
                    'Value' => $Param{ItemData}{$Key} || '',
                },
            );
        }
    }
    return;
}

sub _GetItemFieldValues {
    my $Self = shift;
    my %Param = @_;
    my %GetParam = ();
    my @Params = qw(ItemData);

    # manage parameters
    foreach (@Params) {
        if (!exists($Param{$_})) {
            return $Self->{LayoutObject}->FatalError(Message => "Need parameter $_");
        }
    }
    # config values
    my %ItemFields = ();
    for (my $i=1;$i<7;$i++) {
        my %ItemConfig = %{$Self->{ConfigObject}->Get('FAQ::Item::Field'.$i)};
        if($ItemConfig{Show}) {
            $ItemFields{"Field".$i} = \%ItemConfig;
        }
    }
    my $String = '';
    foreach my $Key (sort( { $ItemFields{$a}{Prio} <=> $ItemFields{$b}{Prio} } keys(%ItemFields))) {
        my %StateTypeData = %{$Self->{FAQObject}->StateTypeGet(
            Name => $ItemFields{$Key}{Show}
        )};
        # show yes /no
        if(exists($Self->{InterfaceStates}{$StateTypeData{ID}})) {
            $String .= $Param{ItemData}{$Key} || '';
            $String .= "\n\n";
        }
    }
    return $String;
}

sub _GetItemVoting {
    my $Self = shift;
    my %Param = @_;

    # check needed parameters
    foreach (qw(ItemData)) {
        if (!$Param{$_}) {
            $Self->{LayoutObject}->FatalError(Message => "Need parameter $_!");
        }
    }
    my %ItemData = %{$Param{ItemData}};

    $Self->{LayoutObject}->Block(
        Name => "Voting"
    );
    my %VoteData = %{$Self->{FAQObject}->VoteGet(
        CreateBy => $Self->{UserID},
        ItemID => $ItemData{ItemID},
        Interface => $Self->{Interface}{ID},
        IP => $ENV{'REMOTE_ADDR'},
    )};

    my $Flag = 0;
    # already voted?
    if(%VoteData) {
        # item/change_time > voting/create_time
        my $ItemChangedSystemTime = $Self->{TimeObject}->TimeStamp2SystemTime(String => $ItemData{Changed} || '');
        my $VoteCreatedSystemTime = $Self->{TimeObject}->TimeStamp2SystemTime(String => $VoteData{Created} || '');

        if ($ItemChangedSystemTime > $VoteCreatedSystemTime) {
            $Flag = 1;
        }
        else {
            push(@{$Self->{Notify}}, ['Info', 'You have already voted!']);
            return;
        }
    }
    else {
        $Flag = 1;
    }
    if ($Self->{Subaction} eq 'Vote' && $Flag) {
        # check needed parameters
        foreach (qw(ItemData)) {
            if (!$Param{$_}) {
                $Self->{LayoutObject}->FatalError(Message => "Need parameter $_!")
            }
        }
        # manage parameters
        my %GetParam = ();
        my @Params = qw(ItemID Rate);
        foreach (@Params) {
            $GetParam{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        if($GetParam{Rate} eq '0' or $GetParam{Rate}) {
            $Self->{FAQObject}->VoteAdd(
                CreatedBy => $Self->{UserID},
                ItemID => $GetParam{ItemID},
                IP => $ENV{'REMOTE_ADDR'},
                Interface => $Self->{Interface}{ID},
                Rate => $GetParam{Rate},
            );
            push(@{$Self->{Notify}}, ['Info', 'Thanks for vote!']);
            return;

        }
        else {
            push(@{$Self->{Notify}}, ['Error', 'No rate selected!']);
            $Self->GetItemVotingForm(
                ItemData => $Param{ItemData}
            );

            return;
        }
    }

    # form
    $Self->_GetItemVotingForm(
        ItemData => $Param{ItemData}
    );

    return;
}

sub _GetItemVotingForm {
    my $Self = shift;
    my %Param = @_;

    # check needed parameters
    foreach (qw(ItemData)) {
        if (!$Param{$_}) {
            $Self->{LayoutObject}->FatalError(Message => "Need parameter $_!")
        }
    }
    $Self->{LayoutObject}->Block(
        Name => 'VoteForm',
        Data => { %Param, %{$Param{ItemData}} }
    );

    my %VotingRates = %{$Self->{ConfigObject}->Get('FAQ::Item::Voting::Rates')};
    foreach my $key ( sort( { $b <=> $a } keys(%VotingRates ) ) ) {
        my %Data = ("Value"=>$key, "Title"=>$VotingRates{$key} );
        $Self->{LayoutObject}->Block(
            Name => "VotingRateRow",
            Data => \%Data,
        );
    }
    return;
}

sub GetItemSearch {
    my $Self = shift;
    my %Param = @_;
    my %GetParam    = ();
    my %Frontend    = ();

    # get params
    foreach (qw(LanguageIDs CategoryIDs)) {
        my @Array = $Self->{ParamObject}->GetArray(Param => $_);
        if (@Array) {
            $GetParam{$_} = \@Array;
        }
    }
    foreach (qw(QuickSearch Number Title What Keyword)) {
        $GetParam{$_} = $Self->{ParamObject}->GetParam(Param => $_);
    }
    # quicksearch in subcategories?
    if ($GetParam{QuickSearch}) {
        if ($Self->{ConfigObject}->Get('FAQ::Explorer::QuickSearch::ShowSubCategoryItems')) {
            if (!$Param{Mode}) {
                $Param{Mode} = 'Public';
            }
            if (!defined($Param{User})) {
                $Param{User} = '';
            }
            my @SubCategoryIDs = @{$Self->{FAQObject}->CategorySubCategoryIDList(
                ParentID => $GetParam{CategoryIDs}->[0],
                ItemStates => $Self->{InterfaceStates},
                CustomerUser => $Param{CustomerUser},
                UserID => $Param{User},
                Mode => $Param{Mode},
            )};
            push(@{$GetParam{CategoryIDs}}, @SubCategoryIDs);
        }
    }
    $Frontend{LanguageOption} = $Self->{LayoutObject}->OptionStrgHashRef(
        Data => { $Self->{FAQObject}->LanguageList() },
        Size => 5,
        Name => 'LanguageIDs',
        Multiple => 1,
        SelectedIDRefArray => $GetParam{LanguageIDs} || [],
        HTMLQuote => 1,
        LanguageTranslation => 0,
    );
    my $Categories = ();
    if ($Param{Mode} && $Param{Mode} eq 'Agent') {
        $Categories = $Self->{FAQObject}->GetUserCategories(
            UserID =>  $Self->{UserID},
            Type => 'rw'
        );
    }
    elsif ($Param{Mode} && $Param{Mode} eq 'Customer') {
        $Categories = $Self->{FAQObject}->GetCustomerCategories(
            CustomerUser =>  $Param{CustomerUser},
            Type => 'rw'
        );
    }
    else {
        $Categories = $Self->{FAQObject}->CategoryList();
    }
    $Frontend{CategoryOption} = $Self->{LayoutObject}->AgentFAQCategoryListOption(
        CategoryList => { %{$Categories} },
        Size => 5,
        Name => 'CategoryIDs',
        Multiple => 1,
        SelectedIDs => $GetParam{CategoryIDs} || [],
        HTMLQuote => 1,
        LanguageTranslation => 0,
    );
    $Self->{LayoutObject}->Block(
        Name => 'Search',
        Data => { %Param, %GetParam, %Frontend },
    );

    # build result
    if ($Self->{ParamObject}->GetParam(Param => 'Submit')) {
        my $CssRow = '';

        my @ItemIDs = $Self->{FAQObject}->FAQSearch(
            %Param,
            %GetParam,
            States => $Self->{InterfaceStates},
            Limit => 25,
        );
        $Self->{LayoutObject}->Block(
            Name => 'SearchResult',
            Data => { %Param, %Frontend },
        );
        foreach (@ItemIDs) {
            %Frontend = ();
            my %Data = $Self->{FAQObject}->FAQGet(ItemID => $_);
            my $Permission = 'ro';
            if ($Param{Mode} && $Param{Mode} eq 'Agent') {
                $Permission = $Self->{FAQObject}->CheckCategoryUserPermission(
                    UserID => $Param{User},
                    CategoryID => $Data{CategoryID},
                );
            }
            elsif ($Param{Mode} && $Param{Mode} eq 'Customer') {
                $Permission = $Self->{FAQObject}->CheckCategoryCustomerPermission(
                    CustomerUser => $Param{CustomerUser},
                    CategoryID => $Data{CategoryID},
                );
            }
            if ($Permission ne '') {
                if($CssRow eq 'searchpassive') {
                    $CssRow = 'searchactive';
                }
                else {
                    $CssRow = 'searchpassive';
                }
                $Data{CssRow} = $CssRow;
                $Frontend{CssColumnVotingResult} = 'color:'.$Self->{LayoutObject}->GetFAQItemVotingRateColor(Rate => $Data{Result}).';';

                $Self->{LayoutObject}->Block(
                    Name => 'SearchResultRow',
                    Data => { %Data, %Frontend },
                );
            }
        }
    }
    return;
}

sub GetSystemHistory {
    my $Self = shift;
    my %Param = @_;
    my %Frontend = ();

    # store last queue screen
    $Self->{SessionObject}->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key => 'LastScreenOverview',
        Value => $Self->{RequestedURL},
    );
    # store last screen
    $Self->{SessionObject}->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key => 'LastScreenView',
        Value => $Self->{RequestedURL},
    );
    $Self->{LayoutObject}->Block(
        Name => 'SystemHistory',
        Data => { %Param },
    );

    $Frontend{CssRow} = '';
    my @History = @{$Self->{FAQObject}->HistoryGet()};
    foreach my $Row (@History) {
        # css configuration
        if($Frontend{CssRow} eq 'searchpassive') {
            $Frontend{CssRow} = 'searchactive';
        }
        else {
            $Frontend{CssRow} = 'searchpassive';
        }
        my %Data = %{$Row};#$Self->{FAQObject}->FAQGet(ItemID => $Row->{ItemID});
        my %User = $Self->{UserObject}->GetUserData(
            UserID => $Row->{CreatedBy},
            Cached => 1,
        );
        $Self->{LayoutObject}->Block(
            Name => 'SystemHistoryRow',
            Data => { %Data, %Frontend, %User, Name => $Row->{Name} },
        );
    }
    return;
}

1;

# --
# Kernel/Modules/AgentFAQ.pm - faq module
# Copyright (C) 2001-2007 OTRS GmbH, http://otrs.org/
# --
# $Id: AgentFAQ.pm,v 1.11 2007/09/26 11:55:34 rk Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl.txt.
# --

package Kernel::Modules::AgentFAQ;

use strict;
use Kernel::System::FAQ;
use Kernel::System::LinkObject;
use Kernel::Modules::FAQ;
use Kernel::System::User;
use Kernel::System::Group;
use Kernel::System::Valid;

use vars qw($VERSION @ISA);
$VERSION = '$Revision: 1.11 $';
$VERSION =~ s/^\$.*:\W(.*)\W.+?$/$1/;

@ISA = qw(Kernel::Modules::FAQ);

sub new {
    my $Type = shift;
    my %Param = @_;

    # allocate new hash for object
    # ********************************************************** #
    my $Self = new Kernel::Modules::FAQ(%Param);
    bless ($Self, $Type);

    # interface settings
    # ********************************************************** #
    $Self->{Interface} = $Self->{FAQObject}->StateTypeGet(
        Name => 'internal'
    );
    $Self->{InterfaceStates} = $Self->{FAQObject}->StateTypeList(
        Types => ['internal','external','public']
    );

    # check needed Objects
    # ********************************************************** #
    foreach (qw(SessionObject)) {
        $Self->{LayoutObject}->FatalError(Message => "Got no $_!") if (!$Self->{$_});
    }
    $Self->{ValidObject} = Kernel::System::Valid->new(%Param);
    $Self->{UserObject} = Kernel::System::User->new(%Param);
    $Self->{GroupObject} = Kernel::System::Group->new(%Param);
    return $Self;
}

sub Run {

    my $Self = shift;
    my %Param = @_;

    # Paramter
    my @Params = ();
    my %GetParam = ();
    my %Frontend = ();
    my %Data = ();

    # Output
    my $Output = '';
    my $Header = '';
    my $HeaderTitle =  '';
    my $HeaderType =  '';
    my $Navigation = '';
    my $Notify = '';
    my $Content = '';
    my $Footer = '';
    my $FooterType =  '';

    my $DefaultHeader = '';
    my $DefaultNavigation = '';
    my $DefaultNotify = '';
    my $DefaultContent = '';
    my $DefaultFooter = '';

    # manage parameters
    # ********************************************************** #
    @Params = qw(ItemID ID Number Name Comment CategoryID ParentID StateID LanguageID Title UserID
    Field1 Field2 Field3 Field4 Field5 Field6 FreeKey1 FreeKey2 FreeKey3 Keywords Order Sort Nav);
    foreach (@Params) {
        $GetParam{$_} = $Self->{ParamObject}->GetParam(Param => $_) || '';
    }

    # navigation ON/OFF
    # ********************************************************** #
    $HeaderType = $Self->{LastFAQNav};
    if($GetParam{Nav} eq 'Normal') {
        $HeaderType = '';
    } elsif ($GetParam{Nav} eq 'None' || $Self->{LastFAQNav}) {
        $HeaderType = 'Small';
        $Navigation = ' ';
        $Notify    = ' ';
    } else {
        $HeaderType = '';
    }

    $Self->{SessionObject}->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key => 'LastFAQNav',
        Value => $HeaderType,
    );

    # store nav param
    $Self->{SessionObject}->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key => 'LastFAQNav',
        Value => $HeaderType,
    );

    # view small
    #$HeaderType = $Self->{LastFAQNav} || '';
    #my $Nav = $Self->{ParamObject}->GetParam(Param => 'Nav') || '';

    # output
    $Output  = '';

    # store search params
    $Self->{SessionObject}->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key => 'LastFAQWhat',
        Value => $Param{What},
    );
    $Self->{SessionObject}->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key => 'LastFAQKeyword',
        Value => $Param{Keyword},
    );

    # ---------------------------------------------------------- #
    # language add
    # ---------------------------------------------------------- #
    if ($Self->{Subaction} eq 'Language') {

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

        # dtl
        $Frontend{Headline} = 'Add';
        $Frontend{AddLink} = '(Click here to add)';
        $Frontend{Subaction} = 'LanguageAdd';
        $Frontend{LanguageOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => { $Self->{FAQObject}->LanguageList(UserID => $Self->{UserID}) },
            Size => 10,
            Name => 'ID',
            HTMLQuote => 1,
            LanguageTranslation => 0,
        );

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

    }

    # ---------------------------------------------------------- #
    # language add action
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'LanguageAction') {

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

        #get Data
        my %ParamData = ();
        $ParamData{UserID} = $Self->{UserID};
        foreach(qw(Name)) {
            if(!($ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_))) {
                $Self->{LayoutObject}->FatalError(Message => "Need $_ !");
            }
        }

        # db action
        if(!$Self->{FAQObject}->LanguageDuplicateCheck(Name=>$ParamData{Name})) {
            if ($Self->{FAQObject}->LanguageAdd(%ParamData, UserID => $Self->{UserID})) {
                return $Self->{LayoutObject}->Redirect(OP => "Action=AgentFAQ&Subaction=Language");
            }
            else {
                return $Self->{LayoutObject}->ErrorScreen();
            }
        }
        else {
            $Self->{LayoutObject}->FatalError(Message => "Language '$ParamData{Name}' already exists!");
        }

    }

    # ---------------------------------------------------------- #
    # language update
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'LanguageChange') {

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

        #get Data
        my %ParamData = ();
        foreach(qw(ID)) {
            if(!($ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_))) {
                $Self->{LayoutObject}->FatalError(Message => "Need $_ !");
            }
        }

        # db action
        my %LanguageData = $Self->{FAQObject}->LanguageGet(ID => $ParamData{ID}, UserID => $Self->{UserID});
        if (!%LanguageData) {
            return $Self->{LayoutObject}->ErrorScreen();
        }

        # dtl
        $Frontend{Headline} = 'Change';
        $Frontend{AddLink} = '';
        $Frontend{Subaction} = 'LanguageChange';
        $Frontend{LanguageOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => { $Self->{FAQObject}->LanguageList(UserID => $Self->{UserID}) },
            Size => 10,
            Name => 'ID',
            SelectedID => $LanguageData{LanguageID},
            HTMLQuote => 1,
            LanguageTranslation => 0,
        );

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

        $Content = $Self->{LayoutObject}->Output(
            Data => {%Param, %Frontend, %LanguageData},
            TemplateFile => 'AgentFAQ',
        );

    }

    # ---------------------------------------------------------- #
    # language update action
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'LanguageChangeAction') {

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

        # get Data
        my %ParamData = ();
        $ParamData{UserID} = $Self->{UserID};
        foreach(qw(ID Name)) {
            if(!($ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_))) {
                $Self->{LayoutObject}->FatalError(Message => "Need $_ !");
            }
        }

        # duplicate check
        if(!$Self->{FAQObject}->LanguageDuplicateCheck(Name=>$ParamData{Name},ID=>$ParamData{ID})) {
            # db action
            if (!$Self->{FAQObject}->LanguageUpdate(%ParamData, UserID => $Self->{UserID})) {
                return $Self->{LayoutObject}->Redirect(OP => "Action=AgentFAQ&Subaction=Language");
            }
            else {
                return $Self->{LayoutObject}->ErrorScreen();
            }
        }
        else {
            $Self->{LayoutObject}->FatalError(Message => "Language '$ParamData{Name}' already exists!");
        }

    }

    # ---------------------------------------------------------- #
    # category update
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'CategoryChange') {

        @Params = qw(CategoryID);

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

        # ceck parameters
        my %ParamData = ();
        foreach(qw(CategoryID)) {
            if(!($ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_))) {
                $Self->{LayoutObject}->FatalError(Message => "Need $_ !");
            }
        }

        # db action
        my %CategoryData = $Self->{FAQObject}->CategoryGet(CategoryID => $ParamData{CategoryID}, UserID => $Self->{UserID});
        if (!%CategoryData) {
            return $Self->{LayoutObject}->ErrorScreen();
        }

        # dtl
        $Frontend{CategoryLongOption} = $Self->{LayoutObject}->AgentFAQCategoryListOption(
            CategoryList => { %{$Self->{FAQObject}->CategoryList(UserID => $Self->{UserID})} },
            Size => 10,
            Name => 'CategoryID',
            SelectedID => $CategoryData{CategoryID},
            HTMLQuote => 1,
            LanguageTranslation => 0,
        );
        $Frontend{CategoryOption} = $Self->{LayoutObject}->AgentFAQCategoryListOption(
            CategoryList => { %{$Self->{FAQObject}->CategoryList(UserID => $Self->{UserID})} },
            Size => 1,
            Name => 'ParentID',
            SelectedID => $CategoryData{ParentID},
            HTMLQuote => 1,
            LanguageTranslation => 0,
            RootElement => 1,
        );

        # build ValidID string
        $Frontend{ValidOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => {$Self->{ValidObject}->ValidList()},
            Name => 'ValidID',
            SelectedID => $CategoryData{ValidID},
        );
        my $SelectedGroups = $Self->{FAQObject}->GetCategoryGroup(CategoryID => $ParamData{CategoryID});
        my %Groups = $Self->{GroupObject}->GroupList(Valid => 1);
        $Frontend{GroupOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => \%Groups,
            Name => 'PermissionGroups',
            Size => 8,
            Multiple => 1,
            SelectedIDRefArray => $SelectedGroups,
        );
        $Param{Headline} = 'Update';
        $Param{FormSubaction} = 'CategoryChangeAction';
        $Self->{LayoutObject}->Block(
            Name => 'Category',
            Data => { %Param, %Frontend, %CategoryData },
        );
        $Self->{LayoutObject}->Block(
            Name => 'CategoryAddLink',
            Data => { %Param },
        );

    }
    # ---------------------------------------------------------- #
    # category update action
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'CategoryChangeAction') {
        # permission check
        if (!$Self->{AccessRw}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        # check parameters
        my %ParamData = ();
        my @RequiredParams = qw(CategoryID Name Comment);
        my @Params = qw(ParentID ValidID);
        foreach (@RequiredParams) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!$ParamData{$_}) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_!");
            }
        }
        foreach (@Params) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!defined($ParamData{$_})) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_!");
            }
        }

        # duplicate check
        if(!$Self->{FAQObject}->CategoryDuplicateCheck(Name=>$ParamData{Name},ID=>$ParamData{CategoryID},ParentID=>$ParamData{ParentID})) {
            # db action
            if ($Self->{FAQObject}->CategoryUpdate(%ParamData, UserID => $Self->{UserID})) {
                # set categorygroups
                my @PermissionGroups = $Self->{ParamObject}->GetArray(Param => "PermissionGroups");
                $Self->{FAQObject}->SetCategoryGroup(
                    CategoryID => $ParamData{CategoryID},
                    GroupIDs => \@PermissionGroups,
                );
                return $Self->{LayoutObject}->Redirect(OP => "Action=AgentFAQ&Subaction=Category");
            }
            else {
                return $Self->{LayoutObject}->ErrorScreen();
            }
        }
        else {
            $Self->{LayoutObject}->FatalError(Message => "Category '$ParamData{Name}' already exists!");
        }
    }

    # ---------------------------------------------------------- #
    # category add
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Category') {
        # permission check

        if (!$Self->{AccessRw}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        # dtl

        $Frontend{CategoryLongOption} = $Self->{LayoutObject}->AgentFAQCategoryListOption(
            CategoryList => { %{$Self->{FAQObject}->CategoryList(UserID => $Self->{UserID})} } ,
            Size => 10,
            Name => 'CategoryID',
            HTMLQuote => 1,
            LanguageTranslation => 0,
        );
        $Frontend{CategoryOption} = $Self->{LayoutObject}->AgentFAQCategoryListOption(
            CategoryList => { %{$Self->{FAQObject}->CategoryList(UserID => $Self->{UserID})} },
            Size => 1,
            Name => 'ParentID',
            HTMLQuote => 1,
            LanguageTranslation => 0,
            RootElement => 1,
        );
        # build ValidID string
        $Frontend{ValidOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => {$Self->{ValidObject}->ValidList()},
            Name => 'ValidID',
        );
        $Param{Headline} = 'Add';
        $Param{FormSubaction} = 'CategoryAddAction';
        # group permission
        my %Groups = $Self->{GroupObject}->GroupList(Valid => 1);
        $Frontend{GroupOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => \%Groups,
            Name => 'PermissionGroups',
            Size => 8,
            Multiple => 1,
        );
        $Self->{LayoutObject}->Block(
            Name => 'Category',
            Data => { %Param, %Frontend },
        );

    }
    # ---------------------------------------------------------- #
    # category add action
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'CategoryAddAction') {
        # permission check
        if (!$Self->{AccessRw}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        # check parameters
        my %ParamData = ();
        my @RequiredParams = qw(Name Comment);
        my @Params = qw(ParentID ValidID);
        foreach (@RequiredParams) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!$ParamData{$_}) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_");
            }
        }
        foreach (@Params) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!defined($ParamData{$_})) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_");
            }
        }

        # duplicate check
        if(!$Self->{FAQObject}->CategoryDuplicateCheck(Name=>$ParamData{Name},ParentID=>$ParamData{ParentID})) {
            my $CategoryID = $Self->{FAQObject}->CategoryAdd(%ParamData, UserID => $Self->{UserID});
            if ($CategoryID) {
                # set categorygroups
                my @PermissionGroups = $Self->{ParamObject}->GetArray(Param => "PermissionGroups");
                $Self->{FAQObject}->SetCategoryGroup(
                    CategoryID => $CategoryID,
                    GroupIDs => \@PermissionGroups,
                );
                return $Self->{LayoutObject}->Redirect(OP => "Action=AgentFAQ&Subaction=Category");
            }
            else {
                return $Self->{LayoutObject}->ErrorScreen();
            }
        }
        else {
            $Self->{LayoutObject}->FatalError(Message => "Category '$ParamData{Name}' already exists!");
        }
    }
    # ---------------------------------------------------------- #
    # item add
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Add') {

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

        # check parameters

        # dtl
        $Frontend{LanguageOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => { $Self->{FAQObject}->LanguageList() },
            Name => 'LanguageID',
            LanguageTranslation => 0,
            Selected => $Self->{UserLanguage},
        );

        my $Categories = $Self->{FAQObject}->GetUserCategories(
            UserID =>  $Self->{UserID},
            Type => 'rw'
        );
        $Frontend{CategoryOption} = $Self->{LayoutObject}->AgentFAQCategoryListOption(
            CategoryList => $Categories,
            Name => 'CategoryID',
            LanguageTranslation => 0,
        );

        $Frontend{StateOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => { $Self->{FAQObject}->StateList() },
            Name => 'StateID',
            Selected => 'internal (agent)',
        );

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

        # fields
        $Self->_GetItemFields(
            ItemData => {}
        );

    }
    # ---------------------------------------------------------- #
    # item add action
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'AddAction') {

        @Params = qw(Title);

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

        # check parameters
        my %ParamData = ();
        my @RequiredParams = qw(CategoryID Title);
        my @Params = qw(StateID LanguageID Field1 Field2 Field3 Field4 Field5 Field6 Keywords Title);
        foreach (@RequiredParams) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!$ParamData{$_}) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_!");
            }
        }
        foreach (@Params) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        # get submit attachment
        my %UploadStuff = $Self->{ParamObject}->GetUploadAll(
            Param => 'file_upload',
            Source => 'String',
        );

        # insert item
        my $ItemID = $Self->{FAQObject}->FAQAdd(
            %ParamData,
            %UploadStuff,
            UserID => $Self->{UserID}
        );

        if ($ItemID) {
            if (!defined($ParamData{ItemID}) && !$ParamData{ItemID}) {
                $ParamData{ItemID} = '';
            }
            return $Self->{LayoutObject}->Redirect(OP => "Action=$Self->{Action}&Subaction=View&ItemID=$ParamData{ItemID}");
        }
        else {
            return $Self->{LayoutObject}->ErrorScreen();
        }
    }
    # ---------------------------------------------------------- #
    # item update
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Update') {

        @Params = qw(ItemID);

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

        # check parameters
        my %ParamData = ();
        my @RequiredParams = qw(CategoryID ItemID);
        foreach (@RequiredParams) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!$ParamData{$_}) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_!");
            }
        }

        # db action
        my %ItemData = $Self->{FAQObject}->FAQGet(ItemID => $ParamData{ItemID}, UserID => $Self->{UserID});
        if (!%ItemData) {
            return $Self->{LayoutObject}->ErrorScreen();
        }

        # dtl
        $Frontend{LanguageOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => { $Self->{FAQObject}->LanguageList() },
            Name => 'LanguageID',
            LanguageTranslation => 0,
            SelectedID => $ItemData{LanguageID},
        );
        my $Categories = $Self->{FAQObject}->GetUserCategories(
            UserID =>  $Self->{UserID},
            Type => 'rw'
        );
        $Frontend{CategoryOption} = $Self->{LayoutObject}->AgentFAQCategoryListOption(
            CategoryList => $Categories,
            Name => 'CategoryID',
            LanguageTranslation => 0,
            SelectedID => $ItemData{CategoryID},
        );
        $Frontend{StateOption} = $Self->{LayoutObject}->OptionStrgHashRef(
            Data => { $Self->{FAQObject}->StateList() },
            Name => 'StateID',
            SelectedID => $ItemData{StateID},
        );
        $Self->{LayoutObject}->Block(
            Name => 'Update',
            Data => { %ItemData, %Frontend },
        );

        # fields
        $Self->_GetItemFields(
            ItemData => \%ItemData
        );

        # output
        $HeaderTitle = 'Edit';
        $Header = $Self->{LayoutObject}->Header(Type => $HeaderType, Title => $HeaderTitle);

    }
    # ---------------------------------------------------------- #
    # item update action
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'UpdateAction') {

        @Params = qw(ItemID Title CategoryID StateID LanguageID);

        # permission check

        if (!$Self->{AccessRw}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        # check parameters
        my %ParamData = ();
        my @RequiredParams = qw(Title CategoryID);
        my @Params = qw(ItemID StateID LanguageID Field1 Field2 Field3 Field4 Field5 Field6 Keywords);
        foreach (@RequiredParams) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!$ParamData{$_}) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_!");
            }
        }
        foreach (@Params) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        # db action

        # get submit attachment
        my %UploadStuff = $Self->{ParamObject}->GetUploadAll(
            Param => 'file_upload',
            Source => 'String',
        );

        # update item
        if ($Self->{FAQObject}->FAQUpdate(%ParamData, %UploadStuff, UserID => $Self->{UserID})) {
            return $Self->{LayoutObject}->Redirect(OP => "Action=$Self->{Action}&Subaction=View&ItemID=$GetParam{ItemID}");
        }
        else {
            return $Self->{LayoutObject}->ErrorScreen();
        }
    }
    # ---------------------------------------------------------- #
    # delete item screen
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Delete') {

        @Params = qw(ItemID);

        # permission check

        if (!$Self->{AccessRw}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }

        # check parameters
        my %ParamData = ();
        my @RequiredParams = qw(CategoryID ItemID);
        foreach (@RequiredParams) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!$ParamData{$_}) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_!");
            }
        }
        foreach (@Params) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
        }

        # db action

        my %ItemData = $Self->{FAQObject}->FAQGet(ItemID => $ParamData{ItemID}, UserID => $Self->{UserID});
        if (!%ItemData) {
            return $Self->{LayoutObject}->ErrorScreen();
        }

        # dtl

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

        # output
        $HeaderTitle = 'Delete';
        $Header = $Self->{LayoutObject}->Header(Type => $HeaderType, Title => $HeaderTitle);
    }

    # ---------------------------------------------------------- #
    # item delete action
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'DeleteAction') {

        @Params = qw(ItemID);

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

        # check parameters
        my %ParamData = ();
        my @RequiredParams = qw(CategoryID ItemID);
        foreach (@RequiredParams) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!$ParamData{$_}) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_!");
            }
        }

        # db action
        my %ItemData = $Self->{FAQObject}->FAQGet(ItemID => $ParamData{ItemID}, UserID => $Self->{UserID});
        if (!%ItemData) {
            return $Self->{LayoutObject}->ErrorScreen();
        }

        if ($Self->{FAQObject}->FAQDelete(%ItemData)) {
            return $Self->{LayoutObject}->Redirect(OP => "Action=$Self->{Action}&Subaction=Explorer&CategoryID=$ParamData{CategoryID}");
        }
        else {
            return $Self->{LayoutObject}->ErrorScreen();
        }
    }

    # ---------------------------------------------------------- #
    # download item
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Download') {
        @Params = qw(ItemID);

        # permission check
        if (!$Self->{AccessRo}) {
            return $Self->{LayoutObject}->NoPermission(WithHeader => 'yes');
        }
        # manage parameters
        foreach (@Params) {
            if(!($GetParam{$_} = $Self->{ParamObject}->GetParam(Param => $_))) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_");
            }
        }
        # db action
        my %ItemData = $Self->{FAQObject}->FAQGet(ItemID => $GetParam{ItemID}, UserID => $Self->{UserID});
        if (!%ItemData) {
            return $Self->{LayoutObject}->ErrorScreen();
        }

        if (%ItemData) {
            return $Self->{LayoutObject}->Attachment(%ItemData);
        }
        else {
            return $Self->{LayoutObject}->ErrorScreen();
        }
    }

    # ---------------------------------------------------------- #
    # search a item
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Search') {
        $Self->GetItemSearch(
            Mode => 'Agent',
            User => $Self->{UserID},
        );
        $HeaderTitle = 'Search';
        $Header = $Self->{LayoutObject}->Header(
            Title => $HeaderTitle,
            Type => $HeaderType,
        );
        $Content = $Self->{LayoutObject}->Output(
            TemplateFile => 'FAQ',
            Data => {%Frontend , %GetParam }
        );
    }

    # ---------------------------------------------------------- #
    # item history
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'History') {

        # check parameters
        my %ParamData = ();
        my @RequiredParams = qw(CategoryID ItemID);
        foreach (@RequiredParams) {
            $ParamData{$_} = $Self->{ParamObject}->GetParam(Param => $_);
            if(!$ParamData{$_}) {
                return $Self->{LayoutObject}->FatalError(Message => "Need $_!");
            }
        }

        # db action
        # ********************************************************** #
        my %ItemData = $Self->{FAQObject}->FAQGet(ItemID => $ParamData{ItemID}, UserID => $Self->{UserID});
        if (!%ItemData) {
            return $Self->{LayoutObject}->ErrorScreen();
        }

        # dtl
        # ********************************************************** #
        $Self->{LayoutObject}->Block(
            Name => 'History',
            Data => { %Param, %ItemData },
        );

        $HeaderTitle = 'History';
        $Frontend{CssRow} = 'searchactive';
        my @History = @{$Self->{FAQObject}->FAQHistoryGet(ItemID => $ItemData{ItemID})};
        foreach my $Row (@History) {

            # css configuration
            if($Frontend{CssRow} eq 'searchpassive') {
                $Frontend{CssRow} = 'searchactive';
            }
            else {
                $Frontend{CssRow} = 'searchpassive';
            }

            $Frontend{Name} = $Row->{Name};
            $Frontend{Created} = $Row->{Created};
            my %User = $Self->{UserObject}->GetUserData(
                UserID => $Row->{CreatedBy},
                Cached => 1,
            );
            $Frontend{CreatedBy} = "$User{UserLogin} ($User{UserFirstname} $User{UserLastname})";

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

        }

    }

    # ---------------------------------------------------------- #
    # system history
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'SystemHistory') {
        $Self->GetSystemHistory();
        $HeaderTitle = 'History';
    }
    # ---------------------------------------------------------- #
    # item print
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Print' && $Self->{ParamObject}->GetParam(Param => 'ItemID')) {
        $Self->GetItemPrint(Links => 1);
        $Header = $Self->{LayoutObject}->PrintHeader(
            Title => $Self->{ItemData}{Subject}
        );
        $Navigation = ' ';
        $Content = $Self->{LayoutObject}->Output(
            TemplateFile => 'FAQ',
            Data => {%Frontend , %GetParam }
        );
        $Footer = $Self->{LayoutObject}->PrintFooter();
    }

    # ---------------------------------------------------------- #
    # explorer
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Explorer') {
        $Self->GetExplorer(Mode => 'Agent');
        $HeaderTitle = 'Explorer';
        $Header = $Self->{LayoutObject}->Header(
            Type => $HeaderType,
            Title => $HeaderTitle
        );
        $Content = $Self->{LayoutObject}->Output(
            TemplateFile => 'FAQ',
            Data => {%Frontend , %GetParam }
        );

    }

    # ---------------------------------------------------------- #
    # item view
    # ---------------------------------------------------------- #
    elsif ($Self->{ParamObject}->GetParam(Param => 'ItemID')) {
        if($Self->{LastFAQNav}) {
            $Self->GetItemSmallView();
        } else {
            my %FAQArticle = $Self->{FAQObject}->FAQGet(
                FAQID => $Self->{ParamObject}->GetParam(Param => 'ItemID'),
            );
            my $Permission = $Self->{FAQObject}->CheckCategoryUserPermission(
                UserID => $Self->{UserID},
                CategoryID => $FAQArticle{CategoryID},
            );
            if ($Permission eq '') {
                $Self->{LayoutObject}->FatalError(Message => "Permission denied!");
            }
            $Self->GetItemView(
                Links => 1,
                Permission => $Permission,
            );
        }
        $HeaderTitle = $Self->{ItemData}{Number};
        $Header = $Self->{LayoutObject}->Header(
            Type => $HeaderType,
            Title => $HeaderTitle
        );
    }

    # ---------------------------------------------------------- #
    # redirect to explorer
    # ---------------------------------------------------------- #
    else {
        if (!defined($Param{Nav})) {
            $Param{Nav} = '';
        }
        return $Self->{LayoutObject}->Redirect(OP => "Action=$Self->{Action}&Subaction=Explorer&Nav=".$Param{Nav});
    }

    # DEFAULT OUTPUT
    $DefaultHeader = $Self->{LayoutObject}->Header(
        Type => $HeaderType,
        Title => $HeaderTitle
    );
    $DefaultNavigation = $Self->{LayoutObject}->NavigationBar();
    $DefaultContent = $Self->{LayoutObject}->Output(
        TemplateFile => 'AgentFAQ',
        Data => {%Frontend , %GetParam }
    );
    $DefaultFooter = $Self->{LayoutObject}->Footer(Type => $FooterType);

    # OUTPUT
    $Output .= $Header || $DefaultHeader;
    $Output .= $Navigation || $DefaultNavigation;
    if(!$Notify) {
        foreach my $Notify (@{$Self->{Notify}}) {
            $Output .= $Self->{LayoutObject}->Notify(
                Priority => $Notify->[0],
                Info => $Notify->[1],
            );
        }
    }
    $Output .= $Content || $DefaultContent;
    $Output .= $Footer || $DefaultFooter;

    return $Output;
}

1;

# --
# Kernel/Modules/CustomerFAQ.pm - faq module
# Copyright (C) 2001-2007 OTRS GmbH, http://otrs.org/
# --
# $Id: CustomerFAQ.pm,v 1.4 2007/01/18 14:11:20 rk Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl.txt.
# --

package Kernel::Modules::CustomerFAQ;

use strict;
use Kernel::System::FAQ;
use Kernel::Modules::FAQ;

use vars qw($VERSION);
$VERSION = '$Revision: 1.4 $';
$VERSION =~ s/^\$.*:\W(.*)\W.+?$/$1/;

our @ISA = qw(Kernel::Modules::FAQ);

# --
sub new {
    my $Type = shift;
    my %Param = @_;

    # allocate new hash for object
    # ********************************************************** #
    my $Self = new Kernel::Modules::FAQ(%Param);
    bless ($Self, $Type);

    # interface settings
    # ********************************************************** #
    $Self->{Interface} = $Self->{FAQObject}->StateTypeGet(
        Name => 'external'
    );
    $Self->{InterfaceStates} = $Self->{FAQObject}->StateTypeList(
        Types => ['external','public']
    );
    # check needed Opjects
    # ********************************************************** #
    foreach (qw(UserObject)) {
        $Self->{LayoutObject}->FatalError(Message => "Got no $_!") if (!$Self->{$_});
    }

    return $Self;
}
# --
sub Run {
    my $Self = shift;
    my %Param = @_;

    # Paramter
    my @Params = ();
    my %GetParam = ();
    my %Frontend = ();

    # Output
    my $Output = '';
    my $Header = '';
    my $HeaderTitle =  '';
    my $HeaderType =  '';
    my $Navigation = '';
    my $Notify = '';
    my $Content = '';
    my $Footer = '';
    my $FooterType =  '';

    my $DefaultHeader = '';
    my $DefaultNavigation = '';
    my $DefaultContent = '';
    my $DefaultFooter = '';

    # navigation ON/OFF
    # ********************************************************** #
    if ($GetParam{Nav}) {
        $HeaderType = 'Small';
    } else {
        $HeaderType = '';
    }
    # store nav param
    $Self->{SessionObject}->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key => 'LastFAQNav',
        Value => $HeaderType,
    );

    # ---------------------------------------------------------- #
    # explorer
    # ---------------------------------------------------------- #
    if ($Self->{Subaction} eq 'Explorer') {
        $Self->GetExplorer(
            Mode => 'Customer',
            CustomerUser => $Self->{UserLogin},
        );
        $HeaderTitle = 'Explorer';
        $Header = $Self->{LayoutObject}->CustomerHeader(
            Type => $HeaderType,
            Title => $HeaderTitle
        );
        $Content = $Self->{LayoutObject}->Output(
            TemplateFile => 'FAQ',
            Data => { %Frontend , %GetParam }
        );
    }
    # ---------------------------------------------------------- #
    # search a item
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Search') {
        $HeaderTitle = 'Search';
        $Header = $Self->{LayoutObject}->CustomerHeader(
            Type => $HeaderType,
            Title => $HeaderTitle
        );
        $Self->GetItemSearch(
            Mode => 'Customer',
            CustomerUser => $Self->{UserLogin},
        );

        $Content = $Self->{LayoutObject}->Output(
            TemplateFile => 'FAQ',
            Data => {%Frontend , %GetParam }
        );
    }
    # ---------------------------------------------------------- #
    # download item
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Download') {
        # get param
        my $ItemID  = $Self->{ParamObject}->GetParam(Param => 'ItemID');
        # db action
        my %ItemData = $Self->{FAQObject}->FAQGet(ItemID => $ItemID);
        if (!%ItemData) {
            return $Self->{LayoutObject}->FatalError(Message => "No FAQ found!");
        }
        if ($ItemData{StateTypeName} eq 'external' || $ItemData{StateTypeName} eq 'public') {
            return $Self->{LayoutObject}->Attachment(%ItemData);
        }
        else {
            return $Self->{LayoutObject}->FatalError(Message => "Permission denied!");
        }
    }
    # ---------------------------------------------------------- #
    # item print
    # ---------------------------------------------------------- #
    elsif ($Self->{Subaction} eq 'Print' && $Self->{ParamObject}->GetParam(Param => 'ItemID')) {
        $Self->GetItemPrint();
        $Header = $Self->{LayoutObject}->PrintHeader(
            Title => $Self->{ItemData}{Subject}
        );
        $Navigation = ' ';
        $Content = $Self->{LayoutObject}->Output(
            TemplateFile => 'FAQ',
            Data => {%Frontend , %GetParam }
        );
        $Footer = $Self->{LayoutObject}->PrintFooter();
    }

    # ---------------------------------------------------------- #
    # item view
    # ---------------------------------------------------------- #
    elsif ($Self->{ParamObject}->GetParam(Param => 'ItemID')) {
        my %FAQArticle = $Self->{FAQObject}->FAQGet(
            FAQID => $Self->{ParamObject}->GetParam(Param => 'ItemID'),
        );
        my $Permission = $Self->{FAQObject}->CheckCategoryCustomerPermission(
            CustomerUser => $Self->{UserLogin},
            CategoryID => $FAQArticle{CategoryID},
        );
        if ($Permission eq '') {
            $Self->{LayoutObject}->FatalError(Message => "Permission denied!");
        }
        $Self->GetItemView(
            Links => 1,
            Permission => $Permission,
        );

        $HeaderTitle = $Self->{ItemData}{Number};
        $Header = $Self->{LayoutObject}->CustomerHeader(
            Type => $HeaderType,
            Title => $HeaderTitle
        );
        $Content = $Self->{LayoutObject}->Output(
            TemplateFile => 'FAQ',
            Data => {%Frontend , %GetParam }
        );
    }

    # ---------------------------------------------------------- #
    # redirect to explorer
    # ---------------------------------------------------------- #
    else {
        return $Self->{LayoutObject}->Redirect(OP => "Action=$Self->{Action}&Subaction=Explorer");
    }

    # DEFAULT OUTPUT
    $DefaultHeader = $Self->{LayoutObject}->CustomerHeader(
        Type => $HeaderType,
        Title => $HeaderTitle
    );
    $DefaultNavigation = $Self->{LayoutObject}->CustomerNavigationBar();
    $DefaultContent = $Self->{LayoutObject}->Output(
        TemplateFile => 'CustomerFAQ',
        Data => { %Frontend , %GetParam }
    );
    $DefaultFooter = $Self->{LayoutObject}->CustomerFooter(Type => $FooterType);

    # OUTPUT
    $Output .= $Header || $DefaultHeader;
    $Output .= $Navigation || $DefaultNavigation;
    if(!$Notify) {
        foreach my $Notify (@{$Self->{Notify}}) {
            $Output .= $Self->{LayoutObject}->Notify(
                Priority => $Notify->[0],
                Info => $Notify->[1],
            );
        }
    }
    $Output .= $Content || $DefaultContent;
    $Output .= $Footer || $DefaultFooter;

    return $Output;
}
# --

1;

IyAtLQojIEtlcm5lbC9Nb2R1bGVzL0N1c3RvbWVyRkFRLnBtIC0gZmFxIG1vZHVsZQojIENvcHlyaWdodCAoQykgMjAwMS0yMDA2IE9UUlMgR21iSCwgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBQdWJsaWNGQVEucG0sdiAxLjIgMjAwNi8xMC8yMCAxMjowNjozNyByayBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpNb2R1bGVzOjpQdWJsaWNGQVE7Cgp1c2Ugc3RyaWN0Owp1c2UgS2VybmVsOjpTeXN0ZW06OkZBUTsKdXNlIEtlcm5lbDo6TW9kdWxlczo6RkFROwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9ICckUmV2aXNpb246IDEuMiAkJzsKJFZFUlNJT04gPX4gcy9eXCQuKjpcVyguKilcVy4rPyQvJDEvOwoKb3VyIEBJU0EgPSBxdyhLZXJuZWw6Ok1vZHVsZXM6OkZBUSk7CgojIC0tCnN1YiBuZXcgewogICAgbXkgJFR5cGUgPSBzaGlmdDsKICAgIG15ICVQYXJhbSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgIyAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqICMKICAgIG15ICRTZWxmID0gbmV3IEtlcm5lbDo6TW9kdWxlczo6RkFRKCVQYXJhbSk7CiAgICBibGVzcyAoJFNlbGYsICRUeXBlKTsKCiAgICAjIGludGVyZmFjZSBzZXR0aW5ncwogICAgIyAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqICMKICAgICRTZWxmLT57SW50ZXJmYWNlfSA9ICRTZWxmLT57RkFRT2JqZWN0fS0+U3RhdGVUeXBlR2V0KAogICAgICAgIE5hbWUgPT4gJ3B1YmxpYycKICAgICk7CiAgICAkU2VsZi0+e0ludGVyZmFjZVN0YXRlc30gPSAkU2VsZi0+e0ZBUU9iamVjdH0tPlN0YXRlVHlwZUxpc3QoCiAgICAgICAgVHlwZXMgPT4gWydwdWJsaWMnXQogICAgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBPcGplY3RzCiAgICAjICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiogIwogICAgZm9yZWFjaCAocXcoKSkgewogICAgICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+RmF0YWxFcnJvcihNZXNzYWdlID0+ICJHb3Qgbm8gJF8hIikgaWYgKCEkU2VsZi0+eyRffSk7CiAgICB9CgogICAgcmV0dXJuICRTZWxmOwp9CiMgLS0Kc3ViIFJ1biB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwogICAgbXkgJVBhcmFtID0gQF87CgogICAgIyBQYXJhbXRlcgogICAgbXkgQFBhcmFtcyA9ICgpOwogICAgbXkgJUdldFBhcmFtID0gKCk7CiAgICBteSAlRnJvbnRlbmQgPSAoKTsKCiAgICAjIE91dHB1dAogICAgbXkgJE91dHB1dCA9ICcnOwogICAgbXkgJEhlYWRlciA9ICcnOwogICAgbXkgJEhlYWRlclRpdGxlID0gICcnOwogICAgbXkgJEhlYWRlclR5cGUgPSAgJyc7CiAgICBteSAkTmF2aWdhdGlvbiA9ICcnOwogICAgbXkgJE5vdGlmeSA9ICcnOwogICAgbXkgJENvbnRlbnQgPSAnJzsKICAgIG15ICRGb290ZXIgPSAnJzsKICAgIG15ICRGb290ZXJUeXBlID0gICcnOwoKICAgIG15ICREZWZhdWx0SGVhZGVyID0gJyc7CiAgICBteSAkRGVmYXVsdE5hdmlnYXRpb24gPSAnJzsKICAgIG15ICREZWZhdWx0Q29udGVudCA9ICcnOwogICAgbXkgJERlZmF1bHRGb290ZXIgPSAnJzsKCiAgICAjIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gIwogICAgIyBleHBsb3JlcgogICAgIyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICMKICAgIGlmICgkU2VsZi0+e1N1YmFjdGlvbn0gZXEgJ0V4cGxvcmVyJykgewogICAgICAgICRIZWFkZXJUaXRsZSA9ICdFeHBsb3Jlcic7CiAgICAgICAgJEhlYWRlciA9ICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+Q3VzdG9tZXJIZWFkZXIoCiAgICAgICAgICAgIFR5cGUgPT4gJEhlYWRlclR5cGUsCiAgICAgICAgICAgIFRpdGxlID0+ICRIZWFkZXJUaXRsZQogICAgICAgICk7CiAgICAgICAgJFNlbGYtPkdldEV4cGxvcmVyKCk7CiAgICAgICAgJENvbnRlbnQgPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPk91dHB1dCgKICAgICAgICAgICAgVGVtcGxhdGVGaWxlID0+ICdGQVEnLAogICAgICAgICAgICBEYXRhID0+IHslRnJvbnRlbmQgLCAlR2V0UGFyYW0gfQogICAgICAgICk7CiAgICB9CgogICAgIyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICMKICAgICMgc2VhcmNoIGEgaXRlbQogICAgIyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICMKICAgIGVsc2lmICgkU2VsZi0+e1N1YmFjdGlvbn0gZXEgJ1NlYXJjaCcpIHsKICAgICAgICAkSGVhZGVyVGl0bGUgPSAnU2VhcmNoJzsKICAgICAgICAkSGVhZGVyID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5DdXN0b21lckhlYWRlcigKICAgICAgICAgICAgVHlwZSA9PiAkSGVhZGVyVHlwZSwKICAgICAgICAgICAgVGl0bGUgPT4gJEhlYWRlclRpdGxlCiAgICAgICAgKTsKICAgICAgICAkU2VsZi0+R2V0SXRlbVNlYXJjaCgpOwogICAgICAgICRDb250ZW50ID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5PdXRwdXQoCiAgICAgICAgICAgIFRlbXBsYXRlRmlsZSA9PiAnRkFRJywKICAgICAgICAgICAgRGF0YSA9PiB7JUZyb250ZW5kICwgJUdldFBhcmFtIH0KICAgICAgICApOwogICAgfQogICAgIyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICMKICAgICMgZG93bmxvYWQgaXRlbQogICAgIyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICMKICAgIGVsc2lmICgkU2VsZi0+e1N1YmFjdGlvbn0gZXEgJ0Rvd25sb2FkJykgewogICAgICAgICMgZ2V0IHBhcmFtCiAgICAgICAgbXkgJEl0ZW1JRCAgPSAkU2VsZi0+e1BhcmFtT2JqZWN0fS0+R2V0UGFyYW0oUGFyYW0gPT4gJ0l0ZW1JRCcpOwogICAgICAgICMgZGIgYWN0aW9uCiAgICAgICAgbXkgJUl0ZW1EYXRhID0gJFNlbGYtPntGQVFPYmplY3R9LT5GQVFHZXQoSXRlbUlEID0+ICRJdGVtSUQpOwogICAgICAgIGlmICghJUl0ZW1EYXRhKSB7CiAgICAgICAgICAgIHJldHVybiAkU2VsZi0+e0xheW91dE9iamVjdH0tPkZhdGFsRXJyb3IoTWVzc2FnZSA9PiAiTm8gRkFRIGZvdW5kISIpOwogICAgICAgIH0KICAgICAgICBpZiAoJEl0ZW1EYXRhe1N0YXRlVHlwZU5hbWV9IGVxICdwdWJsaWMnKSB7CiAgICAgICAgICAgIHJldHVybiAkU2VsZi0+e0xheW91dE9iamVjdH0tPkF0dGFjaG1lbnQoJUl0ZW1EYXRhKTsKICAgICAgICB9CiAgICAgICAgZWxzZSB7CiAgICAgICAgICAgIHJldHVybiAkU2VsZi0+e0xheW91dE9iamVjdH0tPkZhdGFsRXJyb3IoTWVzc2FnZSA9PiAiUGVybWlzc2lvbiBkZW5pZWQhIik7CiAgICAgICAgfQogICAgfQoKICAgICMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSAjCiAgICAjIGl0ZW0gcHJpbnQKICAgICMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSAjCiAgICBlbHNpZiAoJFNlbGYtPntTdWJhY3Rpb259IGVxICdQcmludCcgJiYgJFNlbGYtPntQYXJhbU9iamVjdH0tPkdldFBhcmFtKFBhcmFtID0+ICdJdGVtSUQnKSkgewogICAgICAgICRIZWFkZXIgPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPlByaW50SGVhZGVyKAogICAgICAgICAgICBUaXRsZSA9PiAkU2VsZi0+e0l0ZW1EYXRhfXtTdWJqZWN0fQogICAgICAgICk7CiAgICAgICAgJFNlbGYtPkdldEl0ZW1QcmludCgpOwogICAgICAgICRDb250ZW50ID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5PdXRwdXQoCiAgICAgICAgICAgIFRlbXBsYXRlRmlsZSA9PiAnRkFRJywKICAgICAgICAgICAgRGF0YSA9PiB7ICVGcm9udGVuZCAsICVHZXRQYXJhbSB9CiAgICAgICAgKTsKICAgICAgICAkRm9vdGVyID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5QcmludEZvb3RlcigpOwogICAgfQoKICAgICMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSAjCiAgICAjIGl0ZW0gdmlldwogICAgIyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICMKICAgIGVsc2lmICgkU2VsZi0+e1BhcmFtT2JqZWN0fS0+R2V0UGFyYW0oUGFyYW0gPT4gJ0l0ZW1JRCcpKSB7CiAgICAgICAgJFNlbGYtPkdldEl0ZW1WaWV3KCk7CiAgICAgICAgJEhlYWRlclRpdGxlID0gJFNlbGYtPntJdGVtRGF0YX17TnVtYmVyfTsKICAgICAgICAkSGVhZGVyID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5DdXN0b21lckhlYWRlcigKICAgICAgICAgICAgVHlwZSA9PiAkSGVhZGVyVHlwZSwKICAgICAgICAgICAgVGl0bGUgPT4gJEhlYWRlclRpdGxlCiAgICAgICAgKTsKICAgICAgICAkQ29udGVudCA9ICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+T3V0cHV0KAogICAgICAgICAgICBUZW1wbGF0ZUZpbGUgPT4gJ0ZBUScsCiAgICAgICAgICAgIERhdGEgPT4geyVGcm9udGVuZCAsICVHZXRQYXJhbSB9CiAgICAgICAgKTsKICAgIH0KCiAgICAjIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gIwogICAgIyByZWRpcmVjdCB0byBleHBsb3JlcgogICAgIyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICMKICAgIGVsc2UgewogICAgICAgIHJldHVybiAkU2VsZi0+e0xheW91dE9iamVjdH0tPlJlZGlyZWN0KE9QID0+ICJBY3Rpb249JFNlbGYtPntBY3Rpb259JlN1YmFjdGlvbj1FeHBsb3JlciIpOwogICAgfQoKICAgICMgREVGQVVMVCBPVVRQVVQKICAgICREZWZhdWx0SGVhZGVyID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5DdXN0b21lckhlYWRlcigKICAgICAgICBUeXBlID0+ICRIZWFkZXJUeXBlLAogICAgICAgIFRpdGxlID0+ICRIZWFkZXJUaXRsZQogICAgKTsKCiAgICAkRGVmYXVsdENvbnRlbnQgPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPk91dHB1dCgKICAgICAgICBUZW1wbGF0ZUZpbGUgPT4gJ1B1YmxpY0ZBUScsCiAgICAgICAgRGF0YSA9PiB7ICVGcm9udGVuZCAsICVHZXRQYXJhbSB9CiAgICApOwogICAgJERlZmF1bHRGb290ZXIgPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPkN1c3RvbWVyRm9vdGVyKFR5cGUgPT4gJEZvb3RlclR5cGUpOwoKICAgICMgT1VUUFVUCiAgICAkT3V0cHV0IC49ICRIZWFkZXIgfHwgJERlZmF1bHRIZWFkZXI7CiAgICBpZighJE5vdGlmeSkgewogICAgICAgIGZvcmVhY2ggbXkgJE5vdGlmeSAoQHskU2VsZi0+e05vdGlmeX19KSB7CiAgICAgICAgICAgICRPdXRwdXQgLj0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5Ob3RpZnkoCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAkTm90aWZ5LT5bMF0sCiAgICAgICAgICAgICAgICBJbmZvID0+ICROb3RpZnktPlsxXSwKICAgICAgICAgICAgKTsKICAgICAgICB9CiAgICB9CiAgICAkT3V0cHV0IC49ICRDb250ZW50IHx8ICREZWZhdWx0Q29udGVudDsKICAgICRPdXRwdXQgLj0gJEZvb3RlciB8fCAkRGVmYXVsdEZvb3RlcjsKCiAgICByZXR1cm4gJE91dHB1dDsKfQojIC0tCgoxOwo=
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9MYXlvdXRGQVEucG0gLSBwcm92aWRlcyBnZW5lcmljIGFnZW50IEhUTUwgb3V0cHV0CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDcgT1RSUyBHbWJILCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IExheW91dEZBUS5wbSx2IDEuMiAyMDA3LzA3LzAzIDE0OjAyOjQ2IHJrIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0RkFROwoKdXNlIHN0cmljdDsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSAnJFJldmlzaW9uOiAxLjIgJCc7CiRWRVJTSU9OID1+IHMvXlwkLio6XFcoLiopXFcuKz8kLyQxLzsKCnN1YiBBZ2VudEZBUUNhdGVnb3J5TGlzdE9wdGlvbiB7CgogICAgbXkgJFNlbGYgPSBzaGlmdDsKICAgIG15ICVQYXJhbSA9IEBfOwogICAgbXkgJE91dHB1dCA9ICcnOwogICAgbXkgJFNpemUgPSBkZWZpbmVkKCRQYXJhbXtTaXplfSkgPyAic2l6ZT0nJFBhcmFte1NpemV9JyIgOiAnJzsKICAgIG15ICRNYXhMZXZlbCA9IGRlZmluZWQoJFBhcmFte01heExldmVsfSkgPyAkUGFyYW17TWF4TGV2ZWx9IDogMTA7CiAgICBteSAkU2VsZWN0ZWQgPSBkZWZpbmVkKCRQYXJhbXtTZWxlY3RlZH0pID8gJFBhcmFte1NlbGVjdGVkfSA6ICcnOwogICAgbXkgJFNlbGVjdGVkSURzID0gJFBhcmFte1NlbGVjdGVkSURzfSB8fCBbXTsKICAgIG15ICRTZWxlY3RlZElEID0gZGVmaW5lZCgkUGFyYW17U2VsZWN0ZWRJRH0pID8gJFBhcmFte1NlbGVjdGVkSUR9IDogJyc7CiAgICBteSAkTXVsdGlwbGUgPSAkUGFyYW17TXVsdGlwbGV9ID8gJ211bHRpcGxlJyA6ICcnOwogICAgbXkgJE9uQ2hhbmdlU3VibWl0ID0gZGVmaW5lZCgkUGFyYW17T25DaGFuZ2VTdWJtaXR9KSA/ICRQYXJhbXtPbkNoYW5nZVN1Ym1pdH0gOiAnJzsKICAgIGlmICgkT25DaGFuZ2VTdWJtaXQpIHsKICAgICAgICAkT25DaGFuZ2VTdWJtaXQgPSAiIG9uY2hhbmdlPVwic3VibWl0KClcIiI7CiAgICB9CiAgICBpZiAoJFBhcmFte09uQ2hhbmdlfSkgewogICAgICAgICRPbkNoYW5nZVN1Ym1pdCA9ICIgb25jaGFuZ2U9XCIkUGFyYW17T25DaGFuZ2V9XCIiOwogICAgfQoKICAgIGlmIChkZWZpbmVkKCRQYXJhbXtTZWxlY3RlZElEfSkpIHsKICAgICAgICAkU2VsZWN0ZWRJRHMgPSBbJFNlbGVjdGVkSURdOwogICAgfQoKICAgIG15ICVDYXRlZ29yeUxpc3QgPSAleyRQYXJhbXtDYXRlZ29yeUxpc3R9fTsKICAgICNpZiAoISgkUGFyYW17Q2F0ZWdvcnlMaXN0fSAmJiByZWYoJFBhcmFte0NhdGVnb3J5TGlzdH0pIGVxICdIQVNIJyAmJiAleyRQYXJhbXtDYXRlZ29yeUxpc3R9fSkpIHsKICAgICMgICAgcmV0dXJuICRTZWxmLT5GYXRhbEVycm9yKAogICAgIyAgICAgICAgTWVzc2FnZSA9PiAnTmVlZCBDYXRlZ29yeUxpc3QgYXMgaGFzaCByZWYgJwogICAgIyAgICApOwogICAgI30KCiAgICAjIGJ1aWxkIHRyZWUgbGlzdAogICAgJE91dHB1dCAuPSAnPHNlbGVjdCBuYW1lPSInLiRQYXJhbXtOYW1lfS4iXCIgJFNpemUgJE11bHRpcGxlICRPbkNoYW5nZVN1Ym1pdD4iOwoKICAgIGlmKCRQYXJhbXtSb290RWxlbWVudH0pIHsKICAgICAgICAkT3V0cHV0IC49ICc8b3B0aW9uIHZhbHVlPSIwIj4tPC9vcHRpb24+JzsKICAgIH0KCiAgICBpZiAoJFBhcmFte0NhdGVnb3J5TGlzdH0gJiYgcmVmKCRQYXJhbXtDYXRlZ29yeUxpc3R9KSBlcSAnSEFTSCcgJiYgJXskUGFyYW17Q2F0ZWdvcnlMaXN0fX0pIHsKICAgICAgICAkT3V0cHV0IC49ICRTZWxmLT5BZ2VudEZBUUNhdGVnb3J5TGlzdE9wdGlvbkVsZW1lbnQoCiAgICAgICAgICAgIENhdGVnb3J5TGlzdCA9PiBcJUNhdGVnb3J5TGlzdCwKICAgICAgICAgICAgTGV2ZWxDb3VudGVyID0+IDAsCiAgICAgICAgICAgIFBhcmVudElEID0+IDAsCiAgICAgICAgICAgIFNlbGVjdGVkSURzID0+ICRTZWxlY3RlZElEcwogICAgICAgICk7CiAgICB9CiAgICAkT3V0cHV0IC49ICc8L3NlbGVjdD4nOwoKICAgIHJldHVybiAkT3V0cHV0Owp9CgpzdWIgQWdlbnRGQVFDYXRlZ29yeUxpc3RPcHRpb25FbGVtZW50IHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CiAgICBteSAlUGFyYW0gPSBAXzsKICAgIG15ICRPdXRwdXQgPSAnJzsKCiAgICBteSAlQ2F0ZWdvcnlMaXN0ID0gJXsgJFBhcmFte0NhdGVnb3J5TGlzdH0gfTsKICAgIG15ICVDYXRlZ29yeUxldmVsTGlzdCA9ICUgeyAkQ2F0ZWdvcnlMaXN0eyRQYXJhbXtQYXJlbnRJRH19fTsKICAgIG15IEBTZWxlY3RlZElEcyA9IEB7JFBhcmFte1NlbGVjdGVkSURzfX07CiAgICBteSAkRmxhZyA9IDA7CgogICAgZm9yZWFjaCBteSAkQ2F0ZWdvcnlJRCAoc29ydCggeyAkQ2F0ZWdvcnlMZXZlbExpc3R7JGF9IGNtcCAkQ2F0ZWdvcnlMZXZlbExpc3R7JGJ9IH0ga2V5cyglQ2F0ZWdvcnlMZXZlbExpc3QpKSkgewogICAgICAgICRPdXRwdXQgLj0gJzxvcHRpb24gdmFsdWU9IicuJENhdGVnb3J5SUQuJyInOwogICAgICAgIGZvcihteSAkaT0wOyRpPEBTZWxlY3RlZElEczskaSsrKSB7CiAgICAgICAgICAgIGlmKCRQYXJhbXtTZWxlY3RlZElEc31bJGldIGVxICRDYXRlZ29yeUlEKSB7CiAgICAgICAgICAgICAgICAkT3V0cHV0IC49ICcgc2VsZWN0ZWQnOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgICRPdXRwdXQgLj0gJz4nOwogICAgICAgIGZvcihteSAkaT0wOyRpPCRQYXJhbXtMZXZlbENvdW50ZXJ9OyRpKyspIHsKICAgICAgICAgICAgJE91dHB1dCAuPSAnJm5ic3A7Jm5ic3A7JzsKICAgICAgICB9CiAgICAgICAgJE91dHB1dCAuPSAkQ2F0ZWdvcnlMZXZlbExpc3R7JENhdGVnb3J5SUR9OwogICAgICAgICRPdXRwdXQgLj0gJzwvb3B0aW9uPic7CiAgICAgICAgaWYoZXhpc3RzKCRDYXRlZ29yeUxpc3R7JENhdGVnb3J5SUR9KSkgewogICAgICAgICAgICAkT3V0cHV0IC49ICRTZWxmLT5BZ2VudEZBUUNhdGVnb3J5TGlzdE9wdGlvbkVsZW1lbnQoCiAgICAgICAgICAgICAgICBDYXRlZ29yeUxpc3QgPT4gXCVDYXRlZ29yeUxpc3QsCiAgICAgICAgICAgICAgICBMZXZlbENvdW50ZXIgPT4gJFBhcmFte0xldmVsQ291bnRlcn0rMSwKICAgICAgICAgICAgICAgIFBhcmVudElEID0+ICRDYXRlZ29yeUlELAogICAgICAgICAgICAgICAgU2VsZWN0ZWRJRHMgPT4gXEBTZWxlY3RlZElEcwogICAgICAgICAgICApOwogICAgICAgIH0KICAgIH0KICAgIHJldHVybiAkT3V0cHV0Owp9CgpzdWIgR2V0RkFRSXRlbVZvdGluZ1JhdGVDb2xvciB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwogICAgbXkgJVBhcmFtID0gQF87CgogICAgaWYgKCEkUGFyYW17UmF0ZX0pIHsKICAgICAgICByZXR1cm4gJFNlbGYtPkZhdGFsRXJyb3IoCiAgICAgICAgICAgIE1lc3NhZ2UgPT4gJ05lZWQgcmF0ZSEnCiAgICAgICAgKTsKICAgIH0KICAgIG15ICRDc3NUbXAgPSAnJzsKICAgIG15ICVWb3RpbmdSZXN1bHRDb2xvcnMgPSAleyRTZWxmLT57Q29uZmlnT2JqZWN0fS0+R2V0KCdGQVE6OkV4cGxvcmVyOjpJdGVtTGlzdDo6Vm90aW5nUmVzdWx0Q29sb3JzJyl9OwogICAgZm9yZWFjaCBteSAkS2V5IChzb3J0KCB7ICRiIDw9PiAkYSB9IGtleXMoJVZvdGluZ1Jlc3VsdENvbG9ycykpKSB7CiAgICAgICAgaWYoJFBhcmFte1JhdGV9IDw9ICRLZXkpIHsKICAgICAgICAgICAgJENzc1RtcCA9ICRWb3RpbmdSZXN1bHRDb2xvcnN7JEtleX07CiAgICAgICAgfQogICAgfQogICAgcmV0dXJuICRDc3NUbXA7Cn0KCjE7Cg==
# --
# FAQ.dtl - provides HTML view for faq articles
# Copyright (C) 2001-2008 OTRS AG, http://otrs.org/
# --
# $Id: FAQ.dtl,v 1.8 2008/03/19 10:54:57 martin Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --


<!-- dtl:block:View -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td colspan="2" class="mainhead">
      $Env{"Box0"}$Text{"View"}: $Config{"FAQ::FAQHook"} $QData{"Number"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td class="menu">
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$Data{"CategoryID"}">$Text{"Back"}</a> -
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Print&CategoryID=$Data{"CategoryID"}&ItemID=$Data{"ItemID"}" target="print">$Text{"Print"}</a>
    </td>
    <td align="right" class="menu">

    </td>
  </tr>
  <tr>
    <td width="75%" class="mainbody">

<!-- dtl:block:FAQPathCategoryElement -->
      $Config{"FAQ::Explorer::Path::Seperator"} <b><a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$Data{"CategoryID"}">$QData{"Name"}</a></b>
<!-- dtl:block:FAQPathCategoryElement -->

      <h1>$QData{"Title"}</h1>

      <table border="0" width="100%" cellspacing="0" cellpadding="4">
        <tr>
          <td colspan="2" class="contenthead">$Text{"Options"}</td>
        </tr>
        <tr>
          <td class="contentbody">
            <table border="0" cellspacing="0" cellpadding="3">
<!-- dtl:block:FAQItemViewAttachment -->
              <tr>
                <td class="contentkey" valign="top">$Text{"Attachment"}:</td>
                <td class="contentvalue"> <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Download&ItemID=$QData{"ItemID"}">$QData{"Filename"}</a></td>
              </tr>
<!-- dtl:block:FAQItemViewAttachment -->
<!-- dtl:block:FAQItemField -->
              <tr>
                <td width="25%" class="contentkey" valign="top">$Text{"$Data{"Caption"}"}:  <br> <i class="small">($Text{"$Data{"StateName"}"})</i></td>
                <td width="75%" class="contentvalue">$Data{"Value"}</td>
              </tr>
<!-- dtl:block:FAQItemField -->

            </table>
          </td>
        </tr>
        <tr>
          <td colspan="2" class="contenthead">&nbsp;</td>
        </tr>
      </table>

    </td>
    <td width="35%" class="menu">

      <table border="0" cellspacing="1" cellpadding="0" width="95%">
        <tr>
          <td class="contentkey">$Text{"Category"}:</td>
          <td class="contentvalue"> <div title="$QData{"CategoryName"}">$QData{"CategoryName","30"}</div></td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"State"}:</td>
          <td class="contentvalue"> <div title="$QData{"State"}">$QData{"State"}</div></td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Language"}:</td>
          <td class="contentvalue"> <div title="$QData{"Language"}">$QData{"Language"}</div></td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Keywords"}:</td>
          <td class="contentvalue"> <div title="$QData{"Keywords"}">$QData{"Keywords","40"}</div></td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Result"}:</td>
          <td style="$Data{"CssColumnVotingResult"}"> $QData{"Result"} &#37;</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Votes"}:</td>
          <td class="contentvalue"> $QData{"Votes"}</td>
        </tr>

        <tr>
          <td>&nbsp;</td>
          <td>&nbsp;</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Created"}: </td>
          <td class="contentvalue"> $Quote{"$TimeLong{"$Data{"Created"}"}"}</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Changed"}: </td>
          <td class="contentvalue"> $Quote{"$TimeLong{"$Data{"Changed"}"}"}</td>
        </tr>

      </table>

<!-- dtl:block:VoteForm -->
      <h3>$Text{"$Config{"FAQ::Item::Voting::QuestionLangKey"}"}</h3>
      <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="spelling">
        <table width="100%">
          <tr>
<!-- dtl:block:VotingRateRow -->
            <td><input type="radio" name="Rate" value="$QData{"Value"}"><br> $QData{"Title"}</td>
<!-- dtl:block:VotingRateRow -->
          </tr>
        </table>

        <input type="submit" name="btn" value="$Text{"Submit"}" class="button">

        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="Vote">
        <input type="hidden" name="CategoryID" value="$Data{"CategoryID"}">
        <input type="hidden" name="ItemID" value="$Data{"ItemID"}">
      </form>
<!-- dtl:block:VoteForm -->

    </td>
  </tr>
</table>
<!-- dtl:block:View -->










<!-- dtl:block:Print -->

<h1>$Data{"Title"}</h1>

<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Config{"FAQ::FAQHook"}: $QData{"Number"}
    </td>
  </tr>
  <tr>
    <td class="contentbody">
      <table border="0" width="100%" cellspacing="0" cellpadding="2">
        <tr>
          <td class="contentkey">$Text{"Category"}: </td>
          <td class="contentvalue"> $Quote{"$Data{"CategoryName"}"}</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"State"}: </td>
          <td class="contentvalue"> $Quote{"$Data{"State"}"}</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Language"}: </td>
          <td class="contentvalue"> $Quote{"$Data{"Language"}"}</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Keywords"}: </td>
          <td class="contentvalue"> $Quote{"$Data{"Keywords"}"}</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Votes"}: </td>
          <td class="contentvalue"> $QData{"Votes"}</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Result"}: </td>
          <td class="contentvalue"> $QData{"Result"} &#37;</td>
        </tr>

        <tr>
          <td class="contentkey">$Text{"Last update"}: </td>
          <td class="contentvalue"> $Quote{"$TimeLong{"$Data{"Changed"}"}"}</td>
        </tr>
        <tr>
          <td colspan="2"><hr></td>
        </tr>

<!-- dtl:block:FAQItemField -->
        <tr>
          <td class="contentkey">$Text{"$Data{"Caption"}"}:  <br> <i class="small">($Text{"$Data{"StateName"}"}</i>) </td>
          <td class="contentvalue">$Data{"Value"}</td>
        </tr>
        <tr>
          <td colspan="2"><hr></td>
        </tr>
<!-- dtl:block:FAQItemField -->

      </table>
    </td>
  </tr>
</table>
<!-- dtl:block:Print -->








<!-- dtl:block:Search -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Search"}: $Config{"FAQ::FAQHook"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td align="center" class="mainbody">
      <br>
      <form action="$Env{"CGIHandle"}" method="post">
        <table width="600" border="0" cellspacing="0" cellpadding="3">
          <tr>
            <td class="contenthead">$Text{"Options"}</td>
          </tr>
          <tr>
            <td class="contentbody">
              <input type="hidden" name="Action" value="$Env{"Action"}">
              <input type="hidden" name="Subaction" value="Search">
              <input type="hidden" name="Submit" value="yes">
              <table border="0" cellspacing="0" cellpadding="3">
                <tr>
                  <td width="25%" class="contentkey">$Text{"Number"}: </td>
                  <td width="75%" class="contentvalue" colspan="3"> <input type="text" name="Number" value="$QData{"Number"}" size="50" maxlength="50"></td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Title"}: </td>
                  <td class="contentvalue" colspan="3"> <input type="text" name="Title" value="$QData{"Title"}" size="50" maxlength="50"></td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Fulltext"}: </td>
                  <td class="contentvalue" colspan="3"> <input type="text" name="What" value="$QData{"What"}" size="50" maxlength="50"></td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Keyword"}: </td>
                  <td class="contentvalue" colspan="3"> <input type="text" name="Keyword" value="$QData{"Keyword"}" size="50" maxlength="50"></td>
                </tr>
                <tr>
                  <td valign="top" class="contentkey">$Text{"Language"}:</td>
                  <td valign="top" class="contentvalue">$Data{"LanguageOption"}</td>
                  <td valign="top" class="contentkey">$Text{"Category"}:</td>
                  <td valign="top" class="contentvalue">$Data{"CategoryOption"}</td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter"> <input accesskey="g" class="button" type="submit" value="$Text{"Search"}"></td>
          </tr>
        </table>
      </form>
      <br>
    </td>
  </tr>
</table>
<!-- dtl:block:Search -->






<!-- dtl:block:SearchResult -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Search"}: $Config{"FAQ::FAQHook"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td class="mainbody">
      <table border="0" width="100%" cellspacing="0" cellpadding="2">
        <tr class="contenthead">
          <td class="contentkey">$Config{"FAQ::FAQHook"}</td>
          <td class="contentkey">$Text{"Title"}</td>
          <td class="contentkey">$Text{"Category"}</td>
          <td class="contentkey">$Text{"Language"}</td>
          <td class="contentkey">$Text{"State"}</td>
          <td class="contentkey">$Text{"Changed"}</td>
        </tr>
<!-- dtl:block:SearchResultRow -->
        <tr class="contentbody">
          <td class="$QData{"CssRow"}"><a href="$Env{"Baselink"}Action=$Env{"Action"}&ItemID=$QData{"ItemID"}">$QData{"Number"}</a></td>
          <td class="$QData{"CssRow"}">$QData{"Title"}</td>
          <td class="$QData{"CssRow"}">$QData{"CategoryName"}</td>
          <td class="$QData{"CssRow"}">$QData{"Language"}</td>
          <td class="$QData{"CssRow"}">$QData{"State"}</td>
          <td class="$QData{"CssRow"}">($TimeLong{"$Data{"Changed"}"})</td>
        </tr>
<!-- dtl:block:SearchResultRow -->
      </table>
    </td>
  </tr>
</table>
<!-- dtl:block:SearchResult -->








<!-- dtl:block:Explorer-->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td colspan="2" class="mainhead">
      $Env{"Box0"}$Text{"Explorer"}: $Config{"FAQ::FAQHook"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td width="75%" class="menu">
<!--<a href="$Env{"Baselink"}$Env{"LastScreenView"}">$Text{"Back"}</a>-->
    </td>
    <td width="35%" class="menu">
      &nbsp;
    </td>
  </tr>
  <tr>
    <td class="mainbody">


<!-- dtl:block:FAQPathCategoryElement -->
      $Config{"FAQ::Explorer::Path::Seperator"} <b><a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=$Env{"Subaction"}&CategoryID=$Data{"CategoryID"}">$QData{"Name"}</a></b>
<!-- dtl:block:FAQPathCategoryElement -->

<!-- dtl:block:ExplorerTitle -->
      <h1>$QData{"Name"}</h1>
      <p>$QData{"Comment"}</p>
<!-- dtl:block:ExplorerTitle -->



<!-- dtl:block:ExplorerCategoryList -->
      <table border="0" cellpadding="2" cellspacing="0" width="100%">
        <colgroup><col width="15" /><col width="*" /><col width="100" /><col width="100" /></colgroup>
        <tr class="contenthead">
          <td class="contentkey" colspan="2">
            $Text{"Name"} /
            $Text{"Comment"}
          </td>
          <td class="contentkey">
            $Text{"SubCategories"}
          </td>
          <td class="contentkey">
            $Text{"Article"}
          </td>
        </tr>
<!-- dtl:block:ExplorerCategoryRow -->
        <tr class="contentbody">
          <td class="$QData{"CssRow"}" valign="top"><img src="$Env{"Images"}$Config{"FAQ::Category::Icon"}"></td>
          <td class="$QData{"CssRow"}">
            <b><a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=$Env{"Subaction"}&CategoryID=$QData{"CategoryID"}">$QData{"Name"}</a></b>
            <br>$QData{"Comment"}&nbsp;
          </td>
          <td class="$QData{"CssRow"}">
            $QData{"CategoryNumber"}
          </td>
          <td class="$QData{"CssRow"}">
            $QData{"ArticleNumber"}
          </td>
        </tr>
<!-- dtl:block:ExplorerCategoryRow -->
        <tr class="contentfooter">
          <td colspan="5">&nbsp;</td>
        </tr>
      </table>
<!-- dtl:block:ExplorerCategoryList -->

      <br>

<!-- dtl:block:ExplorerFAQItemList -->
      <table border="0" width="100%" cellspacing="0" cellpadding="2">
        <tr class="contenthead">
          <td class="contentkey">
            $Text{"Title"}
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$QData{"CategoryID"}&Order=Title&Sort=up"><img border="0" src="$Env{"Images"}up-small.png" alt="$Text{"up"}"></a>&nbsp;/
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$QData{"CategoryID"}&Order=Title&Sort=down"><img border="0" src="$Env{"Images"}down-small.png" alt="$Text{"down"}"></a>
          </td>
          <td class="contentkey">
            $Text{"Language"}
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$QData{"CategoryID"}&Order=Language&Sort=up"><img border="0" src="$Env{"Images"}up-small.png" alt="$Text{"up"}"></a>&nbsp;/
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$QData{"CategoryID"}&Order=Language&Sort=down"><img border="0" src="$Env{"Images"}down-small.png" alt="$Text{"down"}"></a>
          </td>
          <td class="contentkey">
            $Text{"State"}
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$QData{"CategoryID"}&Order=State&Sort=up"><img border="0" src="$Env{"Images"}up-small.png" alt="$Text{"up"}"></a>&nbsp;/
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$QData{"CategoryID"}&Order=State&Sort=down"><img border="0" src="$Env{"Images"}down-small.png" alt="$Text{"down"}"></a>
          </td>
          <td class="contentkey">
            $Text{"Result"} (&#37;)
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$QData{"CategoryID"}&Order=Result&Sort=up"><img border="0" src="$Env{"Images"}up-small.png" alt="$Text{"up"}"></a>&nbsp;/
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$QData{"CategoryID"}&Order=Result&Sort=down"><img border="0" src="$Env{"Images"}down-small.png" alt="$Text{"down"}"></a>
          </td>
          <td class="contentkey">
            $Text{"Votes"}
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$Data{"CategoryID"}&Order=Votes&Sort=up"><img border="0" src="$Env{"Images"}up-small.png" alt="$Text{"up"}"></a>&nbsp;/
            <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$Data{"CategoryID"}&Order=Votes&Sort=down"><img border="0" src="$Env{"Images"}down-small.png" alt="$Text{"down"}"></a>
          </td>
        </tr>

<!-- dtl:block:ExplorerFAQItemRow -->
        <tr class="contentbody">
          <td class="$QData{"CssRow"}"><img src="$Env{"Images"}$Config{"FAQ::Item::Icon"}" style="vertical-align:middle;">&nbsp;<a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$QData{"CategoryID"}&ItemID=$QData{"ItemID"}" title="$QData{"Title"}">$QData{"Title","60"}</a></td>
          <td class="$QData{"CssRow"}">$QData{"Language"}</td>
          <td class="$QData{"CssRow"}">$QData{"State"}</td>
          <td class="$QData{"CssRow"}" style="$Data{"CssColumnVotingResult"}">$QData{"Result"}</td>
          <td class="$QData{"CssRow"}">$QData{"Votes"}</td>
        </tr>
<!-- dtl:block:ExplorerFAQItemRow -->
        <tr class="contentfooter">
          <td colspan="5">&nbsp;</td>
        </tr>
      </table>

<!-- dtl:block:ExplorerFAQItemList -->





    </td>
    <td class="menu">


<!-- dtl:block:ExplorerQuickSearch -->
      <h3>$Text{"QuickSearch"}</h3>
      <form action="$Env{"CGIHandle"}" method="get" style="margin:0px;padding:0px;">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="Search">
        <input type="hidden" name="Submit" value="yes">

<!-- dtl:block:ExplorerQuickSearchCategory -->
        <input type="hidden" name="CategoryIDs" value="$Data{"CategoryID"}">
<!-- dtl:block:ExplorerQuickSearchCategory -->

        <input type="hidden" name="QuickSearch" value="1">
        <input type="hidden" name="CategoryIDs" value="$Data{"CategoryID"}">
        <input type="text" name="What" value="$QData{"What"}" size="30" maxlength="50"><input class="button" type="submit" value="$Text{"Search"}"><br>
        <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Search">$Text{"DetailSearch"}</a>
      </form>
<!-- dtl:block:ExplorerQuickSearch -->


<!-- dtl:block:ExplorerLatestCreate -->
      <h3>$Text{"LatestCreatedItems"}</h3>
      <table>
<!-- dtl:block:ExplorerLatestCreateFAQItemRow -->
        <tr>
          <td valign="top"><img src="$Env{"Images"}$Config{"FAQ::Item::Icon"}"></td>
          <td><a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$QData{"CategoryID"}&ItemID=$QData{"ItemID"}" title="$QData{"Title"} ($QData{"CategoryName"})">$QData{"Title","40"} ($QData{"CategoryName","40"})</a>
            <br/>
            <i class="small">($QData{"Language"}) $TimeShort{"$Data{"Created"}"}</i>
          </td>
        </tr>
<!-- dtl:block:ExplorerLatestCreateFAQItemRow -->
      </table>
<!-- dtl:block:ExplorerLatestCreate -->


<!-- dtl:block:ExplorerLatestChange -->
      <h3>$Text{"LatestChangedItems"}</h3>
      <table>
<!-- dtl:block:ExplorerLatestChangeFAQItemRow -->
        <tr>
          <td valign="top"><img src="$Env{"Images"}$Config{"FAQ::Item::Icon"}"></td>
          <td><a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$QData{"CategoryID"}&ItemID=$QData{"ItemID"}" title="$QData{"Title"} ($QData{"CategoryName"})">$QData{"Title","40"} ($QData{"CategoryName","40"})</a>
            <br/>
            <i class="small">($QData{"Language"}) $TimeShort{"$Data{"Changed"}"}</i>
          </td>
        </tr>
<!-- dtl:block:ExplorerLatestChangeFAQItemRow -->
      </table>
<!-- dtl:block:ExplorerLatestChange -->


    </td>
  </tr>
</table>
<!-- dtl:block:Explorer -->

# --
# AgentFAQ.dtl - provides HTML view for faq articles
# Copyright (C) 2001-2008 OTRS AG, http://otrs.org/
# --
# $Id: AgentFAQ.dtl,v 1.14 2008/03/19 10:54:57 martin Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --

<!-- dtl:block:View -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td colspan="2" class="mainhead">
      $Env{"Box0"}$Text{"View"}: $Config{"FAQ::FAQHook"} $QData{"Number"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td class="menu">
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$Data{"CategoryID"}">$Text{"Back"}</a> -
<!-- dtl:block:FAQItemViewLinkUpdate -->
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Update&CategoryID=$Data{"CategoryID"}&ItemID=$Data{"ItemID"}">$Text{"Edit"}</a> -
<!-- dtl:block:FAQItemViewLinkUpdate -->
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=History&CategoryID=$Data{"CategoryID"}&ItemID=$Data{"ItemID"}">$Text{"History"}</a> -
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Print&CategoryID=$Data{"CategoryID"}&ItemID=$Data{"ItemID"}" target="print">$Text{"Print"}</a>
<!-- dtl:block:FAQItemViewLinkDelete -->
      - <a href="$Env{"Baselink"}Action=AgentLinkObject&SourceObject=FAQ&CategoryID=$Data{"CategoryID"}&SourceID=$Data{"ItemID"}">$Text{"Link"}</a> -
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Delete&CategoryID=$Data{"CategoryID"}&ItemID=$Data{"ItemID"}">$Text{"Delete"}</a>
<!-- dtl:block:FAQItemViewLinkDelete -->
    </td>
    <td align="right" class="menu">

    </td>
  </tr>
  <tr>
    <td width="75%" class="mainbody">


<!-- dtl:block:FAQPathCategoryElement -->
      $Config{"FAQ::Explorer::Path::Seperator"} <b><a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$Data{"CategoryID"}">$QData{"Name"}</a></b>
<!-- dtl:block:FAQPathCategoryElement -->

<!-- dtl:block:FAQPathItemElement -->
      $Config{"FAQ::Explorer::Path::Seperator"} <b><a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$QData{"CategoryID"}&ItemID=$QData{"ItemID"}">$QData{"Title"}</a></b>
<!-- dtl:block:FAQPathItemElement -->


      <h1>$QData{"Title"}</h1>

      <table border="0" width="100%" cellspacing="0" cellpadding="4">
        <tr>
          <td colspan="2" class="contenthead">$Text{"Options"}</td>
        </tr>
        <tr>
          <td class="contentbody">
            <table border="0" cellspacing="0" cellpadding="3">
<!-- dtl:block:FAQItemViewAttachment -->
              <tr>
                <td class="contentkey">$Text{"Attachment"}:</td>
                <td class="contentvalue"> <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Download&ItemID=$QData{"ItemID"}">$QData{"Filename"}</a></td>
              </tr>
<!-- dtl:block:FAQItemViewAttachment -->
<!-- dtl:block:FAQItemField -->
              <tr>
                <td width="25%" class="contentkey" valign="top">$Text{"$Data{"Caption"}"}:  <br> <i class="small">($Text{"$Data{"StateName"}"})</i></td>
                <td width="75%" class="contentvalue">$Data{"Value"}</td>
              </tr>
<!-- dtl:block:FAQItemField -->

            </table>
          </td>
        </tr>
        <tr>
          <td colspan="2" class="contenthead">&nbsp;</td>
        </tr>
      </table>

    </td>
    <td width="35%" class="menu">
      <table border="0" cellspacing="1" cellpadding="0" width="95%">
        <tr>
          <td class="contentkey">$Text{"Category"}:</td>
          <td class="contentvalue"> <div title="$QData{"CategoryName"}">$QData{"CategoryName","25"}</div></td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"State"}:</td>
          <td class="contentvalue"> <div title="$QData{"State"}">$QData{"State"}</div></td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Language"}:</td>
          <td class="contentvalue"> <div title="$QData{"Language"}">$QData{"Language"}</div></td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Keywords"}:</td>
          <td class="contentvalue"> <div title="$QData{"Keywords"}">$QData{"Keywords","30"}</div></td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Result"}:</td>
          <td style="$Data{"CssColumnVotingResult"}"> $QData{"Result"} &#37;</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Votes"}:</td>
          <td class="contentvalue"> $QData{"Votes"}</td>
        </tr>

        <tr>
          <td>&nbsp;</td>
          <td>&nbsp;</td>
        </tr>

        <tr>
          <td class="contentkey">$Text{"Created"}:</td>
          <td class="contentvalue">$TimeLong{"$Data{"Created"}"} ($Data{"CreatedByLogin"})</td>
        </tr>
        <tr>
          <td class="contentkey">$Text{"Changed"}:</td>
          <td class="contentvalue">$TimeLong{"$Data{"Changed"}"} ($Data{"ChangedByLogin"})</td>
        </tr>
        <br>
        <tr>
          <td>&nbsp;</td>
          <td>&nbsp;</td>
        </tr>
# object links
        <tr>
          <td class="contentkey">$Text{"Linked"}:</td>
          <td class="contentvalue"></td>
        </tr>

<!-- dtl:block:Link -->
        <tr>
          <td class="contentkey">&nbsp;&nbsp;$Text{"$QData{"LinkType"}"}:</td>
          <td class="contentvalue">
<!-- dtl:block:LinkItem -->
            <a href="$Env{"Baselink"}$Data{"FrontendDest"}$QData{"ID"}" title="$QData{"Object"}: $QData{"Number"} $QData{"Title"}" $Data{"Css"}>$QData{"Text"}</a>
<!-- dtl:block:LinkItem -->
          </td>
        </tr>
<!-- dtl:block:Link -->

      </table>

<!-- dtl:block:VoteForm -->
      <h3>$Text{"$Config{"FAQ::Item::Voting::QuestionLangKey"}"}</h3>
      <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="spelling">
        <table width="100%">
          <tr>
<!-- dtl:block:VotingRateRow -->
            <td><input type="radio" name="Rate" value="$QData{"Value"}"><br> $QData{"Title"}</td>
<!-- dtl:block:VotingRateRow -->
          </tr>
        </table>

        <input type="submit" name="btn" value="$Text{"Submit"}" class="button">

        <input type="hidden" name="Action" value="AgentFAQ">
        <input type="hidden" name="Subaction" value="Vote">
        <input type="hidden" name="CategoryID" value="$Data{"CategoryID"}">
        <input type="hidden" name="ItemID" value="$Data{"ItemID"}">
      </form>
<!-- dtl:block:VoteForm -->

    </td>
  </tr>
</table>
<!-- dtl:block:View -->



<!-- dtl:block:ViewSmall -->
<script language="JavaScript" type="text/javascript">
<!--

function updateMessage()
{
    if (parent.opener.closed) {
        alert('$JSText{"The message being composed has been closed.  Exiting."}');
        this.close();
        return;
    }

    if (!parent.opener.document.compose) {
        alert('$JSText{"This window must be called from compose window"}');
        this.close();
        return;
    }

    /* copy subject */
    parent.opener.document.compose.Subject.value = document.spelling.Title.value;

    /* create short alias */
    var input = parent.opener.document.compose.Body;

    /* focus orig input field */
    input.focus();

    /* IE */
    if (typeof document.selection != 'undefined') {
        var range = parent.opener.document.selection.createRange();
        var insText = range.text;
        range.text = document.spelling.Body.value + insText;
        range = document.selection.createRange();
        range.moveStart('character', document.spelling.Body.value.length + insText.length);
        range.select();
    }
    /* gecko */
    else if (typeof input.selectionStart != 'undefined') {
        var Start = input.selectionStart;
        var End = input.selectionEnd;
        /* set focus to start, if curser it not dedicated set to the end (only done automaticaly if more then 400 chars avail)*/
        if (input.value.length > 400 && End == input.value.length && Start == input.value.length) {
            Start = 0;
            End = 0;
        }
        var insText = input.value.substring(Start, End);
        input.value = input.value.substr(0, Start) + document.spelling.Body.value + insText + input.value.substr(End);
        var pos;
        if (insText.length == 0) {
            pos = Start + document.spelling.Body.value.length;
        }
        else {
            pos = Start + document.spelling.Body.value.length + insText.length;
        }
        input.selectionStart = pos;
        input.selectionEnd = pos;
    }
    /* other, insert on top */
    else {
        parent.opener.document.compose.Body.value = document.spelling.Body.value + parent.opener.document.compose.Body.value;
    }

    this.close();
}

//-->
</script>

<form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="spelling">

  <table border="0" width="100%" cellspacing="0" cellpadding="3">
    <tr>
      <td colspan="2" class="mainhead">
        $Env{"Box0"}$Config{"FAQ::FAQHook"}: $QData{"Number"}$Env{"Box1"}
      </td>
    </tr>
    <tr align="left" valign="top">
      <td width="100%" class="mainbody">


<!-- dtl:block:FAQPathCategoryElement -->
        $Config{"FAQ::Explorer::Path::Seperator"} <b><a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Explorer&CategoryID=$Data{"CategoryID"}">$QData{"Name"}</a></b>
<!-- dtl:block:FAQPathCategoryElement -->

<!-- dtl:block:FAQPathItemElement -->
        $Config{"FAQ::Explorer::Path::Seperator"} <b><a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$QData{"CategoryID"}&ItemID=$QData{"ItemID"}">$QData{"Title"}</a></b>
<!-- dtl:block:FAQPathItemElement -->


        <h1>$QData{"Title"}</h1>

        <table border="0" cellspacing="0" cellpadding="3">
          <tr>
            <td class="contentkey">$Config{"FAQ::FAQHook"}: </td>
            <td class="contentvalue"> $QData{"Number"}</td>
          </tr>
          <tr>
            <td class="contentkey">$Text{"Title"}: </td>
            <td class="contentvalue"> $QData{"Title"}</td>
          </tr>
          <tr>
            <td class="contentkey">$Text{"Category"}: </td>
            <td class="contentvalue"> $QData{"Category"}</td>
          </tr>
          <tr>
            <td class="contentkey">$Text{"State"}: </td>
            <td class="contentvalue"> $QData{"State"}</td>
          </tr>
          <tr>
            <td class="contentkey">$Text{"Language"}: </td>
            <td class="contentvalue"> $QData{"Language"}</td>
          </tr>
          <tr>
            <td class="contentkey">$Text{"Keywords"}: </td>
            <td class="contentvalue"> $QData{"Keywords"}</td>
          </tr>
<!-- dtl:block:FAQItemViewAttachment -->
          <tr>
            <td class="contentkey">$Text{"Attachment"}: </td>
            <td class="contentvalue"> <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Download&ItemID=$QData{"ItemID"}">$QData{"Filename"}</a></td>
          </tr>
<!-- dtl:block:FAQItemViewAttachment -->
          <tr valign="top">
            <td class="contentkey">$Text{"Result"}:</td>
            <td style="$Data{"CssColumnVotingResult"}"> $QData{"Result"} &#37;</td>
          </tr>
          <tr valign="top">
            <td class="contentkey">$Text{"Votes"}:</td>
            <td class="contentvalue"> $QData{"Votes"}</td>
          </tr>
          <tr>
            <td colspan="2"><hr></td>
          </tr>
          <tr>
            <td colspan="2">
              <input type="hidden" name="Title" value="$QData{"Title"}">
              <textarea name="Body" rows="14" cols="$Config{"Ticket::Frontend::TextAreaEmail"}" wrap="$Env{"BrowserWrap"}">$Data{"ItemFieldValues"}</textarea>
            </td>
          </tr>
          <tr>
            <td colspan="2"><hr></td>
          </tr>
        </table>

        <input class="button" type="button" onclick="updateMessage();" value="$Text{"Done"}">
        <i>($Text{"Return to the compose screen"})</i>

        <input class="button" type="button" onclick="window.close();" value="$Text{"Cancel"}">
        <i>($Text{"Discard all changes and return to the compose screen"})</i>

        <hr>

      </td>
    </tr>
  </table>
  </form>
<!-- dtl:block:ViewSmall -->




<!-- dtl:block:History -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"History"}: $Config{"FAQ::FAQHook"} $QData{"Number"}$Env{"Box1"}
    </td>
    </tr>
  <tr>
    <td class="menu">
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$Data{"CategoryID"}&ItemID=$Data{"ItemID"}">$Text{"Back"}</a>
    </td>
  </tr>
  <tr>
    <td class="mainbody">
    <table border="0" width="100%" cellspacing="0" cellpadding="2">
      <tr class="contenthead">
        <td class="contentkey">$Text{"Action"}</td>
        <td class="contentkey">$Text{"Time"}</td>
        <td class="contentkey">$Text{"User"}</td>
      </tr>
<!-- dtl:block:HistoryRow -->
      <tr class="contentbody">
        <td class="$Data{"CssRow"}">$Text{"$Data{"Name"}"}</td>
        <td class="$Data{"CssRow"}">($TimeLong{"$Data{"Created"}"})</td>
        <td class="$Data{"CssRow"}">$Data{"CreatedBy"}</td>
      </tr>
<!-- dtl:block:HistoryRow -->
    </table>
    </td>
  </tr>
</table>
<!-- dtl:block:History -->



<!-- dtl:block:Add -->
<script language="JavaScript" type="text/javascript">
<!--
function submit_compose()
{
    if (document.compose.Title.value == "") {
        alert('$JSText{"A article should have a title!"}');
        document.compose.Title.focus();
        return false;
    }
    else {
        return true;
    }
}
//-->
</script>

<table border="0" width="100%" cellspacing="0" cellpadding="3">
<tr>
  <td align="left" colspan="2" class="mainhead">
    $Env{"Box0"}$Text{"New FAQ Article"}$Env{"Box1"}
  </td>
</tr>
<tr>
  <td width="100%" class="mainbody">
    <br>
    <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="compose">
    <table border="0" width="700" align="center" cellspacing="0" cellpadding="4">
      <tr>
        <td colspan="2" class="contenthead">$Text{"Options"}</td>
      </tr>
      <tr>
        <td class="contentbody">
            <input type="hidden" name="Action" value="$Env{"Action"}">
            <input type="hidden" name="Subaction" value="AddAction">
            <input type="hidden" name="ItemID" value="$Data{"ItemID"}">
            <table border="0" cellspacing="0" cellpadding="3">
              <tr>
                <td class="contentkey">$Text{"Title"}: </td>
                <td class="contentvalue"> <input type="text" name="Title" value="$Data{"Title"}" size="75" maxlength="120"></td>
              </tr>
              <tr>
                <td class="contentkey">$Text{"Keywords"}: </td>
                <td class="contentvalue"> <input type="text" name="Keywords" value="$Data{"Keywords"}" size="75" maxlength="120"></td>
              </tr>
              <tr>
                <td width="20%" class="contentkey">$Text{"Category"}: </td>
                <td width="80%" class="contentvalue"> $Data{"CategoryOption"}</td>
              </tr>
              <tr>
                <td class="contentkey">$Text{"State"}: </td>
                <td class="contentvalue"> $Data{"StateOption"}</td>
              </tr>
              <tr>
                <td class="contentkey">$Text{"Language"}: </td>
                <td class="contentvalue"> $Data{"LanguageOption"} </td>
              </tr>
              <tr>
                <td class="contentkey">$Text{"Attachment"}:</td>
                <td class="contentvalue"><input name="file_upload" type="file" size="40" class="fixed">&nbsp;<a name="attachment"></a></td>
              </tr>
              <tr>
                <td colspan="2"><hr></td>
              </tr>

<!-- dtl:block:FAQItemField -->
              <tr>
                <td class="contentkey">$Text{"$Data{"Caption"}"}:  <br> <i class="small">($Text{"$Data{"StateName"}"})</i> </td>
                <td class="contentvalue"><textarea name="$Data{"Key"}" rows="12" cols="75" wrap="$Env{"BrowserWrap"}">$Data{"Value"}</textarea></td>
              </tr>
              <tr>
                <td colspan="2"><hr></td>
              </tr>
<!-- dtl:block:FAQItemField -->
                </table>
              </td>
            </tr>
            <tr>
              <td class="contentfooter"><input accesskey="g" class="button" type="submit" value="$Text{"Submit"}" onclick="return submit_compose();"></td>
            </tr>
          </table>
        </form>
        <br>
      </td>
    </tr>
  </table>
<!-- dtl:block:Add -->




<!-- dtl:block:Update -->
<script language="JavaScript" type="text/javascript">
<!--
function submit_compose()
{
    if (document.compose.Title.value == "") {
        alert('$JSText{"A article should have a title!"}');
        document.compose.Title.focus();
        return false;
    }
    else {
        return true;
    }
}
//-->
</script>


  <table border="0" width="100%" cellspacing="0" cellpadding="3">
    <tr>
      <td class="mainhead">
        $Env{"Box0"}$Text{"Edit"}: $Config{"FAQ::FAQHook"} $QData{"Number"}$Env{"Box1"}
      </td>
    </tr>
    <tr>
      <td class="menu">
        <a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$Data{"CategoryID"}&ItemID=$Data{"ItemID"}" class="menuitem">$Text{"Back"}</a>
      </td>
    </tr>
    <tr>
      <td class="mainbody">
        <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="compose">
        <table border="0" width="700" cellspacing="0" cellpadding="4" align="center">
          <tr>
            <td colspan="2" class="contenthead">$Text{"Options"}</td>
          </tr>
          <tr>
            <td class="contentbody">
                <input type="hidden" name="Action" value="$Env{"Action"}">
                <input type="hidden" name="Subaction" value="UpdateAction">
                <input type="hidden" name="ItemID" value="$Data{"ItemID"}">
                <table border="0" cellspacing="0" cellpadding="3">
                  <tr>
                    <td class="contentkey">$Text{"Title"}: </td>
                    <td class="contentvalue"> <input type="text" name="Title" value="$QData{"Title"}" size="75" maxlength="120"></td>
                  </tr>
                  <tr>
                    <td class="contentkey">$Text{"Keywords"}: </td>
                    <td class="contentvalue"> <input type="text" name="Keywords" value="$QData{"Keywords"}" size="75" maxlength="120"></td>
                  </tr>
                  <tr>
                    <td width="20%" class="contentkey">$Text{"Category"}: </td>
                    <td width="80%" class="contentvalue"> $Data{"CategoryOption"}</td>
                  </tr>
                  <tr>
                    <td class="contentkey">$Text{"State"}: </td>
                    <td class="contentvalue"> $Data{"StateOption"}</td>
                  </tr>
                  <tr>
                    <td class="contentkey">$Text{"Attachment"}:</td>
                    <td class="contentvalue">
                        <input name="file_upload" type="file" size="30" class="fixed">&nbsp;<a name="attachment"></a>
                        <br/>
                        $Data{"Filename"}
#                        <input class="button" type="submit" name="AttachmentDelete" value="$Text{"Delete"}" onclick="return submit_compose();">
                    </td>
                  </tr>
                  <tr>
                    <td class="contentkey">$Text{"Language"}: </td>
                    <td class="contentvalue"> $Data{"LanguageOption"} </td>
                  </tr>
                  <tr>
                    <td colspan="2"><hr></td>
                  </tr>
<!-- dtl:block:FAQItemField -->
                  <tr>
                    <td class="contentkey">$Text{"$Data{"Caption"}"}:  <br> <i class="small">($Text{"$Data{"StateName"}"})</i> </td>
                    <td class="contentvalue"><textarea name="$Data{"Key"}" rows="12" cols="75" wrap="$Env{"BrowserWrap"}">$QData{"Value"}</textarea></td>
                  </tr>
                  <tr>
                    <td colspan="2"><hr></td>
                  </tr>
<!-- dtl:block:FAQItemField -->
                  </table>
                  </td>
                </tr>
                <tr>
                  <td class="contentfooter"><input accesskey="g" class="button" type="submit" value="$Text{"Submit"}" onclick="return submit_compose();"></td>
                </tr>
              </table>
            </form>
          </td>
        </tr>
      </table>
<!-- dtl:block:Update -->



<!-- dtl:block:Delete -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"Delete"}: $Config{"Artefact::ArtefactHook"} $QData{"Number"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td class="menu">
      <a href="$Env{"Baselink"}Action=$Env{"Action"}&CategoryID=$Data{"CategoryID"}&ItemID=$Data{"ItemID"}" class="menuitem">$Text{"Back"}</a>
    </td>
  </tr>
  <tr>
    <td class="mainbody">
      <br>
      <table border="0" width="350" align="center" cellspacing="0" cellpadding="4">
        <tr>
          <td colspan="2" class="contenthead">$Text{"Options"}</td>
        </tr>
        <tr>
          <td class="contentbody">
            <table border="0" cellspacing="0" cellpadding="3">
              <tr>
                <td width="25%" class="contentkey">$Config{"FAQ::FAQHook"}:</td>
                <td width="75%" class="contentvalue">$QData{"Number"}</td>
              </tr>
              <tr>
                <td class="contentkey">$Text{"Title"}:</td>
                <td class="contentvalue">$Data{"Title"}</td>
              </tr>
              <tr>
                <td colspan="2">&nbsp;</td>
              </tr>
              <tr>
                <td colspan="2" class="contentkey">$Text{"Do you really want to delete this Object?"}:</td>
              </tr>
            </table>
          </td>
        </tr>
        <tr>
          <td colspan="2" class="contentfooter">
            <table width="100%" border="0" cellspacing="0" cellpadding="1">
              <tr>
                <td class="contentfooter">
                  <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data">
                    <input type="hidden" name="Action" value="$Env{"Action"}">
                    <input type="hidden" name="CategoryID" value="$QData{"CategoryID"}">
                    <input type="hidden" name="ItemID" value="$QData{"ItemID"}">
                    <input class="button" type="submit" value="$Text{"No"}">
                  </form>
                </td>
                <td width="10" class="contentfooter">
                  <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data">
                    <input type="hidden" name="Action" value="$Env{"Action"}">
                    <input type="hidden" name="Subaction" value="DeleteAction">
                    <input type="hidden" name="CategoryID" value="$QData{"CategoryID"}">
                    <input type="hidden" name="ItemID" value="$QData{"ItemID"}">
                    <input class="button" type="submit" value="$Text{"Yes"}">
                  </form>
                </td>
              </tr>
            </table>
          </td>
        </tr>
      </table>
      <br>
    </td>
  </tr>
</table>
<!-- dtl:block:Delete -->



<!-- dtl:block:SystemHistory -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td class="mainhead">
      $Env{"Box0"}$Text{"System History"}: $Config{"FAQ::FAQHook"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td align="center" class="mainbody">
      <table border="0" width="100%" cellspacing="0" cellpadding="2">
        <tr class="contenthead">
          <td class="contentkey">$Config{"FAQ::FAQHook"}</td>
          <td class="contentkey">$Text{"Title"}</td>
          <td class="contentkey">$Text{"Category"}</td>
          <td class="contentkey">$Text{"Name"}</td>
          <td class="contentkey">$Text{"Time"}</td>
        </tr>
<!-- dtl:block:SystemHistoryRow -->
        <tr class="contentbody">
          <td class="$Data{"CssRow"}"><a href="$Env{"Baselink"}Action=$Env{"Action"}&ItemID=$QData{"ItemID"}">$QData{"Number"}</a></td>
          <td class="$Data{"CssRow"}">$QData{"Subject"}</td>
          <td class="$Data{"CssRow"}">$QData{"Category"}</td>
          <td class="$Data{"CssRow"}">$QData{"Action"}</td>
          <td class="$Data{"CssRow"}">($TimeLong{"$Data{"Created"}"})</td>
        </tr>
<!-- dtl:block:SystemHistoryRow -->
      </table>
    </td>
  </tr>
</table>
<!-- dtl:block:SystemHistory -->


<!-- dtl:block:Language -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td colspan="2" class="mainhead">
      $Env{"Box0"}$Text{"FAQ Language"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td width="40%" class="mainbody">
      <form action="$Env{"CGIHandle"}" method="post">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="LanguageChange">
        <table border="0" width="100%" align="center" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Change"}</td>
          </tr>
          <tr>
            <td class="contentbody">$Data{"LanguageOption"}</td>
          </tr>
          <tr>
            <td class="contentfooter"><input class="button" type="submit" value="$Text{"Change"}"></td>
          </tr>
        </table>
      </form>
    </td>
    <td width="60%" class="mainbody">
      <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="compose">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="$Env{"Subaction"}Action">
        <input type="hidden" name="ID" value="$Data{"ID"}">
        <table border="0" width="100%" align="center" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"$Data{"Headline"}"} <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Language">$Text{"$Data{"AddLink"}"}</a></td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
                <tr>
                  <td class="contentkey">$Text{"Language"}: </td>
                  <td class="contentvalue"> <input type="text" name="Name" value="$QData{"Name"}" size="5" maxlength="50"></td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" value="$Text{"$Data{"Headline"}"}" onclick="return submit_compose();">
            </td>
          </tr>
        </table>
        </form>
      </td>
    </tr>
  </table>
<!-- dtl:block:Language -->



<!-- dtl:block:Category -->
<script language="JavaScript" type="text/javascript">
<!--
function submit_compose() {
    if (document.compose.Name.value == "") {
        alert('$JSText{"A category should have a name!"}');
        document.compose.Name.focus();
        return false;
    }
    else if (document.compose.Comment.value == "") {
        alert('$JSText{"A category should have a comment!"}');
        document.compose.Comment.focus();
        return false;
    }
    return true;
}
//-->
</script>
  <table border="0" width="100%" cellspacing="0" cellpadding="3">
    <tr>
      <td align="left" colspan="2" class="mainhead">
        $Env{"Box0"}$Text{"FAQ Category"}$Env{"Box1"}
      </td>
    </tr>
    <tr>
      <td width="40%" class="mainbody">
        <form action="$Env{"CGIHandle"}" method="post">
          <input type="hidden" name="Action" value="$Env{"Action"}">
          <input type="hidden" name="Subaction" value="CategoryChange">
          <table border="0" width="100%" align="center" cellspacing="0" cellpadding="4">
            <tr>
              <td class="contenthead">$Text{"Change"}</td>
            </tr>
            <tr>
              <td width="46%" class="contentbody">$Data{"CategoryLongOption"}</td>
            </tr>
            <tr>
              <td class="contentfooter"><input class="button" type="submit" value="$Text{"Change"}"></td>
            </tr>
          </table>
        </form>
      </td>
      <td width="60%" class="mainbody">
        <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" name="compose">
          <input type="hidden" name="Action" value="$Env{"Action"}">
          <input type="hidden" name="Subaction" value="$Data{"FormSubaction"}">
          <input type="hidden" name="CategoryID" value="$Data{"CategoryID"}">
          <table border="0" width="100%" align="center" cellspacing="0" cellpadding="4">
            <tr>
              <td class="contenthead">
                $Text{"$Data{"Headline"}"}
                <!-- dtl:block:CategoryAddLink -->
                <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Category">$Text{"(Click here to add)"}</a>
                <!-- dtl:block:CategoryAddLink -->
              </td>
            </tr>
            <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
                <tr>
                  <td class="contentvalue">
                    <table border="0" cellspacing="0" cellpadding="3">
                      <tr>
                        <td class="contentkey">$Text{"Category"}: </td>
                        <td class="contentvalue"> <input type="text" name="Name" value="$QData{"Name"}" size="35" maxlength="50"></td>
                      </tr>
                      <tr>
                        <td class="contentkey">$Text{"SubCategoryOf"}: </td>
                        <td class="contentbody">$Data{"CategoryOption"}</td>
                      </tr>
                      <tr>
                        <td class="contentkey">$Text{"Comment"}: </td>
                        <td class="contentvalue"> <input type="text" name="Comment" value="$QData{"Comment"}" size="50" maxlength="80"></td>
                      </tr>
                      <tr>
                        <td class="contentkey">$Text{"Valid"}: </td>
                        <td class="contentbody">$Data{"ValidOption"}</td>
                      </tr>
                      <tr>
                        <td class="contentkey">$Text{"Permission"}: </td>
                        <td class="contentkey">$Data{"GroupOption"}</td>
                      </tr>
                    </table>
                  </td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" onclick="return submit_compose();" value="$Text{"$Data{"Headline"}"}">
            </td>
          </tr>
        </table>
      </form>
    </td>
  </tr>
</table>
<!-- dtl:block:Category -->

IyAtLQojIEN1c3RvbWVyRkFRLmR0bCAtIHByb3ZpZGVzIEhUTUwgdmlldyBmb3IgZmFxIGFydGljbGVzCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDUgTWFydGluIEVkZW5ob2ZlciA8bWFydGluK2NvZGVAb3Rycy5vcmc+CiMgLS0KIyAkSWQ6IEN1c3RvbWVyRkFRLmR0bCx2IDEuMS4xLjEgMjAwNi8wNi8yOSAwOToyOTo1MSBjdCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC50eHQuCiMgLS0K
IyAtLQojIFB1YmxpY0ZBUS5kdGwgLSBwcm92aWRlcyBIVE1MIHZpZXcgZm9yIGZhcSBhcnRpY2xlcwojIENvcHlyaWdodCAoQykgMjAwMS0yMDA1IE1hcnRpbiBFZGVuaG9mZXIgPG1hcnRpbitjb2RlQG90cnMub3JnPgojIC0tCiMgJElkOiBQdWJsaWNGQVEuZHRsLHYgMS4xLjEuMSAyMDA2LzA2LzI5IDA5OjI5OjUxIGN0IEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLnR4dC4KIyAtLQo=
# --
# Kernel/System/FAQ.pm - all faq funktions
# Copyright (C) 2001-2008 OTRS AG, http://otrs.org/
# --
# $Id: FAQ.pm,v 1.16 2008/03/14 15:50:44 martin Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --

package Kernel::System::FAQ;

use strict;
use MIME::Base64;
use Kernel::System::Encode;
use Kernel::System::Group;
use Kernel::System::CustomerGroup;

use vars qw(@ISA $VERSION);
$VERSION = '$Revision: 1.16 $';
$VERSION =~ s/^\$.*:\W(.*)\W.+?$/$1/;

=head1 NAME

Kernel::System::FAQ - faq lib

=head1 SYNOPSIS

All faq functions. E. g. to add faqs or to get faqs.

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create a faq object

    use Kernel::Config;
    use Kernel::System::Log;
    use Kernel::System::DB;
    use Kernel::System::FAQ;

    my $ConfigObject = Kernel::Config->new();
    my $LogObject    = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        LogObject => $LogObject,
    );
    my $FAQObject = Kernel::System::FAQ->new(
        ConfigObject => $ConfigObject,
        LogObject => $LogObject,
        DBObject => $DBObject,
    );

=cut

sub new {
    my $Type = shift;
    my %Param = @_;

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

    # check needed objects
    foreach (qw(DBObject ConfigObject LogObject UserID)) {
        $Self->{$_} = $Param{$_} || die "Got no $_!";
    }
    $Self->{GroupObject} = Kernel::System::Group->new(%Param);
    $Self->{CustomerGroupObject} = Kernel::System::CustomerGroup->new(%Param);
    $Self->{EncodeObject} = Kernel::System::Encode->new(%Param);
    return $Self;
}

=item FAQGet()

get an article

    my %Article = $FAQObject->FAQGet(
        ItemID => 1,
    );

=cut

sub FAQGet {
    my $Self = shift;
    my %Param = @_;

    # Failures rename from ItemID to FAQID
    if($Param{FAQID}) {
        $Param{ItemID} = $Param{FAQID};
    }

    # check needed stuff
    foreach (qw(ItemID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return {};
        }
    }
    # db quote
    foreach (qw(ItemID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    my %Data = ();
    my $SQL = "SELECT i.f_name, i.f_language_id, i.f_subject, ".
            " i.f_field1, i.f_field2, i.f_field3, ".
            " i.f_field4, i.f_field5, i.f_field6, ".
            " i.free_key1, i.free_value1, i.free_key2, i.free_value2, ".
            " i.free_key3, i.free_value3, i.free_key4, i.free_value4, ".
            " i.created, i.created_by, i.changed, i.changed_by, ".
            " i.category_id, i.state_id, c.name, s.name, l.name, i.f_keywords, i.f_number, st.id, st.name ".
            " FROM faq_item i, faq_category c, faq_state s, faq_state_type st, faq_language l ".
            " WHERE ".
            " i.state_id = s.id AND ".
            " s.type_id = st.id AND ".
            " i.category_id = c.id AND ".
            " i.f_language_id = l.id AND ".
            " i.id = $Param{ItemID}";
    #$Self->{LogObject}->Log(Priority => 'error', Message => $SQL);
    $Self->{DBObject}->Prepare(SQL => $SQL);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        my %VoteData = %{$Self->ItemVoteDataGet(ItemID=>$Param{ItemID})};
        %Data = (
            # var for old versions
            ID => $Param{ItemID},
            FAQID => $Param{ItemID},
            #
            ItemID => $Param{ItemID},
            Name => $Row[0],
            LanguageID => $Row[1],
            Title => $Row[2],
            Field1 => $Row[3],
            Field2 => $Row[4],
            Field3 => $Row[5],
            Field4 => $Row[6],
            Field5 => $Row[7],
            Field6 => $Row[8],
            FreeKey1 => $Row[9],
            FreeText1 => $Row[10],
            FreeKey2 => $Row[11],
            FreeText2 => $Row[12],
            FreeKey3 => $Row[13],
            FreeText3 => $Row[14],
            FreeKey4 => $Row[15],
            FreeText4 => $Row[16],
            Created => $Row[17],
            CreatedBy => $Row[18],
            Changed => $Row[19],
            ChangedBy => $Row[20],
            CategoryID => $Row[21],
            StateID => $Row[22],
            CategoryName => $Row[23],
            State => $Row[24],
            Language => $Row[25],
            Keywords => $Row[26],
            Number => $Row[27],
            StateTypeID => $Row[28],
            StateTypeName => $Row[29],
            Result => sprintf("%0.".$Self->{ConfigObject}->Get("FAQ::Explorer::ItemList::VotingResultDecimalPlaces")."f",$VoteData{Result} || 0),
            Votes => $VoteData{Votes},
        );
    }
    if (!%Data) {
        $Self->{LogObject}->Log(Priority => 'error', Message => "No such ItemID $Param{ItemID}!");
        return;
    }
    # update number
    if (!$Data{Number}) {
        my $Number = $Self->{ConfigObject}->Get('SystemID')."00".$Data{ItemID};
        $Self->{DBObject}->Do(
            SQL => "UPDATE faq_item SET f_number = '$Number' WHERE id = $Data{ItemID}",
        );
        $Data{Number} = $Number;
    }
    # get attachment
    $SQL = "SELECT filename, content_type, content_size, content ".
        " FROM faq_attachment WHERE faq_id = $Param{ItemID}";
    $Self->{DBObject}->Prepare(SQL => $SQL, Encode => [1,1,1,0]);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        # decode attachment if it's a postgresql backend and not BLOB
        if (!$Self->{DBObject}->GetDatabaseFunction('DirectBlob')) {
            $Row[3] = decode_base64($Row[3]);
        }
        $Data{Filename} = $Row[0];
        $Data{ContentType} = $Row[1];
        $Data{ContentSize} = $Row[2];
        $Data{Content} = $Row[3];
    }
    my $Hash = $Self->GetCategoryTree();
    $Data{CategoryName} = $Hash->{$Data{CategoryID}};
    return %Data;
}

=item ItemVoteDataGet()

returns a hash with number and result of a item

    my %Flag = $FAQObject->ItemVoteDataGet(
        ItemID => 1,
    );

=cut
sub ItemVoteDataGet {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return 0;
        }
    }

    # db quote
    foreach (qw(ItemID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }

    my $SQL = " SELECT count(*), avg(rate) FROM faq_voting WHERE item_id = " . $Param{ItemID};

    my %Hash = ();
    $Self->{DBObject}->Prepare(SQL => $SQL, Limit => $Param{Limit} || 500);
    if(my @Data = $Self->{DBObject}->FetchrowArray()) {
        $Hash{Votes}  = $Data[0];
        $Hash{Result} = $Data[1];
    }
    return \%Hash;
}

=item FAQAdd()

add an article

    my $ItemID = $FAQObject->FAQAdd(
        Number => '13402',
        Title => 'Some Text',
        CategoryID => 1,
        StateID => 1,
        LanguageID => 1,
        Keywords => 'some keywords',
        Field1 => 'Problem...',
        Field2 => 'Solution...',
        FreeKey1 => 'Software',
        FreeText1 => 'Apache 3.4.2',
        FreeKey2 => 'OS',
        FreeText2 => 'OpenBSD 4.2.2',
        # attachment options (not required)
        Filename => $Filename,
        Content => $Content,
        ContentType => $ContentType,
    );

=cut

sub FAQAdd {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(CategoryID StateID LanguageID Title)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return 0;
        }
    }
    # check name
    if (!$Param{Name}) {
        $Param{Name} = time().'-'.rand(100);
    }
    # check number
    if (!$Param{Number}) {
        $Param{Number} = $Self->{ConfigObject}->Get('SystemID').rand(100);
    }
    # db quote (just not Content, use db Bind values)
    foreach (qw(Number Name Title Keywords Field1 Field2 Field3 Field4 Field5 Field6 FreeKey1 FreeText1 FreeKey2 FreeText2 FreeKey3 FreeText3 FreeKey4 FreeText4 Filename ContentType Filesize)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_});
    }
    foreach (qw(CategoryID StateID LanguageID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }

    foreach my $Type (qw(Field FreeKey FreeText)) {
        foreach (1..6) {
            if (!defined($Param{$Type.$_})) {
                $Param{$Type.$_} = '';
            }
        }
    }
    my $SQL = "INSERT INTO faq_item (f_number, f_name, f_language_id, f_subject, ".
            " category_id, state_id, f_keywords, ".
            " f_field1, f_field2, f_field3, f_field4, f_field5, f_field6, ".
            " free_key1, free_value1, free_key2, free_value2, ".
            " free_key3, free_value3, free_key4, free_value4, ".
            " created, created_by, changed, changed_by)".
            " VALUES ".
            " ('$Param{Number}', '$Param{Name}', $Param{LanguageID}, '$Param{Title}', ".
            " $Param{CategoryID}, $Param{StateID}, '$Param{Keywords}', ".
            " '$Param{Field1}', '$Param{Field2}', '$Param{Field3}', ".
            " '$Param{Field4}', '$Param{Field5}', '$Param{Field6}', ".
            " '$Param{FreeKey1}', '$Param{FreeText1}', ".
            " '$Param{FreeKey2}', '$Param{FreeText2}', ".
            " '$Param{FreeKey3}', '$Param{FreeText3}', ".
            " '$Param{FreeKey4}', '$Param{FreeText4}', ".
            " current_timestamp, $Self->{UserID}, ".
            " current_timestamp, $Self->{UserID})";

    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        # get id
        $Self->{DBObject}->Prepare(
            SQL => "SELECT id FROM faq_item WHERE ".
            "f_name = '$Param{Name}' AND f_language_id = $Param{LanguageID} ".
            " AND f_subject = '$Param{Title}'",
        );
        my $ID = 0;
        while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
            $ID = $Row[0];
        }
        # update number
        my $Number = $Self->{ConfigObject}->Get('SystemID')."00".$ID;
        $Self->{DBObject}->Do(
            SQL => "UPDATE faq_item SET f_number = '$Number' WHERE id = $ID",
        );
        # add attachment
        if($Param{Content} && $Param{ContentType} && $Param{Filename}) {
            $Self->AttachmentAdd(
                ItemID => $ID,
                Content => $Param{Content},
                ContentType => $Param{ContentType},
                Filename => $Param{Filename}
            );
        }

        $Self->FAQHistoryAdd(
            Name => 'Created',
            ItemID => $ID,
        );
        return $ID;
    }
    else {
        return;
    }

}

=item AttachmentAdd()

=cut

sub AttachmentAdd {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID Content ContentType Filename)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }

    # get attachment size
    {
        use bytes;
        $Param{Filesize} = length($Param{Content});
        no bytes;
    }
    # encode attachemnt if it's a postgresql backend!!!
    if (!$Self->{DBObject}->GetDatabaseFunction('DirectBlob')) {
        $Self->{EncodeObject}->EncodeOutput(\$Param{Content});
        $Param{Content} = encode_base64($Param{Content});
    }

    my $SQL = "INSERT INTO faq_attachment ".
        " (faq_id, filename, content_type, content_size, content, ".
        " created, created_by, changed, changed_by) " .
        " VALUES ".
        " ($Param{ItemID}, '$Param{Filename}', '$Param{ContentType}', '$Param{Filesize}', ?, ".
        " current_timestamp, $Self->{UserID}, current_timestamp, $Self->{UserID})";
    # write attachment to db
    if ($Self->{DBObject}->Do(SQL => $SQL, Bind => [\$Param{Content}])) {
        return 1;
    }
    return 0;
}

=item AttachmentGet()

=cut

sub AttachmentGet {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID Content ContentType Filename)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
}

=item AttachmentDelete()

=cut

sub AttachmentDelete {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID Content ContentType Filename)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
}

=item AttachmentSearch()

=cut

sub AttachmentSearch {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID Content ContentType Filename)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
}

=item FAQUpdate()

update an article

    $FAQObject->FAQUpdate(
        ItemID => 123,
        CategoryID => 1,
        StateID => 1,
        LanguageID => 1,
        Title => 'Some Text',
        Field1 => 'Problem...',
        Field2 => 'Solution...',
        FreeKey1 => 'Software',
        FreeText1 => 'Apache 3.4.2',
        FreeKey2 => 'OS',
        FreeText2 => 'OpenBSD 4.2.2',
        # attachment options (not required)
        Filename => $Filename,
        Content => $Content,
        ContentType => $ContentType,
    );

=cut

sub FAQUpdate {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID CategoryID StateID LanguageID Title)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # check name
    if (!$Param{Name}) {
        my %Article = $Self->FAQGet(%Param);
        $Param{Name} = $Article{Name};
    }
    # db quote (just not Content, use db Bind values)
    foreach (qw(Number Name Title Keywords Field1 Field2 Field3 Field4 Field5 Field6 FreeKey1 FreeText1 FreeKey2 FreeText2 FreeKey3 FreeText3 FreeKey4 FreeText4 Filename ContentType Filesize)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_});
    }
    foreach (qw(ItemID CategoryID StateID LanguageID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    # fill up empty stuff
    foreach my $Type (qw(Field FreeKey FreeText)) {
        foreach (1..6) {
            if (!$Param{$Type.$_}) {
                $Param{$Type.$_} = '';
            }
        }
    }

    my $SQL = "UPDATE faq_item SET f_name = '$Param{Name}', ".
            " f_language_id = $Param{LanguageID}, f_subject = '$Param{Title}', ".
            " category_id = $Param{CategoryID}, state_id = $Param{StateID}, ".
            " f_keywords = '$Param{Keywords}', ".
            " f_field1 = '$Param{Field1}', f_field2 = '$Param{Field2}', ".
            " f_field3 = '$Param{Field3}', f_field4 = '$Param{Field4}', ".
            " f_field5 = '$Param{Field5}', f_field6 = '$Param{Field6}', ".
            " free_key1 = '$Param{FreeKey1}', free_value1 = '$Param{FreeText1}', ".
            " free_key2 = '$Param{FreeKey2}', free_value2 = '$Param{FreeText2}', ".
            " free_key3 = '$Param{FreeKey3}', free_value3 = '$Param{FreeText3}', ".
            " free_key4 = '$Param{FreeKey4}', free_value4 = '$Param{FreeText4}', ".
            " changed = current_timestamp, changed_by = $Self->{UserID} ".
            " WHERE id = $Param{ItemID} ";

    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        # add attachment
        if ($Param{Content} && $Param{ContentType} && $Param{Filename}) {
            # get attachment size
            {
                use bytes;
                $Param{Filesize} = length($Param{Content});
                no bytes;
            }
            # encode attachemnt if it's a postgresql backend!!!
            if (!$Self->{DBObject}->GetDatabaseFunction('DirectBlob')) {
                $Self->{EncodeObject}->EncodeOutput(\$Param{Content});
                $Param{Content} = encode_base64($Param{Content});
            }
            # delete old attachment
            $Self->{DBObject}->Do(
                SQL => "DELETE FROM faq_attachment WHERE faq_id = $Param{ItemID}",
            );
            my $SQL = "INSERT INTO faq_attachment ".
                " (faq_id, filename, content_type, content_size, content, ".
                " created, created_by, changed, changed_by) " .
                " VALUES ".
                " ($Param{ItemID}, '$Param{Filename}', '$Param{ContentType}', '$Param{Filesize}', ?, ".
                " current_timestamp, $Self->{UserID}, current_timestamp, $Self->{UserID})";
            # write attachment to db
            if ($Self->{DBObject}->Do(SQL => $SQL, Bind => [\$Param{Content}])) {

            }
        }
        $Self->FAQHistoryAdd(
            Name => 'Updated',
            ItemID => $Param{ItemID},
        );
        return 1;
    }
    else {
        return;
    }
}

=item FAQCount()

count an article

    $FAQObject->FAQCount(
        CategoryIDs => [1,2,3,4],
    );

=cut

sub FAQCount {
    my $Self = shift;
    my %Param = @_;

    # check needed stuff
    foreach (qw(CategoryIDs ItemStates)) {
        if (!defined($Param{$_})) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }

    my $SQL = "";
    my $Ext = "";
    $SQL = "SELECT COUNT(*)" .
        " FROM faq_item i, faq_state s".
        " WHERE i.category_id IN ('${\(join '\', \'', @{$Param{CategoryIDs}})}')".
        " AND i.state_id = s.id";
    if ($Param{ItemStates} && ref($Param{ItemStates}) eq 'HASH' && %{$Param{ItemStates}}) {
        $Ext .= " AND s.type_id IN ('${\(join '\', \'', keys(%{$Param{ItemStates}}))}')";
    }
    $Ext .= " GROUP BY category_id";
    $SQL .= $Ext;

    $Self->{DBObject}->Prepare(SQL => $SQL, Limit => 200);

    if(my @Row = $Self->{DBObject}->FetchrowArray()) {
        return $Row[0];
    } else {
        return 0;
    }
}

=item VoteAdd()

add an article

    my $Ok = $FAQObject->VoteAdd(
        CreatedBy => 'Some Text',
        ItemID => '123456',
        IP => '54.43.30.1',
        Interface => 'Some Text',
        Rate => 100,
    );

=cut

sub VoteAdd {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(CreatedBy ItemID IP Interface)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return 0;
        }
    }

    # db quote
    foreach (qw(CreatedBy Interface IP)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_});
    }
    foreach (qw(ItemID Rate)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }

    my $SQL = "INSERT INTO faq_voting ( ".
        " created_by, item_id, ip, interface, rate, created".
        " ) VALUES (".
        " '$Param{CreatedBy}', ".
        " $Param{ItemID}, ".
        " '$Param{IP}', ".
        " '$Param{Interface}', ".
        " $Param{Rate}, ".
        " current_timestamp ".
        " )";
    #$Self->{LogObject}->Log(Priority => 'error', Message => $SQL);
    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        return 1;
    }
    else {
        return 0;
    }

}

=item VoteGet()

add an article

    my %VoteData = %{$FAQObject->VoteGet(
        CreateBy => 'Some Text',
        ItemID => '123456',
        IP     => '127.0.0.1',
        Interface => 'Some Text',
    )};

=cut

sub VoteGet {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(CreateBy ItemID Interface IP)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return {};
        }
    }

    # db quote
    foreach (qw(CreatedBy Interface IP)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_});
    }
    foreach (qw(ItemID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }

    my $Ext = "";
    my $SQL = " SELECT created_by, item_id, interface, ip, created, rate FROM faq_voting WHERE";

    # public
    if($Param{Interface} eq '3') {
        $Ext .= " ip LIKE '$Param{IP}' AND".
                " item_id = $Param{ItemID}";

    # customer
    } elsif($Param{Interface} eq '2') {
        $Ext .= " created_by LIKE '$Param{CreateBy}' AND".
                " item_id = $Param{ItemID}";

    # internal
    } elsif($Param{Interface} eq '1') {
        $Ext .= " created_by LIKE '$Param{CreateBy}' AND".
                " item_id = $Param{ItemID}";
    }
    $SQL .= $Ext;
    #$Self->{LogObject}->Log(Priority => 'error', Message => $SQL);

    $Self->{DBObject}->Prepare(SQL => $SQL);
    my %Data = ();
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        %Data = (
            CreatedBy => $Row[0],
            ItemID => $Row[1],
            Interface => $Row[2],
            IP => $Row[3],
            Created => $Row[4],
            Rate => $Row[5],
        );
    }
    if (!%Data) {
        #$Self->{LogObject}->Log(Priority => 'error', Message => "No voting for this faq article! Kernel::System::FAQ::VoteGet()");
        return {};
    }
    return \%Data;
}

=item VoteSearch()

returns a array with VoteIDs

    my @VoteIDs = @{$FAQObject->VoteSearch(
        ItemID => 1,
    )};

=cut

sub VoteSearch {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return [];
        }
    }

    # db quote
    foreach (qw()) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_});
    }
    foreach (qw(ItemID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }

    my $Ext = "";
    my $SQL = " SELECT id FROM faq_voting WHERE";

    if(defined($Param{ItemID})) {
        $Ext .= " item_id = " . $Param{ItemID};
    }

    $SQL .= $Ext;

    my @List = ();
    $Self->{DBObject}->Prepare(SQL => $SQL, Limit => $Param{Limit} || 500);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        push (@List, $Row[0]);
    }
    return \@List;
}

=item VoteDelete()

add an article

    my $Ok = $FAQObject->VoteDelete(
        VoteID => 1,
    );

=cut

sub VoteDelete {
    my $Self = shift;
    my %Param = @_;

    # check needed stuff
    foreach (qw(VoteID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return 0;
        }
    }

    # db quote
    foreach (qw(VoteID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }

    my $SQL = "DELETE FROM faq_voting WHERE id = ".$Param{VoteID};

    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        return 1;
    }
    else {
        return 0;
    }

}

=item FAQDelete()

delete an article

    $Flag = $FAQObject->FAQDelete(ItemID => 1);

=cut

sub FAQDelete {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return 0;
        }
    }
    # db quote
    foreach (qw(ItemID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    #if ($Self->AttachmentDelete(%Param)) {
        my @VoteIDs = @{$Self->VoteSearch(ItemID => $Param{ItemID})};
        foreach my $TmpVoteID (@VoteIDs) {
            if(!$Self->VoteDelete(VoteID => $TmpVoteID)) {
                return;
            }
        }
        if ($Self->FAQHistoryDelete(%Param)) {
            if ($Self->{DBObject}->Prepare(SQL => "DELETE FROM faq_item WHERE id = $Param{ItemID}")) {
                return 1;
            }
        }
    #}
    return 0;
}

=item FAQHistoryAdd()

add an history to an article

    $Flag = $FAQObject->FAQHistoryAdd(
        ItemID => 1,
        Name => 'Updated Article.',
    );

=cut

sub FAQHistoryAdd {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID Name)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return 0;
        }
    }
    # db quote
    foreach (qw(Name)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    foreach (qw(ItemID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    my $SQL = "INSERT INTO faq_history (name, item_id, ".
            " created, created_by, changed, changed_by)".
            " VALUES ".
            " ('$Param{Name}', $Param{ItemID}, ".
            " current_timestamp, $Self->{UserID}, ".
            " current_timestamp, $Self->{UserID})";

    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        return 1;
    }
    else {
        return 0;
    }
}

=item FAQHistoryGet()

get a array with hachref (Name, Created) with history of an article back

    my @Data = @{$FAQObject->FAQHistoryGet(
        ItemID => 1,
    )};

=cut

sub FAQHistoryGet {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return [];
        }
    }
    # db quote
    foreach (qw(ItemID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    my @Data = ();
    $Self->{DBObject}->Prepare(
        SQL => "SELECT name, created, created_by FROM faq_history WHERE item_id = $Param{ItemID}",
    );
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        my %Record = (
            Name => $Row[0],
            Created => $Row[1],
            CreatedBy => $Row[2],
        );
        push (@Data, \%Record);
    }
    return \@Data;
}

=item FAQHistoryDelete()

delete an history of an article

    $FAQObject->FAQHistoryDelete(
        ItemID => 1,
    );

=cut

sub FAQHistoryDelete {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ItemID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return 0;
        }
    }
    # db quote
    foreach (qw(ItemID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    $Self->{DBObject}->Prepare(
        SQL => "DELETE FROM faq_history WHERE item_id = $Param{ItemID}",
    );
    return 1;
}

=item HistoryGet()

get the system history

    my @Data = @{$FAQObject->HistoryGet()};

=cut

sub HistoryGet {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw()) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # query
    my $SQL = "SELECT i.id, h.name, h.created, h.created_by, c.name, i.f_subject, i.f_number FROM".
        " faq_item i, faq_state s, faq_history h, faq_category c WHERE".
        " s.id = i.state_id AND h.item_id = i.id AND i.category_id = c.id";
    if ($Param{States} && ref($Param{States}) eq 'ARRAY' && @{$Param{States}}) {
        $SQL .= " AND s.name IN ('${\(join '\', \'', @{$Param{States}})}') ";
    }
    $SQL .= ' ORDER BY created DESC';
    my @Data = ();
    $Self->{DBObject}->Prepare(SQL => $SQL, Limit => 200);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        my %Record = (
            ItemID => $Row[0],
            Action => $Row[1],
            Created => $Row[2],
            CreatedBy => $Row[3],
            Category => $Row[4],
            Subject => $Row[5],
            Number => $Row[6],
        );
        push (@Data, \%Record);
    }
    return \@Data;
}

=item CategoryList()

get the category list as hash

    my %Categories = %{$FAQObject->CategoryList(
        Valid => 1,
    )};

=cut

sub CategoryList {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw()) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return {};
        }
    }
    # sql
    my $SQL = 'SELECT id, parent_id, name FROM faq_category ';
    if(defined($Param{Valid})) {
        $SQL .= 'WHERE valid_id = 1';
    }
    my %Data = ();
    $Self->{DBObject}->Prepare(SQL => $SQL);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $Data{$Row[1]}{$Row[0]} = $Row[2];
    }

    return \%Data;
}

=item CategorySearch()

get the category search as hash

    my @CategorieIDs = @{$FAQObject->CategorySearch(
        Name => "Name"
    )};

=cut

sub CategorySearch {
    my $Self = shift;
    my %Param = @_;

    # sql
    my $SQL = "SELECT id FROM faq_category WHERE valid_id = 1 ";
    my $Ext = '';

    # WHERE
    if (defined($Param{Name})) {
        $Ext .= " AND name LIKE '%".$Self->{DBObject}->Quote($Param{Name})."%'";
    }
    elsif (defined($Param{ParentID})) {
        $Ext .= " AND parent_id = '".$Self->{DBObject}->Quote($Param{ParentID}, 'Integer')."'";
    }
    elsif (defined($Param{ParentIDs}) && ref($Param{ParentIDs}) eq 'ARRAY' && @{$Param{ParentIDs}}) {
        $Ext = " AND parent_id IN (";
        foreach my $ParentID (@{$Param{ParentIDs}}) {
            $Ext .= $Self->{DBObject}->Quote($ParentID, 'Integer').",";
        }
        $Ext = substr($Ext,0,-1);
        $Ext .= ")";
    }
    elsif (defined($Param{CategoryIDs}) && ref($Param{CategoryIDs}) eq 'ARRAY' && @{$Param{CategoryIDs}}) {
        $Ext = " AND id IN (";
        foreach my $CategoryID (@{$Param{CategoryIDs}}) {
            $Ext .= $Self->{DBObject}->Quote($CategoryID, 'Integer').",";
        }
        $Ext = substr($Ext,0,-1);
        $Ext .= ")";
    }
    #if (defined($Param{ValidID})) {
    #    $Ext .= " AND valid_id = '".$Self->{DBObject}->Quote($Param{ValidID}, 'Integer')."' ";
    #}

    # ORDER BY
    if ($Param{Order}) {
        $Ext .= " ORDER BY ";
        if($Param{Order} eq 'Name') {
            $Ext .= "name";
        }
        #default
        else {
            $Ext .= "name";
        }
        # SORT
        if ($Param{Sort}) {
            if ($Param{Sort} eq 'up') {
                $Ext .= " ASC";
            }
            elsif ($Param{Sort} eq 'down') {
                $Ext .= " DESC";
            }
        }
    }

    # SQL STATEMENT
    $SQL .= $Ext;

    my @List = ();
    $Self->{DBObject}->Prepare(SQL => $SQL, Limit => 500);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        push (@List, $Row[0]);
    }
    return \@List;
}

=item CategoryGet()

get a category as hash

    my %Category = $FAQObject->CategoryGet(
        ID => 1,
    );

=cut

sub CategoryGet {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(CategoryID)) {
        if (!defined($Param{$_})) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # db quote
    foreach (qw(CategoryID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    # sql
    my %Data = ();
    $Self->{DBObject}->Prepare(
        SQL => "SELECT id, parent_id, name, comments, valid_id FROM faq_category WHERE id = $Param{CategoryID} ",
    );
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        %Data = (
            CategoryID => $Row[0],
            ParentID => $Row[1],
            Name => $Row[2],
            Comment => $Row[3],
            ValidID => $Row[4],
        );
    }
    return %Data;
}

=item CategorySubCategoryIDList()

get all subcategory ids of of a category

    my %Category = $FAQObject->CategorySubCategorieIDList(
        ParentID => 1,
        ItemStates => [1,2,3]
    );

=cut

sub CategorySubCategoryIDList {
    my $Self = shift;
    my %Param = @_;

    # check needed stuff
    foreach (qw(ParentID ItemStates)) {
        if (!defined($Param{$_})) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return [];
        }
    }
    my @SubCategoryIDs = ();
    if ($Param{Mode} && $Param{Mode} eq 'Agent') {
        # add subcategoryids
        @SubCategoryIDs = @{$Self->AgentCategorySearch(
            ParentID => $Param{ParentID},
            States => $Param{ItemStates},
            Order => 'Created',
            Sort => 'down',
            UserID => $Param{UserID},
        )};
    }
    elsif ($Param{Mode} && $Param{Mode} eq 'Customer') {
        # add subcategoryids
        @SubCategoryIDs = @{$Self->CustomerCategorySearch(
            ParentID => $Param{ParentID},
            States => $Param{ItemStates},
            Order => 'Created',
            Sort => 'down',
            CustomerUser => $Param{CustomerUser},
        )};
    }
    else {
        # add subcategoryids
        @SubCategoryIDs = @{$Self->CategorySearch(
            ParentID => $Param{ParentID},
            States => $Param{ItemStates},
            Order => 'Created',
            Sort => 'down',
        )};
    }
    my @Result = @SubCategoryIDs;
    foreach my $SubCategoryID (@SubCategoryIDs) {
        my @Temp = @{$Self->CategorySubCategoryIDList(
            ParentID => $SubCategoryID,
            ItemStates => $Param{ItemStates},
            Mode => $Param{Mode},
            CustomerUser => $Param{CustomerUser},
            UserID => $Param{UserID},
        )};
        if(@Temp) {
            push(@Result, @Temp);
        }
    }
    return \@Result;
}

=item CategoryAdd()

add a category

    my $ID = $FAQObject->CategoryAdd(
        Name => 'Some Category',
        Comment => 'some comment ...',
    );

=cut

sub CategoryAdd {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ParentID Name)) {
        if (!defined($Param{$_})) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # db quote
    foreach (qw(Name Comment)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    foreach (qw(ParentID UserID ValidID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    my $SQL = "INSERT INTO faq_category (name, parent_id, comments, valid_id, ".
            " created, created_by, changed, changed_by)".
            " VALUES ".
            " ('$Param{Name}', $Param{ParentID}, '$Param{Comment}', $Param{ValidID}, ".
            " current_timestamp, $Self->{UserID}, ".
            " current_timestamp, $Self->{UserID})";
    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        # get new category id
        $SQL = "SELECT id ".
            " FROM " .
            " faq_category " .
            " WHERE " .
            " name = '$Param{Name}'";
        my $ID = '';
        $Self->{DBObject}->Prepare(SQL => $SQL);
        while (my @Row = $Self->{DBObject}->FetchrowArray()) {
            $ID = $Row[0];
        }
        # log notice
        $Self->{LogObject}->Log(
            Priority => 'notice',
            Message => "FAQCategory: '$Param{Name}' ID: '$ID' created successfully ($Self->{UserID})!",
        );
        return $ID;
    }
    else {
        return;
    }
}

=item CategoryUpdate()

update a category

    $FAQObject->CategoryUpdate(
        ID => 1,
        Name => 'Some Category',
        Comment => 'some comment ...',
    );

=cut

sub CategoryUpdate {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(CategoryID ParentID Name)) {
        if (!defined($Param{$_})) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # db quote
    foreach (qw(Name Comment)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    foreach (qw(CategoryID ParentID ValidID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }

    # sql
    my $SQL = "UPDATE faq_category SET ".
        " parent_id = $Param{ParentID}, ".
        " name = '$Param{Name}', ".
        " comments = '$Param{Comment}', ".
        " valid_id = '$Param{ValidID}', ".
        " changed = current_timestamp, changed_by = $Self->{UserID} ".
        " WHERE id = $Param{CategoryID}";
    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        # log notice
        $Self->{LogObject}->Log(
            Priority => 'notice',
            Message => "FAQCategory: '$Param{Name}' ID: '$Param{CategoryID}' updated successfully ($Self->{UserID})!",
        );
        return 1;
    }
    else {
        return;
    }
}

=item CategoryDuplicateCheck()

check a category

    $FAQObject->CategoryDuplicateCheck(
        ID => 1, # or
        Name => 'Some Name',
        ParentID => 1,
    );

=cut

sub CategoryDuplicateCheck {
    my $Self = shift;
    my %Param = @_;

    # db quote
    foreach (qw(Name)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    foreach (qw(ID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }

    # sql
    my $SQL = "SELECT id FROM faq_category WHERE ";
    if(defined($Param{Name})) {
        $SQL .= "name = '$Param{Name}' AND parent_id = $Param{ParentID} ";
        if(defined($Param{ID})) {
            $SQL .= "AND id != '$Param{ID}' ";
        }
    }
    $Self->{DBObject}->Prepare(SQL => $SQL);
    if (my @Row = $Self->{DBObject}->FetchrowArray()) {
        return 1;
    }
    return 0;
}

=item CategoryCount()

count an article

    $FAQObject->CategoryCount(
        ParentIDs => [1,2,3,4],
    );

=cut

sub CategoryCount {
    my $Self = shift;
    my %Param = @_;

    # check needed stuff
    foreach (qw(ParentIDs)) {
        if (!defined($Param{$_})) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }

    my $SQL = "";
    my $Ext = "";

    $SQL = "SELECT COUNT(*)" .
        " FROM faq_category WHERE valid_id = 1";

    if(defined($Param{ParentIDs})) {
        $Ext = " AND parent_id IN (";
        foreach my $ParentID (@{$Param{ParentIDs}}) {
            $Ext .= $Self->{DBObject}->Quote($ParentID,'Integer').",";
        }
        $Ext = substr($Ext,0,-1);
        $Ext .= ")";
    }
    $Ext .= " GROUP BY parent_id";

    $SQL .= $Ext;
    $Self->{DBObject}->Prepare(SQL => $SQL, Limit => 200);

    my %Data = ();
    if(my @Row = $Self->{DBObject}->FetchrowArray()) {
        return $Row[0];
    } else {
        return 0;
    }

}

=item StateTypeList()

get the state type list as hash

    my %StateTypes = $FAQObject->StateTypeList();

=cut

sub StateTypeList {
    my $Self = shift;
    my %Param = @_;

    my $SQL = '';
    my $Ext = '';
    $SQL = "SELECT id, name FROM faq_state_type";

    if($Param{Types}) {
        my @States = @{$Param{Types}};
        $Ext = " WHERE";
        foreach my $State (@States) {
            $Ext .= " name LIKE '".$Self->{DBObject}->Quote($State)."' OR";
        }
        $Ext = substr($Ext,0,-3);
    }
    $SQL .= $Ext;

    # sql
    my %List = ();
    $Self->{DBObject}->Prepare(SQL => $SQL);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $List{$Row[0]} = $Row[1];
    }
    return \%List;
}

=item StateList()

get the state list as hash

    my %States = $FAQObject->StateList();

=cut

sub StateList {
    my $Self = shift;
    my %Param = @_;
    # sql
    my %List = ();
    $Self->{DBObject}->Prepare(SQL => 'SELECT id, name FROM faq_state');
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $List{$Row[0]} = $Row[1];
    }
    return %List;
}

=item StateUpdate()

update a state

    $FAQObject->StateUpdate(
        ID => 1,
        Name => 'public',
        TypeID => 1,
    );

=cut

sub StateUpdate {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ID Name TypeID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # db quote
    foreach (qw(Name)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    foreach (qw(ID TypeID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    # sql
    my $SQL = "UPDATE faq_state SET name = '$Param{Name}', type_id = $Param{TypeID}, ".
        " WHERE id = $Param{ID}";
    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        return 1;
    }
    else {
        return;
    }
}

=item StateAdd()

add a state

    my $ID = $FAQObject->StateAdd(
        ID => 1,
        Name => 'public',
        TypeID => 1,
    );

=cut

sub StateAdd {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(Name TypeID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # db quote
    foreach (qw(Name)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    foreach (qw(TypeID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    my $SQL = "INSERT INTO faq_state (name, type_id) ".
        " VALUES ".
        " ('$Param{Name}', $Param{TypeID}) ";

    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        return 1;
    }
    else {
        return;
    }
}

=item StateGet()

get a state as hash

    my %State = $FAQObject->StateGet(
        ID => 1,
    );

=cut

sub StateGet {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # db quote
    foreach (qw(ID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    # sql
    my %Data = ();
    $Self->{DBObject}->Prepare(
        SQL => "SELECT id, name FROM faq_state WHERE id = $Param{ID}",
    );
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        %Data = (
            ID => $Row[0],
            Name => $Row[1],
            Comment => $Row[2],
        );
    }
    return %Data;
}

=item StateTypeGet()

get a state as hash

    my %State = $FAQObject->StateTypeGet(
        ID => 1, # or
        Name => 'internal',
    );

=cut

sub StateTypeGet {
    my $Self = shift;
    my %Param = @_;

    my $SQL = "";
    my $Ext = "";

    $SQL = "SELECT id, name ".
        " FROM faq_state_type WHERE";

    if(defined($Param{ID})) {
        $Ext .= " id = ".$Self->{DBObject}->Quote($Param{ID}, 'Integer')
    }
    elsif(defined($Param{Name})) {
        $Ext .= " name LIKE '".$Self->{DBObject}->Quote($Param{Name})."'"
    }
    $SQL .= $Ext;
    # sql
    my %Data = ();
    $Self->{DBObject}->Prepare(
        SQL => $SQL,
    );
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        %Data = (
            ID => $Row[0],
            Name => $Row[1],
        );
    }
    return \%Data;
}

=item LanguageList()

get the language list as hash

    my %Languages = $FAQObject->LanguageList();

=cut

sub LanguageList {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw()) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # sql
    my %List = ();
    $Self->{DBObject}->Prepare(SQL => 'SELECT id, name FROM faq_language');
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $List{$Row[0]} = $Row[1];
    }
    return %List;
}

=item LanguageUpdate()

update a language

    $FAQObject->LanguageUpdate(
        ID => 1,
        Name => 'Some Category',
    );

=cut

sub LanguageUpdate {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ID Name)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # db quote
    foreach (qw(Name)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    foreach (qw(ID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    # sql
    my $SQL = "UPDATE faq_language SET name = '$Param{Name}' ".
        " WHERE id = $Param{ID}";
    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        return 1;
    }
    else {
        return;
    }
}

=item LanguageDuplicateCheck()

check a language

    $FAQObject->LanguageDuplicateCheck(
        Name => 'Some Name',
        ID => 1, # for update
    );

=cut

sub LanguageDuplicateCheck {
    my $Self = shift;
    my %Param = @_;

    # db quote
    foreach (qw(Name)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    foreach (qw(ID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }

    # sql
    my $SQL = "SELECT id FROM faq_language WHERE ";
    if(defined($Param{Name})) {
        $SQL .= "name = '$Param{Name}' ";
    }
    if(defined($Param{ID})) {
        $SQL .= "AND id != '$Param{ID}' ";
    }
    $Self->{DBObject}->Prepare(SQL => $SQL);
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        return 1;
    }
    return 0;
}

=item LanguageAdd()

add a language

    my $ID = $FAQObject->LanguageAdd(
        Name => 'Some Category',
    );

=cut

sub LanguageAdd {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(Name)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # db quote
    foreach (qw(Name)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}) || '';
    }
    my $SQL = "INSERT INTO faq_language (name) ".
            " VALUES ".
            " ('$Param{Name}') ";

    if ($Self->{DBObject}->Do(SQL => $SQL)) {
        return 1;
    }
    else {
        return;
    }
}

=item LanguageGet()

get a language as hash

    my %Language = $FAQObject->LanguageGet(
        ID => 1,
    );

=cut

sub LanguageGet {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw(ID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    # db quote
    foreach (qw(ID)) {
        $Param{$_} = $Self->{DBObject}->Quote($Param{$_}, 'Integer');
    }
    # sql
    my %Data = ();
    $Self->{DBObject}->Prepare(
        SQL => "SELECT id, name FROM faq_language WHERE id = $Param{ID}",
    );
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        %Data = (
            ID => $Row[0],
            Name => $Row[1],
        );
    }
    return %Data;
}

=item FAQSearch()

search in articles

    my @IDs = $FAQObject->FAQSearch(
        Number => '*134*',
        What => '*some text*',
        Keyword => '*webserver*',
        States => ['public', 'internal'],
        Order => 'Changed',
        Sort => 'ASC',
        Limit => 150,
    );

=cut

sub FAQSearch {
    my $Self = shift;
    my %Param = @_;
    # check needed stuff
    foreach (qw()) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }

    # sql
    my $SQL = "SELECT i.id, count( v.item_id ) as votes, avg( v.rate ) as vrate".
        " FROM faq_item i ".
        " LEFT JOIN faq_voting v ON v.item_id = i.id".
        " LEFT JOIN faq_state s ON s.id = i.state_id".
        " WHERE";
    my $Ext = '';
    foreach my $Key (qw(f_subject f_keywords f_field1 f_field2 f_field3 f_field4 f_field5 f_field6)) {
        if ($Ext) {
            $Ext .= ' OR ';
        }
        else {
            $Ext .= ' (';
        }
        if ($Param{What}) {
            my @List = split(/;/, $Param{What});
            my $What = '';
            foreach my $Value (@List) {
                $Value = "\%$Value\%";
                $Value =~ s/\*/%/g;
                $Value =~ s/%%/%/g;
                if ($What) {
                    $What .= ' OR ';
                }
# prepared for OTRS 2.3.x
                my $ValueInsert = $Self->{DBObject}->Quote($Value);
#                my $ValueInsert = $Self->{DBObject}->Quote($Value, 'Like');
#                if ( $Self->{DBObject}->GetDatabaseFunction('NoLowerInLargeText') ) {
                if ( $Self->{DBObject}->GetDatabaseFunction('Type') eq 'mssql' ) {
                    $What .= " i.$Key LIKE '$ValueInsert'";
                }
# prepared for OTRS 2.3.x
#                elsif ( $Self->{DBObject}->GetDatabaseFunction('LcaseLikeInLargeText') ) {
                elsif ( $Self->{DBObject}->GetDatabaseFunction('Type') eq 'db2' ) {
                    $What .= " LCASE(i.$Key) LIKE LCASE('$ValueInsert')";
                }
                else {
                    $What .= " LOWER(i.$Key) LIKE LOWER('$ValueInsert')";
                }
            }
            $Ext .= $What;
        }
        else {
            $Ext .= " i.$Key LIKE '%'";
        }
    }
    $Ext .= ' )';
    if ($Param{Number}) {
        $Param{Number} =~ s/\*/%/g;
        $Param{Number} =~ s/%%/%/g;
        $Param{Number} = $Self->{DBObject}->Quote($Param{Number});
        $Ext .= " AND LOWER(i.f_number) LIKE LOWER('$Param{Number}')";
    }
    if ($Param{Title}) {
        $Param{Title} = "\%$Param{Title}\%";
        $Param{Title} =~ s/\*/%/g;
        $Param{Title} =~ s/%%/%/g;
        $Param{Title} = $Self->{DBObject}->Quote($Param{Title});
        $Ext .= " AND LOWER(i.f_subject) LIKE LOWER('".$Param{Title}."')";
    }
    if ($Param{LanguageIDs} && ref($Param{LanguageIDs}) eq 'ARRAY' && @{$Param{LanguageIDs}}) {
        $Ext .= " AND i.f_language_id IN (";
        foreach my $LanguageID (@{$Param{LanguageIDs}}) {
            $Ext .= $Self->{DBObject}->Quote($LanguageID, 'Integer').",";
        }
        $Ext = substr($Ext,0,-1);
        $Ext .= ")";
    }
    if ($Param{CategoryIDs} && ref($Param{CategoryIDs}) eq 'ARRAY' && @{$Param{CategoryIDs}}) {
        $Ext .= " AND (i.category_id IN  (";
        my $Counter=0;
        foreach my $CategoryID (@{$Param{CategoryIDs}}) {
            $Ext .= $Self->{DBObject}->Quote($CategoryID, 'Integer').",";
            $Counter++;
            if(!($Counter%500)) {
                $Ext = substr($Ext,0,-1);
                $Ext .= ")";
                $Ext .= " OR i.category_id IN  (";
            }
        }
        $Ext = substr($Ext,0,-1);
        $Ext .= "))";
    }
    if ($Param{States} && ref($Param{States}) eq 'HASH' && %{$Param{States}}) {
        $Ext .= " AND s.type_id IN (";
        foreach my $StateID (keys(%{$Param{States}})) {
            $Ext .= $Self->{DBObject}->Quote($StateID, 'Integer').",";
        }
        $Ext = substr($Ext,0,-1);
        $Ext .= ")";
    }
    if ($Param{Keyword}) {
        $Param{Keyword} = "\%$Param{Keyword}\%";
        $Param{Keyword} =~ s/\*/%/g;
        $Param{Keyword} =~ s/%%/%/g;
        $Param{Keyword} = $Self->{DBObject}->Quote($Param{Keyword});
# prepared for OTRS 2.3.x
#        $Param{Keyword} = $Self->{DBObject}->Quote($Param{Keyword}, 'Like');
#        if ( $Self->{DBObject}->GetDatabaseFunction('NoLowerInLargeText') ) {
        if ( $Self->{DBObject}->GetDatabaseFunction('Type') eq 'mssql' ) {
            $Ext .= " AND i.f_keywords LIKE '".$Param{Keyword}."'";
        }
# prepared for OTRS 2.3.x
#        elsif ( $Self->{DBObject}->GetDatabaseFunction('LcaseLikeInLargeText') ) {
        elsif ( $Self->{DBObject}->GetDatabaseFunction('Type') eq 'db2' ) {
            $Ext .= " AND LCASE(i.f_keywords) LIKE LCASE('".$Param{Keyword}."')";
        }
        else {
            $Ext .= " AND LOWER(i.f_keywords) LIKE LOWER('".$Param{Keyword}."')";
        }
    }
    $Ext .= " GROUP BY i.id, i.f_subject, i.f_language_id, i.created, i.changed, s.name, v.item_id";
    if ($Param{Order}) {
        $Ext .= " ORDER BY ";

        # title
        if ($Param{Order} eq 'Title') {
            $Ext .= "i.f_subject";
        }
        # language
        elsif ($Param{Order} eq 'Language') {
            $Ext .= "i.f_language_id";
        }
        # state
        elsif ($Param{Order} eq 'State') {
            $Ext .= "s.name";
        }
        # votes
        elsif ($Param{Order} eq 'Votes') {
            $Ext .= "votes";
        }
        # rates
        elsif ($Param{Order} eq 'Result') {
            $Ext .= "vrate";
        }
        # changed
        elsif ($Param{Order} eq 'Created') {
            $Ext .= "i.created";
        }
        # created
        elsif ($Param{Order} eq 'Changed') {
            $Ext .= "i.changed";
        }

        if ($Param{Sort}) {
            if ($Param{Sort} eq 'up') {
                $Ext .= " ASC";
            }
            elsif ($Param{Sort} eq 'down') {
                $Ext .= " DESC";
            }
        }
    }
    $SQL .= $Ext;
    my @List = ();
    $Self->{DBObject}->Prepare(SQL => $SQL, Limit => $Param{Limit} || 500);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        push (@List, $Row[0]);
    }
    return @List;
}
=item FAQPathListGet()

returns a category array

    my @IDs = $FAQObject->FAQPathListGet(
        CategoryID => 150,
    );

=cut

sub FAQPathListGet {
    my $Self = shift;
    my %Param = @_;

    my @CategoryList = ();
    my $TempCategoryID = $Param{CategoryID};
    while($TempCategoryID) {
        my %Data = $Self->CategoryGet(CategoryID => $TempCategoryID);
        if(%Data) {
            $CategoryList[$#CategoryList+1] = \%Data;
        }
        $TempCategoryID = $Data{ParentID};
    }
    @CategoryList = reverse(@CategoryList);

    return \@CategoryList;

}

=item GetCategoryTree()

get all categories as tree

    my $Hash = $FAQObject->GetCategoryTree(
        Valid => 0, # if 1 then check valid category
    );

=cut

sub GetCategoryTree {
    my $Self = shift;
    my %Param = @_;
    my %Hash = ();
    my $SQL = '';
    # sql
    $SQL = "SELECT id, parent_id, name FROM faq_category";
    if ($Param{Valid}) {
        $SQL .= " WHERE valid_id = 1";
    }
    $SQL .= " ORDER BY name";
    $Self->{DBObject}->Prepare(SQL => $SQL);
    # tree
    my @Data;
    while (my @Row = $Self->{DBObject}->FetchrowArray()) {
        push (@Data, \@Row);
    }
    foreach my $Row (@Data) {
        my @RowData = @{$Row};
        $Hash{$RowData[1]}{$RowData[0]} = $RowData[2];
    }
    # return tree
    my $Categories = $Self->_MakeTree(ParentID => 0, Parent => '', Hash => \%Hash, Tree => {},);
    if (!$Categories) {
        $Categories = {};
    }
    return $Categories;
}

sub _MakeTree {
    my $Self = shift;
    my %Param = @_;
    foreach my $ID (keys(%{$Param{Hash}->{$Param{ParentID}}})) {
        $Param{Tree}->{$ID} = $Param{Parent}.$Param{Hash}->{$Param{ParentID}}{$ID};
        if (defined($Param{Hash}->{$ID})) {
            $Self->_MakeTree(
                ParentID => $ID,
                Hash => $Param{Hash},
                Tree => $Param{Tree},
                Parent => $Param{Tree}->{$ID}.'::',
            )
        }
    }
    return $Param{Tree};
}

=item SetCategoryGroup()
set groups to a category

        $FAQObject->SetCategoryGroup(
            CategoryID => 3,
            GroupIDs => [2,4,1,5,77],
        );

=cut

sub SetCategoryGroup {
    my $Self = shift;
    my %Param = @_;
    my $SQL = '';
    # check needed stuff
    foreach (qw(CategoryID GroupIDs)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    $Param{CategoryID} = $Self->{DBObject}->Quote($Param{CategoryID}, 'Integer');
    # delete old groups
    $Self->{DBObject}->Do(
        SQL => "DELETE FROM faq_category_group WHERE category_id = $Param{CategoryID}",
    );
    # insert groups
    foreach my $Key (@{$Param{GroupIDs}}) {
        my $GroupID = $Self->{DBObject}->Quote($Key, 'Integer');
        $SQL = "INSERT INTO faq_category_group ".
            " (category_id, group_id, changed, changed_by, created, created_by) VALUES" .
            " ($Param{CategoryID}, $GroupID, current_timestamp, $Self->{UserID}, ".
            " current_timestamp, $Self->{UserID})";
        # write attachment to db
        if (!$Self->{DBObject}->Do(SQL => $SQL)) {
            return 0;
        }
    }
    return 1;
}

=item GetCategoryGroup()
get groups from a category

    $FAQObject->GetCategoryGroup(
        CategoryID => 3,
    );

=cut

sub GetCategoryGroup {
    my $Self = shift;
    my %Param = @_;
    my $SQL = '';
    my @Groups = ();
    # check needed stuff
    foreach (qw(CategoryID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    $Param{CategoryID} = $Self->{DBObject}->Quote($Param{CategoryID}, 'Integer');
    # get groups
    $SQL = "SELECT group_id FROM faq_category_group WHERE category_id = $Param{CategoryID}";
    $Self->{DBObject}->Prepare(SQL => $SQL);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        push(@Groups, $Row[0]);
    }
    return \@Groups;
}

=item GetAllCategoryGroup()
get all category-groups

    $FAQObject->GetAllCategoryGroup();

=cut

sub GetAllCategoryGroup {
    my $Self = shift;
    my %Param = @_;
    my $SQL = '';
    my %Groups = ();
    # get groups
    $SQL = "SELECT group_id, category_id FROM faq_category_group";
    $Self->{DBObject}->Prepare(SQL => $SQL);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $Groups{$Row[1]}->{$Row[0]} = 1;
    }
    return \%Groups;
}

=item GetUserCategories()
get all category-groups

    my $Hashref = $FAQObject->GetUserCategories(
        UserID => '123456',
        Type => 'rw'
    );

=cut

sub GetUserCategories {
    my $Self = shift;
    my %Param = @_;

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

    my $Categories = $Self->CategoryList(Valid => 1);
    my $CategoryGroups = $Self->GetAllCategoryGroup();
    my %UserGroups = $Self->{GroupObject}->GroupMemberList(
        UserID => $Param{UserID},
        Type => $Param{Type},
        Result => 'HASH',
    );

    my %Hash = ();
    $Self->_UserCategories(
        Categories => $Categories,
        CategoryGroups => $CategoryGroups,
        UserGroups => \%UserGroups,
        ParentID => 0,
        NewHash => \%Hash,
    );
    return \%Hash;
}

sub _UserCategories {
    my $Self = shift;
    my %Param = @_;
    my %Hash = ();
    foreach my $CategoryID (keys %{$Param{Categories}->{$Param{ParentID}}}) {
        # check category groups
        if (defined($Param{CategoryGroups}->{$CategoryID})) {
            # check user groups
            foreach my $GroupID (keys %{$Param{CategoryGroups}->{$CategoryID}}) {
                if (defined($Param{UserGroups}->{$GroupID})) {
                    # add category to new hash
                    $Hash{$CategoryID} = $Param{Categories}->{$Param{ParentID}}{$CategoryID};
                    last;
                }
            }
        }
        else {
            next;
        }
        # recursion
        $Self->_UserCategories(
            Categories => $Param{Categories},
            CategoryGroups => $Param{CategoryGroups},
            UserGroups => $Param{UserGroups},
            ParentID => $CategoryID,
            NewHash => $Param{NewHash},
        );
    }
    $Param{NewHash}->{$Param{ParentID}} = \%Hash;
    return;
}

=item GetCustomerCategories()
get all category-groups

    my $Hashref = $FAQObject->GetCustomerCategories(
        CustomerUser => 'hans',
        Type => 'rw'
    );

=cut

sub GetCustomerCategories {
    my $Self = shift;
    my %Param = @_;

    # check needed stuff
    foreach (qw(CustomerUser Type)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }

    my $Categories = $Self->CategoryList(Valid => 1);
    my $CategoryGroups = $Self->GetAllCategoryGroup();
#    my %UserGroups = $Self->{GroupObject}->GroupMemberList(
#        UserID => $Param{UserID},
#        Type => $Param{Type},
#        Result => 'HASH',
#    );

    my %UserGroups = $Self->{CustomerGroupObject}->GroupMemberList(
        UserID => $Param{CustomerUser},
        Type => 'ro',
        Result => 'HASH',
    );
    my %Hash = ();
    $Self->_UserCategories(
        Categories => $Categories,
        CategoryGroups => $CategoryGroups,
        UserGroups => \%UserGroups,
        ParentID => 0,
        NewHash => \%Hash,
    );
    return \%Hash;
}

=item CheckCategoryUserPermission()
get userpermission from a category

    $FAQObject->CheckCategoryUserPermission(
        UserID => '123456',
        CategoryID => '123',
    );

=cut

sub CheckCategoryUserPermission {
    my $Self = shift;
    my %Param = @_;

    # check needed stuff
    foreach (qw(UserID CategoryID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    foreach my $Permission (qw(rw ro)) {
        my $Hash = $Self->GetUserCategories(
            UserID => $Param{UserID},
            Type => 'ro'
        );
        foreach my $ParentID (keys %{$Hash}) {
            my $CategoryHash = $Hash->{$ParentID};
            foreach my $CategoryID (keys %{$CategoryHash}) {
                if ($CategoryID == $Param{CategoryID}) {
                    return $Permission;
                }
            }
        }
    }
    return '';
}

=item CheckCategoryCustomerPermission()
get userpermission from a category

    $FAQObject->CheckCategoryCustomerPermission(
        UserID => '123456',
        CategoryID => '123',
    );

=cut

sub CheckCategoryCustomerPermission {
    my $Self = shift;
    my %Param = @_;

    # check needed stuff
    foreach (qw(CustomerUser CategoryID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    foreach my $Permission (qw(rw ro)) {
        my $Hash = $Self->GetCustomerCategories(
            CustomerUser => $Param{CustomerUser},
            Type => 'ro'
        );
        foreach my $ParentID (keys %{$Hash}) {
            my $CategoryHash = $Hash->{$ParentID};
            foreach my $CategoryID (keys %{$CategoryHash}) {
                if ($CategoryID == $Param{CategoryID}) {
                    return $Permission;
                }
            }
        }
    }
    return '';
}

=item AgentCategorySearch()

get the category search as hash

    my @CategorieIDs = @{$FAQObject->AgentCategorySearch(
        Name => "Name"
    )};

=cut

sub AgentCategorySearch {
    my $Self = shift;
    my %Param = @_;
    my @List = ();

    # check needed stuff
    foreach (qw(UserID)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return;
        }
    }
    if (!defined($Param{ParentID})) {
        $Param{ParentID} = 0;
    }
    my $Hashref = $Self->GetUserCategories(
        UserID => $Param{UserID},
        Type => 'ro'
    );
    my %CategoryHash = %{$Hashref->{$Param{ParentID}}};
    @List = sort {$CategoryHash{$a} cmp $CategoryHash{$b}} (keys %CategoryHash);
    return \@List;
}

=item CustomerCategorySearch()

get the category search as hash

    my @CategorieIDs = @{$FAQObject->CustomerCategorySearch(
        Name => "Name"
    )};

=cut

sub CustomerCategorySearch {
    my $Self = shift;
    my %Param = @_;
    my @List = ();

    # check needed stuff
    foreach (qw(CustomerUser)) {
        if (!$Param{$_}) {
            $Self->{LogObject}->Log(Priority => 'error', Message => "Need $_!");
            return ();
        }
    }
    if (!defined($Param{ParentID})) {
        $Param{ParentID} = 0;
    }
    my $Hashref = $Self->GetCustomerCategories(
        CustomerUser => $Param{CustomerUser},
        Type => 'ro',
    );
    my %CategoryHash = %{$Hashref->{$Param{ParentID}}};
    @List = sort {$CategoryHash{$a} cmp $CategoryHash{$b}} (keys %CategoryHash);
    return \@List;
}

=item PublicCategorySearch()

get the category search as hash

    my @CategorieIDs = @{$FAQObject->PublicCategorySearch(
        Name => "Name"
    )};

=cut

sub PublicCategorySearch {
    my $Self = shift;
    my %Param = @_;
    my $SQL = '';
    my $State = 0;

    # get 'public' state id
    $SQL = "SELECT id from faq_state_type WHERE name = 'public'";
    $Self->{DBObject}->Prepare(SQL => $SQL, Limit => 500);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        $State = $Row[0];
    }

    # get category ids
    $SQL = "SELECT distinct(faq_item.category_id) FROM faq_item ".
        "LEFT JOIN faq_category ON faq_item.category_id = faq_category.id ".
        "WHERE faq_category.valid_id = 1 AND faq_item.state_id = $State";
    if (defined($Param{ParentID})) {
        $SQL .= " AND faq_category.parent_id = ".$Self->{DBObject}->Quote($Param{ParentID}, 'Integer');
    }

    my @List = ();
    $Self->{DBObject}->Prepare(SQL => $SQL, Limit => 500);
    while  (my @Row = $Self->{DBObject}->FetchrowArray()) {
        push (@List, $Row[0]);
    }
    return \@List;
}

1;

=back
=head1 TERMS AND CONDITIONS

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

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.

=cut

=head1 VERSION

$Revision: 1.16 $ $Date: 2008/03/14 15:50:44 $

=cut

IyAtLQojIEtlcm5lbC9TeXN0ZW0vTGlua09iamVjdC9GQVEucG0gLSB0byBsaW5rIGZhcSBvYmplY3RzCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDggT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBGQVEucG0sdiAxLjQgMjAwOC8wMy8wMiAyMzowMDo0NCBtYXJ0aW4gRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMi4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6TGlua09iamVjdDo6RkFROwoKdXNlIHN0cmljdDsKdXNlIEtlcm5lbDo6U3lzdGVtOjpGQVE7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gJyRSZXZpc2lvbjogMS40ICQnOwokVkVSU0lPTiA9fiBzL15cJC4qOlxXKC4qKVxXLis/JC8kMS87CgpzdWIgSW5pdCB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwogICAgbXkgJVBhcmFtID0gQF87CgogICAgJFNlbGYtPntGQVFPYmplY3R9ID0gS2VybmVsOjpTeXN0ZW06OkZBUS0+bmV3KCV7JFNlbGZ9KTsKCiAgICByZXR1cm4gMTsKfQoKc3ViIEZpbGxEYXRhTWFwIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CiAgICBteSAlUGFyYW0gPSBAXzsKICAgIGZvcmVhY2ggKHF3KElEKSkgewogICAgICAgIGlmICghJFBhcmFteyRffSkgewogICAgICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyhQcmlvcml0eSA9PiAnZXJyb3InLCBNZXNzYWdlID0+ICJOZWVkICRfISIpOwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQogICAgbXkgJUFydGljbGUgPSAkU2VsZi0+e0ZBUU9iamVjdH0tPkZBUUdldCgKICAgICAgICBGQVFJRCA9PiAkUGFyYW17SUR9LAogICAgKTsKICAgIHJldHVybiAoCiAgICAgICAgVGV4dCA9PiAnRjonLiRBcnRpY2xle051bWJlcn0sCiAgICAgICAgTnVtYmVyID0+ICRBcnRpY2xle051bWJlcn0sCiAgICAgICAgVGl0bGUgID0+ICRBcnRpY2xle1RpdGxlfSwKICAgICAgICBJRCA9PiAkUGFyYW17SUR9LAogICAgICAgIE9iamVjdCA9PiAnRkFRJywKICAgICAgICBGcm9udGVuZERlc3QgPT4gIkFjdGlvbj1BZ2VudEZBUSZJdGVtSUQ9IiwKICAgICk7Cn0KCnN1YiBCYWNrZW5kTGlua09iamVjdCB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwogICAgbXkgJVBhcmFtID0gQF87CiAgICByZXR1cm4gMTsKfQoKc3ViIEJhY2tlbmRVbmxpbmtPYmplY3QgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKICAgIG15ICVQYXJhbSA9IEBfOwogICAgcmV0dXJuIDE7Cn0KCnN1YiBMaW5rU2VhcmNoUGFyYW1zIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CiAgICBteSAlUGFyYW0gPSBAXzsKICAgIHJldHVybiAoCiAgICAgICAgeyBOYW1lID0+ICdGQVFOdW1iZXInLCBUZXh0ID0+ICdGQVEjJ30sCiAgICAgICAgeyBOYW1lID0+ICdGQVFUaXRsZScsIFRleHQgPT4gJ1RpdGxlJ30sCiAgICAgICAgeyBOYW1lID0+ICdGQVFGdWxsdGV4dCcsIFRleHQgPT4gJ0Z1bGx0ZXh0J30sCiAgICApOwp9CgpzdWIgTGlua1NlYXJjaCB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwogICAgbXkgJVBhcmFtID0gQF87CiAgICBteSBAUmVzdWx0V2l0aERhdGEgPSAoKTsKICAgIG15IEBSZXN1bHQgPSAkU2VsZi0+e0ZBUU9iamVjdH0tPkZBUVNlYXJjaCgKICAgICAgICBOdW1iZXIgPT4gJFBhcmFte0ZBUU51bWJlcn0sCiAgICAgICAgVGl0bGUgPT4gJFBhcmFte0ZBUVRpdGxlfSwKICAgICAgICBXaGF0ID0+ICRQYXJhbXtGQVFGdWxsdGV4dH0sCiAgICApOwogICAgZm9yZWFjaCAoQFJlc3VsdCkgewogICAgICAgIG15ICVBcnRpY2xlID0gJFNlbGYtPntGQVFPYmplY3R9LT5GQVFHZXQoRkFRSUQgPT4gJF8pOwogICAgICAgIHB1c2ggKEBSZXN1bHRXaXRoRGF0YSwKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgJUFydGljbGUsCiAgICAgICAgICAgICAgICBJRCA9PiAkQXJ0aWNsZXtJdGVtSUR9LAogICAgICAgICAgICB9LAogICAgICAgICk7CiAgICB9CiAgICByZXR1cm4gQFJlc3VsdFdpdGhEYXRhOwp9CgpzdWIgTGlua0l0ZW1EYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CiAgICBteSAlUGFyYW0gPSBAXzsKICAgIG15ICVBcnRpY2xlID0gJFNlbGYtPntGQVFPYmplY3R9LT5GQVFHZXQoCiAgICAgICAgSXRlbUlEID0+ICRQYXJhbXtJRH0sCiAgICApOwoKICAgIG15ICRCb2R5ID0gJyc7CiAgICBmb3JlYWNoICgxLi4xMCkgewogICAgICAgIGlmICgkQXJ0aWNsZXsiRmllbGQkXyJ9KSB7CiAgICAgICAgICAgICRCb2R5IC49ICRBcnRpY2xleyJGaWVsZCRfIn07CiAgICAgICAgfQogICAgfQoKICAgIHJldHVybiAoCiAgICAgICAgJUFydGljbGUsCiAgICAgICAgSUQgPT4gJEFydGljbGV7SXRlbUlEfSwKICAgICAgICBCb2R5ID0+ICRCb2R5LAogICAgICAgIERldGFpbExpbmsgPT4gIkFjdGlvbj1BZ2VudEZBUSZJdGVtSUQ9JFBhcmFte0lEfSIsCiAgICApOwp9CgoxOwo=
IyAtLQojIEZBUS50IC0gRkFRIHRlc3RzCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDggT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBGQVEudCx2IDEuMyAyMDA4LzAzLzA1IDIzOjQ2OjEyIG1hcnRpbiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0yLjAudHh0LgojIC0tCgp1c2UgS2VybmVsOjpTeXN0ZW06OkZBUTsKCiRTZWxmLT57RkFRT2JqZWN0fSA9IEtlcm5lbDo6U3lzdGVtOjpGQVEtPm5ldygleyRTZWxmfSwgVXNlcklEID0+IDEpOwoKbXkgJEZBUUlEID0gJFNlbGYtPntGQVFPYmplY3R9LT5GQVFBZGQoCiAgICBUaXRsZSA9PiAnU29tZSBUZXh0JywKICAgIENhdGVnb3J5SUQgPT4gMSwKICAgIFN0YXRlSUQgPT4gMSwKICAgIExhbmd1YWdlSUQgPT4gMSwKICAgIEtleXdvcmRzID0+ICdzb21lIGtleXdvcmRzJywKICAgIEZpZWxkMSA9PiAnUHJvYmxlbS4uLicsCiAgICBGaWVsZDIgPT4gJ1NvbHV0aW9uLi4uJywKICAgIEZyZWVLZXkxID0+ICdTb2Z0d2FyZScsCiAgICBGcmVlVGV4dDEgPT4gJ0FwYWNoZSAzLjQuMicsCiAgICBGcmVlS2V5MiA9PiAnT1MnLAogICAgRnJlZVRleHQyID0+ICdPcGVuQlNEIDQuMi4yJywKICAgIEZyZWVLZXkzID0+ICdLZXkzJywKICAgIEZyZWVUZXh0MyA9PiAnVmFsdWUzJywKICAgIEZyZWVLZXk0ID0+ICdLZXk0JywKICAgIEZyZWVUZXh0NCA9PiAnVmFsdWU0JywKKTsKCiRTZWxmLT5UcnVlKAogICAgJEZBUUlEIHx8IDAsCiAgICAiRkFRQWRkKCkgLSAxIiwKKTsKCm15ICVGQVEgPSAkU2VsZi0+e0ZBUU9iamVjdH0tPkZBUUdldCgKICAgIEl0ZW1JRCA9PiAkRkFRSUQsCik7CgpteSAlRkFRVGVzdCA9ICgKICAgIFRpdGxlID0+ICdTb21lIFRleHQnLAogICAgQ2F0ZWdvcnlJRCA9PiAxLAogICAgU3RhdGVJRCA9PiAxLAogICAgTGFuZ3VhZ2VJRCA9PiAxLAogICAgS2V5d29yZHMgPT4gJ3NvbWUga2V5d29yZHMnLAogICAgRmllbGQxID0+ICdQcm9ibGVtLi4uJywKICAgIEZpZWxkMiA9PiAnU29sdXRpb24uLi4nLAogICAgRnJlZUtleTEgPT4gJ1NvZnR3YXJlJywKICAgIEZyZWVUZXh0MSA9PiAnQXBhY2hlIDMuNC4yJywKICAgIEZyZWVLZXkyID0+ICdPUycsCiAgICBGcmVlVGV4dDIgPT4gJ09wZW5CU0QgNC4yLjInLAogICAgRnJlZUtleTMgPT4gJ0tleTMnLAogICAgRnJlZVRleHQzID0+ICdWYWx1ZTMnLAogICAgRnJlZUtleTQgPT4gJ0tleTQnLAogICAgRnJlZVRleHQ0ID0+ICdWYWx1ZTQnLAopOwoKZm9yIG15ICRUZXN0ICggc29ydCBrZXlzICVGQVFUZXN0ICkgewogICAgJFNlbGYtPklzKAogICAgICAgICRGQVF7JFRlc3R9IHx8IDAsCiAgICAgICAgJEZBUVRlc3R7JFRlc3R9IHx8ICcnLAogICAgICAgICJGQVFHZXQoKSAtICRUZXN0IiwKICAgICk7Cn0KCm15ICRGQVFVcGRhdGUgPSAkU2VsZi0+e0ZBUU9iamVjdH0tPkZBUVVwZGF0ZSgKICAgIEl0ZW1JRCA9PiAkRkFRSUQsCiAgICBDYXRlZ29yeUlEID0+IDEsCiAgICBTdGF0ZUlEID0+IDIsCiAgICBMYW5ndWFnZUlEID0+IDIsCiAgICBUaXRsZSA9PiAnU29tZSBUZXh0MicsCiAgICBLZXl3b3JkcyA9PiAnc29tZSBrZXl3b3JkczInLAogICAgRmllbGQxID0+ICdQcm9ibGVtLi4uMicsCiAgICBGaWVsZDIgPT4gJ1NvbHV0aW9uLi4uMicsCiAgICBGcmVlS2V5MSA9PiAnU29mdHdhcmUyJywKICAgIEZyZWVUZXh0MSA9PiAnQXBhY2hlIDMuNC4yMicsCiAgICBGcmVlS2V5MiA9PiAnT1MyJywKICAgIEZyZWVUZXh0MiA9PiAnT3BlbkJTRCA0LjIuMjInLAogICAgRnJlZUtleTMgPT4gJ0tleTMyJywKICAgIEZyZWVUZXh0MyA9PiAnVmFsdWUzMicsCiAgICBGcmVlS2V5NCA9PiAnS2V5NDInLAogICAgRnJlZVRleHQ0ID0+ICdWYWx1ZTQyJywKKTsKCiVGQVEgPSAkU2VsZi0+e0ZBUU9iamVjdH0tPkZBUUdldCgKICAgIEl0ZW1JRCA9PiAkRkFRSUQsCik7CgolRkFRVGVzdCA9ICgKICAgIFRpdGxlID0+ICdTb21lIFRleHQyJywKICAgIENhdGVnb3J5SUQgPT4gMSwKICAgIFN0YXRlSUQgPT4gMiwKICAgIExhbmd1YWdlSUQgPT4gMiwKICAgIEtleXdvcmRzID0+ICdzb21lIGtleXdvcmRzMicsCiAgICBGaWVsZDEgPT4gJ1Byb2JsZW0uLi4yJywKICAgIEZpZWxkMiA9PiAnU29sdXRpb24uLi4yJywKICAgIEZyZWVLZXkxID0+ICdTb2Z0d2FyZTInLAogICAgRnJlZVRleHQxID0+ICdBcGFjaGUgMy40LjIyJywKICAgIEZyZWVLZXkyID0+ICdPUzInLAogICAgRnJlZVRleHQyID0+ICdPcGVuQlNEIDQuMi4yMicsCiAgICBGcmVlS2V5MyA9PiAnS2V5MzInLAogICAgRnJlZVRleHQzID0+ICdWYWx1ZTMyJywKICAgIEZyZWVLZXk0ID0+ICdLZXk0MicsCiAgICBGcmVlVGV4dDQgPT4gJ1ZhbHVlNDInLAopOwoKZm9yIG15ICRUZXN0ICggc29ydCBrZXlzICVGQVFUZXN0ICkgewogICAgJFNlbGYtPklzKAogICAgICAgICRGQVFUZXN0eyRUZXN0fSB8fCAnJywKICAgICAgICAkRkFReyRUZXN0fSB8fCAwLAogICAgICAgICJGQVFHZXQoKSAtICRUZXN0IiwKICAgICk7Cn0KCm15ICRPayA9ICRTZWxmLT57RkFRT2JqZWN0fS0+Vm90ZUFkZCgKICAgIENyZWF0ZWRCeSA9PiAnU29tZSBUZXh0JywKICAgIEl0ZW1JRCA9PiAkRkFRSUQsCiAgICBJUCA9PiAnNTQuNDMuMzAuMScsCiAgICBJbnRlcmZhY2UgPT4gJzInLAogICAgUmF0ZSA9PiAxMDAsCik7CgokU2VsZi0+VHJ1ZSgKICAgICRPayB8fCAwLAogICAgIlZvdGVBZGQoKSIsCik7CgpteSAkVm90ZSA9ICRTZWxmLT57RkFRT2JqZWN0fS0+Vm90ZUdldCgKICAgIENyZWF0ZUJ5ID0+ICdTb21lIFRleHQnLAogICAgSXRlbUlEID0+ICRGQVFJRCwKICAgIElQID0+ICc1NC40My4zMC4xJywKICAgIEludGVyZmFjZSA9PiAnMicsCik7CgokU2VsZi0+SXMoCiAgICAkVm90ZS0+e0lQfSB8fCAwLAogICAgJzU0LjQzLjMwLjEnLAogICAgIlZvdGVHZXQoKSAtIElQIiwKKTsKCm15ICRGQVFJRDIgPSAkU2VsZi0+e0ZBUU9iamVjdH0tPkZBUUFkZCgKICAgIFRpdGxlID0+ICdUaXRsZScsCiAgICBDYXRlZ29yeUlEID0+IDEsCiAgICBTdGF0ZUlEID0+IDEsCiAgICBMYW5ndWFnZUlEID0+IDEsCiAgICBLZXl3b3JkcyA9PiAnJywKICAgIEZpZWxkMSA9PiAnUHJvYmxlbTEuLi4nLAogICAgRmllbGQyID0+ICdTb2x1dGlvbjEuLi4nLAogICAgRnJlZUtleTEgPT4gJ1NvZnR3YXJlMScsCiAgICBGcmVlVGV4dDEgPT4gJ0FwYWNoZSAzLjQuMicsCiAgICBGcmVlS2V5MiA9PiAnT1MnLAogICAgRnJlZVRleHQyID0+ICdPcGVuQlNEIDQuMi4yJywKICAgIEZyZWVLZXkzID0+ICdLZXkzJywKICAgIEZyZWVUZXh0MyA9PiAnVmFsdWUzJywKICAgIEZyZWVLZXk0ID0+ICdLZXk0JywKICAgIEZyZWVUZXh0NCA9PiAnVmFsdWU0JywKKTsKCiRTZWxmLT5UcnVlKAogICAgJEZBUUlEMiB8fCAwLAogICAgIkZBUUFkZCgpIC0gMiIsCik7CgpteSBARkFRSURzID0gJFNlbGYtPntGQVFPYmplY3R9LT5GQVFTZWFyY2goCiAgICBOdW1iZXIgPT4gJyonLAogICAgV2hhdCA9PiAnKnMqJywKICAgIEtleXdvcmQgPT4gJ3NvbWUqJywKICAgIFN0YXRlcyA9PiBbJ3B1YmxpYycsICdpbnRlcm5hbCddLAogICAgT3JkZXIgPT4gJ1ZvdGVzJywKICAgIFNvcnQgPT4gJ0FTQycsCiAgICBMaW1pdCA9PiAxNTAsCik7CgpteSAkRkFRU2VhcmNoRm91bmQgPSAwOwpteSAkRkFRU2VhcmNoRm91bmQyID0gMDsKZm9yIG15ICRGQVFJRFNlYXJjaCAoIEBGQVFJRHMgKSB7CiAgICBpZiAoICRGQVFJRFNlYXJjaCBlcSAkRkFRSUQgKSB7CiAgICAgICAgJEZBUVNlYXJjaEZvdW5kID0gMTsKICAgIH0KICAgIGlmICggJEZBUUlEU2VhcmNoIGVxICRGQVFJRDIgKSB7CiAgICAgICAgJEZBUVNlYXJjaEZvdW5kMiA9IDE7CiAgICB9Cn0KJFNlbGYtPlRydWUoCiAgICAkRkFRU2VhcmNoRm91bmQsCiAgICAiRkFRU2VhcmNoKCkgLSAkRkFRSUQiLAopOwokU2VsZi0+RmFsc2UoCiAgICAkRkFRU2VhcmNoRm91bmQyLAogICAgIkZBUVNlYXJjaCgpIC0gJEZBUUlEMiIsCik7CgpARkFRSURzID0gJFNlbGYtPntGQVFPYmplY3R9LT5GQVFTZWFyY2goCiAgICBOdW1iZXIgPT4gJyonLAogICAgVGl0bGUgPT4gJ3RJVExlJywKICAgIFdoYXQgID0+ICdsJywKICAgIFN0YXRlcyA9PiBbJ3B1YmxpYycsICdpbnRlcm5hbCddLAogICAgT3JkZXIgPT4gJ0NyZWF0ZWQnLAogICAgU29ydCA9PiAnQVNDJywKICAgIExpbWl0ID0+IDE1MCwKKTsKCiRGQVFTZWFyY2hGb3VuZCA9IDA7CiRGQVFTZWFyY2hGb3VuZDIgPSAwOwpmb3IgbXkgJEZBUUlEU2VhcmNoICggQEZBUUlEcyApIHsKICAgIGlmICggJEZBUUlEU2VhcmNoIGVxICRGQVFJRCApIHsKICAgICAgICAkRkFRU2VhcmNoRm91bmQgPSAxOwogICAgfQogICAgaWYgKCAkRkFRSURTZWFyY2ggZXEgJEZBUUlEMiApIHsKICAgICAgICAkRkFRU2VhcmNoRm91bmQyID0gMTsKICAgIH0KfQokU2VsZi0+RmFsc2UoCiAgICAkRkFRU2VhcmNoRm91bmQsCiAgICAiRkFRU2VhcmNoKCkgLSAkRkFRSUQiLAopOwokU2VsZi0+VHJ1ZSgKICAgICRGQVFTZWFyY2hGb3VuZDIsCiAgICAiRkFRU2VhcmNoKCkgLSAkRkFRSUQyIiwKKTsKCm15IEBWb3RlSURzID0gQHskU2VsZi0+e0ZBUU9iamVjdH0tPlZvdGVTZWFyY2goCiAgICBJdGVtSUQgPT4gJEZBUUlELAopfTsKCmZvciBteSAkVm90ZUlEICggQFZvdGVJRHMgKSB7CiAgICBteSAkVm90ZURlbGV0ZSA9ICRTZWxmLT57RkFRT2JqZWN0fS0+Vm90ZURlbGV0ZSgKICAgICAgICBWb3RlSUQgPT4gMSwKICAgICk7CiAgICAkU2VsZi0+VHJ1ZSgKICAgICAgICAkVm90ZURlbGV0ZSB8fCAwLAogICAgICAgICJWb3RlRGVsZXRlKCkiLAogICAgKTsKfQoKbXkgJEZBUURlbGV0ZSA9ICRTZWxmLT57RkFRT2JqZWN0fS0+RkFRRGVsZXRlKAogICAgSXRlbUlEID0+ICRGQVFJRCwKKTsKJFNlbGYtPlRydWUoCiAgICAkRkFRRGVsZXRlIHx8IDAsCiAgICAiRkFRRGVsZXRlKCkiLAopOwoKbXkgJEZBUURlbGV0ZTIgPSAkU2VsZi0+e0ZBUU9iamVjdH0tPkZBUURlbGV0ZSgKICAgIEl0ZW1JRCA9PiAkRkFRSUQyLAopOwokU2VsZi0+VHJ1ZSgKICAgICRGQVFEZWxldGUyIHx8IDAsCiAgICAiRkFRRGVsZXRlKCkiLAopOwoKMTsK
iVBORw0KGgoAAAANSUhEUgAAABIAAAAQCAIAAACUZLgLAAABgklEQVQokZWTMUsjQRiGny/GUvIDDjsLQRCikP8g2Jx2FoKFiAjRKxQEEQ5ETg5EBcHCvyCYzsLG0kIFjTZeJVhYaIJhszuzO/mu2M1uQhTxLQZ2Zp/vfeeFEVUFgPHVfygCKgIKXP0d4jOpqqoWVx680HnGJatxnnHFlQf9RHmgWL47+zPSsAAIiSeMDjyOlUMVBFE03r8+GAXyAM42wu4ILSTHxq9JQEFy0IIcCsWlq5vD8Tygka0HH11AmFmr9uz2J27q7JsF6AMXr0ofLG5W1xcGZ0uFlDi9Z3232sZarh6AsLHVNXtu5sdsqbB3gZM4enYUu4V1w8729cTPsf2prkDLJ0lJqZwN21hkagHO2Oc3po+TwSKdP7dZJTImC/lucH7wUoM0T2oiab8gOD/IKmlYIj+oeT1Mx1eMu2aGhZ4h9L373z1td6tSuZw/94EcQCtKx3wp12ymbtGrTxj4lcrll1gU+IDEL2B48ujp1iYn0nEjFaSjGaX5VM6w7+o/Xh/qfCGTpFEAAAAASUVORK5CYII=
iVBORw0KGgoAAAANSUhEUgAAAAwAAAARCAIAAABmX9r1AAABeklEQVQokYWRvUtCYRTGz1WuphDUIoX/gFQiREPgcKGpghanBqOWKAfBoKkWawsaLvedhYaGcCqTjFDpvhEO4VCD2od4DfwqvVlZ2IechmtXA8lnejjnx+E55zCICN2k6Uq0IEopIYRpEyGEUtqCKKUcxxn7B4LhM6lUjd8UfP6jl4aB4ziFYxDR4/FYx+zTU5OMRqMvbgNArDaXf3yOBvZMPR88z4MS/DYnF+SanPAiIiLKCa+fpnwH52azGRGbUPK+rBBywqsa8eIaABCxGTxfqaW1S0/JjQy7rC7Va9QrhkFEQsgra7IMjxh1LMtqRw27ACDhwvFJqCSleJ7XAIDNZlt3zZ5GwncZKVcoKsT+YWBtxeVwOJqT1CsE/Vt9OimdrWiNFufiZusZ+CsAqMvxRuPrvRRJ0dX2FqPihJDEZdQ5M5jOVmJXn0O2Cbfb3WHSWzXzXS8/ZEKBnfk/LdUJgmAft6r7C4LQAUJEURQVQhTF9nor0z/6AQncFoIjEATtAAAAAElFTkSuQmCC
iVBORw0KGgoAAAANSUhEUgAAABYAAAAWCAYAAADEtGw7AAAABmJLR0QA/wDpAIB4rBT8AAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH1gUPDBkpSJgCpAAABRhJREFUOMvVlWto3fUZxz+//+3cTy4nOznJqWmaxjQGpY1EITHOpZrVigEtoogS71GE0rlBYLCNbbgZdayUMelopdgO9IWV1tZ6r7fWqqm21SRq6+mapjnpyeXcci7/c/6X316IeyHK3g32hefFA9/n4eF5vjxf+H+D+G+EmXeOd+VOTt0V1J1fGCp+o6GezLFxckLZXY6v+PW1vxxJ/mjxkSPTQ7v3fLTiu/zTEwuR17a+2jN5/+iOmc4rk4XBQemePi2dZFI6c3Myf/fdctHjkye7rpqa/HC850cnHr73pV3JlP/mcFj/wPDoFCtc1nfxSOdjJ8ZwN2xAHR1FX7cOoWlIoJjOUHjrPQq/+S1n/vi7P9x05+2//35jDaBcNejvv6Th40+Xbs2WJTcONqF92Y71xGuwZg16Po+YmoJcDlwXMZPE7unj/LYd1IblD+94fj5rPLLl+CvTSW7o6Ahh2hLdUNk42MDt4jOM555Di8ehsxOEgHwe9+mnqdY38P4dD3DUU523rep8NpslFou9MDQ09Pfu7u6sNn3+Yk1ra3j9wPoIbx9dAFVlw/oYK3c8ie/L/YBAdncj+vogEsEtm1imhbH9GX6yezsXbvp5VFOU6M6dO/H7/Y8fOHBg47Zt24bUrstvC0x9Y49OnC7T0RHGF9RJpm16Zo4Su7IN66mnYGgIdB23UsHRdErXDbA0/CDnzl/kM6dMqVhkcmKCarXK3NxcS2NjY7/WN9BeE1nlcGqyRLkqaYz7WNkaoOm+xylZeZQ9e7DHxpDlMlJRcNJpquemkSObKdxwM4MHdxHL5VkDHAY+lpJ4PH6NFjEXRn6mzLPh4atxVRXFNFEmJ9H/uh9l714EoA0Po65diwyFsBcWcfftQ/3Ln4h39eCLBWlYWqRDwJ3AM0CNz3dWy3w+wcpfbUZtaUExDNRCAW12Frl6NXLLFpRNm9CiUdRUCmnbuJdfgXvLbRRffxdr7M/4SyWchQVcwAA2A28mEnu1Qks76tiTMP4J2DZOczNWby/092MsL6M9/zzqoUOQSIDjoHl9eAeuJz3yGJne66g7eRgnl/+PzFygPh5HUy6kcO69B3vkIZAShED/6isCW7fi2bcPJZXCRWDaEsuy8C5lCex6lhWH3yF5fQe1AzahTpfKGUFxXFIxBeHmZrSmNw7cGHpiFHnppSiOg7K0hDo/D6pKQTFIqD6WxDJWvYCgjhH0EKn3EP9wkVCzn/A1fsJVH5QcrCGTiX96qI03oznSaRNnEngSCWQ4TKWunploE2nSzE79C8UnMWI+alsNmtYZNP40hN7QQDHXSuCiieYVoHjBddCbXSIjIQKt0aJ48W//6K89eWq7Xsy1Wl6/lY3FLwR61y5HJkZ6tUyGgO4n2uanpj2MqKsB3QOaFxA4lo2omCjFNJQKUFzmm6Umjiz0RQVAJpNRTNO8WghRjsVip6ZPHLw/cvzWZ42AgV4TgmAYAkEIhsDrA90ACdhV3HyOUrqAWirgs5f5Ot3JJ9VNUQ2grq7OBT767rLpmc8vafE7oKugat+GroPhAX8APF5SF0qkzoVIZa9wFjP2qzo27bXJa1FETW101bff7fuouMYXp821rNLPomsCdAG6im05JKZMZtMtFKtt++uaul5oWR0/flVj41kpJV9/Mb6iUlh8NBaJmT/oIIcOvizijbV3zJ45tlHXlWFbuiAUbNdzLFi/6oOGWOuLgUDgRFtbm/0/97x/A0fIGqQPAQaeAAAAAElFTkSuQmCC
iVBORw0KGgoAAAANSUhEUgAAABYAAAAWCAYAAADEtGw7AAAABmJLR0QA/wD/AP+gvaeTAAAB3ElEQVQ4y7WVPW4UQRCFv/nxD7IEEgkiceAjWCKBhIwLEJmAtdCKY3AKSMYWSOQcgICIEyACRESAOICRmJnuriqC7vX02LPekaUdqVTV6u7Xb1696SnMjG089eN3f94f7FeLvaokeCE4JTjBpxyc4jfk4ATxioog/V/UXXyoD/arxbPjB/QO+g66HrouRQttB22bIq9b0BKcgROwMjItigtwnxf1nd2avgfnYngH3qcIEAKIpMhqzWKkZrVDwS713R3jycNhQwgQEkjIwue1H7L3hgiYGWYgzvj4qaS+twdPD4cDm6ZhuVxubI5ZBFNVVA1Vw8xwvfHla0l5G1esA13VmFGj84C7t4djO73+NQkax1Ca3s7HK7Y5qJkhqpgZ9ZQUTdOMxlOaD3JYatzqgBXwBOM5zZtinEtRz2meql1qmjfturZD3uiKqY03gaoqYDc3b93G9aAzpMg3mRl2dkQBVCv7vfhxpWlpXSJaTvl40vTXXLGGdSJaTxE+Pz8bjU9OXl4ynaN9BFYdLT49fTXJZs5bRSnSl6lieB/iAonNEY1a5XeBPv+GqqU5Rf91aNJUbQB1zsV7+dGb73Z0v0K8IEFi9mGog6D5XMoqutZNP3+3FNv655Vs6fkPvQLKVHtth28AAAAASUVORK5CYII=
IyEvdXNyL2Jpbi9wZXJsIC13CiMgLS0KIyBmYXEucGwgLSB0aGUgZ2xvYmFsIENHSSBoYW5kbGUgZmlsZSBmb3IgT1RSUwojIENvcHlyaWdodCAoQykgMjAwMS0yMDA2IE9UUlMgR21iSCwgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBmYXEucGwsdiAxLjEgMjAwNi8wOS8yOCAxNzoyODoxMCByayBFeHAgJAojIC0tCiMgVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU7IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkKIyBpdCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieQojIHRoZSBGcmVlIFNvZnR3YXJlIEZvdW5kYXRpb247IGVpdGhlciB2ZXJzaW9uIDIgb2YgdGhlIExpY2Vuc2UsIG9yCiMgKGF0IHlvdXIgb3B0aW9uKSBhbnkgbGF0ZXIgdmVyc2lvbi4KIwojIFRoaXMgcHJvZ3JhbSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLAojIGJ1dCBXSVRIT1VUIEFOWSBXQVJSQU5UWTsgd2l0aG91dCBldmVuIHRoZSBpbXBsaWVkIHdhcnJhbnR5IG9mCiMgTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiAgU2VlIHRoZQojIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGZvciBtb3JlIGRldGFpbHMuCiMKIyBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZQojIGFsb25nIHdpdGggdGhpcyBwcm9ncmFtOyBpZiBub3QsIHdyaXRlIHRvIHRoZSBGcmVlIFNvZnR3YXJlCiMgRm91bmRhdGlvbiwgSW5jLiwgNTkgVGVtcGxlIFBsYWNlLCBTdWl0ZSAzMzAsIEJvc3RvbiwgTUEgIDAyMTExLTEzMDcgIFVTQQojIC0tCgojIHVzZSAuLi8uLi8gYXMgbGliIGxvY2F0aW9uCnVzZSBGaW5kQmluIHF3KCRCaW4pOwp1c2UgbGliICIkQmluLy4uLy4uIjsKdXNlIGxpYiAiJEJpbi8uLi8uLi9LZXJuZWwvY3Bhbi1saWIiOwoKdXNlIHN0cmljdDsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OIEBJTkMpOwokVkVSU0lPTiA9ICckUmV2aXNpb246IDEuMSAkJzsKJFZFUlNJT04gPX4gcy9eXCQuKjpcVyguKilcVy4rPyQvJDEvOwoKIyAtLQojIDA9b2ZmOzE9b247CiMgLS0KbXkgJERlYnVnID0gMDsKCiMgLS0KIyBjaGVjayBASU5DIGZvciBtb2RfcGVybCAoYWRkIGxpYiBwYXRoIGZvciAicmVxdWlyZSBtb2R1bGUiISkKIyAtLQpwdXNoIChASU5DLCAiJEJpbi8uLi8uLiIsICIkQmluLy4uLy4uL0tlcm5lbC9jcGFuLWxpYiIpOwoKcHJpbnQgImxvY2F0aW9uOiBwdWJsaWMucGw/QWN0aW9uPVB1YmxpY0ZBUVxuIjsKcHJpbnQgIlxuIjsKcHJpbnQgIjxhIGhyZWY9J3B1YmxpYy5wbD9BY3Rpb249UHVibGljRkFRJz5tb3ZlZDwvYT5cbiI7Cgo=
IyEvdXNyL2Jpbi9wZXJsIC13CiMgLS0KIyBmYXEucGwgLSB0aGUgZ2xvYmFsIENHSSBoYW5kbGUgZmlsZSBmb3IgT1RSUwojIENvcHlyaWdodCAoQykgMjAwMS0yMDA2IE9UUlMgR21iSCwgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBmYXEuZnBsLHYgMS4xIDIwMDYvMDkvMjggMTc6Mjg6MTAgcmsgRXhwICQKIyAtLQojIFRoaXMgcHJvZ3JhbSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IgbW9kaWZ5CiMgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkKIyB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIgdmVyc2lvbiAyIG9mIHRoZSBMaWNlbnNlLCBvcgojIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiMKIyBUaGlzIHByb2dyYW0gaXMgZGlzdHJpYnV0ZWQgaW4gdGhlIGhvcGUgdGhhdCBpdCB3aWxsIGJlIHVzZWZ1bCwKIyBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgojIE1FUkNIQU5UQUJJTElUWSBvciBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRS4gIFNlZSB0aGUKIyBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmb3IgbW9yZSBkZXRhaWxzLgojCiMgWW91IHNob3VsZCBoYXZlIHJlY2VpdmVkIGEgY29weSBvZiB0aGUgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UKIyBhbG9uZyB3aXRoIHRoaXMgcHJvZ3JhbTsgaWYgbm90LCB3cml0ZSB0byB0aGUgRnJlZSBTb2Z0d2FyZQojIEZvdW5kYXRpb24sIEluYy4sIDU5IFRlbXBsZSBQbGFjZSwgU3VpdGUgMzMwLCBCb3N0b24sIE1BICAwMjExMS0xMzA3ICBVU0EKIyAtLQoKIyB1c2UgLi4vLi4vIGFzIGxpYiBsb2NhdGlvbgp1c2UgRmluZEJpbiBxdygkQmluKTsKdXNlIGxpYiAiJEJpbi8uLi8uLiI7CnVzZSBsaWIgIiRCaW4vLi4vLi4vS2VybmVsL2NwYW4tbGliIjsKCnVzZSBzdHJpY3Q7Cgp1c2UgdmFycyBxdygkVkVSU0lPTiBASU5DKTsKJFZFUlNJT04gPSAnJFJldmlzaW9uOiAxLjEgJCc7CiRWRVJTSU9OID1+IHMvXlwkLio6XFcoLiopXFcuKz8kLyQxLzsKCiMgLS0KIyAwPW9mZjsxPW9uOwojIC0tCm15ICREZWJ1ZyA9IDA7CgojIC0tCiMgY2hlY2sgQElOQyBmb3IgbW9kX3BlcmwgKGFkZCBsaWIgcGF0aCBmb3IgInJlcXVpcmUgbW9kdWxlIiEpCiMgLS0KcHVzaCAoQElOQywgIiRCaW4vLi4vLi4iLCAiJEJpbi8uLi8uLi9LZXJuZWwvY3Bhbi1saWIiKTsKCnByaW50ICJsb2NhdGlvbjogcHVibGljLmZwbD9BY3Rpb249UHVibGljRkFRXG4iOwpwcmludCAiXG4iOwpwcmludCAiPGEgaHJlZj0ncHVibGljLmZwbD9BY3Rpb249UHVibGljRkFRJz5tb3ZlZDwvYT5cbiI7Cgo=
en
de
internal
external
public
internal (agent)
1
external (customer)
2
public (all)
3
Misc
Misc Comment
1
current_timestamp
1
current_timestamp
1
0
Created
1
current_timestamp
1
current_timestamp
1