-
Notifications
You must be signed in to change notification settings - Fork 683
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-contain] CQ vs shadow boundaries #5984
Comments
And with CQ you probably mean container queries, right? Sebastian |
Yes. (Cadit quaestio) |
To clarify, neither approach uses a selector to target the container. Both of these would select a @container (width > 40em) { .card { ... } }
.card:container(width > 40em) { ... } Maybe they still have different shadow-DOM implications under the hood? |
For the implicit case, it follows from the layout tree, but what if you have a web component where you want to have your styles depend on the closest container and there is no such container inside the component. Do you really want to query a container outside of your component? Would it be natural that the shadow host is the outermost container to query and that the CQ fallback happens on the shadow host level? For the container-name proposal it's a question of name clashes. If you have a @container mycontainer (inline-size >= 200px), you probably don't want to match an arbitrary mycontainer outside of your web component when you don't have a mycontainer in your ancestor chain inside the component? |
I think I would often want the ability to query the host page - this feature is all about access to contextual information - but that the host element might suffice in most situations (especially for dimensional queries), if we need that limitation. If I was building a component library with shadow-DOM, I would want content-based components to query external layout containers - and I would likely try to do that with containers named by type (e.g. |
I think a bunch of people are playing with container queries, have we collected any feedback on specifically this from others? It seems like it would be very valuable... Has someone asked for either? |
The current implementation in Blink uses the flat tree for looking up containers. That was the most straightforward way to implement it. I agree that ::part() and pseudo elements like ::placeholder are problematic if they match containers inside the shadow tree. This issue should probably be considered together with issue #6711. |
I think this is an interesting case. Two selectors for the same element in different scopes querying the closest container for inline-size: <!doctype html>
<style>
host-element, host-child {
container-type: inline-size;
}
@container (width <= 300px) {
host-child {}
}
</style>
<host-element>
<template shadowroot="open">
<style>
inner-container { container-type: inline-size; }
@container (width <= 200px) {
::slotted(host-child) {}
}
</style>
<inner-container>
<slot></slot>
</inner-container>
</template>
<host-child>Light content</host-child>
</host-element> Do we want host-child to have host-element as its container when the query/selector is in the host-child tree, and have inner-container as the container for the ::slotted rule? If we just look for the container in the element's own tree, we would not be able to style slotted elements inside the shadow tree. Is that acceptable? |
With container-name it gets even more interesting. Should container-name be tree-scoped names? In the example below, using tree-scoped names would mean that the container rules for host-child below would match differerent containers based on the scope of the rule: <!doctype html>
<style>
host-element {
container-type: inline-size;
container-name: a;
}
@container a (width <= 300px) {
host-child {}
}
</style>
<host-element>
<template shadowroot="open">
<style>
inner-container {
container-type: inline-size;
container-name: a;
}
@container a (width <= 200px) {
::slotted(host-child) {}
}
</style>
<inner-container>
<slot></slot>
</inner-container>
</template>
<host-child>Light content</host-child>
</host-element> |
Container names don't need to be tree-scoped; they're not globally registered/visible, but are instead solely looked up via an ancestor search. We could potentially still censor names past a shadow boundary if we wanted, but that would be an independent decision; it's not forced by this issue. |
I like the idea of allowing slotted elements to be styled based on the slot location in the shadow tree. It seems likely to me that both styles from inside and outside the shadow boundary are attempting to determine actual space available, and in most cases the shadow parent will give the better answer - but that may not be true in all cases. |
@tabatkins By ancestor search, do you mean looking up ancestors in the light tree without walking past shadow roots? If so, it means host-child will never match any containers inside host's shadow tree: <style>
@container name (min-width: 100px) { host-child {} }
@container (min-width: 100px) { host-child {} }
</style>
<host-element><host-child></host-child></host-element> What about slotted rules inside a shadow tree: <style>
@container name (min-width: 100px) { ::slotted(div) {} }
@container (min-width: 100px) { ::slotted(div) {} }
</style>
<div>
<slot></slot>
</div> Do we start walking from the slot instead of the slotted element to find the container so that it's possible to match containers inside the shadow tree? |
The current spec covers first case if "descendants" are light tree descendants. If ::slotted rules should match containers walking from the slot instead of the slotted element being matched, the spec needs to say something about that. |
Possibly stupid question: why is this worse than e.g. |
No, I was referring to jumping up shadow roots as well; CSS almost always operates on the flat tree and doesn't care about shadow boundaries, particularly when layout is involved. That said, while shadows being able to see ancestor light containers seems reasonable, i agree that parts being able to see shadow containers isn't good.
Name collisions among variables, while possible, aren't too likely in general. On the other hand, the common use-case for CQs doesn't even refer to the container by name, so you'll often end up accidentally being intercepted by the shadow's container, when you intended to be querying off of some light-dom container that's an ancestor to the shadow. |
OK, so what about specifying that container selection can't see containers in tree scopes below the scope where the |
That's per definition shadow-including descendants/ancestors |
No, that's basically flat tree - a ::part(), selecting an element in the shadow, would see shadow ancestors before it escapes back into the light tree and starts seeing light ancestors. Anders is talking about something more akin to the tree-scoped names rule, where the tree scope of the @container rule affects what container elements are visible to it. A I think that makes sense? |
I meant that the part about: "... but skips containers that are below the "originating tree scope" ..." is about shadow-including descendants/ancestors. |
(After decoding cryptic DOM spec language): Shadow-including ancestors is almost what we're talking about, but we need to skip the "originating tree scope" based on where the container rule comes from, not based on where the element is in the tree. Otherwise ::slotted will not work as expected. |
After an off-line discussion with @andruud we think the discussed behavior could be specified as: In https://drafts.csswg.org/css-contain-3/#query-container, change: "... styling its descendants ..." to: "... styling its shadow-including descendants ..." And add: "For selectors with pseudo elements, query containers can be established in the shadow-including inclusive ancestors of the originating element." Resolving on this could also resolve issue #6711.
I have uploaded a Chromium CL with supporting tests for the Shadow DOM cases here: https://chromium-review.googlesource.com/c/chromium/src/+/3304155 |
👍 But regarding:
we need to describe how to look for containers without mentioning the "kind" of selector. For e.g. |
I don't think it looks like the spec is written with multiple pseudo elements in mind: https://drafts.csswg.org/selectors-4/#pseudo-element-attachment As you say, we either need to clarify that |
The implementation behind a flag for container queries in Blink now implements the behavior proposed in #5984 (comment) Tentative tests landed here: https://wpt.live/css/css-contain/container-queries/container-for-shadow-dom.tentative.html See #6711 (comment) for tests for pseudo elements. |
Excellent! And I agree that the proposed behavior (use shadow-inclusive descendants/ancestors, with pseudos jumping straight to their originating element) sounds good; it uses only information available at selector-evaluation time, and respects tree scopes reasonably. |
The CSS Working Group just discussed
The full IRC log of that discussion<dael> Topic: [css-contain] CQ vs shadow boundaries<dael> github: https://github.com//issues/5984 <dael> RESOLVED: Accept proposal to in all cases use originating element |
I think this is true for basically all queries: the goal is to measure your nearest context. But style queries and query units in particular expose the ways a container query is more akin to 'inheritance' than selection. The shadow DOM has always provided context for slotted elements, and containers are explicitly designed for providing that sort of context. If we need to make container names private to a shadow tree, or allow container names to be marked as private, that feels like a separate issue to me. |
The CSS Working Group just discussed
The full IRC log of that discussion<flackr> emilio: the current behavior of cq and cq units is weird for authors, especially the units. For stuff like style queries, current behavior is using the regular dom which kind of makes sense. But for style queries you really want the flat tree<miriam> q+ <TabAtkins> q+ <flackr> emilio: we opened this as a result of another issue filed on the units which are right now defined to behave like the queries. The original resolution isn't the best for authors, I think it's a bit weird. miriam thinks so as well and maybe this is worth reverting. <flackr> emilio: firefox implements what i'm proposing and we haven't run into any issues so i think the compat risk is small <astearns> ack miriam <flackr> miriam: I've felt strongly from the start that we went the wrong way. CQ are very much akin to inheritance in a lot of ways and should be treated in that way rather than as selectors in terms of relation to shadow dom. They're all about context and shadow elements and slotting creates context which should be able to be accessed <astearns> ack TabAtkins <kizu> q+ <flackr> TabAtkins: ultimately I agree [missed] <astearns> ack kizu <flackr> kizu: may be worth mentioning named CQ where we cannot assess a named entity from outside of the shadow dom <flackr> astearns: any other comments / concerns? <flackr> emilio: Would be interested in hearing rune's thoughts, can weigh in on issue <flackr> Proposed resolution: Container queries and units use the flat tree <flackr> RESOLVED: Container queries and units use the flat tree <emilio> scribenick: emilio |
As per CSSWG resolution in: w3c/csswg-drafts#5984 (comment) Differential Revision: https://phabricator.services.mozilla.com/D212412 bugzilla-url: https://bugzilla.mozilla.org/show_bug.cgi?id=1790886 gecko-commit: bc97304e5b3d4d2110fd8f2120ba9914406a116a gecko-reviewers: jwatt
…lution. r=jwatt As per CSSWG resolution in: w3c/csswg-drafts#5984 (comment) Differential Revision: https://phabricator.services.mozilla.com/D212412
As per CSSWG resolution in: w3c/csswg-drafts#5984 (comment) Differential Revision: https://phabricator.services.mozilla.com/D212412 bugzilla-url: https://bugzilla.mozilla.org/show_bug.cgi?id=1790886 gecko-commit: bc97304e5b3d4d2110fd8f2120ba9914406a116a gecko-reviewers: jwatt
…lution. r=jwatt As per CSSWG resolution in: w3c/csswg-drafts#5984 (comment) Differential Revision: https://phabricator.services.mozilla.com/D212412
…lution. r=jwatt As per CSSWG resolution in: w3c/csswg-drafts#5984 (comment) Differential Revision: https://phabricator.services.mozilla.com/D212412
...Unfortunately they do not work across slot boundaries. The container unit is sized by the light dom container, not the shadow dom container. See w3c/csswg-drafts#5984 Solution: use percentage units instead. Oddly enough, these do work across shadow boundaries.
* Add exports * Stop using container units ...Unfortunately they do not work across slot boundaries. The container unit is sized by the light dom container, not the shadow dom container. See w3c/csswg-drafts#5984 Solution: use percentage units instead. Oddly enough, these do work across shadow boundaries. * Update all classes to use Common prefix
I am trying to write up a PR for the spec change, but not entirely sure how to express the pseudo element case. We cannot keep the "ultimately originating element" as the starting candidate as that would make us incorrectly skip flat tree ancestors for I was thinking about keeping the current wording for pseudo elements which are not part-like pseudo-elements, but what about pseudo elements for UA controls like which represent Perhaps it works with just explicitly call out |
Per resolution in: w3c#5984 (comment)
Per resolution in: #5984 (comment) Co-authored-by: Rune Lillesveen <[email protected]>
Tests for size and style queries have also been updated. Closing. |
Can a CQ see past shadow boundaries? If so, exactly how?
If the CQ is done via selectors, the answer is obvious - you can only CQ against elements you can see via selectors. So from within a shadow, the highest ancestor you can CQ against is your host element, no higher.
If CQ is done via an at-rule, the answer is not quite as clear; theoretically, we can walk the box tree (flat tree) to find ancestors. However, we do not want a ::part() to be able to see an ancestor within the targeted shadow (it violates shadow encapsulation). And letting a shadow see ancestors outside in the light DOM gives us similar power to :host-context(), which already makes browser vendors unhappy, so maybe we don't want that either. And that's all the cases, so presumably we don't want it to work in either case.
As a related issue, we'll need to specifically define that the CQ pseudo-class matches on host elements, since they're featureless and don't match anything besides :host by default otherwise.
The text was updated successfully, but these errors were encountered: