如何在Shadow DOM/Web组件中覆盖CSS

开发 前端
本文介绍了使用CSS自定义属性穿透Shadow DOM并使你的Web组件可自定义的基础知识。

Web组件(Web Components)的主要目的之一是提供封装——能够隐藏HTML标记结构和CSS样式,并与页面上的其他代码分离,这样不同的部分就不会冲突,通过这种方式,这样代码就可以保持漂亮和干净。

影子DOM(Shadow DOM)为我们提供了范围限定的样式封装,并提供了一种让我们随意选择进入(尽可能少)外界的方法。

但是,如果我想使我的组件可自定义某些样式属性,该怎么办?

本文介绍了使用CSS自定义属性穿透Shadow DOM并使你的Web组件可自定义的基础知识。

创建一个HTML元素

我们将使用扩展基本HTML Element的JavaScript类创建自定义HTML元素。然后,我们将使用要创建的标签名称和刚刚创建的类调用 customElements.define()。

  1. class AppCard extends HTMLElement {...} 
  2. window.customElements.define('app-card', AppCard); 

在此示例中,我们将创建此简单的Material Design卡片,当我们在HTML上添加此元素时将显示该元素:<app-card></ app-card>

如何在Shadow DOM/Web组件中覆盖CSS

首先,我们创建Shadow DOM root,然后将HTML和CSS字符串分配给Shadow DOM root 的 innerHTML,如下所示。

  1. class AppCard extends HTMLElement { 
  2.   constructor() { 
  3.     super(); 
  4.     const shadowRoot = this.attachShadow({mode: 'open'}); 
  5.         shadowRoot.innerHTML = ` 
  6.       <style> 
  7.         .card { 
  8.           background-color: #fff; 
  9.           ... 
  10.         } 
  11.       </style> 
  12.       <div class="card"> 
  13.         <div>Card title</div> 
  14.       </div> 
  15.     `; 
  16.   }}window.customElements.define('app-card', AppCard); 

覆盖尝试

在此示例中,我们要修改卡的背景颜色。如果它是HTML中的简单 div 元素,则可以覆盖 card 类或通过CSS选择器访问 div 元素。但是,以下尝试将无效:

  1. // access the div  
  2. app-card > div { 
  3.   background-color: #2196F3; 
  4. // override card class 
  5. app-card > .card { 
  6.   background-color: #2196F3; 

使用CSS自定义属性

为了解决这个问题,我们可以使用CSS自定义属性(CSS变量)。可以使用CSS中定义的CSS自定义属性来更改自定义元素中的某些CSS属性。

按照我们的例子,我们将使用属性 background-color 上的变量 card-bg 来获取谁在使用自定义元素所定义的颜色。

  1. class AppCard extends HTMLElement { 
  2.   constructor() { 
  3.     super(); 
  4.     const shadowRoot = this.attachShadow({mode: 'open'}); 
  5.         shadowRoot.innerHTML = ` 
  6.       <style> 
  7.         .card { 
  8.           background-color: var(--card-bg, #fff); 
  9.           ... 
  10.         } 
  11.       </style> 
  12.       <div class="card"> 
  13.         <div>Card title</div> 
  14.       </div> 
  15.     `; 
  16.   }}window.customElements.define('app-card', AppCard); 

现在,我们将使用 app-card 自定义元素,并在Body元素的CSS中创建 card-bg 变量,我们将十六进制颜色 #2196F3 分配给变量。

  1. <html> 
  2.   <head> 
  3.     <style> 
  4.       body { 
  5.         --card-bg: #2196F3; 
  6.       } 
  7.     </style> 
  8.   </head> 
  9.   <body> 
  10.     <app-card></app-card> 
  11.   </body> 
  12. </html> 
如何在Shadow DOM/Web组件中覆盖CSS

总结

使用这种策略,我们可以在你的文档中拥有一个封装的CSS元素,同时我们可以允许使用CSS对一些属性进行自定义。你可以在这里访问一个完整的例子。

 

 

责任编辑:赵宁宁 来源: 今日头条
相关推荐

2014-05-26 16:29:12

Shadow DomWeb Compone

2022-02-10 22:24:05

DOM结构工具

2014-05-26 16:16:59

Shadow DomWeb Compone

2021-04-09 18:01:03

前端ReactDOM

2021-02-26 15:10:00

前端React组件交互

2020-07-14 08:31:42

CSS网格布局项目

2022-11-08 11:39:34

Web浏览器深色模式

2009-04-15 10:08:00

Wi-Fi无线网络优化

2022-05-13 21:20:23

组件库样式选择器

2016-01-25 12:25:58

UbuntuFedoraGitLab

2021-07-07 08:01:48

CSS DOM解析

2009-01-30 09:12:11

SpringApplicationJavaWeb

2010-09-08 16:50:11

JavaScriptDOM操作

2011-08-19 15:53:01

Windows7Windows日历组件

2011-12-29 15:35:39

Web

2022-06-09 13:52:35

Vue协作开发项目

2021-03-28 07:47:01

JavaWeb项目windows

2021-01-11 07:51:16

DOM对象节点树

2022-09-20 08:43:37

Go编程语言Web

2021-01-18 07:15:22

虚拟DOM真实DOMJavaScript
点赞
收藏

51CTO技术栈公众号