Vue中Scope是怎么做样式隔离的?

开发 前端
在 Vue 2.x 中,使用 ::v-deep​ 或 /deep/​ 可以穿透 scoped​ 样式的限制的原因与其实现方式有关。这两个选择器的作用是告诉样式引擎在处理选择器时要忽略 scoped 样式中的作用域,从而可以选择到子组件中的元素。

scope样式隔离

在 Vue 中,样式隔离是通过 scoped 特性实现的。当在一个组件的 <style> 标签上添加 scoped 特性时,Vue 会自动为这个样式块中的所有选择器添加一个唯一的属性,以确保这些样式只对当前组件生效,从而达到样式隔离的效果。

以下是 scoped 样式隔离的基本原理:

  1. 「唯一属性添加:」 在编译过程中,Vue 会将样式块中的每个选择器都转换为一个带有唯一属性的选择器。例如,如果有一个类似 .my-class 的选择器,可能会被转换成类似 .my-class[data-v-xxxxxxx] 的形式,其中 data-v-xxxxxxx 是一个唯一的属性。
  2. 「属性注入到组件元素:」 在组件渲染时,Vue 会自动将这个唯一的属性注入到组件的根元素上。这样,样式块中的选择器就会匹配到具有相同唯一属性的元素,从而确保样式只会应用到当前组件。

下面是一个简单的示例:

<template>
  <div class="my-component">
    <p class="my-class">This is a paragraph.</p>
  </div>
</template>


<style scoped>
.my-class {
  color: red;
}
</style>

在这个例子中,<style scoped> 中的样式会被转换,使得 .my-class 成为 .my-class[data-v-xxxxxxx]。然后,这个唯一属性 data-v-xxxxxxx 会被注入到组件根元素上,使得样式只会应用到当前组件内的元素,而不会影响到其他组件。

这种样式隔离的机制保证了组件之间的样式不会相互干扰,是 Vue 组件化开发中的一个重要特性。

如何打破scope的限制?

在 Vue 中,scoped 样式的设计初衷是为了实现组件样式的隔离,以防止样式污染和冲突。然而,有时候可能需要打破 scoped 样式的限制。以下是一些常见的方法:

1. 使用 ::v-deep 或 /deep/:

在 scoped 样式中,可以使用 ::v-deep 或 /deep/ 伪类选择器来打破样式的封装,以便选择子组件中的元素。需要注意的是,这两个选择器在 Vue 2.x 中是等效的,但在 Vue 3.x 中 ::v-deep 被废弃,取而代之的是 /deep/。

<style scoped>
::v-deep .my-class {
  color: blue;
}


/* 或者在 Vue 3.x 中可以使用 /deep/ */
/deep/ .my-class {
  color: blue;
}
</style>

这样可以让样式穿透到子组件中。

2. 使用全局样式:

如果确实需要在多个组件之间共享样式,可以考虑将样式定义为全局样式而不使用 scoped 特性。

<style>
.my-class {
  color: red;
}
</style>

3. 使用深度选择器:

在 Vue 3.x 中,可以使用 ::v-slotted 深度选择器来选择插槽中的元素。这在处理插槽样式时非常有用。

<style scoped>
::v-slotted(.my-class) {
  color: green;
}
</style>

在使用这些方法时,需要谨慎考虑样式的全局性和可维护性,以确保样式的修改不会产生意外的副作用。打破 scoped 样式的限制可能会导致不易维护的代码,因此建议仅在确实需要的情况下使用。

deep

在 Vue 2.x 中,使用 ::v-deep 或 /deep/ 可以穿透 scoped 样式的限制的原因与其实现方式有关。这两个选择器的作用是告诉样式引擎在处理选择器时要忽略 scoped 样式中的作用域,从而可以选择到子组件中的元素。

实际上,::v-deep 和 /deep/ 是 Vue 对样式处理引擎进行了处理,使其能够正确解析这些选择器,然后将样式应用到相应的 DOM 元素上。

在 Vue 3.x 中,::v-deep 被废弃,取而代之的是 /deep/ 或 ::v-slotted。这种变化是为了更好地与 Web 标准兼容,因为 ::v-deep 不是 CSS 规范的一部分。

但是,穿透 scoped 样式的做法实际上打破了组件的封装性,可能导致样式的不可预测性和不易维护性。在使用这种方式时需要权衡利弊,并确保清晰地了解可能产生的影响。最好的做法是遵循组件的封装性原则,将样式限制在组件内部,以避免全局样式的冲突和混乱。

责任编辑:武晓燕 来源: 海燕技术栈
相关推荐

2012-05-24 14:58:55

开源代码

2011-03-11 09:53:46

FacebookMySQL

2017-07-20 13:11:46

Code ReviewPR评审

2022-10-19 14:16:18

样式隔离前缀css

2016-01-05 16:17:59

云梦数据仓

2018-01-08 14:18:14

代码互联网持续集成

2015-07-22 11:35:26

2022-12-02 16:28:47

2022-03-10 11:25:51

InnoDB优化

2023-09-27 22:44:18

数据迁移数据库

2015-07-30 09:01:59

2021-05-13 07:32:17

培训代码同事

2019-09-10 09:25:27

分布式系统Elasticsear

2016-09-21 10:18:26

阿里Dubbo性能测试

2021-05-05 10:48:33

渗透测试漏洞网络攻击

2024-01-15 08:08:27

2022-08-15 12:31:32

Vue3TypeScript

2022-06-27 14:21:09

Spring语言

2020-07-28 08:36:54

数据安全数据泄露数据

2023-07-10 15:35:46

点赞
收藏

51CTO技术栈公众号