shadow dom 是什么?

顾名思义,shadow dom直译的话就是影子dom,但我更愿把它理解为DOM中的DOM。因为他能够为Web组件中的 DOM和 CSS提供了封装,实际上是在浏览器渲染文档的时候会给指定的DOM结构插入编写好的DOM元素,但是插入的Shadow DOM 会与主文档的DOM保持分离,也就是说Shadow DOM不存在于主DOM树上。

并且Shadow DOM封装出来的DOM元素是独立的,外部的配置不会影响到内部,内部的配置也不会影响外部。

如果这篇文章有帮助到你,️关注+点赞️鼓励一下作者,文章公众号首发,关注 前端南玖 第一时间获取最新文章~

思考

理解完它的概念,我们再来思考一个问题:

为什么我们用的一些标签明明就是一个空元素,但他却能够渲染出各种复杂的场景?

  • input
  • video
  • audio
  • textarea
  • 等...

可能很多同学都没想过为什么这些标签跟我们常用的div标签不一样,它们就简单写个标签就能渲染出对应的样式与功能;

或者有些同学理解成这都是底层渲染的事,我们不必关心。

是的,这些标签内部的内容确实都是底层渲染的,不过我们也不是看不到它们内部的实现原理。

查看html原生标签的Shadow DOM

在html中写入以下标签,然后到浏览器控制台去查看

<input type="text">
<input type="range">
<video src="http://clips.vorwaerts-gmbh.de/big_buck_bunny.mp4" controls></video>
<textarea></textarea>

很多人看到的是这样的,但这和我们写的没有任何区别呀?别急,这就带你看看他们的真实面目~

首先打开浏览器控制台的设置选项

然后再找到Preference -> Elements,把show user anent shadow dom勾上

这时候我们再来看一下此时的dom元素发生了什么变化

我们会发现这些标签内部都大有乾坤,在这些标签下面都多了一个shadow root,在它里面才是这些标签的真实布局。

既然这些标签内部都有一些子元素布局,那么我们能不能通过JavaScript来访问到它们呢?

const input = document.querySelector('input')
console.log(input.firstChild) // null

很明显,这是不可以的!

因为它为web开发者设定了一个边界,界定了哪些是你可以访问的,哪些实现细节是访问不到的。然而,浏览器本身却可以随意跨越这个边界。设置这样一个边界之后,它们就可以在你看不见的地方使用熟悉的web技术、同样的HTML元素去创建更多的功能,而不是像你一样要在页面上用div和span来堆。

shadow dom 结构

Shadow DOM 允许将隐藏的 DOM 树附加到常规的 DOM 树中——它以 shadow root 节点为起始根节点,在这个根节点的下方,可以是任意元素,和普通的 DOM 元素一样。

就是因为这个特点所以我们才能看到上面那些单个空标签就能够渲染出各种各样的复杂场景。

上面这张图非常直观的表现了shadow dom的结构以及它与真实dom的关系。

shadow host

一个常规 DOM 节点,Shadow DOM 会被附加到这个节点上。

shadow bounday

Shadow DOM 结束的地方,也是常规 DOM 开始的地方。

shadow tree

Shadow DOM 内部的 DOM 树。

shadow root

Shadow tree 的根节点。

如何使用shadow dom?

创建一个shadow dom

我们可以使用attachShadow给指定元素挂载一个shadow dom,并且返回对shadow root的引用。

const shadowroot = root.attachShadow({mode: 'open'})
const template = `
<div>前端南玖</div>
`
shadowroot.innerHTML = template

shadow dom mode

当调用Element.attachShadow()方法t时,必须通过传递一个对象作为参数来指定shadow DOM树的封装模式,否则将会抛出一个TypeError。该对象必须具有mode属性,值为 openclosed

  • open shadow root 元素可以从 js 外部访问根节点,例如使用 Element.shadowRoot:
element.shadowRoot; // 返回一个 ShadowRoot 对象
  • closed 拒绝从 js 外部访问关闭的 shadow root 节点
element.shadowRoot; // 返回 null

浏览器通常用关闭的 shadow roo 来使某些元素的实现内部不可访问,而且不可从JavaScript更改。

对于一些不希望公开shadow root 的Web组件来说,封闭的shadow DOM看起来非常方便,然而在实践中绕过封闭的shadow DOM并不难。但是完全隐藏shadow DOM所需的工作量也大大超过了它的价值。

哪些元素可以挂载shadow dom?

这里需要注意的是并非所有html元素都可以挂载shadow dom,只有以下这些元素可以充当shadow dom的 shadow host

article aside blockquote body
div footer h1 h2
h3 h4 h5 h6
header main nav p
section span 任何带有有效的名称且可独立存在的自定义元素

当我们尝试在其它元素挂在shadow dom时,浏览器则会抛出异常。

const input = document.querySelector('input')
const inputRoot = input.attachShadow({mode: 'open'})

shadow dom的特点

从前面的介绍,我们知道shadow dom是游离在 DOM 树之外的节点树,但是它是基于普通 DOM 元素(非 document)创建的,并且创建后的 Shadow-dom 节点可以从界面上直观的看到。最重要的一点是Shadow-dom 具有良好的密封性。

