Skip to content

Commit 947b59d

Browse files
authoredFeb 14, 2025··
fix(SelectPanel): Correctly recalculate position on overflow (#5562)
* wip: SelectPanel overflow * fix(useAnchoredPosition): refine reposition logic * Create silent-cameras-care.md * fix: lint * test(AnchoredOverlay): update snapshot * reorganize stories * fix(Overlay): add default max height * fix(SelectPanel): revert preventOverflow changes * test(vrt): update snapshots * Revert "test(vrt): update snapshots" This reverts commit 9a0fb5a. * test(vrt): update snapshots * fix(useResizeObserver): SSR compatibility * Revert "test(vrt): update snapshots" This reverts commit 08a9f9c. * test(vrt): update snapshots * fix(SelectPanel): fix flashing race condition and cleanup code * docs(AnchoredOverlay): document new pinPosition pro * fix tests * fix tests * fix tests * test(vrt): update snapshots * Revert "test(vrt): update snapshots" This reverts commit 8c1e8cf. * remove test code --------- Co-authored-by: francinelucca <francinelucca@users.noreply.github.com>
1 parent 37a91b5 commit 947b59d

13 files changed

+274
-7
lines changed
 

‎.changeset/silent-cameras-care.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@primer/react": patch
3+
---
4+
5+
fix(SelectPanel): Correctly recalculate position on overflow
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,110 @@
1+
import type {Meta} from '@storybook/react'
2+
import React, {useState} from 'react'
3+
4+
import {Button} from '../Button'
5+
import {AnchoredOverlay} from '.'
6+
import {Stack} from '../Stack'
7+
import {Dialog, Spinner} from '..'
8+
9+
const meta = {
10+
title: 'Components/AnchoredOverlay/Dev',
11+
component: AnchoredOverlay,
12+
} satisfies Meta<typeof AnchoredOverlay>
13+
14+
export default meta
15+
16+
export const RepositionAfterContentGrows = () => {
17+
const [open, setOpen] = useState(false)
18+
19+
const [loading, setLoading] = useState(true)
20+
21+
React.useEffect(() => {
22+
window.setTimeout(() => {
23+
if (open) setLoading(false)
24+
}, 2000)
25+
}, [open])
26+
27+
return (
28+
<Stack direction="vertical" justify="space-between" style={{height: 'calc(100vh - 200px)'}}>
29+
<div>
30+
What to expect:
31+
<ul>
32+
<li>The anchored overlay should open below the anchor (default position)</li>
33+
<li>After 2000ms, the amount of content in the overlay grows</li>
34+
<li>the overlay should reposition itself above the anchor so that it stays inside the window</li>
35+
</ul>
36+
</div>
37+
<AnchoredOverlay
38+
renderAnchor={props => (
39+
<Button {...props} sx={{width: 'fit-content'}}>
40+
Button
41+
</Button>
42+
)}
43+
open={open}
44+
onOpen={() => setOpen(true)}
45+
onClose={() => {
46+
setOpen(false)
47+
setLoading(true)
48+
}}
49+
>
50+
{loading ? (
51+
<>
52+
<Spinner />
53+
loading for 2000ms
54+
</>
55+
) : (
56+
<div style={{height: '300px'}}>content with 300px height</div>
57+
)}
58+
</AnchoredOverlay>
59+
</Stack>
60+
)
61+
}
62+
63+
export const RepositionAfterContentGrowsWithinDialog = () => {
64+
const [open, setOpen] = useState(false)
65+
66+
const [loading, setLoading] = useState(true)
67+
68+
React.useEffect(() => {
69+
window.setTimeout(() => {
70+
if (open) setLoading(false)
71+
}, 2000)
72+
}, [open])
73+
74+
return (
75+
<Dialog onClose={() => {}}>
76+
<Stack direction="vertical" justify="space-between" style={{height: 'calc(100vh - 300px)'}}>
77+
<div>
78+
What to expect:
79+
<ul>
80+
<li>The anchored overlay should open below the anchor (default position)</li>
81+
<li>After 2000ms, the amount of content in the overlay grows</li>
82+
<li>the overlay should reposition itself above the anchor so that it stays inside the window</li>
83+
</ul>
84+
</div>
85+
<AnchoredOverlay
86+
renderAnchor={props => (
87+
<Button {...props} sx={{width: 'fit-content'}}>
88+
Button
89+
</Button>
90+
)}
91+
open={open}
92+
onOpen={() => setOpen(true)}
93+
onClose={() => {
94+
setOpen(false)
95+
setLoading(true)
96+
}}
97+
>
98+
{loading ? (
99+
<>
100+
<Spinner />
101+
loading for 2000ms
102+
</>
103+
) : (
104+
<div style={{height: '300px'}}>content with 300px height</div>
105+
)}
106+
</AnchoredOverlay>
107+
</Stack>
108+
</Dialog>
109+
)
110+
}

‎packages/react/src/AnchoredOverlay/AnchoredOverlay.docs.json

+6
Original file line numberDiff line numberDiff line change
@@ -147,6 +147,12 @@
147147
"required": false,
148148
"description": "",
149149
"defaultValue": ""
150+
}, {
151+
"name": "pinPosition",
152+
"type": "boolean",
153+
"required": false,
154+
"description": "If true, the overlay will attempt to prevent position shifting when sitting at the top of the anchor.",
155+
"defaultValue": "false"
150156
}
151157
]
152158
}

