chore(PIcker): split PickerToolbar component (#10982)

This commit is contained in:
neverland 2022-08-28 10:43:14 +08:00 committed by GitHub
parent 8bcb34724f
commit c25acb8d46
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 114 additions and 65 deletions

View File

@ -9,6 +9,7 @@ import {
// Utils // Utils
import { import {
pick,
extend, extend,
unitToPx, unitToPx,
truthProp, truthProp,
@ -17,18 +18,18 @@ import {
preventDefault, preventDefault,
makeStringProp, makeStringProp,
makeNumericProp, makeNumericProp,
createNamespace,
HAPTICS_FEEDBACK,
BORDER_UNSET_TOP_BOTTOM, BORDER_UNSET_TOP_BOTTOM,
type Numeric, type Numeric,
} from '../utils'; } from '../utils';
import { import {
bem,
name,
isOptionExist, isOptionExist,
getColumnsType, getColumnsType,
findOptionByValue, findOptionByValue,
assignDefaultFields,
formatCascadeColumns, formatCascadeColumns,
getFirstEnabledOption, getFirstEnabledOption,
assignDefaultFields,
} from './utils'; } from './utils';
// Composables // Composables
@ -38,6 +39,11 @@ import { useExpose } from '../composables/use-expose';
// Components // Components
import { Loading } from '../loading'; import { Loading } from '../loading';
import Column, { PICKER_KEY } from './PickerColumn'; import Column, { PICKER_KEY } from './PickerColumn';
import Toolbar, {
pickerToolbarPropKeys,
pickerToolbarProps,
pickerToolbarSlots,
} from './PickerToolbar';
// Types // Types
import type { import type {
@ -48,10 +54,8 @@ import type {
PickerToolbarPosition, PickerToolbarPosition,
} from './types'; } from './types';
const [name, bem, t] = createNamespace('picker'); export const pickerSharedProps = extend(
{
export const pickerSharedProps = {
title: String,
loading: Boolean, loading: Boolean,
readonly: Boolean, readonly: Boolean,
allowHtml: Boolean, allowHtml: Boolean,
@ -59,9 +63,9 @@ export const pickerSharedProps = {
showToolbar: truthProp, showToolbar: truthProp,
swipeDuration: makeNumericProp(1000), swipeDuration: makeNumericProp(1000),
visibleOptionNum: makeNumericProp(6), visibleOptionNum: makeNumericProp(6),
cancelButtonText: String, },
confirmButtonText: String, pickerToolbarProps
}; );
const pickerProps = extend({}, pickerSharedProps, { const pickerProps = extend({}, pickerSharedProps, {
columns: makeArrayProp<PickerOption | PickerColumn>(), columns: makeArrayProp<PickerOption | PickerColumn>(),
@ -167,53 +171,6 @@ export default defineComponent({
selectedOptions: selectedOptions.value, selectedOptions: selectedOptions.value,
}); });
const renderTitle = () => {
if (slots.title) {
return slots.title();
}
if (props.title) {
return <div class={[bem('title'), 'van-ellipsis']}>{props.title}</div>;
}
};
const renderCancel = () => {
const text = props.cancelButtonText || t('cancel');
return (
<button
type="button"
class={[bem('cancel'), HAPTICS_FEEDBACK]}
onClick={cancel}
>
{slots.cancel ? slots.cancel() : text}
</button>
);
};
const renderConfirm = () => {
const text = props.confirmButtonText || t('confirm');
return (
<button
type="button"
class={[bem('confirm'), HAPTICS_FEEDBACK]}
onClick={confirm}
>
{slots.confirm ? slots.confirm() : text}
</button>
);
};
const renderToolbar = () => {
if (props.showToolbar) {
return (
<div class={bem('toolbar')}>
{slots.toolbar
? slots.toolbar()
: [renderCancel(), renderTitle(), renderConfirm()]}
</div>
);
}
};
const renderColumnItems = () => const renderColumnItems = () =>
currentColumns.value.map((options, columnIndex) => ( currentColumns.value.map((options, columnIndex) => (
<Column <Column
@ -260,6 +217,19 @@ export default defineComponent({
); );
}; };
const renderToolbar = () => {
if (props.showToolbar) {
return (
<Toolbar
v-slots={pick(slots, pickerToolbarSlots)}
{...pick(props, pickerToolbarPropKeys)}
onConfirm={confirm}
onCancel={cancel}
/>
);
}
};
watch( watch(
currentColumns, currentColumns,
(columns) => { (columns) => {

View File

@ -0,0 +1,75 @@
import { defineComponent } from 'vue';
import { bem, t } from './utils';
import { createNamespace, HAPTICS_FEEDBACK } from '../utils';
const [name] = createNamespace('picker-toolbar');
export const pickerToolbarProps = {
title: String,
cancelButtonText: String,
confirmButtonText: String,
};
export const pickerToolbarSlots = ['cancel', 'confirm', 'title', 'toolbar'];
export type PickerToolbarPropKeys = Array<keyof typeof pickerToolbarProps>;
export const pickerToolbarPropKeys = Object.keys(
pickerToolbarProps
) as PickerToolbarPropKeys;
export default defineComponent({
name,
props: pickerToolbarProps,
emits: ['confirm', 'cancel'],
setup(props, { emit, slots }) {
const renderTitle = () => {
if (slots.title) {
return slots.title();
}
if (props.title) {
return <div class={[bem('title'), 'van-ellipsis']}>{props.title}</div>;
}
};
const onCancel = () => emit('cancel');
const onConfirm = () => emit('confirm');
const renderCancel = () => {
const text = props.cancelButtonText || t('cancel');
return (
<button
type="button"
class={[bem('cancel'), HAPTICS_FEEDBACK]}
onClick={onCancel}
>
{slots.cancel ? slots.cancel() : text}
</button>
);
};
const renderConfirm = () => {
const text = props.confirmButtonText || t('confirm');
return (
<button
type="button"
class={[bem('confirm'), HAPTICS_FEEDBACK]}
onClick={onConfirm}
>
{slots.confirm ? slots.confirm() : text}
</button>
);
};
return () => (
<div class={bem('toolbar')}>
{slots.toolbar
? slots.toolbar()
: [renderCancel(), renderTitle(), renderConfirm()]}
</div>
);
},
});

View File

@ -1,7 +1,11 @@
import { isDef, clamp, extend, type Numeric } from '../utils'; import { isDef, clamp, extend, createNamespace, type Numeric } from '../utils';
import type { Ref } from 'vue'; import type { Ref } from 'vue';
import type { PickerOption, PickerColumn, PickerFieldNames } from './types'; import type { PickerOption, PickerColumn, PickerFieldNames } from './types';
const [name, bem, t] = createNamespace('picker');
export { name, bem, t };
export const getFirstEnabledOption = ( export const getFirstEnabledOption = (
options: PickerOption[] options: PickerOption[]
): PickerOption | undefined => ): PickerOption | undefined =>