|
| 1 | +import useBoolean from 'ahooks/es/useBoolean' |
| 2 | +import React from 'react' |
| 3 | +import { isDOMTypeElement, isElement } from '../is' |
| 4 | + |
| 5 | +export interface UseModalEnhancedContext { |
| 6 | + enhancedAction: UseModalEnhancedAction |
| 7 | +} |
| 8 | + |
| 9 | +export interface UseModalEnhancedProps { |
| 10 | + children?: React.ReactNode |
| 11 | + content?: React.ReactNode |
| 12 | + onClick?: (e: React.MouseEvent<HTMLElement>, enhancedAction: UseModalEnhancedAction) => void |
| 13 | + actionRef?: React.RefObject<UseModalEnhancedAction> |
| 14 | +} |
| 15 | + |
| 16 | +export interface UseModalEnhancedAction { |
| 17 | + close: () => void |
| 18 | + open: () => void |
| 19 | +} |
| 20 | + |
| 21 | +function useModalEnhanced(props: UseModalEnhancedProps = {}) { |
| 22 | + const { |
| 23 | + children, |
| 24 | + content, |
| 25 | + onClick, |
| 26 | + actionRef: actionRefProp, |
| 27 | + } = props |
| 28 | + |
| 29 | + const [visible, { setTrue: open, setFalse: close }] = useBoolean(false) |
| 30 | + |
| 31 | + const actionRef = React.useRef<UseModalEnhancedAction>({ open, close }) |
| 32 | + |
| 33 | + React.useImperativeHandle(actionRefProp, () => actionRef.current) |
| 34 | + |
| 35 | + const handleClick = (event: React.MouseEvent<HTMLElement>) => { |
| 36 | + if (onClick) |
| 37 | + return onClick(event, actionRef.current) |
| 38 | + |
| 39 | + return open() |
| 40 | + } |
| 41 | + |
| 42 | + // ======================== Trigger ======================== |
| 43 | + let trigger: React.ReactNode = children |
| 44 | + if (React.isValidElement(children)) |
| 45 | + trigger = React.cloneElement<any>(children, { onClick: handleClick }) |
| 46 | + |
| 47 | + // ======================== Content ======================== |
| 48 | + let contentNode: React.ReactNode = content |
| 49 | + if (isElement<UseModalEnhancedContext>(contentNode) && !isDOMTypeElement(contentNode)) { |
| 50 | + contentNode = React.cloneElement<UseModalEnhancedContext>(contentNode, { |
| 51 | + enhancedAction: actionRef.current, |
| 52 | + }) |
| 53 | + } |
| 54 | + |
| 55 | + const contextHolder = { trigger, content: contentNode } |
| 56 | + const action = { open, close } |
| 57 | + |
| 58 | + return [visible, action, contextHolder] as const |
| 59 | +} |
| 60 | + |
| 61 | +export default useModalEnhanced |
0 commit comments