import * as tslib_1 from "tslib";
import { searchAndInitialize, preventDefault, find, remove, internetExplorerOrEdgeVersion, scrollIntoView } from "../Utils";
import DomElement from "../DomElement";
import * as Inputs from "../Inputs";
import * as Dom from "../DomFunctions";
var CLASS_PLACEHOLDER = "select__placeholder";
var CLASS_THUMB = "select__thumb";
var CLASS_BUTTON = "select__button";
var CLASS_DROPDOWN = "select__dropdown";
var CLASS_OPEN = "select--open";
var CLASS_CLOSED = "select--closed";
var CLASS_DISABLED = "select--disabled";
var CLASS_FILTERABLE = "select--filterable";
var CLASS_ITEM = "dropdown-item";
var CLASS_ITEM_SELECTED = "dropdown-item--selected";
var CLASS_ITEM_FOCUSED = "dropdown-item--focused";
var CLASS_ITEM_DISABLED = "dropdown-item--disabled";
var CLASS_GROUP_ITEM = "dropdown-group";
var CLASS_GROUP_HEADER = "dropdown-group__item";
var QUERY_MESSAGE = ".message";
var TIMEOUT_CLOSE = 150;
var TIMEOUT_BLUR = 400;
/**
* The select component API.
*/
var Select = /** @class */ (function (_super) {
tslib_1.__extends(Select, _super);
function Select(element) {
var _this = _super.call(this, element) || this;
// Minimum filter length
_this._minFilterLength = 2;
// The options the Select was initially created upon
// These will be used as a basis for filtering
_this._initialOptions = Array.prototype.slice.call(_this.element.children);
_this._openByFocus = false;
// Check for multi-selection
_this._multiselection = _this.element.hasAttribute("multiple") === true;
// Setup event context
_this._clickHandler = _this._handleClick.bind(_this);
_this._handleDropdownClick = _this._handleClick.bind(_this);
_this._keydownHandler = _this._handleKeydown.bind(_this);
_this._focusHandler = _this._handleFocus.bind(_this);
_this._blurHandler = _this._handleBlur.bind(_this);
_this._windowClickHandler = _this._handleWindowClick.bind(_this);
_this._filterKeydownHandler = _this._handleFilterKeydown.bind(_this);
_this._filterKeyupHandler = _this._handleFilterKeyup.bind(_this);
_this._filterFocusHandler = _this._handleFilterFocus.bind(_this);
_this._initialize();
return _this;
}
/**
* Initializes the select component.
*
* This method inspects the select definition and its options and
* generates new stylable DOM elements around the original select-element
* definitions.
* @private
*/
Select.prototype._initialize = function () {
var e_1, _a;
var selectedOption = this.element.querySelector("option[selected]");
var firstOption = this.element.querySelector("option");
// Per default, set the last selected option to either the option with a "selected" attribute,
// or, if not found, to the first available option
this._lastSelectedOption = selectedOption || firstOption;
this._wrapperElement = new DomElement(this.element.parentElement)
.addClass(CLASS_CLOSED);
try {
for (var _b = tslib_1.__values(this.classes), _c = _b.next(); !_c.done; _c = _b.next()) {
var cls = _c.value;
this._wrapperElement.addClass(cls);
}
}
catch (e_1_1) { e_1 = { error: e_1_1 }; }
finally {
try {
if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
}
finally { if (e_1) throw e_1.error; }
}
this._dropdownElement = new DomElement("div")
.addClass(CLASS_DROPDOWN);
if (internetExplorerOrEdgeVersion() > 0 && internetExplorerOrEdgeVersion() < 12) {
// This is a workaround for IE browsers 11 and earlier where focusing
// a scrollable dropdown list will close the dropdown prematurely.
this._dropdownElement.element.addEventListener("mousedown", function (event) { return event.preventDefault(); });
}
this._setupTarget();
this._setupPlaceholder();
this._wrapperElement.appendChild(this._dropdownElement);
this._createOptions(this.element);
this._updateSize();
this._updateMessage();
if (this.element.disabled) {
this.disable();
}
else {
this.enable();
}
};
Select.prototype._setupTarget = function () {
// move the id from the select element to the wrapper
var id = this.element.getAttribute("id");
if (id) {
this.element.removeAttribute("id");
this._wrapperElement.setAttribute("id", id);
}
// Apply the tab index
var tabIndex = this.element.getAttribute("tabindex");
if (tabIndex) {
this._wrapperElement.setAttribute("tabIndex", tabIndex);
}
};
Select.prototype._setupPlaceholder = function () {
var _this = this;
if (!this._selectButtonElement) {
this._selectButtonElement = new DomElement("div")
.addClass(CLASS_BUTTON);
this._wrapperElement.appendChild(this._selectButtonElement);
}
if (!this._thumbElement) {
this._thumbElement = new DomElement("div")
.addClass(CLASS_THUMB);
var thumbIcon = new DomElement("div")
.addClass("thumb-icon");
var loader = new DomElement("div")
.addClass("loader-spinner")
.addClass("loader-spinner--small");
this._thumbElement.appendChild(loader);
this._thumbElement.appendChild(thumbIcon);
this._selectButtonElement.appendChild(this._thumbElement);
}
var placeholderText = "";
this._placeholderOption = this.element.querySelector("option[selected][disabled]") || undefined;
if (this._placeholderOption) {
placeholderText = Dom.text(this._placeholderOption);
if (this._multiselection === true) {
this._placeholderOption.selected = false;
}
}
var selectedOption = this.element.querySelector("option[selected]:not([disabled])");
if (selectedOption) {
placeholderText = Dom.text(selectedOption);
}
if (!this._placeholderElement) {
// When the Select is filterable, create an "input" as the placeholder element, otherwise a "span"
if (this._isFilterable()) {
this._placeholderElement = new DomElement("input");
this._placeholderElement.addEventListener("keyup", function (e) { return _this._handleFilterKeyup(e); });
this._placeholderElement.addEventListener("keydown", function (e) { return _this._handleFilterKeydown(e); });
this._placeholderElement.addEventListener("focus", function (e) { return _this._handleFilterFocus(e); });
}
else {
this._placeholderElement = new DomElement("span");
}
this._placeholderElement.addClass(CLASS_PLACEHOLDER);
this._selectButtonElement.appendChild(this._placeholderElement);
}
this._setPlaceholder(placeholderText);
this._placeholderText = placeholderText;
if (selectedOption && selectedOption !== this._placeholderOption) {
this._updatePlaceholder(true);
}
};
Select.prototype._updateMessage = function () {
var messageNode = this._wrapperElement.element.querySelector(QUERY_MESSAGE);
if (messageNode !== null) {
this._wrapperElement.appendChild(new DomElement(messageNode));
}
};
Select.prototype._isOptGroup = function (element) {
return element.tagName.toUpperCase() === "OPTGROUP";
};
Select.prototype._isOption = function (element) {
return element.tagName.toUpperCase() === "OPTION";
};
Select.prototype._createOptions = function (element) {
for (var i = 0; i < element.children.length; i++) {
var child = element.children[i];
if (this._isOptGroup(child)) {
this._appendGroup(child);
}
if (this._isOption(child)) {
var option = this._createOption(child);
if (option) {
this._dropdownElement.appendChild(option);
}
}
}
};
Select.prototype._createOption = function (option) {
var html = option.innerHTML;
if (this._activeFilter) {
var sanitizedActiveFilter = this._activeFilter.replace(/[-\\^$*+?.()|[\]{}]/g, "\\$&");
html = html.replace(new RegExp("(" + sanitizedActiveFilter + ")", "gi"), "$1 ");
}
var opt = new DomElement("div")
.addClass(CLASS_ITEM)
.setHtml(html);
if (option.selected) {
opt.addClass(CLASS_ITEM_SELECTED);
}
if (option.disabled) {
opt.addClass(CLASS_ITEM_DISABLED);
}
if (!this._isPlaceholder(option)) {
opt.setAttribute("data-value", option.value);
return opt;
}
return undefined;
};
Select.prototype._appendGroup = function (optgroup) {
var e_2, _a;
var label = optgroup.getAttribute("label");
var group = new DomElement("div")
.addClass(CLASS_GROUP_ITEM);
var groupHeader = new DomElement("div")
.addClass(CLASS_GROUP_HEADER)
.setHtml(label);
group.appendChild(groupHeader);
var options = optgroup.querySelectorAll("option");
try {
for (var options_1 = tslib_1.__values(options), options_1_1 = options_1.next(); !options_1_1.done; options_1_1 = options_1.next()) {
var entry = options_1_1.value;
var option = this._createOption(entry);
if (option) {
group.appendChild(option);
}
}
}
catch (e_2_1) { e_2 = { error: e_2_1 }; }
finally {
try {
if (options_1_1 && !options_1_1.done && (_a = options_1.return)) _a.call(options_1);
}
finally { if (e_2) throw e_2.error; }
}
this._dropdownElement.appendChild(group);
return group;
};
Select.prototype._updateSize = function () {
var e_3, _a;
// Note: Mirroring the DOM and measuring the items using their clientWidth was very
// unreliable, therefore measuring was switched to the new HTML5 measureText method
// margins and paddings arround the text are copied from the original placeholder items
// dimension
var placeholderStyle = window.getComputedStyle(this._placeholderElement.element);
var paddingRight = parseFloat(placeholderStyle.paddingRight);
var paddingLeft = parseFloat(placeholderStyle.paddingLeft);
var font = this._placeholderElement.css("font");
var textWidth = Dom.textWidth(this._placeholderText, font);
var maxWidth = paddingLeft + paddingRight + textWidth;
var options = this._wrapperElement.element.querySelectorAll("." + CLASS_ITEM);
try {
for (var options_2 = tslib_1.__values(options), options_2_1 = options_2.next(); !options_2_1.done; options_2_1 = options_2.next()) {
var entry = options_2_1.value;
var width = Dom.textWidth(Dom.text(entry), font) + paddingLeft + paddingRight;
if (width > maxWidth) {
maxWidth = width;
}
}
}
catch (e_3_1) { e_3 = { error: e_3_1 }; }
finally {
try {
if (options_2_1 && !options_2_1.done && (_a = options_2.return)) _a.call(options_2);
}
finally { if (e_3) throw e_3.error; }
}
};
Select.prototype._isButtonTarget = function (target) {
return (target === this._wrapperElement.element ||
target === this._placeholderElement.element ||
target === this._selectButtonElement.element ||
target === this._thumbElement.element);
};
Select.prototype._isDropdownTarget = function (target) {
var current = target;
while (current !== this._dropdownElement.element && current.parentElement) {
current = current.parentElement;
}
return current === this._dropdownElement.element;
};
/**
* Updates the UI if the selection has changed and makes sure the
* select control and the generated markup are synchronized.
* @private
*/
Select.prototype._selectedItemChanged = function (newItem, autoClose, multiselect) {
var _this = this;
if (autoClose === void 0) { autoClose = true; }
if (multiselect === void 0) { multiselect = false; }
var oldItems = this._dropdownElement.element.querySelectorAll("." + CLASS_ITEM_SELECTED);
if (!newItem) {
setTimeout(function () { return _this.close(); }, TIMEOUT_CLOSE);
return;
}
if (Dom.hasClass(newItem, CLASS_ITEM_DISABLED)) {
return;
}
if ((oldItems.length === 0) && !newItem) {
throw new Error("Can not select undefined elements");
}
var oldItem = oldItems[0];
if (multiselect === true) {
oldItem = find(oldItems, function (x) { return x.getAttribute("data-value") === newItem.getAttribute("data-value"); });
}
var isDeselect = false;
if (newItem && oldItem && oldItem === newItem) {
// Click on a previously selected element -> deselect
isDeselect = true;
if (!this._placeholderOption && !multiselect) {
// If there is no placeholder option, non multiselect options cannot be deselected
return;
}
delete this._lastSelectedOption;
}
if (oldItem) {
// Remove selection on the element
var oldValue_1 = oldItem.getAttribute("data-value");
var optElement = find(this.element.options, function (x) { return !x.disabled && x.value === oldValue_1; });
if (!optElement) {
throw new Error("The option with value " + oldValue_1 + " does not exist");
}
// Unset Select value
optElement.selected = false;
Dom.removeClass(oldItem, CLASS_ITEM_SELECTED);
}
if (!isDeselect) { // Select an option
// Select a new item
var newValue_1 = newItem.getAttribute("data-value");
var optElement = find(this.element.options, function (x) { return !x.disabled && x.value === newValue_1; });
if (!optElement) {
throw new Error("The option with value " + newValue_1 + " does not exist");
}
// Set Select value
optElement.selected = true;
Dom.addClass(newItem, CLASS_ITEM_SELECTED);
// Preserve selection
this._lastSelectedOption = optElement;
}
else { // Deselect an option
// Keep track of falling back to the placeholder (if any)
if (this._placeholderOption) {
this._lastSelectedOption = this._placeholderOption;
}
}
var hasSelectedItems = true;
if (this._multiselection === false && isDeselect) {
// Handle no selection for non-multiselect states
this._placeholderOption.selected = true;
hasSelectedItems = false;
}
if (this._multiselection === true && this._getSelectedOptions().length === 0) {
hasSelectedItems = false;
}
// Reset the filter if filterable
if (this._activeFilter) {
this._clearFilter();
}
this._updatePlaceholder(hasSelectedItems);
// Dispatch the changed event
this.dispatchEvent("change");
if (autoClose && !multiselect) {
setTimeout(function () {
_this.close();
}, TIMEOUT_CLOSE);
}
};
Select.prototype._updatePlaceholder = function (hasSelectedItems) {
var e_4, _a;
var text = this._placeholderOption ? Dom.text(this._placeholderOption) : " ";
if (hasSelectedItems === true) {
var selectedItems = this._getSelectedOptions();
if (selectedItems.length > 0) {
text = "";
try {
for (var selectedItems_1 = tslib_1.__values(selectedItems), selectedItems_1_1 = selectedItems_1.next(); !selectedItems_1_1.done; selectedItems_1_1 = selectedItems_1.next()) {
var item = selectedItems_1_1.value;
text += Dom.text(item) + ", ";
}
}
catch (e_4_1) { e_4 = { error: e_4_1 }; }
finally {
try {
if (selectedItems_1_1 && !selectedItems_1_1.done && (_a = selectedItems_1.return)) _a.call(selectedItems_1);
}
finally { if (e_4) throw e_4.error; }
}
text = text.substring(0, text.length - 2);
}
}
this._setPlaceholder(text);
};
Select.prototype._getSelectedOptions = function () {
var selectedOptions = [];
if (this.element.options) {
[].forEach.call(this.element.options, (function (option) {
if (option.selected && !option.disabled) {
selectedOptions.push(option);
}
}));
}
return selectedOptions;
};
/**
* Clone all of the initially set options (and optgroups) and returns them in a new array.
* This serves as the basis for filtering. If a filter is present, it will be respected.
*/
Select.prototype.getInitialOptions = function () {
var filter = this._activeFilter || "";
var filtered = [];
var initialOptions = this._initialOptions;
for (var i = 0; i < initialOptions.length; i++) {
var child = initialOptions[i];
if (this._isOptGroup(child)) { // handle
var optGroupClone = child.cloneNode(false);
var found = false;
for (var j = 0; j < child.children.length; j++) {
var optionClone = child.children[j].cloneNode(true);
// Append on match
if (this._containsWord(optionClone.innerHTML, filter)) {
optGroupClone.appendChild(optionClone);
found = true;
}
}
// Push if any matches found
if (found) {
filtered.push(optGroupClone);
}
}
else if (this._isOption(child)) { // handle
var optionClone = child.cloneNode(true);
// Push on match
if (this._containsWord(optionClone.innerHTML, filter)) {
filtered.push(optionClone);
}
}
}
return filtered;
};
/**
* Returns true if a text contains a given keyword, e.g. in "ca" in "Car"
*/
Select.prototype._containsWord = function (text, keyword) {
return text.toLowerCase().indexOf(keyword.toLowerCase()) > -1;
};
Select.prototype._handleFocus = function () {
var _this = this;
this.open();
this._openByFocus = true;
setTimeout(function () {
_this._openByFocus = false;
}, TIMEOUT_BLUR);
};
Select.prototype._handleBlur = function () {
this.close();
};
Select.prototype._handleClick = function (event) {
var handled = false;
if (this._lastHandledEvent === event) {
this._lastHandledEvent = undefined;
return;
}
if (this._isButtonTarget(event.target) && this._openByFocus === false) {
// handle header item clicks and toggle dropdown
this.toggle();
handled = true;
}
var newItem = event.target;
if (!handled && Dom.hasClass(newItem, CLASS_ITEM)) {
// handle clicks on dropdown items
this._selectedItemChanged(newItem, true, this._multiselection);
handled = true;
}
if (handled) {
this._lastHandledEvent = event;
preventDefault(event);
}
};
Select.prototype._handleWindowClick = function (event) {
if (this._isDropdownTarget(event.target) || this._isButtonTarget(event.target)) {
return;
}
this.close();
};
Select.prototype._focusOptionStartingWith = function (keycode, startIndex, options) {
for (var index = startIndex; index < options.length; index++) {
var item = new DomElement(options[index]);
var value = item.innerText.toLowerCase();
if (index > options.length) {
index = 0;
}
if (value.startsWith(Inputs.getKeyValue(keycode))) {
var newOption = new DomElement(options[index]);
if (!newOption.hasClass(CLASS_ITEM_DISABLED)) {
scrollIntoView(options[index]);
newOption.addClass(CLASS_ITEM_FOCUSED);
return newOption;
}
}
}
return undefined;
};
Select.prototype._handleKeydown = function (event) {
var evt = event || window.event;
var keycode = event.which || event.keyCode;
if (keycode === Inputs.KEY_ESCAPE) {
// handle Escape key (ESC)
if (this.isOpen()) {
this.close();
}
evt.preventDefault();
return;
}
if (keycode === Inputs.KEY_ARROW_UP || keycode === Inputs.KEY_ARROW_DOWN) {
// Up and down arrows
var options = this._wrapperElement.element.querySelectorAll("." + CLASS_ITEM);
if (options.length > 0) {
var newIndex = 0;
var oldOption = void 0;
var focusedElement = this._wrapperElement.find("." + CLASS_ITEM_FOCUSED);
var searchFor = focusedElement ? CLASS_ITEM_FOCUSED : CLASS_ITEM_SELECTED;
var newElement = void 0;
for (var index = 0; index < options.length; index++) {
var direction = keycode === Inputs.KEY_ARROW_DOWN ? 1 : -1;
var item = new DomElement(options[index]);
// search for selected or focusedElement elements
if (item.hasClass(searchFor)) {
oldOption = item;
newIndex = index;
// get the next not disabled element in the appropriate direction
for (var count = 0; count < options.length; count++) {
newIndex += direction;
newIndex %= options.length;
if (newIndex < 0) {
newIndex = options.length - 1;
}
newElement = new DomElement(options[newIndex]);
if (!newElement.hasClass(CLASS_ITEM_DISABLED)) {
break;
}
}
}
}
// set the new element focused
scrollIntoView(options[newIndex]);
var newOption = new DomElement(options[newIndex]);
newOption.addClass(CLASS_ITEM_FOCUSED);
if (oldOption) {
oldOption.removeClass(CLASS_ITEM_FOCUSED);
}
}
evt.preventDefault();
return;
}
if (Inputs.getKeyValue(keycode) && !this._isFilterable()) {
// Keyboard keys
var options = this._wrapperElement.element.querySelectorAll("." + CLASS_ITEM);
if (options.length > 0) {
var oldFocusIndex = 0;
var hasFocusedOption = false;
for (var index = 0; index < options.length; index++) {
var item = new DomElement(options[index]);
if (item.hasClass(CLASS_ITEM_FOCUSED)) {
item.removeClass(CLASS_ITEM_FOCUSED);
var value = item.innerText.toLowerCase();
if (value.startsWith(Inputs.getKeyValue(keycode))) {
hasFocusedOption = true;
oldFocusIndex = index;
}
}
}
var newOption = this._focusOptionStartingWith(keycode, hasFocusedOption ? oldFocusIndex + 1 : 0, options);
if (newOption === undefined) {
this._focusOptionStartingWith(keycode, 0, options);
}
}
evt.preventDefault();
return;
}
if (keycode === Inputs.KEY_ENTER || keycode === Inputs.KEY_TAB) {
// Handle enter and tab key by selecting the currently focused element
var newItem = this._dropdownElement.element.querySelector("." + CLASS_ITEM_FOCUSED);
this._selectedItemChanged(newItem, true, this._multiselection);
}
};
/**
* Fired when the user presses a key in the filter field
*/
Select.prototype._handleFilterKeydown = function (e) {
var keycode = e.which || e.keyCode;
// If the user hits the enter key while filtering and there's a single match, select it
if (keycode === Inputs.KEY_ENTER) {
var dropdownElements = this._dropdownElement.element.querySelectorAll("." + CLASS_ITEM);
if (dropdownElements.length === 1) {
this._selectedItemChanged(dropdownElements[0], true, this._multiselection);
e.stopPropagation();
}
}
};
/**
* Fired when the user releases a key in the filter field
*/
Select.prototype._handleFilterKeyup = function (e) {
var target = e.target;
// Filter has changed
if (target.value !== this._activeFilter && target.value !== this._placeholderText && target.value !== this._lastSelectedOption.innerHTML) {
this._setFilter(target.value);
}
};
/**
* Fired when the user focusses the filter input field
*/
Select.prototype._handleFilterFocus = function (e) {
var target = e.target;
setTimeout(function () {
target.select();
});
};
/**
* Filters the Select by a given filter keyword
* @param filter Keyword to filter by
*/
Select.prototype._setFilter = function (filter) {
if (filter === void 0) { filter = ""; }
this._activeFilter = (filter.length >= this._minFilterLength) ? filter : "";
this.setOptions(this.getInitialOptions());
};
/**
* Resets the filter
*/
Select.prototype._clearFilter = function () {
delete this._activeFilter;
this.setOptions(this.getInitialOptions());
};
/**
* Set new content and reload the Select
* @param elements Array of new option (or optgroup) elements to display
*/
Select.prototype.setOptions = function (options) {
var _this = this;
this._emptyNode(this.element);
options.forEach(function (option) {
_this.element.appendChild(option);
});
// Preserve selected value if the selected
this.element.value = this._lastSelectedOption.value;
this.reload();
};
/**
* Clear all children of a given node
* @param node Node
*/
Select.prototype._emptyNode = function (node) {
while (node.firstChild) {
node.removeChild(node.firstChild);
}
};
/**
* Returns whether an option is a placeholder option
*/
Select.prototype._isPlaceholder = function (option) {
return option.hasAttribute("disabled") && option.hasAttribute("selected");
};
/**
* Update placeholder value
* @param text Content of the placeholder
*/
Select.prototype._setPlaceholder = function (text) {
if (this._placeholderElement && text) {
if (this._isFilterable()) {
this._placeholderElement.element.value = text;
}
else {
this._placeholderElement.setHtml(text);
}
}
};
Object.defineProperty(Select.prototype, "value", {
/**
* Gets the value of the currently selected option.
* If multiple selection is enabled this property returns an array of values.
*/
get: function () {
if (this._multiselection) {
return this._getSelectedOptions().map(function (x) { return x.value; });
}
if (this.element.value === "") {
return null;
}
return this.element.value;
},
enumerable: true,
configurable: true
});
Object.defineProperty(Select.prototype, "disabled", {
/**
* Enables or disables the select component depending on the
* 'value' parameter.
* @param {value} If true disables the control; false enables it.
*/
set: function (value) {
if (value) {
this.disable();
}
else {
this.enable();
}
},
enumerable: true,
configurable: true
});
/**
* Reloads the dropdown's option data definitions from the DOM and updates
* the generated dropdown display items.
*/
Select.prototype.reload = function () {
// Remove all existing child elements
this._emptyNode(this._dropdownElement.element);
if (this._activeFilter === undefined) { // If the user is filtering, let the placeholder "input" alive
this._setupPlaceholder();
}
this._createOptions(this.element);
this._updateSize();
this._updateMessage();
if (!this._isFilterable()) {
this._updatePlaceholder(!!this.value);
}
};
/**
* Sets the select control to the enabled state.
*/
Select.prototype.enable = function () {
this.element.removeAttribute("disabled");
this._wrapperElement.removeClass(CLASS_DISABLED);
window.addEventListener("click", this._windowClickHandler);
this._wrapperElement.element.addEventListener("click", this._clickHandler);
this._wrapperElement.element.addEventListener("keydown", this._keydownHandler);
this._wrapperElement.element.addEventListener("focus", this._focusHandler);
this._wrapperElement.element.addEventListener("blur", this._blurHandler);
};
/**
* Sets the select control to the disabled state.
*/
Select.prototype.disable = function () {
this.element.setAttribute("disabled", "");
this._wrapperElement.addClass(CLASS_DISABLED);
window.removeEventListener("click", this._windowClickHandler);
this._wrapperElement.element.removeEventListener("click", this._clickHandler);
this._wrapperElement.element.removeEventListener("keydown", this._keydownHandler);
this._wrapperElement.element.removeEventListener("focus", this._focusHandler);
this._wrapperElement.element.removeEventListener("blur", this._blurHandler);
this.close();
};
/**
* Toggles the open/closed state of the select dropdown.
*/
Select.prototype.toggle = function () {
if (this.isOpen()) {
this.close();
}
else {
this.open();
}
};
/**
* Gets if the select dropdown is open or closed.
* @return {boolean} True if open; otherwise false.
*/
Select.prototype.isOpen = function () {
return this._wrapperElement.hasClass(CLASS_OPEN);
};
/**
* Opens the select dropdown.
*/
Select.prototype.open = function () {
if (!this.isOpen()) {
this._openByFocus = false;
this._wrapperElement.removeClass(CLASS_CLOSED);
this._wrapperElement.addClass(CLASS_OPEN);
this._dropdownElement.element.addEventListener("click", this._handleDropdownClick);
this._dropdownElement.element.addEventListener("tap", this._handleDropdownClick);
}
};
/**
* Closes the select dropdown.
*/
Select.prototype.close = function () {
if (this.isOpen()) {
this._openByFocus = false;
this._wrapperElement.removeClass(CLASS_OPEN);
this._wrapperElement.addClass(CLASS_CLOSED);
// If the Select is filterable and therefore has an input field,
// reset the value of it to the chosen option
if (this._isFilterable()) {
// Unfocus input field
this._placeholderElement.element.blur();
if (!this._activeFilter || this._activeFilter === this._lastSelectedOption.innerHTML) {
this._setPlaceholder(this._lastSelectedOption.innerHTML);
}
}
this._dropdownElement.element.removeEventListener("click", this._handleDropdownClick);
this._dropdownElement.element.removeEventListener("tap", this._handleDropdownClick);
var focusedItem = this._wrapperElement.find("." + CLASS_ITEM_FOCUSED);
if (focusedItem) {
focusedItem.removeClass(CLASS_ITEM_FOCUSED);
}
}
};
/**
* Returns true when the element has the filter modifier class
*/
Select.prototype._isFilterable = function () {
return this._wrapperElement.hasClass(CLASS_FILTERABLE);
};
/**
* Destroys the component and clears all references.
*/
Select.prototype.destroy = function () {
window.removeEventListener("click", this._windowClickHandler);
if (this._dropdownElement) {
this._dropdownElement.element.removeEventListener("click", this._handleDropdownClick);
this._dropdownElement.element.removeEventListener("tap", this._handleDropdownClick);
remove(this._dropdownElement.element);
this._dropdownElement = undefined;
}
if (this._placeholderElement) {
this._placeholderElement.removeEventListener("keydown", this._filterKeydownHandler);
this._placeholderElement.removeEventListener("keyup", this._filterKeyupHandler);
this._placeholderElement.removeEventListener("focus", this._filterFocusHandler);
}
if (this._wrapperElement) {
this._wrapperElement.element.removeEventListener("click", this._clickHandler);
this._wrapperElement.element.removeEventListener("keydown", this._keydownHandler);
this._wrapperElement.element.removeEventListener("focus", this._focusHandler);
this._wrapperElement.element.removeEventListener("blur", this._blurHandler);
this._wrapperElement = undefined;
}
if (this._selectButtonElement) {
remove(this._selectButtonElement.element);
this._selectButtonElement = undefined;
}
this.removeClass(CLASS_CLOSED);
};
return Select;
}(DomElement));
export function init() {
searchAndInitialize("select", function (e) {
new Select(e);
});
}
export default Select;
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["main/src/form/Select.ts"],"names":[],"mappings":"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","file":"main/src/form/Select.js","sourcesContent":["import { searchAndInitialize, preventDefault, find, remove, internetExplorerOrEdgeVersion, scrollIntoView } from \"../Utils\"\nimport DomElement from \"../DomElement\"\nimport * as Inputs from \"../Inputs\"\nimport * as Dom from \"../DomFunctions\"\n\nconst CLASS_PLACEHOLDER = \"select__placeholder\"\nconst CLASS_THUMB = \"select__thumb\"\nconst CLASS_BUTTON = \"select__button\"\nconst CLASS_DROPDOWN = \"select__dropdown\"\n\nconst CLASS_OPEN = \"select--open\"\nconst CLASS_CLOSED = \"select--closed\"\nconst CLASS_DISABLED = \"select--disabled\"\nconst CLASS_FILTERABLE = \"select--filterable\"\n\nconst CLASS_ITEM = \"dropdown-item\"\nconst CLASS_ITEM_SELECTED = \"dropdown-item--selected\"\nconst CLASS_ITEM_FOCUSED = \"dropdown-item--focused\"\nconst CLASS_ITEM_DISABLED = \"dropdown-item--disabled\"\n\nconst CLASS_GROUP_ITEM = \"dropdown-group\"\nconst CLASS_GROUP_HEADER = \"dropdown-group__item\"\n\nconst QUERY_MESSAGE = \".message\"\n\nconst TIMEOUT_CLOSE = 150\nconst TIMEOUT_BLUR = 400\n\n/**\n * The select component API.\n */\nclass Select extends DomElement<HTMLSelectElement> {\n  private _openByFocus: boolean\n  private _multiselection: boolean\n  private _clickHandler: (e: Event) => void\n  private _handleDropdownClick: (e: Event) => void\n  private _keydownHandler: (e: Event) => void\n  private _focusHandler: (e: Event) => void\n  private _blurHandler: (e: Event) => void\n  private _windowClickHandler: (e: Event) => void\n  private _filterKeydownHandler: (e: Event) => void\n  private _filterKeyupHandler: (e: Event) => void\n  private _filterFocusHandler: (e: Event) => void\n\n  private _wrapperElement!: DomElement\n  private _dropdownElement!: DomElement<HTMLElement>\n\n  private _selectButtonElement!: DomElement\n  private _thumbElement!: DomElement\n\n  private _placeholderOption?: HTMLOptionElement\n  private _placeholderElement!: DomElement\n  private _placeholderText!: string\n\n  private _lastHandledEvent?: Event\n  private _lastSelectedOption?: HTMLOptionElement\n\n  // Minimum filter length\n  private _minFilterLength = 2\n\n  // The keyword the Select is currently filtered by\n  private _activeFilter?: string\n\n  // The options the Select was initially created upon\n  // These will be used as a basis for filtering\n  private _initialOptions = Array.prototype.slice.call(this.element.children)\n\n  constructor(element: HTMLSelectElement) {\n    super(element)\n\n    this._openByFocus = false\n\n    // Check for multi-selection\n    this._multiselection = this.element.hasAttribute(\"multiple\") === true\n\n    // Setup event context\n    this._clickHandler = this._handleClick.bind(this)\n    this._handleDropdownClick = this._handleClick.bind(this)\n    this._keydownHandler = this._handleKeydown.bind(this)\n    this._focusHandler = this._handleFocus.bind(this)\n    this._blurHandler = this._handleBlur.bind(this)\n    this._windowClickHandler = this._handleWindowClick.bind(this)\n    this._filterKeydownHandler = this._handleFilterKeydown.bind(this)\n    this._filterKeyupHandler = this._handleFilterKeyup.bind(this)\n    this._filterFocusHandler = this._handleFilterFocus.bind(this)\n\n    this._initialize()\n  }\n\n  /**\n   * Initializes the select component.\n   *\n   * This method inspects the select definition and its options and\n   * generates new stylable DOM elements around the original select-element\n   * definitions.\n   * @private\n   */\n  protected _initialize() {\n    const selectedOption = this.element.querySelector(\"option[selected]\") as HTMLOptionElement\n    const firstOption = this.element.querySelector(\"option\") as HTMLOptionElement\n\n    // Per default, set the last selected option to either the option with a \"selected\" attribute,\n    // or, if not found, to the first available option\n    this._lastSelectedOption = selectedOption || firstOption\n\n    this._wrapperElement = new DomElement(this.element.parentElement!)\n      .addClass(CLASS_CLOSED)\n\n    for (let cls of this.classes) {\n      this._wrapperElement.addClass(cls)\n    }\n\n    this._dropdownElement = new DomElement<HTMLElement>(\"div\")\n      .addClass(CLASS_DROPDOWN)\n\n    if (internetExplorerOrEdgeVersion() > 0 && internetExplorerOrEdgeVersion() < 12) {\n      // This is a workaround for IE browsers 11 and earlier where focusing\n      // a scrollable dropdown list will close the dropdown prematurely.\n      this._dropdownElement.element.addEventListener(\"mousedown\", (event: MouseEvent) => event.preventDefault())\n    }\n\n    this._setupTarget()\n    this._setupPlaceholder()\n\n    this._wrapperElement.appendChild(this._dropdownElement)\n\n    this._createOptions(this.element)\n\n    this._updateSize()\n    this._updateMessage()\n\n    if (this.element.disabled) {\n      this.disable()\n    } else {\n      this.enable()\n    }\n  }\n\n  protected _setupTarget() {\n    // move the id from the select element to the wrapper\n    const id = this.element.getAttribute(\"id\")\n    if (id) {\n      this.element.removeAttribute(\"id\")\n      this._wrapperElement.setAttribute(\"id\", id)\n    }\n\n    // Apply the tab index\n    const tabIndex = this.element.getAttribute(\"tabindex\")\n    if (tabIndex) {\n      this._wrapperElement.setAttribute(\"tabIndex\", tabIndex)\n    }\n  }\n\n  protected _setupPlaceholder() {\n    if (!this._selectButtonElement) {\n      this._selectButtonElement = new DomElement(\"div\")\n        .addClass(CLASS_BUTTON)\n\n      this._wrapperElement.appendChild(this._selectButtonElement)\n    }\n\n    if (!this._thumbElement) {\n      this._thumbElement = new DomElement(\"div\")\n        .addClass(CLASS_THUMB)\n\n      let thumbIcon = new DomElement(\"div\")\n        .addClass(\"thumb-icon\")\n\n      let loader = new DomElement(\"div\")\n        .addClass(\"loader-spinner\")\n        .addClass(\"loader-spinner--small\")\n\n      this._thumbElement.appendChild(loader)\n      this._thumbElement.appendChild(thumbIcon)\n      this._selectButtonElement.appendChild(this._thumbElement)\n    }\n\n    let placeholderText = \"\"\n\n    this._placeholderOption = this.element.querySelector(\"option[selected][disabled]\") as HTMLOptionElement || undefined\n\n    if (this._placeholderOption) {\n      placeholderText = Dom.text(this._placeholderOption)\n\n      if (this._multiselection === true) {\n        this._placeholderOption.selected = false\n      }\n    }\n\n    let selectedOption = this.element.querySelector(\"option[selected]:not([disabled])\")\n\n    if (selectedOption) {\n      placeholderText = Dom.text(selectedOption)\n    }\n\n    if (!this._placeholderElement) {\n      // When the Select is filterable, create an \"input\" as the placeholder element, otherwise a \"span\"\n      if (this._isFilterable()) {\n        this._placeholderElement = new DomElement(\"input\")\n        this._placeholderElement.addEventListener(\"keyup\", (e) => this._handleFilterKeyup(e))\n        this._placeholderElement.addEventListener(\"keydown\", (e) => this._handleFilterKeydown(e))\n        this._placeholderElement.addEventListener(\"focus\", (e) => this._handleFilterFocus(e))\n      } else {\n        this._placeholderElement = new DomElement(\"span\")\n      }\n\n      this._placeholderElement.addClass(CLASS_PLACEHOLDER)\n      this._selectButtonElement.appendChild(this._placeholderElement)\n    }\n\n    this._setPlaceholder(placeholderText)\n    this._placeholderText = placeholderText\n\n    if (selectedOption && selectedOption !== this._placeholderOption) {\n      this._updatePlaceholder(true)\n    }\n  }\n\n  protected _updateMessage() {\n    const messageNode = this._wrapperElement.element.querySelector(QUERY_MESSAGE)\n    if (messageNode !== null) {\n      this._wrapperElement.appendChild(new DomElement(messageNode))\n    }\n  }\n\n  private _isOptGroup(element: Element): element is HTMLOptGroupElement {\n    return element.tagName.toUpperCase() === \"OPTGROUP\"\n  }\n\n  private _isOption(element: Element): element is HTMLOptionElement {\n    return element.tagName.toUpperCase() === \"OPTION\"\n  }\n\n  protected _createOptions(element: HTMLSelectElement) {\n    for (let i = 0; i < element.children.length; i++) {\n      let child = element.children[i]\n\n      if (this._isOptGroup(child)) {\n        this._appendGroup(child as HTMLOptGroupElement)\n      }\n\n      if (this._isOption(child)) {\n        let option = this._createOption(child as HTMLOptionElement)\n\n        if (option) {\n          this._dropdownElement.appendChild(option)\n        }\n      }\n    }\n  }\n\n  protected _createOption(option: HTMLOptionElement) {\n    let html = option.innerHTML\n\n    if (this._activeFilter) {\n      const sanitizedActiveFilter = this._activeFilter.replace(/[-\\\\^$*+?.()|[\\]{}]/g, \"\\\\$&\")\n      html = html.replace(new RegExp(`(${sanitizedActiveFilter})`, \"gi\"), \"<strong>$1</strong>\")\n    }\n\n    let opt = new DomElement(\"div\")\n      .addClass(CLASS_ITEM)\n      .setHtml(html)\n\n    if (option.selected) {\n      opt.addClass(CLASS_ITEM_SELECTED)\n    }\n\n    if (option.disabled) {\n      opt.addClass(CLASS_ITEM_DISABLED)\n    }\n\n    if (!this._isPlaceholder(option)) {\n      opt.setAttribute(\"data-value\", option.value)\n      return opt\n    }\n\n    return undefined\n  }\n\n  protected _appendGroup(optgroup: HTMLOptGroupElement) {\n    let label = optgroup.getAttribute(\"label\")!\n\n    let group = new DomElement(\"div\")\n      .addClass(CLASS_GROUP_ITEM)\n\n    let groupHeader = new DomElement(\"div\")\n      .addClass(CLASS_GROUP_HEADER)\n      .setHtml(label)\n\n    group.appendChild(groupHeader)\n\n    let options = optgroup.querySelectorAll(\"option\")\n    for (let entry of options) {\n      let option = this._createOption(entry)\n      if (option) {\n        group.appendChild(option)\n      }\n    }\n\n    this._dropdownElement.appendChild(group)\n    return group\n  }\n\n  protected _updateSize() {\n    // Note: Mirroring the DOM and measuring the items using their clientWidth was very\n    // unreliable, therefore measuring was switched to the new HTML5 measureText method\n    // margins and paddings arround the text are copied from the original placeholder items\n    // dimension\n    const placeholderStyle = window.getComputedStyle(this._placeholderElement.element)\n\n    let paddingRight = parseFloat(placeholderStyle.paddingRight!)\n    let paddingLeft = parseFloat(placeholderStyle.paddingLeft!)\n\n    let font = this._placeholderElement.css(\"font\")\n    let textWidth = Dom.textWidth(this._placeholderText, font)\n    let maxWidth = paddingLeft + paddingRight + textWidth\n\n    let options = this._wrapperElement.element.querySelectorAll(`.${CLASS_ITEM}`)\n    for (let entry of options) {\n      let width = Dom.textWidth(Dom.text(entry), font) + paddingLeft + paddingRight\n\n      if (width > maxWidth) {\n        maxWidth = width\n      }\n    }\n\n  }\n\n  protected _isButtonTarget(target: EventTarget) {\n    return (target === this._wrapperElement.element ||\n      target === this._placeholderElement.element ||\n      target === this._selectButtonElement.element ||\n      target === this._thumbElement.element)\n  }\n\n  protected _isDropdownTarget(target: EventTarget) {\n    let current = target as HTMLElement\n    while (current !== this._dropdownElement.element && current.parentElement) {\n      current = current.parentElement\n    }\n\n    return current === this._dropdownElement.element\n  }\n\n  /**\n   * Updates the UI if the selection has changed and makes sure the\n   * select control and the generated markup are synchronized.\n   * @private\n   */\n  protected _selectedItemChanged(\n    newItem: Element,\n    autoClose = true,\n    multiselect = false\n  ) {\n    const oldItems = this._dropdownElement.element.querySelectorAll(`.${CLASS_ITEM_SELECTED}`)\n\n    if (!newItem) {\n      setTimeout(() => this.close(), TIMEOUT_CLOSE)\n      return\n    }\n\n    if (Dom.hasClass(newItem, CLASS_ITEM_DISABLED)) {\n      return\n    }\n\n    if ((oldItems.length === 0) && !newItem) {\n      throw new Error(\"Can not select undefined elements\")\n    }\n\n    let oldItem = oldItems[0]\n\n    if (multiselect === true) {\n      oldItem = find(oldItems, (x) => x.getAttribute(\"data-value\") === newItem.getAttribute(\"data-value\"))!\n    }\n\n    let isDeselect = false\n\n    if (newItem && oldItem && oldItem === newItem) {\n      // Click on a previously selected element -> deselect\n      isDeselect = true\n\n      if (!this._placeholderOption && !multiselect) {\n        // If there is no placeholder option, non multiselect options cannot be deselected\n        return\n      }\n\n      delete this._lastSelectedOption\n    }\n\n    if (oldItem) {\n      // Remove selection on the element\n      let oldValue = oldItem.getAttribute(\"data-value\")\n      let optElement = find(this.element.options, (x) => !x.disabled && x.value === oldValue)\n\n      if (!optElement) {\n        throw new Error(`The option with value ${oldValue} does not exist`)\n      }\n\n      // Unset Select value\n      optElement.selected = false\n      Dom.removeClass(oldItem, CLASS_ITEM_SELECTED)\n    }\n\n    if (!isDeselect) { // Select an option\n      // Select a new item\n      let newValue = newItem.getAttribute(\"data-value\")\n      let optElement = find(this.element.options, (x) => !x.disabled && x.value === newValue)\n\n      if (!optElement) {\n        throw new Error(`The option with value ${newValue} does not exist`)\n      }\n\n      // Set Select value\n      optElement.selected = true\n      Dom.addClass(newItem, CLASS_ITEM_SELECTED)\n\n      // Preserve selection\n      this._lastSelectedOption = optElement\n\n    } else { // Deselect an option\n      // Keep track of falling back to the placeholder (if any)\n      if (this._placeholderOption) {\n        this._lastSelectedOption = this._placeholderOption\n      }\n    }\n\n    let hasSelectedItems = true\n\n    if (this._multiselection === false && isDeselect) {\n      // Handle no selection for non-multiselect states\n      this._placeholderOption!.selected = true\n      hasSelectedItems = false\n    }\n\n    if (this._multiselection === true && this._getSelectedOptions().length === 0) {\n      hasSelectedItems = false\n    }\n\n    // Reset the filter if filterable\n    if (this._activeFilter) {\n      this._clearFilter()\n    }\n\n    this._updatePlaceholder(hasSelectedItems)\n\n    // Dispatch the changed event\n    this.dispatchEvent(\"change\")\n\n    if (autoClose && !multiselect) {\n      setTimeout(() => {\n        this.close()\n      }, TIMEOUT_CLOSE)\n    }\n  }\n\n  protected _updatePlaceholder(hasSelectedItems: boolean) {\n    let text = this._placeholderOption ? Dom.text(this._placeholderOption) : \" \"\n\n    if (hasSelectedItems === true) {\n      let selectedItems = this._getSelectedOptions()\n\n      if (selectedItems.length > 0) {\n        text = \"\"\n        for (let item of selectedItems) {\n          text += `${Dom.text(item)}, `\n        }\n        text = text.substring(0, text.length - 2)\n      }\n    }\n\n    this._setPlaceholder(text)\n  }\n\n  protected _getSelectedOptions() {\n    let selectedOptions: HTMLOptionElement[] = []\n    if (this.element.options) {\n      [].forEach.call(this.element.options, ((option: HTMLOptionElement) => {\n        if (option.selected && !option.disabled) {\n          selectedOptions.push(option)\n        }\n      }))\n    }\n    return selectedOptions\n  }\n\n  /**\n   * Clone all of the initially set options (and optgroups) and returns them in a new array.\n   * This serves as the basis for filtering. If a filter is present, it will be respected.\n   */\n  private getInitialOptions(): Element[] {\n    const filter = this._activeFilter || \"\"\n    const filtered: Element[] = []\n    const initialOptions = this._initialOptions\n\n    for (let i = 0; i < initialOptions.length; i++) {\n      const child: Element = initialOptions[i] as Element\n\n      if (this._isOptGroup(child)) { // handle <optgroup>\n        const optGroupClone: Element = child.cloneNode(false) as Element\n        let found = false\n\n        for (let j = 0; j < child.children.length; j++) {\n          const optionClone: Element = child.children[j].cloneNode(true) as Element\n\n          // Append on match\n          if (this._containsWord(optionClone.innerHTML, filter)) {\n            optGroupClone.appendChild(optionClone)\n            found = true\n          }\n        }\n\n        // Push if any matches found\n        if (found) {\n          filtered.push(optGroupClone)\n        }\n\n      } else if (this._isOption(child)) { // handle <option>\n        const optionClone: Element = child.cloneNode(true) as Element\n\n        // Push on match\n        if (this._containsWord(optionClone.innerHTML, filter)) {\n          filtered.push(optionClone)\n        }\n      }\n    }\n\n    return filtered\n  }\n\n  /**\n   * Returns true if a text contains a given keyword, e.g. in \"ca\" in \"Car\"\n   */\n  private _containsWord(text: string, keyword: string): boolean {\n    return text.toLowerCase().indexOf(keyword.toLowerCase()) > -1\n  }\n\n  protected _handleFocus() {\n    this.open()\n    this._openByFocus = true\n\n    setTimeout(() => {\n      this._openByFocus = false\n    }, TIMEOUT_BLUR)\n  }\n\n  protected _handleBlur() {\n    this.close()\n  }\n\n  protected _handleClick(event: MouseEvent) {\n    let handled = false\n\n    if (this._lastHandledEvent === event) {\n      this._lastHandledEvent = undefined\n      return\n    }\n\n    if (this._isButtonTarget(event.target!) && this._openByFocus === false) {\n      // handle header item clicks and toggle dropdown\n      this.toggle()\n      handled = true\n    }\n\n    let newItem = event.target as Element\n\n    if (!handled && Dom.hasClass(newItem, CLASS_ITEM)) {\n      // handle clicks on dropdown items\n      this._selectedItemChanged(newItem, true, this._multiselection)\n      handled = true\n    }\n\n    if (handled) {\n      this._lastHandledEvent = event\n      preventDefault(event)\n    }\n  }\n\n  protected _handleWindowClick(event: MouseEvent) {\n    if (this._isDropdownTarget(event.target!) || this._isButtonTarget(event.target!)) {\n      return\n    }\n\n    this.close()\n  }\n\n  protected _focusOptionStartingWith(keycode: number, startIndex: number, options: NodeListOf<HTMLElement>) {\n    for (let index = startIndex; index < options.length; index++) {\n      let item = new DomElement(options[index])\n      let value = item.innerText.toLowerCase()\n\n      if (index > options.length) {\n        index = 0\n      }\n\n      if (value.startsWith(Inputs.getKeyValue(keycode))) {\n        let newOption = new DomElement(options[index])\n\n        if (!newOption.hasClass(CLASS_ITEM_DISABLED)) {\n          scrollIntoView(options[index])\n          newOption.addClass(CLASS_ITEM_FOCUSED)\n          return newOption\n        }\n      }\n    }\n    return undefined\n  }\n\n  protected _handleKeydown(event: KeyboardEvent) {\n    let evt = event || window.event\n    let keycode = event.which || event.keyCode\n\n    if (keycode === Inputs.KEY_ESCAPE) {\n      // handle Escape key (ESC)\n      if (this.isOpen()) {\n        this.close()\n      }\n      evt.preventDefault()\n      return\n    }\n\n    if (keycode === Inputs.KEY_ARROW_UP || keycode === Inputs.KEY_ARROW_DOWN) {\n      // Up and down arrows\n\n      let options = this._wrapperElement.element.querySelectorAll(`.${CLASS_ITEM}`) as NodeListOf<HTMLElement>\n      if (options.length > 0) {\n\n        let newIndex = 0\n        let oldOption\n\n        let focusedElement = this._wrapperElement.find(`.${CLASS_ITEM_FOCUSED}`)\n        let searchFor = focusedElement ? CLASS_ITEM_FOCUSED : CLASS_ITEM_SELECTED\n\n        let newElement\n\n        for (let index = 0; index < options.length; index++) {\n          let direction = keycode === Inputs.KEY_ARROW_DOWN ? 1 : -1\n\n          let item = new DomElement(options[index])\n\n          // search for selected or focusedElement elements\n          if (item.hasClass(searchFor)) {\n            oldOption = item\n            newIndex = index\n\n            // get the next not disabled element in the appropriate direction\n            for (let count = 0; count < options.length; count++) {\n              newIndex += direction\n              newIndex %= options.length\n\n              if (newIndex < 0) {\n                newIndex = options.length - 1\n              }\n\n              newElement = new DomElement(options[newIndex])\n              if (!newElement.hasClass(CLASS_ITEM_DISABLED)) {\n                break\n              }\n            }\n          }\n        }\n\n        // set the new element focused\n        scrollIntoView(options[newIndex])\n        let newOption = new DomElement(options[newIndex])\n        newOption.addClass(CLASS_ITEM_FOCUSED)\n\n        if (oldOption) {\n          oldOption.removeClass(CLASS_ITEM_FOCUSED)\n        }\n      }\n\n      evt.preventDefault()\n      return\n    }\n\n    if (Inputs.getKeyValue(keycode) && !this._isFilterable()) {\n      // Keyboard keys\n\n      let options = this._wrapperElement.element.querySelectorAll(`.${CLASS_ITEM}`) as NodeListOf<HTMLElement>\n      if (options.length > 0) {\n\n        let oldFocusIndex = 0\n        let hasFocusedOption = false\n\n        for (let index = 0; index < options.length; index++) {\n          let item = new DomElement(options[index])\n\n          if (item.hasClass(CLASS_ITEM_FOCUSED)) {\n            item.removeClass(CLASS_ITEM_FOCUSED)\n\n            let value = item.innerText.toLowerCase()\n            if (value.startsWith(Inputs.getKeyValue(keycode))) {\n              hasFocusedOption = true\n              oldFocusIndex = index\n            }\n          }\n        }\n\n        let newOption = this._focusOptionStartingWith(keycode, hasFocusedOption ? oldFocusIndex + 1 : 0, options)\n        if (newOption === undefined){\n          this._focusOptionStartingWith(keycode, 0, options)\n        }\n      }\n\n      evt.preventDefault()\n      return\n    }\n\n    if (keycode === Inputs.KEY_ENTER || keycode === Inputs.KEY_TAB) {\n      // Handle enter and tab key by selecting the currently focused element\n      let newItem = this._dropdownElement.element.querySelector(`.${CLASS_ITEM_FOCUSED}`)!\n      this._selectedItemChanged(newItem, true, this._multiselection)\n    }\n  }\n\n  /**\n   * Fired when the user presses a key in the filter field\n   */\n  private _handleFilterKeydown(e: KeyboardEvent): void {\n    const keycode = e.which || e.keyCode\n\n    // If the user hits the enter key while filtering and there's a single match, select it\n    if (keycode === Inputs.KEY_ENTER) {\n      const dropdownElements = this._dropdownElement.element.querySelectorAll(`.${CLASS_ITEM}`)\n\n      if (dropdownElements.length === 1) {\n        this._selectedItemChanged(dropdownElements[0], true, this._multiselection)\n        e.stopPropagation()\n      }\n    }\n  }\n\n  /**\n   * Fired when the user releases a key in the filter field\n   */\n  private _handleFilterKeyup(e: KeyboardEvent): void {\n    const target = e.target as HTMLInputElement\n\n    // Filter has changed\n    if (target.value !== this._activeFilter && target.value !== this._placeholderText && target.value !== this._lastSelectedOption!.innerHTML) {\n      this._setFilter(target.value)\n    }\n  }\n\n  /**\n   * Fired when the user focusses the filter input field\n   */\n  private _handleFilterFocus(e: FocusEvent): void {\n    const target = e.target as HTMLInputElement\n\n    setTimeout(() => {\n      target.select()\n    })\n  }\n\n  /**\n   * Filters the Select by a given filter keyword\n   * @param filter Keyword to filter by\n   */\n  private _setFilter(filter: string = \"\"): void {\n    this._activeFilter = (filter.length >= this._minFilterLength) ? filter : \"\"\n    this.setOptions(this.getInitialOptions())\n  }\n\n  /**\n   * Resets the filter\n   */\n  private _clearFilter(): void {\n    delete this._activeFilter\n    this.setOptions(this.getInitialOptions())\n  }\n\n  /**\n   * Set new content and reload the Select\n   * @param elements Array of new option (or optgroup) elements to display\n   */\n  private setOptions(options: Element[]): void {\n    this._emptyNode(this.element)\n\n    options.forEach((option) => {\n      this.element.appendChild(option)\n    })\n\n    // Preserve selected value if the selected\n    this.element.value = this._lastSelectedOption!.value\n\n    this.reload()\n  }\n\n  /**\n   * Clear all children of a given node\n   * @param node Node\n   */\n  private _emptyNode(node: Node): void {\n    while (node.firstChild) {\n      node.removeChild(node.firstChild)\n    }\n  }\n\n  /**\n   * Returns whether an option is a placeholder option\n   */\n  private _isPlaceholder(option: HTMLOptionElement): boolean {\n    return option.hasAttribute(\"disabled\") && option.hasAttribute(\"selected\")\n  }\n\n  /**\n   * Update placeholder value\n   * @param text Content of the placeholder\n   */\n  protected _setPlaceholder(text: string): void {\n    if (this._placeholderElement && text) {\n      if (this._isFilterable()) {\n        (this._placeholderElement as DomElement<HTMLInputElement>).element.value = text\n      } else {\n        this._placeholderElement.setHtml(text)\n      }\n    }\n  }\n\n  /**\n   * Gets the value of the currently selected option.\n   * If multiple selection is enabled this property returns an array of values.\n   */\n  get value() {\n    if (this._multiselection) {\n      return this._getSelectedOptions().map((x) => x.value)\n    }\n\n    if (this.element.value === \"\") {\n      return null\n    }\n\n    return this.element.value\n  }\n\n  /**\n   * Enables or disables the select component depending on the\n   * 'value' parameter.\n   * @param {value} If true disables the control; false enables it.\n   */\n  set disabled(value: boolean) {\n    if (value) {\n      this.disable()\n    } else {\n      this.enable()\n    }\n  }\n\n  /**\n   * Reloads the dropdown's option data definitions from the DOM and updates\n   * the generated dropdown display items.\n   */\n  reload() {\n    // Remove all existing child elements\n    this._emptyNode(this._dropdownElement.element)\n\n    if (this._activeFilter === undefined) { // If the user is filtering, let the placeholder \"input\" alive\n      this._setupPlaceholder()\n    }\n\n    this._createOptions(this.element)\n\n    this._updateSize()\n    this._updateMessage()\n\n    if (!this._isFilterable()) {\n      this._updatePlaceholder(!!this.value)\n    }\n  }\n\n  /**\n   * Sets the select control to the enabled state.\n   */\n  enable() {\n    this.element.removeAttribute(\"disabled\")\n    this._wrapperElement.removeClass(CLASS_DISABLED)\n\n    window.addEventListener(\"click\", this._windowClickHandler)\n\n    this._wrapperElement.element.addEventListener(\"click\", this._clickHandler)\n    this._wrapperElement.element.addEventListener(\"keydown\", this._keydownHandler)\n    this._wrapperElement.element.addEventListener(\"focus\", this._focusHandler)\n    this._wrapperElement.element.addEventListener(\"blur\", this._blurHandler)\n  }\n\n  /**\n   * Sets the select control to the disabled state.\n   */\n  disable() {\n    this.element.setAttribute(\"disabled\", \"\")\n    this._wrapperElement.addClass(CLASS_DISABLED)\n\n    window.removeEventListener(\"click\", this._windowClickHandler)\n\n    this._wrapperElement.element.removeEventListener(\"click\", this._clickHandler)\n    this._wrapperElement.element.removeEventListener(\"keydown\", this._keydownHandler)\n    this._wrapperElement.element.removeEventListener(\"focus\", this._focusHandler)\n    this._wrapperElement.element.removeEventListener(\"blur\", this._blurHandler)\n\n    this.close()\n  }\n\n  /**\n   * Toggles the open/closed state of the select dropdown.\n   */\n  toggle() {\n    if (this.isOpen()) {\n      this.close()\n    } else {\n      this.open()\n    }\n  }\n\n  /**\n   * Gets if the select dropdown is open or closed.\n   * @return {boolean} True if open; otherwise false.\n   */\n  isOpen() {\n    return this._wrapperElement.hasClass(CLASS_OPEN)\n  }\n\n  /**\n   * Opens the select dropdown.\n   */\n  open() {\n    if (!this.isOpen()) {\n      this._openByFocus = false\n\n      this._wrapperElement.removeClass(CLASS_CLOSED)\n      this._wrapperElement.addClass(CLASS_OPEN)\n\n      this._dropdownElement.element.addEventListener(\"click\", this._handleDropdownClick)\n      this._dropdownElement.element.addEventListener(\"tap\", this._handleDropdownClick)\n    }\n  }\n\n  /**\n   * Closes the select dropdown.\n   */\n  close() {\n    if (this.isOpen()) {\n      this._openByFocus = false\n\n      this._wrapperElement.removeClass(CLASS_OPEN)\n      this._wrapperElement.addClass(CLASS_CLOSED)\n\n      // If the Select is filterable and therefore has an input field,\n      // reset the value of it to the chosen option\n      if (this._isFilterable()) {\n        // Unfocus input field\n        (this._placeholderElement.element as HTMLInputElement).blur()\n\n        if (!this._activeFilter || this._activeFilter === this._lastSelectedOption!.innerHTML) {\n          this._setPlaceholder(this._lastSelectedOption!.innerHTML)\n        }\n      }\n\n      this._dropdownElement.element.removeEventListener(\"click\", this._handleDropdownClick)\n      this._dropdownElement.element.removeEventListener(\"tap\", this._handleDropdownClick)\n\n      let focusedItem = this._wrapperElement.find(`.${CLASS_ITEM_FOCUSED}`)\n\n      if (focusedItem) {\n        focusedItem.removeClass(CLASS_ITEM_FOCUSED)\n      }\n    }\n  }\n\n  /**\n   * Returns true when the element has the filter modifier class\n   */\n  private _isFilterable(): boolean {\n    return this._wrapperElement.hasClass(CLASS_FILTERABLE)\n  }\n\n  /**\n   * Destroys the component and clears all references.\n   */\n  destroy() {\n    window.removeEventListener(\"click\", this._windowClickHandler)\n\n    if (this._dropdownElement) {\n      this._dropdownElement.element.removeEventListener(\"click\", this._handleDropdownClick)\n      this._dropdownElement.element.removeEventListener(\"tap\", this._handleDropdownClick)\n\n      remove(this._dropdownElement.element);\n      (this as any)._dropdownElement = undefined\n    }\n\n    if (this._placeholderElement) {\n      this._placeholderElement.removeEventListener(\"keydown\", this._filterKeydownHandler)\n      this._placeholderElement.removeEventListener(\"keyup\", this._filterKeyupHandler)\n      this._placeholderElement.removeEventListener(\"focus\", this._filterFocusHandler)\n    }\n\n    if (this._wrapperElement) {\n      this._wrapperElement.element.removeEventListener(\"click\", this._clickHandler)\n      this._wrapperElement.element.removeEventListener(\"keydown\", this._keydownHandler)\n      this._wrapperElement.element.removeEventListener(\"focus\", this._focusHandler)\n      this._wrapperElement.element.removeEventListener(\"blur\", this._blurHandler);\n\n      (this as any)._wrapperElement = undefined\n    }\n\n    if (this._selectButtonElement) {\n      remove(this._selectButtonElement.element);\n      (this as any)._selectButtonElement = undefined\n    }\n\n    this.removeClass(CLASS_CLOSED)\n  }\n}\n\nexport function init() {\n  searchAndInitialize<HTMLSelectElement>(\"select\", (e) => {\n    new Select(e)\n  })\n}\n\nexport default Select\n"],"sourceRoot":"../../../../.."}