mirror of
https://gitee.com/vant-contrib/vant-weapp.git
synced 2025-04-06 03:58:05 +08:00
[bugfix] Picker: 回滚movable-view重构
This commit is contained in:
parent
66e2233e54
commit
faf43f3448
@ -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;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -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;
|
||||||
|
|
||||||
|
@ -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];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -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>
|
||||||
|
@ -117,7 +117,7 @@ Page({
|
|||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
const citys = {
|
const citys = {
|
||||||
'浙江': ['杭州', { text: '宁波', disabled: true }, '温州', '嘉兴', '湖州'],
|
'浙江': ['杭州', '宁波', '温州', '嘉兴', '湖州'],
|
||||||
'福建': ['福州', '厦门', '莆田', '三明', '泉州']
|
'福建': ['福州', '厦门', '莆田', '三明', '泉州']
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -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
|
||||||
|
Loading…
x
Reference in New Issue
Block a user