Vue的作用域样式scoped CSS和深度作用选择器>>>、/deep/、::v-deep的原理

不同于 React 和 Angular,Vue 使用的 .vue 文件是一个自定义的文件类型,用类 HTML 语法描述一个 Vue 组件。每个 .vue 文件包含三种类型的顶级语言块 <template><script><style>,分别用于包含组件的 Html模板、JS实例定义 和 组件的样式;这使得 Vue 能轻松并正确无误地确定组件对应的样式规则;所以在样式隔离上,Vue 可以使用不同于 CSS Modules规范 的方案。本文主要讲解 Vue 的 Scoped CSS 和 深度作用选择器 的原理,以及其它的基于 Vue 的这种集合 Html模板、JS实例定义 和 组件的样式 的单文件组件场景下 样式隔离的实现方案。

目录

内容

1. Scoped CSS的原理

原理

Vue的作用域样式 Scoped CSS 的实现思路如下:

  1. 为每个组件实例(注意:是组件的实例,不是组件类)生成一个能唯一标识组件实例的标识符,我称它为组件实例标识,简称实例标识,记作 InstanceID;
  2. 给组件模板中的每一个标签对应的Dom元素(组件标签对应的Dom元素是该组件的根元素)添加一个标签属性,格式为 data-v-实例标识,示例:<div data-v-e0f690c0="" >
  3. 给组件的作用域样式 <style scoped> 的每一个选择器的最后一个选择器单元增加一个属性选择器 原选择器[data-v-实例标识] ,示例:假设原选择器为 .cls #id > div,则更改后的选择器为 .cls #id > div[data-v-e0f690c0]

示例

父组件:Parent.vue

<template>
  <div class="parent" id="app">
    <h1>我是父组件</h1>
    <div class="gby">
      <p>我是一个段落</p>
    </div>

    <child></child>
  </div>
</template>

<style scoped>
  .parent {
    background-color: green;
  }

  .gby p {
    background-color: red;
  }
</style>

子组件:Child.vue

<template>
    <div class="child">
        <h1>我是子组件</h1>
        <div class="dyx">
            <p>我是子组件的段落</p>
        </div>
    </div>
</template>

<style scoped>
    .child .dyx p {
        background-color: blue;
    }
</style>

转换后,相当于

父组件:Parent.vue

<template>
  <div class="parent" id="app" data-v-e0f690c0="" >
    <h1 data-v-e0f690c0="" >我是父组件</h1>
    <div class="gby" data-v-e0f690c0="" >
      <p data-v-e0f690c0="" >我是一个段落</p>
    </div>

    <child data-v-e0f690c0="" ></child>
  </div>
</template>

<style scoped>
  .parent[data-v-e0f690c0] {
    background-color: green;
  }

  .gby p[data-v-e0f690c0] {
    background-color: red;
  }
</style>

子组件:Child.vue

<template>
    <div class="child" data-v-32d296d4="" >
        <h1 data-v-32d296d4="" >我是子组件</h1>
        <div class="dyx" data-v-32d296d4="" >
            <p data-v-32d296d4="" >我是子组件的段落</p>
        </div>
    </div>
</template>

<style scoped>
    .child .dyx p[data-v-32d296d4] {
        background-color: blue;
    }
</style>

生成的Dom树如下

示例dom

特点

  1. 将组件的样式的作用范围限制在了组件自身的标签,即:组件内部,包含子组件的根标签,但不包含子组件的除根标签之外的其它标签;所以 组件的css选择器也不能选择到子组件及后代组件的中的元素(子组件的根元素除外);

    因为它给选择器的最后一个选择器单元增加了属性选择器 [data-v-实例标识] ,而该属性选择器只能选中当前组件模板中的标签;而对于子组件,只有根元素 即有 能代表子组件的标签属性 data-v-子实例标识,又有能代表当前组件(父组件)的 签属性 data-v-父实例标识,子组件的其它非根元素,仅有能代表子组件的标签属性 data-v-子实例标识

  2. 如果递归组件有后代选择器,则该选择器会打破特性1中所说的子组件限制,从而选中递归子组件的中元素;

    原因:假设递归组件A的作用域样式中有选择器有后代选择器 div p ,则在每次递归中都会为本次递归创建新的组件实例,同时也会为该实例生成对应的选择器 div p[data-v-当前递归组件实例的实例标识],对于递归组件的除了第一个递归实例之外的所有递归实例来说,虽然 div p[data-v-当前递归组件实例的实例标识] 不会选中子组件实例(递归子组件的实例)中的 p 元素(具体原因已在特性1中讲解),但是它会选中当前组件实例中所有的 p 元素,因为 父组件实例(递归父组件的实例)中有匹配的 div 元素;

