mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-05 19:41:42 +08:00
243 lines
5.4 KiB
TypeScript
243 lines
5.4 KiB
TypeScript
import {
|
|
ref,
|
|
watch,
|
|
nextTick,
|
|
PropType,
|
|
onMounted,
|
|
CSSProperties,
|
|
TeleportProps,
|
|
onBeforeUnmount,
|
|
defineComponent,
|
|
} from 'vue';
|
|
import { Instance, createPopper, offsetModifier } from '@vant/popperjs';
|
|
|
|
// Utils
|
|
import {
|
|
pick,
|
|
extend,
|
|
truthProp,
|
|
unknownProp,
|
|
BORDER_BOTTOM,
|
|
createNamespace,
|
|
ComponentInstance,
|
|
} from '../utils';
|
|
|
|
// Composables
|
|
import { useClickAway } from '@vant/use';
|
|
|
|
// Components
|
|
import { Icon } from '../icon';
|
|
import { Popup } from '../popup';
|
|
|
|
const [name, bem] = createNamespace('popover');
|
|
|
|
const popupProps = [
|
|
'show',
|
|
'overlay',
|
|
'duration',
|
|
'teleport',
|
|
'overlayStyle',
|
|
'overlayClass',
|
|
'closeOnClickOverlay',
|
|
] as const;
|
|
|
|
export type PopoverTheme = 'light' | 'dark';
|
|
export type PopoverTrigger = 'manual' | 'click';
|
|
export type PopoverPlacement =
|
|
| 'top'
|
|
| 'top-start'
|
|
| 'top-end'
|
|
| 'left'
|
|
| 'left-start'
|
|
| 'left-end'
|
|
| 'right'
|
|
| 'right-start'
|
|
| 'right-end'
|
|
| 'bottom'
|
|
| 'bottom-start'
|
|
| 'bottom-end';
|
|
|
|
export type PopoverAction = {
|
|
text: string;
|
|
icon?: string;
|
|
color?: string;
|
|
disabled?: boolean;
|
|
className?: string;
|
|
};
|
|
|
|
export default defineComponent({
|
|
name,
|
|
|
|
props: {
|
|
show: Boolean,
|
|
overlay: Boolean,
|
|
duration: [Number, String],
|
|
iconPrefix: String,
|
|
overlayClass: unknownProp,
|
|
overlayStyle: Object as PropType<CSSProperties>,
|
|
closeOnClickAction: truthProp,
|
|
closeOnClickOverlay: truthProp,
|
|
closeOnClickOutside: truthProp,
|
|
offset: {
|
|
type: (Array as unknown) as PropType<[number, number]>,
|
|
default: () => [0, 8],
|
|
},
|
|
theme: {
|
|
type: String as PropType<PopoverTheme>,
|
|
default: 'light',
|
|
},
|
|
trigger: {
|
|
type: String as PropType<PopoverTrigger>,
|
|
default: 'click',
|
|
},
|
|
actions: {
|
|
type: Array as PropType<PopoverAction[]>,
|
|
default: () => [],
|
|
},
|
|
placement: {
|
|
type: String as PropType<PopoverPlacement>,
|
|
default: 'bottom',
|
|
},
|
|
teleport: {
|
|
type: [String, Object] as PropType<TeleportProps['to']>,
|
|
default: 'body',
|
|
},
|
|
},
|
|
|
|
emits: ['select', 'touchstart', 'update:show'],
|
|
|
|
setup(props, { emit, slots, attrs }) {
|
|
let popper: Instance | null;
|
|
|
|
const wrapperRef = ref<HTMLElement>();
|
|
const popoverRef = ref<ComponentInstance>();
|
|
|
|
const createPopperInstance = () => {
|
|
return createPopper(wrapperRef.value!, popoverRef.value!.popupRef.value, {
|
|
placement: props.placement,
|
|
modifiers: [
|
|
{
|
|
name: 'computeStyles',
|
|
options: {
|
|
adaptive: false,
|
|
gpuAcceleration: false,
|
|
},
|
|
},
|
|
extend({}, offsetModifier, {
|
|
options: {
|
|
offset: props.offset,
|
|
},
|
|
}),
|
|
],
|
|
});
|
|
};
|
|
|
|
const updateLocation = () => {
|
|
nextTick(() => {
|
|
if (!props.show) {
|
|
return;
|
|
}
|
|
|
|
if (!popper) {
|
|
popper = createPopperInstance();
|
|
} else {
|
|
popper.setOptions({
|
|
placement: props.placement,
|
|
});
|
|
}
|
|
});
|
|
};
|
|
|
|
const updateShow = (value: boolean) => emit('update:show', value);
|
|
|
|
const onClickWrapper = () => {
|
|
if (props.trigger === 'click') {
|
|
updateShow(!props.show);
|
|
}
|
|
};
|
|
|
|
const onTouchstart = (event: TouchEvent) => {
|
|
event.stopPropagation();
|
|
emit('touchstart', event);
|
|
};
|
|
|
|
const onClickAction = (action: PopoverAction, index: number) => {
|
|
if (action.disabled) {
|
|
return;
|
|
}
|
|
|
|
emit('select', action, index);
|
|
|
|
if (props.closeOnClickAction) {
|
|
updateShow(false);
|
|
}
|
|
};
|
|
|
|
const onClickAway = () => {
|
|
if (
|
|
props.closeOnClickOutside &&
|
|
(!props.overlay || props.closeOnClickOverlay)
|
|
) {
|
|
updateShow(false);
|
|
}
|
|
};
|
|
|
|
const renderAction = (action: PopoverAction, index: number) => {
|
|
const { icon, text, color, disabled, className } = action;
|
|
return (
|
|
<div
|
|
role="menuitem"
|
|
class={[bem('action', { disabled, 'with-icon': icon }), className]}
|
|
style={{ color }}
|
|
onClick={() => onClickAction(action, index)}
|
|
>
|
|
{icon && (
|
|
<Icon
|
|
name={icon}
|
|
classPrefix={props.iconPrefix}
|
|
class={bem('action-icon')}
|
|
/>
|
|
)}
|
|
<div class={[bem('action-text'), BORDER_BOTTOM]}>{text}</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
onMounted(updateLocation);
|
|
onBeforeUnmount(() => {
|
|
if (popper) {
|
|
popper.destroy();
|
|
popper = null;
|
|
}
|
|
});
|
|
|
|
watch(() => [props.show, props.placement], updateLocation);
|
|
|
|
useClickAway(wrapperRef, onClickAway, { eventName: 'touchstart' });
|
|
|
|
return () => (
|
|
<>
|
|
<span ref={wrapperRef} class={bem('wrapper')} onClick={onClickWrapper}>
|
|
{slots.reference?.()}
|
|
</span>
|
|
<Popup
|
|
ref={popoverRef}
|
|
class={bem([props.theme])}
|
|
position={''}
|
|
transition="van-popover-zoom"
|
|
lockScroll={false}
|
|
onTouchstart={onTouchstart}
|
|
{...attrs}
|
|
{...pick(props, popupProps)}
|
|
{...{ 'onUpdate:show': updateShow }}
|
|
>
|
|
<div class={bem('arrow')} />
|
|
<div role="menu" class={bem('content')}>
|
|
{slots.default ? slots.default() : props.actions.map(renderAction)}
|
|
</div>
|
|
</Popup>
|
|
</>
|
|
);
|
|
},
|
|
});
|