Skip to content

[#279] Fix position-try demos and polyfill logic#424

Open
jgerigmeyer wants to merge 5 commits into
mainfrom
jgerigmeyer/279-fix-position-fallback
Open

[#279] Fix position-try demos and polyfill logic#424
jgerigmeyer wants to merge 5 commits into
mainfrom
jgerigmeyer/279-fix-position-fallback

Conversation

@jgerigmeyer

@jgerigmeyer jgerigmeyer commented Jun 27, 2026

Copy link
Copy Markdown
Member

Resolves #279

Background

The position-try demos didn't update on scroll, and were broken in native browsers too, not just under the polyfill. Investigating in real native Chrome (150) confirmed two distinct problems:

  1. Demo authoring bug (affects native). Per spec, position-try options are evaluated by checking whether the target overflows its inset-modified containing blocknot the scrollport of an ancestor scroll container. Every demo target's containing block was its .scroll-container (position: relative), which also contains the anchor. So scrolling moved the anchor and target together and the target never overflowed its containing block → fallbacks never triggered, natively or otherwise. The polyfill only appeared to work because it measured overflow against the scrollport, diverging from native.

  2. Polyfill inheritance leak. Properties shifted into custom properties (height, insets, margins, anchor-name, …) are all non-inherited in CSS, but custom properties inherit by default. So height: 400px on a .scroll-container leaked through --height-<uuid> into every descendant target and got stamped into every generated fallback.

Changes

  • Demos (public/position-try*.css): make the position-try scroll containers position: static so each target's containing block is promoted to the surrounding .demo-elements wrapper (an ancestor outside the scroller). Now scrolling pushes the target to overflow and fallbacks trigger — natively and under the polyfill. Added overflow: clip on the wrapper so a target whose anchor has scrolled out of view doesn't float over other content, and gave the combined demo's anchor an offset so its base position fits at rest. Removed the "this example is broken" warning in index.html.

  • Non-inherited custom properties (src/cascade.ts, src/dom.ts): register the shifted custom properties with CSS.registerProperty({ syntax: '*', inherits: false }) so a value set on an ancestor is no longer read back as if it were set directly on a descendant. Registration is global and runs once; it no-ops where CSS.registerProperty is unavailable. Updated the now-inaccurate inheritance comments.

  • Overflow basis (src/polyfill.ts): rewrote checkOverflow to measure the target's margin-box against its containing block (offsetParent) using getBoundingClientRect, matching the spec/native behavior instead of floating-ui's scrollport-based detectOverflow. Both rects are read in viewport coordinates, so the page's scroll offset cancels out (the previous standalone detectOverflow call produced wildly wrong values once the containing block was no longer a near-viewport scroll container). Dropped the now-unused detectOverflow/MiddlewareState imports and platformWithCache.

  • Test (tests/e2e/polyfill.test.ts): the target's offsetParent is now the (non-scrollable) wrapper, so the @position-fallback test scrolls .scroll-container directly and captures the pre-polyfill position rather than hardcoding it.

Behavior change to flag for review

checkOverflow now matches native semantics (overflow vs. containing block). This preserves existing behavior for the common cases — an absolutely-positioned target whose containing block is a scroll container still flips against that scroller's visible box, and a fixed target flips against the viewport — so nothing in the suite regressed. But it is a change to core polyfill logic and worth a careful look.

Verification

  • Native (Chrome 150): all position-try targets now flip on scroll. The combined demo flips but non-monotonically, which is inherent to its dual-inline-axis options plus native scroll-offset compensation.
  • Polyfill (Chromium 124): all five targets flip cleanly on scroll; no height: 400px leaks into generated fallbacks; no console errors.

@netlify

netlify Bot commented Jun 27, 2026

Copy link
Copy Markdown

Deploy Preview for anchor-polyfill ready!

Name Link
🔨 Latest commit 363ad7c
🔍 Latest deploy log https://app.netlify.com/projects/anchor-polyfill/deploys/6a42e1248bc3ae000857fca1
😎 Deploy Preview https://deploy-preview-424--anchor-polyfill.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify project configuration.

@netlify

netlify Bot commented Jun 27, 2026

Copy link
Copy Markdown

Deploy Preview for anchor-position-wpt canceled.

Name Link
🔨 Latest commit 363ad7c
🔍 Latest deploy log https://app.netlify.com/projects/anchor-position-wpt/deploys/6a42e124923feb0008ef8193

Comment thread src/fallback.ts
Comment on lines -579 to -582
const anchorPosition: AnchorPosition = {};
if (order) {
anchorPosition.order = order;
}

Copy link
Copy Markdown
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Claude's explanation of this change:

In the polyfill, a comma-separated selector list that uses position-try-fallbacks (e.g. #a, #b { position-try-fallbacks: flip-block, … }) caused the second target to be positioned using the first target's fallbacks. Visibly, in Firefox the second target stayed static (un-anchored) and targets could revert to a non-anchored position when a fallback couldn't be placed.

Cause

In parsePositionFallbacks, the anchorPosition object was declared outside the selectors.forEach loop, so it was shared across every selector in the list. Because try-tactic fallbacks are keyed per selector (${selector}-${tactics}), each selector's fallbacks accumulated into that one shared object — and since validPositions[selectorA] and validPositions[selectorB] both referenced it, the second target's fallback list began with the first target's fallbacks (which resolve against the wrong anchor). @position-try (at-rule) fallbacks were unaffected because their names are shared across selectors.

Fix

Give each selector its own anchorPosition, and separate the two concerns that were conflated under a single dedup guard:

  • Add a fallback to this selector's position — now tracked per selector.
  • Inject the generated @position-try rule into the stylesheet — still deduped once per stylesheet.

Tests

Added a regression unit test asserting that each selector in a shared rule gets only its own try-tactic fallback. Existing unit (186) and e2e (37) suites pass.

Comment thread src/cascade.ts
Comment on lines +58 to +62
CSS.registerProperty({
name: customProperty,
syntax: '*',
inherits: false,
});

Copy link
Copy Markdown
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TIL about CSS.registerProperty 🎉

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was avoiding relying on CSS.registerProperty - it moves Safari support from Safari 10 to Safari 16.4 (Release date: 2023-03-27), and Firefox Support from Firefox 54 to Firefox 128 (Release date: 2024-07-09).

We would need to add this as a caveat. I'm not sure there's another way...

@jgerigmeyer jgerigmeyer marked this pull request as ready for review June 30, 2026 16:45
@jgerigmeyer jgerigmeyer requested a review from jamesnw June 30, 2026 16:45

@jamesnw jamesnw left a comment

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some testing notes-

In Firefox 124- Demo doesn't work. Adding padding (removing .tight from .demo-elements) makes it work, but the target goes outside of the scroll container before falling back. When it falls back, it still has the inheritance bug (CSS.registerProperty came to Firefox in 128). Fallback only happens on the y-axis.

In Firefox 128- Demo works. Adding padding makes the the targets go outside of the scroll container before falling back.

I'm not sure if this is because the custom properties are required to make it fallback correctly, or if the incorrect values are the making the fallback happen. We should figure out how crucial non-inheritance is to this change, and see if there's an alternate way of addressing that.

Comment thread src/cascade.ts
* this only needs to run once.
*/
let propertiesRegistered = false;
export function registerShiftedProperties() {

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of eagerly registering all properties, would it make sense to check if it's registered in shiftUnsupportedProperties first, and only registering those?

Comment thread src/cascade.ts
Comment on lines +58 to +62
CSS.registerProperty({
name: customProperty,
syntax: '*',
inherits: false,
});

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was avoiding relying on CSS.registerProperty - it moves Safari support from Safari 10 to Safari 16.4 (Release date: 2023-03-27), and Firefox Support from Firefox 54 to Firefox 128 (Release date: 2024-07-09).

We would need to add this as a caveat. I'm not sure there's another way...

content. The wrapper is the target's containing block, so this does not
affect where fallbacks are evaluated. */
#position-try-tactics-combined .demo-elements {
overflow: clip;

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It seems like this shouldn't be required to have fallback work within a scrollport, but it seems to be the case. This is repeated a lot, I'm wondering if we should make it a class?

Also, the lack of padding on the elements (tight) is essential to making this pattern work- I think moving it in here would make sense.

Comment thread src/polyfill.ts
@@ -1,7 +1,5 @@
import {
autoUpdate,
detectOverflow,

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice, this was simple enough to replace. I wonder if this reduces the polyfill size, and if there are other simple-to-replace parts that would help with that?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

[BUG] Position-try-fallbacks demo breaks at certain screen widths

2 participants