vant/src/popover/Popover.tsx
2021-08-07 16:22:02 +08:00

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>
</>
);
},
});