Skip to content

Commit a1f1445

Browse files
authored
fix(size): fill viewport along an axis if shift is enabled on that axis (#3027)
1 parent 0dd000f commit a1f1445

File tree

65 files changed

+225
-24
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

65 files changed

+225
-24
lines changed

.changeset/beige-meals-draw.md

+5

packages/core/src/middleware/shift.ts

+4
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,10 @@ export const shift = (
9191
data: {
9292
x: limitedCoords.x - x,
9393
y: limitedCoords.y - y,
94+
enabled: {
95+
[mainAxis]: checkMainAxis,
96+
[crossAxis]: checkCrossAxis,
97+
},
9498
},
9599
};
96100
},

packages/core/src/middleware/size.ts

+5-10
Original file line numberDiff line numberDiff line change
@@ -82,16 +82,11 @@ export const size = (
8282
let availableHeight = overflowAvailableHeight;
8383
let availableWidth = overflowAvailableWidth;
8484

85-
if (isYAxis) {
86-
availableWidth =
87-
alignment || noShift
88-
? min(overflowAvailableWidth, maximumClippingWidth)
89-
: maximumClippingWidth;
90-
} else {
91-
availableHeight =
92-
alignment || noShift
93-
? min(overflowAvailableHeight, maximumClippingHeight)
94-
: maximumClippingHeight;
85+
if (state.middlewareData.shift?.enabled.x) {
86+
availableWidth = maximumClippingWidth;
87+
}
88+
if (state.middlewareData.shift?.enabled.y) {
89+
availableHeight = maximumClippingHeight;
9590
}
9691

9792
if (noShift && !alignment) {

packages/core/src/types.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import type {
2+
Axis,
23
ClientRectObject,
34
Coords,
45
Dimensions,
@@ -77,7 +78,9 @@ export interface MiddlewareData {
7778
escapedOffsets?: SideObject;
7879
};
7980
offset?: Coords & {placement: Placement};
80-
shift?: Coords;
81+
shift?: Coords & {
82+
enabled: {[key in Axis]: boolean};
83+
};
8184
}
8285

8386
export interface ComputePositionConfig {

packages/dom/test/functional/size.test.ts

+62-2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import {expect, test} from '@playwright/test';
22

33
import {allPlacements} from '../visual/utils/allPlacements';
44
import {click} from './utils/click';
5+
import {resize} from './utils/resize';
56
import {scroll} from './utils/scroll';
67

78
allPlacements.forEach((placement) => {
@@ -96,7 +97,8 @@ test('center aligned placements can fill the whole viewport along the crossAxis
9697
page,
9798
}) => {
9899
await page.goto('http://localhost:1234/size');
99-
await click(page, `[data-testid="flipshift-true"]`);
100+
await click(page, `[data-testid="flip-true"]`);
101+
await click(page, `[data-testid="shift-before"]`);
100102

101103
await scroll(page, {x: 325, y: 605});
102104

@@ -107,10 +109,44 @@ test('center aligned placements can fill the whole viewport along the crossAxis
107109

108110
test('edge aligned placements can fill the whole viewport along the crossAxis with shift', async ({
109111
page,
112+
}) => {
113+
await page.goto('http://localhost:1234/size');
114+
await click(page, `[data-testid="placement-bottom-start"]`);
115+
await click(page, `[data-testid="shift-before"]`);
116+
117+
await scroll(page, {x: 620});
118+
119+
expect(await page.locator('.container').screenshot()).toMatchSnapshot(
120+
`edge-aligned-shift-left-start.png`,
121+
);
122+
123+
await resize(page, {width: 420});
124+
125+
expect(await page.locator('.container').screenshot()).toMatchSnapshot(
126+
`edge-aligned-shift-left-end.png`,
127+
);
128+
129+
await resize(page, {width: 300});
130+
await scroll(page, {x: 395});
131+
132+
expect(await page.locator('.container').screenshot()).toMatchSnapshot(
133+
`edge-aligned-shift-right-start.png`,
134+
);
135+
136+
await resize(page, {width: 400});
137+
138+
expect(await page.locator('.container').screenshot()).toMatchSnapshot(
139+
`edge-aligned-shift-right-end.png`,
140+
);
141+
});
142+
143+
test('edge aligned placements can fill the whole viewport on one side along the crossAxis when shift is after', async ({
144+
page,
110145
}) => {
111146
await page.goto('http://localhost:1234/size');
112147
await click(page, `[data-testid="placement-right-start"]`);
113-
await click(page, `[data-testid="flipshift-true"]`);
148+
await click(page, `[data-testid="flip-true"]`);
149+
await click(page, `[data-testid="shift-after"]`);
114150

115151
await scroll(page, {x: 600, y: 800});
116152

@@ -160,3 +196,27 @@ test('edge aligned placements can fill the whole viewport along the crossAxis wi
160196
`left-start-shift.png`,
161197
);
162198
});
199+
200+
test('can fill the whole viewport along the main axis with shift.crossAxis', async ({
201+
page,
202+
}) => {
203+
await page.goto('http://localhost:1234/size');
204+
await click(page, `[data-testid="shift-before"]`);
205+
await click(page, `[data-testid="shift.crossAxis-true"]`);
206+
207+
expect(await page.locator('.container').screenshot()).toMatchSnapshot(
208+
`shift-crossAxis-whole.png`,
209+
);
210+
211+
await resize(page, {height: 170});
212+
213+
expect(await page.locator('.container').screenshot()).toMatchSnapshot(
214+
`shift-crossAxis-top-start.png`,
215+
);
216+
217+
await resize(page, {height: 300});
218+
219+
expect(await page.locator('.container').screenshot()).toMatchSnapshot(
220+
`shift-crossAxis-top-end.png`,
221+
);
222+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
import type {Dimensions} from '@floating-ui/core';
2+
3+
/**
4+
* Resizes an element to the provided dimensions.
5+
*/
6+
export async function resize(
7+
page: any,
8+
resizeDimensions: Partial<Dimensions>,
9+
selector = '.resize',
10+
) {
11+
await page.waitForSelector(selector);
12+
return await page.evaluate(
13+
({width, height, selector}: Partial<Dimensions> & {selector: string}) => {
14+
const resize = document.querySelector(selector);
15+
if (resize && resize instanceof HTMLElement) {
16+
if (width != null) {
17+
resize.style.width = `${width}px`;
18+
}
19+
if (height != null) {
20+
resize.style.height = `${height}px`;
21+
}
22+
}
23+
},
24+
{...resizeDimensions, selector},
25+
);
26+
}

packages/dom/test/visual/index.css

+8
Original file line numberDiff line numberDiff line change
@@ -196,6 +196,14 @@ main {
196196
display: none;
197197
}
198198

199+
.resize {
200+
resize: both;
201+
max-height: 480px;
202+
max-width: 480px;
203+
min-height: 120px;
204+
min-width: 120px;
205+
}
206+
199207
.prose {
200208
font-size: 1.125rem;
201209
color: #555;

packages/dom/test/visual/spec/Size.tsx

+83-11
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import type {Placement} from '@floating-ui/core';
22
import {
33
autoUpdate,
44
flip,
5+
limitShift,
56
shift,
67
size,
78
useFloating,
@@ -10,21 +11,34 @@ import {useState} from 'react';
1011

1112
import {allPlacements} from '../utils/allPlacements';
1213
import {Controls} from '../utils/Controls';
14+
import {useResize} from '../utils/useResize';
1315
import {useScroll} from '../utils/useScroll';
1416

17+
type ShiftOrder = 'none' | 'before' | 'after';
18+
const SHIFT_ORDERS: ShiftOrder[] = ['none', 'before', 'after'];
19+
1520
export function Size() {
1621
const [rtl, setRtl] = useState(false);
1722
const [placement, setPlacement] = useState<Placement>('bottom');
18-
const [addFlipShift, setAddFlipShift] = useState(false);
23+
const [addFlip, setAddFlip] = useState(false);
24+
const [addShift, setAddShift] = useState<ShiftOrder>('none');
25+
const [shiftCrossAxis, setShiftCrossAxis] = useState(false);
26+
const [shiftLimiter, setShiftLimiter] = useState(false);
1927

2028
const hasEdgeAlignment = placement.includes('-');
2129

30+
const shiftOptions = {
31+
padding: 10,
32+
crossAxis: shiftCrossAxis,
33+
limiter: shiftLimiter ? limitShift({offset: 50}) : undefined,
34+
};
35+
2236
const {x, y, strategy, update, refs} = useFloating({
2337
placement,
2438
whileElementsMounted: autoUpdate,
2539
middleware: [
26-
addFlipShift && flip({padding: 10}),
27-
addFlipShift && !hasEdgeAlignment && shift({padding: 10}),
40+
addFlip && flip({padding: 10}),
41+
addShift === 'before' && shift(shiftOptions),
2842
size({
2943
apply({availableHeight, availableWidth, elements}) {
3044
Object.assign(elements.floating.style, {
@@ -34,19 +48,20 @@ export function Size() {
3448
},
3549
padding: 10,
3650
}),
37-
addFlipShift && hasEdgeAlignment && shift({padding: 10}),
51+
addShift === 'after' && shift(shiftOptions),
3852
],
3953
});
4054

4155
const {scrollRef, indicator} = useScroll({refs, update, rtl});
56+
useResize(scrollRef, update);
4257

4358
return (
4459
<>
4560
<h1>Size</h1>
4661
<p></p>
4762
<div className="container" style={{direction: rtl ? 'rtl' : 'ltr'}}>
4863
<div
49-
className="scroll"
64+
className="scroll resize"
5065
data-x
5166
style={{position: 'relative'}}
5267
ref={scrollRef}
@@ -64,8 +79,13 @@ export function Size() {
6479
left: x ?? '',
6580
width: 400,
6681
height: 300,
67-
...(addFlipShift && {
68-
width: 600,
82+
...(addShift !== 'none' && {
83+
width:
84+
addShift === 'before' && shiftCrossAxis
85+
? 100
86+
: addShift === 'before' && hasEdgeAlignment
87+
? 360
88+
: 600,
6989
height: 600,
7090
}),
7191
}}
@@ -107,21 +127,73 @@ export function Size() {
107127
))}
108128
</Controls>
109129

110-
<h2>Add flip and shift</h2>
130+
<h2>Add flip</h2>
111131
<Controls>
112132
{[true, false].map((bool) => (
113133
<button
114134
key={String(bool)}
115-
data-testid={`flipshift-${bool}`}
116-
onClick={() => setAddFlipShift(bool)}
135+
data-testid={`flip-${bool}`}
136+
onClick={() => setAddFlip(bool)}
117137
style={{
118-
backgroundColor: addFlipShift === bool ? 'black' : '',
138+
backgroundColor: addFlip === bool ? 'black' : '',
119139
}}
120140
>
121141
{String(bool)}
122142
</button>
123143
))}
124144
</Controls>
145+
146+
<h2>Add shift</h2>
147+
<Controls>
148+
{SHIFT_ORDERS.map((str) => (
149+
<button
150+
key={str}
151+
data-testid={`shift-${str}`}
152+
onClick={() => setAddShift(str)}
153+
style={{
154+
backgroundColor: addShift === str ? 'black' : '',
155+
}}
156+
>
157+
{str}
158+
</button>
159+
))}
160+
</Controls>
161+
162+
{addShift !== 'none' && (
163+
<>
164+
<h3>shift.crossAxis</h3>
165+
<Controls>
166+
{[true, false].map((bool) => (
167+
<button
168+
key={String(bool)}
169+
data-testid={`shift.crossAxis-${bool}`}
170+
onClick={() => setShiftCrossAxis(bool)}
171+
style={{
172+
backgroundColor: shiftCrossAxis === bool ? 'black' : '',
173+
}}
174+
>
175+
{String(bool)}
176+
</button>
177+
))}
178+
</Controls>
179+
180+
<h3>shift.limiter</h3>
181+
<Controls>
182+
{[true, false].map((bool) => (
183+
<button
184+
key={String(bool)}
185+
data-testid={`shift.limiter-${bool}`}
186+
onClick={() => setShiftLimiter(bool)}
187+
style={{
188+
backgroundColor: shiftLimiter === bool ? 'black' : '',
189+
}}
190+
>
191+
{String(bool)}
192+
</button>
193+
))}
194+
</Controls>
195+
</>
196+
)}
125197
</>
126198
);
127199
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
import type {MutableRefObject} from 'react';
2+
import {useEffect, useRef} from 'react';
3+
4+
export const useResize = (
5+
ref: MutableRefObject<Element | null>,
6+
callback: ResizeObserverCallback,
7+
) => {
8+
const callbackRef = useRef<ResizeObserverCallback | null>(null);
9+
10+
useEffect(() => {
11+
callbackRef.current = callback;
12+
});
13+
14+
useEffect(() => {
15+
let cleanup;
16+
if (typeof ResizeObserver === 'function') {
17+
const el = ref.current;
18+
const observer = new ResizeObserver(
19+
(entries, observer) => callbackRef.current?.(entries, observer),
20+
);
21+
el && observer.observe(el);
22+
cleanup = () => {
23+
el && observer.unobserve(el);
24+
};
25+
}
26+
return cleanup;
27+
}, [ref, callback]);
28+
};

0 commit comments

Comments
 (0)