UNPKG

slim-select

Version:

Slim advanced select dropdown

1,104 lines 65.2 kB
class L { main; // Placeholder placeholder; // Values values; single; max; value; valueText; valueDelete; valueOut; // Deselect deselect; deselectPath; // Not a class but whatever // Arrow arrow; arrowClose; // Not a class but whatever arrowOpen; // Not a class but whatever // Content content; contentOpen; dirAbove; dirBelow; // Search search; searchHighlighter; searching; addable; addablePath; // Not a class but whatever // List optgroups/options list; // Optgroup optgroup; optgroupLabel; optgroupLabelText; optgroupActions; optgroupSelectAll; // optgroup select all optgroupSelectAllBox; // Not a class but whatever optgroupSelectAllCheck; // Not a class but whatever optgroupClosable; // Option option; optionDelete; // Not a class but whatever highlighted; // Misc mainOpen; close; selected; error; disabled; hide; constructor(e) { e || (e = {}); let t = (s = "", i = "") => `${s} ${i}`.trim(); this.main = t("ss-main", e.main), this.placeholder = t("ss-placeholder", e.placeholder), this.values = t("ss-values", e.values), this.single = t("ss-single", e.single), this.max = t("ss-max", e.max), this.value = t("ss-value", e.value), this.valueText = t("ss-value-text", e.valueText), this.valueDelete = t("ss-value-delete", e.valueDelete), this.valueOut = t("ss-value-out", e.valueOut), this.deselect = t("ss-deselect", e.deselect), this.deselectPath = e.deselectPath || "M10,10 L90,90 M10,90 L90,10", this.arrow = t("ss-arrow", e.arrow), this.arrowClose = e.arrowClose || "M10,30 L50,70 L90,30", this.arrowOpen = e.arrowOpen || "M10,70 L50,30 L90,70", this.content = t("ss-content", e.content), this.contentOpen = t("ss-open", e.contentOpen), this.dirAbove = t("ss-dir-above", e.dirAbove), this.dirBelow = t("ss-dir-below", e.dirBelow), this.search = t("ss-search", e.search), this.searchHighlighter = t("ss-search-highlight", e.searchHighlighter), this.searching = t("ss-searching", e.searching), this.addable = t("ss-addable", e.addable), this.addablePath = e.addablePath || "M50,10 L50,90 M10,50 L90,50", this.list = t("ss-list", e.list), this.optgroup = t("ss-optgroup", e.optgroup), this.optgroupLabel = t("ss-optgroup-label", e.optgroupLabel), this.optgroupLabelText = t("ss-optgroup-label-text", e.optgroupLabelText), this.optgroupActions = t("ss-optgroup-actions", e.optgroupActions), this.optgroupSelectAll = t("ss-selectall", e.optgroupSelectAll), this.optgroupSelectAllBox = e.optgroupSelectAllBox || "M60,10 L10,10 L10,90 L90,90 L90,50", this.optgroupSelectAllCheck = e.optgroupSelectAllCheck || "M30,45 L50,70 L90,10", this.optgroupClosable = t("ss-closable", e.optgroupClosable), this.option = t("ss-option", e.option), this.optionDelete = e.optionDelete || "M10,10 L90,90 M10,90 L90,10", this.highlighted = t("ss-highlighted", e.highlighted), this.mainOpen = t("ss-open", e.mainOpen), this.close = t("ss-close", e.close), this.selected = t("ss-selected", e.selected), this.error = t("ss-error", e.error), this.disabled = t("ss-disabled", e.disabled), this.hide = t("ss-hide", e.hide); } getFirst(e) { return this[e].split(" ")[0]; } } function y() { return Math.random().toString(36).substring(2, 10); } function A(p, e) { function t(i, l) { return l && i && i.classList && i.classList.contains(l) || l && i && i.dataset && i.dataset.id && i.dataset.id === e ? i : null; } function s(i, l) { return !i || i === document ? null : t(i, l) ? i : s(i.parentNode, l); } return t(p, e) || s(p, e); } function S(p, e = 50, t = !1) { let s; return function(...i) { const l = self, n = () => { s = null, t || p.apply(l, i); }, a = t && !s; clearTimeout(s), s = setTimeout(n, e), a && p.apply(l, i); }; } function O(p, e) { return JSON.stringify(p) === JSON.stringify(e); } function D(p) { const e = p.replace(/[A-Z\u00C0-\u00D6\u00D8-\u00DE]/g, (t) => "-" + t.toLowerCase()); return p[0] === p[0].toUpperCase() ? e.substring(1) : e; } class u { id; value; text; html; defaultSelected; selected; display; disabled; placeholder; class; style; data; mandatory; constructor(e) { this.id = !e.id || e.id === "" ? y() : e.id, this.value = e.value === void 0 ? e.text || "" : e.value || "", this.text = e.text || "", this.html = e.html || "", this.defaultSelected = e.defaultSelected !== void 0 ? e.defaultSelected : !1, this.selected = e.selected !== void 0 ? e.selected : !1, this.display = e.display !== void 0 ? e.display : !0, this.disabled = e.disabled !== void 0 ? e.disabled : !1, this.mandatory = e.mandatory !== void 0 ? e.mandatory : !1, this.placeholder = e.placeholder !== void 0 ? e.placeholder : !1, this.class = e.class || "", this.style = e.style || "", this.data = e.data || {}; } } class g { id; label; selectAll; selectAllText; closable; options; constructor(e) { if (this.id = !e.id || e.id === "" ? y() : e.id, this.label = e.label || "", this.selectAll = e.selectAll === void 0 ? !1 : e.selectAll, this.selectAllText = e.selectAllText || "Select All", this.closable = e.closable || "off", this.options = [], e.options) for (const t of e.options) this.options.push(new u(t)); } } class k { selectType = "single"; // Main data set, never null data = []; selectedOrder = []; constructor(e, t) { this.selectType = e, this.setData(t); } // Validate DataArrayPartial validateDataArray(e) { if (!Array.isArray(e)) return new Error("Data must be an array"); for (let t of e) if (t) if (t instanceof g || "label" in t) { if (!("label" in t)) return new Error("Optgroup must have a label"); if ("options" in t && t.options) for (let s of t.options) { const i = this.validateOption(s); if (i) return i; } } else if (t instanceof u || "text" in t) { const s = this.validateOption(t); if (s) return s; } else return new Error("Data object must be a valid optgroup or option"); return null; } // Validate Option validateOption(e) { return "text" in e ? null : new Error("Option must have a text"); } partialToFullData(e) { let t = []; return e.forEach((s) => { if (s) { if (s instanceof g || "label" in s) { let i = []; "options" in s && s.options && s.options.forEach((l) => { i.push(new u(l)); }), i.length > 0 && t.push(new g(s)); } (s instanceof u || "text" in s) && t.push(new u(s)); } }), t; } setData(e, t = !1) { const s = this.partialToFullData(e); if (t) { const i = this.getSelectedOptions(), l = []; i.forEach((n) => { let a = !1; for (const o of s) { if (o instanceof u && o.id === n.id) { a = !0; break; } if (o instanceof g) { for (const h of o.options) if (h.id === n.id) { a = !0; break; } } } a || l.push(n); }), this.data = [...l, ...s]; } else this.data = s; this.selectType === "single" && this.setSelectedBy("id", this.getSelected()); } // Get data will return all the data getData() { return this.filter(null, !0); } // Get data options will return the data as a // flat array of just options getDataOptions() { return this.filter(null, !1); } addOption(e, t = !1) { if (t) { let s = [new u(e)]; this.setData(s.concat(this.getData())); } else this.setData(this.getData().concat(new u(e))); } // Pass in an array of id that will loop through // each option and set the selected property to true // but also clean selected by determining selectType setSelectedBy(e, t) { let s = null, i = !1; const l = []; for (let a of this.data) { if (a instanceof g) for (let o of a.options) { s || (s = o); let h = o[e] || ""; o.selected = i ? !1 : t.includes(h), o.selected && (l.push(o), this.selectType === "single" && (i = !0)); } a instanceof u && (s || (s = a), a.selected = i ? !1 : t.includes(a[e]), a.selected && (l.push(a), this.selectType === "single" && (i = !0))); } this.selectType === "single" && s && !i && (s.selected = !0, l.push(s)); const n = t.map((a) => l.find((o) => o[e] === a)?.id || ""); this.selectedOrder = n; } getSelected() { return this.getSelectedOptions().map((e) => e.id); } getSelectedValues() { return this.getSelectedOptions().map((e) => e.value); } getSelectedOptions() { return this.filter((e) => e.selected, !1); } getOptgroupByID(e) { for (let t of this.data) if (t instanceof g && t.id === e) return t; return null; } getOptionByID(e) { let t = this.filter((s) => s.id === e, !1); return t.length ? t[0] : null; } getSelectType() { return this.selectType; } getFirstOption() { let e = null; for (let t of this.data) if (t instanceof g ? e = t.options[0] : t instanceof u && (e = t), e) break; return e; } // Take in search string and return filtered list of values search(e, t) { return e = e.trim(), e === "" ? this.getData() : this.filter((s) => t(s, e), !0); } // Filter takes in a function that will be used to filter the data // This will also keep optgroups of sub options meet the filter requirements filter(e, t) { const s = []; return this.data.forEach((i) => { if (i instanceof g) { let l = []; if (i.options.forEach((a) => { (!e || e(a)) && (t ? l.push(new u(a)) : s.push(new u(a))); }), l.length > 0) { let a = new g(i); a.options = l, s.push(a); } } i instanceof u && (!e || e(i)) && s.push(new u(i)); }), s; } // Take in an array of options and reoder them based upon the selected order selectedOrderOptions(e) { const t = []; return this.selectedOrder.forEach((s) => { const i = e.find((l) => l.id === s); i && t.push(i); }), e.forEach((s) => { let i = !1; t.forEach((l) => { if (s.id === l.id) { i = !0; return; } }), i || t.push(s); }), t; } } class N { settings; store; callbacks; // Used to compute the range selection lastSelectedOption; // Timeout tracking for cleanup closeAnimationTimeout = null; // Elements main; content; // Classes classes; constructor(e, t, s, i) { this.store = s, this.settings = e, this.classes = t, this.callbacks = i, this.lastSelectedOption = null, this.main = this.mainDiv(), this.content = this.contentDiv(), this.updateClassStyles(), this.updateAriaAttributes(), this.settings.contentPosition !== "relative" && (this.content.main.style.top = "-9999px", this.content.main.style.left = "-9999px", this.content.main.style.margin = "0", this.content.main.style.width = "auto"), this.settings.contentLocation && this.settings.contentLocation.appendChild(this.content.main); } // Helper method to add classes that may contain spaces // Splits by spaces and adds each class individually to avoid DOMException addClasses(e, t) { if (!t || t.trim() === "") return; const s = t.split(" ").filter((i) => i.trim() !== ""); for (const i of s) e.classList.add(i.trim()); } // Helper method to remove classes that may contain spaces removeClasses(e, t) { if (!t || t.trim() === "") return; const s = t.split(" ").filter((i) => i.trim() !== ""); for (const i of s) e.classList.remove(i.trim()); } // Remove disabled classes enable() { this.removeClasses(this.main.main, this.classes.disabled), this.main.main.setAttribute("aria-disabled", "false"), this.content.search.input.disabled = !1; } // Set disabled classes disable() { this.addClasses(this.main.main, this.classes.disabled), this.main.main.setAttribute("aria-disabled", "true"), this.content.search.input.disabled = !0; } open() { this.main.arrow.path.setAttribute("d", this.classes.arrowOpen), this.main.main.setAttribute("aria-expanded", "true"), this.closeAnimationTimeout && (clearTimeout(this.closeAnimationTimeout), this.closeAnimationTimeout = null); const t = this.settings.openPosition === "up" ? this.classes.dirAbove : this.classes.dirBelow; this.addClasses(this.main.main, t), this.addClasses(this.content.main, t), this.addClasses(this.content.main, this.classes.contentOpen), this.content.search.input.removeAttribute("aria-hidden"), this.moveContent(); const s = this.store.getSelectedOptions(); if (s.length) { const i = s[s.length - 1].id, l = this.content.list.querySelector('[data-id="' + i + '"]'); l && this.ensureElementInView(this.content.list, l); } } close() { this.main.main.setAttribute("aria-expanded", "false"), this.main.arrow.path.setAttribute("d", this.classes.arrowClose), this.removeClasses(this.content.main, this.classes.contentOpen), this.content.search.input.setAttribute("aria-hidden", "true"), this.main.main.removeAttribute("aria-activedescendant"); const e = this.getAnimationTiming(); this.closeAnimationTimeout = setTimeout(() => { this.removeClasses(this.main.main, this.classes.dirAbove), this.removeClasses(this.main.main, this.classes.dirBelow), this.removeClasses(this.content.main, this.classes.dirAbove), this.removeClasses(this.content.main, this.classes.dirBelow), this.closeAnimationTimeout = null; }, e); } getAnimationTiming() { const t = getComputedStyle(this.content.main).getPropertyValue("--ss-animation-timing").trim(); if (t) { if (t.endsWith("ms")) return parseFloat(t); if (t.endsWith("s")) return parseFloat(t) * 1e3; } return 200; } updateClassStyles() { if (this.main.main.className = "", this.main.main.removeAttribute("style"), this.content.main.className = "", this.content.main.removeAttribute("style"), this.addClasses(this.main.main, this.classes.main), this.addClasses(this.content.main, this.classes.content), this.settings.style !== "" && (this.main.main.style.cssText = this.settings.style, this.content.main.style.cssText = this.settings.style), this.settings.class.length) for (const e of this.settings.class) e.trim() !== "" && (this.main.main.classList.add(e.trim()), this.content.main.classList.add(e.trim())); (this.settings.contentPosition === "relative" || this.settings.contentPosition === "fixed") && this.content.main.classList.add("ss-" + this.settings.contentPosition); } updateAriaAttributes() { const e = this.content.list.id; this.main.main.role = "combobox", this.main.main.setAttribute("aria-haspopup", "listbox"), this.main.main.setAttribute("aria-controls", e), this.main.main.setAttribute("aria-expanded", "false"), this.content.list.setAttribute("role", "listbox"), this.content.list.setAttribute("aria-label", this.settings.ariaLabel + " listbox"), this.settings.isMultiple && this.content.list.setAttribute("aria-multiselectable", "true"), this.content.search.input.setAttribute("aria-controls", e); } mainDiv() { const e = document.createElement("div"); e.dataset.id = this.settings.id, e.setAttribute("aria-label", this.settings.ariaLabel), e.tabIndex = 0, e.onkeydown = (h) => { switch (h.key) { case "ArrowUp": case "ArrowDown": return this.callbacks.open(), h.key === "ArrowDown" ? this.highlight("down") : this.highlight("up"), !1; case "Tab": return this.callbacks.close(), !0; // Continue doing normal tabbing case "Enter": case " ": this.callbacks.open(); const r = this.content.list.querySelector( "." + this.classes.getFirst("highlighted") ); return r && r.click(), !1; case "Escape": return this.callbacks.close(), !1; } return h.key.length === 1 && this.callbacks.open(), !0; }, e.onclick = (h) => { this.settings.disabled || (this.settings.isOpen ? this.callbacks.close() : this.callbacks.open()); }; const t = document.createElement("div"); this.addClasses(t, this.classes.values), e.appendChild(t); const s = document.createElement("div"); this.addClasses(s, this.classes.deselect); const i = this.store?.getSelectedOptions(); !this.settings.allowDeselect || this.settings.isMultiple && i && i.length <= 0 ? this.addClasses(s, this.classes.hide) : this.removeClasses(s, this.classes.hide), s.onclick = (h) => { if (h.stopPropagation(), this.settings.disabled) return; let r = !0; const c = this.store.getSelectedOptions(), f = []; if (this.callbacks.beforeChange && (r = this.callbacks.beforeChange(f, c) === !0), r) { if (this.settings.isMultiple) this.callbacks.setSelected([], !1), this.updateDeselectAll(); else { const d = this.store.getFirstOption(), m = d ? d.id : ""; this.callbacks.setSelected(m, !1); } this.settings.closeOnSelect && this.callbacks.close(), this.callbacks.afterChange && this.callbacks.afterChange(this.store.getSelectedOptions()); } }; const l = document.createElementNS("http://www.w3.org/2000/svg", "svg"); l.setAttribute("viewBox", "0 0 100 100"); const n = document.createElementNS("http://www.w3.org/2000/svg", "path"); n.setAttribute("d", this.classes.deselectPath), l.appendChild(n), s.appendChild(l), e.appendChild(s); const a = document.createElementNS("http://www.w3.org/2000/svg", "svg"); this.addClasses(a, this.classes.arrow), a.setAttribute("viewBox", "0 0 100 100"); const o = document.createElementNS("http://www.w3.org/2000/svg", "path"); return o.setAttribute("d", this.classes.arrowClose), this.settings.alwaysOpen && this.addClasses(a, this.classes.hide), a.appendChild(o), e.appendChild(a), { main: e, values: t, deselect: { main: s, svg: l, path: n }, arrow: { main: a, path: o } }; } mainFocus(e) { e !== "click" && this.main.main.focus({ preventScroll: !0 }); } placeholder() { const e = this.store.filter((i) => i.placeholder, !1); let t = this.settings.placeholderText; e.length && (e[0].html !== "" ? t = e[0].html : e[0].text !== "" && (t = e[0].text)); const s = document.createElement("div"); return this.addClasses(s, this.classes.placeholder), s.innerHTML = t, s; } // Get selected values and append to multiSelected values container // and remove those who shouldnt exist renderValues() { if (!this.settings.isMultiple) { this.renderSingleValue(); return; } this.renderMultipleValues(), this.updateDeselectAll(); } renderSingleValue() { const e = this.store.filter((s) => s.selected && !s.placeholder, !1), t = e.length > 0 ? e[0] : null; if (!t) this.main.values.innerHTML = this.placeholder().outerHTML; else { const s = document.createElement("div"); this.addClasses(s, this.classes.single), t.html ? s.innerHTML = t.html : s.innerText = t.text, this.main.values.innerHTML = s.outerHTML; } !this.settings.allowDeselect || !e.length ? this.addClasses(this.main.deselect.main, this.classes.hide) : this.removeClasses(this.main.deselect.main, this.classes.hide); } renderMultipleValues() { let e = this.main.values.childNodes, t = this.store.filter((i) => i.selected && i.display, !1); if (t.length === 0) { this.main.values.innerHTML = this.placeholder().outerHTML; return; } else { const i = this.main.values.querySelector("." + this.classes.getFirst("placeholder")); i && i.remove(); } if (t.length > this.settings.maxValuesShown) { const i = document.createElement("div"); this.addClasses(i, this.classes.max), i.textContent = this.settings.maxValuesMessage.replace("{number}", t.length.toString()), this.main.values.innerHTML = i.outerHTML; return; } else { const i = this.main.values.querySelector("." + this.classes.getFirst("max")); i && i.remove(); } this.settings.keepOrder && (t = this.store.selectedOrderOptions(t)); let s = []; for (let i = 0; i < e.length; i++) { const l = e[i], n = l.getAttribute("data-id"); n && (t.filter((o) => o.id === n, !1).length || s.push(l)); } for (const i of s) this.addClasses(i, this.classes.valueOut), setTimeout(() => { this.main.values.hasChildNodes() && this.main.values.contains(i) && this.main.values.removeChild(i); }, 100); e = this.main.values.childNodes; for (let i = 0; i < t.length; i++) { let l = !0; for (let n = 0; n < e.length; n++) t[i].id === String(e[n].dataset.id) && (l = !1); l && (this.settings.keepOrder ? this.main.values.appendChild(this.multipleValue(t[i])) : e.length === 0 ? this.main.values.appendChild(this.multipleValue(t[i])) : i === 0 ? this.main.values.insertBefore(this.multipleValue(t[i]), e[i]) : e[i - 1].insertAdjacentElement("afterend", this.multipleValue(t[i]))); } } multipleValue(e) { const t = document.createElement("div"); this.addClasses(t, this.classes.value), t.dataset.id = e.id; const s = document.createElement("div"); if (this.addClasses(s, this.classes.valueText), s.textContent = e.text, t.appendChild(s), !e.mandatory) { const i = document.createElement("div"); this.addClasses(i, this.classes.valueDelete), i.setAttribute("tabindex", "0"), i.onclick = (a) => { if (a.preventDefault(), a.stopPropagation(), this.settings.disabled) return; let o = !0; const h = this.store.getSelectedOptions(), r = h.filter((c) => c.selected && c.id !== e.id, !0); if (!(this.settings.minSelected && r.length < this.settings.minSelected) && (this.callbacks.beforeChange && (o = this.callbacks.beforeChange(r, h) === !0), o)) { let c = []; for (const f of r) { if (f instanceof g) for (const d of f.options) d.id && c.push(d.id); f instanceof u && c.push(f.id); } this.callbacks.setSelected(c, !1), this.settings.closeOnSelect && this.callbacks.close(), this.callbacks.afterChange && this.callbacks.afterChange(r), this.updateDeselectAll(); } }; const l = document.createElementNS("http://www.w3.org/2000/svg", "svg"); l.setAttribute("viewBox", "0 0 100 100"); const n = document.createElementNS("http://www.w3.org/2000/svg", "path"); n.setAttribute("d", this.classes.optionDelete), l.appendChild(n), i.appendChild(l), t.appendChild(i), i.onkeydown = (a) => { a.key === "Enter" && i.click(); }; } return t; } contentDiv() { const e = document.createElement("div"); e.dataset.id = this.settings.id; const t = this.searchDiv(); e.appendChild(t.main); const s = this.listDiv(); return e.appendChild(s), { main: e, search: t, list: s }; } moveContent() { if (this.settings.contentPosition === "relative") { this.moveContentBelow(); return; } if (this.settings.openPosition === "down") { this.moveContentBelow(); return; } else if (this.settings.openPosition === "up") { this.moveContentAbove(); return; } this.putContent() === "up" ? this.moveContentAbove() : this.moveContentBelow(); } searchDiv() { const e = document.createElement("div"), t = document.createElement("input"), s = document.createElement("div"); this.addClasses(e, this.classes.search); const i = { main: e, input: t }; if (this.settings.showSearch || (this.addClasses(e, this.classes.hide), t.readOnly = !0), t.type = "search", t.placeholder = this.settings.searchPlaceholder, t.tabIndex = -1, t.setAttribute("aria-label", this.settings.searchPlaceholder), t.setAttribute("aria-autocomplete", "list"), t.setAttribute("autocapitalize", "off"), t.setAttribute("autocomplete", "off"), t.setAttribute("autocorrect", "off"), t.setAttribute("aria-hidden", "true"), t.oninput = S((l) => { this.callbacks.search(l.target.value); }, 100), t.onkeydown = (l) => { switch (l.key) { case "ArrowUp": case "ArrowDown": return l.key === "ArrowDown" ? this.highlight("down") : this.highlight("up"), !1; case "Tab": return this.callbacks.close(), !0; // Continue doing normal tabbing case "Escape": return this.callbacks.close(), !1; case " ": const n = this.content.list.querySelector( "." + this.classes.getFirst("highlighted") ); return n ? (n.click(), !1) : !0; case "Enter": const a = this.content.list.querySelector( "." + this.classes.getFirst("highlighted") ); return a ? (a.click(), !1) : this.callbacks.addable ? (s.click(), !1) : !0; } return !0; }, e.appendChild(t), this.callbacks.addable) { this.addClasses(s, this.classes.addable); const l = document.createElementNS("http://www.w3.org/2000/svg", "svg"); l.setAttribute("viewBox", "0 0 100 100"); const n = document.createElementNS("http://www.w3.org/2000/svg", "path"); n.setAttribute("d", this.classes.addablePath), l.appendChild(n), s.appendChild(l), s.onclick = (a) => { if (a.preventDefault(), a.stopPropagation(), !this.callbacks.addable) return; const o = this.content.search.input.value.trim(); if (o === "") { this.content.search.input.focus(); return; } const h = (c) => { let f = new u(c); if (this.callbacks.addOption(f), this.settings.isMultiple) { let d = this.store.getSelected(); d.push(f.id), this.callbacks.setSelected(d, !0); } else this.callbacks.setSelected([f.id], !0); this.callbacks.search(""), this.settings.closeOnSelect && setTimeout(() => { this.callbacks.close(); }, 100); }, r = this.callbacks.addable(o); r === !1 || r === void 0 || r === null || (r instanceof Promise ? r.then((c) => { typeof c == "string" ? h({ text: c, value: c }) : r instanceof Error ? this.renderError(r.message) : h(c); }) : typeof r == "string" ? h({ text: r, value: r }) : r instanceof Error ? this.renderError(r.message) : h(r)); }, e.appendChild(s), i.addable = { main: s, svg: l, path: n }; } return i; } searchFocus() { this.content.search.input.focus({ preventScroll: !0 }); } getOptions(e = !1, t = !1, s = !1) { let i = "." + this.classes.getFirst("option"); return e && (i += ":not(." + this.classes.getFirst("placeholder") + ")"), t && (i += ":not(." + this.classes.getFirst("disabled") + ")"), s && (i += ":not(." + this.classes.getFirst("hide") + ")"), Array.from(this.content.list.querySelectorAll(i)); } // highlightUp is used to highlight the previous option in the list highlight(e) { const t = this.getOptions(!0, !0, !0); if (t.length === 0) return; if (t.length === 1 && !t[0].classList.contains(this.classes.getFirst("highlighted"))) { this.addClasses(t[0], this.classes.highlighted); return; } let s = !1; for (const l of t) l.classList.contains(this.classes.getFirst("highlighted")) && (s = !0); if (!s) { for (const l of t) if (l.classList.contains(this.classes.getFirst("selected"))) { this.addClasses(l, this.classes.highlighted); break; } } for (let l = 0; l < t.length; l++) if (t[l].classList.contains(this.classes.getFirst("highlighted"))) { const n = t[l]; this.removeClasses(n, this.classes.highlighted); const a = n.parentElement; if (a && a.classList.contains(this.classes.getFirst("mainOpen"))) { const r = a.querySelector("." + this.classes.getFirst("optgroupLabel")); r && r.click(); } let o = t[e === "down" ? l + 1 < t.length ? l + 1 : 0 : l - 1 >= 0 ? l - 1 : t.length - 1]; this.addClasses(o, this.classes.highlighted), this.ensureElementInView(this.content.list, o), o.id && this.main.main.setAttribute("aria-activedescendant", o.id); const h = o.parentElement; if (h && h.classList.contains(this.classes.getFirst("close"))) { const r = h.querySelector( "." + this.classes.getFirst("optgroupLabel") ); r && r.click(); } return; } const i = t[e === "down" ? 0 : t.length - 1]; this.addClasses(i, this.classes.highlighted), i.id && this.main.main.setAttribute("aria-activedescendant", i.id), this.ensureElementInView(this.content.list, i); } // Create main container that options will reside listDiv() { const e = document.createElement("div"); this.addClasses(e, this.classes.list); const t = this.settings.id + "-list"; return e.id = t, e.dataset.id = t, e; } renderError(e) { this.content.list.innerHTML = ""; const t = document.createElement("div"); this.addClasses(t, this.classes.error), t.textContent = e, this.content.list.appendChild(t); } renderSearching() { this.content.list.innerHTML = ""; const e = document.createElement("div"); this.addClasses(e, this.classes.searching), e.textContent = this.settings.searchingText, this.content.list.appendChild(e); } // Take in data and add options to renderOptions(e) { if (this.content.list.innerHTML = "", e.length === 0) { const s = document.createElement("div"); this.addClasses(s, this.classes.search), this.callbacks.addable ? s.innerHTML = this.settings.addableText.replace("{value}", this.content.search.input.value) : s.innerHTML = this.settings.searchText, this.content.list.appendChild(s); return; } this.settings.allowDeselect && !this.settings.isMultiple && (this.store.filter((i) => i.placeholder, !1).length || this.store.addOption( new u({ text: "", value: "", selected: !1, placeholder: !0 }), !0 )); const t = document.createDocumentFragment(); for (const s of e) { if (s instanceof g) { const i = document.createElement("div"); this.addClasses(i, this.classes.optgroup); const l = document.createElement("div"); this.addClasses(l, this.classes.optgroupLabel), i.appendChild(l); const n = document.createElement("div"); this.addClasses(n, this.classes.optgroupLabelText), n.textContent = s.label, l.appendChild(n); const a = document.createElement("div"); if (this.addClasses(a, this.classes.optgroupActions), l.appendChild(a), this.settings.isMultiple && s.selectAll) { const o = document.createElement("div"); this.addClasses(o, this.classes.optgroupSelectAll); let h = !0; for (const m of s.options) if (!m.selected) { h = !1; break; } h && this.addClasses(o, this.classes.selected); const r = document.createElement("span"); r.textContent = s.selectAllText, o.appendChild(r); const c = document.createElementNS("http://www.w3.org/2000/svg", "svg"); c.setAttribute("viewBox", "0 0 100 100"), o.appendChild(c); const f = document.createElementNS("http://www.w3.org/2000/svg", "path"); f.setAttribute("d", this.classes.optgroupSelectAllBox), c.appendChild(f); const d = document.createElementNS("http://www.w3.org/2000/svg", "path"); d.setAttribute("d", this.classes.optgroupSelectAllCheck), c.appendChild(d), o.addEventListener("click", (m) => { m.preventDefault(), m.stopPropagation(); const v = this.store.getSelected(); if (h) { const C = v.filter((w) => { for (const b of s.options) if (w === b.id) return !1; return !0; }); this.callbacks.setSelected(C, !0); return; } else { let C = s.options.map((b) => b.id).filter((b) => b !== void 0); const w = v.concat(C); for (const b of s.options) b.id && !this.store.getOptionByID(b.id) && this.callbacks.addOption(new u(b)); this.callbacks.setSelected(w, !0); return; } }), a.appendChild(o); } if (s.closable !== "off") { const o = document.createElement("div"); this.addClasses(o, this.classes.optgroupClosable); const h = document.createElementNS("http://www.w3.org/2000/svg", "svg"); h.setAttribute("viewBox", "0 0 100 100"), this.addClasses(h, this.classes.arrow), o.appendChild(h); const r = document.createElementNS("http://www.w3.org/2000/svg", "path"); h.appendChild(r), s.options.some((c) => c.selected) || this.content.search.input.value.trim() !== "" ? (this.addClasses(o, this.classes.mainOpen), r.setAttribute("d", this.classes.arrowOpen)) : s.closable === "open" ? (this.addClasses(i, this.classes.mainOpen), r.setAttribute("d", this.classes.arrowOpen)) : s.closable === "close" && (this.addClasses(i, this.classes.close), r.setAttribute("d", this.classes.arrowClose)), l.addEventListener("click", (c) => { c.preventDefault(), c.stopPropagation(), i.classList.contains(this.classes.getFirst("close")) ? (this.removeClasses(i, this.classes.close), this.addClasses(i, this.classes.mainOpen), r.setAttribute("d", this.classes.arrowOpen)) : (this.removeClasses(i, this.classes.mainOpen), this.addClasses(i, this.classes.close), r.setAttribute("d", this.classes.arrowClose)); }), a.appendChild(o); } i.appendChild(l); for (const o of s.options) i.appendChild(this.option(new u(o))), t.appendChild(i); } s instanceof u && t.appendChild(this.option(s)); } this.content.list.appendChild(t); } // Create option div element option(e) { if (e.placeholder) { const s = document.createElement("div"); return this.addClasses(s, this.classes.option), this.addClasses(s, this.classes.hide), s; } const t = document.createElement("div"); return t.dataset.id = e.id, t.id = this.settings.id + "-" + e.id, this.addClasses(t, this.classes.option), t.setAttribute("role", "option"), e.class && e.class.split(" ").forEach((s) => { t.classList.add(s); }), e.style && (t.style.cssText = e.style), this.settings.searchHighlight && this.content.search.input.value.trim() !== "" ? t.innerHTML = this.highlightText( e.html !== "" ? e.html : e.text, this.content.search.input.value, this.classes.searchHighlighter ) : e.html !== "" ? t.innerHTML = e.html : t.textContent = e.text, this.settings.showOptionTooltips && t.textContent && t.setAttribute("title", t.textContent), e.display || this.addClasses(t, this.classes.hide), e.disabled && this.addClasses(t, this.classes.disabled), e.selected && this.settings.hideSelected && this.addClasses(t, this.classes.hide), e.selected ? (this.addClasses(t, this.classes.selected), t.setAttribute("aria-selected", "true"), this.main.main.setAttribute("aria-activedescendant", t.id)) : (this.removeClasses(t, this.classes.selected), t.setAttribute("aria-selected", "false")), t.addEventListener("click", (s) => { s.preventDefault(), s.stopPropagation(); const i = this.store.getSelected(), l = s.currentTarget, n = String(l.dataset.id), a = s.ctrlKey || s.metaKey; if (e.disabled || !this.settings.isMultiple && e.selected && !this.settings.allowDeselect || e.selected && e.mandatory || this.settings.isMultiple && this.settings.maxSelected <= i.length && !e.selected || this.settings.isMultiple && this.settings.minSelected >= i.length && e.selected && !a) return; let o = !1; const h = this.store.getSelectedOptions(); let r = []; if (this.settings.isMultiple) { const c = h.some((d) => d.id === n); if (s.shiftKey && this.lastSelectedOption) { const d = this.store.getDataOptions(), m = d.findIndex((C) => C.id === this.lastSelectedOption.id), v = d.findIndex((C) => C.id === e.id); if (m >= 0 && v >= 0) { const C = Math.min(m, v), w = Math.max(m, v), E = d.slice(C, w + 1).filter((x) => !h.find((T) => T.id === x.id)); h.length + E.length <= this.settings.maxSelected ? r = h.concat(E) : r = h; } else r = h; } else a ? (c ? r = h.filter((d) => d.id !== n) : r = h.concat(e), this.lastSelectedOption = e) : (c ? r = h.filter((d) => d.id !== n) : r = h.concat(e), this.lastSelectedOption = e); } if (this.settings.isMultiple || (e.selected ? r = [] : r = [e]), this.callbacks.beforeChange || (o = !0), this.callbacks.beforeChange && (this.callbacks.beforeChange(r, h) === !1 ? o = !1 : o = !0), o) { this.store.getOptionByID(n) || this.callbacks.addOption(e), this.callbacks.setSelected( r.map((d) => d.id), !1 ); const c = s.ctrlKey || s.metaKey || s.shiftKey; this.settings.closeOnSelect && !(this.settings.isMultiple && c) && this.callbacks.close(), this.callbacks.afterChange && this.callbacks.afterChange(r); } }), t; } destroy() { this.closeAnimationTimeout && (clearTimeout(this.closeAnimationTimeout), this.closeAnimationTimeout = null), this.main.main.remove(), this.content.main.remove(); } highlightText(e, t, s) { const i = t.trim(); if (i === "") return e; const l = i.replace(/[.*+?^${}()|[\]\\]/g, "\\$&"), n = document.createElement("div"); n.innerHTML = e; const a = (o) => { if (o.nodeType === Node.TEXT_NODE) { const h = o.textContent || "", r = new RegExp("(" + l + ")", "i"); if (r.test(h)) { const c = document.createElement("span"); h.split(r).forEach((d, m) => { if (d && r.test(d)) { const v = document.createElement("mark"); v.className = s, v.textContent = d, c.appendChild(v); } else d && c.appendChild(document.createTextNode(d)); }), o.parentNode?.replaceChild(c, o); } } else o.nodeType === Node.ELEMENT_NODE && Array.from(o.childNodes).forEach((h) => a(h)); }; return Array.from(n.childNodes).forEach((o) => a(o)), n.innerHTML; } setContentDirection(e) { const t = e === "above", s = t ? this.classes.dirAbove : this.classes.dirBelow, i = t ? this.classes.dirBelow : this.classes.dirAbove; if (this.removeClasses(this.main.main, i), this.addClasses(this.main.main, s), this.removeClasses(this.content.main, i), this.addClasses(this.content.main, s), t) { const l = this.main.main.offsetHeight, n = this.content.main.offsetHeight; this.content.main.style.margin = "-" + (l + n - 1) + "px 0px 0px 0px"; } else this.content.main.style.margin = "-1px 0px 0px 0px"; } setContentPosition() { if (this.settings.contentPosition === "relative") return; const e = this.main.main.getBoundingClientRect(); let t, s; if (this.settings.contentPosition === "fixed") t = e.top + e.height, s = e.left; else { const i = this.content.main.offsetParent, l = i ? i.getBoundingClientRect() : { top: 0, left: 0 }; t = e.top - l.top + e.height - (i?.clientTop || 0), s = e.left - l.left - (i?.clientLeft || 0); } this.content.main.style.top = t + "px", this.content.main.style.left = s + "px", this.content.main.style.width = e.width + "px"; } moveContentAbove() { this.setContentDirection("above"), this.setContentPosition(); } moveContentBelow() { this.setContentDirection("below"), this.setContentPosition(); } ensureElementInView(e, t) { const s = e.scrollTop + e.offsetTop, i = s + e.clientHeight, l = t.offsetTop, n = l + t.clientHeight; l < s ? e.scrollTop -= s - l : n > i && (e.scrollTop += n - i); } putContent() { const e = this.main.main.offsetHeight, t = this.main.main.getBoundingClientRect(), s = this.content.main.offsetHeight; return window.innerHeight - (t.top + e) <= s && t.top > s ? "up" : "down"; } // Updates deselect based on item count and allowDeselect setting updateDeselectAll() { if (!this.store || !this.settings) return; const e = this.store.getSelectedOptions(), t = e && e.length > 0, s = this.settings.isMultiple, i = this.settings.allowDeselect, l = this.main.deselect.main, n = this.classes.hide; i && !(s && !t) ? this.removeClasses(l, n) : this.addClasses(l, n); } } class P { select; // Mutation observer fields onValueChange; onClassChange; onDisabledChange; onOptionsChange; onLabelClick; // Change observers listen = !1; observer = null; isUpdating = !1; pendingOptionsChange = null; // Event handlers for preventing native select behavior (especially on iOS Safari) preventNativeSelect = null; preventNativeSelectMousedown = null; preventNativeSelectFocus = null; constructor(e) { this.select = e, this.valueChange = this.valueChange.bind(this), this.select.addEventListener("change", this.valueChange, { // allow bubbling of event passive: !0 }), this.observer = new MutationObserver(this.observeCall.bind(this)), this.changeListen(!0); } enable() { this.select.disabled = !1; } disable() { this.select.disabled = !0; } hideUI() { this.select.tabIndex = -1, this.select.style.position = "absolute", this.select.style.width = "1px", this.select.style.height = "1px", this.select.style.opacity = "0", this.select.style.overflow = "hidden", this.select.style.pointerEvents = "none", this.select.style.margin = "0", this.select.style.padding = "0", this.select.style.borderWidth = "0", this.select.style.clip = "rect(0 0 0 0)", this.select.setAttribute("aria-hidden", "true"), this.preventNativeSelect || (this.preventNativeSelect = (e) => { e.preventDefault(), e.stopPropagation(), e.stopImmediatePropagation(); }, this.preventNativeSelectMousedown = (e) => { e.preventDefault(), e.stopPropagation(), e.stopImmediatePropagation(); }, this.preventNativeSelectFocus = (e) => { e.preventDefault(), e.stopPropagation(), e.stopImmediatePropagation(); }, this.select.addEventListener("click", this.preventNativeSelect, { capture: !0, passive: !1 }), this.select.addEventListener("mousedown", this.preventNativeSelectMousedown, { capture: !0, passive: !1 }), this.select.addEventListener("focus", this.preventNativeSelectFocus, { capture: !0, passive: !1 })); } showUI() { this.select.removeAttribute("tabindex"), this.select.style.position = "", this.select.style.width = "", this.select.style.height = "", this.select.style.opacity = "", this.select.style.overflow = "", this.select.style.pointerEvents = "", this.select.style.margin = "", this.select.style.padding = "", this.select.style.borderWidth = "", this.select.style.clip = "", this.select.removeAttribute("aria-hidden"), this.preventNativeSelect && (this.select.removeEventListener("click", this.preventNativeSelect, { capture: !0 }), this.preventNativeSelect = null), this.preventNativeSelectMousedown && (this.select.removeEventListener("mousedown", this.preventNativeSelectMousedown, { capture: !0 }), this.preventNativeSelectMousedown = null), this.preventNativeSelectFocus && (this.select.removeEventListener("focus", this.preventNativeSelectFocus, { capture: !0 }), this.preventNativeSelectFocus = null); } changeListen(e) { this.listen = e, e && this.observer && this.observer.observe(this.select, { subtree: !0, // subtree for optgroups options childList: !0, // children changes attributes: !0 // attributes changes }), e || this.observer && this.observer.disconnect(); } // This function get triggers when the select value changes // and will call the onValueChange function if it exists valueChange(e) { return this.listen && this.onValueChange && this.onValueChange(this.getSelectedOptions()), !0; } observeCall(e) { if (!this.listen) return; let t = !1, s = !1, i = !1, l = !1; for (const n of e) { if (n.target === this.select && (n.attributeName === "disabled" && (s = !0), n.attributeName === "class" && (t = !0), n.type === "childList")) { for (const a of Array.from(n.addedNodes)) if (a.nodeName === "OPTION" && a.value === this.select.value) { l = !0; break; } i = !0; } (n.target.nodeName === "OPTGROUP" || n.target.nodeName === "OPTION") && (i = !0); } if (t && this.onClassChange && this.onClassChange(this.select.className.split(" ")), s && this.onDisabledChange && (this.changeListen(!1), this.onDisabledChange(this.select.disabled), this.changeListen(!0)), i && this.onOptionsChange) { if (this.isUpdating) { if (this.select.options.length > 0) { const n = this.getData(); n.length > 0 && (this.pendingOptionsChange = n); } l && this.select.dispatchEvent(new Event("change")); return; } this.changeListen(!1), this.onOptionsChange(this.getData()), this.changeListen(!0); } l && this.select.dispatchEvent(new Event("change")); } // From the select element pull optgroup and options into data getData() { let e = []; const t = this.select.childNodes; for (const s of t) s.nodeName === "OPTGROUP" && e.push(this.getDataFromOptgroup(s)), s.nodeName === "OPTION" && e.push(this.getDataFromOption(s)); return e; } getDataFromOptgroup(e) { let t = { id: e.id, label: e.label, selectAll: e.dataset ? e.dataset.selectall === "true" : !1, selectAllText: e.dataset ? e.dataset.selectalltext : "Select all", closable: e.dataset ? e.dataset.closable : "off", options: [] }; const s = e.childNodes; for (const i of s) i.nodeName === "OPTION" && t.options.push(this.getDataFromOption(i)); return t; } // From passed in option pull pieces of usable information getDataFromOption(e) { return { id: e.id, value: e.value, text: e.text, html: e.dataset && e.dataset.html ? e.dataset.html : "", defaultSelected: e.defaultSelected, selected: e.selected, display: e.style.display !== "none", disabled: e.disabled, mandatory: e.dataset ? e.dataset.mandatory === "true" : !1, placeholder: e.dataset.placeholder === "true", class: e.className, style: e.style.cssText, data: e.dataset }; } getSelectedOptions() { let e = []; const t = this.select.childNodes; for (const s of t) { if (s.nodeName === "OPTGROUP") { const i = s.childNodes; for (const l of i) if (l.nodeName === "OPTION") { const n = l; n.selected && e.push(this.getDataFromOption(n)); } } if (s.nodeName === "OPTION") { const i = s; i.selected && e.push(this.getDataFromOption(i)); } } return e; } getSelectedValues() { return this.getSelectedOptions().map((e) => e.value); } setSelected(e) { this.changeListen(!1); const t = this.select.childNodes; for (const s of t) { if (s.nodeName === "OPTGROUP") { const l = s.childNodes; for (const n of l) if (n.nodeName === "OPTION") { const a = n; a.selected = e.includes(a.id); } } if (s.nodeName === "OPTION") { const i = s; i.selected = e.includes(i.id); } } this.changeListen(!0); } // Set selected options by value instead of id // This is useful when the id is not known // and only the value is known // but the value is not unique and can be duplicated setSelectedByValue(e) { this.changeListen(!1); const t = this.select.childNodes; for (const s of t) { if (s.nodeName === "OPTGROUP") { const l = s.childNodes; for (const n of l) if (n.nodeName === "OPTION") { const a = n; a.selected = e.includes(a.value); } } if (s.nodeName === "OPTION") { const i = s; i.selected = e.includes(i.value); } } this.changeListen(!0); } updateSelect(e, t, s) { this.changeListen(!1), e && (this.select.dataset.id = e), t && (this.select.style.cssText = t), s && (this.select.className = "", s.forEach((i) => { i.trim() !== "" && this.select.classList.add(i.trim()); })), this.changeListen(!0); } updateOptions(e) { if (!(!e || e.length === 0)) { this.isUpdating = !0, this.pendingOptionsChange = null, this.changeListen(!1), this.select.innerHTML = ""; for (const t of e) t instanceof g && this.select.appendChild(this.createOptgroup(t)), t instanceof u && this.select.appendChild(this.createOption(t)); if (this.select.dispatchEvent(new Event("change", { bubbles: !0 })), this.changeListen(!0), this.isUpdating = !1, this.pendingOptionsChange !== null) { const t = this.pendingOptionsChange; t.length > 0 && this.onOptionsChange ? (this.pendingOptionsChange = null, this.changeListen(!1), this.onOptionsChange(t), this.changeListen(!0)) : this.pendingOptionsChange = null; } } } createOptgrou