样式

<style>
.wx_name {
color:aqua;
}
</style>
<body>
<div class="wx_name">我是真实dom</div>
<div id="root"></div> <script>
const shadowroot = root.attachShadow({mode: 'open'})
const template = `
<div class="wx_name">shadow dom - 前端南玖</div>
`
shadowroot.innerHTML = template
</script>
</body>

它渲染出来是下面这样的:

上面我们说了shadow dom是游离在 DOM 树之外的节点树,所以我们文档上的CSS就不会作用在他身上。

样式化host元素

host伪类选择器允许你从shadow root中的任何地方访问shadow host

const shadowroot = root.attachShadow({mode: 'open'})
const template = `
<div class="wx_name">shadow dom - 前端南玖</div>
<style>
:host {
border: 1px solid #ccc;
color: pink;
}
</style>
`
shadowroot.innerHTML = template

需要注意的是:host仅在shadow root中有效,并且在shadow root之外定义的样式规则比:host中定义的规则具有更高的特殊性。

样式钩子

shadow dom还有一个非常重要的一个特点就是可以使用CSS自定义属性来创建样式占位符,并允许用户填充。

<style>
#root {
--bg: coral;
--color: #fff:
}
</style>
<div id="root"></div> <script> const shadowroot = root.attachShadow({mode: 'open'})
const template = `
<div class="wx_name">shadow dom - 前端南玖</div>
<style>
.wx_name {
background: var(--bg, red);
color: var(--color, #000)
}
</style>
`
shadowroot.innerHTML = template
</script>

通过CSS访问shadow

如果我们想要自定义一些原生标签的样式应该怎样做呢,很显然常规的CSS选择器并不能获取到shadow dom内部元素。那我们就一点办法没有了吗?其实这里我们可以通过一些伪元素来实现,比如:

<input type="range">

它默认长这样

那我们怎么去改变他的样式呢,比如给它换种背景色

直接给input写背景色能实现吗?

input{
background: #ccc;
}

很显然这是一种大聪明行为,那它就这一个元素,究竟怎样才能改变它的背景色呢,上面我们不是说了吗,它内部是有shadow dom的

input[type=range]::-webkit-slider-runnable-track {
-webkit-appearance: none;
background-color: chocolate;
}

我们可以通过伪元素来访问到shadow的内部元素并改变其样式。

事件

在shadow DOM内触发的事件可以穿过shadow边界并冒泡到light DOM;但是Event.target的值会自动更改,因此它看起来好像该事件源自其包含的shadow树而不是实际元素的host元素。

此更改称为事件重定向,其背后的原因是保留shadow DOM封装。

<div id="root"></div>

<script>
const shadowroot = root.attachShadow({mode: 'open'})
const template = `
<div class="wx_name">shadow dom - aaa</div>
<div class="wx_name">shadow dom - bbb</div>
<div class="wx_name">shadow dom - ccc</div>
`
shadowroot.innerHTML = template
document.addEventListener('click', e => {
console.log(e.target)
})
</script>

当点击shadow dom中的任何元素时,打印出来的都是root,监听器无法看到调度该事件的真实元素。

自定义元素托管shadow DOM

模拟微信小程序标签

Custom Elements API 创建的自定义元素可以像其他元素一样托管shadow DOM。

<body>
<wx-text>前端南玖</wx-text>
<script>
class wxText extends HTMLElement {
constructor() {
super()
// console.log(this.innerText)
const text = this.innerText
this.innerText = null
const shadowRoot = this.attachShadow({mode: 'open'})
shadowRoot.innerHTML = `
<span>${text}</span>
`
}
} customElements.define('wx-text', wxText)
</script>
</body>

上面这段代码就是模拟微信小程序的标签实现,这里为什么又跳到了小程序?因为微信小程序的实现原理跟这类似,我们知道小程序的图是在WebView里渲染的,那搭建视图的方式自然就需要用到HTML语言。然后为了管控安全,肯定不可能让开发者直接使用html来进行开发,所以就自己实现了一套组件组织框架Exparser内置在小程序基础库中。这里的Exparser框架模型上与WebComponents的ShadowDOM高度相似,但不依赖浏览器的原生支持,也没有其他依赖库。

如何查看小程序编译后的标签

很多人可能会有疑惑,我们在小程序中写的标签不是这样的呀,它们都不带有wx前缀。是的,为了开发者使用方便,开发时是不需要带wx前缀的,在编译过程会自动识别比对转换成真实DOM。

  • 我们可以打开微信开发者工具,调试微信开发者工具,然后在控制台输入document.getElementsByTagName('webview')[0].showDevTools(true,null)
  • 然后它会打开另外一个控制台,在这里就能看到类似我们上面实现的内容了

推荐阅读

原文首发地址点这里,欢迎大家关注公众号 「前端南玖」,如果你想进前端交流群一起学习,请点这里

我是南玖,我们下期见!!!

