[bugfix] Picker: 回滚movable-view重构

This commit is contained in:
rex 2019-04-23 17:42:19 +08:00 committed by GitHub
parent 66e2233e54
commit faf43f3448
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 132 additions and 96 deletions

View File

@ -1,22 +1,25 @@
import Page from '../../common/page'; import Page from '../../common/page';
import Toast from '../../dist/toast/toast'; import Toast from '../../dist/toast/toast';
const citys = {
'浙江': ['杭州', { text: '宁波', disabled: true }, '温州', '嘉兴', '湖州'],
'福建': ['福州', '厦门', '莆田', '三明', '泉州']
};
Page({ Page({
data: { data: {
column1: ['杭州', '宁波', '温州', '嘉兴', '湖州'], column1: ['杭州', '宁波', '温州', '嘉兴', '湖州'],
column2: citys['浙江'], column2: [
{ text: '杭州', disabled: true },
{ text: '宁波' },
{ text: '温州' }
],
column3: {
浙江: ['杭州', '宁波', '温州', '嘉兴', '湖州'],
福建: ['福州', '厦门', '莆田', '三明', '泉州']
},
column4: [ column4: [
{ {
values: Object.keys(citys), values: ['浙江', '福建'],
className: 'column1' className: 'column1'
}, },
{ {
values: citys['浙江'], values: ['杭州', '宁波', '温州', '嘉兴', '湖州'],
className: 'column2', className: 'column2',
defaultIndex: 2 defaultIndex: 2
} }
@ -39,6 +42,7 @@ Page({
onChange2(event) { onChange2(event) {
const { picker, value } = event.detail; const { picker, value } = event.detail;
picker.setColumnValues(1, citys[value[0]]); picker.setColumnValues(1, this.data.column3[value[0]]);
getApp().picker = picker;
} }
}); });

View File

@ -1,12 +1,7 @@
@import '../common/style/var'; @import '../common/style/var';
:host {
overflow: hidden;
}
.van-picker-column { .van-picker-column {
width: 100%; overflow: hidden;
overflow: visible;
font-size: 16px; font-size: 16px;
text-align: center; text-align: center;

View File

@ -1,13 +1,7 @@
import { VantComponent } from '../common/component'; import { VantComponent } from '../common/component';
import { isObj, range } from '../common/utils'; import { isObj, range } from '../common/utils';
function isDisabled(option: any) { const DEFAULT_DURATION = 200;
return isObj(option) && option.disabled;
}
function getOptionText(option: any, valueKey: string) {
return isObj(option) && valueKey in option ? option[valueKey] : option;
}
VantComponent({ VantComponent({
classes: ['active-class'], classes: ['active-class'],
@ -23,18 +17,17 @@ VantComponent({
}, },
defaultIndex: { defaultIndex: {
type: Number, type: Number,
value: 0, value: 0
observer(value) {
this.setIndex(value);
}
} }
}, },
data: { data: {
startY: 0,
offset: 0, offset: 0,
duration: 0,
startOffset: 0,
options: [], options: [],
currentIndex: 0, currentIndex: 0
animation: false
}, },
created() { created() {
@ -43,38 +36,71 @@ VantComponent({
this.set({ this.set({
currentIndex: defaultIndex, currentIndex: defaultIndex,
options: initialOptions options: initialOptions
}).then(() => this.setIndex(defaultIndex)); }).then(() => {
this.setIndex(defaultIndex);
});
},
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: { methods: {
onChange(event: Weapp.Event) { onTouchStart(event: Weapp.TouchEvent) {
if (!event.detail.source) { this.set({
return; startY: event.touches[0].clientY,
} startOffset: this.data.offset,
duration: 0
this.offset = event.detail.y; });
}, },
onTouchStart() { onTouchMove(event: Weapp.TouchEvent) {
// open animate at first touch const { data } = this;
if (!this.data.animation) { const deltaY = event.touches[0].clientY - data.startY;
this.set({ animation: true }); this.set({
} offset: range(
data.startOffset + deltaY,
-(data.count * data.itemHeight),
data.itemHeight
)
});
}, },
onTouchEnd() { onTouchEnd() {
const { options = [], itemHeight, offset } = this.data; const { data } = this;
if (data.offset !== data.startOffset) {
if (this.offset === offset) { this.set({
return; duration: DEFAULT_DURATION
});
const index = range(
Math.round(-data.offset / data.itemHeight),
0,
data.count - 1
);
this.setIndex(index, true);
} }
const index = range(
Math.round(-this.offset / itemHeight),
0,
options.length - 1
);
this.setIndex(index, true);
}, },
onClickItem(event: Weapp.Event) { onClickItem(event: Weapp.Event) {
@ -83,27 +109,33 @@ VantComponent({
}, },
adjustIndex(index: number) { adjustIndex(index: number) {
const { options = [] } = this.data; const { data } = this;
const count = options.length; index = range(index, 0, data.count);
index = range(index, 0, count); for (let i = index; i < data.count; i++) {
if (!this.isDisabled(data.options[i])) return i;
for (let i = index; i < count; i++) {
if (!isDisabled(options[i])) return i;
} }
for (let i = index - 1; i >= 0; i--) { for (let i = index - 1; i >= 0; i--) {
if (!isDisabled(options[i])) return i; if (!this.isDisabled(data.options[i])) return i;
} }
return 0; },
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) { setIndex(index: number, userAction: boolean) {
const { itemHeight, currentIndex } = this.data; const { data } = this;
index = this.adjustIndex(index); index = this.adjustIndex(index) || 0;
const offset = -index * itemHeight; const offset = -index * data.itemHeight;
this.offset = offset; if (index !== data.currentIndex) {
if (index !== currentIndex) {
return this.set({ offset, currentIndex: index }).then(() => { return this.set({ offset, currentIndex: index }).then(() => {
userAction && this.$emit('change', index); userAction && this.$emit('change', index);
}); });
@ -113,17 +145,18 @@ VantComponent({
}, },
setValue(value: string) { setValue(value: string) {
const { options = [], valueKey } = this.data; const { options } = this.data;
for (let i = 0; i < options.length; i++) {
const index = options.findIndex( if (this.getOptionText(options[i]) === value) {
(item: any) => getOptionText(item, valueKey) === value return this.setIndex(i);
); }
return index !== -1 ? this.setIndex(index) : Promise.resolve(); }
return Promise.resolve();
}, },
getValue() { getValue() {
const { options = [], currentIndex } = this.data; const { data } = this;
return options[currentIndex]; return data.options[data.currentIndex];
} }
} }
}); });

View File

@ -1,27 +1,31 @@
<wxs src="../wxs/utils.wxs" module="utils" /> <view
<movable-area
class="van-picker-column custom-class" class="van-picker-column custom-class"
style="height: {{ itemHeight }}px; top: {{ itemHeight * (visibleItemCount - 1) / 2 }}px;" style="height: {{ itemHeight * visibleItemCount }}px"
bind:touchstart="onTouchStart"
catch:touchmove="onTouchMove"
bind:touchend="onTouchEnd"
bind:touchcancel="onTouchEnd"
> >
<movable-view <view style="{{ wrapperStyle }}">
direction="vertical"
out-of-bounds
damping="{{ 50 }}"
y="{{ offset }}"
animation="{{ animation }}"
style="line-height: {{ itemHeight }}px; width: 100%; height: {{ itemHeight * options.length }}px;"
bindchange="onChange"
bindtouchstart="onTouchStart"
bindtouchend="onTouchEnd"
>
<view <view
wx:for="{{ options }}" wx:for="{{ options }}"
wx:for-item="option"
wx:key="index" wx:key="index"
data-index="{{ index }}" data-index="{{ index }}"
style="height: {{ itemHeight }}px" style="height: {{ itemHeight }}px"
class="van-ellipsis {{ utils.bem('picker-column__item', { disabled: item && item.disabled, selected: index === currentIndex }) }} {{ index === currentIndex ? 'active-class' : '' }}" class="van-ellipsis van-picker-column__item {{ option && option.disabled ? 'van-picker-column__item--disabled' : '' }} {{ index === currentIndex ? 'van-picker-column__item--selected active-class' : '' }}"
bindtap="onClickItem" bindtap="onClickItem"
>{{ item && item[valueKey] ? item[valueKey] : item }}</view> >{{ getOptionText(option, valueKey) }}</view>
</movable-view> </view>
</movable-area> </view>
<wxs module="getOptionText">
function isObj(x) {
var type = typeof x;
return x !== null && (type === 'object' || type === 'function');
}
module.exports = function (option, valueKey) {
return isObj(option) && option[valueKey] ? option[valueKey] : option;
}
</wxs>

View File

@ -117,7 +117,7 @@ Page({
```javascript ```javascript
const citys = { const citys = {
'浙江': ['杭州', { text: '宁波', disabled: true }, '温州', '嘉兴', '湖州'], '浙江': ['杭州', '宁波', '温州', '嘉兴', '湖州'],
'福建': ['福州', '厦门', '莆田', '三明', '泉州'] '福建': ['福州', '厦门', '莆田', '三明', '泉州']
}; };

View File

@ -32,17 +32,17 @@
catch:touchmove="noop" catch:touchmove="noop"
> >
<picker-column <picker-column
class="van-picker__column"
wx:for="{{ isSimple(columns) ? [columns] : columns }}" wx:for="{{ isSimple(columns) ? [columns] : columns }}"
wx:key="{{ index }}" wx:key="{{ index }}"
class="van-picker__column" data-index="{{ index }}"
custom-class="column-class"
value-key="{{ valueKey }}" value-key="{{ valueKey }}"
initial-options="{{ isSimple(columns) ? item : item.values }}" initial-options="{{ isSimple(columns) ? item : item.values }}"
default-index="{{ item.defaultIndex || defaultIndex }}" default-index="{{ item.defaultIndex || defaultIndex }}"
item-height="{{ itemHeight }}" item-height="{{ itemHeight }}"
visible-item-count="{{ visibleItemCount }}" visible-item-count="{{ visibleItemCount }}"
custom-class="column-class"
active-class="active-class" active-class="active-class"
data-index="{{ index }}"
bind:change="onChange" bind:change="onChange"
/> />
<view <view