2017-10-19 15:51:08 -04:00
|
|
|
import SelectBoxKitComponent from "select-box-kit/components/select-box-kit";
|
|
|
|
import computed from "ember-addons/ember-computed-decorators";
|
2017-11-09 13:57:53 -05:00
|
|
|
const { get, isNone, isEmpty } = Ember;
|
2017-10-19 15:51:08 -04:00
|
|
|
|
|
|
|
export default SelectBoxKitComponent.extend({
|
2017-11-09 13:57:53 -05:00
|
|
|
classNames: "multi-combo-box",
|
2017-10-19 15:51:08 -04:00
|
|
|
headerComponent: "multi-combo-box/multi-combo-box-header",
|
|
|
|
filterComponent: null,
|
|
|
|
headerText: "select_box.default_header_text",
|
|
|
|
allowAny: true,
|
2017-11-09 13:57:53 -05:00
|
|
|
allowValueMutation: false,
|
|
|
|
autoSelectFirst: false,
|
|
|
|
autoFilterable: true,
|
|
|
|
selectedNameComponent: "multi-combo-box/selected-name",
|
|
|
|
|
|
|
|
init() {
|
|
|
|
this._super();
|
|
|
|
|
|
|
|
if (isNone(this.get("value"))) { this.set("value", []); }
|
|
|
|
|
|
|
|
this.set("headerComponentOptions", Ember.Object.create({
|
|
|
|
selectedNameComponent: this.get("selectedNameComponent")
|
|
|
|
}));
|
|
|
|
},
|
2017-10-19 15:51:08 -04:00
|
|
|
|
|
|
|
@computed("filter")
|
|
|
|
templateForCreateRow() {
|
|
|
|
return (rowComponent) => {
|
2017-11-09 13:57:53 -05:00
|
|
|
return I18n.t("select_box.create", { content: rowComponent.get("content.name")});
|
2017-10-19 15:51:08 -04:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
keyDown(event) {
|
|
|
|
const keyCode = event.keyCode || event.which;
|
|
|
|
const $filterInput = this.$filterInput();
|
|
|
|
|
2017-11-09 13:57:53 -05:00
|
|
|
if (this.get("isFocused") === true && this.get("isExpanded") === false && keyCode === this.keys.BACKSPACE) {
|
|
|
|
this.expand();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// select all choices
|
|
|
|
if (event.metaKey === true && keyCode === 65 && isEmpty(this.get("filter"))) {
|
|
|
|
this.$(".choices .selected-name:not(.is-locked)").addClass("is-highlighted");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// clear selection when multiple
|
|
|
|
if (Ember.isEmpty(this.get("filter")) && this.$(".selected-name.is-highlighted").length >= 1 && keyCode === this.keys.BACKSPACE) {
|
|
|
|
const highlightedValues = [];
|
|
|
|
$.each(this.$(".selected-name.is-highlighted"), (i, el) => {
|
|
|
|
highlightedValues.push($(el).attr("data-value"));
|
|
|
|
});
|
2017-10-19 15:51:08 -04:00
|
|
|
|
2017-11-09 13:57:53 -05:00
|
|
|
this.send("onDeselect", highlightedValues);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// try to remove last item from the list
|
|
|
|
if (Ember.isEmpty(this.get("filter")) && keyCode === this.keys.BACKSPACE) {
|
|
|
|
let $lastSelectedValue = $(this.$(".choices .selected-name:not(.is-locked)").last());
|
|
|
|
|
|
|
|
if ($lastSelectedValue.length === 0) { return; }
|
|
|
|
|
|
|
|
if ($lastSelectedValue.hasClass("is-highlighted") || $(document.activeElement).is($lastSelectedValue)) {
|
|
|
|
this.send("onDeselect", this.get("selectedContent.lastObject.value"));
|
2017-10-19 15:51:08 -04:00
|
|
|
$filterInput.focus();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-11-09 13:57:53 -05:00
|
|
|
if ($filterInput.not(":visible") && $lastSelectedValue.length > 0) {
|
|
|
|
$lastSelectedValue.click();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-10-19 15:51:08 -04:00
|
|
|
if ($filterInput.val() === "") {
|
|
|
|
if ($filterInput.is(":focus")) {
|
2017-11-09 13:57:53 -05:00
|
|
|
if ($lastSelectedValue.length > 0) { $lastSelectedValue.click(); }
|
2017-10-19 15:51:08 -04:00
|
|
|
} else {
|
|
|
|
if ($lastSelectedValue.length > 0) {
|
2017-11-09 13:57:53 -05:00
|
|
|
$lastSelectedValue.click();
|
2017-10-19 15:51:08 -04:00
|
|
|
} else {
|
|
|
|
$filterInput.focus();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
@computed("value.[]")
|
2017-11-09 13:57:53 -05:00
|
|
|
computedValue(value) { return value.map(v => this._castInteger(v)); },
|
2017-10-19 15:51:08 -04:00
|
|
|
|
2017-11-09 13:57:53 -05:00
|
|
|
@computed("value.[]", "computedContent.[]")
|
|
|
|
selectedContent(value, computedContent) {
|
2017-10-19 15:51:08 -04:00
|
|
|
const contents = [];
|
2017-11-09 13:57:53 -05:00
|
|
|
value.forEach(v => {
|
|
|
|
const content = computedContent.findBy("value", v);
|
|
|
|
if (!isNone(content)) { contents.push(content); }
|
2017-10-19 15:51:08 -04:00
|
|
|
});
|
|
|
|
return contents;
|
|
|
|
},
|
|
|
|
|
2017-11-09 13:57:53 -05:00
|
|
|
filteredContentFunction(computedContent, computedValue, filter) {
|
|
|
|
computedContent = computedContent.filter(c => {
|
|
|
|
return !computedValue.includes(get(c, "value"));
|
|
|
|
});
|
|
|
|
|
|
|
|
if (isEmpty(filter)) { return computedContent; }
|
|
|
|
|
|
|
|
const lowerFilter = filter.toLowerCase();
|
|
|
|
return computedContent.filter(c => {
|
|
|
|
return get(c, "name").toLowerCase().indexOf(lowerFilter) > -1;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
willCreateContent() {
|
|
|
|
this.set("highlightedValue", null);
|
|
|
|
},
|
|
|
|
|
|
|
|
didCreateContent() {
|
|
|
|
this.clearFilter();
|
|
|
|
this.autoHighlightFunction();
|
|
|
|
},
|
|
|
|
|
|
|
|
createContentFunction(input) {
|
|
|
|
if (!this.get("content").includes(input)) {
|
|
|
|
this.get("content").pushObject(input);
|
|
|
|
this.get("value").pushObject(input);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
deselectValuesFunction(values) {
|
|
|
|
const contents = this._computeRemovableContentsForValues(values);
|
|
|
|
this.get("value").removeObjects(values);
|
|
|
|
this.get("content").removeObjects(contents);
|
|
|
|
},
|
|
|
|
|
|
|
|
highlightValueFunction(value) {
|
|
|
|
this.set("highlightedValue", value);
|
|
|
|
},
|
|
|
|
|
|
|
|
selectValuesFunction(values) {
|
|
|
|
this.get("value").pushObjects(values);
|
|
|
|
},
|
|
|
|
|
|
|
|
willSelectValues() {
|
|
|
|
this.expand();
|
|
|
|
this.set("highlightedValue", null);
|
|
|
|
},
|
|
|
|
|
|
|
|
didSelectValues() {
|
|
|
|
this.focus();
|
|
|
|
this.clearFilter();
|
|
|
|
this.autoHighlightFunction();
|
|
|
|
},
|
|
|
|
|
|
|
|
willDeselectValues() {
|
|
|
|
this.set("highlightedValue", null);
|
|
|
|
},
|
|
|
|
|
|
|
|
didDeselectValues() {
|
|
|
|
this.autoHighlightFunction();
|
|
|
|
},
|
|
|
|
|
|
|
|
willHighlightValue() {},
|
|
|
|
|
|
|
|
didHighlightValue() {},
|
|
|
|
|
|
|
|
autoHighlightFunction() {
|
|
|
|
Ember.run.schedule("afterRender", () => {
|
|
|
|
if (this.get("isExpanded") === false) { return; }
|
|
|
|
if (this.get("renderedBodyOnce") === false) { return; }
|
|
|
|
if (!isNone(this.get("highlightedValue"))) { return; }
|
|
|
|
|
|
|
|
if (isEmpty(this.get("filteredContent"))) {
|
|
|
|
if (!isEmpty(this.get("filter"))) {
|
|
|
|
this.send("onHighlight", this.get("filter"));
|
|
|
|
} else if (this.get("none") && !isEmpty(this.get("selectedContent"))) {
|
|
|
|
this.send("onHighlight", this.noneValue);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
this.send("onHighlight", this.get("filteredContent.firstObject.value"));
|
|
|
|
}
|
|
|
|
});
|
2017-10-19 15:51:08 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
actions: {
|
|
|
|
onClearSelection() {
|
2017-11-09 13:57:53 -05:00
|
|
|
const values = this.get("selectedContent").map(c => get(c, "value"));
|
|
|
|
this.send("onDeselect", values);
|
|
|
|
},
|
|
|
|
|
|
|
|
onHighlight(value) {
|
|
|
|
value = this._originalValueForValue(value);
|
|
|
|
this.willHighlightValue(value);
|
|
|
|
this.set("highlightedValue", value);
|
|
|
|
this.highlightValueFunction(value);
|
|
|
|
this.didHighlightValue(value);
|
|
|
|
},
|
|
|
|
|
|
|
|
onCreateContent(input) {
|
|
|
|
this.willCreateContent(input);
|
|
|
|
this.createContentFunction(input);
|
|
|
|
this.didCreateContent(input);
|
2017-10-19 15:51:08 -04:00
|
|
|
},
|
|
|
|
|
2017-11-09 13:57:53 -05:00
|
|
|
onSelect(values) {
|
|
|
|
values = Ember.makeArray(values).map(v => this._originalValueForValue(v));
|
|
|
|
this.willSelectValues(values);
|
|
|
|
this.selectValuesFunction(values);
|
|
|
|
this.didSelectValues(values);
|
2017-10-19 15:51:08 -04:00
|
|
|
},
|
|
|
|
|
2017-11-09 13:57:53 -05:00
|
|
|
onDeselect(values) {
|
|
|
|
values = Ember.makeArray(this._computeRemovableValues(values));
|
|
|
|
this.willDeselectValues(values);
|
|
|
|
this.deselectValuesFunction(values);
|
|
|
|
this.didSelectValues(values);
|
2017-10-19 15:51:08 -04:00
|
|
|
}
|
2017-11-09 13:57:53 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
_computeRemovableContentsForValues(values) {
|
|
|
|
const removableContents = [];
|
|
|
|
values.forEach(v => {
|
|
|
|
if (!this.get("_initialValues").includes(v)) {
|
|
|
|
const content = this._contentForValue(v);
|
|
|
|
if (!isNone(content)) { removableContents.push(content); }
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return removableContents;
|
|
|
|
},
|
|
|
|
|
|
|
|
_computeRemovableValues(values) {
|
|
|
|
return Ember.makeArray(values)
|
|
|
|
.map(v => this._originalValueForValue(v))
|
|
|
|
.filter(v => {
|
|
|
|
return get(this._computedContentForValue(v), "locked") !== true;
|
|
|
|
});
|
2017-10-19 15:51:08 -04:00
|
|
|
}
|
|
|
|
});
|