究竟什么是Shadow DOM?的更多相关文章

  1. 【shadow dom入UI】web components思想如何应用于实际项目

    回顾 经过昨天的优化处理([前端优化之拆分CSS]前端三剑客的分分合合),我们在UI一块做了几个关键动作: ① CSS入UI ② CSS作为组件的一个节点而存在,并且会被“格式化”,即选择器带id前缀 ...

  2. 封印术:shadow dom

    置顶文章:<纯CSS打造银色MacBook Air(完整版)> 上一篇:<鼠标滚动插件smoovejs和wowjs> 作者主页:myvin 博主QQ:851399101(点击Q ...

  3. 使用shadow dom封装web组件

    什么是shadow dom? 首先我们先来看看它长什么样子.在HTML5中,我们只用写如下简单的两行代码,就可以通过 <video> 标签来创建一个浏览器自带的视频播放器控件. <v ...

  4. shadow dom

    初识shadow dom 我们先看个input="range"的表现: what amazing ! 一个dom能表现出这么多样式嘛? 无论是初学者和老鸟都是不肯相信的,于是在好奇 ...

  5. shadow dom 隔离代码 封装

    Shadow DOM是指浏览器的一种能力,它允许在文档(document)渲染时插入一棵DOM元素子树,但是这棵子树不在主DOM树中.   Shadow DOM 解决了 DOM 树的封装问题.     ...

  6. 纯CSS菜单样式,及其Shadow DOM,Json接口 实现

    先声明,要看懂这篇博客要求你具备少量基础CSS知识, 当然如果你只是要用的话就随便了,不用了解任何知识 完整项目github链接:https://github.com/git-Code-Shelf/M ...

  7. JavaScript 是如何工作:Shadow DOM 的内部结构 + 如何编写独立的组件!

    这是专门探索 JavaScript 及其所构建的组件的系列文章的第 17 篇. 如果你错过了前面的章节,可以在这里找到它们: JavaScript 是如何工作的:引擎,运行时和调用堆栈的概述! Jav ...

  8. Shadow DOM及自定义标签

    参考链接:点我 一.什么是Shadow DOM Shadow DOM,直接翻译的话就是 影子 DOM,可以理解为潜藏在 DOM 结构中并且我们无法直接控制操纵的 DOM 结构.类似于下面这种结构 Sh ...

  9. 理解Shadow DOM(一)

    1. 什么是Shadow DOM? Shadow DOM 如果按照英文翻译的话可以理解为 影子DOM, 何为影子DOM呢?可以理解为一般情况下使用肉眼看不到的DOM结构,那如果一般情况下看不到的话,那 ...

随机推荐

  1. springboot2.7.x 集成log4j2配置写入日志到mysql自定义表格

    在阅读之前请先查看[springboot集成log4j2] 本文暂不考虑抽象等实现方式,只限于展示如何自定义配置log4j2并写入mysql数据库(自定义结构) 先看下log4j2的配置 <?x ...

  2. Windows-VS2017创建.NET项目

    首先新建->项目 选择如下, 注意要选择.NET Framework4.x 选择对应的项目类型 建议选上Web窗体(如果是用于实验的话) 完成后进行测试 如果出现 HTTP Error 403. ...

  3. 教你用VS code 生成vue-cli代码片段

    可以自定义设置名字:name.json { "Print to console": { "prefix": "vue", "bod ...

  4. oracle-安装与访问、卸载

    安装oracle 官网(http://oracle.com/ )下载oracle -->Oracle Database -->点击接受Accept --> 下载11g(Downloa ...

  5. docker容器内修改文件

    1.找到容器对应的ID 使用docker ps命令找到对应的镜像id 2.根据容器id进入到对应文件夹 执行命令:docker exec -it 镜像id /bin/bash 3.进入对应目录(以My ...

  6. 测试右移:线上质量监控 ELK 实战

    目录 [测试右移]介绍 ELK Stack 介绍 ELK 监控体系搭建 ES & Kibana 搭建 Nginx 日志自动采集 Nginx Agent 安装 Nginx 服务器 数据分析 Lo ...

  7. win10设置Python程序定时运行(设置计划任务)

    今天来设置一下定时执行Pycharm内的脚本: 这个要基于win10 的任务计划程序(设置 > 控制面板 > 系统和安全 > 管理工具 > 任务计划程序) 1. create ...

  8. go-zero微服务实战系列(十、分布式事务如何实现)

    在分布式应用场景中,分布式事务问题是不可回避的,在目前流行的微服务场景下更是如此.比如在我们的商城系统中,下单操作涉及创建订单和库存扣减操作两个操作,而订单服务和商品服务是两个独立的微服务,因为每个微 ...

  9. Tomcat深入浅出——Session与Cookie(四)

    一.Cookie 1.1 Cookie概念 Cookie:有时也用其复数形式 Cookies.类型为"小型文本文件",是某些网站为了辨别用户身份,进行Session跟踪而储存在用户 ...

  10. 本地使用 Docker Compose 与 Nestjs 快速构建基于 Dapr 的 Redis 发布/订阅分布式应用

    Dapr(分布式应用程序运行时)介绍 Dapr 是一个可移植的.事件驱动的运行时,它使任何开发人员能够轻松构建出弹性的.无状态和有状态的应用程序,并可运行在云平台或边缘计算中,它同时也支持多种编程语言 ...