网站首页 > 资源文章 正文
简介
Vue3 中可以在 style 标签中使用 :deep() 的方法进行样式穿透,主要是解决在使用第三方的 UI 库(如 element-plus 等)时导致的对其样式设置不生效的问题。
1. scoped属性
在介绍deep样式穿透之前,我们先看一下scoped属性。
在一个 Vue3 组件的 style 标签中可以对其设置 scoped 属性 <style scoped>,</style>。该属性的作用主要有以下三个方面:
- [ Vue 会为该组件 template 中定义的 HTML 元素中添加一个 data-v-{一个hash值} 的属性选择器 ]
- [ 在 style 标签中写的 CSS 样式 Vue 都会在样式选择器的最后段添加上 data-v-{一个hash值} 的属性标签 ]
- [ 而在使用第三方的 UI 库时,只会为根元素添加 data-v-{一个hash值} 属性,而子元素则不会添加 ]
说的简单一点,其实就是在 HTML 的标签下添加了一个 属性选择器,可以根据这个属性选择器来保证各个组件的样式不被相互污染,从而实现一个模块化的效果。
举个栗子(element-plus):
<template>
<div class="main">
<el-input class="ipt"></el-input>
</div>
</template>
<script setup></script>
<style scoped>
.ipt {
width: 300px;
}
</style>
在这个栗子中,主要使用了一个 标签,并为其设置了一个宽度。
那么随之而来就产生一个问题,当使用以下方法修改样式时并不能生效:
.ipt .el-input__wrapper {
background-color: red;
}
但该选择器确实是添加进去了。
出现这种结果的原因就在于 Vue 将 [data-v-7a7a37b1] 属性添加到 .el-input__wrapper 之后, 而 .el-input__wrapper 的标签上并不存在 [data-v-7a7a37b1] 属性。那么 deep 样式穿透随之而来。
2. deep 样式穿透
使用方法
:deep(.ipt .el-input__wrapper) {
background-color: red;
}
:deep() 函数会把属性选择器放在最前面,那么就可以捕获到啦!
3. 源码解析
目录:
core-main/packages/compiler-sfc/src/compileStyle.ts
export function doCompileStyle(
scoped = false,
): SFCStyleCompileResults | Promise<SFCStyleCompileResults> {
......
if (scoped) {
plugins.push(scopedPlugin(longId))
}
......
}
在这个函数中,如果存在 scoped 属性,就会调用 postcss 这个插件,这个插件的主要作用就是把 CSS 转换成抽象语法树 (AST) 便于之后的操作。
function processRule(id: string, rule: Rule) {
......
rule.selector = selectorParser(selectorRoot => {
selectorRoot.each(selector => {
rewriteSelector(id, selector, selectorRoot)
})
}).processSync(rule.selector)
}
之后在 processRule 函数中调用 rewriteSelector() 方法对 CSS 选择器进行重写。
function rewriteSelector(
id: string,
selector: selectorParser.Selector,
selectorRoot: selectorParser.Root,
slotted = false,
) {
let node: selectorParser.Node | null = null
let shouldInject = true
// find the last child node to insert attribute selector
selector.each(n => {
......
if (n.type === 'pseudo') {
const { value } = n
// deep: inject [id] attribute at the node before the ::v-deep
// combinator.
if (value === ':deep' || value === '::v-deep') {
if (n.nodes.length) {
// .foo ::v-deep(.bar) -> [xxxxxxx] .foo .bar
// replace the current node with ::v-deep's inner selector
let last: selectorParser.Selector['nodes'][0] = n
n.nodes[0].each(ss => {
selector.insertAfter(last, ss)
last = ss
})
// insert a space combinator before if it doesn't already have one
const prev = selector.at(selector.index(n) - 1)
if (!prev || !isSpaceCombinator(prev)) {
selector.insertAfter(
n,
selectorParser.combinator({
value: ' ',
}),
)
}
selector.removeChild(n)
} else {
......
}
......
}
当遇到 :deep 时,会将原来的属性选择器添加到前面元素中,即:.foo ::v-deep(.bar) -> [xxxxxxx] .foo .bar 通过这种方法就能定位到第三方 UI库中的 CSS 选择器了。
结语
在 Vue3 中,当使用一些第三方的 UI 库时,由于 Vue3 实现了模块化封装,那么再设置 UI 库的 CSS 样式时有时会出现设置不成功的问题,那么这个时候可以考虑使用 :deep() 来进行样式穿透。
- 上一篇: Python高性能编程:五种核心优化技术的原理与Python代码
- 下一篇:已经是最后一篇了
猜你喜欢
- 2025-07-23 Python高性能编程:五种核心优化技术的原理与Python代码
- 2025-07-23 竞技味十足的11代斯巴鲁WRX STi,战斗的同时还带点"骚气"
- 2025-07-23 Python 代码优化四大杀器:性能提升 10 倍的硬核技巧
- 2025-07-23 今年最火开源前端框架——Svelte 5正式发布稳定版、彻底重写、新增$语法 、star数近8万
- 2025-07-23 水声通信网基于载波侦听多路访问的MAC协议
- 2025-07-23 固定翼无人机的机翼和舵面(固定翼无人机有什么布局)
- 2025-07-23 SERES and Huawei struggle to deliver Harmony
- 2025-07-23 翻译(翻译中文)
- 2025-07-23 Blender4.4版本正式发布,5大重要功能更新详解!
- 2025-07-23 最全紧固件中英文对照,外贸必备词典一
你 发表评论:
欢迎- 07-23Vue3中deep样式穿透的使用细节及源码解析
- 07-23Python高性能编程:五种核心优化技术的原理与Python代码
- 07-23竞技味十足的11代斯巴鲁WRX STi,战斗的同时还带点"骚气"
- 07-23Python 代码优化四大杀器:性能提升 10 倍的硬核技巧
- 07-23今年最火开源前端框架——Svelte 5正式发布稳定版、彻底重写、新增$语法 、star数近8万
- 07-23水声通信网基于载波侦听多路访问的MAC协议
- 07-23固定翼无人机的机翼和舵面(固定翼无人机有什么布局)
- 07-23SERES and Huawei struggle to deliver Harmony
- 最近发表
-
- Vue3中deep样式穿透的使用细节及源码解析
- Python高性能编程:五种核心优化技术的原理与Python代码
- 竞技味十足的11代斯巴鲁WRX STi,战斗的同时还带点"骚气"
- Python 代码优化四大杀器:性能提升 10 倍的硬核技巧
- 今年最火开源前端框架——Svelte 5正式发布稳定版、彻底重写、新增$语法 、star数近8万
- 水声通信网基于载波侦听多路访问的MAC协议
- 固定翼无人机的机翼和舵面(固定翼无人机有什么布局)
- SERES and Huawei struggle to deliver Harmony
- 翻译(翻译中文)
- Blender4.4版本正式发布,5大重要功能更新详解!
- 标签列表
-
- 电脑显示器花屏 (79)
- 403 forbidden (65)
- linux怎么查看系统版本 (54)
- 补码运算 (63)
- 缓存服务器 (61)
- 定时重启 (59)
- plsql developer (73)
- 对话框打开时命令无法执行 (61)
- excel数据透视表 (72)
- oracle认证 (56)
- 网页不能复制 (84)
- photoshop外挂滤镜 (58)
- 网页无法复制粘贴 (55)
- vmware workstation 7 1 3 (78)
- jdk 64位下载 (65)
- phpstudy 2013 (66)
- 卡通形象生成 (55)
- psd模板免费下载 (67)
- shift (58)
- localhost打不开 (58)
- 检测代理服务器设置 (55)
- frequency (66)
- indesign教程 (55)
- 运行命令大全 (61)
- ping exe (64)
本文暂时没有评论,来添加一个吧(●'◡'●)