2023-10-10 14:38:59 -04:00
|
|
|
import Component from "@ember/component";
|
2020-05-23 00:56:13 -04:00
|
|
|
import EmberObject, { computed, get } from "@ember/object";
|
2023-10-10 14:38:59 -04:00
|
|
|
import { guidFor } from "@ember/object/internals";
|
|
|
|
import Mixin from "@ember/object/mixin";
|
2020-12-18 08:18:52 -05:00
|
|
|
import { bind, cancel, next, schedule, throttle } from "@ember/runloop";
|
2024-03-25 17:58:11 -04:00
|
|
|
import { service } from "@ember/service";
|
2020-02-18 17:41:15 -05:00
|
|
|
import { isEmpty, isNone, isPresent } from "@ember/utils";
|
2020-09-14 12:33:14 -04:00
|
|
|
import { createPopper } from "@popperjs/core";
|
2023-10-10 14:38:59 -04:00
|
|
|
import { Promise } from "rsvp";
|
|
|
|
import { INPUT_DELAY } from "discourse-common/config/environment";
|
2020-12-18 08:18:52 -05:00
|
|
|
import discourseDebounce from "discourse-common/lib/debounce";
|
2023-10-10 14:38:59 -04:00
|
|
|
import deprecated from "discourse-common/lib/deprecated";
|
2017-11-22 04:34:12 -05:00
|
|
|
import { makeArray } from "discourse-common/lib/helpers";
|
2024-03-25 17:58:11 -04:00
|
|
|
import { bind as bindDecorator } from "discourse-common/utils/decorators";
|
2023-10-18 06:07:09 -04:00
|
|
|
import I18n from "discourse-i18n";
|
2023-10-10 14:38:59 -04:00
|
|
|
import PluginApiMixin, {
|
|
|
|
applyContentPluginApiCallbacks,
|
|
|
|
applyOnChangePluginApiCallbacks,
|
|
|
|
} from "select-kit/mixins/plugin-api";
|
|
|
|
import UtilsMixin from "select-kit/mixins/utils";
|
2017-11-21 05:53:09 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
export const MAIN_COLLECTION = "MAIN_COLLECTION";
|
|
|
|
export const ERRORS_COLLECTION = "ERRORS_COLLECTION";
|
|
|
|
|
|
|
|
const EMPTY_OBJECT = Object.freeze({});
|
2020-02-05 11:14:42 -05:00
|
|
|
const SELECT_KIT_OPTIONS = Mixin.create({
|
2022-07-12 15:38:17 -04:00
|
|
|
concatenatedProperties: ["selectKitOptions"],
|
2020-02-03 08:22:14 -05:00
|
|
|
selectKitOptions: EMPTY_OBJECT,
|
|
|
|
});
|
|
|
|
|
2020-10-22 10:08:11 -04:00
|
|
|
function isDocumentRTL() {
|
2021-11-19 05:17:23 -05:00
|
|
|
return document.documentElement.classList.contains("rtl");
|
2020-10-22 10:08:11 -04:00
|
|
|
}
|
|
|
|
|
2019-10-23 12:30:52 -04:00
|
|
|
export default Component.extend(
|
2020-02-03 08:22:14 -05:00
|
|
|
SELECT_KIT_OPTIONS,
|
2017-11-21 05:53:09 -05:00
|
|
|
PluginApiMixin,
|
2020-02-03 08:22:14 -05:00
|
|
|
UtilsMixin,
|
2017-11-21 05:53:09 -05:00
|
|
|
{
|
2021-08-23 04:44:19 -04:00
|
|
|
tagName: "details",
|
2017-11-21 05:53:09 -05:00
|
|
|
pluginApiIdentifiers: ["select-kit"],
|
2018-02-05 12:25:39 -05:00
|
|
|
classNames: ["select-kit"],
|
2017-11-21 05:53:09 -05:00
|
|
|
classNameBindings: [
|
2020-02-03 08:22:14 -05:00
|
|
|
"selectKit.isLoading:is-loading",
|
|
|
|
"selectKit.isExpanded:is-expanded",
|
2020-10-13 05:31:07 -04:00
|
|
|
"selectKit.options.disabled:is-disabled",
|
2020-02-03 08:22:14 -05:00
|
|
|
"selectKit.isHidden:is-hidden",
|
2020-02-05 10:01:58 -05:00
|
|
|
"selectKit.hasSelection:has-selection",
|
2017-11-21 05:53:09 -05:00
|
|
|
],
|
2018-06-15 11:03:24 -04:00
|
|
|
tabindex: 0,
|
2017-11-21 05:53:09 -05:00
|
|
|
content: null,
|
2020-02-03 08:22:14 -05:00
|
|
|
value: null,
|
|
|
|
selectKit: null,
|
|
|
|
mainCollection: null,
|
|
|
|
errorsCollection: null,
|
|
|
|
options: null,
|
|
|
|
valueProperty: "id",
|
|
|
|
nameProperty: "name",
|
2020-02-11 09:54:56 -05:00
|
|
|
singleSelect: false,
|
|
|
|
multiSelect: false,
|
2020-05-28 02:30:31 -04:00
|
|
|
labelProperty: null,
|
|
|
|
titleProperty: null,
|
2021-01-18 09:34:18 -05:00
|
|
|
langProperty: null,
|
2024-03-25 17:58:11 -04:00
|
|
|
appEvents: service(),
|
2018-06-15 11:03:24 -04:00
|
|
|
|
|
|
|
init() {
|
2019-01-10 05:06:01 -05:00
|
|
|
this._super(...arguments);
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
this._searchPromise = null;
|
|
|
|
|
|
|
|
this.set("errorsCollection", []);
|
|
|
|
this._collections = [ERRORS_COLLECTION, MAIN_COLLECTION];
|
|
|
|
|
|
|
|
!this.options && this.set("options", EmberObject.create({}));
|
|
|
|
|
|
|
|
this.handleDeprecations();
|
|
|
|
|
2018-09-12 06:19:04 -04:00
|
|
|
this.set(
|
2020-02-03 08:22:14 -05:00
|
|
|
"selectKit",
|
2019-10-29 15:23:50 -04:00
|
|
|
EmberObject.create({
|
2023-11-27 06:16:31 -05:00
|
|
|
uniqueID: this.id || guidFor(this),
|
2020-02-03 08:22:14 -05:00
|
|
|
valueProperty: this.valueProperty,
|
|
|
|
nameProperty: this.nameProperty,
|
2020-05-28 02:30:31 -04:00
|
|
|
labelProperty: this.labelProperty,
|
|
|
|
titleProperty: this.titleProperty,
|
2021-01-18 09:34:18 -05:00
|
|
|
langProperty: this.langProperty,
|
2020-02-03 08:22:14 -05:00
|
|
|
options: EmberObject.create(),
|
|
|
|
|
|
|
|
isLoading: false,
|
|
|
|
isHidden: false,
|
|
|
|
isExpanded: false,
|
|
|
|
isFilterExpanded: false,
|
2021-02-01 05:07:11 -05:00
|
|
|
enterDisabled: false,
|
2020-02-03 08:22:14 -05:00
|
|
|
hasSelection: false,
|
|
|
|
hasNoContent: true,
|
|
|
|
highlighted: null,
|
|
|
|
noneItem: null,
|
2020-02-07 08:12:17 -05:00
|
|
|
newItem: null,
|
2020-02-03 08:22:14 -05:00
|
|
|
filter: null,
|
|
|
|
|
|
|
|
modifyContent: bind(this, this._modifyContentWrapper),
|
|
|
|
modifySelection: bind(this, this._modifySelectionWrapper),
|
|
|
|
modifyComponentForRow: bind(this, this._modifyComponentForRowWrapper),
|
|
|
|
modifyContentForCollection: bind(
|
|
|
|
this,
|
|
|
|
this._modifyContentForCollectionWrapper
|
|
|
|
),
|
|
|
|
modifyComponentForCollection: bind(
|
|
|
|
this,
|
|
|
|
this._modifyComponentForCollectionWrapper
|
|
|
|
),
|
|
|
|
|
|
|
|
toggle: bind(this, this._toggle),
|
|
|
|
close: bind(this, this._close),
|
|
|
|
open: bind(this, this._open),
|
|
|
|
highlightNext: bind(this, this._highlightNext),
|
|
|
|
highlightPrevious: bind(this, this._highlightPrevious),
|
2021-08-23 04:44:19 -04:00
|
|
|
highlightLast: bind(this, this._highlightLast),
|
|
|
|
highlightFirst: bind(this, this._highlightFirst),
|
2023-07-25 11:00:02 -04:00
|
|
|
deselectLast: bind(this, this._deselectLast),
|
2020-02-03 08:22:14 -05:00
|
|
|
change: bind(this, this._onChangeWrapper),
|
|
|
|
select: bind(this, this.select),
|
|
|
|
deselect: bind(this, this.deselect),
|
2021-02-22 08:34:31 -05:00
|
|
|
deselectByValue: bind(this, this.deselectByValue),
|
2021-01-06 06:57:13 -05:00
|
|
|
append: bind(this, this.append),
|
2021-08-23 04:44:19 -04:00
|
|
|
cancelSearch: bind(this, this._cancelSearch),
|
|
|
|
triggerSearch: bind(this, this.triggerSearch),
|
|
|
|
focusFilter: bind(this, this._focusFilter),
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
onOpen: bind(this, this._onOpenWrapper),
|
|
|
|
onClose: bind(this, this._onCloseWrapper),
|
|
|
|
onInput: bind(this, this._onInput),
|
|
|
|
onClearSelection: bind(this, this._onClearSelection),
|
|
|
|
onHover: bind(this, this._onHover),
|
|
|
|
onKeydown: bind(this, this._onKeydownWrapper),
|
2021-08-23 04:44:19 -04:00
|
|
|
|
|
|
|
mainElement: bind(this, this._mainElement),
|
|
|
|
headerElement: bind(this, this._headerElement),
|
|
|
|
bodyElement: bind(this, this._bodyElement),
|
2018-09-12 06:19:04 -04:00
|
|
|
})
|
|
|
|
);
|
2020-02-03 08:22:14 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
_modifyComponentForRowWrapper(collection, item) {
|
|
|
|
let component = this.modifyComponentForRow(collection, item);
|
|
|
|
return component || "select-kit/select-kit-row";
|
|
|
|
},
|
|
|
|
|
|
|
|
modifyComponentForRow() {},
|
|
|
|
|
|
|
|
_modifyContentForCollectionWrapper(identifier) {
|
|
|
|
let collection = this.modifyContentForCollection(identifier);
|
|
|
|
|
|
|
|
if (!collection) {
|
|
|
|
switch (identifier) {
|
|
|
|
case ERRORS_COLLECTION:
|
|
|
|
collection = this.errorsCollection;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
collection = this.mainCollection;
|
|
|
|
break;
|
|
|
|
}
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
return collection;
|
|
|
|
},
|
|
|
|
|
|
|
|
modifyContentForCollection() {},
|
|
|
|
|
|
|
|
_modifyComponentForCollectionWrapper(identifier) {
|
|
|
|
let component = this.modifyComponentForCollection(identifier);
|
|
|
|
|
|
|
|
if (!component) {
|
|
|
|
switch (identifier) {
|
|
|
|
case ERRORS_COLLECTION:
|
|
|
|
component = "select-kit/errors-collection";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
component = "select-kit/select-kit-collection";
|
|
|
|
break;
|
|
|
|
}
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
return component;
|
|
|
|
},
|
|
|
|
|
|
|
|
modifyComponentForCollection() {},
|
|
|
|
|
|
|
|
didUpdateAttrs() {
|
|
|
|
this._super(...arguments);
|
|
|
|
|
|
|
|
this.handleDeprecations();
|
|
|
|
},
|
|
|
|
|
2023-01-13 10:13:13 -05:00
|
|
|
didInsertElement() {
|
|
|
|
this._super(...arguments);
|
|
|
|
|
2024-03-25 17:58:11 -04:00
|
|
|
this.appEvents.on("keyboard-visibility-change", this, this._updatePopper);
|
|
|
|
|
2023-01-13 10:13:13 -05:00
|
|
|
if (this.selectKit.options.expandedOnInsert) {
|
2024-04-16 07:58:06 -04:00
|
|
|
next(() => {
|
|
|
|
this._open();
|
|
|
|
});
|
2023-01-13 10:13:13 -05:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2021-09-03 12:29:16 -04:00
|
|
|
click(event) {
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
2021-08-23 04:44:19 -04:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
willDestroyElement() {
|
|
|
|
this._super(...arguments);
|
|
|
|
|
2021-08-23 04:44:19 -04:00
|
|
|
this._cancelSearch();
|
2020-02-03 08:22:14 -05:00
|
|
|
|
2024-03-25 17:58:11 -04:00
|
|
|
this.appEvents.off(
|
|
|
|
"keyboard-visibility-change",
|
|
|
|
this,
|
|
|
|
this._updatePopper
|
|
|
|
);
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
if (this.popper) {
|
|
|
|
this.popper.destroy();
|
|
|
|
this.popper = null;
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
2018-01-24 05:48:20 -05:00
|
|
|
},
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
didReceiveAttrs() {
|
|
|
|
this._super(...arguments);
|
|
|
|
|
2023-03-09 08:44:31 -05:00
|
|
|
const deprecatedOptions = this._resolveDeprecatedOptions();
|
2022-07-12 15:38:17 -04:00
|
|
|
const mergedOptions = Object.assign({}, ...this.selectKitOptions);
|
|
|
|
Object.keys(mergedOptions).forEach((key) => {
|
2021-11-12 08:04:48 -05:00
|
|
|
if (isPresent(this.options[key])) {
|
|
|
|
this.selectKit.options.set(key, this.options[key]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-03-09 08:44:31 -05:00
|
|
|
if (isPresent(deprecatedOptions[`options.${key}`])) {
|
|
|
|
this.selectKit.options.set(key, deprecatedOptions[`options.${key}`]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-07-12 15:38:17 -04:00
|
|
|
const value = mergedOptions[key];
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
if (
|
|
|
|
key === "componentForRow" ||
|
|
|
|
key === "contentForCollection" ||
|
|
|
|
key === "componentForCollection"
|
|
|
|
) {
|
|
|
|
if (typeof value === "string") {
|
2021-11-12 08:04:48 -05:00
|
|
|
this.selectKit.options.set(key, () => value);
|
2020-02-03 08:22:14 -05:00
|
|
|
} else {
|
2021-11-12 08:04:48 -05:00
|
|
|
this.selectKit.options.set(key, bind(this, value));
|
2020-02-03 08:22:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
if (
|
|
|
|
typeof value === "string" &&
|
2022-07-17 14:48:36 -04:00
|
|
|
!value.includes(".") &&
|
2020-02-03 08:22:14 -05:00
|
|
|
value in this
|
|
|
|
) {
|
|
|
|
const computedValue = get(this, value);
|
|
|
|
if (typeof computedValue !== "function") {
|
2021-11-12 08:04:48 -05:00
|
|
|
this.selectKit.options.set(key, computedValue);
|
2020-02-03 08:22:14 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2021-11-12 08:04:48 -05:00
|
|
|
|
|
|
|
this.selectKit.options.set(key, value);
|
2020-02-03 08:22:14 -05:00
|
|
|
});
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
this.selectKit.setProperties({
|
2020-02-05 11:14:42 -05:00
|
|
|
hasSelection: !isEmpty(this.value),
|
2020-02-07 08:12:17 -05:00
|
|
|
noneItem: this._modifyNoSelectionWrapper(),
|
|
|
|
newItem: null,
|
2020-02-03 08:22:14 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
if (this.selectKit.isExpanded) {
|
2020-03-13 07:41:08 -04:00
|
|
|
this.triggerSearch();
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
if (this.computeContent) {
|
|
|
|
this._deprecated(
|
|
|
|
`The \`computeContent()\` function is deprecated pass a \`content\` attribute or define a \`content\` computed property in your component.`
|
|
|
|
);
|
|
|
|
|
|
|
|
this.set("content", this.computeContent());
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
selectKitOptions: {
|
2021-11-12 08:04:48 -05:00
|
|
|
allowAny: false,
|
2020-02-03 08:22:14 -05:00
|
|
|
showFullTitle: true,
|
|
|
|
none: null,
|
|
|
|
translatedNone: null,
|
|
|
|
filterable: false,
|
|
|
|
autoFilterable: "autoFilterable",
|
|
|
|
filterIcon: "search",
|
2021-08-23 04:44:19 -04:00
|
|
|
filterPlaceholder: null,
|
2020-04-22 14:39:55 -04:00
|
|
|
translatedFilterPlaceholder: null,
|
2020-02-03 08:22:14 -05:00
|
|
|
icon: null,
|
|
|
|
icons: null,
|
|
|
|
maximum: null,
|
2020-02-05 11:47:20 -05:00
|
|
|
maximumLabel: null,
|
2020-02-03 08:22:14 -05:00
|
|
|
minimum: null,
|
|
|
|
autoInsertNoneItem: true,
|
|
|
|
closeOnChange: true,
|
2023-07-25 11:00:02 -04:00
|
|
|
useHeaderFilter: false,
|
2020-02-03 08:22:14 -05:00
|
|
|
limitMatches: null,
|
2020-10-22 10:08:11 -04:00
|
|
|
placement: isDocumentRTL() ? "bottom-end" : "bottom-start",
|
2022-11-10 01:42:37 -05:00
|
|
|
verticalOffset: 3,
|
2020-02-03 08:22:14 -05:00
|
|
|
filterComponent: "select-kit/select-kit-filter",
|
2020-02-14 04:00:40 -05:00
|
|
|
selectedNameComponent: "selected-name",
|
2021-08-23 04:44:19 -04:00
|
|
|
selectedChoiceComponent: "selected-choice",
|
2020-04-21 11:11:40 -04:00
|
|
|
castInteger: false,
|
2020-05-23 00:56:13 -04:00
|
|
|
focusAfterOnChange: true,
|
|
|
|
triggerOnChangeOnTab: true,
|
2021-02-23 05:20:32 -05:00
|
|
|
autofocus: false,
|
2021-12-02 19:47:33 -05:00
|
|
|
placementStrategy: null,
|
|
|
|
mobilePlacementStrategy: null,
|
|
|
|
desktopPlacementStrategy: null,
|
2022-08-19 12:39:58 -04:00
|
|
|
hiddenValues: null,
|
2022-09-29 10:57:36 -04:00
|
|
|
disabled: false,
|
2023-01-13 10:13:13 -05:00
|
|
|
expandedOnInsert: false,
|
2023-07-10 07:43:33 -04:00
|
|
|
formName: null,
|
2020-02-03 08:22:14 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
autoFilterable: computed("content.[]", "selectKit.filter", function () {
|
|
|
|
return (
|
|
|
|
this.selectKit.filter &&
|
|
|
|
this.options.autoFilterable &&
|
|
|
|
this.content.length > 15
|
2018-06-15 11:03:24 -04:00
|
|
|
);
|
2020-02-03 08:22:14 -05:00
|
|
|
}),
|
|
|
|
|
|
|
|
collections: computed(
|
|
|
|
"selectedContent.[]",
|
|
|
|
"mainCollection.[]",
|
|
|
|
"errorsCollection.[]",
|
|
|
|
function () {
|
|
|
|
return this._collections.map((identifier) => {
|
|
|
|
return {
|
|
|
|
identifier,
|
|
|
|
content: this.selectKit.modifyContentForCollection(identifier),
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
),
|
|
|
|
|
|
|
|
createContentFromInput(input) {
|
|
|
|
return input;
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
validateCreate(filter, content) {
|
|
|
|
this.clearErrors();
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
return (
|
|
|
|
filter.length > 0 &&
|
|
|
|
content &&
|
|
|
|
!content.map((c) => this.getValue(c)).includes(filter) &&
|
|
|
|
!makeArray(this.value).includes(filter)
|
2019-02-07 08:43:33 -05:00
|
|
|
);
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
validateSelect() {
|
|
|
|
this.clearErrors();
|
|
|
|
|
2020-02-18 18:57:58 -05:00
|
|
|
const selection = makeArray(this.value);
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
const maximum = this.selectKit.options.maximum;
|
|
|
|
if (maximum && selection.length >= maximum) {
|
|
|
|
const key =
|
|
|
|
this.selectKit.options.maximumLabel ||
|
|
|
|
"select_kit.max_content_reached";
|
|
|
|
this.addError(I18n.t(key, { count: maximum }));
|
|
|
|
return false;
|
2018-08-03 16:41:37 -04:00
|
|
|
}
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
return true;
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
2017-11-22 07:49:45 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
addError(error) {
|
2024-03-19 00:47:18 -04:00
|
|
|
if (!this.errorsCollection.includes(error)) {
|
|
|
|
this.errorsCollection.pushObject(error);
|
|
|
|
}
|
2020-02-03 08:22:14 -05:00
|
|
|
|
2024-03-25 17:58:11 -04:00
|
|
|
this._safeAfterRender(() => this._updatePopper());
|
2017-11-22 07:49:45 -05:00
|
|
|
},
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
clearErrors() {
|
|
|
|
if (!this.element || this.isDestroyed || this.isDestroying) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.set("errorsCollection", []);
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
prependCollection(identifier) {
|
|
|
|
this._collections.unshift(identifier);
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
2017-11-27 13:50:04 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
appendCollection(identifier) {
|
|
|
|
this._collections.push(identifier);
|
|
|
|
},
|
2018-02-26 05:42:57 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
insertCollectionAtIndex(identifier, index) {
|
|
|
|
this._collections.insertAt(index, identifier);
|
|
|
|
},
|
2018-03-22 06:29:55 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
insertBeforeCollection(identifier, insertedIdentifier) {
|
|
|
|
const index = this._collections.indexOf(identifier);
|
|
|
|
this.insertCollectionAtIndex(insertedIdentifier, index - 1);
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
2018-03-22 06:29:55 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
insertAfterCollection(identifier, insertedIdentifier) {
|
|
|
|
const index = this._collections.indexOf(identifier);
|
|
|
|
this.insertCollectionAtIndex(insertedIdentifier, index + 1);
|
2018-03-22 06:29:55 -04:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_onInput(event) {
|
2024-03-25 17:58:11 -04:00
|
|
|
this._updatePopper();
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
if (this._searchPromise) {
|
|
|
|
cancel(this._searchPromise);
|
|
|
|
}
|
|
|
|
|
2021-08-23 04:44:19 -04:00
|
|
|
this.selectKit.set("isLoading", true);
|
|
|
|
|
|
|
|
discourseDebounce(
|
|
|
|
this,
|
|
|
|
this._debouncedInput,
|
|
|
|
event.target.value,
|
|
|
|
INPUT_DELAY
|
|
|
|
);
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_debouncedInput(filter) {
|
2021-08-23 04:44:19 -04:00
|
|
|
this.selectKit.set("filter", filter);
|
2020-03-13 07:41:08 -04:00
|
|
|
this.triggerSearch(filter);
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_onChangeWrapper(value, items) {
|
|
|
|
this.selectKit.set("filter", null);
|
|
|
|
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
if (
|
|
|
|
!this.selectKit.valueProperty &&
|
|
|
|
this.selectKit.noneItem === value
|
|
|
|
) {
|
|
|
|
value = null;
|
|
|
|
items = [];
|
|
|
|
}
|
|
|
|
|
2020-02-11 09:54:56 -05:00
|
|
|
value = makeArray(value);
|
|
|
|
items = makeArray(items);
|
|
|
|
|
|
|
|
if (this.multiSelect) {
|
|
|
|
items = items.filter(
|
|
|
|
(i) =>
|
|
|
|
i !== this.newItem &&
|
|
|
|
i !== this.noneItem &&
|
|
|
|
this.getValue(i) !== null
|
|
|
|
);
|
|
|
|
|
|
|
|
if (this.selectKit.options.maximum === 1) {
|
|
|
|
value = value.slice(0, 1);
|
|
|
|
items = items.slice(0, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.singleSelect) {
|
2020-02-18 17:41:15 -05:00
|
|
|
value = isPresent(value.firstObject) ? value.firstObject : null;
|
|
|
|
items = isPresent(items.firstObject) ? items.firstObject : null;
|
2020-02-11 09:54:56 -05:00
|
|
|
}
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
this._boundaryActionHandler("onChange", value, items);
|
2020-05-06 11:16:20 -04:00
|
|
|
|
|
|
|
applyOnChangePluginApiCallbacks(value, items, this);
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
resolve(items);
|
|
|
|
}).finally(() => {
|
|
|
|
if (!this.isDestroying && !this.isDestroyed) {
|
2021-11-29 08:29:11 -05:00
|
|
|
if (
|
|
|
|
this.selectKit.options.closeOnChange ||
|
|
|
|
(isPresent(value) && this.selectKit.options.maximum === 1)
|
|
|
|
) {
|
2021-09-03 12:29:16 -04:00
|
|
|
this.selectKit.close(event);
|
2020-02-03 08:22:14 -05:00
|
|
|
}
|
|
|
|
|
2020-05-16 05:20:37 -04:00
|
|
|
if (this.selectKit.options.focusAfterOnChange) {
|
2020-05-09 04:26:23 -04:00
|
|
|
this._safeAfterRender(() => {
|
|
|
|
this._focusFilter();
|
2024-03-25 17:58:11 -04:00
|
|
|
this._updatePopper();
|
2020-05-09 04:26:23 -04:00
|
|
|
});
|
|
|
|
}
|
2020-02-03 08:22:14 -05:00
|
|
|
}
|
|
|
|
});
|
2018-04-05 10:45:19 -04:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_modifyContentWrapper(content) {
|
|
|
|
content = this.modifyContent(content);
|
|
|
|
|
2020-05-06 11:16:20 -04:00
|
|
|
return applyContentPluginApiCallbacks(content, this);
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
modifyContent(content) {
|
|
|
|
return content;
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_modifyNoSelectionWrapper() {
|
2020-05-06 11:16:20 -04:00
|
|
|
return this.modifyNoSelection();
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
modifyNoSelection() {
|
|
|
|
if (this.selectKit.options.translatedNone) {
|
|
|
|
return this.defaultItem(null, this.selectKit.options.translatedNone);
|
|
|
|
}
|
|
|
|
|
|
|
|
let none = this.selectKit.options.none;
|
2020-09-22 10:28:28 -04:00
|
|
|
if (isNone(none) && !this.selectKit.options.allowAny) {
|
|
|
|
return null;
|
|
|
|
}
|
2020-02-03 08:22:14 -05:00
|
|
|
|
2017-11-21 05:53:09 -05:00
|
|
|
if (
|
2020-02-03 08:22:14 -05:00
|
|
|
isNone(none) &&
|
|
|
|
this.selectKit.options.allowAny &&
|
|
|
|
!this.selectKit.isExpanded
|
2017-11-21 05:53:09 -05:00
|
|
|
) {
|
2020-04-22 16:17:53 -04:00
|
|
|
return null;
|
2017-11-21 05:53:09 -05:00
|
|
|
}
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
let item;
|
|
|
|
switch (typeof none) {
|
|
|
|
case "string":
|
|
|
|
item = this.defaultItem(null, I18n.t(none));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
item = none;
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
2018-03-29 07:42:00 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
return item;
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
2017-11-21 05:53:09 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_modifySelectionWrapper(item) {
|
|
|
|
return this.modifySelection(item);
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
modifySelection(item) {
|
|
|
|
return item;
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
2017-11-21 05:53:09 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_onKeydownWrapper(event) {
|
|
|
|
return this._boundaryActionHandler("onKeydown", event);
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
2017-11-21 05:53:09 -05:00
|
|
|
|
2021-08-23 04:44:19 -04:00
|
|
|
_mainElement() {
|
|
|
|
return document.querySelector(`#${this.selectKit.uniqueID}`);
|
|
|
|
},
|
|
|
|
|
|
|
|
_headerElement() {
|
|
|
|
return this.selectKit.mainElement().querySelector("summary");
|
|
|
|
},
|
|
|
|
|
|
|
|
_bodyElement() {
|
|
|
|
return this.selectKit.mainElement().querySelector(".select-kit-body");
|
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_onHover(value, item) {
|
|
|
|
throttle(this, this._highlight, item, 25, true);
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
2018-01-11 03:39:51 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_highlight(item) {
|
|
|
|
this.selectKit.set("highlighted", item);
|
2018-03-22 06:29:55 -04:00
|
|
|
},
|
2018-01-11 03:39:51 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_boundaryActionHandler(actionName, ...params) {
|
|
|
|
if (!this.element || this.isDestroying || this.isDestroyed) {
|
|
|
|
return;
|
|
|
|
}
|
2017-11-21 05:53:09 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
let boundaryAction = true;
|
2017-11-21 05:53:09 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
const privateActionName = `_${actionName}`;
|
|
|
|
const privateAction = get(this, privateActionName);
|
|
|
|
if (privateAction) {
|
|
|
|
boundaryAction = privateAction.call(this, ...params);
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
2018-03-22 06:29:55 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
if (this.actions) {
|
|
|
|
const componentAction = get(this.actions, actionName);
|
|
|
|
if (boundaryAction && componentAction) {
|
|
|
|
boundaryAction = componentAction.call(this, ...params);
|
|
|
|
}
|
|
|
|
}
|
2018-03-22 06:29:55 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
const action = get(this, actionName);
|
|
|
|
if (boundaryAction && action) {
|
|
|
|
boundaryAction = action.call(this, ...params);
|
|
|
|
}
|
|
|
|
|
|
|
|
return boundaryAction;
|
2018-03-22 06:29:55 -04:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
deselect() {
|
|
|
|
this.clearErrors();
|
|
|
|
this.selectKit.change(null, null);
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
|
|
|
|
2021-02-22 08:34:31 -05:00
|
|
|
deselectByValue(value) {
|
|
|
|
if (!value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const item = this.itemForValue(value, this.selectedContent);
|
|
|
|
this.deselect(item);
|
|
|
|
},
|
|
|
|
|
2021-01-06 06:57:13 -05:00
|
|
|
append() {
|
|
|
|
// do nothing on general case
|
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
search(filter) {
|
|
|
|
let content = this.content || [];
|
|
|
|
if (filter) {
|
|
|
|
filter = this._normalize(filter);
|
|
|
|
content = content.filter((c) => {
|
|
|
|
const name = this._normalize(this.getName(c));
|
2022-07-17 14:48:36 -04:00
|
|
|
return name?.includes(filter);
|
2020-02-03 08:22:14 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
return content;
|
2018-02-13 11:23:12 -05:00
|
|
|
},
|
|
|
|
|
2020-03-13 07:41:08 -04:00
|
|
|
triggerSearch(filter) {
|
2021-08-23 04:44:19 -04:00
|
|
|
this._searchPromise && cancel(this._searchPromise);
|
|
|
|
|
2020-03-13 07:41:08 -04:00
|
|
|
this._searchPromise = this._searchWrapper(
|
|
|
|
filter || this.selectKit.filter
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_searchWrapper(filter) {
|
2021-08-23 04:44:19 -04:00
|
|
|
if (this.isDestroyed || this.isDestroying) {
|
|
|
|
return Promise.resolve([]);
|
|
|
|
}
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
this.clearErrors();
|
2021-02-01 05:07:11 -05:00
|
|
|
this.setProperties({
|
|
|
|
mainCollection: [],
|
|
|
|
"selectKit.isLoading": true,
|
|
|
|
"selectKit.enterDisabled": true,
|
|
|
|
});
|
2024-03-25 17:58:11 -04:00
|
|
|
this._safeAfterRender(() => this._updatePopper());
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
let content = [];
|
|
|
|
|
2021-02-01 05:07:11 -05:00
|
|
|
return Promise.resolve(this.search(filter))
|
|
|
|
.then((result) => {
|
2021-08-23 04:44:19 -04:00
|
|
|
if (this.isDestroyed || this.isDestroying) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2024-03-19 00:47:18 -04:00
|
|
|
if (this.selectKit.options.maximum === 0) {
|
|
|
|
this.set("selectKit.isLoading", false);
|
|
|
|
this.set("selectKit.hasNoContent", false);
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2021-02-01 05:07:11 -05:00
|
|
|
content = content.concat(makeArray(result));
|
|
|
|
content = this.selectKit.modifyContent(content).filter(Boolean);
|
2020-02-03 08:22:14 -05:00
|
|
|
|
2021-02-01 05:07:11 -05:00
|
|
|
if (this.selectKit.valueProperty) {
|
|
|
|
content = content.uniqBy(this.selectKit.valueProperty);
|
|
|
|
} else {
|
|
|
|
content = content.uniq();
|
2020-02-03 08:22:14 -05:00
|
|
|
}
|
|
|
|
|
2021-02-01 05:07:11 -05:00
|
|
|
if (this.selectKit.options.limitMatches) {
|
|
|
|
content = content.slice(0, this.selectKit.options.limitMatches);
|
|
|
|
}
|
2020-02-03 08:22:14 -05:00
|
|
|
|
2021-02-01 05:07:11 -05:00
|
|
|
const noneItem = this.selectKit.noneItem;
|
|
|
|
if (
|
|
|
|
this.selectKit.options.allowAny &&
|
|
|
|
filter &&
|
|
|
|
this.getName(noneItem) !== filter
|
|
|
|
) {
|
|
|
|
filter = this.createContentFromInput(filter);
|
|
|
|
if (this.validateCreate(filter, content)) {
|
|
|
|
this.selectKit.set("newItem", this.defaultItem(filter, filter));
|
|
|
|
content.unshift(this.selectKit.newItem);
|
|
|
|
}
|
|
|
|
}
|
2020-02-03 08:22:14 -05:00
|
|
|
|
2021-02-01 05:07:11 -05:00
|
|
|
const hasNoContent = isEmpty(content);
|
|
|
|
if (
|
|
|
|
this.selectKit.hasSelection &&
|
|
|
|
noneItem &&
|
|
|
|
this.selectKit.options.autoInsertNoneItem
|
|
|
|
) {
|
|
|
|
content.unshift(noneItem);
|
|
|
|
}
|
2020-02-03 08:22:14 -05:00
|
|
|
|
2021-02-01 05:07:11 -05:00
|
|
|
this.set("mainCollection", content);
|
|
|
|
|
|
|
|
this.selectKit.setProperties({
|
|
|
|
highlighted:
|
|
|
|
this.singleSelect && this.value
|
|
|
|
? this.itemForValue(this.value, this.mainCollection)
|
2021-08-23 04:44:19 -04:00
|
|
|
: isEmpty(this.selectKit.filter)
|
|
|
|
? null
|
2021-02-01 05:07:11 -05:00
|
|
|
: this.mainCollection.firstObject,
|
|
|
|
isLoading: false,
|
|
|
|
hasNoContent,
|
|
|
|
});
|
|
|
|
|
|
|
|
this._safeAfterRender(() => {
|
2021-08-23 13:33:50 -04:00
|
|
|
if (this.selectKit.isExpanded) {
|
2024-03-25 17:58:11 -04:00
|
|
|
this._updatePopper();
|
2021-08-23 13:33:50 -04:00
|
|
|
this._focusFilter();
|
|
|
|
}
|
2021-02-01 05:07:11 -05:00
|
|
|
});
|
|
|
|
})
|
|
|
|
.finally(() => {
|
|
|
|
if (this.isDestroyed || this.isDestroying) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.set("selectKit.enterDisabled", false);
|
2020-02-03 08:22:14 -05:00
|
|
|
});
|
2018-05-28 10:18:25 -04:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_safeAfterRender(fn) {
|
|
|
|
next(() => {
|
|
|
|
schedule("afterRender", () => {
|
|
|
|
if (!this.element || this.isDestroyed || this.isDestroying) {
|
|
|
|
return;
|
|
|
|
}
|
2018-01-09 04:52:32 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
fn();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
2019-01-03 12:03:01 -05:00
|
|
|
|
2021-08-23 04:44:19 -04:00
|
|
|
_scrollToRow(rowItem, preventScroll = true) {
|
2020-02-03 08:22:14 -05:00
|
|
|
const value = this.getValue(rowItem);
|
2021-10-20 07:07:58 -04:00
|
|
|
|
|
|
|
let rowContainer;
|
|
|
|
if (isPresent(value)) {
|
|
|
|
rowContainer = this.element.querySelector(
|
|
|
|
`.select-kit-row[data-value="${value}"]`
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
rowContainer = this.element.querySelector(".select-kit-row.is-none");
|
|
|
|
}
|
|
|
|
|
2021-12-08 06:22:18 -05:00
|
|
|
rowContainer?.focus({ preventScroll });
|
2021-08-23 04:44:19 -04:00
|
|
|
},
|
2018-03-22 06:29:55 -04:00
|
|
|
|
2021-08-23 04:44:19 -04:00
|
|
|
_highlightLast() {
|
|
|
|
const highlighted = this.mainCollection.objectAt(
|
|
|
|
this.mainCollection.length - 1
|
|
|
|
);
|
|
|
|
if (highlighted) {
|
|
|
|
this._scrollToRow(highlighted, false);
|
|
|
|
this.set("selectKit.highlighted", highlighted);
|
|
|
|
}
|
|
|
|
},
|
2020-02-03 08:22:14 -05:00
|
|
|
|
2021-08-23 04:44:19 -04:00
|
|
|
_highlightFirst() {
|
|
|
|
const highlighted = this.mainCollection.objectAt(0);
|
|
|
|
if (highlighted) {
|
|
|
|
this._scrollToRow(highlighted, false);
|
|
|
|
this.set("selectKit.highlighted", highlighted);
|
2020-02-03 08:22:14 -05:00
|
|
|
}
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
2018-01-11 03:39:51 -05:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_highlightNext() {
|
2020-05-07 10:41:26 -04:00
|
|
|
let highlightedIndex = this.mainCollection.indexOf(
|
2020-02-03 08:22:14 -05:00
|
|
|
this.selectKit.highlighted
|
|
|
|
);
|
|
|
|
const count = this.mainCollection.length;
|
|
|
|
|
|
|
|
if (highlightedIndex < count - 1) {
|
2020-05-07 10:41:26 -04:00
|
|
|
highlightedIndex = highlightedIndex + 1;
|
2020-02-03 08:22:14 -05:00
|
|
|
} else {
|
2021-08-23 04:44:19 -04:00
|
|
|
if (this.selectKit.isFilterExpanded) {
|
|
|
|
this._focusFilter();
|
2024-01-11 08:00:04 -05:00
|
|
|
this.set("selectKit.highlighted", null);
|
|
|
|
return;
|
2021-08-23 04:44:19 -04:00
|
|
|
} else {
|
|
|
|
highlightedIndex = 0;
|
|
|
|
}
|
2020-02-03 08:22:14 -05:00
|
|
|
}
|
|
|
|
|
2020-05-07 10:41:26 -04:00
|
|
|
const highlighted = this.mainCollection.objectAt(highlightedIndex);
|
2020-02-03 08:22:14 -05:00
|
|
|
if (highlighted) {
|
2021-08-23 04:44:19 -04:00
|
|
|
this._scrollToRow(highlighted, false);
|
2020-02-03 08:22:14 -05:00
|
|
|
this.set("selectKit.highlighted", highlighted);
|
|
|
|
}
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_highlightPrevious() {
|
2020-05-07 10:41:26 -04:00
|
|
|
let highlightedIndex = this.mainCollection.indexOf(
|
2020-02-03 08:22:14 -05:00
|
|
|
this.selectKit.highlighted
|
|
|
|
);
|
|
|
|
const count = this.mainCollection.length;
|
|
|
|
|
|
|
|
if (highlightedIndex > 0) {
|
2020-05-07 10:41:26 -04:00
|
|
|
highlightedIndex = highlightedIndex - 1;
|
2020-02-03 08:22:14 -05:00
|
|
|
} else {
|
2021-08-23 04:44:19 -04:00
|
|
|
if (this.selectKit.isFilterExpanded) {
|
|
|
|
this._focusFilter();
|
2024-01-11 08:00:04 -05:00
|
|
|
this.set("selectKit.highlighted", null);
|
|
|
|
return;
|
2021-08-23 04:44:19 -04:00
|
|
|
} else {
|
|
|
|
highlightedIndex = count - 1;
|
|
|
|
}
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
|
|
|
|
2020-05-07 10:41:26 -04:00
|
|
|
const highlighted = this.mainCollection.objectAt(highlightedIndex);
|
2020-02-03 08:22:14 -05:00
|
|
|
if (highlighted) {
|
2021-08-23 04:44:19 -04:00
|
|
|
this._scrollToRow(highlighted, false);
|
2020-02-03 08:22:14 -05:00
|
|
|
this.set("selectKit.highlighted", highlighted);
|
|
|
|
}
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
|
|
|
|
2023-07-25 11:00:02 -04:00
|
|
|
_deselectLast() {
|
|
|
|
if (this.selectKit.hasSelection) {
|
|
|
|
this.deselectByValue(this.value[this.value.length - 1]);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
select(value, item) {
|
2020-02-18 17:41:15 -05:00
|
|
|
if (!isPresent(value)) {
|
2021-09-03 10:40:20 -04:00
|
|
|
this._onClearSelection();
|
2020-02-03 08:22:14 -05:00
|
|
|
} else {
|
|
|
|
const existingItem = this.findValue(this.mainCollection, item);
|
|
|
|
if (existingItem) {
|
|
|
|
if (!this.validateSelect(item)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.selectKit.change(value, item || this.defaultItem(value, value));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_onClearSelection() {
|
|
|
|
this.selectKit.change(null, null);
|
|
|
|
},
|
|
|
|
|
2020-05-06 11:16:20 -04:00
|
|
|
_onOpenWrapper() {
|
|
|
|
return this._boundaryActionHandler("onOpen");
|
2018-03-22 06:29:55 -04:00
|
|
|
},
|
|
|
|
|
2021-08-23 04:44:19 -04:00
|
|
|
_cancelSearch() {
|
|
|
|
this._searchPromise && cancel(this._searchPromise);
|
|
|
|
},
|
|
|
|
|
2020-05-06 11:16:20 -04:00
|
|
|
_onCloseWrapper() {
|
2021-08-23 04:44:19 -04:00
|
|
|
this._cancelSearch();
|
2020-02-03 08:22:14 -05:00
|
|
|
this.set("selectKit.highlighted", null);
|
|
|
|
|
2020-05-06 11:16:20 -04:00
|
|
|
return this._boundaryActionHandler("onClose");
|
2018-03-22 06:29:55 -04:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_toggle(event) {
|
|
|
|
if (this.selectKit.isExpanded) {
|
|
|
|
this._close(event);
|
|
|
|
} else {
|
|
|
|
this._open(event);
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
2018-03-22 06:29:55 -04:00
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_close(event) {
|
|
|
|
if (!this.selectKit.isExpanded) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-03 12:29:16 -04:00
|
|
|
this.selectKit.mainElement().open = false;
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
this.clearErrors();
|
2018-02-13 11:23:12 -05:00
|
|
|
|
2023-08-15 05:50:41 -04:00
|
|
|
const inModal = this.element.closest(".fixed-modal");
|
2023-04-07 08:10:49 -04:00
|
|
|
if (inModal && this.site.mobileView) {
|
2021-08-23 04:44:19 -04:00
|
|
|
const modalBody = inModal.querySelector(".modal-body");
|
|
|
|
modalBody.style = "";
|
|
|
|
}
|
|
|
|
|
2020-05-06 11:16:20 -04:00
|
|
|
this.selectKit.onClose(event);
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
this.selectKit.setProperties({
|
|
|
|
isExpanded: false,
|
|
|
|
filter: null,
|
|
|
|
});
|
2018-06-15 11:03:24 -04:00
|
|
|
},
|
2018-03-22 06:29:55 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_open(event) {
|
|
|
|
if (this.selectKit.isExpanded) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-03 12:29:16 -04:00
|
|
|
this.selectKit.mainElement().open = true;
|
2020-02-03 08:22:14 -05:00
|
|
|
this.clearErrors();
|
2020-05-06 11:16:20 -04:00
|
|
|
this.selectKit.onOpen(event);
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
if (!this.popper) {
|
2023-08-15 05:50:41 -04:00
|
|
|
const inModal = this.element.closest(".fixed-modal .modal-body");
|
2020-02-03 08:22:14 -05:00
|
|
|
const anchor = document.querySelector(
|
2021-02-15 06:32:09 -05:00
|
|
|
`#${this.selectKit.uniqueID}-header`
|
2020-02-03 08:22:14 -05:00
|
|
|
);
|
|
|
|
const popper = document.querySelector(
|
2021-02-15 06:32:09 -05:00
|
|
|
`#${this.selectKit.uniqueID}-body`
|
2020-02-03 08:22:14 -05:00
|
|
|
);
|
2021-12-02 19:47:33 -05:00
|
|
|
const strategy = this._computePlacementStrategy();
|
2020-04-22 07:02:20 -04:00
|
|
|
|
2024-03-27 12:26:24 -04:00
|
|
|
let bottomOffset = 0;
|
|
|
|
if (this.capabilities.isIOS) {
|
|
|
|
bottomOffset +=
|
|
|
|
parseInt(
|
|
|
|
getComputedStyle(document.documentElement)
|
|
|
|
.getPropertyValue("--safe-area-inset-bottom")
|
|
|
|
.trim(),
|
|
|
|
10
|
|
|
|
) || 0;
|
|
|
|
}
|
|
|
|
if (this.site.mobileView) {
|
|
|
|
bottomOffset +=
|
|
|
|
parseInt(
|
|
|
|
getComputedStyle(document.documentElement)
|
|
|
|
.getPropertyValue("--footer-nav-height")
|
|
|
|
.trim(),
|
|
|
|
10
|
|
|
|
) || 0;
|
|
|
|
}
|
|
|
|
|
2020-09-14 12:33:14 -04:00
|
|
|
this.popper = createPopper(anchor, popper, {
|
2020-02-03 08:22:14 -05:00
|
|
|
eventsEnabled: false,
|
2021-12-02 19:47:33 -05:00
|
|
|
strategy,
|
2020-02-03 08:22:14 -05:00
|
|
|
placement: this.selectKit.options.placement,
|
|
|
|
modifiers: [
|
2022-08-05 13:09:38 -04:00
|
|
|
{
|
|
|
|
name: "eventListeners",
|
|
|
|
options: {
|
2024-03-18 18:29:42 -04:00
|
|
|
resize: this.site.desktopView,
|
|
|
|
scroll: this.site.desktopView,
|
2022-08-05 13:09:38 -04:00
|
|
|
},
|
|
|
|
},
|
2022-03-30 20:58:22 -04:00
|
|
|
{
|
|
|
|
name: "flip",
|
|
|
|
enabled: !inModal,
|
|
|
|
options: {
|
|
|
|
padding: {
|
|
|
|
top:
|
|
|
|
parseInt(
|
|
|
|
document.documentElement.style.getPropertyValue(
|
|
|
|
"--header-offset"
|
|
|
|
),
|
|
|
|
10
|
|
|
|
) || 0,
|
2024-03-27 12:26:24 -04:00
|
|
|
bottom: bottomOffset,
|
2022-03-30 20:58:22 -04:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2020-04-22 04:54:02 -04:00
|
|
|
{
|
|
|
|
name: "offset",
|
|
|
|
options: {
|
2022-11-10 01:42:37 -05:00
|
|
|
offset: [0, this.selectKit.options.verticalOffset],
|
2020-04-22 04:54:02 -04:00
|
|
|
},
|
|
|
|
},
|
2020-05-05 06:59:47 -04:00
|
|
|
{
|
|
|
|
name: "applySmallScreenOffset",
|
|
|
|
enabled: window.innerWidth <= 450,
|
|
|
|
phase: "main",
|
|
|
|
fn({ state }) {
|
2020-05-07 03:10:29 -04:00
|
|
|
if (!inModal) {
|
|
|
|
let { x } = state.elements.reference.getBoundingClientRect();
|
2021-12-02 19:47:33 -05:00
|
|
|
if (strategy === "fixed") {
|
|
|
|
state.modifiersData.popperOffsets.x = 0 + 10;
|
|
|
|
} else {
|
|
|
|
state.modifiersData.popperOffsets.x = -x + 10;
|
|
|
|
}
|
2020-05-07 03:10:29 -04:00
|
|
|
}
|
2020-05-05 06:59:47 -04:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "applySmallScreenMaxWidth",
|
|
|
|
enabled: window.innerWidth <= 450,
|
|
|
|
phase: "beforeWrite",
|
2020-06-22 09:48:00 -04:00
|
|
|
fn: ({ state }) => {
|
2020-05-07 03:10:29 -04:00
|
|
|
if (inModal) {
|
|
|
|
const innerModal = document.querySelector(
|
2023-08-15 05:50:41 -04:00
|
|
|
".fixed-modal div.modal-inner-container"
|
2020-05-07 03:10:29 -04:00
|
|
|
);
|
|
|
|
|
|
|
|
if (innerModal) {
|
2020-06-22 09:48:00 -04:00
|
|
|
if (this.multiSelect) {
|
|
|
|
state.styles.popper.width = `${this.element.offsetWidth}px`;
|
|
|
|
} else {
|
|
|
|
state.styles.popper.width = `${
|
|
|
|
innerModal.clientWidth - 20
|
|
|
|
}px`;
|
|
|
|
}
|
2020-05-07 03:10:29 -04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
state.styles.popper.width = `${window.innerWidth - 20}px`;
|
|
|
|
}
|
2020-05-05 06:59:47 -04:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2022-09-20 22:59:47 -04:00
|
|
|
name: "minWidth",
|
2020-05-05 07:01:02 -04:00
|
|
|
enabled: window.innerWidth > 450,
|
2020-05-05 06:59:47 -04:00
|
|
|
phase: "beforeWrite",
|
|
|
|
requires: ["computeStyles"],
|
|
|
|
fn: ({ state }) => {
|
2021-08-26 10:37:04 -04:00
|
|
|
state.styles.popper.minWidth = `${Math.max(
|
|
|
|
state.rects.reference.width,
|
|
|
|
220
|
|
|
|
)}px`;
|
2020-05-05 06:59:47 -04:00
|
|
|
},
|
|
|
|
effect: ({ state }) => {
|
2021-08-26 10:37:04 -04:00
|
|
|
state.elements.popper.style.minWidth = `${Math.max(
|
|
|
|
state.elements.reference.offsetWidth,
|
|
|
|
220
|
|
|
|
)}px`;
|
2020-05-05 06:59:47 -04:00
|
|
|
},
|
|
|
|
},
|
2020-02-03 08:22:14 -05:00
|
|
|
{
|
2021-08-23 04:44:19 -04:00
|
|
|
name: "modalHeight",
|
|
|
|
enabled: !!(inModal && this.site.mobileView),
|
2020-02-03 08:22:14 -05:00
|
|
|
phase: "afterWrite",
|
2021-08-23 04:44:19 -04:00
|
|
|
fn: ({ state }) => {
|
2023-08-03 14:00:07 -04:00
|
|
|
inModal.style = "";
|
|
|
|
inModal.style.height =
|
|
|
|
inModal.clientHeight + state.rects.popper.height + "px";
|
2020-02-03 08:22:14 -05:00
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
this.selectKit.setProperties({
|
|
|
|
isExpanded: true,
|
|
|
|
isFilterExpanded:
|
|
|
|
this.selectKit.options.filterable || this.selectKit.options.allowAny,
|
|
|
|
});
|
|
|
|
|
2023-08-02 09:59:50 -04:00
|
|
|
if (this.selectKit.options.useHeaderFilter) {
|
|
|
|
this._focusFilterInput();
|
|
|
|
}
|
|
|
|
|
2020-03-13 07:41:08 -04:00
|
|
|
this.triggerSearch();
|
2020-02-03 08:22:14 -05:00
|
|
|
|
|
|
|
this._safeAfterRender(() => {
|
|
|
|
this._focusFilter();
|
2020-04-21 10:48:49 -04:00
|
|
|
this._scrollToCurrent();
|
2024-03-25 17:58:11 -04:00
|
|
|
this._updatePopper();
|
2020-02-03 08:22:14 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2020-04-21 10:48:49 -04:00
|
|
|
_scrollToCurrent() {
|
|
|
|
if (this.value && this.mainCollection) {
|
|
|
|
let highlighted;
|
|
|
|
if (this.valueProperty) {
|
|
|
|
highlighted = this.mainCollection.findBy(
|
|
|
|
this.valueProperty,
|
|
|
|
this.value
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
const index = this.mainCollection.indexOf(this.value);
|
|
|
|
highlighted = this.mainCollection.objectAt(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (highlighted) {
|
2021-12-08 06:22:18 -05:00
|
|
|
this._scrollToRow(highlighted, false);
|
2020-04-21 10:48:49 -04:00
|
|
|
this.set("selectKit.highlighted", highlighted);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_focusFilter(forceHeader = false) {
|
2021-08-23 04:44:19 -04:00
|
|
|
if (!this.selectKit.mainElement()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this.selectKit.mainElement().open) {
|
|
|
|
const headerContainer = this.getHeader();
|
|
|
|
headerContainer && headerContainer.focus({ preventScroll: true });
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-08-02 09:59:50 -04:00
|
|
|
// setting focus as early as possible is best for iOS
|
|
|
|
// because render/promise delays may cause keyboard not to show
|
|
|
|
if (!forceHeader) {
|
|
|
|
this._focusFilterInput();
|
|
|
|
}
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
this._safeAfterRender(() => {
|
|
|
|
const input = this.getFilterInput();
|
|
|
|
if (!forceHeader && input) {
|
2023-08-02 09:59:50 -04:00
|
|
|
this._focusFilterInput();
|
2021-04-16 11:10:02 -04:00
|
|
|
} else if (!this.selectKit.options.preventHeaderFocus) {
|
2020-02-03 08:22:14 -05:00
|
|
|
const headerContainer = this.getHeader();
|
|
|
|
headerContainer && headerContainer.focus({ preventScroll: true });
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
2020-02-03 08:22:14 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2023-08-02 09:59:50 -04:00
|
|
|
_focusFilterInput() {
|
|
|
|
const input = this.getFilterInput();
|
|
|
|
|
|
|
|
if (input && document.activeElement !== input) {
|
|
|
|
input.focus({ preventScroll: true });
|
|
|
|
|
|
|
|
if (typeof input.selectionStart === "number") {
|
|
|
|
input.selectionStart = input.selectionEnd = input.value.length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
getFilterInput() {
|
2021-02-15 06:32:09 -05:00
|
|
|
return document.querySelector(`#${this.selectKit.uniqueID}-filter input`);
|
2020-02-03 08:22:14 -05:00
|
|
|
},
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
getHeader() {
|
2021-02-15 06:32:09 -05:00
|
|
|
return document.querySelector(`#${this.selectKit.uniqueID}-header`);
|
2020-02-03 08:22:14 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
handleDeprecations() {
|
|
|
|
this._deprecateValueAttribute();
|
|
|
|
this._deprecateMutations();
|
2023-04-07 08:10:49 -04:00
|
|
|
this._handleDeprecatedArgs();
|
2020-02-03 08:22:14 -05:00
|
|
|
},
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2024-03-25 17:58:11 -04:00
|
|
|
@bindDecorator
|
|
|
|
_updatePopper() {
|
|
|
|
this.popper?.update?.();
|
|
|
|
},
|
|
|
|
|
2021-12-02 19:47:33 -05:00
|
|
|
_computePlacementStrategy() {
|
|
|
|
let placementStrategy = this.selectKit.options.placementStrategy;
|
|
|
|
|
|
|
|
if (placementStrategy) {
|
|
|
|
return placementStrategy;
|
|
|
|
}
|
|
|
|
|
2023-04-07 08:10:49 -04:00
|
|
|
if (this.capabilities.isIpadOS || this.site.mobileView) {
|
2021-12-02 19:47:33 -05:00
|
|
|
placementStrategy =
|
|
|
|
this.selectKit.options.mobilePlacementStrategy || "absolute";
|
|
|
|
} else {
|
|
|
|
placementStrategy =
|
|
|
|
this.selectKit.options.desktopPlacementStrategy || "fixed";
|
|
|
|
}
|
|
|
|
|
|
|
|
return placementStrategy;
|
|
|
|
},
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_deprecated(text) {
|
2023-04-28 11:58:41 -04:00
|
|
|
deprecated(text, {
|
|
|
|
since: "v2.4.0",
|
|
|
|
dropFrom: "2.9.0.beta1",
|
|
|
|
id: "discourse.select-kit",
|
|
|
|
});
|
2020-02-03 08:22:14 -05:00
|
|
|
},
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_deprecateValueAttribute() {
|
2020-02-07 04:39:39 -05:00
|
|
|
if (this.valueAttribute || this.valueAttribute === null) {
|
2020-02-03 08:22:14 -05:00
|
|
|
this._deprecated(
|
|
|
|
"The `valueAttribute` is deprecated. Use `valueProperty` instead"
|
|
|
|
);
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
this.set("valueProperty", this.valueAttribute);
|
|
|
|
}
|
|
|
|
},
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
_deprecateMutations() {
|
2023-08-17 12:07:06 -04:00
|
|
|
this.actions ??= {};
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2023-11-27 06:16:31 -05:00
|
|
|
if (!this.onChange && !this.actions.onChange) {
|
2020-02-03 08:22:14 -05:00
|
|
|
this._deprecated(
|
|
|
|
"Implicit mutation has been deprecated, please use `onChange` handler"
|
|
|
|
);
|
2018-06-15 11:03:24 -04:00
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
this.actions.onChange =
|
2023-11-27 06:16:31 -05:00
|
|
|
this.onSelect ||
|
2020-02-03 08:22:14 -05:00
|
|
|
this.actions.onSelect ||
|
|
|
|
((value) => this.set("value", value));
|
2018-06-15 11:03:24 -04:00
|
|
|
}
|
2020-02-03 08:22:14 -05:00
|
|
|
},
|
|
|
|
|
2023-03-09 08:44:31 -05:00
|
|
|
_resolveDeprecatedOptions() {
|
2020-02-03 08:22:14 -05:00
|
|
|
const migrations = {
|
|
|
|
allowAny: "options.allowAny",
|
|
|
|
allowCreate: "options.allowAny",
|
|
|
|
filterable: "options.filterable",
|
|
|
|
excludeCategoryId: "options.excludeCategoryId",
|
|
|
|
scopedCategoryId: "options.scopedCategoryId",
|
|
|
|
allowUncategorized: "options.allowUncategorized",
|
|
|
|
none: "options.none",
|
|
|
|
rootNone: "options.none",
|
2020-10-13 05:31:07 -04:00
|
|
|
disabled: "options.disabled",
|
2022-04-05 13:01:09 -04:00
|
|
|
isDisabled: "options.disabled",
|
2020-02-03 08:22:14 -05:00
|
|
|
rootNoneLabel: "options.none",
|
|
|
|
showFullTitle: "options.showFullTitle",
|
|
|
|
title: "options.translatedNone",
|
|
|
|
maximum: "options.maximum",
|
|
|
|
minimum: "options.minimum",
|
|
|
|
i18nPostfix: "options.i18nPostfix",
|
2020-02-14 04:00:40 -05:00
|
|
|
i18nPrefix: "options.i18nPrefix",
|
2023-06-15 22:08:26 -04:00
|
|
|
btnCustomClasses: "options.btnCustomClasses",
|
2020-02-14 04:00:40 -05:00
|
|
|
castInteger: "options.castInteger",
|
2020-02-03 08:22:14 -05:00
|
|
|
};
|
|
|
|
|
2023-03-09 08:44:31 -05:00
|
|
|
const resolvedDeprecations = {};
|
|
|
|
|
|
|
|
Object.keys(migrations).forEach((from) => {
|
|
|
|
const to = migrations[from];
|
|
|
|
if (this.get(from) && !this.get(to)) {
|
|
|
|
this._deprecated(
|
|
|
|
`The \`${from}\` attribute is deprecated. Use \`${to}\` instead`
|
|
|
|
);
|
|
|
|
|
|
|
|
resolvedDeprecations[(to, this.get(from))];
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return resolvedDeprecations;
|
|
|
|
},
|
|
|
|
|
2023-04-07 08:10:49 -04:00
|
|
|
_handleDeprecatedArgs() {
|
2023-03-09 08:44:31 -05:00
|
|
|
const migrations = {
|
|
|
|
headerIcon: "icon",
|
|
|
|
onExpand: "onOpen",
|
|
|
|
onCollapse: "onClose",
|
|
|
|
};
|
|
|
|
|
2020-02-03 08:22:14 -05:00
|
|
|
Object.keys(migrations).forEach((from) => {
|
|
|
|
const to = migrations[from];
|
|
|
|
if (this.get(from) && !this.get(to)) {
|
|
|
|
this._deprecated(
|
|
|
|
`The \`${from}\` attribute is deprecated. Use \`${to}\` instead`
|
|
|
|
);
|
|
|
|
|
|
|
|
this.set(to, this.get(from));
|
|
|
|
}
|
|
|
|
});
|
2017-11-21 05:53:09 -05:00
|
|
|
},
|
|
|
|
}
|
|
|
|
);
|