Vue scoped与深度选择器deep的原理

JS引入模块化概念后,变得更易于开发维护,但是css样式由于其特殊性,一直没有实现模块化,scoped的出现就是为了实现样式模块化,其本质利用属性选择器实现的一种伪模块化,并非真正意义上的模块化,但这已经让css模块化前进了一大步,要知道JS的模块化也是以这种方式开始的,比如早期的seajs,requirejs都是利用闭包封装达到模块化的效果,后来慢慢的出现了ES6的模块化规范import/export,说不定未来的某一天css也会出现真正的模块化,我们就不用学习这种知识点了。

说回正题,该篇文章主要从以下几个问题,进行探索scoped和deep相关的实现原理:

  1. scoped生成的dom和style有什么特点;
  2. 父组件引入子组件,生成的dom和style有什么区别;
  3. 父组件传入的子组件slot,生成的dom和style是什么样的;
  4. 父组件如何修改子组件的样式;

scoped的作用

scoped主要用于vue中style部分,加上scoped后,最终生成的dom和style都会被加上一个唯一的动态属性,这样样式只会对当前组件有效,不会污染全局样式。如下图所示:

未加scoped






Vue scoped与深度选择器deep的原理_第1张图片

加上scoped






Vue scoped与深度选择器deep的原理_第2张图片

通过上面两种情况对比,我们发现:

  1. 加上scoped后,dom会被添加上一个唯一的属性值,生成的style样式也会使用该属性作为属性选择器设置样式,这样使得样式只对该组件有效,避免了全局样式污染;
  2. 每个组件内的dom标签都会被设置上同一个data属性值;

对于上面两条规律,父组件内引入子组件的情况下,是否一样呢?

父组件:






子组件:






Vue scoped与深度选择器deep的原理_第3张图片

根据上图我们会发现:

  1. 父组件内出现的dom标签都会被添加上同一个属性值,包括给子组件传入的slot内容;
  2. 父组件的属性值只会设置在子组件的最外层,并不会设置到子组件的内部dom元素上;

常用的组件库,比如element、vux、vant提供的组件是不带scoped的。但是我们写的子组件大部分都是带有scoped的,看下生成的dom和style如下图:

Vue scoped与深度选择器deep的原理_第4张图片

deep作用

看到这里,我们清楚了scoped对于生成的dom和style的影响,以及避免全局污染的原理。但是在使用scoped的父组件中如何修改子组件样式呢?我们一般会这么写


Vue scoped与深度选择器deep的原理_第5张图片

但是最终生成的样式是以父组件的属性值作为选择器的,这样父组件就只能修改子组件最外层的div样式,但是修改子组件内层元素的样式是不可行的。

想要修改子组件的内层元素样式,就需要使用/deep/了,/deep/是less中深度选择器的>的另一种写法,只是因为>在vue模板中不能正常解析,所以用/deep/代替。修改下父组件:


Vue scoped与深度选择器deep的原理_第6张图片

总结

这个知识点我们经常使用,但是很多人却不知道为什么,实际看下生成的style就知道了。

  1. scoped本质上是给dom增加一个唯一属性,然后利用这个属性作为属性选择器设置样式达到模块化的目的(这里的属性值是vue-template-complier编译时动态添加的,具体看下源码);
  2. /deep/为了解决scoped引发的父组件修改子组件内部样式问题出现的;

扫一扫 关注我的公众号【前端名狮】,更多精彩内容陪伴你!

Vue scoped与深度选择器deep的原理_第7张图片

你可能感兴趣的:(Web前端,vue.js,css)