import { computed, getCurrentInstance, watch, onMounted } from 'vue' import { isFunction } from '@vue/shared' import { isClient } from '@vueuse/core' import { isBool } from '@element-plus/utils/util' import { buildProp, definePropType } from '@element-plus/utils/props' import type { RouteLocationNormalizedLoaded } from 'vue-router' import type { Ref, ComponentPublicInstance, ExtractPropTypes } from 'vue' export const createModelToggleComposable = (name: string) => { const useModelToggleProps = { [name]: buildProp({ type: definePropType(Boolean), default: null, } as const), [`onUpdate:${name}`]: buildProp({ type: definePropType<(val: boolean) => void>(Function), } as const), } const useModelToggleEmits = [`update:${name}`] const useModelToggle = ({ indicator, shouldHideWhenRouteChanges, shouldProceed, onShow, onHide, }: ModelToggleParams) => { const instance = getCurrentInstance()! const props = instance.props as UseModelToggleProps & { disabled: boolean } const { emit } = instance const updateEventKey = `update:${name}` const hasUpdateHandler = computed(() => isFunction(props[`onUpdate:${name}`]) ) // when it matches the default value we say this is absent // though this could be mistakenly passed from the user but we need to rule out that // condition const isModelBindingAbsent = computed(() => props[name] === null) const doShow = () => { if (indicator.value === true) { return } indicator.value = true if (isFunction(onShow)) { onShow() } } const doHide = () => { if (indicator.value === false) { return } indicator.value = false if (isFunction(onHide)) { onHide() } } const show = () => { if ( props.disabled === true || (isFunction(shouldProceed) && !shouldProceed()) ) return const shouldEmit = hasUpdateHandler.value && isClient if (shouldEmit) { emit(updateEventKey, true) } if (isModelBindingAbsent.value || !shouldEmit) { doShow() } } const hide = () => { if (props.disabled === true || !isClient) return const shouldEmit = hasUpdateHandler.value && isClient if (shouldEmit) { emit(updateEventKey, false) } if (isModelBindingAbsent.value || !shouldEmit) { doHide() } } const onChange = (val: boolean) => { if (!isBool(val)) return if (props.disabled && val) { if (hasUpdateHandler.value) { emit(updateEventKey, false) } } else if (indicator.value !== val) { if (val) { doShow() } else { doHide() } } } const toggle = () => { if (indicator.value) { hide() } else { show() } } watch(() => props[name], onChange as any) if ( shouldHideWhenRouteChanges && instance.appContext.config.globalProperties.$route !== undefined ) { watch( () => ({ ...( instance.proxy as ComponentPublicInstance<{ $route: RouteLocationNormalizedLoaded }> ).$route, }), () => { if (shouldHideWhenRouteChanges.value && indicator.value) { hide() } } ) } onMounted(() => { onChange(props[name] as boolean) }) return { hide, show, toggle, } } return { useModelToggle, useModelToggleProps, useModelToggleEmits, } } const { useModelToggle, useModelToggleProps, useModelToggleEmits } = createModelToggleComposable('modelValue') export { useModelToggle, useModelToggleEmits, useModelToggleProps } export type UseModelToggleProps = ExtractPropTypes export type ModelToggleParams = { indicator: Ref shouldHideWhenRouteChanges?: Ref shouldProceed?: () => boolean onShow?: () => void onHide?: () => void }