feat(Picker): add scroll-into event (#11757)

This commit is contained in:
neverland 2023-04-16 21:25:49 +08:00 committed by GitHub
parent 9e1640180b
commit 5e4bbab00c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 71 additions and 11 deletions

View File

@ -83,7 +83,14 @@ export default defineComponent({
props: pickerProps, props: pickerProps,
emits: ['confirm', 'cancel', 'change', 'clickOption', 'update:modelValue'], emits: [
'confirm',
'cancel',
'change',
'scrollInto',
'clickOption',
'update:modelValue',
],
setup(props, { emit, slots }) { setup(props, { emit, slots }) {
const columnsRef = ref<HTMLElement>(); const columnsRef = ref<HTMLElement>();
@ -165,11 +172,14 @@ export default defineComponent({
}); });
}; };
const onClickOption = (currentOption: PickerOption, columnIndex: number) => const onClickOption = (
emit( currentOption: PickerOption,
'clickOption', columnIndex: number
extend({ columnIndex, currentOption }, getEventParams()) ) => {
); const params = { columnIndex, currentOption };
emit('clickOption', extend(getEventParams(), params));
emit('scrollInto', params);
};
const confirm = () => { const confirm = () => {
children.forEach((child) => child.stopMomentum()); children.forEach((child) => child.stopMomentum());
@ -202,6 +212,12 @@ export default defineComponent({
onClickOption={(option: PickerOption) => onClickOption={(option: PickerOption) =>
onClickOption(option, columnIndex) onClickOption(option, columnIndex)
} }
onScrollInto={(option: PickerOption) => {
emit('scrollInto', {
currentOption: option,
columnIndex,
});
}}
/> />
)); ));

View File

@ -1,5 +1,6 @@
import { import {
ref, ref,
computed,
watchEffect, watchEffect,
defineComponent, defineComponent,
type PropType, type PropType,
@ -56,7 +57,7 @@ export default defineComponent({
visibleOptionNum: makeRequiredProp(numericProp), visibleOptionNum: makeRequiredProp(numericProp),
}, },
emits: ['change', 'clickOption'], emits: ['change', 'clickOption', 'scrollInto'],
setup(props, { emit, slots }) { setup(props, { emit, slots }) {
let moving: boolean; let moving: boolean;
@ -113,6 +114,8 @@ export default defineComponent({
const getIndexByOffset = (offset: number) => const getIndexByOffset = (offset: number) =>
clamp(Math.round(-offset / props.optionHeight), 0, count() - 1); clamp(Math.round(-offset / props.optionHeight), 0, count() - 1);
const currentIndex = computed(() => getIndexByOffset(currentOffset.value));
const momentum = (distance: number, duration: number) => { const momentum = (distance: number, duration: number) => {
const speed = Math.abs(distance / duration); const speed = Math.abs(distance / duration);
@ -166,16 +169,23 @@ export default defineComponent({
preventDefault(event, true); preventDefault(event, true);
} }
currentOffset.value = clamp( const newOffset = clamp(
startOffset + touch.deltaY.value, startOffset + touch.deltaY.value,
-(count() * props.optionHeight), -(count() * props.optionHeight),
props.optionHeight props.optionHeight
); );
const newIndex = getIndexByOffset(newOffset);
if (newIndex !== currentIndex.value) {
emit('scrollInto', props.options[newIndex]);
}
currentOffset.value = newOffset;
const now = Date.now(); const now = Date.now();
if (now - touchStartTime > MOMENTUM_TIME) { if (now - touchStartTime > MOMENTUM_TIME) {
touchStartTime = now; touchStartTime = now;
momentumOffset = currentOffset.value; momentumOffset = newOffset;
} }
}; };

View File

@ -357,8 +357,9 @@ export default {
| --- | --- | --- | | --- | --- | --- |
| confirm | Emitted when the confirm button is clicked | _{ selectedValues, selectedOptions, selectedIndexes }_ | | confirm | Emitted when the confirm button is clicked | _{ selectedValues, selectedOptions, selectedIndexes }_ |
| cancel | Emitted when the cancel button is clicked | _{ selectedValues, selectedOptions, selectedIndexes }_ | | cancel | Emitted when the cancel button is clicked | _{ selectedValues, selectedOptions, selectedIndexes }_ |
| change | Emitted when current option is changed | _{ selectedValues, selectedOptions,selectedIndexes, columnIndex }_ | | change | Emitted when current selected option is changed | _{ selectedValues, selectedOptions,selectedIndexes, columnIndex }_ |
| click-option | Emitted when an option is clicked | _{ currentOption, selectedValues, selectedOptions, selectedIndexes, columnIndex }_ | | click-option | Emitted when an option is clicked | _{ currentOption, selectedValues, selectedOptions, selectedIndexes, columnIndex }_ |
| scroll-into `v4.2.1` | Emitted when an option is scrolled into the middle selection area by clicking or dragging | _{ currentOption, columnIndex }_ |
### Slots ### Slots

View File

@ -378,8 +378,9 @@ export default {
| --- | --- | --- | | --- | --- | --- |
| confirm | 点击完成按钮时触发 | _{ selectedValues, selectedOptions, selectedIndexes }_ | | confirm | 点击完成按钮时触发 | _{ selectedValues, selectedOptions, selectedIndexes }_ |
| cancel | 点击取消按钮时触发 | _{ selectedValues, selectedOptions, selectedIndexes }_ | | cancel | 点击取消按钮时触发 | _{ selectedValues, selectedOptions, selectedIndexes }_ |
| change | 选项改变时触发 | _{ selectedValues, selectedOptions, selectedIndexes, columnIndex }_ | | change | 选中的选项改变时触发 | _{ selectedValues, selectedOptions, selectedIndexes, columnIndex }_ |
| click-option | 点击选项时触发 | _{ currentOption, selectedValues, selectedOptions, selectedIndexes, columnIndex }_ | | click-option | 点击选项时触发 | _{ currentOption, selectedValues, selectedOptions, selectedIndexes, columnIndex }_ |
| scroll-into `v4.2.1` | 当用户通过点击或拖拽让一个选项滚动到中间的选择区域时触发 | _{ currentOption, columnIndex }_ |
### Slots ### Slots

View File

@ -143,6 +143,38 @@ test('should emit click-option event after clicking an option', async () => {
]); ]);
}); });
test('should emit scroll-into event after draging the column', async () => {
const wrapper = mount(Picker, {
props: {
columns: simpleColumn,
},
});
triggerDrag(wrapper.find('.van-picker-column'), 0, -100);
await wrapper.find('.van-picker-column ul').trigger('transitionend');
expect(wrapper.emitted('scrollInto')).toEqual([
[{ columnIndex: 0, currentOption: { text: '1992', value: '1992' } }],
]);
});
test('should emit scroll-into event after clicking an option', async () => {
const wrapper = mount(Picker, {
props: {
showToolbar: true,
columns: simpleColumn,
},
});
await wrapper.find('.van-picker-column__item').trigger('click');
expect(wrapper.emitted('scrollInto')![0]).toEqual([
{
columnIndex: 0,
currentOption: { text: '1990', value: '1990' },
},
]);
});
test('should render bottom toolbar when toolbar-position is bottom', () => { test('should render bottom toolbar when toolbar-position is bottom', () => {
const wrapper = mount(Picker, { const wrapper = mount(Picker, {
props: { props: {