‎packages/react/src/AnchoredOverlay/AnchoredOverlay.tsx

+7-1
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,10 @@ interface AnchoredOverlayBaseProps extends Pick<OverlayProps, 'height' | 'width'
8989
* If `preventOverflow` is `true`, the width of the `Overlay` will not be adjusted.
9090
*/
9191
preventOverflow?: boolean
92+
/**
93+
* If true, the overlay will attempt to prevent position shifting when sitting at the top of the anchor.
94+
*/
95+
pinPosition?: boolean
9296
}
9397

9498
export type AnchoredOverlayProps = AnchoredOverlayBaseProps &
@@ -112,11 +116,12 @@ export const AnchoredOverlay: React.FC<React.PropsWithChildren<AnchoredOverlayPr
112116
overlayProps,
113117
focusTrapSettings,
114118
focusZoneSettings,
115-
side = 'outside-bottom',
119+
side = overlayProps?.['anchorSide'] || 'outside-bottom',
116120
align = 'start',
117121
alignmentOffset,
118122
anchorOffset,
119123
className,
124+
pinPosition,
120125
preventOverflow = true,
121126
}) => {
122127
const anchorRef = useProvidedRefOrCreate(externalAnchorRef)
@@ -155,6 +160,7 @@ export const AnchoredOverlay: React.FC<React.PropsWithChildren<AnchoredOverlayPr
155160
{
156161
anchorElementRef: anchorRef,
157162
floatingElementRef: overlayRef,
163+
pinPosition,
158164
side,
159165
align,
160166
alignmentOffset,

‎packages/react/src/Overlay/Overlay.module.css

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
width: auto;
1414
min-width: 192px;
1515
height: auto;
16+
max-height: 100vh;
1617
overflow: hidden;
1718
background-color: var(--overlay-bgColor);
1819
border-radius: var(--borderRadius-large);

‎packages/react/src/Overlay/Overlay.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -75,7 +75,7 @@ const StyledOverlay = toggleStyledComponent(
7575
min-width: 192px;
7676
max-width: ${props => props.maxWidth && widthMap[props.maxWidth]};
7777
height: ${props => heightMap[props.height || 'auto']};
78-
max-height: ${props => props.maxHeight && heightMap[props.maxHeight]};
78+
max-height: ${props => (props.maxHeight ? heightMap[props.maxHeight] : '100vh')};
7979
width: ${props => widthMap[props.width || 'auto']};
8080
border-radius: 12px;
8181
overflow: ${props => (props.overflow ? props.overflow : 'hidden')};

‎packages/react/src/SelectPanel/SelectPanel.examples.stories.tsx

+95
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@ import type {OverlayProps} from '../Overlay'
88
import {TriangleDownIcon} from '@primer/octicons-react'
99
import {ActionList} from '../deprecated/ActionList'
1010
import FormControl from '../FormControl'
11+
import {Stack} from '../Stack'
12+
import {Dialog} from '../experimental'
1113

1214
const meta = {
1315
title: 'Components/SelectPanel/Examples',
@@ -442,3 +444,96 @@ export const ItemsInScope = () => {
442444
</FormControl>
443445
)
444446
}
447+
448+
export const RepositionAfterLoading = () => {
449+
const [selected, setSelected] = React.useState<ItemInput[]>([items[0], items[1]])
450+
const [open, setOpen] = useState(false)
451+
const [filter, setFilter] = React.useState('')
452+
const [filteredItems, setFilteredItems] = React.useState<typeof items>([])
453+
454+
const [loading, setLoading] = useState(true)
455+
456+
React.useEffect(() => {
457+
if (!open) setLoading(true)
458+
window.setTimeout(() => {
459+
if (open) {
460+
setFilteredItems(items.filter(item => item.text.toLowerCase().startsWith(filter.toLowerCase())))
461+
setLoading(false)
462+
}
463+
}, 2000)
464+
// eslint-disable-next-line react-hooks/exhaustive-deps
465+
}, [open])
466+
467+
React.useEffect(() => {
468+
if (!loading) {
469+
setFilteredItems(items.filter(item => item.text.toLowerCase().startsWith(filter.toLowerCase())))
470+
}
471+
// eslint-disable-next-line react-hooks/exhaustive-deps
472+
}, [filter])
473+
474+
return (
475+
<>
476+
<Stack direction="vertical" justify="space-between" style={{height: 'calc(100vh - 300px)', width: 'fit-content'}}>
477+
<h1>Reposition panel after loading</h1>
478+
<SelectPanel
479+
loading={loading}
480+
title="Select labels"
481+
placeholderText="Filter Labels"
482+
open={open}
483+
onOpenChange={setOpen}
484+
items={filteredItems}
485+
selected={selected}
486+
onSelectedChange={setSelected}
487+
onFilterChange={setFilter}
488+
/>
489+
</Stack>
490+
</>
491+
)
492+
}
493+
494+
export const SelectPanelRepositionInsideDialog = () => {
495+
const [selected, setSelected] = React.useState<ItemInput[]>([items[0], items[1]])
496+
const [open, setOpen] = useState(false)
497+
const [filter, setFilter] = React.useState('')
498+
const [filteredItems, setFilteredItems] = React.useState<typeof items>([])
499+
500+
const [loading, setLoading] = useState(true)
501+
502+
React.useEffect(() => {
503+
if (!open) setLoading(true)
504+
window.setTimeout(() => {
505+
if (open) {
506+
setFilteredItems(items.filter(item => item.text.toLowerCase().startsWith(filter.toLowerCase())))
507+
setLoading(false)
508+
}
509+
}, 2000)
510+
// eslint-disable-next-line react-hooks/exhaustive-deps
511+
}, [open])
512+
513+
React.useEffect(() => {
514+
if (!loading) {
515+
setFilteredItems(items.filter(item => item.text.toLowerCase().startsWith(filter.toLowerCase())))
516+
}
517+
// eslint-disable-next-line react-hooks/exhaustive-deps
518+
}, [filter])
519+
520+
return (
521+
<Dialog title="SelectPanel reposition after loading inside Dialog" onClose={() => {}}>
522+
<Stack direction="vertical" justify="space-between" style={{height: 'calc(100vh - 500px)', width: 'fit-content'}}>
523+
<p>other content</p>
524+
<SelectPanel
525+
loading={loading}
526+
title="Select labels"
527+
placeholderText="Filter Labels"
528+
open={open}
529+
onOpenChange={setOpen}
530+
items={filteredItems}
531+
selected={selected}
532+
onSelectedChange={setSelected}
533+
onFilterChange={setFilter}
534+
overlayProps={{anchorSide: 'outside-top'}}
535+
/>
536+
</Stack>
537+
</Dialog>
538+
)
539+
}

‎packages/react/src/SelectPanel/SelectPanel.tsx

+5-2
Original file line numberDiff line numberDiff line change
@@ -132,7 +132,7 @@ interface SelectPanelBaseProps {
132132

133133
export type SelectPanelProps = SelectPanelBaseProps &
134134
Omit<FilteredActionListProps, 'selectionVariant'> &
135-
Pick<AnchoredOverlayProps, 'open' | 'height'> &
135+
Pick<AnchoredOverlayProps, 'open' | 'height' | 'width'> &
136136
AnchoredOverlayWrapperAnchorProps &
137137
(SelectPanelSingleSelection | SelectPanelMultiSelection)
138138

@@ -185,8 +185,9 @@ export function SelectPanel({
185185
sx,
186186
loading,
187187
initialLoadingType = 'spinner',
188-
height,
189188
className,
189+
height,
190+
width,
190191
id,
191192
...listProps
192193
}: SelectPanelProps): JSX.Element {
@@ -451,7 +452,9 @@ export function SelectPanel({
451452
focusTrapSettings={focusTrapSettings}
452453
focusZoneSettings={focusZoneSettings}
453454
height={height}
455+
width={width}
454456
anchorId={id}
457+
pinPosition={!height}
455458
>
456459
<LiveRegionOutlet />
457460
{usingModernActionList ? null : (

‎packages/react/src/__tests__/__snapshots__/AnchoredOverlay.test.tsx.snap

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ exports[`AnchoredOverlay should render consistently when open 1`] = `
1313
position: absolute;
1414
min-width: 192px;
1515
height: auto;
16+
max-height: 100vh;
1617
width: auto;
1718
border-radius: 12px;
1819
overflow: hidden;

‎packages/react/src/hooks/useAnchoredPosition.ts

+42-2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import useLayoutEffect from '../utils/useIsomorphicLayoutEffect'
88
export interface AnchoredPositionHookSettings extends Partial<PositionSettings> {
99
floatingElementRef?: React.RefObject<Element>
1010
anchorElementRef?: React.RefObject<Element>
11+
pinPosition?: boolean
1112
}
1213

1314
/**
@@ -30,22 +31,61 @@ export function useAnchoredPosition(
3031
const floatingElementRef = useProvidedRefOrCreate(settings?.floatingElementRef)
3132
const anchorElementRef = useProvidedRefOrCreate(settings?.anchorElementRef)
3233
const [position, setPosition] = React.useState<AnchorPosition | undefined>(undefined)
34+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
35+
const [_, setPrevHeight] = React.useState<number | undefined>(undefined)
36+
37+
const topPositionChanged = (prevPosition: AnchorPosition | undefined, newPosition: AnchorPosition) => {
38+
return (
39+
prevPosition &&
40+
['outside-top', 'inside-top'].includes(prevPosition.anchorSide) &&
41+
// either the anchor changed or the element is trying to shrink in height
42+
(prevPosition.anchorSide !== newPosition.anchorSide || prevPosition.top < newPosition.top)
43+
)
44+
}
45+
46+
const updateElementHeight = () => {
47+
let heightUpdated = false
48+
setPrevHeight(prevHeight => {
49+
// if the element is trying to shrink in height, restore to old height to prevent it from jumping
50+
if (prevHeight && prevHeight > (floatingElementRef.current?.clientHeight ?? 0)) {
51+
requestAnimationFrame(() => {
52+
;(floatingElementRef.current as HTMLElement).style.height = `${prevHeight}px`
53+
})
54+
heightUpdated = true
55+
}
56+
return prevHeight
57+
})
58+
return heightUpdated
59+
}
3360

3461
const updatePosition = React.useCallback(
3562
() => {
3663
if (floatingElementRef.current instanceof Element && anchorElementRef.current instanceof Element) {
37-
setPosition(getAnchoredPosition(floatingElementRef.current, anchorElementRef.current, settings))
64+
const newPosition = getAnchoredPosition(floatingElementRef.current, anchorElementRef.current, settings)
65+
setPosition(prev => {
66+
if (settings?.pinPosition && topPositionChanged(prev, newPosition)) {
67+
const anchorTop = anchorElementRef.current?.getBoundingClientRect().top ?? 0
68+
const elementStillFitsOnTop = anchorTop > (floatingElementRef.current?.clientHeight ?? 0)
69+
70+
if (elementStillFitsOnTop && updateElementHeight()) {
71+
return prev
72+
}
73+
}
74+
return newPosition
75+
})
3876
} else {
3977
setPosition(undefined)
4078
}
79+
setPrevHeight(floatingElementRef.current?.clientHeight)
4180
},
4281
// eslint-disable-next-line react-hooks/exhaustive-deps
4382
[floatingElementRef, anchorElementRef, ...dependencies],
4483
)
4584

4685
useLayoutEffect(updatePosition, [updatePosition])
4786

48-
useResizeObserver(updatePosition)
87+
useResizeObserver(updatePosition) // watches for changes in window size
88+
useResizeObserver(updatePosition, floatingElementRef as React.RefObject<HTMLElement>) // watches for changes in floating element size
4989

5090
return {
5191
floatingElementRef,

0 commit comments

Comments
 (0)
Please sign in to comment.