{"version":3,"file":"form-autocomplete.min.js","sources":["https:\/\/recursosorientacion.agenciaeducacion.gob.cl\/lib\/amd\/src\/form-autocomplete.js"],"sourcesContent":["\/\/ This file is part of Moodle - http:\/\/moodle.org\/\n\/\/\n\/\/ Moodle is free software: you can redistribute it and\/or modify\n\/\/ it under the terms of the GNU General Public License as published by\n\/\/ the Free Software Foundation, either version 3 of the License, or\n\/\/ (at your option) any later version.\n\/\/\n\/\/ Moodle is distributed in the hope that it will be useful,\n\/\/ but WITHOUT ANY WARRANTY; without even the implied warranty of\n\/\/ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n\/\/ GNU General Public License for more details.\n\/\/\n\/\/ You should have received a copy of the GNU General Public License\n\/\/ along with Moodle. If not, see .\n\n\/**\n * Autocomplete wrapper for select2 library.\n *\n * @module core\/form-autocomplete\n * @copyright 2015 Damyon Wiese \n * @license http:\/\/www.gnu.org\/copyleft\/gpl.html GNU GPL v3 or later\n * @since 3.0\n *\/\ndefine([\n 'jquery',\n 'core\/log',\n 'core\/str',\n 'core\/templates',\n 'core\/notification',\n 'core\/loadingicon',\n 'core\/aria',\n 'core_form\/changechecker',\n], function(\n $,\n log,\n str,\n templates,\n notification,\n LoadingIcon,\n Aria,\n FormChangeChecker\n) {\n \/\/ Private functions and variables.\n \/** @var {Object} KEYS - List of keycode constants. *\/\n var KEYS = {\n DOWN: 40,\n ENTER: 13,\n SPACE: 32,\n ESCAPE: 27,\n COMMA: 44,\n UP: 38,\n LEFT: 37,\n RIGHT: 39\n };\n\n var uniqueId = Date.now();\n\n \/**\n * Make an item in the selection list \"active\".\n *\n * @method activateSelection\n * @private\n * @param {Number} index The index in the current (visible) list of selection.\n * @param {Object} state State variables for this autocomplete element.\n * @return {Promise}\n *\/\n var activateSelection = function(index, state) {\n \/\/ Find the elements in the DOM.\n var selectionElement = $(document.getElementById(state.selectionId));\n\n \/\/ Count the visible items.\n var length = selectionElement.children('[aria-selected=true]').length;\n \/\/ Limit the index to the upper\/lower bounds of the list (wrap in both directions).\n index = index % length;\n while (index < 0) {\n index += length;\n }\n \/\/ Find the specified element.\n var element = $(selectionElement.children('[aria-selected=true]').get(index));\n \/\/ Create an id we can assign to this element.\n var itemId = state.selectionId + '-' + index;\n\n \/\/ Deselect all the selections.\n selectionElement.children().attr('data-active-selection', null).attr('id', '');\n\n \/\/ Select only this suggestion and assign it the id.\n element.attr('data-active-selection', true).attr('id', itemId);\n\n \/\/ Tell the input field it has a new active descendant so the item is announced.\n selectionElement.attr('aria-activedescendant', itemId);\n selectionElement.attr('data-active-value', element.attr('data-value'));\n\n return $.Deferred().resolve();\n };\n\n \/**\n * Get the actively selected element from the state object.\n *\n * @param {Object} state\n * @returns {jQuery}\n *\/\n var getActiveElementFromState = function(state) {\n var selectionRegion = $(document.getElementById(state.selectionId));\n var activeId = selectionRegion.attr('aria-activedescendant');\n\n if (activeId) {\n var activeElement = $(document.getElementById(activeId));\n if (activeElement.length) {\n \/\/ The active descendent still exists.\n return activeElement;\n }\n }\n\n \/\/ Ensure we are creating a properly formed selector based on the active value.\n var activeValue = selectionRegion.attr('data-active-value')?.replace(\/\"\/g, '\\\\\"');\n return selectionRegion.find('[data-value=\"' + activeValue + '\"]');\n };\n\n \/**\n * Update the active selection from the given state object.\n *\n * @param {Object} state\n *\/\n var updateActiveSelectionFromState = function(state) {\n var activeElement = getActiveElementFromState(state);\n var activeValue = activeElement.attr('data-value');\n\n var selectionRegion = $(document.getElementById(state.selectionId));\n if (activeValue) {\n \/\/ Find the index of the currently selected index.\n var activeIndex = selectionRegion.find('[aria-selected=true]').index(activeElement);\n\n if (activeIndex !== -1) {\n activateSelection(activeIndex, state);\n return;\n }\n }\n\n \/\/ Either the active index was not set, or it could not be found.\n \/\/ Select the first value instead.\n activateSelection(0, state);\n };\n\n \/**\n * Update the element that shows the currently selected items.\n *\n * @method updateSelectionList\n * @private\n * @param {Object} options Original options for this autocomplete element.\n * @param {Object} state State variables for this autocomplete element.\n * @param {JQuery} originalSelect The JQuery object matching the hidden select list.\n * @return {Promise}\n *\/\n var updateSelectionList = function(options, state, originalSelect) {\n var pendingKey = 'form-autocomplete-updateSelectionList-' + state.inputId;\n M.util.js_pending(pendingKey);\n\n \/\/ Build up a valid context to re-render the template.\n var items = [];\n var newSelection = $(document.getElementById(state.selectionId));\n originalSelect.children('option').each(function(index, ele) {\n if ($(ele).prop('selected')) {\n var label;\n if ($(ele).data('html')) {\n label = $(ele).data('html');\n } else {\n label = $(ele).html();\n }\n if (label !== '') {\n items.push({label: label, value: $(ele).attr('value')});\n }\n }\n });\n\n if (!hasItemListChanged(state, items)) {\n M.util.js_complete(pendingKey);\n return Promise.resolve();\n }\n\n state.items = items;\n\n var context = $.extend(options, state);\n \/\/ Render the template.\n return templates.render(options.templates.items, context)\n .then(function(html, js) {\n \/\/ Add it to the page.\n templates.replaceNodeContents(newSelection, html, js);\n\n updateActiveSelectionFromState(state);\n\n return;\n })\n .then(function() {\n return M.util.js_complete(pendingKey);\n })\n .catch(notification.exception);\n };\n\n \/**\n * Check whether the list of items stored in the state has changed.\n *\n * @param {Object} state\n * @param {Array} items\n * @returns {Boolean}\n *\/\n var hasItemListChanged = function(state, items) {\n if (state.items.length !== items.length) {\n return true;\n }\n\n \/\/ Check for any items in the state items which are not present in the new items list.\n return state.items.filter(item => items.indexOf(item) === -1).length > 0;\n };\n\n \/**\n * Notify of a change in the selection.\n *\n * @param {jQuery} originalSelect The jQuery object matching the hidden select list.\n *\/\n var notifyChange = function(originalSelect) {\n FormChangeChecker.markFormChangedFromNode(originalSelect[0]);\n\n \/\/ Note, jQuery .change() was not working here. Better to\n \/\/ use plain JavaScript anyway.\n originalSelect[0].dispatchEvent(new Event('change'));\n };\n\n \/**\n * Remove the given item from the list of selected things.\n *\n * @method deselectItem\n * @private\n * @param {Object} options Original options for this autocomplete element.\n * @param {Object} state State variables for this autocomplete element.\n * @param {Element} item The item to be deselected.\n * @param {Element} originalSelect The original select list.\n * @return {Promise}\n *\/\n var deselectItem = function(options, state, item, originalSelect) {\n var selectedItemValue = $(item).attr('data-value');\n\n \/\/ Look for a match, and toggle the selected property if there is a match.\n originalSelect.children('option').each(function(index, ele) {\n if ($(ele).attr('value') == selectedItemValue) {\n $(ele).prop('selected', false);\n \/\/ We remove newly created custom tags from the suggestions list when they are deselected.\n if ($(ele).attr('data-iscustom')) {\n $(ele).remove();\n }\n }\n });\n \/\/ Rerender the selection list.\n return updateSelectionList(options, state, originalSelect)\n .then(function() {\n \/\/ Notify that the selection changed.\n notifyChange(originalSelect);\n\n return;\n });\n };\n\n \/**\n * Make an item in the suggestions \"active\" (about to be selected).\n *\n * @method activateItem\n * @private\n * @param {Number} index The index in the current (visible) list of suggestions.\n * @param {Object} state State variables for this instance of autocomplete.\n * @return {Promise}\n *\/\n var activateItem = function(index, state) {\n \/\/ Find the elements in the DOM.\n var inputElement = $(document.getElementById(state.inputId));\n var suggestionsElement = $(document.getElementById(state.suggestionsId));\n\n \/\/ Count the visible items.\n var length = suggestionsElement.children(':not([aria-hidden])').length;\n \/\/ Limit the index to the upper\/lower bounds of the list (wrap in both directions).\n index = index % length;\n while (index < 0) {\n index += length;\n }\n \/\/ Find the specified element.\n var element = $(suggestionsElement.children(':not([aria-hidden])').get(index));\n \/\/ Find the index of this item in the full list of suggestions (including hidden).\n var globalIndex = $(suggestionsElement.children('[role=option]')).index(element);\n \/\/ Create an id we can assign to this element.\n var itemId = state.suggestionsId + '-' + globalIndex;\n\n \/\/ Deselect all the suggestions.\n suggestionsElement.children().attr('aria-selected', false).attr('id', '');\n \/\/ Select only this suggestion and assign it the id.\n element.attr('aria-selected', true).attr('id', itemId);\n \/\/ Tell the input field it has a new active descendant so the item is announced.\n inputElement.attr('aria-activedescendant', itemId);\n\n \/\/ Scroll it into view.\n var scrollPos = element.offset().top\n - suggestionsElement.offset().top\n + suggestionsElement.scrollTop()\n - (suggestionsElement.height() \/ 2);\n return suggestionsElement.animate({\n scrollTop: scrollPos\n }, 100).promise();\n };\n\n \/**\n * Find the index of the current active suggestion, and activate the next one.\n *\n * @method activateNextItem\n * @private\n * @param {Object} state State variable for this auto complete element.\n * @return {Promise}\n *\/\n var activateNextItem = function(state) {\n \/\/ Find the list of suggestions.\n var suggestionsElement = $(document.getElementById(state.suggestionsId));\n \/\/ Find the active one.\n var element = suggestionsElement.children('[aria-selected=true]');\n \/\/ Find it's index.\n var current = suggestionsElement.children(':not([aria-hidden])').index(element);\n \/\/ Activate the next one.\n return activateItem(current + 1, state);\n };\n\n \/**\n * Find the index of the current active selection, and activate the previous one.\n *\n * @method activatePreviousSelection\n * @private\n * @param {Object} state State variables for this instance of autocomplete.\n * @return {Promise}\n *\/\n var activatePreviousSelection = function(state) {\n \/\/ Find the list of selections.\n var selectionsElement = $(document.getElementById(state.selectionId));\n \/\/ Find the active one.\n var element = selectionsElement.children('[data-active-selection]');\n if (!element) {\n return activateSelection(0, state);\n }\n \/\/ Find it's index.\n var current = selectionsElement.children('[aria-selected=true]').index(element);\n \/\/ Activate the next one.\n return activateSelection(current - 1, state);\n };\n\n \/**\n * Find the index of the current active selection, and activate the next one.\n *\n * @method activateNextSelection\n * @private\n * @param {Object} state State variables for this instance of autocomplete.\n * @return {Promise}\n *\/\n var activateNextSelection = function(state) {\n \/\/ Find the list of selections.\n var selectionsElement = $(document.getElementById(state.selectionId));\n\n \/\/ Find the active one.\n var element = selectionsElement.children('[data-active-selection]');\n var current = 0;\n\n if (element) {\n \/\/ The element was found. Determine the index and move to the next one.\n current = selectionsElement.children('[aria-selected=true]').index(element);\n current = current + 1;\n } else {\n \/\/ No selected item found. Move to the first.\n current = 0;\n }\n\n return activateSelection(current, state);\n };\n\n \/**\n * Find the index of the current active suggestion, and activate the previous one.\n *\n * @method activatePreviousItem\n * @private\n * @param {Object} state State variables for this autocomplete element.\n * @return {Promise}\n *\/\n var activatePreviousItem = function(state) {\n \/\/ Find the list of suggestions.\n var suggestionsElement = $(document.getElementById(state.suggestionsId));\n\n \/\/ Find the active one.\n var element = suggestionsElement.children('[aria-selected=true]');\n\n \/\/ Find it's index.\n var current = suggestionsElement.children(':not([aria-hidden])').index(element);\n\n \/\/ Activate the previous one.\n return activateItem(current - 1, state);\n };\n\n \/**\n * Close the list of suggestions.\n *\n * @method closeSuggestions\n * @private\n * @param {Object} state State variables for this autocomplete element.\n * @return {Promise}\n *\/\n var closeSuggestions = function(state) {\n \/\/ Find the elements in the DOM.\n var inputElement = $(document.getElementById(state.inputId));\n var suggestionsElement = $(document.getElementById(state.suggestionsId));\n\n if (inputElement.attr('aria-expanded') === \"true\") {\n \/\/ Announce the list of suggestions was closed.\n inputElement.attr('aria-expanded', false);\n }\n \/\/ Read the current list of selections.\n inputElement.attr('aria-activedescendant', state.selectionId);\n\n \/\/ Hide the suggestions list (from screen readers too).\n Aria.hide(suggestionsElement.get());\n suggestionsElement.hide();\n\n return $.Deferred().resolve();\n };\n\n \/**\n * Rebuild the list of suggestions based on the current values in the select list, and the query.\n *\n * @method updateSuggestions\n * @private\n * @param {Object} options The original options for this autocomplete.\n * @param {Object} state The state variables for this autocomplete.\n * @param {String} query The current text for the search string.\n * @param {JQuery} originalSelect The JQuery object matching the hidden select list.\n * @return {Promise}\n *\/\n var updateSuggestions = function(options, state, query, originalSelect) {\n var pendingKey = 'form-autocomplete-updateSuggestions-' + state.inputId;\n M.util.js_pending(pendingKey);\n\n \/\/ Find the elements in the DOM.\n var inputElement = $(document.getElementById(state.inputId));\n var suggestionsElement = $(document.getElementById(state.suggestionsId));\n\n \/\/ Used to track if we found any visible suggestions.\n var matchingElements = false;\n \/\/ Options is used by the context when rendering the suggestions from a template.\n var suggestions = [];\n originalSelect.children('option').each(function(index, option) {\n if ($(option).prop('selected') !== true) {\n suggestions[suggestions.length] = {label: option.innerHTML, value: $(option).attr('value')};\n }\n });\n\n \/\/ Re-render the list of suggestions.\n var searchquery = state.caseSensitive ? query : query.toLocaleLowerCase();\n var context = $.extend({options: suggestions}, options, state);\n var returnVal = templates.render(\n 'core\/form_autocomplete_suggestions',\n context\n )\n .then(function(html, js) {\n \/\/ We have the new template, insert it in the page.\n templates.replaceNode(suggestionsElement, html, js);\n\n \/\/ Get the element again.\n suggestionsElement = $(document.getElementById(state.suggestionsId));\n\n \/\/ Show it if it is hidden.\n Aria.unhide(suggestionsElement.get());\n suggestionsElement.show();\n\n \/\/ For each option in the list, hide it if it doesn't match the query.\n suggestionsElement.children().each(function(index, node) {\n node = $(node);\n if ((options.caseSensitive && node.text().indexOf(searchquery) > -1) ||\n (!options.caseSensitive && node.text().toLocaleLowerCase().indexOf(searchquery) > -1)) {\n Aria.unhide(node.get());\n node.show();\n matchingElements = true;\n } else {\n node.hide();\n Aria.hide(node.get());\n }\n });\n \/\/ If we found any matches, show the list.\n inputElement.attr('aria-expanded', true);\n if (originalSelect.attr('data-notice')) {\n \/\/ Display a notice rather than actual suggestions.\n suggestionsElement.html(originalSelect.attr('data-notice'));\n } else if (matchingElements) {\n \/\/ We only activate the first item in the list if tags is false,\n \/\/ because otherwise \"Enter\" would select the first item, instead of\n \/\/ creating a new tag.\n if (!options.tags) {\n activateItem(0, state);\n }\n } else {\n \/\/ Nothing matches. Tell them that.\n str.get_string('nosuggestions', 'form').done(function(nosuggestionsstr) {\n suggestionsElement.html(nosuggestionsstr);\n });\n }\n\n return suggestionsElement;\n })\n .then(function() {\n return M.util.js_complete(pendingKey);\n })\n .catch(notification.exception);\n\n return returnVal;\n };\n\n \/**\n * Create a new item for the list (a tag).\n *\n * @method createItem\n * @private\n * @param {Object} options The original options for the autocomplete.\n * @param {Object} state State variables for the autocomplete.\n * @param {JQuery} originalSelect The JQuery object matching the hidden select list.\n * @return {Promise}\n *\/\n var createItem = function(options, state, originalSelect) {\n \/\/ Find the element in the DOM.\n var inputElement = $(document.getElementById(state.inputId));\n \/\/ Get the current text in the input field.\n var query = inputElement.val();\n var tags = query.split(',');\n var found = false;\n\n $.each(tags, function(tagindex, tag) {\n \/\/ If we can only select one at a time, deselect any current value.\n tag = tag.trim();\n if (tag !== '') {\n if (!options.multiple) {\n originalSelect.children('option').prop('selected', false);\n }\n \/\/ Look for an existing option in the select list that matches this new tag.\n originalSelect.children('option').each(function(index, ele) {\n if ($(ele).attr('value') == tag) {\n found = true;\n $(ele).prop('selected', true);\n }\n });\n \/\/ Only create the item if it's new.\n if (!found) {\n var option = $('