Vue3通信方式之defineProps、defineEmits、useAttrs、插件mitt和v-model

1、使用defineProps

props可以实现父子组件通信,在vue3中我们可以通过defineProps获取父组件传递的数据。且在组件内部不需要引入defineProps方法可以直接使用!

父组件给子组件传递数据

 <Child info="我爱祖国" :money="money"></Child>

 子组件获取父组件传递数据:方式1

    let props = defineProps({
      info:{
       type:String,//接受的数据类型
       default:'默认参数',//接受默认数据
      },
      money:{
       type:Number,
       default:0
    }})

子组件获取父组件传递数据:方式2

    let props = defineProps(["info",'money']);

子组件获取到props数据就可以在模板中使用了,但是切记props是只读的(只能读取,不能修改)

2、使用defineEmits接受自定义事件 

在vue框架中事件分为两种:一种是原生的DOM事件,另外一种自定义事件。

原生DOM事件可以让用户与网页进行交互,比如click、dbclick、change、mouseenter、mouseleave…

自定义事件可以实现子组件给父组件传递数据

 

2.1原生DOM事件

代码如下:

     <pre @click="handler">
          我是祖国的老花骨朵
     </pre>

当前代码级给pre标签绑定原生DOM事件点击事件,默认会给事件回调注入event事件对象。当然点击事件想注入多个参数可以按照下图操作。但是切记注入的事件对象务必叫做$event. 

 

      <div @click="handler1(1,2,3,$event)">我要传递多个参数</div>

在Event2子组件内部触发这个自定义事件

    <template>
      <div>
        <h1>我是子组件2</h1>
        <button @click="handler">点击我触发xxx自定义事件</button>
      </div>
    </template>
    
    <script setup lang="ts">
    let $emit = defineEmits(["xxx"]);
    const handler = () => {
      $emit("xxx", "法拉利", "茅台");
    };
    </script>
    <style scoped>
    </style>

我们会发现在script标签内部,使用了defineEmits方法,此方法是vue3提供的方法,不需要引入直接使用。defineEmits方法执行,传递一个数组,数组元素即为将来组件需要触发的自定义事件类型,此方执行会返回一个$emit方法用于触发自定义事件。

当点击按钮的时候,事件回调内部调用$emit方法去触发自定义事件,第一个参数为触发事件类型,第二个、三个、N个参数即为传递给父组件的数据。

需要注意的是:代码如下

    <Event2  @xxx="handler3" @click="handler"></Event2>

 正常说组件标签书写@click应该为原生DOM事件,但是如果子组件内部通过defineEmits定义就变为自定义事件了

    let $emit = defineEmits(["xxx",'click']);

 3、全局事件总线(插件mitt)

全局事件总线可以实现任意组件通信,在vue2中可以根据VM与VC关系推出全局事件总线。

但是在vue3中没有Vue构造函数,也就没有Vue.prototype.以及组合式API写法没有this,

那么在Vue3想实现全局事件的总线功能就有点不现实啦,如果想在Vue3中使用全局事件总线功能

可以使用插件mitt实现。

 

 新建个文件mitt.js,填入以下代码,即封装了一个mitt方法

import mitt from 'mitt'
const emitter = mitt()
export default emitter

二、组件A发射指令

组件中引入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接收指令

组件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>

 4、v-model

v-model指令可是收集表单数据(数据双向绑定),除此之外它也可以实现父子组件数据同步。

而v-model实指利用props[modelValue]与自定义事件[update:modelValue]实现的。

下方代码:相当于给组件Child传递一个props(modelValue)与绑定一个自定义事件update:modelValue
实现父子组件数据同步

 

    <Child v-model="msg"></Child>

 在vue3中一个组件可以通过使用多个v-model,让父子组件多个数据同步,下方代码相当于给组件Child传递两个props分别是pageNo与pageSize,以及绑定两个自定义事件update:pageNo与update:pageSize实现父子数据同步

    <Child v-model:pageNo="msg" v-model:pageSize="msg1"></Child>

 

5、useAttrs

使用vue中的useAttrs方法可以获取组件标签上的属性和事件。

两个组件分别为父组件app.vue和子组件helloworld.vue

父组件:

<template>
  <el-button type="primary" size="small" :icon="Edit"></el-button>
  <HelloWorld msg="我是helloword子组件" type="primary" size="small" :icon="Edit" title="你好编辑" />
</template>

<script setup lang="ts">
  import HelloWorld from './components/HelloWorld.vue'
  import {
    Edit,
    Delete
  } from '@element-plus/icons-vue'
</script>

<style scoped>

</style>

子组件:

<template>


  <div class="card" :title="$attrs.title">
    <h3>{{ msg }}</h3>
    <el-button :="$attrs"></el-button>
  </div>

</template>

<script setup lang="ts">
  import {
    ref
  } from 'vue'
  import {
    useAttrs
  } from 'vue'

  let $attrs = useAttrs()
  console.log($attrs)

  defineProps < {
    msg: string
  } > ()

  const count = ref(0)
</script>

<style scoped>
  .card {
    background-color: antiquewhite;
  }

  .read-the-docs {
    color: #888;
  }
</style>

上述代码中,首先在父组件中引入子组件:

import HelloWorld from './components/HelloWorld.vue'

在父组件中,子组件标签上加上所需属性:

<HelloWorld msg="我是helloword子组件" type="primary" size="small" :icon="Edit" title="你好编辑" />

然后在子组件中引入vue中的useAttrs方法:

  import {
    useAttrs
  } from 'vue'

useAttrs方法返回的是一个对象

  let $attrs = useAttrs()
  console.log($attrs)

 

可以看到输出结果中,能接受到父组件标签中的属性值。

最后在子组件中属性绑定$attrs的值:

<el-button :="$attrs"></el-button>

意事项(重点)

需要注意的是:props和useAttrs方法都可以获取到父组件传过来来的属性和属性值;
但是如果一旦用prop接受了其中的某个属性和属性值,那么useAttrs就接受不到这个属性和属性值。

  • 28
    点赞
  • 32
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值