ant-design-vue/components/vc-dropdown/Dropdown.tsx

147 lines
4.5 KiB
Vue

import type { CSSProperties, PropType } from 'vue';
import { Fragment, computed, defineComponent, ref, watch } from 'vue';
import PropTypes from '../_util/vue-types';
import Trigger from '../vc-trigger';
import placements from './placements';
import { cloneElement } from '../_util/vnode';
import classNames from '../_util/classNames';
import { skipFlattenKey } from '../_util/props-util';
export default defineComponent({
compatConfig: { MODE: 3 },
props: {
minOverlayWidthMatchTrigger: { type: Boolean, default: undefined },
arrow: { type: Boolean, default: false },
prefixCls: PropTypes.string.def('rc-dropdown'),
transitionName: String,
overlayClassName: PropTypes.string.def(''),
openClassName: String,
animation: PropTypes.any,
align: PropTypes.object,
overlayStyle: { type: Object as PropType<CSSProperties>, default: undefined as CSSProperties },
placement: PropTypes.string.def('bottomLeft'),
overlay: PropTypes.any,
trigger: PropTypes.oneOfType([PropTypes.string, PropTypes.arrayOf(PropTypes.string)]).def(
'hover',
),
alignPoint: { type: Boolean, default: undefined },
showAction: PropTypes.array,
hideAction: PropTypes.array,
getPopupContainer: Function,
visible: { type: Boolean, default: undefined },
defaultVisible: { type: Boolean, default: false },
mouseEnterDelay: PropTypes.number.def(0.15),
mouseLeaveDelay: PropTypes.number.def(0.1),
},
emits: ['visibleChange', 'overlayClick'],
slots: ['overlay'],
setup(props, { slots, emit, expose }) {
const triggerVisible = ref(!!props.visible);
watch(
() => props.visible,
val => {
if (val !== undefined) {
triggerVisible.value = val;
}
},
);
const triggerRef = ref();
expose({
triggerRef,
});
const onClick = (e: MouseEvent) => {
if (props.visible === undefined) {
triggerVisible.value = false;
}
emit('overlayClick', e);
};
const onVisibleChange = (visible: boolean) => {
if (props.visible === undefined) {
triggerVisible.value = visible;
}
emit('visibleChange', visible);
};
const getMenuElement = () => {
const overlayElement = slots.overlay?.();
const extraOverlayProps = {
prefixCls: `${props.prefixCls}-menu`,
onClick,
getPopupContainer: () => triggerRef.value.getPopupDomNode(),
};
return (
<Fragment key={skipFlattenKey}>
{props.arrow && <div class={`${props.prefixCls}-arrow`} />}
{cloneElement(overlayElement, extraOverlayProps, false)}
</Fragment>
);
};
const minOverlayWidthMatchTrigger = computed(() => {
const { minOverlayWidthMatchTrigger: matchTrigger = !props.alignPoint } = props;
return matchTrigger;
});
const renderChildren = () => {
const children = slots.default?.();
return triggerVisible.value && children
? cloneElement(
children[0],
{ class: props.openClassName || `${props.prefixCls}-open` },
false,
)
: children;
};
const triggerHideAction = computed(() => {
if (!props.hideAction && props.trigger.indexOf('contextmenu') !== -1) {
return ['click'];
}
return props.hideAction;
});
return () => {
const {
prefixCls,
arrow,
showAction,
overlayStyle,
trigger,
placement,
align,
getPopupContainer,
transitionName,
animation,
overlayClassName,
...otherProps
} = props;
return (
<Trigger
{...otherProps}
prefixCls={prefixCls}
ref={triggerRef}
popupClassName={classNames(overlayClassName, {
[`${prefixCls}-show-arrow`]: arrow,
})}
popupStyle={overlayStyle}
builtinPlacements={placements}
action={trigger}
showAction={showAction}
hideAction={triggerHideAction.value || []}
popupPlacement={placement}
popupAlign={align}
popupTransitionName={transitionName}
popupAnimation={animation}
popupVisible={triggerVisible.value}
stretch={minOverlayWidthMatchTrigger.value ? 'minWidth' : ''}
onPopupVisibleChange={onVisibleChange}
getPopupContainer={getPopupContainer}
v-slots={{ popup: getMenuElement, default: renderChildren }}
></Trigger>
);
};
},
});