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

<template>
    <div class="parent">
        <h1>前端名狮</h1>
    </div>
</template>
<script>
export default {};
</script>
<style lang="less">
.parent {
    color: red;
    h1 {
        font-size: 30px;
    }
}
</style>

加上scoped

<template>
    <div class="parent">
        <h1>前端名狮</h1>
    </div>
</template>
<script>
export default {};
</script>
<style lang="less" scoped>
.parent {
    color: red;
    h1 {
        font-size: 30px;
    }
}
</style>

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

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

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

父组件:

<template>
	<div class="parent">
		<h1>前端名狮</h1>
		<child>
		<div class="time">时间:2020年9月16日</div>
		</child>
	</div>
</template>
<script>
import Child from "./Child.vue";
export default {
  components: {
    Child,
  },
};
</script>
<style lang="less" scoped>
.parent {
	color: red;
	h1 {
		font-size: 30px;
	}
}
</style>

子组件:

<template>
    <div class="child">
        <div class="author">作者:诀九</div>
        <div class="introduce">介绍:定期推送前端技术相关文章,面试题详解</div>
        <slot />
    </div>
</template>
<script>
export default {};
</script>
<style lang="less">
.child {
    font-size: 20px;
    .author {
        font-weight: 600;
        color: red;
    }
    .introduce {
        color: blue;
    }
}
</style>

根据上图我们会发现:

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

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

deep作用

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

我们一般会这么写

<style lang="less" scoped>
.parent {
	color: red;
	h1 {
		font-size: 30px;
	}
	.child {
	    font-size: 20px;
	    .author {
	        color: orange;
	    }
	}
}
</style>

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

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

修改下父组件:

<style lang="less" scoped>
.parent {
	color: red;
	h1 {
		font-size: 30px;
	}
/deep/	.child {
	    font-size: 20px;
	    .author {
	        color: orange;
	    }
	}
}
</style>

image

总结

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

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

以上为个人经验,希望能给大家一个参考,也希望大家多多支持Devmax。

vue scoped与深度选择器deep的原理分析的更多相关文章

  1. Vue如何指定不编译的文件夹和favicon.ico

    这篇文章主要介绍了Vue如何指定不编译的文件夹和favicon.ico,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

  2. vue自定义加载指令v-loading占位图指令v-showimg

    这篇文章主要为大家介绍了vue自定义加载指令和v-loading占位图指令v-showimg的示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  3. vue使用动画实现滚动表格效果

    这篇文章主要为大家详细介绍了vue使用动画实现滚动表格效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  4. 关于Vue 监控数组的问题

    这篇文章主要介绍了Vue 监控数组的示例,主要包括Vue 是如何追踪数据发生变化,Vue 如何更新数组以及为什么有些数组的数据变更不能被 Vue 监测到,对vue监控数组知识是面试比较常见的问题,感兴趣的朋友一起看看吧

  5. Vue子组件props从父组件接收数据并存入data

    这篇文章主要介绍了Vue子组件props从父组件接收数据并存入data的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  6. Vue h函数的使用详解

    本文主要介绍了Vue h函数的使用详解,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  7. VUE响应式原理的实现详解

    这篇文章主要为大家详细介绍了VUE响应式原理的实现,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助

  8. vue+Element ui实现照片墙效果

    这篇文章主要为大家详细介绍了vue+Element ui实现照片墙效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  9. vue+elemet实现表格手动合并行列

    这篇文章主要为大家详细介绍了vue+elemet实现表格手动合并行列,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  10. iview+vue实现导入EXCEL预览功能

    这篇文章主要为大家详细介绍了iview+vue实现导入EXCEL预览功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

随机推荐

  1. js中‘!.’是什么意思

  2. Vue如何指定不编译的文件夹和favicon.ico

    这篇文章主要介绍了Vue如何指定不编译的文件夹和favicon.ico,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

  3. 基于JavaScript编写一个图片转PDF转换器

    本文为大家介绍了一个简单的 JavaScript 项目,可以将图片转换为 PDF 文件。你可以从本地选择任何一张图片,只需点击一下即可将其转换为 PDF 文件,感兴趣的可以动手尝试一下

  4. jquery点赞功能实现代码 点个赞吧!

    点赞功能很多地方都会出现,如何实现爱心点赞功能,这篇文章主要为大家详细介绍了jquery点赞功能实现代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  5. AngularJs上传前预览图片的实例代码

    使用AngularJs进行开发,在项目中,经常会遇到上传图片后,需在一旁预览图片内容,怎么实现这样的功能呢?今天小编给大家分享AugularJs上传前预览图片的实现代码,需要的朋友参考下吧

  6. JavaScript面向对象编程入门教程

    这篇文章主要介绍了JavaScript面向对象编程的相关概念,例如类、对象、属性、方法等面向对象的术语,并以实例讲解各种术语的使用,非常好的一篇面向对象入门教程,其它语言也可以参考哦

  7. jQuery中的通配符选择器使用总结

    通配符在控制input标签时相当好用,这里简单进行了jQuery中的通配符选择器使用总结,需要的朋友可以参考下

  8. javascript 动态调整图片尺寸实现代码

    在自己的网站上更新文章时一个比较常见的问题是:文章插图太宽,使整个网页都变形了。如果对每个插图都先进行缩放再插入的话,太麻烦了。

  9. jquery ajaxfileupload异步上传插件

    这篇文章主要为大家详细介绍了jquery ajaxfileupload异步上传插件,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  10. React学习之受控组件与数据共享实例分析

    这篇文章主要介绍了React学习之受控组件与数据共享,结合实例形式分析了React受控组件与组件间数据共享相关原理与使用技巧,需要的朋友可以参考下

返回
顶部