161 lines
3.4 KiB
TypeScript

import { VantComponent } from '../common/component';
import { isObj, range } from '../common/utils';
const DEFAULT_DURATION = 200;
VantComponent({
classes: ['active-class'],
props: {
valueKey: String,
className: String,
itemHeight: Number,
visibleItemCount: Number,
initialOptions: {
type: Array,
value: []
},
defaultIndex: {
type: Number,
value: 0
}
},
data: {
startY: 0,
offset: 0,
duration: 0,
startOffset: 0,
options: [],
currentIndex: 0
},
created() {
const { defaultIndex, initialOptions } = this.data;
this.set({
currentIndex: defaultIndex,
options: initialOptions
});
},
computed: {
count() {
return this.data.options.length;
},
baseOffset() {
const { data } = this;
return (data.itemHeight * (data.visibleItemCount - 1)) / 2;
},
wrapperStyle() {
const { data } = this;
return [
`transition: ${data.duration}ms`,
`transform: translate3d(0, ${data.offset + data.baseOffset}px, 0)`,
`line-height: ${data.itemHeight}px`
].join('; ');
}
},
watch: {
defaultIndex(value: number) {
this.setIndex(value);
}
},
methods: {
onTouchStart(event: Weapp.TouchEvent) {
this.set({
startY: event.touches[0].clientY,
startOffset: this.data.offset,
duration: 0
});
},
onTouchMove(event: Weapp.TouchEvent) {
const { data } = this;
const deltaY = event.touches[0].clientY - data.startY;
this.set({
offset: range(
data.startOffset + deltaY,
-(data.count * data.itemHeight),
data.itemHeight
)
});
},
onTouchEnd() {
const { data } = this;
if (data.offset !== data.startOffset) {
this.set({
duration: DEFAULT_DURATION
});
const index = range(
Math.round(-data.offset / data.itemHeight),
0,
data.count - 1
);
this.setIndex(index, true);
}
},
onClickItem(event: Weapp.Event) {
const { index } = event.currentTarget.dataset;
this.setIndex(index, true);
},
adjustIndex(index: number) {
const { data } = this;
index = range(index, 0, data.count);
for (let i = index; i < data.count; i++) {
if (!this.isDisabled(data.options[i])) return i;
}
for (let i = index - 1; i >= 0; i--) {
if (!this.isDisabled(data.options[i])) return i;
}
},
isDisabled(option: any) {
return isObj(option) && option.disabled;
},
getOptionText(option: any) {
const { data } = this;
return isObj(option) && data.valueKey in option
? option[data.valueKey]
: option;
},
setIndex(index: number, userAction: boolean) {
const { data } = this;
index = this.adjustIndex(index) || 0;
this.set({
offset: -index * data.itemHeight
});
if (index !== data.currentIndex) {
this.set({
currentIndex: index
});
userAction && this.$emit('change', index);
}
},
setValue(value: string) {
const { options } = this.data;
for (let i = 0; i < options.length; i++) {
if (this.getOptionText(options[i]) === value) {
return this.setIndex(i);
}
}
},
getValue() {
const { data } = this;
return data.options[data.currentIndex];
}
}
});