Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[css-grid-3] intrinsic track sizing algorithm for masonry can go exponential in complexity. #10053

Closed
bfgeek opened this issue Mar 8, 2024 · 10 comments
Labels

Comments

@bfgeek
Copy link

bfgeek commented Mar 8, 2024

https://drafts.csswg.org/css-grid-3/#track-sizing

Currently you need to place everything in every possible position. With subgrid this means that the complexity can go ~exponential (or sub-exponential).

<div style="grid-template: masonry / repeat(10, auto);">
  <div style="grid-template: subgrid / subgrid; grid-column: auto / span 6;"> <!-- 5 positions -->
    <div style="grid-template: subgrid / subgrid; grid-column: auto / span 3;"> <!-- 4 positions -->
      <div></div> <!-- 3 positions -->
    </div>
  </div>
</div>

In the above example there are 60 positions you need to test (each subgrid level might have different margin/border/padding which needs to be taken into account for the items contribution in a particular position along with many other factors).

In Blink we've spent most of our performance work in layout fixing exponential time bugs - developers really care about layout time performance. While these issues were quality of implementation bugs (and not a core problem with a particular spec), having exponential time complexity in a spec seems bad[1].

[1] - Exponential time complexity doesn't care how powerful your CPU is - it'll always win.

@bfgeek bfgeek added the css-grid-3 Masonry Layout label Mar 8, 2024
@fantasai fantasai changed the title [css-grid-3] Current track sizing algorithm can go exponential in complexity. [css-grid-3] intrinsic track sizing algorithm for masonry can go exponential in complexity. Mar 14, 2024
@ByteEater-pl
Copy link

ByteEater-pl commented Mar 17, 2024

Exponential in what? The subgrid nesting level? If so, is it really such a bad thing? Do deeply nested subgrids with masonry have compelling use cases, leading to people using them without realizing how poorly they'd perform?

@fantasai
Copy link
Collaborator

Discussed with the Apple/WebKit team...

We agree that subgridded masonry within masonry can create an exponential calculation. However this only affects intrinsic track sizing, so the dangers here are limited to: intrinsically sized masonry tracks combined with multiple levels of subgridded masonry nesting combined with large numbers of tracks.

We don't think the cases where these calculations become unmanageable will be common:

  • We expect most use cases with large numbers of tracks to use non-intrinsic track sizes.
  • We don't expect deep nesting of masonry within masonry in general, and even less likely with intrinsic track sizing.

