|  | @@ -1,84 +1,167 @@
 | 
	
		
			
				|  |  |  'use client'
 | 
	
		
			
				|  |  | -import { useBoolean } from 'ahooks'
 | 
	
		
			
				|  |  | -import React, { useEffect, useRef, useState } from 'react'
 | 
	
		
			
				|  |  | -import type { FC } from 'react'
 | 
	
		
			
				|  |  | -import { createRoot } from 'react-dom/client'
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -type IPortalToFollowElementProps = {
 | 
	
		
			
				|  |  | -  portalElem: React.ReactNode
 | 
	
		
			
				|  |  | -  children: React.ReactNode
 | 
	
		
			
				|  |  | -  controlShow?: number
 | 
	
		
			
				|  |  | -  controlHide?: number
 | 
	
		
			
				|  |  | +import React from 'react'
 | 
	
		
			
				|  |  | +import {
 | 
	
		
			
				|  |  | +  FloatingPortal,
 | 
	
		
			
				|  |  | +  autoUpdate,
 | 
	
		
			
				|  |  | +  flip,
 | 
	
		
			
				|  |  | +  offset,
 | 
	
		
			
				|  |  | +  shift,
 | 
	
		
			
				|  |  | +  useDismiss,
 | 
	
		
			
				|  |  | +  useFloating,
 | 
	
		
			
				|  |  | +  useFocus,
 | 
	
		
			
				|  |  | +  useHover,
 | 
	
		
			
				|  |  | +  useInteractions,
 | 
	
		
			
				|  |  | +  useMergeRefs,
 | 
	
		
			
				|  |  | +  useRole,
 | 
	
		
			
				|  |  | +} from '@floating-ui/react'
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +import type { Placement } from '@floating-ui/react'
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +type PortalToFollowElemOptions = {
 | 
	
		
			
				|  |  | +  /*
 | 
	
		
			
				|  |  | +  * top, bottom, left, right
 | 
	
		
			
				|  |  | +  * start, end. Default is middle
 | 
	
		
			
				|  |  | +  * combine: top-start, top-end
 | 
	
		
			
				|  |  | +  */
 | 
	
		
			
				|  |  | +  placement?: Placement
 | 
	
		
			
				|  |  | +  open?: boolean
 | 
	
		
			
				|  |  | +  offset?: number
 | 
	
		
			
				|  |  | +  onOpenChange?: (open: boolean) => void
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -const PortalToFollowElement: FC<IPortalToFollowElementProps> = ({
 | 
	
		
			
				|  |  | -  portalElem,
 | 
	
		
			
				|  |  | +export function usePortalToFollowElem({
 | 
	
		
			
				|  |  | +  placement = 'bottom',
 | 
	
		
			
				|  |  | +  open,
 | 
	
		
			
				|  |  | +  offset: offsetValue = 0,
 | 
	
		
			
				|  |  | +  onOpenChange: setControlledOpen,
 | 
	
		
			
				|  |  | +}: PortalToFollowElemOptions = {}) {
 | 
	
		
			
				|  |  | +  const setOpen = setControlledOpen
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const data = useFloating({
 | 
	
		
			
				|  |  | +    placement,
 | 
	
		
			
				|  |  | +    open,
 | 
	
		
			
				|  |  | +    onOpenChange: setOpen,
 | 
	
		
			
				|  |  | +    whileElementsMounted: autoUpdate,
 | 
	
		
			
				|  |  | +    middleware: [
 | 
	
		
			
				|  |  | +      offset(offsetValue),
 | 
	
		
			
				|  |  | +      flip({
 | 
	
		
			
				|  |  | +        crossAxis: placement.includes('-'),
 | 
	
		
			
				|  |  | +        fallbackAxisSideDirection: 'start',
 | 
	
		
			
				|  |  | +        padding: 5,
 | 
	
		
			
				|  |  | +      }),
 | 
	
		
			
				|  |  | +      shift({ padding: 5 }),
 | 
	
		
			
				|  |  | +    ],
 | 
	
		
			
				|  |  | +  })
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const context = data.context
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const hover = useHover(context, {
 | 
	
		
			
				|  |  | +    move: false,
 | 
	
		
			
				|  |  | +    enabled: open == null,
 | 
	
		
			
				|  |  | +  })
 | 
	
		
			
				|  |  | +  const focus = useFocus(context, {
 | 
	
		
			
				|  |  | +    enabled: open == null,
 | 
	
		
			
				|  |  | +  })
 | 
	
		
			
				|  |  | +  const dismiss = useDismiss(context)
 | 
	
		
			
				|  |  | +  const role = useRole(context, { role: 'tooltip' })
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  const interactions = useInteractions([hover, focus, dismiss, role])
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  return React.useMemo(
 | 
	
		
			
				|  |  | +    () => ({
 | 
	
		
			
				|  |  | +      open,
 | 
	
		
			
				|  |  | +      setOpen,
 | 
	
		
			
				|  |  | +      ...interactions,
 | 
	
		
			
				|  |  | +      ...data,
 | 
	
		
			
				|  |  | +    }),
 | 
	
		
			
				|  |  | +    [open, setOpen, interactions, data],
 | 
	
		
			
				|  |  | +  )
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +type ContextType = ReturnType<typeof usePortalToFollowElem> | null
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +const PortalToFollowElemContext = React.createContext<ContextType>(null)
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +export function usePortalToFollowElemContext() {
 | 
	
		
			
				|  |  | +  const context = React.useContext(PortalToFollowElemContext)
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  if (context == null)
 | 
	
		
			
				|  |  | +    throw new Error('PortalToFollowElem components must be wrapped in <PortalToFollowElem />')
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  return context
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +export function PortalToFollowElem({
 | 
	
		
			
				|  |  |    children,
 | 
	
		
			
				|  |  | -  controlShow,
 | 
	
		
			
				|  |  | -  controlHide,
 | 
	
		
			
				|  |  | -}) => {
 | 
	
		
			
				|  |  | -  const [isShowContent, { setTrue: showContent, setFalse: hideContent, toggle: toggleContent }] = useBoolean(false)
 | 
	
		
			
				|  |  | -  const [wrapElem, setWrapElem] = useState<HTMLDivElement | null>(null)
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -  useEffect(() => {
 | 
	
		
			
				|  |  | -    if (controlShow)
 | 
	
		
			
				|  |  | -      showContent()
 | 
	
		
			
				|  |  | -  }, [controlShow])
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -  useEffect(() => {
 | 
	
		
			
				|  |  | -    if (controlHide)
 | 
	
		
			
				|  |  | -      hideContent()
 | 
	
		
			
				|  |  | -  }, [controlHide])
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -  // todo use click outside hidden
 | 
	
		
			
				|  |  | -  const triggerElemRef = useRef<HTMLDivElement>(null)
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -  const calLoc = () => {
 | 
	
		
			
				|  |  | -    const triggerElem = triggerElemRef.current
 | 
	
		
			
				|  |  | -    if (!triggerElem) {
 | 
	
		
			
				|  |  | -      return {
 | 
	
		
			
				|  |  | -        display: 'none',
 | 
	
		
			
				|  |  | -      }
 | 
	
		
			
				|  |  | -    }
 | 
	
		
			
				|  |  | -    const {
 | 
	
		
			
				|  |  | -      left: triggerLeft,
 | 
	
		
			
				|  |  | -      top: triggerTop,
 | 
	
		
			
				|  |  | -      height,
 | 
	
		
			
				|  |  | -    } = triggerElem.getBoundingClientRect()
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -    return {
 | 
	
		
			
				|  |  | -      position: 'fixed',
 | 
	
		
			
				|  |  | -      left: triggerLeft,
 | 
	
		
			
				|  |  | -      top: triggerTop + height,
 | 
	
		
			
				|  |  | -      zIndex: 999,
 | 
	
		
			
				|  |  | -    }
 | 
	
		
			
				|  |  | -  }
 | 
	
		
			
				|  |  | +  ...options
 | 
	
		
			
				|  |  | +}: { children: React.ReactNode } & PortalToFollowElemOptions) {
 | 
	
		
			
				|  |  | +  // This can accept any props as options, e.g. `placement`,
 | 
	
		
			
				|  |  | +  // or other positioning options.
 | 
	
		
			
				|  |  | +  const tooltip = usePortalToFollowElem(options)
 | 
	
		
			
				|  |  | +  return (
 | 
	
		
			
				|  |  | +    <PortalToFollowElemContext.Provider value={tooltip}>
 | 
	
		
			
				|  |  | +      {children}
 | 
	
		
			
				|  |  | +    </PortalToFollowElemContext.Provider>
 | 
	
		
			
				|  |  | +  )
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +export const PortalToFollowElemTrigger = React.forwardRef<
 | 
	
		
			
				|  |  | +HTMLElement,
 | 
	
		
			
				|  |  | +React.HTMLProps<HTMLElement> & { asChild?: boolean }
 | 
	
		
			
				|  |  | +>(({ children, asChild = false, ...props }, propRef) => {
 | 
	
		
			
				|  |  | +  const context = usePortalToFollowElemContext()
 | 
	
		
			
				|  |  | +  const childrenRef = (children as any).ref
 | 
	
		
			
				|  |  | +  const ref = useMergeRefs([context.refs.setReference, propRef, childrenRef])
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  useEffect(() => {
 | 
	
		
			
				|  |  | -    if (isShowContent) {
 | 
	
		
			
				|  |  | -      const holder = document.createElement('div')
 | 
	
		
			
				|  |  | -      const root = createRoot(holder)
 | 
	
		
			
				|  |  | -      const style = calLoc()
 | 
	
		
			
				|  |  | -      root.render(
 | 
	
		
			
				|  |  | -        <div style={style as React.CSSProperties}>
 | 
	
		
			
				|  |  | -          {portalElem}
 | 
	
		
			
				|  |  | -        </div>,
 | 
	
		
			
				|  |  | -      )
 | 
	
		
			
				|  |  | -      document.body.appendChild(holder)
 | 
	
		
			
				|  |  | -      setWrapElem(holder)
 | 
	
		
			
				|  |  | -      console.log(holder)
 | 
	
		
			
				|  |  | -    }
 | 
	
		
			
				|  |  | -    else {
 | 
	
		
			
				|  |  | -      wrapElem?.remove?.()
 | 
	
		
			
				|  |  | -      setWrapElem(null)
 | 
	
		
			
				|  |  | -    }
 | 
	
		
			
				|  |  | -  }, [isShowContent])
 | 
	
		
			
				|  |  | +  // `asChild` allows the user to pass any element as the anchor
 | 
	
		
			
				|  |  | +  if (asChild && React.isValidElement(children)) {
 | 
	
		
			
				|  |  | +    return React.cloneElement(
 | 
	
		
			
				|  |  | +      children,
 | 
	
		
			
				|  |  | +      context.getReferenceProps({
 | 
	
		
			
				|  |  | +        ref,
 | 
	
		
			
				|  |  | +        ...props,
 | 
	
		
			
				|  |  | +        ...children.props,
 | 
	
		
			
				|  |  | +        'data-state': context.open ? 'open' : 'closed',
 | 
	
		
			
				|  |  | +      }),
 | 
	
		
			
				|  |  | +    )
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    return (
 | 
	
		
			
				|  |  | -    <div ref={triggerElemRef as React.RefObject<HTMLDivElement>} onClick={toggleContent}>
 | 
	
		
			
				|  |  | +    <div
 | 
	
		
			
				|  |  | +      ref={ref}
 | 
	
		
			
				|  |  | +      className='inline-block'
 | 
	
		
			
				|  |  | +      // The user can style the trigger based on the state
 | 
	
		
			
				|  |  | +      data-state={context.open ? 'open' : 'closed'}
 | 
	
		
			
				|  |  | +      {...context.getReferenceProps(props)}
 | 
	
		
			
				|  |  | +    >
 | 
	
		
			
				|  |  |        {children}
 | 
	
		
			
				|  |  |      </div>
 | 
	
		
			
				|  |  |    )
 | 
	
		
			
				|  |  | -}
 | 
	
		
			
				|  |  | +})
 | 
	
		
			
				|  |  | +PortalToFollowElemTrigger.displayName = 'PortalToFollowElemTrigger'
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +export const PortalToFollowElemContent = React.forwardRef<
 | 
	
		
			
				|  |  | +HTMLDivElement,
 | 
	
		
			
				|  |  | +React.HTMLProps<HTMLDivElement>
 | 
	
		
			
				|  |  | +>(({ style, ...props }, propRef) => {
 | 
	
		
			
				|  |  | +  const context = usePortalToFollowElemContext()
 | 
	
		
			
				|  |  | +  const ref = useMergeRefs([context.refs.setFloating, propRef])
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  if (!context.open)
 | 
	
		
			
				|  |  | +    return null
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  return (
 | 
	
		
			
				|  |  | +    <FloatingPortal>
 | 
	
		
			
				|  |  | +      <div
 | 
	
		
			
				|  |  | +        ref={ref}
 | 
	
		
			
				|  |  | +        style={{
 | 
	
		
			
				|  |  | +          ...context.floatingStyles,
 | 
	
		
			
				|  |  | +          ...style,
 | 
	
		
			
				|  |  | +        }}
 | 
	
		
			
				|  |  | +        {...context.getFloatingProps(props)}
 | 
	
		
			
				|  |  | +      />
 | 
	
		
			
				|  |  | +    </FloatingPortal>
 | 
	
		
			
				|  |  | +  )
 | 
	
		
			
				|  |  | +})
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -export default React.memo(PortalToFollowElement)
 | 
	
		
			
				|  |  | +PortalToFollowElemContent.displayName = 'PortalToFollowElemContent'
 |