import { useRef } from 'react'; import { Keys } from 'react-hotkeys-hook'; import { autoUpdate, flip, offset, Placement, useFloating, } from '@floating-ui/react'; import { Key } from 'ts-key-enum'; import { DropdownScope } from '@/ui/layout/dropdown/scopes/DropdownScope'; import { HotkeyEffect } from '@/ui/utilities/hotkey/components/HotkeyEffect'; import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; import { HotkeyScope } from '@/ui/utilities/hotkey/types/HotkeyScope'; import { useListenClickOutside } from '@/ui/utilities/pointer-event/hooks/useListenClickOutside'; import { getScopeIdFromComponentId } from '@/ui/utilities/recoil-scope/utils/getScopeIdFromComponentId'; import { isDefined } from '~/utils/isDefined'; import { useDropdown } from '../hooks/useDropdown'; import { useInternalHotkeyScopeManagement } from '../hooks/useInternalHotkeyScopeManagement'; import { DropdownMenu } from './DropdownMenu'; import { DropdownOnToggleEffect } from './DropdownOnToggleEffect'; type DropdownProps = { className?: string; clickableComponent?: JSX.Element | JSX.Element[]; dropdownComponents: JSX.Element | JSX.Element[]; hotkey?: { key: Keys; scope: string; }; dropdownHotkeyScope: HotkeyScope; dropdownId: string; dropdownPlacement?: Placement; dropdownMenuWidth?: `${string}px` | `${number}%` | 'auto' | number; dropdownOffset?: { x?: number; y?: number }; disableBlur?: boolean; onClickOutside?: () => void; onClose?: () => void; onOpen?: () => void; }; export const Dropdown = ({ className, clickableComponent, dropdownComponents, dropdownMenuWidth, hotkey, dropdownId, dropdownHotkeyScope, dropdownPlacement = 'bottom-end', dropdownOffset = { x: 0, y: 0 }, disableBlur = false, onClickOutside, onClose, onOpen, }: DropdownProps) => { const containerRef = useRef(null); const { isDropdownOpen, toggleDropdown, closeDropdown, dropdownWidth } = useDropdown(dropdownId); const offsetMiddlewares = []; if (isDefined(dropdownOffset.x)) { offsetMiddlewares.push(offset({ crossAxis: dropdownOffset.x })); } if (isDefined(dropdownOffset.y)) { offsetMiddlewares.push(offset({ mainAxis: dropdownOffset.y })); } const { refs, floatingStyles } = useFloating({ placement: dropdownPlacement, middleware: [flip(), ...offsetMiddlewares], whileElementsMounted: autoUpdate, }); const handleHotkeyTriggered = () => { toggleDropdown(); }; useListenClickOutside({ refs: [containerRef], callback: () => { onClickOutside?.(); if (isDropdownOpen) { closeDropdown(); } }, }); useInternalHotkeyScopeManagement({ dropdownScopeId: getScopeIdFromComponentId(dropdownId), dropdownHotkeyScopeFromParent: dropdownHotkeyScope, }); useScopedHotkeys( [Key.Escape], () => { closeDropdown(); }, dropdownHotkeyScope.scope, [closeDropdown], ); return (
{clickableComponent && (
{ toggleDropdown(); onClickOutside?.(); }} className={className} > {clickableComponent}
)} {hotkey && ( )} {isDropdownOpen && ( {dropdownComponents} )}
); };