组件通讯是vue3组合式开发的核心之一,现在我在写代码时,一个组件的代码超过了200行,基本都会拆分组件。组件拆分后,组件之间的通讯就很重要,总结了一下,目前有这么几种组件通讯类型:
安装。作为第三方插件,先安装再使用
npm i mitt -D
导出
新建个文件mitt.js,填入以下代码,即封装了一个mitt方法
// 第一种是创建一个工具js,不如说mitt.js,哪个组件要用到就引入该文件就好了。
import mitt from 'mitt'
const emitter = mitt()
export default emitter
组件中引入mitt,并通过事件发射指令
import mitt from '@/utils/mitt'
const emitter = mitt
const mittName = ['showOverall', 'showDriverAge', 'showAge', 'showOffences', 'showHelmetBelt', 'showOccRequire', 'showCumulativeScore']
const btns = ref([
{ title: '总体情况', type: 'warning' },
{ title: '驾龄', type: 'success' },
{ title: '年龄', type: 'success' },
{ title: '违法', type: 'success' },
{ title: '一盔一带', type: 'success' },
{ title: '从业资格', type: 'success' },
{ title: '累计计分', type: 'success' },
])
const chooseTag = i => {
btns.value.forEach(item => item.type = 'success')
btns.value[i].type = btns.value[i].type === 'warning' ? 'success' : 'warning'
emitter.emit(mittName[i])
}
上面的代码挺复杂,业务逻辑是我点击某个按钮,就显示对应的页面,按钮组如下:
每次点击一个按钮,就修改这个按钮的类型,同时发射这个按钮对应的指令名
组件B作为组件A的兄弟(广义的兄弟,五湖四海皆兄弟的意思)组件,当A发出指令后,它的其他兄弟都可以接收这个指令
<template>
<Overall v-if="showOverall"></Overall>
<DriverAge v-if="showDriverAge"></DriverAge>
<Age v-if="showAge"></Age>
<Offences v-if="showOffences"></Offences>
<HelmetBelt v-if="showHelmetBelt"></HelmetBelt>
<OccRequire v-if="showOccRequire"></OccRequire>
<CumulativeScore v-if="showCumulativeScore"></CumulativeScore>
</template>
<script setup>
import { ref } from 'vue'
import Overall from './tags/Overall.vue'
import DriverAge from './tags/DriverAge.vue';
import Age from './tags/Age.vue'
import Offences from './tags/Offences.vue'
import HelmetBelt from './tags/HelmetBelt.vue'
import OccRequire from './tags/OccRequire.vue'
import CumulativeScore from './tags/CumulativeScore.vue'
import mitt from '@/utils/mitt'
const emitter = mitt
const showOverall = ref(true)
const showDriverAge = ref(false)
const showAge = ref(false)
const showOffences = ref(false)
const showHelmetBelt = ref(false)
const showOccRequire = ref(false)
const showCumulativeScore = ref(false)
const events = [
{ emitterName: 'showOverall', showState: showOverall },
{ emitterName: 'showDriverAge', showState: showDriverAge },
{ emitterName: 'showAge', showState: showAge },
{ emitterName: 'showOffences', showState: showOffences },
{ emitterName: 'showHelmetBelt', showState: showHelmetBelt },
{ emitterName: 'showOccRequire', showState: showOccRequire },
{ emitterName: 'showCumulativeScore', showState: showCumulativeScore },
]
events.forEach(item => {
emitter.on(item.emitterName, () => {
// ChatGPT提供的代码,牛逼
events.forEach(i => {
i.showState.value = (i === item);
});
})
})
</script>
<style lang="scss" scoped>
.el-header {
// background-color: #333;
box-shadow: rgba(99, 99, 99, 0.2) 0px 2px 8px 0px;
display: flex;
}
</style>
同样,上面接收指令的代码也挺复杂的,还借助了chatAI的力量才实现的
解释一下吧,就是我在一个页面上写了很多个不同的组件,通过v-if来控制是否显示这个组件,而v-if则是通过组件A的按钮点击事件来确定的
const events = [
{ emitterName: 'showOverall', showState: showOverall },
{ emitterName: 'showDriverAge', showState: showDriverAge },
{ emitterName: 'showAge', showState: showAge },
{ emitterName: 'showOffences', showState: showOffences },
{ emitterName: 'showHelmetBelt', showState: showHelmetBelt },
{ emitterName: 'showOccRequire', showState: showOccRequire },
{ emitterName: 'showCumulativeScore', showState: showCumulativeScore },
]
events是定义的一个事件数组,emitterName是组件A发过来的事件名,showState则是显示状态
events.forEach(item => {
emitter.on(item.emitterName, () => {
// ChatGPT提供的代码,牛逼
events.forEach(i => {
i.showState.value = (i === item);
});
})
})
遍历训练events数组,监听里面的每个指令,当监听到了之后,就把它的showState置为true,只是多了个环节,判断当前的指令和当前点击的按钮是否相同,巧妙的实现了页面的点击响应,主要是能省不少代码。。。
俗事太多,很长一段时间没写代码,甚是怀恋,也不知道自己还能坚持几年