element-plus/packages/hooks/use-model-toggle/index.ts
dopamine 8618a6bcd0
refactor: import shared utilities from @element-plus/utils (#18048)
chore: import shared utilities from `@element-plus/utils`
2024-08-29 09:04:03 +08:00

204 lines
4.7 KiB
TypeScript

import { computed, getCurrentInstance, onMounted, watch } from 'vue'
import {
buildProp,
definePropType,
isBoolean,
isClient,
isFunction,
} from '@element-plus/utils'
import type { ExtractPropType } from '@element-plus/utils'
import type { RouteLocationNormalizedLoaded } from 'vue-router'
import type { ComponentPublicInstance, ExtractPropTypes, Ref } from 'vue'
const _prop = buildProp({
type: definePropType<boolean | null>(Boolean),
default: null,
} as const)
const _event = buildProp({
type: definePropType<(val: boolean) => void>(Function),
} as const)
export type UseModelTogglePropsRaw<T extends string> = {
[K in T]: typeof _prop
} & {
[K in `onUpdate:${T}`]: typeof _event
}
export type UseModelTogglePropsGeneric<T extends string> = {
[K in T]: ExtractPropType<typeof _prop>
} & {
[K in `onUpdate:${T}`]: ExtractPropType<typeof _event>
}
export const createModelToggleComposable = <T extends string>(name: T) => {
const updateEventKey = `update:${name}` as const
const updateEventKeyRaw = `onUpdate:${name}` as const
const useModelToggleEmits = [updateEventKey]
const useModelToggleProps = {
[name]: _prop,
[updateEventKeyRaw]: _event,
} as UseModelTogglePropsRaw<T>
const useModelToggle = ({
indicator,
toggleReason,
shouldHideWhenRouteChanges,
shouldProceed,
onShow,
onHide,
}: ModelToggleParams) => {
const instance = getCurrentInstance()!
const { emit } = instance
const props = instance.props as UseModelTogglePropsGeneric<T> & {
disabled: boolean
}
const hasUpdateHandler = computed(() =>
isFunction(props[updateEventKeyRaw])
)
// 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 = (event?: Event) => {
if (indicator.value === true) {
return
}
indicator.value = true
if (toggleReason) {
toggleReason.value = event
}
if (isFunction(onShow)) {
onShow(event)
}
}
const doHide = (event?: Event) => {
if (indicator.value === false) {
return
}
indicator.value = false
if (toggleReason) {
toggleReason.value = event
}
if (isFunction(onHide)) {
onHide(event)
}
}
const show = (event?: Event) => {
if (
props.disabled === true ||
(isFunction(shouldProceed) && !shouldProceed())
)
return
const shouldEmit = hasUpdateHandler.value && isClient
if (shouldEmit) {
emit(updateEventKey, true)
}
if (isModelBindingAbsent.value || !shouldEmit) {
doShow(event)
}
}
const hide = (event?: Event) => {
if (props.disabled === true || !isClient) return
const shouldEmit = hasUpdateHandler.value && isClient
if (shouldEmit) {
emit(updateEventKey, false)
}
if (isModelBindingAbsent.value || !shouldEmit) {
doHide(event)
}
}
const onChange = (val: boolean) => {
if (!isBoolean(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)
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])
})
return {
hide,
show,
toggle,
hasUpdateHandler,
}
}
return {
useModelToggle,
useModelToggleProps,
useModelToggleEmits,
}
}
const { useModelToggle, useModelToggleProps, useModelToggleEmits } =
createModelToggleComposable('modelValue')
export { useModelToggle, useModelToggleEmits, useModelToggleProps }
export type UseModelToggleProps = ExtractPropTypes<typeof useModelToggleProps>
export type ModelToggleParams = {
indicator: Ref<boolean>
toggleReason?: Ref<Event | undefined>
shouldHideWhenRouteChanges?: Ref<boolean>
shouldProceed?: () => boolean
onShow?: (event?: Event) => void
onHide?: (event?: Event) => void
}