Vue3 属性穿透

Vue3 属性穿透

Vue3 属性穿透包括自动穿透和属性绑定两种

  • 自动穿透 如果子组件只有一个根节点,父组件上的属性(class,id,style,事件,属性)就会向下流转到子组件的根节点上。
  • 属性绑定 如果子组件有多个根节点,父组件上的属性流转到子组件时,不确定绑定到哪个根节点上,这时就需要手动绑定属性到节点。

自动穿透

子组件只有一个根节点

简单样式穿透

vue 复制代码
<template>
    <button>提交</button>
</template>

父组件调用改子组件并传递 css,该样式会自动流转到子组件中。

vue 复制代码
<template>
  <div>
    <my-button class="my-button-background-color"></my-button>
  </div>
</template>
<script setup>
import MyButton from './components/my-button.vue'
</script>
<style>
.my-button-background-color{
  background-color: red;
}
</style>

如上,my-button-background-color 样式被自动添加到子组件的根节点上。

样式合并

子组件不变,父组件增加行内样式。

vue 复制代码
<template>
  <div>
    <my-button class="my-button-background-color" style="border: 2px solid green;"></my-button>
  </div>
</template>
<script setup>
import MyButton from './components/my-button.vue'
</script>
<style>
.my-button-background-color{
  background-color: red;
}
</style>

如上,行内样式也增加到了根节点。

监听事件

子组件不变,父组件增加监听事件。

vue 复制代码
<template>
  <div>
    <my-button class="my-button-background-color" style="border: 2px solid green;" @click="handleClick"></my-button>
  </div>
</template>
<script setup>
import MyButton from './components/my-button.vue'
const handleClick = () => {
  alert('监听点击事件');
}
</script>
<style>
.my-button-background-color{
  background-color: red;
}
</style>

click 监听器会被添加到子组件的根元素上。当原生的 button 被点击,会触发父组件的 onClick 方法。同样的,如果原生 button 元素自身也通过 v-on 绑定了一个事件监听器,则这个监听器和从父组件继承的监听器都会被触发。

隔代穿透

儿子根节点是另一个组件,属性会继续流转。 父组件不变

vue 复制代码
# 儿子组件
<template>
    <base-button></base-button>
</template>
<script setup>
import BaseButton from './base-button.vue'
</script>
vue 复制代码
# 孙子组件
<template>
    <button>孙子组件</button>
</template>

如上,属性通过儿子组件流转到孙子组件

注意:透传的属性不包含儿子组件上声明过的 props 或是针对 emits 声明事件的 v-on 侦听函数

禁用属性穿透

在某些情况下,你可能不希望子组件继承所有的属性,而是希望对属性的传递进行更精确的控制

js 复制代码
<script setup>
defineOptions({
  inheritAttrs: false
})
</script>

在儿子节点中增加如上代码

vue 复制代码
<template>
    <base-button></base-button>
</template>
<script setup>
import BaseButton from './base-button.vue'
import {defineOptions} from 'vue'
defineOptions({
  inheritAttrs: false
})
</script>

如上,属性没有流转到儿子节点。

注意:这些流转进来的属性可以在模板的表达式中直接用 $attrs 访问到

在儿子节点中增加模版代码

vue 复制代码
<template>
    <span>流转进来的属性: {{ $attrs }}</span>
    <base-button></base-button>
</template>
<script setup>
import BaseButton from './base-button.vue'
import {defineOptions} from 'vue'
defineOptions({
  inheritAttrs: false
})
</script>

如上,可以显示传递进来的属性。

注意:

  • 流转到儿子组件的属性在JavaScript中保留了它们原始的大小写,所以像 my-button-background-color 这样的一个属性需要通过 $attrs'my-button-background-color' 来访问
  • 事件会暴露出来一个函数 $attrs.handleClick

一般在模版中我们都会设置一个根节点

vue 复制代码
<template>
    <div>
        <base-button></base-button>
    </div>
</template>

如上 这样流转进来的属性会默认绑定到根节点(div)上,如果需要绑定到 <base-button></base-button> 组件上,需要手动绑定。

vue 复制代码
<template>
    <div>
        <base-button v-bind="$attrs"></base-button>
    </div>
</template>
<script setup>
import BaseButton from './base-button.vue'
import {defineOptions} from 'vue'
defineOptions({
  inheritAttrs: false
})
</script>

多根节点属性流转

多个根节点的组件没有自动属性流转行为,需要手动绑定到一个节点。如果没有手动绑定某个节点,会有警告。

在JS中访问流转属性

使用 useAttrs() 来访问一个组件的所有流转属性。

vue 复制代码
<template>
    <base-button></base-button>
</template>
<script setup>
import BaseButton from './base-button.vue'
import { useAttrs } from 'vue';
const attrs = useAttrs()
console.log(attrs)
</script>

结语

Vue3 中,属性穿透让父组件的非props属性自动传递给子组件根元素。通过设置inheritAttrs: false可禁用此功能以实现更精细控制。禁用后,利用$attrs手动绑定未声明的属性至所需位置,确保了属性传递的灵活性与可控性。

相关推荐
时光足迹4 小时前
腾讯云 TRTC UniApp SDK 从入门到上线
前端·vue.js·uni-app
时光足迹4 小时前
uni-app 里把加密视频嵌入页面播放?我分析了 4 种方案,只有 1 种接近完美
前端·vue.js·uni-app
时光足迹4 小时前
JPush UniApp UTS 插件完全参考手册:API、事件与厂商通道一网打尽
vue.js·ios·uni-app
时光足迹4 小时前
极光推送全攻略(下):uni-app 代码实现与 iOS 排查实战
vue.js·ios·uni-app
To_OC4 小时前
万字解析《JS 语言精粹》之第五章:继承 5 大核心精髓(JS 原型核心)
前端·javascript·代码规范
时光足迹5 小时前
极光推送全攻略(上):被iOS证书折磨了三天,我写了一份前端也能看懂的避坑指南
前端·ios·uni-app
DyLatte5 小时前
AI 时代,最危险的不是被替代,而是努力不沉淀
前端·后端·程序员
mCell5 小时前
【锐评】桌面端技术营销:别拿跑分当工程判断
前端·rust·electron
柒和远方5 小时前
从一次工程审查看 AI 学习产品的边界兜底:RAG 资料链路一致性实战
前端·后端·架构
疯狂的魔鬼5 小时前
一个"懂分寸"的文本省略组件是怎样炼成的
前端·vue.js·设计