* Chore(front): Create Storybook tests for the DropdownMenu component Co-authored-by: Benjamin Mayanja V <vibenjamin6@gmail.com> Co-authored-by: FellipeMTX <fellipefacdir@gmail.com> * Fix the tests Co-authored-by: Benjamin Mayanja V <vibenjamin6@gmail.com> Co-authored-by: FellipeMTX <fellipefacdir@gmail.com> * Simplify Dropdown * Remove console.log --------- Co-authored-by: Benjamin Mayanja V <vibenjamin6@gmail.com> Co-authored-by: FellipeMTX <fellipefacdir@gmail.com> Co-authored-by: Charles Bochet <charles@twenty.com>
118 lines
3.0 KiB
TypeScript
118 lines
3.0 KiB
TypeScript
import { useRef } from 'react';
|
|
import { Keys } from 'react-hotkeys-hook';
|
|
import { flip, offset, Placement, useFloating } from '@floating-ui/react';
|
|
import { Key } from 'ts-key-enum';
|
|
|
|
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 { useDropdown } from '../hooks/useDropdown';
|
|
import { useInternalHotkeyScopeManagement } from '../hooks/useInternalHotkeyScopeManagement';
|
|
|
|
import { DropdownMenu } from './DropdownMenu';
|
|
import { DropdownToggleEffect } from './DropdownToggleEffect';
|
|
|
|
type DropdownProps = {
|
|
clickableComponent?: JSX.Element | JSX.Element[];
|
|
dropdownComponents: JSX.Element | JSX.Element[];
|
|
hotkey?: {
|
|
key: Keys;
|
|
scope: string;
|
|
};
|
|
dropdownHotkeyScope: HotkeyScope;
|
|
dropdownPlacement?: Placement;
|
|
dropdownMenuWidth?: number;
|
|
dropdownOffset?: { x?: number; y?: number };
|
|
onClickOutside?: () => void;
|
|
onClose?: () => void;
|
|
onOpen?: () => void;
|
|
};
|
|
|
|
export const Dropdown = ({
|
|
clickableComponent,
|
|
dropdownComponents,
|
|
dropdownMenuWidth = 160,
|
|
hotkey,
|
|
dropdownHotkeyScope,
|
|
dropdownPlacement = 'bottom-end',
|
|
dropdownOffset = { x: 0, y: 0 },
|
|
onClickOutside,
|
|
onClose,
|
|
onOpen,
|
|
}: DropdownProps) => {
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
|
|
const { isDropdownOpen, toggleDropdown, closeDropdown } = useDropdown();
|
|
|
|
const offsetMiddlewares = [];
|
|
if (dropdownOffset.x) {
|
|
offsetMiddlewares.push(offset({ crossAxis: dropdownOffset.x }));
|
|
}
|
|
|
|
if (dropdownOffset.y) {
|
|
offsetMiddlewares.push(offset({ mainAxis: dropdownOffset.y }));
|
|
}
|
|
|
|
const { refs, floatingStyles } = useFloating({
|
|
placement: dropdownPlacement,
|
|
middleware: [flip(), ...offsetMiddlewares],
|
|
});
|
|
|
|
const handleHotkeyTriggered = () => {
|
|
toggleDropdown();
|
|
};
|
|
|
|
useListenClickOutside({
|
|
refs: [containerRef],
|
|
callback: () => {
|
|
onClickOutside?.();
|
|
|
|
if (isDropdownOpen) {
|
|
closeDropdown();
|
|
}
|
|
},
|
|
});
|
|
|
|
useInternalHotkeyScopeManagement({
|
|
dropdownHotkeyScopeFromParent: dropdownHotkeyScope,
|
|
});
|
|
|
|
useScopedHotkeys(
|
|
Key.Escape,
|
|
() => {
|
|
closeDropdown();
|
|
},
|
|
dropdownHotkeyScope.scope,
|
|
[closeDropdown],
|
|
);
|
|
|
|
return (
|
|
<div ref={containerRef}>
|
|
{clickableComponent && (
|
|
<div ref={refs.setReference} onClick={toggleDropdown}>
|
|
{clickableComponent}
|
|
</div>
|
|
)}
|
|
{hotkey && (
|
|
<HotkeyEffect
|
|
hotkey={hotkey}
|
|
onHotkeyTriggered={handleHotkeyTriggered}
|
|
/>
|
|
)}
|
|
{isDropdownOpen && (
|
|
<DropdownMenu
|
|
width={dropdownMenuWidth}
|
|
data-select-disable
|
|
ref={refs.setFloating}
|
|
style={floatingStyles}
|
|
>
|
|
{dropdownComponents}
|
|
</DropdownMenu>
|
|
)}
|
|
<DropdownToggleEffect onDropdownClose={onClose} onDropdownOpen={onOpen} />
|
|
</div>
|
|
);
|
|
};
|