| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | import { Popover, Transition } from '@headlessui/react' | 
					
						
							|  |  |  | import { Fragment, cloneElement, useRef } from 'react' | 
					
						
							|  |  |  | import s from './style.module.css' | 
					
						
							| 
									
										
										
										
											2024-07-09 15:05:40 +08:00
										 |  |  | import cn from '@/utils/classnames' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-07 09:47:31 +08:00
										 |  |  | export type HtmlContentProps = { | 
					
						
							|  |  |  |   onClose?: () => void | 
					
						
							|  |  |  |   onClick?: () => void | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | type IPopover = { | 
					
						
							|  |  |  |   className?: string | 
					
						
							| 
									
										
										
										
											2023-09-07 09:47:31 +08:00
										 |  |  |   htmlContent: React.ReactElement<HtmlContentProps> | 
					
						
							| 
									
										
										
										
											2023-12-19 17:22:54 +08:00
										 |  |  |   popupClassName?: string | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   trigger?: 'click' | 'hover' | 
					
						
							| 
									
										
										
										
											2024-04-24 15:02:29 +08:00
										 |  |  |   position?: 'bottom' | 'br' | 'bl' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   btnElement?: string | React.ReactNode | 
					
						
							|  |  |  |   btnClassName?: string | ((open: boolean) => string) | 
					
						
							| 
									
										
										
										
											2023-08-18 17:37:31 +08:00
										 |  |  |   manualClose?: boolean | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const timeoutDuration = 100 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default function CustomPopover({ | 
					
						
							|  |  |  |   trigger = 'hover', | 
					
						
							|  |  |  |   position = 'bottom', | 
					
						
							|  |  |  |   htmlContent, | 
					
						
							| 
									
										
										
										
											2023-12-19 17:22:54 +08:00
										 |  |  |   popupClassName, | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   btnElement, | 
					
						
							|  |  |  |   className, | 
					
						
							|  |  |  |   btnClassName, | 
					
						
							| 
									
										
										
										
											2023-08-18 17:37:31 +08:00
										 |  |  |   manualClose, | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | }: IPopover) { | 
					
						
							|  |  |  |   const buttonRef = useRef<HTMLButtonElement>(null) | 
					
						
							|  |  |  |   const timeOutRef = useRef<NodeJS.Timeout | null>(null) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const onMouseEnter = (isOpen: boolean) => { | 
					
						
							|  |  |  |     timeOutRef.current && clearTimeout(timeOutRef.current) | 
					
						
							|  |  |  |     !isOpen && buttonRef.current?.click() | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const onMouseLeave = (isOpen: boolean) => { | 
					
						
							|  |  |  |     timeOutRef.current = setTimeout(() => { | 
					
						
							|  |  |  |       isOpen && buttonRef.current?.click() | 
					
						
							|  |  |  |     }, timeoutDuration) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <Popover className="relative"> | 
					
						
							|  |  |  |       {({ open }: { open: boolean }) => { | 
					
						
							|  |  |  |         return ( | 
					
						
							|  |  |  |           <> | 
					
						
							|  |  |  |             <div | 
					
						
							|  |  |  |               {...(trigger !== 'hover' | 
					
						
							|  |  |  |                 ? {} | 
					
						
							|  |  |  |                 : { | 
					
						
							|  |  |  |                   onMouseLeave: () => onMouseLeave(open), | 
					
						
							|  |  |  |                   onMouseEnter: () => onMouseEnter(open), | 
					
						
							| 
									
										
										
										
											2023-08-16 10:31:08 +08:00
										 |  |  |                 })} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |             > | 
					
						
							|  |  |  |               <Popover.Button | 
					
						
							|  |  |  |                 ref={buttonRef} | 
					
						
							| 
									
										
										
										
											2023-12-19 17:22:54 +08:00
										 |  |  |                 className={`group ${s.popupBtn} ${open ? '' : 'bg-gray-100'} ${!btnClassName | 
					
						
							|  |  |  |                   ? '' | 
					
						
							|  |  |  |                   : typeof btnClassName === 'string' | 
					
						
							|  |  |  |                     ? btnClassName | 
					
						
							|  |  |  |                     : btnClassName?.(open) | 
					
						
							| 
									
										
										
										
											2023-08-16 10:31:08 +08:00
										 |  |  |                 }`}
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |               > | 
					
						
							|  |  |  |                 {btnElement} | 
					
						
							|  |  |  |               </Popover.Button> | 
					
						
							| 
									
										
										
										
											2023-08-16 10:31:08 +08:00
										 |  |  |               <Transition as={Fragment}> | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                 <Popover.Panel | 
					
						
							| 
									
										
										
										
											2024-04-24 15:02:29 +08:00
										 |  |  |                   className={cn( | 
					
						
							|  |  |  |                     s.popupPanel, | 
					
						
							|  |  |  |                     position === 'bottom' && '-translate-x-1/2 left-1/2', | 
					
						
							|  |  |  |                     position === 'bl' && 'left-0', | 
					
						
							|  |  |  |                     position === 'br' && 'right-0', | 
					
						
							|  |  |  |                     className, | 
					
						
							|  |  |  |                   )} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                   {...(trigger !== 'hover' | 
					
						
							|  |  |  |                     ? {} | 
					
						
							|  |  |  |                     : { | 
					
						
							|  |  |  |                       onMouseLeave: () => onMouseLeave(open), | 
					
						
							|  |  |  |                       onMouseEnter: () => onMouseEnter(open), | 
					
						
							| 
									
										
										
										
											2023-08-18 17:37:31 +08:00
										 |  |  |                     }) | 
					
						
							|  |  |  |                   } | 
					
						
							| 
									
										
										
										
											2023-08-16 10:31:08 +08:00
										 |  |  |                 > | 
					
						
							|  |  |  |                   {({ close }) => ( | 
					
						
							|  |  |  |                     <div | 
					
						
							| 
									
										
										
										
											2023-12-19 17:22:54 +08:00
										 |  |  |                       className={cn(s.panelContainer, popupClassName)} | 
					
						
							| 
									
										
										
										
											2023-08-16 10:31:08 +08:00
										 |  |  |                       {...(trigger !== 'hover' | 
					
						
							|  |  |  |                         ? {} | 
					
						
							|  |  |  |                         : { | 
					
						
							|  |  |  |                           onMouseLeave: () => onMouseLeave(open), | 
					
						
							|  |  |  |                           onMouseEnter: () => onMouseEnter(open), | 
					
						
							| 
									
										
										
										
											2023-08-18 17:37:31 +08:00
										 |  |  |                         }) | 
					
						
							|  |  |  |                       } | 
					
						
							| 
									
										
										
										
											2023-08-16 10:31:08 +08:00
										 |  |  |                     > | 
					
						
							| 
									
										
										
										
											2023-09-07 09:47:31 +08:00
										 |  |  |                       {cloneElement(htmlContent as React.ReactElement<HtmlContentProps>, { | 
					
						
							| 
									
										
										
										
											2023-08-18 17:37:31 +08:00
										 |  |  |                         onClose: () => onMouseLeave(open), | 
					
						
							|  |  |  |                         ...(manualClose | 
					
						
							|  |  |  |                           ? { | 
					
						
							|  |  |  |                             onClick: close, | 
					
						
							|  |  |  |                           } | 
					
						
							|  |  |  |                           : {}), | 
					
						
							| 
									
										
										
										
											2023-08-16 10:31:08 +08:00
										 |  |  |                       })} | 
					
						
							|  |  |  |                     </div> | 
					
						
							|  |  |  |                   )} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                 </Popover.Panel> | 
					
						
							|  |  |  |               </Transition> | 
					
						
							|  |  |  |             </div> | 
					
						
							|  |  |  |           </> | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |       }} | 
					
						
							|  |  |  |     </Popover> | 
					
						
							|  |  |  |   ) | 
					
						
							|  |  |  | } |