mirror of
https://github.com/tenrok/vue-select.git
synced 2026-06-22 10:30:34 +03:00
Merge remote-tracking branch 'origin/feature/input-slot' into release/v3.0
# Conflicts: # src/components/Select.vue
This commit is contained in:
+1
-1
@@ -33,7 +33,7 @@
|
|||||||
"@babel/plugin-transform-runtime": "^7.2.0",
|
"@babel/plugin-transform-runtime": "^7.2.0",
|
||||||
"@babel/preset-env": "^7.3.1",
|
"@babel/preset-env": "^7.3.1",
|
||||||
"@babel/runtime": "^7.3.1",
|
"@babel/runtime": "^7.3.1",
|
||||||
"@vue/test-utils": "1.0.0-beta.20",
|
"@vue/test-utils": "^1.0.0-beta.29",
|
||||||
"babel-core": "^7.0.0-bridge.0",
|
"babel-core": "^7.0.0-bridge.0",
|
||||||
"babel-loader": "^8.0.0",
|
"babel-loader": "^8.0.0",
|
||||||
"chokidar": "^2.0.4",
|
"chokidar": "^2.0.4",
|
||||||
|
|||||||
+109
-31
@@ -323,29 +323,9 @@
|
|||||||
</span>
|
</span>
|
||||||
</slot>
|
</slot>
|
||||||
|
|
||||||
<input
|
<slot name="search" v-bind="scope.search">
|
||||||
ref="search"
|
<input v-bind="scope.search.attributes" v-on="scope.search.events">
|
||||||
v-model="search"
|
</slot>
|
||||||
@keydown.delete="maybeDeleteValue"
|
|
||||||
@keyup.esc="onEscape"
|
|
||||||
@keydown.up.prevent="typeAheadUp"
|
|
||||||
@keydown.down.prevent="typeAheadDown"
|
|
||||||
@keydown.enter.prevent="typeAheadSelect"
|
|
||||||
@keydown.tab="onTab"
|
|
||||||
@blur="onSearchBlur"
|
|
||||||
@focus="onSearchFocus"
|
|
||||||
type="search"
|
|
||||||
class="form-control"
|
|
||||||
:autocomplete="autocomplete"
|
|
||||||
:disabled="disabled"
|
|
||||||
:placeholder="searchPlaceholder"
|
|
||||||
:tabindex="tabindex"
|
|
||||||
:readonly="!searchable"
|
|
||||||
:id="inputId"
|
|
||||||
role="combobox"
|
|
||||||
:aria-expanded="dropdownOpen"
|
|
||||||
aria-label="Search for option"
|
|
||||||
>
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="vs__actions">
|
<div class="vs__actions">
|
||||||
@@ -601,7 +581,7 @@
|
|||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Enable/disable creating options from searchInput.
|
* Enable/disable creating options from searchEl.
|
||||||
* @type {Boolean}
|
* @type {Boolean}
|
||||||
*/
|
*/
|
||||||
taggable: {
|
taggable: {
|
||||||
@@ -732,6 +712,7 @@
|
|||||||
type: String,
|
type: String,
|
||||||
default: 'auto'
|
default: 'auto'
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* When true, hitting the 'tab' key will select the current select value
|
* When true, hitting the 'tab' key will select the current select value
|
||||||
* @type {Boolean}
|
* @type {Boolean}
|
||||||
@@ -739,6 +720,20 @@
|
|||||||
selectOnTab: {
|
selectOnTab: {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false
|
default: false
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Query Selector used to find the search input
|
||||||
|
* when the 'search' scoped slot is used.
|
||||||
|
*
|
||||||
|
* Must be a valid CSS selector string.
|
||||||
|
*
|
||||||
|
* @see https://developer.mozilla.org/en-US/docs/Web/API/Document/querySelector
|
||||||
|
* @type {String}
|
||||||
|
*/
|
||||||
|
searchInputQuerySelector: {
|
||||||
|
type: String,
|
||||||
|
default: '[type=search]'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -805,7 +800,7 @@
|
|||||||
*/
|
*/
|
||||||
multiple(val) {
|
multiple(val) {
|
||||||
this.mutableValue = val ? [] : null
|
this.mutableValue = val ? [] : null
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -892,7 +887,7 @@
|
|||||||
onAfterSelect(option) {
|
onAfterSelect(option) {
|
||||||
if (this.closeOnSelect) {
|
if (this.closeOnSelect) {
|
||||||
this.open = !this.open
|
this.open = !this.open
|
||||||
this.$refs.search.blur()
|
this.searchEl.blur()
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.clearSearchOnSelect) {
|
if (this.clearSearchOnSelect) {
|
||||||
@@ -906,14 +901,14 @@
|
|||||||
* @return {void}
|
* @return {void}
|
||||||
*/
|
*/
|
||||||
toggleDropdown(e) {
|
toggleDropdown(e) {
|
||||||
if (e.target === this.$refs.openIndicator || e.target === this.$refs.search || e.target === this.$refs.toggle ||
|
if (e.target === this.$refs.openIndicator || e.target === this.searchEl || e.target === this.$refs.toggle ||
|
||||||
e.target.classList.contains('selected-tag') || e.target === this.$el) {
|
e.target.classList.contains('selected-tag') || e.target === this.$el) {
|
||||||
if (this.open) {
|
if (this.open) {
|
||||||
this.$refs.search.blur() // dropdown will close on blur
|
this.searchEl.blur() // dropdown will close on blur
|
||||||
} else {
|
} else {
|
||||||
if (!this.disabled) {
|
if (!this.disabled) {
|
||||||
this.open = true
|
this.open = true
|
||||||
this.$refs.search.focus()
|
this.searchEl.focus()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -975,7 +970,7 @@
|
|||||||
*/
|
*/
|
||||||
onEscape() {
|
onEscape() {
|
||||||
if (!this.search.length) {
|
if (!this.search.length) {
|
||||||
this.$refs.search.blur()
|
this.searchEl.blur()
|
||||||
} else {
|
} else {
|
||||||
this.search = ''
|
this.search = ''
|
||||||
}
|
}
|
||||||
@@ -1029,7 +1024,7 @@
|
|||||||
* @return {this.value}
|
* @return {this.value}
|
||||||
*/
|
*/
|
||||||
maybeDeleteValue() {
|
maybeDeleteValue() {
|
||||||
if (!this.$refs.search.value.length && this.mutableValue && this.clearable) {
|
if (!this.searchEl.value.length && this.mutableValue && this.clearable) {
|
||||||
return this.multiple ? this.mutableValue.pop() : this.mutableValue = null
|
return this.multiple ? this.mutableValue.pop() : this.mutableValue = null
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -1077,11 +1072,94 @@
|
|||||||
*/
|
*/
|
||||||
onMousedown() {
|
onMousedown() {
|
||||||
this.mousedown = true
|
this.mousedown = true
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search 'input' KeyBoardEvent handler.
|
||||||
|
* @param e {KeyboardEvent}
|
||||||
|
* @return {Function}
|
||||||
|
*/
|
||||||
|
onSearchKeyDown (e) {
|
||||||
|
switch (e.keyCode) {
|
||||||
|
case 8:
|
||||||
|
// delete
|
||||||
|
return this.maybeDeleteValue();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search 'input' KeyBoardEvent handler.
|
||||||
|
* @param e {KeyboardEvent}
|
||||||
|
* @return {Function}
|
||||||
|
*/
|
||||||
|
onSearchKeyUp (e) {
|
||||||
|
switch (e.keyCode) {
|
||||||
|
case 27:
|
||||||
|
// esc
|
||||||
|
return this.onEscape();
|
||||||
|
case 38:
|
||||||
|
// up.prevent
|
||||||
|
e.preventDefault();
|
||||||
|
return this.typeAheadUp();
|
||||||
|
case 40:
|
||||||
|
// down.prevent
|
||||||
|
e.preventDefault();
|
||||||
|
return this.typeAheadDown();
|
||||||
|
case 13:
|
||||||
|
// enter.prevent
|
||||||
|
e.preventDefault();
|
||||||
|
return this.typeAheadSelect();
|
||||||
|
case 9:
|
||||||
|
// tab
|
||||||
|
return this.onTab();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Find the search input DOM element.
|
||||||
|
* @returns {HTMLInputElement}
|
||||||
|
*/
|
||||||
|
searchEl () {
|
||||||
|
return !!this.$scopedSlots['search']
|
||||||
|
? this.$refs.selectedOptions.querySelector(this.searchInputQuerySelector)
|
||||||
|
: this.$refs.search;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The object to be bound to the $slots.search scoped slot.
|
||||||
|
* @returns {Object}
|
||||||
|
*/
|
||||||
|
scope () {
|
||||||
|
return {
|
||||||
|
search: {
|
||||||
|
attributes: {
|
||||||
|
'disabled': this.disabled,
|
||||||
|
'placeholder': this.searchPlaceholder,
|
||||||
|
'tabindex': this.tabindex,
|
||||||
|
'readonly': !this.searchable,
|
||||||
|
'id': this.inputId,
|
||||||
|
'aria-expanded': this.dropdownOpen,
|
||||||
|
'aria-label': 'Search for option',
|
||||||
|
'ref': 'search',
|
||||||
|
'role': 'combobox',
|
||||||
|
'type': 'search',
|
||||||
|
'autocomplete': 'off',
|
||||||
|
'class': 'form-control',
|
||||||
|
},
|
||||||
|
events: {
|
||||||
|
'keydown': this.onSearchKeyDown,
|
||||||
|
'keyup': this.onSearchKeyUp,
|
||||||
|
'blur': this.onSearchBlur,
|
||||||
|
'focus': this.onSearchFocus,
|
||||||
|
'input': (e) => this.search = e.target.value,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Classes to be output on .dropdown
|
* Classes to be output on .dropdown
|
||||||
* @return {Object}
|
* @return {Object}
|
||||||
|
|||||||
@@ -57,4 +57,4 @@ module.exports = {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+31
-3
@@ -1,5 +1,6 @@
|
|||||||
import { shallowMount } from "@vue/test-utils";
|
import { shallowMount } from "@vue/test-utils";
|
||||||
import VueSelect from "../src/components/Select.vue";
|
import VueSelect from "../src/components/Select.vue";
|
||||||
|
import Vue from 'vue';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Trigger a submit event on the search
|
* Trigger a submit event on the search
|
||||||
@@ -12,9 +13,7 @@ export const searchSubmit = (Wrapper, searchText = false) => {
|
|||||||
if (searchText) {
|
if (searchText) {
|
||||||
Wrapper.vm.search = searchText;
|
Wrapper.vm.search = searchText;
|
||||||
}
|
}
|
||||||
Wrapper.find({ ref: "search" }).trigger("keydown", {
|
Wrapper.find({ ref: "search" }).trigger("keyup.enter")
|
||||||
keyCode: 13
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -26,3 +25,32 @@ export const searchSubmit = (Wrapper, searchText = false) => {
|
|||||||
export const selectWithProps = (propsData = {}) => {
|
export const selectWithProps = (propsData = {}) => {
|
||||||
return shallowMount(VueSelect, { propsData });
|
return shallowMount(VueSelect, { propsData });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a Wrapper with a v-select component.
|
||||||
|
* @param options
|
||||||
|
* @return {Wrapper<Vue>}
|
||||||
|
*/
|
||||||
|
export const mountDefault = (options = {}) =>
|
||||||
|
shallowMount(VueSelect, {
|
||||||
|
propsData: { options: ["one", "two", "three"],
|
||||||
|
...options
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a v-select component directly.
|
||||||
|
* @param props
|
||||||
|
* @param options
|
||||||
|
* @return {Vue | Element | Vue[] | Element[]}
|
||||||
|
*/
|
||||||
|
export const mountWithoutTestUtils = (props = {}, options = {}) => {
|
||||||
|
return new Vue({
|
||||||
|
render: createEl => createEl('vue-select', {
|
||||||
|
ref: 'select',
|
||||||
|
props: {options: ['one', 'two', 'three'], ...props},
|
||||||
|
...options
|
||||||
|
}),
|
||||||
|
components: {VueSelect},
|
||||||
|
}).$mount().$refs.select;
|
||||||
|
};
|
||||||
|
|||||||
@@ -58,9 +58,7 @@ describe("VS - Selecting Values", () => {
|
|||||||
|
|
||||||
const spy = jest.spyOn(Select.vm, "typeAheadSelect");
|
const spy = jest.spyOn(Select.vm, "typeAheadSelect");
|
||||||
|
|
||||||
Select.find({ ref: "search" }).trigger("keydown", {
|
Select.find({ ref: "search" }).trigger("keyup.tab");
|
||||||
keyCode: 9
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(spy).toHaveBeenCalledWith();
|
expect(spy).toHaveBeenCalledWith();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,15 +1,20 @@
|
|||||||
import { shallowMount } from "@vue/test-utils";
|
import { shallowMount } from '@vue/test-utils';
|
||||||
import VueSelect from "../../src/components/Select";
|
import VueSelect from "../../src/components/Select";
|
||||||
|
import { mountDefault, mountWithoutTestUtils } from '../helpers';
|
||||||
|
import typeAheadMixin from '../../src/mixins/typeAheadPointer';
|
||||||
|
import Vue from 'vue';
|
||||||
|
|
||||||
describe("Moving the Typeahead Pointer", () => {
|
describe("Moving the Typeahead Pointer", () => {
|
||||||
const mountDefault = () =>
|
|
||||||
shallowMount(VueSelect, {
|
it('should set the pointer to zero when the filteredOptions watcher is called', async () => {
|
||||||
propsData: { options: ["one", "two", "three"] }
|
const Select = shallowMount(VueSelect, {
|
||||||
|
propsData: { options: ['one', 'two', 'three'] },
|
||||||
|
sync: false
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should set the pointer to zero when the filteredOptions change", () => {
|
Select.vm.search = 'one';
|
||||||
const Select = mountDefault();
|
|
||||||
Select.vm.search = "two";
|
await Select.vm.$nextTick();
|
||||||
expect(Select.vm.typeAheadPointer).toEqual(0);
|
expect(Select.vm.typeAheadPointer).toEqual(0);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -18,7 +23,7 @@ describe("Moving the Typeahead Pointer", () => {
|
|||||||
|
|
||||||
Select.vm.typeAheadPointer = 1;
|
Select.vm.typeAheadPointer = 1;
|
||||||
|
|
||||||
Select.find({ ref: "search" }).trigger("keydown", { keyCode: 38 });
|
Select.find({ ref: "search" }).trigger("keyup.up");
|
||||||
|
|
||||||
expect(Select.vm.typeAheadPointer).toEqual(0);
|
expect(Select.vm.typeAheadPointer).toEqual(0);
|
||||||
});
|
});
|
||||||
@@ -28,7 +33,7 @@ describe("Moving the Typeahead Pointer", () => {
|
|||||||
|
|
||||||
Select.vm.typeAheadPointer = 1;
|
Select.vm.typeAheadPointer = 1;
|
||||||
|
|
||||||
Select.find({ ref: "search" }).trigger("keydown", { keyCode: 40 });
|
Select.find({ ref: "search" }).trigger("keyup.down");
|
||||||
|
|
||||||
expect(Select.vm.typeAheadPointer).toEqual(2);
|
expect(Select.vm.typeAheadPointer).toEqual(2);
|
||||||
});
|
});
|
||||||
@@ -48,7 +53,7 @@ describe("Moving the Typeahead Pointer", () => {
|
|||||||
|
|
||||||
Select.vm.typeAheadPointer = 1;
|
Select.vm.typeAheadPointer = 1;
|
||||||
|
|
||||||
Select.find({ ref: "search" }).trigger("keydown", { keyCode: 38 });
|
Select.find({ ref: "search" }).trigger("keyup.up");
|
||||||
expect(spy).toHaveBeenCalled();
|
expect(spy).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -58,11 +63,16 @@ describe("Moving the Typeahead Pointer", () => {
|
|||||||
|
|
||||||
Select.vm.typeAheadPointer = 1;
|
Select.vm.typeAheadPointer = 1;
|
||||||
|
|
||||||
Select.find({ ref: "search" }).trigger("keydown", { keyCode: 40 });
|
Select.find({ ref: "search" }).trigger("keyup.down");
|
||||||
expect(spy).toHaveBeenCalled();
|
expect(spy).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should check if the scroll position needs to be adjusted when filtered options changes", () => {
|
/**
|
||||||
|
* This test fails despite working in the browser.
|
||||||
|
* After many attempts to get it to pass, it's been
|
||||||
|
* rewritten below.
|
||||||
|
*/
|
||||||
|
it.skip("should check if the scroll position needs to be adjusted when filtered options changes", () => {
|
||||||
const Select = mountDefault();
|
const Select = mountDefault();
|
||||||
const spy = jest.spyOn(Select.vm, "maybeAdjustScroll");
|
const spy = jest.spyOn(Select.vm, "maybeAdjustScroll");
|
||||||
|
|
||||||
|
|||||||
@@ -770,11 +770,12 @@
|
|||||||
resolved "https://registry.yarnpkg.com/@vue/preload-webpack-plugin/-/preload-webpack-plugin-1.1.0.tgz#d768dba004261c029b53a77c5ea2d5f9ee4f3cce"
|
resolved "https://registry.yarnpkg.com/@vue/preload-webpack-plugin/-/preload-webpack-plugin-1.1.0.tgz#d768dba004261c029b53a77c5ea2d5f9ee4f3cce"
|
||||||
integrity sha512-rcn2KhSHESBFMPj5vc5X2pI9bcBNQQixvJXhD5gZ4rN2iym/uH2qfDSQfUS5+qwiz0a85TCkeUs6w6jxFDudbw==
|
integrity sha512-rcn2KhSHESBFMPj5vc5X2pI9bcBNQQixvJXhD5gZ4rN2iym/uH2qfDSQfUS5+qwiz0a85TCkeUs6w6jxFDudbw==
|
||||||
|
|
||||||
"@vue/test-utils@1.0.0-beta.20":
|
"@vue/test-utils@^1.0.0-beta.29":
|
||||||
version "1.0.0-beta.20"
|
version "1.0.0-beta.29"
|
||||||
resolved "https://registry.yarnpkg.com/@vue/test-utils/-/test-utils-1.0.0-beta.20.tgz#ef4505341b802f3de1c06b3cb8651378c87371fa"
|
resolved "https://registry.yarnpkg.com/@vue/test-utils/-/test-utils-1.0.0-beta.29.tgz#c942cf25e891cf081b6a03332b4ae1ef430726f0"
|
||||||
integrity sha1-70UFNBuALz3hwGs8uGUTeMhzcfo=
|
integrity sha512-yX4sxEIHh4M9yAbLA/ikpEnGKMNBCnoX98xE1RwxfhQVcn0MaXNSj1Qmac+ZydTj6VBSEVukchBogXBTwc+9iA==
|
||||||
dependencies:
|
dependencies:
|
||||||
|
dom-event-types "^1.0.0"
|
||||||
lodash "^4.17.4"
|
lodash "^4.17.4"
|
||||||
|
|
||||||
"@vue/web-component-wrapper@^1.2.0":
|
"@vue/web-component-wrapper@^1.2.0":
|
||||||
@@ -3022,6 +3023,11 @@ dom-converter@~0.1:
|
|||||||
dependencies:
|
dependencies:
|
||||||
utila "~0.3"
|
utila "~0.3"
|
||||||
|
|
||||||
|
dom-event-types@^1.0.0:
|
||||||
|
version "1.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/dom-event-types/-/dom-event-types-1.0.0.tgz#5830a0a29e1bf837fe50a70cd80a597232813cae"
|
||||||
|
integrity sha512-2G2Vwi2zXTHBGqXHsJ4+ak/iP0N8Ar+G8a7LiD2oup5o4sQWytwqqrZu/O6hIMV0KMID2PL69OhpshLO0n7UJQ==
|
||||||
|
|
||||||
dom-serializer@0, dom-serializer@~0.1.0:
|
dom-serializer@0, dom-serializer@~0.1.0:
|
||||||
version "0.1.0"
|
version "0.1.0"
|
||||||
resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-0.1.0.tgz#073c697546ce0780ce23be4a28e293e40bc30c82"
|
resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-0.1.0.tgz#073c697546ce0780ce23be4a28e293e40bc30c82"
|
||||||
|
|||||||
Reference in New Issue
Block a user