2018-03-27 09:58:27 +02:00
|
|
|
|
<template>
|
2019-04-09 11:42:07 +08:00
|
|
|
|
<div @click="onHeaderClick" :class="headCls">
|
2019-04-08 15:47:07 +08:00
|
|
|
|
<span :class="[prefixCls + '-prefix']" v-if="$slots.prefix || prefix">
|
|
|
|
|
<slot name="prefix">
|
|
|
|
|
<Icon :type="prefix" v-if="prefix" />
|
|
|
|
|
</slot>
|
|
|
|
|
</span>
|
2019-04-09 10:45:01 +08:00
|
|
|
|
<div
|
|
|
|
|
class="ivu-tag ivu-tag-checked"
|
|
|
|
|
v-for="(item, index) in selectedMultiple"
|
|
|
|
|
v-if="maxTagCount === undefined || index < maxTagCount">
|
2018-03-27 09:58:27 +02:00
|
|
|
|
<span class="ivu-tag-text">{{ item.label }}</span>
|
2018-06-25 13:16:06 +08:00
|
|
|
|
<Icon type="ios-close" @click.native.stop="removeTag(item)"></Icon>
|
2019-04-09 10:45:01 +08:00
|
|
|
|
</div><div class="ivu-tag ivu-tag-checked" v-if="maxTagCount !== undefined && selectedMultiple.length > maxTagCount">
|
|
|
|
|
<span class="ivu-tag-text ivu-select-max-tag">
|
2019-04-15 10:05:42 +08:00
|
|
|
|
<template v-if="maxTagPlaceholder">{{ maxTagPlaceholder(selectedMultiple.length - maxTagCount) }}</template>
|
2019-04-09 10:45:01 +08:00
|
|
|
|
<template v-else>+ {{ selectedMultiple.length - maxTagCount }}...</template>
|
|
|
|
|
</span>
|
2018-03-27 09:58:27 +02:00
|
|
|
|
</div>
|
|
|
|
|
<span
|
|
|
|
|
:class="singleDisplayClasses"
|
|
|
|
|
v-show="singleDisplayValue"
|
|
|
|
|
>{{ singleDisplayValue }}</span>
|
|
|
|
|
<input
|
|
|
|
|
:id="inputElementId"
|
|
|
|
|
type="text"
|
|
|
|
|
v-if="filterable"
|
|
|
|
|
v-model="query"
|
|
|
|
|
:disabled="disabled"
|
|
|
|
|
:class="[prefixCls + '-input']"
|
|
|
|
|
:placeholder="showPlaceholder ? localePlaceholder : ''"
|
|
|
|
|
:style="inputStyle"
|
|
|
|
|
autocomplete="off"
|
|
|
|
|
spellcheck="false"
|
|
|
|
|
@keydown="resetInputState"
|
|
|
|
|
@keydown.delete="handleInputDelete"
|
|
|
|
|
@focus="onInputFocus"
|
2019-01-07 11:41:56 +08:00
|
|
|
|
@blur="onInputBlur"
|
2018-03-27 09:58:27 +02:00
|
|
|
|
|
|
|
|
|
ref="input">
|
2018-06-25 13:16:06 +08:00
|
|
|
|
<Icon type="ios-close-circle" :class="[prefixCls + '-arrow']" v-if="resetSelect" @click.native.stop="onClear"></Icon>
|
2019-04-09 17:17:47 +08:00
|
|
|
|
<Icon :type="arrowType" :custom="customArrowType" :size="arrowSize" :class="[prefixCls + '-arrow']" v-if="!resetSelect && !remote"></Icon>
|
2018-03-27 09:58:27 +02:00
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
<script>
|
|
|
|
|
import Icon from '../icon';
|
|
|
|
|
import Emitter from '../../mixins/emitter';
|
|
|
|
|
import Locale from '../../mixins/locale';
|
|
|
|
|
|
|
|
|
|
const prefixCls = 'ivu-select';
|
|
|
|
|
|
|
|
|
|
export default {
|
|
|
|
|
name: 'iSelectHead',
|
|
|
|
|
mixins: [ Emitter, Locale ],
|
|
|
|
|
components: { Icon },
|
|
|
|
|
props: {
|
|
|
|
|
disabled: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
default: false
|
|
|
|
|
},
|
|
|
|
|
filterable: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
default: false
|
|
|
|
|
},
|
|
|
|
|
multiple: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
default: false
|
|
|
|
|
},
|
|
|
|
|
remote: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
default: false
|
|
|
|
|
},
|
|
|
|
|
initialLabel: {
|
2018-06-07 09:58:13 +08:00
|
|
|
|
type: [String, Number, Array],
|
2018-03-27 09:58:27 +02:00
|
|
|
|
},
|
|
|
|
|
values: {
|
|
|
|
|
type: Array,
|
|
|
|
|
default: () => []
|
|
|
|
|
},
|
|
|
|
|
clearable: {
|
|
|
|
|
type: [Function, Boolean],
|
|
|
|
|
default: false,
|
|
|
|
|
},
|
|
|
|
|
inputElementId: {
|
|
|
|
|
type: String
|
|
|
|
|
},
|
|
|
|
|
placeholder: {
|
|
|
|
|
type: String
|
|
|
|
|
},
|
|
|
|
|
queryProp: {
|
|
|
|
|
type: String,
|
|
|
|
|
default: ''
|
2019-04-08 15:47:07 +08:00
|
|
|
|
},
|
|
|
|
|
prefix: {
|
|
|
|
|
type: String
|
|
|
|
|
},
|
2019-04-09 10:45:01 +08:00
|
|
|
|
// 3.4.0
|
|
|
|
|
maxTagCount: {
|
|
|
|
|
type: Number
|
|
|
|
|
},
|
|
|
|
|
// 3.4.0
|
|
|
|
|
maxTagPlaceholder: {
|
2019-04-15 10:05:42 +08:00
|
|
|
|
type: Function
|
2019-04-09 10:45:01 +08:00
|
|
|
|
}
|
2018-03-27 09:58:27 +02:00
|
|
|
|
},
|
|
|
|
|
data () {
|
|
|
|
|
return {
|
|
|
|
|
prefixCls: prefixCls,
|
|
|
|
|
query: '',
|
|
|
|
|
inputLength: 20,
|
|
|
|
|
remoteInitialLabel: this.initialLabel,
|
|
|
|
|
preventRemoteCall: false,
|
|
|
|
|
};
|
|
|
|
|
},
|
|
|
|
|
computed: {
|
|
|
|
|
singleDisplayClasses(){
|
|
|
|
|
const {filterable, multiple, showPlaceholder} = this;
|
|
|
|
|
return [{
|
2019-04-08 15:47:07 +08:00
|
|
|
|
[prefixCls + '-head-with-prefix']: this.$slots.prefix || this.prefix,
|
2018-03-27 09:58:27 +02:00
|
|
|
|
[prefixCls + '-placeholder']: showPlaceholder && !filterable,
|
|
|
|
|
[prefixCls + '-selected-value']: !showPlaceholder && !multiple && !filterable,
|
|
|
|
|
}];
|
|
|
|
|
},
|
|
|
|
|
singleDisplayValue(){
|
|
|
|
|
if ((this.multiple && this.values.length > 0) || this.filterable) return '';
|
|
|
|
|
return `${this.selectedSingle}` || this.localePlaceholder;
|
|
|
|
|
},
|
|
|
|
|
showPlaceholder () {
|
|
|
|
|
let status = false;
|
|
|
|
|
if (!this.multiple) {
|
|
|
|
|
const value = this.values[0];
|
|
|
|
|
if (typeof value === 'undefined' || String(value).trim() === ''){
|
|
|
|
|
status = !this.remoteInitialLabel;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (!this.values.length > 0) {
|
|
|
|
|
status = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return status;
|
|
|
|
|
},
|
|
|
|
|
resetSelect(){
|
|
|
|
|
return !this.showPlaceholder && this.clearable;
|
|
|
|
|
},
|
|
|
|
|
inputStyle () {
|
|
|
|
|
let style = {};
|
|
|
|
|
|
|
|
|
|
if (this.multiple) {
|
|
|
|
|
if (this.showPlaceholder) {
|
|
|
|
|
style.width = '100%';
|
|
|
|
|
} else {
|
|
|
|
|
style.width = `${this.inputLength}px`;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return style;
|
|
|
|
|
},
|
|
|
|
|
localePlaceholder () {
|
|
|
|
|
if (this.placeholder === undefined) {
|
|
|
|
|
return this.t('i.select.placeholder');
|
|
|
|
|
} else {
|
|
|
|
|
return this.placeholder;
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
selectedSingle(){
|
|
|
|
|
const selected = this.values[0];
|
|
|
|
|
return selected ? selected.label : (this.remoteInitialLabel || '');
|
|
|
|
|
},
|
|
|
|
|
selectedMultiple(){
|
|
|
|
|
return this.multiple ? this.values : [];
|
2019-04-09 11:42:07 +08:00
|
|
|
|
},
|
|
|
|
|
// 使用 prefix 时,在 filterable
|
|
|
|
|
headCls () {
|
|
|
|
|
return {
|
|
|
|
|
[`${prefixCls}-head-flex`]: this.filterable && (this.$slots.prefix || this.prefix)
|
|
|
|
|
};
|
2019-04-09 17:17:47 +08:00
|
|
|
|
},
|
|
|
|
|
// 3.4.0, global setting customArrow 有值时,arrow 赋值空
|
|
|
|
|
arrowType () {
|
|
|
|
|
let type = 'ios-arrow-down';
|
|
|
|
|
|
|
|
|
|
if (this.$IVIEW) {
|
|
|
|
|
if (this.$IVIEW.select.customArrow) {
|
|
|
|
|
type = '';
|
|
|
|
|
} else if (this.$IVIEW.select.arrow) {
|
|
|
|
|
type = this.$IVIEW.select.arrow;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return type;
|
|
|
|
|
},
|
|
|
|
|
// 3.4.0, global setting
|
|
|
|
|
customArrowType () {
|
|
|
|
|
let type = '';
|
|
|
|
|
|
|
|
|
|
if (this.$IVIEW) {
|
|
|
|
|
if (this.$IVIEW.select.customArrow) {
|
|
|
|
|
type = this.$IVIEW.select.customArrow;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return type;
|
|
|
|
|
},
|
|
|
|
|
// 3.4.0, global setting
|
|
|
|
|
arrowSize () {
|
|
|
|
|
let size = '';
|
|
|
|
|
|
|
|
|
|
if (this.$IVIEW) {
|
|
|
|
|
if (this.$IVIEW.select.arrowSize) {
|
|
|
|
|
size = this.$IVIEW.select.arrowSize;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return size;
|
2018-03-27 09:58:27 +02:00
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
methods: {
|
2019-01-07 11:41:56 +08:00
|
|
|
|
onInputFocus(){
|
|
|
|
|
this.$emit('on-input-focus');
|
|
|
|
|
},
|
|
|
|
|
onInputBlur () {
|
|
|
|
|
if (!this.values.length) this.query = ''; // #5155
|
|
|
|
|
this.$emit('on-input-blur');
|
2018-03-27 09:58:27 +02:00
|
|
|
|
},
|
|
|
|
|
removeTag (value) {
|
|
|
|
|
if (this.disabled) return false;
|
|
|
|
|
this.dispatch('iSelect', 'on-select-selected', value);
|
|
|
|
|
},
|
|
|
|
|
resetInputState () {
|
|
|
|
|
this.inputLength = this.$refs.input.value.length * 12 + 20;
|
2018-09-28 11:17:45 +08:00
|
|
|
|
this.$emit('on-keydown');
|
2018-03-27 09:58:27 +02:00
|
|
|
|
},
|
2019-09-04 16:07:40 +08:00
|
|
|
|
handleInputDelete (e) {
|
|
|
|
|
const targetValue = e.target.value;
|
|
|
|
|
if (this.multiple && this.selectedMultiple.length && this.query === '' && targetValue === '' ) {
|
2018-03-27 09:58:27 +02:00
|
|
|
|
this.removeTag(this.selectedMultiple[this.selectedMultiple.length - 1]);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
onHeaderClick(e){
|
|
|
|
|
if (this.filterable && e.target === this.$el){
|
|
|
|
|
this.$refs.input.focus();
|
|
|
|
|
}
|
2018-05-08 12:47:03 +02:00
|
|
|
|
},
|
|
|
|
|
onClear(){
|
|
|
|
|
this.$emit('on-clear');
|
2018-03-27 09:58:27 +02:00
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
watch: {
|
|
|
|
|
values ([value]) {
|
|
|
|
|
if (!this.filterable) return;
|
|
|
|
|
this.preventRemoteCall = true;
|
|
|
|
|
if (this.multiple){
|
|
|
|
|
this.query = '';
|
|
|
|
|
this.preventRemoteCall = false; // this should be after the query change setter above
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
// #982
|
|
|
|
|
if (typeof value === 'undefined' || value === '' || value === null) this.query = '';
|
|
|
|
|
else this.query = value.label;
|
2018-05-11 08:45:14 +02:00
|
|
|
|
this.$nextTick(() => this.preventRemoteCall = false); // this should be after the query change setter above
|
2018-03-27 09:58:27 +02:00
|
|
|
|
},
|
|
|
|
|
query (val) {
|
|
|
|
|
if (this.preventRemoteCall) {
|
|
|
|
|
this.preventRemoteCall = false;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.$emit('on-query-change', val);
|
|
|
|
|
},
|
|
|
|
|
queryProp(query){
|
|
|
|
|
if (query !== this.query) this.query = query;
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
</script>
|