OTRS API Reference JavaScript

Source: Core.Agent.CustomerUserAddressBook.js

  1. // --
  2. // Copyright (C) 2001-2020 OTRS AG, https://otrs.com/
  3. // --
  4. // This software comes with ABSOLUTELY NO WARRANTY. For details, see
  5. // the enclosed file COPYING for license information (GPL). If you
  6. // did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
  7. // --
  8. "use strict";
  9. var Core = Core || {};
  10. Core.Agent = Core.Agent || {};
  11. /**
  12. * @namespace Core.Agent.CustomerUserAddressBook
  13. * @memberof Core.Agent
  14. * @author OTRS AG
  15. * @description
  16. * This namespace contains the special module functions for the search.
  17. */
  18. Core.Agent.CustomerUserAddressBook = (function (TargetNS) {
  19. /**
  20. * @private
  21. * @name RecipientUserLoginElementSelector
  22. * @memberof Core.Agent.CustomerUserAddressBook
  23. * @member {String}
  24. * @description
  25. * The recipient user login element selector for the checkbox.
  26. */
  27. var RecipientUserLoginElementSelector = 'div.Overview table td input[type="checkbox"][name=RecipientUserLogin]';
  28. /**
  29. * @private
  30. * @name CheckForSearchedValues
  31. * @memberof Core.Agent.CustomerUserAddressBook
  32. * @function
  33. * @returns {Boolean} False if no values were found, true if values where there.
  34. * @description
  35. * Checks if any values were entered in the search.
  36. * If nothing at all exists, it alerts with translated:
  37. * "Please enter at least one search value or * to find anything"
  38. */
  39. function CheckForSearchedValues() {
  40. var SearchValueFlag = false;
  41. $('#SearchForm label').each(function () {
  42. var ElementName,
  43. $Element;
  44. // Those with ID's are used for searching.
  45. if ($(this).attr('id')) {
  46. // substring "Label" (e.g. first five characters) from the
  47. // label id, use the remaining name as name string for accessing
  48. // the form input's value
  49. ElementName = $(this).attr('id').substring(5);
  50. $Element = $('#SearchForm input[name=' + ElementName + ']');
  51. // If there's no input element with the selected name
  52. // find the next "select" element and use that one for checking
  53. if (!$Element.length) {
  54. $Element = $(this).next().find('select');
  55. }
  56. if ($Element.length) {
  57. if ($Element.val() && $Element.val() !== '') {
  58. SearchValueFlag = true;
  59. }
  60. }
  61. }
  62. });
  63. if (!SearchValueFlag) {
  64. alert(Core.Language.Translate('Please enter at least one search value or * to find anything.'));
  65. }
  66. return SearchValueFlag;
  67. }
  68. /**
  69. * @private
  70. * @name CheckForSearchedValues
  71. * @memberof Core.Agent.CustomerUserAddressBook
  72. * @function
  73. * @description
  74. * Adds the given recipients to the ticket for the selected recipient field and close the dialog.
  75. */
  76. function AddRecipientsToTicket() {
  77. var RecipientField = $('#RecipientField').val(),
  78. Recipients = Core.JSON.Parse($('#RecipientSelectedJSON').val());
  79. $.each(Recipients, function(UserLogin, CustomerTicketText) {
  80. if (!CustomerTicketText) {
  81. return true;
  82. }
  83. parent.Core.App.Publish('Event.CustomerUserAddressBook.AddTicketCustomer.Callback.' + RecipientField, [UserLogin, CustomerTicketText]);
  84. });
  85. parent.Core.UI.Dialog.CloseDialog($('.Dialog', parent.document));
  86. }
  87. /**
  88. * @private
  89. * @name ShowWaitingLoader
  90. * @memberof Core.Agent.CustomerUserAddressBook
  91. * @function
  92. * @param {String} Selector - The selector to add the loader icon to the html.
  93. * @description
  94. * Shows waiting loader until search screen is ready.
  95. */
  96. function ShowWaitingLoader(Selector) {
  97. $('.Dialog .Footer', parent.document).html('');
  98. $(Selector).after('<div class="Spacing Center"><span class="AJAXLoader" title="' + Core.Config.Get('LoadingMsg') + '"></span></div>');
  99. $(Selector).remove();
  100. }
  101. /**
  102. * @name InitSearchDialog
  103. * @memberof Core.Agent.CustomerUserAddressBook
  104. * @function
  105. * @description
  106. * This function init the search dialog.
  107. */
  108. TargetNS.InitSearchDialog = function () {
  109. // move search button to dialog footer
  110. $('.Dialog .Footer', parent.document).html($('.SearchFormButton').html());
  111. $('.Dialog .Footer', parent.document).addClass('Center');
  112. $('.SearchFormButton').remove();
  113. // hide add template block
  114. $('#SearchProfileAddBlock').hide();
  115. // hide save changes in template block
  116. $('#SaveProfile').parent().hide().prev().hide().prev().hide();
  117. // search profile is selected
  118. if ($('#SearchProfile').val() && $('#SearchProfile').val() !== 'last-search') {
  119. // show delete button
  120. $('#SearchProfileDelete').show();
  121. // show save changes in template block
  122. $('#SaveProfile').parent().show().prev().show().prev().show();
  123. // set SaveProfile to 0
  124. $('#SaveProfile').prop('checked', false);
  125. }
  126. Core.Agent.Search.AddSearchAttributes();
  127. Core.Agent.Search.AdditionalAttributeSelectionRebuild();
  128. Core.UI.InputFields.Activate($('.Dialog:visible'));
  129. // register add of attribute
  130. $('#Attribute').on('change', function () {
  131. var Attribute = $('#Attribute').val();
  132. Core.Agent.Search.SearchAttributeAdd(Attribute);
  133. Core.Agent.Search.AdditionalAttributeSelectionRebuild();
  134. return false;
  135. });
  136. // register return key
  137. $('#SearchForm').off('keypress.FilterInput').on('keypress.FilterInput', function (Event) {
  138. if ((Event.charCode || Event.keyCode) === 13) {
  139. if (!CheckForSearchedValues()) {
  140. return false;
  141. }
  142. else {
  143. $('#SearchFormSubmit', parent.document).trigger('click');
  144. }
  145. return false;
  146. }
  147. });
  148. // add new profile
  149. $('#SearchProfileAddAction').on('click', function () {
  150. var ProfileName, $Element1;
  151. // get name
  152. ProfileName = $('#SearchProfileAddName').val();
  153. // check name
  154. if (!ProfileName.length || ProfileName.length < 2) {
  155. return;
  156. }
  157. // add name to profile selection
  158. $Element1 = $('#SearchProfile').children().first().clone();
  159. $Element1.text(ProfileName);
  160. $Element1.attr('value', ProfileName);
  161. $Element1.prop('selected', true);
  162. $('#SearchProfile').append($Element1).trigger('redraw.InputField');
  163. // set input box to empty
  164. $('#SearchProfileAddName').val('');
  165. // hide add template block
  166. $('#SearchProfileAddBlock').hide();
  167. // hide save changes in template block
  168. $('#SaveProfile').parent().hide().prev().hide().prev().hide();
  169. // set SaveProfile to 1
  170. $('#SaveProfile').prop('checked', true);
  171. // show delete button
  172. $('#SearchProfileDelete').show();
  173. });
  174. // Close the dialog.
  175. $('#FormCancel', parent.document).on('click', function() {
  176. parent.Core.UI.Dialog.CloseDialog($('.Dialog', parent.document));
  177. });
  178. // Register submit for search.
  179. $('#SearchFormSubmit', parent.document).on('click', function () {
  180. var ShownAttributes = '',
  181. ExcludeUserLogins = [];
  182. // Remember shown attributes.
  183. $('#SearchInsert label').each(function () {
  184. if ($(this).attr('id')) {
  185. ShownAttributes = ShownAttributes + ';' + $(this).attr('id');
  186. }
  187. });
  188. $('#SearchForm #ShownAttributes').val(ShownAttributes);
  189. // Check for already added customer users entries.
  190. $('.CustomerKey', parent.document).each(function() {
  191. if($(this).val()) {
  192. ExcludeUserLogins.push($(this).val());
  193. }
  194. });
  195. if (ExcludeUserLogins) {
  196. $('#SearchForm #ExcludeUserLogins').val(Core.JSON.Stringify(ExcludeUserLogins));
  197. }
  198. if (!CheckForSearchedValues()) {
  199. return false;
  200. }
  201. else {
  202. $('#SearchForm').submit();
  203. ShowWaitingLoader('.ContentColumn');
  204. }
  205. });
  206. // load profile
  207. $('#SearchProfile').on('change', function () {
  208. var SearchProfile = $('#SearchProfile').val(),
  209. SearchProfileEmptySearch = $('#EmptySearch').val(),
  210. SearchProfileAction = $('#SearchAction').val(),
  211. RecipientType = $('#RecipientType').val(),
  212. RecipientField = $('#RecipientField').val(),
  213. RecipientFieldLabel = $('#RecipientFieldLabel').val();
  214. window.location.href = Core.Config.Get('Baselink') + 'Action=' + SearchProfileAction + ';RecipientType=' + RecipientType + ';RecipientField=' + RecipientField + ';RecipientFieldLabel=' + RecipientFieldLabel + ';Subaction=LoadProfile;EmptySearch=' +
  215. encodeURIComponent(SearchProfileEmptySearch) + ';Profile=' + encodeURIComponent(SearchProfile);
  216. return false;
  217. });
  218. // show add profile block or not
  219. $('#SearchProfileNew').on('click', function (Event) {
  220. $('#SearchProfileAddBlock').toggle();
  221. $('#SearchProfileAddName').focus();
  222. Event.preventDefault();
  223. return false;
  224. });
  225. // delete profile
  226. $('#SearchProfileDelete').on('click', function (Event) {
  227. var SearchProfileAction = $('#SearchAction').val(),
  228. RecipientType = $('#RecipientType').val(),
  229. RecipientField = $('#RecipientField').val(),
  230. RecipientFieldLabel = $('#RecipientFieldLabel').val();
  231. Event.preventDefault();
  232. // strip all already used attributes
  233. $('#SearchProfile').find('option:selected').each(function () {
  234. if ($(this).attr('value') !== 'last-search') {
  235. window.location.href = Core.Config.Get('Baselink') + 'Action=' + SearchProfileAction + ';RecipientType=' + RecipientType + ';RecipientField=' + RecipientField + ';RecipientFieldLabel=' + RecipientFieldLabel + ';Subaction=DeleteProfile;Profile=' +
  236. encodeURIComponent($(this).val());
  237. return false;
  238. }
  239. });
  240. });
  241. };
  242. /**
  243. * @name InitRecipientSelection
  244. * @memberof Core.Agent.CustomerUserAddressBook
  245. * @function
  246. * @description
  247. * This function init the recipient selection.
  248. */
  249. TargetNS.InitRecipientSelection = function () {
  250. var InitialRecipients = Core.JSON.Parse($('#RecipientSelectedJSON').val());
  251. // Add the current field label and move recipient selection button in the dialog footer
  252. // for the result view, to have the same view in all dialog widgets.
  253. $('#RecipientSelect span').append(' ' + $('#RecipientFieldLabel').val() + ' (' + Object.keys(InitialRecipients).length + ')');
  254. $('.Dialog .Footer', parent.document).html($('.RecipientButton').html());
  255. $('.RecipientButton').remove();
  256. if (!Object.keys(InitialRecipients).length) {
  257. $('#RecipientSelect', parent.document).prop('disabled', true);
  258. $('#RecipientSelect', parent.document).addClass('Disabled');
  259. }
  260. $('#SelectAllCustomerUser').on('click', function () {
  261. var Status = $(this).prop('checked'),
  262. Recipients = Core.JSON.Parse($('#RecipientSelectedJSON').val()),
  263. AddRecipientsButtonHTML = $('#RecipientSelect span', parent.document).html();
  264. $(RecipientUserLoginElementSelector).prop('checked', Status).triggerHandler('click');
  265. $(RecipientUserLoginElementSelector).each(function() {
  266. var UserLogin = $(this).val();
  267. if (Status) {
  268. Recipients[UserLogin] = $(this).data('customer-ticket-text');
  269. }
  270. else {
  271. delete Recipients[UserLogin];
  272. }
  273. });
  274. // Disable the recipient select button, if no recipient is selected.
  275. if (!Object.keys(Recipients).length) {
  276. $('#RecipientSelect', parent.document).prop('disabled', true);
  277. $('#RecipientSelect', parent.document).addClass('Disabled');
  278. }
  279. else {
  280. $('#RecipientSelect', parent.document).prop('disabled', false);
  281. $('#RecipientSelect', parent.document).removeClass('Disabled');
  282. }
  283. $('#RecipientSelect span', parent.document).html(AddRecipientsButtonHTML.replace(/[ ]?\(?\d*\)?$/, ' (' + Object.keys(Recipients).length + ')'));
  284. $('#RecipientSelectedJSON').val(Core.JSON.Stringify(Recipients));
  285. });
  286. $(RecipientUserLoginElementSelector).on('click', function (Event) {
  287. var Status = $(this).prop('checked'),
  288. UserLogin = $(this).val(),
  289. Recipients = Core.JSON.Parse($('#RecipientSelectedJSON').val()),
  290. AddRecipientsButtonHTML = $('#RecipientSelect span', parent.document).html();
  291. Event.stopPropagation();
  292. Core.Form.SelectAllCheckboxes($(this), $('#SelectAllCustomerUser'));
  293. if (Status) {
  294. Recipients[UserLogin] = $(this).data('customer-ticket-text');
  295. }
  296. else {
  297. delete Recipients[UserLogin];
  298. }
  299. $('#RecipientSelect span', parent.document).html(AddRecipientsButtonHTML.replace(/[ ]?\(?\d*\)?$/, ' (' + Object.keys(Recipients).length + ')'));
  300. // Disable the recipient select button, if no recipient is selected.
  301. if (!Object.keys(Recipients).length) {
  302. $('#RecipientSelect', parent.document).prop('disabled', true);
  303. $('#RecipientSelect', parent.document).addClass('Disabled');
  304. }
  305. else {
  306. $('#RecipientSelect', parent.document).prop('disabled', false);
  307. $('#RecipientSelect', parent.document).removeClass('Disabled');
  308. }
  309. $('#RecipientSelectedJSON').val(Core.JSON.Stringify(Recipients));
  310. });
  311. $('.MasterAction').off('click').on('click', function () {
  312. $(this).find('td input[type="checkbox"][name=RecipientUserLogin]').trigger('click');
  313. });
  314. // Execute the pagination as a post action instead of get.
  315. $('span.Pagination a').off('click').on('click', function (Event) {
  316. var Href = $(this).attr('href'),
  317. Parts = Href.split('?'),
  318. URL = Parts[0],
  319. Params = Parts[1].split(';'),
  320. Key,
  321. InputParams,
  322. InputFields = '';
  323. Event.stopPropagation();
  324. Event.preventDefault();
  325. for (Key in Params) {
  326. InputParams = Params[Key].split('=');
  327. InputFields += "<input type='hidden' name='" + InputParams[0] + "' value='" + InputParams[1] + "' />";
  328. }
  329. // Get the selected recipients json string from the current page to not lose the selections.
  330. InputFields += "<input type='hidden' name='" + $('#RecipientSelectedJSON').attr('name') + "' value='" + $('#RecipientSelectedJSON').val() + "' />";
  331. $("body").append('<form action="' + URL + '" method="post" id="SearchResultPagination">' + InputFields + '</form>');
  332. $("#SearchResultPagination").submit();
  333. });
  334. // Close the dialog.
  335. $('#FormCancel', parent.document).on('click', function() {
  336. parent.Core.UI.Dialog.CloseDialog($('.Dialog', parent.document));
  337. });
  338. // Register Apply button event.
  339. $('#RecipientSelect', parent.document).on('click', function () {
  340. AddRecipientsToTicket();
  341. });
  342. };
  343. /**
  344. * @name Init
  345. * @memberof Core.Agent.CustomerUserAddressBook
  346. * @function
  347. * @description
  348. * This function init the customer user adress book search dialog or the result screen.
  349. */
  350. TargetNS.Init = function () {
  351. var ShowSearchDialog = Core.Config.Get('ShowSearchDialog');
  352. if (typeof ShowSearchDialog !== 'undefined' && parseInt(ShowSearchDialog, 10) === 1) {
  353. Core.Agent.CustomerUserAddressBook.InitSearchDialog();
  354. }
  355. else {
  356. Core.Agent.CustomerUserAddressBook.InitRecipientSelection();
  357. }
  358. };
  359. Core.Init.RegisterNamespace(TargetNS, 'APP_MODULE');
  360. return TargetNS;
  361. }(Core.Agent.CustomerUserAddressBook || {}));

^ Use Elevator