forked from github/plane
2dcaccd4ec
* chore: dynamic position dropdown (#2138) * chore: dynamic position state dropdown for issue view * style: state select dropdown styling * fix: state icon attribute names * chore: state select dynamic dropdown * chore: member select dynamic dropdown * chore: label select dynamic dropdown * chore: priority select dynamic dropdown * chore: label select dropdown improvement * refactor: state dropdown location * chore: dropdown improvement and code refactor * chore: dynamic dropdown hook type added --------- Co-authored-by: Aaryan Khandelwal <aaryankhandu123@gmail.com> * fix: fields not getting selected in the create issue form (#2212) * fix: hydration error and draft issue workflow * fix: build error * fix: properties getting de-selected after create, module & cycle not getting auto-select on the form * fix: display layout, props being updated directly * chore: sub issues count in individual issue (#2221) * fix: service imports * chore: rename csv service file --------- Co-authored-by: Anmol Singh Bhatia <121005188+anmolsinghbhatia@users.noreply.github.com> Co-authored-by: Dakshesh Jain <65905942+dakshesh14@users.noreply.github.com> Co-authored-by: Bavisetti Narayan <72156168+NarayanBavisetti@users.noreply.github.com>
65 lines
1.9 KiB
TypeScript
65 lines
1.9 KiB
TypeScript
import React, { useCallback, useEffect } from "react";
|
|
|
|
// hook
|
|
import useOutsideClickDetector from "./use-outside-click-detector";
|
|
|
|
/**
|
|
* Custom hook for dynamic dropdown position calculation.
|
|
* @param isOpen - Indicates whether the dropdown is open.
|
|
* @param handleClose - Callback to handle closing the dropdown.
|
|
* @param buttonRef - Ref object for the button triggering the dropdown.
|
|
* @param dropdownRef - Ref object for the dropdown element.
|
|
*/
|
|
|
|
const useDynamicDropdownPosition = (
|
|
isOpen: boolean,
|
|
handleClose: () => void,
|
|
buttonRef: React.RefObject<any>,
|
|
dropdownRef: React.RefObject<any>
|
|
) => {
|
|
const handlePosition = useCallback(() => {
|
|
const button = buttonRef.current;
|
|
const dropdown = dropdownRef.current;
|
|
|
|
if (!dropdown || !button) return;
|
|
|
|
const buttonRect = button.getBoundingClientRect();
|
|
const dropdownRect = dropdown.getBoundingClientRect();
|
|
|
|
const { innerHeight, innerWidth, scrollX, scrollY } = window;
|
|
|
|
let top: number = buttonRect.bottom + scrollY;
|
|
if (top + dropdownRect.height > innerHeight) top = innerHeight - dropdownRect.height;
|
|
|
|
let left: number = buttonRect.left + scrollX + (buttonRect.width - dropdownRect.width) / 2;
|
|
if (left + dropdownRect.width > innerWidth) left = innerWidth - dropdownRect.width;
|
|
|
|
dropdown.style.top = `${Math.max(top, 5)}px`;
|
|
dropdown.style.left = `${Math.max(left, 5)}px`;
|
|
}, [buttonRef, dropdownRef]);
|
|
|
|
useEffect(() => {
|
|
if (isOpen) handlePosition();
|
|
}, [handlePosition, isOpen]);
|
|
|
|
useOutsideClickDetector(dropdownRef, () => {
|
|
if (isOpen) handleClose();
|
|
});
|
|
|
|
const handleResize = useCallback(() => {
|
|
if (isOpen) {
|
|
handlePosition();
|
|
}
|
|
}, [handlePosition, isOpen]);
|
|
|
|
useEffect(() => {
|
|
window.addEventListener("resize", handleResize);
|
|
|
|
return () => {
|
|
window.removeEventListener("resize", handleResize);
|
|
};
|
|
}, [isOpen, handleResize]);
|
|
};
|
|
|
|
export default useDynamicDropdownPosition;
|