diff --git a/AtSheet/explainer.md b/AtSheet/explainer.md new file mode 100644 index 00000000..84456672 --- /dev/null +++ b/AtSheet/explainer.md @@ -0,0 +1,256 @@ +# `@sheet` + +## Authors: + +- Andy Luhrs +- Kurt Catti-Schmidt + +Much of this explainer is consolidating and iterating on a CSSWG discussion around [Justin Fagnani](https://github.com/justinfagnani)'s proposal for multiple stylesheets in a single file [here](https://github.com/w3c/csswg-drafts/issues/5629). + +## Participate +- [Issue tracker](https://github.com/MicrosoftEdge/MSEdgeExplainers/labels/AtSheet) +- [Discussion forum](https://github.com/w3c/csswg-drafts/issues/5629) + +## Status of this Document + +This document is intended as a starting point for engaging the community and +standards bodies in developing collaborative solutions fit for standardization. +As the solutions to problems described in this document progress along the +standards-track, we will retain this document as an archive and use this section +to keep the community up-to-date with the most current standards venue and +content location of future work and discussions. + +* This document status: **Active** +* Expected venue: [CSS Working Group](https://www.w3.org/Style/CSS/) +* Current version: this document + +## Introduction +When developing web components, web authors often encounter challenges with distributing global styles into shadow roots and sharing styles across different shadow roots. Declarative shadow DOM (DSD) enables creation of shadow DOM without JavaScript. However, adding styles to DSD requires the developer to either use JavaScript to put a shared stylesheet into `adoptedStyleSheets`, or to duplicate the styles in a ` +``` + +This will import only the rules for `foo` - in this case, the `div { color: red; }` rule. This will *not* import any rules from `sheet.css` outside of "foo". + +### Importing a specific sheet via the `` tag +```html + +``` + +This will also import only this rules for "foo" - in this case, the `div { color: red; }` rule. This will *not* import any rules from `sheet.css` outside of "foo". + +### Importing a base set of inline styles into a Declarative Shadow DOM +Shadow DOM isolates styles, but fragment identifiers from the light DOM are global and referenceable from shadow DOM (but not vice versa). This enables Declarative Shadow DOM to import `@sheet` references from the light DOM: + +```html + + +``` +or imported from JavaScript: +```html + +``` + +## Detailed design discussion + +#### Named Imports with Imperative Shadow DOM + +`sheet.JavaScript` can also be imported via JavaScript as follows: + +```JavaScript +import baz, { bar } from 'sheet.css' with { type: 'css' } +``` + +`baz` will reference style rules outside of any `@sheet` blocks as a Default Import (in this case, the `div { color: blue; } ` rule). + +`bar` will reference style rules within the `@sheet bar` block as a Named Import (in this case, the `div { color: red; } ` rule). + +Named imports may be renamed as part of this import process: + +```JavaScript +import baz, { bar as renamed } from 'sheet.css' with { type: 'css' } +``` + +`bar` will be renamed to `renamed`. + +The default import may be omitted, importing only the named `@sheet`: + +```JavaScript +import { bar } from 'sheet.css' with { type: 'css' } +``` + +Any of these `import` examples can then be used to set the `adoptedStyleSheets` attribute on a Shadow DOM node: + +```JavaScript +import { bar } from 'sheet.css' with { type: 'css' } +document.adoptedStyleSheets = [bar]; +shadowRoot.adoptedStyleSheets = [bar]; +``` + +#### Performance + +`@sheet` has several performance benefits. The primary benefit is reduced network requests, as it allows mutliple stylesheets to be downloaded in one network request. + +`@sheet` can also reduce the work done by the style engine. Currently, Declarative Shadow DOM requires either a) that style rules are duplicated within shadow roots, or b) entire stylesheets are shared via `` tags. `@sheet` allows for granular control over style rules shared between the parent document and shadow roots without additional network requests or duplicated styles. + +Using `@sheet` may also yield some benefits to file compression. With a dictionary-based compression scheme, if two stylesheets contain many similar tokens (e.g. CSS rules and selectors), combining them via `@sheet` and then compressing may yield a higher compression ratio than compressing them as separate files. + +```JavaScript +// The following two imports should only make a single network request. +import { foo } from 'sheet.css' with { type: 'css' }; +import { bar } from 'sheet.css' with { type: 'css' } +``` + +```html + +``` + +```html + + + +``` + +#### Interaction with CSSOM + + +Named `@sheet` references augment the [existing](https://drafts.csswg.org/cssom/#stylesheet) `StyleSheet` interface with an optional `name` attribute reflecting the `@sheet` identifier: + +``` +[Exposed=Window] +interface StyleSheet { + readonly attribute DOMString? name; +}; +``` + +This also expands the [existing](https://drafts.csswg.org/cssom/#cssstylesheet) CSSOM `CSSStyleSheet` definition with a `StyleSheetList` of nested `CSSStyleSheet` objects to access nested `@sheet` references: + +``` +[Exposed=Window] +interface CSSStyleSheet : StyleSheet { + [SameObject] readonly attribute StyleSheetList nestedStyleSheets; +}; +``` + +## Considered alternatives + +1. [Declarative CSS Modules](https://github.com/MicrosoftEdge/MSEdgeExplainers/blob/main/ShadowDOM/explainer.md) are another mechanism for sharing styles between Declarative Shadow DOM and light DOM without the use of JavaScript. +2. Some additional alternatives to parts of the problems discussed here are discussed in the [Alternate proposals](https://github.com/MicrosoftEdge/MSEdgeExplainers/blob/main/ShadowDOM/explainer.md#alternate-proposals) section of that explainer. + +## Open Issues + +1. Whether rules are applied automatically for `@sheet` definitions, or whether they need to be imported to apply. The CSS Working Group did not have a consensus. +2. Fragment-only identifiers (without a URL) should allow inline `@sheet` references on the same document to be included globally (even within shadow roots). This wasn't brought up in the CSSWG discussions at all, but is important for DSD without requiring an external file (to avoid FOUC). +3. Behavior of `@import` - should `@import` be possible within `@sheet` at all, should it be allowed if it's the first/only statement, or should it be blocked? There was discussion of this in the CSSWG, but no conclusion was reached. This was briefly discussed in this CSSWG conversation: https://lists.w3.org/Archives/Public/www-style/2023Apr/0004.html +4. What happens with multiple `@sheet` definitions with the same identifier? First-definition wins, or do they get merged like `@layer`? Again, this was brought up in the CSSWG but not resolved (https://github.com/w3c/csswg-drafts/issues/5629#issuecomment-1498299448). Note that it's possible to have a "Flash of other-styled content" if it's last-defintion-wins, as the first definition may apply, then a later definition from an external CSS file may override it. +5. Do we want to be able to access sheets declared in shadow DOM from light DOM? For example: +```html + + + +I'm in the light DOM +``` +6. The name `nestedStyleSheets` is up for discussion. +7. Should we add `name` to the `StyleSheet` interface or overload the existing `title` attribute instead? +8. If a stylesheet contains named `@sheet` references *and* rules outside of the `@sheet` references, what happens in all cases when a fragment identifier is *not* specified? For example: + +sheet.css: + +```css +@sheet foo { + div{ + color: red; + } +} +div { + color: blue; +} +``` + +```html + + +``` + +## References & acknowledgements +Many thanks for valuable feedback and advice from: + +- Alison Maher +- Daniel Clark +- Justin Fagnani +- Tab Atkins Jr. +- Tien Mai +- Westbrook Johnson diff --git a/README.md b/README.md index 71081dac..76122669 100644 --- a/README.md +++ b/README.md @@ -82,7 +82,7 @@ we move them into the [Alumni section](#alumni-) below. | [Handwriting attribute](Handwriting/explainer.md) | ![GitHub issues by-label](https://img.shields.io/github/issues/MicrosoftEdge/MSEdgeExplainers/Handwriting?label=issues) | [New issue...](https://github.com/MicrosoftEdge/MSEdgeExplainers/issues/new?assignees=adettenb&labels=Handwriting&template=Handwriting.md&title=%5BHandwriting%5D+Issue) | HTML | | [AudioContext Interrupted State](https://github.com/MicrosoftEdge/MSEdgeExplainers/blob/main/AudioContextInterruptedState/explainer.md) | ![GitHub issues by-label](https://img.shields.io/github/issues/MicrosoftEdge/MSEdgeExplainers/AudioContext%20Interrupted%20State?label=issues) | [New Issue...](https://github.com/MicrosoftEdge/MSEdgeExplainers/issues/new?assignees=gabrielbrito&labels=AudioContext+Interrupted+State&title=%5BAudioContext+Interrupted+State%5D+%3CTITLE+HERE%3E) | WebAudio | | [IndexedDB getAllRecords()](https://github.com/MicrosoftEdge/MSEdgeExplainers/blob/main/IndexedDbGetAllEntries/explainer.md) | ![GitHub issues by-label](https://img.shields.io/github/issues/MicrosoftEdge/MSEdgeExplainers/IndexedDB%20GetAllRecords?label=issues) | [New Issue...](https://github.com/MicrosoftEdge/MSEdgeExplainers/issues/new?assignees=SteveBeckerMSFT&labels=IndexedDB%20GetAllRecords&title=%5BIndexedDB+getAllRecords()%5D+%3CTITLE+HERE%3E) | IndexedDB | - +| [Mulitple Stylesheets Per File (@sheet)](https://github.com/MicrosoftEdge/MSEdgeExplainers/blob/main/AtSheet/explainer.md) | ![GitHub issues by-label](https://img.shields.io/github/issues/MicrosoftEdge/MSEdgeExplainers/AtSheet?label=issues) | [New Issue...](https://github.com/MicrosoftEdge/MSEdgeExplainers/issues/new?assignees=aluhrs13&labels=AtSheet&title=%5B%40sheet%5D+%3CTITLE+HERE%3E) | CSS | # Alumni 🎓 diff --git a/ShadowDOM/explainer.md b/ShadowDOM/explainer.md index 13dd8b7c..1049af55 100644 --- a/ShadowDOM/explainer.md +++ b/ShadowDOM/explainer.md @@ -46,6 +46,8 @@ content location of future work and discussions. ## Background With the use of web components in web development, web authors often encounter challenges in managing styles, such as distributing global styles into shadow roots and sharing styles across different shadow roots. Markup-based shadow DOM, or [Declarative shadow DOM (DSD)](https://developer.chrome.com/docs/css-ui/declarative-shadow-dom), is a new concept that makes it easier and more efficient to create a shadow DOM definition directly in HTML, without needing JavaScript for setup. [Shadow DOM](https://developer.mozilla.org/en-US/docs/Web/API/Web_components/Using_shadow_DOM) provides isolation for CSS, JavaScript, and HTML. Each shadow root has its own separate scope, which means styles defined inside one shadow root do not affect another or the main document. +We're currently investigating this and [@sheet](/AtSheet/explainer.md) in parallel, and anticipate that we'll be prioritizing only one of these two in the immediate future. + ## Problem Sites that make use of Declarative Shadow DOM (DSD) have reported that the lack of a way to reference repeated stylesheets creates large payloads that add large amounts of latency. Authors have repeatedly asked for a way to reference stylesheets from other DSD instances in the same way that frameworks leverage internal data structures to share constructable style sheets via `adoptedStyleSheets`. This Explainer explores several potential solutions.