mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
205 lines
4.9 KiB
JavaScript
205 lines
4.9 KiB
JavaScript
import { use } from '../utils';
|
|
import { prevent } from '../utils/event';
|
|
import { deepClone } from '../utils/deep-clone';
|
|
import { pickerProps } from './shared';
|
|
import Loading from '../loading';
|
|
import PickerColumn from './PickerColumn';
|
|
|
|
const [sfc, bem, t] = use('picker');
|
|
|
|
export default sfc({
|
|
props: {
|
|
...pickerProps,
|
|
columns: Array,
|
|
valueKey: {
|
|
type: String,
|
|
default: 'text'
|
|
}
|
|
},
|
|
|
|
data() {
|
|
return {
|
|
children: []
|
|
};
|
|
},
|
|
|
|
computed: {
|
|
simple() {
|
|
return this.columns.length && !this.columns[0].values;
|
|
}
|
|
},
|
|
|
|
watch: {
|
|
columns() {
|
|
this.setColumns();
|
|
}
|
|
},
|
|
|
|
methods: {
|
|
setColumns() {
|
|
const columns = this.simple ? [{ values: this.columns }] : this.columns;
|
|
columns.forEach((column, index) => {
|
|
this.setColumnValues(index, deepClone(column.values));
|
|
});
|
|
},
|
|
|
|
emit(event) {
|
|
if (this.simple) {
|
|
this.$emit(event, this.getColumnValue(0), this.getColumnIndex(0));
|
|
} else {
|
|
this.$emit(event, this.getValues(), this.getIndexes());
|
|
}
|
|
},
|
|
|
|
onChange(columnIndex) {
|
|
if (this.simple) {
|
|
this.$emit(
|
|
'change',
|
|
this,
|
|
this.getColumnValue(0),
|
|
this.getColumnIndex(0)
|
|
);
|
|
} else {
|
|
this.$emit('change', this, this.getValues(), columnIndex);
|
|
}
|
|
},
|
|
|
|
// get column instance by index
|
|
getColumn(index) {
|
|
return this.children[index];
|
|
},
|
|
|
|
// get column value by index
|
|
getColumnValue(index) {
|
|
const column = this.getColumn(index);
|
|
return column && column.getValue();
|
|
},
|
|
|
|
// set column value by index
|
|
setColumnValue(index, value) {
|
|
const column = this.getColumn(index);
|
|
column && column.setValue(value);
|
|
},
|
|
|
|
// get column option index by column index
|
|
getColumnIndex(columnIndex) {
|
|
return (this.getColumn(columnIndex) || {}).currentIndex;
|
|
},
|
|
|
|
// set column option index by column index
|
|
setColumnIndex(columnIndex, optionIndex) {
|
|
const column = this.getColumn(columnIndex);
|
|
column && column.setIndex(optionIndex);
|
|
},
|
|
|
|
// get options of column by index
|
|
getColumnValues(index) {
|
|
return (this.children[index] || {}).options;
|
|
},
|
|
|
|
// set options of column by index
|
|
setColumnValues(index, options) {
|
|
const column = this.children[index];
|
|
if (
|
|
column &&
|
|
JSON.stringify(column.options) !== JSON.stringify(options)
|
|
) {
|
|
column.options = options;
|
|
column.setIndex(0);
|
|
}
|
|
},
|
|
|
|
// get values of all columns
|
|
getValues() {
|
|
return this.children.map(child => child.getValue());
|
|
},
|
|
|
|
// set values of all columns
|
|
setValues(values) {
|
|
values.forEach((value, index) => {
|
|
this.setColumnValue(index, value);
|
|
});
|
|
},
|
|
|
|
// get indexes of all columns
|
|
getIndexes() {
|
|
return this.children.map(child => child.currentIndex);
|
|
},
|
|
|
|
// set indexes of all columns
|
|
setIndexes(indexes) {
|
|
indexes.forEach((optionIndex, columnIndex) => {
|
|
this.setColumnIndex(columnIndex, optionIndex);
|
|
});
|
|
},
|
|
|
|
onConfirm() {
|
|
this.emit('confirm');
|
|
},
|
|
|
|
onCancel() {
|
|
this.emit('cancel');
|
|
}
|
|
},
|
|
|
|
render(h) {
|
|
const { itemHeight } = this;
|
|
const columns = this.simple ? [this.columns] : this.columns;
|
|
|
|
const frameStyle = {
|
|
height: `${itemHeight}px`
|
|
};
|
|
|
|
const columnsStyle = {
|
|
height: `${itemHeight * this.visibleItemCount}px`
|
|
};
|
|
|
|
const Toolbar = this.showToolbar && (
|
|
<div class={['van-hairline--top-bottom', bem('toolbar')]}>
|
|
{this.slots() || [
|
|
<div class={bem('cancel')} onClick={this.onCancel}>
|
|
{this.cancelButtonText || t('cancel')}
|
|
</div>,
|
|
this.slots('title') ||
|
|
(this.title && (
|
|
<div class={['van-ellipsis', bem('title')]}>{this.title}</div>
|
|
)),
|
|
<div class={bem('confirm')} onClick={this.onConfirm}>
|
|
{this.confirmButtonText || t('confirm')}
|
|
</div>
|
|
]}
|
|
</div>
|
|
);
|
|
|
|
return (
|
|
<div class={bem()}>
|
|
{Toolbar}
|
|
{this.loading && (
|
|
<div class={bem('loading')}>
|
|
<Loading />
|
|
</div>
|
|
)}
|
|
<div class={bem('columns')} style={columnsStyle} onTouchmove={prevent}>
|
|
{columns.map((item, index) => (
|
|
<PickerColumn
|
|
valueKey={this.valueKey}
|
|
className={item.className}
|
|
itemHeight={this.itemHeight}
|
|
defaultIndex={item.defaultIndex}
|
|
visibleItemCount={this.visibleItemCount}
|
|
initialOptions={this.simple ? item : item.values}
|
|
onChange={() => {
|
|
this.onChange(index);
|
|
}}
|
|
/>
|
|
))}
|
|
<div
|
|
class={['van-hairline--top-bottom', bem('frame')]}
|
|
style={frameStyle}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
});
|