import { ref, computed, PropType, InjectionKey, CSSProperties, defineComponent, ExtractPropTypes, } from 'vue'; // Utils import { isDef, truthProp, createNamespace, ComponentInstance } from '../utils'; // Composables import { useRect, useChildren, useClickAway, useScrollParent, useEventListener, } from '@vant/use'; // Types import type { DropdownMenuProvide, DropdownMenuDirection } from './types'; const [name, bem] = createNamespace('dropdown-menu'); const props = { overlay: truthProp, zIndex: [Number, String], activeColor: String, closeOnClickOutside: truthProp, closeOnClickOverlay: truthProp, duration: { type: [Number, String], default: 0.2, }, direction: { type: String as PropType, default: 'down', }, }; export type DropdownMenuProps = ExtractPropTypes; export const DROPDOWN_KEY: InjectionKey = Symbol(name); export default defineComponent({ name, props, setup(props, { slots }) { const root = ref(); const barRef = ref(); const offset = ref(0); const { children, linkChildren } = useChildren(DROPDOWN_KEY); const scrollParent = useScrollParent(root); const opened = computed(() => children.some((item) => item.state.showWrapper) ); const barStyle = computed(() => { if (opened.value && isDef(props.zIndex)) { return { zIndex: +props.zIndex + 1, }; } }); const onClickAway = () => { if (props.closeOnClickOutside) { children.forEach((item) => { item.toggle(false); }); } }; const updateOffset = () => { if (barRef.value) { const rect = useRect(barRef); if (props.direction === 'down') { offset.value = rect.bottom; } else { offset.value = window.innerHeight - rect.top; } } }; const onScroll = () => { if (opened.value) { updateOffset(); } }; const toggleItem = (active: number) => { children.forEach((item, index) => { if (index === active) { updateOffset(); item.toggle(); } else if (item.state.showPopup) { item.toggle(false, { immediate: true }); } }); }; const renderTitle = (item: ComponentInstance, index: number) => { const { showPopup } = item.state; const { disabled, titleClass } = item; return (
{ if (!disabled) { toggleItem(index); } }} >
{item.renderTitle()}
); }; linkChildren({ props, offset }); useClickAway(root, onClickAway); useEventListener('scroll', onScroll, { target: scrollParent }); return () => (
{children.map(renderTitle)}
{slots.default?.()}
); }, });