Skip to content

Commit

Permalink
feat: adjust contextual dropdown vert position
Browse files Browse the repository at this point in the history
  • Loading branch information
petermakowski committed Jan 9, 2024
1 parent 80a8005 commit 68dabcf
Show file tree
Hide file tree
Showing 3 changed files with 83 additions and 8 deletions.
3 changes: 2 additions & 1 deletion src/components/ContextualMenu/ContextualMenu.stories.mdx
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { ArgsTable, Canvas, Meta, Story } from "@storybook/addon-docs";

import Button from "../Button";
import ContextualMenu from "./ContextualMenu";

<Meta
Expand Down Expand Up @@ -37,6 +37,7 @@ export const ScrollTemplate = (args) => (
voluptas odit aspernatur alias molestias facere.
</p>
)}
<ContextualMenu {...args} />
</div>
);

Expand Down
1 change: 1 addition & 0 deletions src/components/ContextualMenu/ContextualMenu.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -191,6 +191,7 @@ const ContextualMenu = <L,>({
const wrapper = useRef<HTMLDivElement | null>(null);
const [positionCoords, setPositionCoords] = useState<DOMRect>();
const [adjustedPosition, setAdjustedPosition] = useState(position);

const hasToggle = hasToggleIcon || toggleLabel;

useEffect(() => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,13 +24,15 @@ export enum Label {
export type MenuLink<L = null> = string | ButtonProps<L> | ButtonProps<L>[];

export type Position = "left" | "center" | "right";
type VerticalPosition = "top" | "bottom";

/**
* The props for the ContextualMenuDropdown component.
* @template L - The type of the link props.
*/
export type Props<L = null> = {
adjustedPosition?: Position;
verticalPosition?: VerticalPosition;
autoAdjust?: boolean;
handleClose?: (evt?: MouseEvent) => void;
constrainPanelWidth?: boolean;
Expand All @@ -55,14 +57,18 @@ export type Props<L = null> = {
*/
const getPositionStyle = (
position: Position,
verticalPosition: VerticalPosition,
positionCoords: Props["positionCoords"],
constrainPanelWidth: Props["constrainPanelWidth"]
): React.CSSProperties => {
if (!positionCoords) {
return null;
}
const { height, left, top, width } = positionCoords;
const topPos = top + height + (window.scrollY || 0);
const topPos =
verticalPosition === "bottom"
? top + height + (window.scrollY || 0)
: top + (window.scrollY || 0);
let leftPos = left;

switch (position) {
Expand Down Expand Up @@ -161,6 +167,23 @@ const generateLink = <L,>(
);
};

const getClosestScrollableParent = (
node: HTMLElement | null
): HTMLElement | null => {
let currentNode = node;
while (currentNode && currentNode !== document.body) {
const { overflowY, overflowX } = window.getComputedStyle(currentNode);
if (
["auto", "scroll", "overlay"].includes(overflowY) &&
["auto", "scroll", "overlay"].includes(overflowX)
) {
return currentNode;
}
currentNode = currentNode.parentElement;
}
return document.body;
};

const ContextualMenuDropdown = <L,>({
adjustedPosition,
autoAdjust,
Expand All @@ -180,30 +203,75 @@ const ContextualMenuDropdown = <L,>({
...props
}: Props<L>): JSX.Element => {
const dropdown = useRef();

const [verticalPosition, setVerticalPosition] = useState<"top" | "bottom">(
"bottom"
);
const [positionStyle, setPositionStyle] = useState(
getPositionStyle(adjustedPosition, positionCoords, constrainPanelWidth)
getPositionStyle(
adjustedPosition,
verticalPosition,
positionCoords,
constrainPanelWidth
)
);
const [maxHeight, setMaxHeight] = useState<number>();

// Update the styles to position the menu.
const updatePositionStyle = useCallback(() => {
setPositionStyle(
getPositionStyle(adjustedPosition, positionCoords, constrainPanelWidth)
getPositionStyle(
adjustedPosition,
verticalPosition,
positionCoords,
constrainPanelWidth
)
);
}, [adjustedPosition, positionCoords, verticalPosition, constrainPanelWidth]);

const updateVerticalPosition = useCallback(() => {
if (!positionNode) {
return null;
}
const scrollableParent = getClosestScrollableParent(positionNode);
if (!scrollableParent) {
return null;
}
const scrollableParentRect = scrollableParent.getBoundingClientRect();
const rect = positionNode.getBoundingClientRect();

// Calculate the rect in relation to the scrollableParent
const relativeRect = {
top: rect.top - scrollableParentRect.top,
bottom: rect.bottom - scrollableParentRect.top,
height: rect.height,
};

const spaceBelow = scrollableParentRect.height - relativeRect.bottom;
const spaceAbove = relativeRect.top;
const dropdownHeight = relativeRect.height;

setVerticalPosition(
spaceBelow >= dropdownHeight || spaceBelow > spaceAbove ? "bottom" : "top"
);
}, [adjustedPosition, positionCoords, constrainPanelWidth]);
}, [positionNode]);

// Update the position when the window fitment info changes.
const onUpdateWindowFitment = useCallback(
(fitsWindow: WindowFitment) => {
if (autoAdjust) {
setAdjustedPosition(adjustForWindow(position, fitsWindow));
updateVerticalPosition();
}
if (scrollOverflow) {
setMaxHeight(fitsWindow.fromBottom.spaceBelow - 16);
}
},
[autoAdjust, position, scrollOverflow, setAdjustedPosition]
[
autoAdjust,
position,
scrollOverflow,
setAdjustedPosition,
updateVerticalPosition,
]
);

// Handle adjusting the horizontal position and scrolling of the dropdown so that it remains on screen.
Expand All @@ -220,6 +288,10 @@ const ContextualMenuDropdown = <L,>({
updatePositionStyle();
}, [adjustedPosition, updatePositionStyle]);

useEffect(() => {
updateVerticalPosition();
}, [updateVerticalPosition]);

return (
// Vanilla Framework uses .p-contextual-menu parent modifier classnames to determine the correct position of the .p-contextual-menu__dropdown dropdown (left, center, right).
// Extra span wrapper is required as the dropdown is rendered in a portal.
Expand All @@ -237,6 +309,7 @@ const ContextualMenuDropdown = <L,>({
...(scrollOverflow
? { maxHeight, minHeight: "2rem", overflowX: "auto" }
: {}),
...(verticalPosition === "top" ? { bottom: "0" } : {}),
}}
{...props}
>
Expand Down

0 comments on commit 68dabcf

Please sign in to comment.