Estilos reutilizables sin problemas
Las hojas de estilo construibles son una forma de crear y distribuir diseños reutilizables cuando se usa el DOM Shadow.
Siempre se ha podido crear hojas de estilo con JavaScript. Sin embargo, históricamente, el proceso consistía en crear un elemento <style>
con document.createElement('style')
y, luego, acceder a su propiedad de hoja para obtener una referencia a la instancia CSSStyleSheet subyacente. Este método puede generar código CSS duplicado y su sobredimensionamiento, y el acto de adjuntar genera un destello de contenido sin estilo, independientemente de que esté sobrecargado o no. La interfaz CSSStyleSheet
es la raíz de una colección de interfaces de representación de CSS denominadas CSSOM, que ofrece una forma programática de manipular las hojas de estilo y eliminar los problemas asociados con el método anterior.
Las hojas de estilo constructibles permiten definir y preparar los diseños CSS compartidos y, luego, aplicar esos estilos a varias shadow roots o al documento con facilidad y sin duplicación. Las actualizaciones de una CSSStyleSheet compartida se aplican a todas las raíces en las que se adoptó, y la adopción de una hoja de estilo es rápido y síncrono una vez que la hoja se carga.
La asociación configurada por las hojas de estilo constructibles se presta bien a una serie de aplicaciones diferentes. Se puede usar para proporcionar un tema centralizado utilizado por muchos componentes: el tema puede ser una instancia CSSStyleSheet
que se pasa a los componentes, con actualizaciones del tema que se propagan automáticamente a los componentes. Se puede usar para distribuir valores de propiedades
personalizadas de CSS en subárboles del DOM específicos sin depender de la
cascada. Incluso se puede usar como una interfaz directa con el analizador de CSS del navegador, lo que facilita la precarga de hojas de estilo sin insertarlas en el DOM.
Construcción de una hoja de estilo
En lugar de introducir una nueva API para lograrlo, la especificación ConstructableStyleSheets permite crear hojas de estilo de manera imperativa al invocar el constructor CSSStyleSheet()
. El objeto CSSStyleSheet resultante tiene dos métodos nuevos que hacen que sea más seguro agregar y actualizar reglas de hojas de estilo sin activar Flash de contenido sin estilo (FOUC).
Los métodos replace()
y replaceSync()
reemplazan la hoja de estilo con una cadena de CSS, y replace()
muestra una promesa. En ambos casos, no se admiten referencias a hojas de estilo externas; se ignorará cualquier regla @import
y se producirá una advertencia.
const sheet = new CSSStyleSheet();
// replace all styles synchronously:
sheet.replaceSync('a { color: red; }');
// replace all styles:
sheet.replace('a { color: blue; }')
.then(() => {
console.log('Styles replaced');
})
.catch(err => {
console.error('Failed to replace styles:', err);
});
// Any @import rules are ignored.
// Both of these still apply the a{} style:
sheet.replaceSync('@import url("styles.css"); a { color: red; }');
sheet.replace('@import url("styles.css"); a { color: red; }');
// Console warning: "@import rules are not allowed here..."
Cómo usar hojas de estilo construidas
La segunda función nueva que presenta Constructable StyleSheets es una propiedad adoptedStyleSheets, que está disponible en Shadow Roots y Documentos. Esto nos permite aplicar de manera explícita los estilos definidos por un CSSStyleSheet
a un subárbol del DOM determinado. Para ello, configuramos la propiedad en un array de una o más hojas de estilo para aplicar a ese elemento.
// Create our shared stylesheet:
const sheet = new CSSStyleSheet();
sheet.replaceSync('a { color: red; }');
// Apply the stylesheet to a document:
document.adoptedStyleSheets.push(sheet);
// Apply the stylesheet to a Shadow Root:
const node = document.createElement('div');
const shadow = node.attachShadow({ mode: 'open' });
shadow.adoptedStyleSheets.push(sheet);
Putting it all together
With Constructable StyleSheets, web developers now have an explicit solution for
creating CSS StyleSheets and applying them to DOM trees. We have a new
Promise-based API for loading StyleSheets from a string of CSS source that uses
the browser's built-in parser and loading semantics. Finally, we have a
mechanism for applying stylesheet updates to all usages of a StyleSheet,
simplifying things like theme changes and color preferences.
Looking ahead
The initial version of Constructable Stylesheets shipped with the API
described here, but there's work underway to make things easier to use. There's
a proposal to extend
the adoptedStyleSheets
FrozenArray with dedicated methods for inserting and
removing stylesheets, which would obviate the need for array cloning and avoid
potential duplicate stylesheet references.
More information