FIX: fixes few regressions in select-kit (#17453)
- following c3fd91670e
`paste` has been typoed into `pase`
- adds two tests for pasting in `multi-select` and `email-group-user-chooser`
- selectKitOptions would not be following the right overriding order
- `category-selector` was using `selectKitOptions` directly which shouldn't be the case as it's not using computed values
- apparently since a recent ember upgrade, paste event is not providing `originalEvent` anymore and `clipboardData` should be retrieved directly on the event
This commit is contained in:
parent
09f1ef6b05
commit
a939609d32
app/assets/javascripts
discourse/tests/integration/components/select-kit
select-kit/addon
components
templates/components/select-kit
|
@ -0,0 +1,51 @@
|
|||
import { module, test } from "qunit";
|
||||
import { setupRenderingTest } from "discourse/tests/helpers/component-test";
|
||||
import { render } from "@ember/test-helpers";
|
||||
import { hbs } from "ember-cli-htmlbars";
|
||||
import selectKit from "discourse/tests/helpers/select-kit-helper";
|
||||
import { paste, query } from "discourse/tests/helpers/qunit-helpers";
|
||||
|
||||
const DEFAULT_CONTENT = [
|
||||
{ id: 1, name: "foo" },
|
||||
{ id: 2, name: "bar" },
|
||||
{ id: 3, name: "baz" },
|
||||
];
|
||||
|
||||
const setDefaultState = (ctx, options) => {
|
||||
const properties = Object.assign(
|
||||
{
|
||||
content: DEFAULT_CONTENT,
|
||||
value: null,
|
||||
},
|
||||
options || {}
|
||||
);
|
||||
ctx.setProperties(properties);
|
||||
};
|
||||
|
||||
module(
|
||||
"Integration | Component | select-kit/email-group-user-chooser",
|
||||
function (hooks) {
|
||||
setupRenderingTest(hooks);
|
||||
|
||||
hooks.beforeEach(function () {
|
||||
this.set("subject", selectKit());
|
||||
});
|
||||
|
||||
test("pasting", async function (assert) {
|
||||
setDefaultState(this);
|
||||
|
||||
await render(hbs`
|
||||
<EmailGroupUserChooser
|
||||
@value={{this.value}}
|
||||
@content={{this.content}}
|
||||
@options={{hash maximum=2}}
|
||||
/>
|
||||
`);
|
||||
|
||||
await this.subject.expand();
|
||||
await paste(query(".filter-input"), "foo,bar");
|
||||
|
||||
assert.equal(this.subject.header().value(), "foo,bar");
|
||||
});
|
||||
}
|
||||
);
|
|
@ -3,6 +3,7 @@ import { setupRenderingTest } from "discourse/tests/helpers/component-test";
|
|||
import { render } from "@ember/test-helpers";
|
||||
import { hbs } from "ember-cli-htmlbars";
|
||||
import selectKit from "discourse/tests/helpers/select-kit-helper";
|
||||
import { paste, query } from "discourse/tests/helpers/qunit-helpers";
|
||||
|
||||
const DEFAULT_CONTENT = [
|
||||
{ id: 1, name: "foo" },
|
||||
|
@ -100,4 +101,21 @@ module("Integration | Component | select-kit/multi-select", function (hooks) {
|
|||
|
||||
assert.ok(this.subject.isExpanded(), "it doesn’t close the dropdown");
|
||||
});
|
||||
|
||||
test("pasting", async function (assert) {
|
||||
setDefaultState(this);
|
||||
|
||||
await render(hbs`
|
||||
<MultiSelect
|
||||
@value={{this.value}}
|
||||
@content={{this.content}}
|
||||
@options={{hash maximum=2}}
|
||||
/>
|
||||
`);
|
||||
|
||||
await this.subject.expand();
|
||||
await paste(query(".filter-input"), "foo|bar");
|
||||
|
||||
assert.equal(this.subject.header().value(), "1,2");
|
||||
});
|
||||
});
|
||||
|
|
|
@ -40,7 +40,7 @@ export default MultiSelectComponent.extend({
|
|||
return this.attrs.options.allowUncategorized;
|
||||
}
|
||||
|
||||
return this.selectKitOptions.allowUncategorized;
|
||||
return this.selectKit.options.allowUncategorized;
|
||||
}
|
||||
|
||||
return (
|
||||
|
|
|
@ -10,7 +10,7 @@ export default MultiSelectFilterComponent.extend({
|
|||
return;
|
||||
}
|
||||
|
||||
const data = event.originalEvent.clipboardData;
|
||||
const data = event?.clipboardData;
|
||||
|
||||
if (!data) {
|
||||
return;
|
||||
|
|
|
@ -19,6 +19,7 @@ export default SelectKitComponent.extend({
|
|||
closeOnChange: false,
|
||||
autoInsertNoneItem: false,
|
||||
headerComponent: "multi-select/multi-select-header",
|
||||
filterComponent: "multi-select/multi-select-filter",
|
||||
autoFilterable: true,
|
||||
caretDownIcon: "caretIcon",
|
||||
caretUpIcon: "caretIcon",
|
||||
|
|
|
@ -3,6 +3,7 @@ import SelectKitFilterComponent from "select-kit/components/select-kit/select-ki
|
|||
import { isEmpty } from "@ember/utils";
|
||||
import discourseComputed from "discourse-common/utils/decorators";
|
||||
import layout from "select-kit/templates/components/select-kit/select-kit-filter";
|
||||
import { action } from "@ember/object";
|
||||
|
||||
export default SelectKitFilterComponent.extend({
|
||||
layout,
|
||||
|
@ -16,24 +17,23 @@ export default SelectKitFilterComponent.extend({
|
|||
return isEmpty(placeholder) ? "" : I18n.t(placeholder);
|
||||
},
|
||||
|
||||
actions: {
|
||||
onPaste(event) {
|
||||
const data = event.originalEvent.clipboardData;
|
||||
@action
|
||||
onPaste(event) {
|
||||
const data = event?.clipboardData;
|
||||
|
||||
if (!data) {
|
||||
return;
|
||||
}
|
||||
if (!data) {
|
||||
return;
|
||||
}
|
||||
|
||||
const parts = data.getData("text").split("|").filter(Boolean);
|
||||
const parts = data.getData("text").split("|").filter(Boolean);
|
||||
|
||||
if (parts.length > 1) {
|
||||
event.stopPropagation();
|
||||
event.preventDefault();
|
||||
if (parts.length > 1) {
|
||||
event.stopPropagation();
|
||||
event.preventDefault();
|
||||
|
||||
this.selectKit.append(parts);
|
||||
this.selectKit.append(parts);
|
||||
|
||||
return false;
|
||||
}
|
||||
},
|
||||
return false;
|
||||
}
|
||||
},
|
||||
});
|
||||
|
|
|
@ -22,7 +22,7 @@ export const ERRORS_COLLECTION = "ERRORS_COLLECTION";
|
|||
|
||||
const EMPTY_OBJECT = Object.freeze({});
|
||||
const SELECT_KIT_OPTIONS = Mixin.create({
|
||||
mergedProperties: ["selectKitOptions"],
|
||||
concatenatedProperties: ["selectKitOptions"],
|
||||
selectKitOptions: EMPTY_OBJECT,
|
||||
});
|
||||
|
||||
|
@ -207,13 +207,14 @@ export default Component.extend(
|
|||
didReceiveAttrs() {
|
||||
this._super(...arguments);
|
||||
|
||||
Object.keys(this.selectKitOptions).forEach((key) => {
|
||||
const mergedOptions = Object.assign({}, ...this.selectKitOptions);
|
||||
Object.keys(mergedOptions).forEach((key) => {
|
||||
if (isPresent(this.options[key])) {
|
||||
this.selectKit.options.set(key, this.options[key]);
|
||||
return;
|
||||
}
|
||||
|
||||
const value = this.selectKitOptions[key];
|
||||
const value = mergedOptions[key];
|
||||
|
||||
if (
|
||||
key === "componentForRow" ||
|
||||
|
|
|
@ -1,7 +1,23 @@
|
|||
{{#unless this.isHidden}}
|
||||
{{!-- filter-input-search prevents 1password from attempting autocomplete --}}
|
||||
{{!-- template-lint-disable no-down-event-binding --}}
|
||||
<Input tabindex={{0}} class="filter-input" placeholder={{this.placeholder}} autocomplete="off" autocorrect="off" autocapitalize="off" name="filter-input-search" spellcheck={{false}} @value={{readonly this.selectKit.filter}} {{on "pase" (action "onPaste")}} {{on "keydown" (action "onKeydown")}} {{on "keyup" (action "onKeyup")}} @type="search" {{on "input" (action "onInput")}} />
|
||||
|
||||
<Input
|
||||
tabindex={{0}}
|
||||
class="filter-input"
|
||||
placeholder={{this.placeholder}}
|
||||
autocomplete="off"
|
||||
autocorrect="off"
|
||||
autocapitalize="off"
|
||||
name="filter-input-search"
|
||||
spellcheck={{false}}
|
||||
@value={{readonly this.selectKit.filter}}
|
||||
@type="search"
|
||||
{{on "paste" (action "onPaste")}}
|
||||
{{on "keydown" (action "onKeydown")}}
|
||||
{{on "keyup" (action "onKeyup")}}
|
||||
{{on "input" (action "onInput")}}
|
||||
/>
|
||||
|
||||
{{#if this.selectKit.options.filterIcon}}
|
||||
{{d-icon this.selectKit.options.filterIcon class="filter-icon"}}
|
||||
|
|
Loading…
Reference in New Issue