* Refactor buttons * Complete components creation * Complete refactoring * fix lint * Complete button work
101 lines
2.8 KiB
TypeScript
101 lines
2.8 KiB
TypeScript
import { useEffect, useRef } from 'react';
|
|
import { Keys } from 'react-hotkeys-hook';
|
|
import styled from '@emotion/styled';
|
|
import { flip, offset, Placement, useFloating } from '@floating-ui/react';
|
|
|
|
import { HotkeyScope } from '@/ui/utilities/hotkey/types/HotkeyScope';
|
|
import { useListenClickOutside } from '@/ui/utilities/pointer-event/hooks/useListenClickOutside';
|
|
import { useRecoilScopedFamilyState } from '@/ui/utilities/recoil-scope/hooks/useRecoilScopedFamilyState';
|
|
import { isDeeplyEqual } from '~/utils/isDeeplyEqual';
|
|
|
|
import { useDropdownButton } from '../hooks/useDropdownButton';
|
|
import { dropdownButtonCustomHotkeyScopeScopedFamilyState } from '../states/dropdownButtonCustomHotkeyScopeScopedFamilyState';
|
|
import { DropdownRecoilScopeContext } from '../states/recoil-scope-contexts/DropdownRecoilScopeContext';
|
|
|
|
import { HotkeyEffect } from './HotkeyEffect';
|
|
|
|
const StyledContainer = styled.div`
|
|
position: relative;
|
|
z-index: 100;
|
|
`;
|
|
|
|
type OwnProps = {
|
|
buttonComponents: JSX.Element | JSX.Element[];
|
|
dropdownComponents: JSX.Element | JSX.Element[];
|
|
dropdownKey: string;
|
|
hotkey?: {
|
|
key: Keys;
|
|
scope: string;
|
|
};
|
|
dropdownHotkeyScope?: HotkeyScope;
|
|
dropdownPlacement?: Placement;
|
|
};
|
|
|
|
export function DropdownButton({
|
|
buttonComponents,
|
|
dropdownComponents,
|
|
dropdownKey,
|
|
hotkey,
|
|
dropdownHotkeyScope,
|
|
dropdownPlacement = 'bottom-end',
|
|
}: OwnProps) {
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
|
|
const { isDropdownButtonOpen, toggleDropdownButton, closeDropdownButton } =
|
|
useDropdownButton({
|
|
key: dropdownKey,
|
|
});
|
|
|
|
const { refs, floatingStyles } = useFloating({
|
|
placement: dropdownPlacement,
|
|
middleware: [flip(), offset()],
|
|
});
|
|
|
|
function handleHotkeyTriggered() {
|
|
toggleDropdownButton();
|
|
}
|
|
|
|
useListenClickOutside({
|
|
refs: [containerRef],
|
|
callback: () => {
|
|
if (isDropdownButtonOpen) {
|
|
closeDropdownButton();
|
|
}
|
|
},
|
|
});
|
|
|
|
const [dropdownButtonCustomHotkeyScope, setDropdownButtonCustomHotkeyScope] =
|
|
useRecoilScopedFamilyState(
|
|
dropdownButtonCustomHotkeyScopeScopedFamilyState,
|
|
dropdownKey,
|
|
DropdownRecoilScopeContext,
|
|
);
|
|
|
|
useEffect(() => {
|
|
if (!isDeeplyEqual(dropdownButtonCustomHotkeyScope, dropdownHotkeyScope)) {
|
|
setDropdownButtonCustomHotkeyScope(dropdownHotkeyScope);
|
|
}
|
|
}, [
|
|
setDropdownButtonCustomHotkeyScope,
|
|
dropdownHotkeyScope,
|
|
dropdownButtonCustomHotkeyScope,
|
|
]);
|
|
|
|
return (
|
|
<StyledContainer ref={containerRef}>
|
|
{hotkey && (
|
|
<HotkeyEffect
|
|
hotkey={hotkey}
|
|
onHotkeyTriggered={handleHotkeyTriggered}
|
|
/>
|
|
)}
|
|
<div ref={refs.setReference}>{buttonComponents}</div>
|
|
{isDropdownButtonOpen && (
|
|
<div ref={refs.setFloating} style={floatingStyles}>
|
|
{dropdownComponents}
|
|
</div>
|
|
)}
|
|
</StyledContainer>
|
|
);
|
|
}
|