To address the "but what if concerns", we're OK with introducing a UA-defined limit on e.g. the depth of masonry subgrid items that can contribute sizing back up to the master grid. A limit like this can prevent exponential explosion, while still allowing

  • A limited amount of masonry nesting with intrinsic track sizing, if needed. (We don't expect much depth is needed here here, maybe 1 or 2 levels would likely satisfy any use cases.)
  • Limitless depth of subgridding with fixed-size tracks.
  • Interleaving Grid and Masonry subgrids in the nesting structure, which we believe is an important use case, and which requires compatible track sizing between Grid and Masonry.

@kizu
Copy link
Member

kizu commented Apr 24, 2024

I really think that spanning items should not contribute to the width of the columns in masonry. Even in a regular grid, this is the behavior that I find unnecessary and often harmful, and I don't think there are many legit use cases for wanting this in a masonry layout. As an author, when I want some element to span other columns/rows, I want it to adapt to whatever there is, not to contribute. (copying this part from #9041 (comment))

When some auto column has only spanning items contributing to it, I would rather consider it be minmax(0px, 1fr) (when with regular auto) or just 0px when it is in a repeating context.

@tabatkins
Copy link
Member

I really think that spanning items should not contribute to the width of the columns in masonry.

I disagree. Like, if you have repeat(5, min-content), and an item spans two of those columns, we do want to make sure the track size is large enough that the two columns it spans will contain its min-content size. After all, the whole reason you gave it a span might be because you expected it to be large. If we don't contribute its size to the tracks, then it'll happen to work fine if it's min-content size is less than twice that of the non-spanning items, but it'll overflow if it is larger than double. That seems pretty bad.

@kizu
Copy link
Member

kizu commented Apr 26, 2024

In my practice, I never encountered use cases for this. Usually, to be resilient, elements tend to have a small min-content, adapting in one way or another to the context. Even the opposite: numerous times I had to give these spanning items contain to stop this behavior. I would be interested in seeing any real-life cases that rely on the spanning items contributing their min-content.

While I think it could be useful to have a control over if any particular spanning item contributes its min-content to the columns for regular grids, I am not talking about changing how it works now, but about a compromise that will enable masonry-type grids with spanning items. I will always take this limitation compared to the absence of a feature completely.

@tabatkins
Copy link
Member

I am not talking about changing how it works now, but about a compromise that will enable masonry-type grids with spanning items. I will always take this limitation compared to the absence of a feature completely.

As we outlined over in #9041 (comment), spanning items and intrinsic tracks are totally fine perf-wise, if all the tracks are the same intrinsic size. (And spanning items across fixed/flexible tracks aren't problematic at all.) It's just when you mix an intrinsic track with some different-sized tracks (intrinsic or not) that you run into these issues.

And so far, we haven't seen any example of Masonry grids that want to exceed that limitation - if they want intrinsic sizes, they want them all the same; if they want different sizes, they want fixed/flexible sizes, rather than intrinsic ones. So we should be just fine handling spanning items properly.

@tabatkins
Copy link
Member

Me, @fantasai, and @bfgeek chatted in person about this today. We reached a few conclusions:

  1. Subgrid/submasonry are indeed problematic, and will need some degree of limitations preventing them from contributing information back up into their parent masonry. We'll need to figure out exactly what they're not allowed to communicate.
  2. Ignoring that case, spanning items can reasonably be grouped into per-span groups, which can then be measured against each possible position. This has complexity of (# of positions)*(# of distinct spans), approximately, which is reasonable.
  3. Non-spanning items have never been problematic.

So, with some restrictions from (1), we think it is reasonable to allow all possible track size combinations; in particular, track lists like 50px auto auto 100px should be allowed.

@fantasai
Copy link
Collaborator

(I posted some details on performance analysis and optimizations here, fwiw.)

@fantasai
Copy link
Collaborator

Proposed resolution? Allow mixed track sizing in masonry; investigate applying limitations on the size contributions of subgrid/submasonry child items to avoid pathological performance characteristics.

@css-meeting-bot
Copy link
Member

The CSS Working Group just discussed [css-grid-3] intrinsic track sizing algorithm for masonry can go exponential in complexity., and agreed to the following:

  • RESOLVED: allowing mixed track sizing, as long as the algorithm does not allow position-dependent effects
  • RESOLVED: add a note about the grouping optimization and baseline handling
The full IRC log of that discussion <noamr> fantasai: TabAtkins, Iank_ and I discussed this on monday, and decided that we think this is doable, want to resolve that we allow mixed track sizing on masonry and investigate
<noamr> fantasai: it is currently a point of contention
<fantasai> Proposed resolution? Allow mixed track sizing in masonry; investigate applying limitations on the size contributions of subgrid/submasonry child items to avoid pathological performance characteristics.
<noamr> IanK_: it would be good to explicitly spec how you're supposed to group things and then apply the sizing algo, otherwise it would get lost and implementers won't do it correctly
<noamr> IanK_: we need to list out how you're supposed to do this, and also mention the positional independence thing
<noamr> IanK_: to get the perf characteristics, you have to group everything by how it's sized, and then apply the track sizing algorithm
<noamr> IanK_: it's different from today where all the items are having applied the track sizing algorithm together
<noamr> fantasai: it's an optimization of the currently spec'ed algo, we can add it as a note for a possible optimization
<noamr> fantasai: I want to keep the algo easy to read
<noamr> IanK_: for an implementor, if they have to dig notes to find important optimizations it's difficult
<noamr> fantasai: there are further optimizations, this is not the only possible optimizations
<noamr> fantasai: It's ok to outline this, but it's not our jobs in specs to detail exactly how to optimize stuff. we might not come to the right solution. This problem exists in all spec
<noamr> fantasai: translating spec directly into code is a bad habit
<TabAtkins> It is useful to give guidance when something should be done in a particular way, fwiw.
<fantasai> Yeah, I'm fine with giving guidance
<noamr> IanK_: I don't want to lose the positional idenependence stuff as it requires for the algorithm to work
<TabAtkins> If there *is* a behavioral difference, tho, then it does matter and needs to be in teh algo, obvs.
<fantasai> obvs
<noamr> IanK_: positional independence is key to get this opt to work
<noamr> IanK_: otherwise we're back to CSS2.1 where there are no algorithm specified
<noamr> IanK_: we can write the basic constraints and build the other constraints on top of that
<noamr> astearns: hearing a few things to resolve. is this merely an optimization?
<noamr> fantasai: yes
<noamr> IanK_: as long as we have positional independence
<noamr> IanK_: the current algorithm does have positional dependence
<noamr> astearns: is the first resolution that we allow mixed track size, with a normative bit on positional independence
<noamr> proposed resolution: allow mixed track sizing on masonry, as long as the algorithm does not allow position-dependent effects
<noamr> RESOLVED: allowing mixed track sizing, as long as the algorithm does not allow position-dependent effects
<noamr> astearns: proposed resolution, add a note about the grouping optimization and baseline handling
<noamr> RESOLVED: add a note about the grouping optimization and baseline handling
<noamr> fantasai: investigate limitations on subgrid contribution of subgrid items to avoid perf issues
<noamr> fantasai: we need to dig more
<noamr> astearns: any other resolutions needed on this one?
<TabAtkins> ScribeNick: TabAtkins

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

No branches or pull requests

7 participants