2. 深度作用选择器(>>>、/deep/、::v-deep)的原理

如果你希望 scoped 样式中的一个选择器能够选择到子组 或 后代组件中的元素,我们可以使用 深度作用选择器,它有三种写法:

  • >>>,示例: .gby div >>> #dyx p
  • /deep/,示例: .gby div /deep/ #dyx p.gby div/deep/ #dyx p
  • ::v-deep,示例: .gby div::v-deep #dyx p.gby div::v-deep #dyx p

它的原理与 Scoped CSS 的原理基本一样,只是第3步有些不同(前2步一样),具体如下:

  1. 为每个组件实例(注意:是组件的实例,不是组件类)生成一个能唯一标识组件的标识符,我称它为实例标识,记作 InstanceID;
  2. 给组件模板中的每一个标签对应的Dom元素(组件标签对应的Dom元素是该组件的根元素)添加一个标签属性,格式为 data-v-实例标识,示例:<div data-v-e0f690c0="" >
  3. 给组件的作用域样式 <style scoped> 的每一个深度作用选择器前面的一个选择器单元增加一个属性选择器[data-v-实例标识] ,示例:假设原选择器为 .cls #id >>> div,则更改后的选择器为 .cls #id[data-v-e0f690c0] div

因为Vue不会为深度作用选择器后面的选择器单元增加 属性选择器[data-v-实例标识],所以,后面的选择器单元能够选择到子组件及后代组件中的元素;

3. 其它实现方案

本文原文的标题是《单文件组件实现样式隔离实现方案与设计》,虽然现在改名并重新编排了,但仍保留了原文中其它的实现方案,原因如下;

  • 让读者了解其它的实现方案;
  • 对比下其它方案的特点,从而让读者明白Vue为什么选择 Scoped CSS 方案;

3.1. 方案1:增加能标识组件实例的父代选择器

思路:

  1. 为每个组件实例(注意:是组件的实例,不是组件类)生成一个能唯一标识组件实例的标识符,我称它为组件实例标识,简称实例标识,记作 InstanceID;
  2. 给组件的根元素增加一个能唯一代表组件的标识,如:给组件增加一个以组件实例标识命名的CSS类 或 标签属性;
  3. 给每个组件的所有CSS选择器都加一个的父代选择器,生成的新选择的格式为 父代选择器 原选择器;这样就使得原选择器的作用范围都被限制在了组件根元素的后代范围中。

父代选择器 可使用能唯一标识组件的 类选择器 或 属性选择器 来实现,比如:原选择器是 .gby div,增加父代选择器后是 [data-组件实例ID] .gby div

特点:

  • 将样式的作用范围限制在了组件内部,即:样式能作用到组件自身及子组件中的元素;所以 组件的css样式选择器能选择到子组件及后代组件的中的元素;

3.2. 方案2:给选择器的第一个选择器单元增加能标识组件实例的属性选择器

这个方案的思路跟 Vue 的 Scoped CSS的原理基本一样,只是第3步中的 属性选择器 [data-v-实例标识]加在了选择器中的第一个选择器单元上,而不是最后一个选择器单元;
思路:

  1. 为每个组件实例(注意:是组件的实例,不是组件类)生成一个能唯一标识组件实例的标识符,我称它为组件实例标识,简称实例标识,记作 InstanceID;
  2. 给组件模板中的每一个标签对应的Dom元素(组件标签对应的Dom元素是该组件的根元素)添加一个标签属性,格式为 data-v-实例标识,示例:<div data-v-e0f690c0="" >
  3. 给组件的作用域样式 <style scoped> 的每一个选择器的第一个选择器单元增加一个属性选择器 [data-v-实例标识] ,示例:假设原选择器为 .cls #id > div,则更改后的选择器为 .cls[data-v-e0f690c0] #id > div

特点:

  • 将样式的作用范围限制在了组件内部,即:样式能作用到组件自身及子组件中的元素;所以组件的css样式选择器能选择到子组件及后代组件的中的元素;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,565评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,021评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,003评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,015评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,020评论 5 370
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,856评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,178评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,824评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,264评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,788评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,913评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,535评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,130评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,102评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,334评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,298评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,622评论 2 343