组件通信
2024/4/30大约 5 分钟
组件通信
自定义属性-props
父组件向子组件传递数据
父组件
<template>
    <div class="parent">
        <h1>父组件</h1>
        <Child :msg="msg" :count="count" />
    </div>
</template>
<script setup>
import Child from './Child.vue'
import { ref } from 'vue'
const msg = ref('Hello World')
const count = ref(0)
</script>子组件
<template>
    <div class="child">
        <h1>子组件</h1>
        <h6>msg:{{ props.msg }}</h6>
        <button @click="handler">count:{{ count }}</button>
    </div>
</template>
<script setup lang="ts">
import { defineProps } from 'vue'
let props = defineProps({
    msg: String,
    count: Number,
});
const handler = () => {
    //props.count += 1; // 这样会报错,因为props是只读的
    console.log(props.count);
};
</script>注意
- 单向数据流
- 子组件不能修改父组件传递的值
自定义事件-$emit
子组件向父组件传递数据
父组件
<template>
    <div class="parent">
        <h1>父组件</h1>
        <p>子组件传递过来的数据:</p>
        <li v-for="item in list">{{ item }}</li>
        <Child @my-event="handle" />
    </div>
</template>
<script setup>
import Child from './Child.vue'
import { ref } from 'vue'
const list = ref([])
function handle(value) {
    console.log('子传父的数据', value)
    list.value.push(value)
}
</script>子组件
<template>
    <div class="child">
        <h1>子组件</h1>
        <button @click="$emit('my-event', 'Hello')">触发事件</button>
        <button @click="handle">触发事件</button>
    </div>
</template>
<script setup lang="ts">
let $emit = defineEmits(['my-event'])
const handle = () => {
    $emit('my-event', 'World')
}
</script>事件总线-$bus
适合任意组件之间的通信,比如:兄弟组件、父子组件等。
事件总线
首先安装mitt
npm install --save mittimport mitt from 'mitt';
const $bus = mitt();
export default $bus;父组件
<template>
    <div class="parent">
        <h1>父组件</h1>
        <button @click="handle">没收玩具</button>
        <Child1 />
        <Child2 />
    </div>
</template>
<script setup>
import Child1 from './Child1.vue'
import Child2 from './Child2.vue'
import $bus from '@/bus';
const handle = () => {
    $bus.emit('clear-toy');
}
</script>哥哥组件
<template>
    <div class="child">
        <h1>哥哥组件</h1>
        <li v-for="toy in toys">{{ toy }}</li>
        <button v-show="toys.length > 0" @click="handle">给弟弟一个玩具</button>
    </div>
</template>
<script setup lang="ts">
import { ref, onMounted } from 'vue'
import $bus from '@/bus'
let toys = ref(['BB枪', '变形金刚', '赛车'])
const handle = () => {
    let toy = toys.value.pop()
    $bus.emit('send-toy', toy)
}
onMounted(() => {
    $bus.on('clear-toy', () => {
        toys.value = []
    })
})
</script>弟弟组件
<template>
    <div class="child">
        <h1>弟弟组件</h1>
        <li v-for="toy in toys">{{ toy }}</li>
    </div>
</template>
<script setup lang="ts">
import { ref, onMounted } from 'vue'
import $bus from '@/bus'
let toys = ref(['奥特曼'])
onMounted(() => {
    $bus.on('send-toy', (data: any) => {
        toys.value.push(data)
    })
    $bus.on('clear-toy', () => {
        toys.value = []
    })
})
</script>v-model
父子组件双向数据同步,相当于:
- 通过props传递modelValue,实现父到子传递数据
- 通过emit触发update:modelValue,实现子到父传递数据
父组件
<template>
    <div class="parent">
        <h1>父组件</h1>
        {{ msg }}
        <Child v-model="msg" />
        <Child :modelValue="msg" @update:modelValue="msg = $event" />
    </div>
</template>
<script setup>
import Child from './Child.vue'
import { ref } from 'vue'
const msg = ref('Hello World')
</script>子组件
<template>
    <div class="child">
        <h1>子组件</h1>
        <h6>msg:{{ props.modelValue }}</h6>
        <button @click="handler">改变</button>
    </div>
</template>
<script setup lang="ts">
import { defineProps } from 'vue'
let props = defineProps(['modelValue']);
let $emit = defineEmits(['update:modelValue'])
const handler = () => {
    let msg = props.modelValue + ' hello '
    $emit('update:modelValue', msg)
};
</script>useAttrs
用于传递属性和事件
父组件
<template>
    <div class="parent">
        <h1>父组件</h1>
        <Child :msg="msg" :count="count" @xxx="handle" />
    </div>
</template>
<script setup>
import Child from './Child.vue'
import { ref } from 'vue'
const msg = ref('Hello World')
const count = ref(0)
const handle = () => {
    count.value++
    console.log(count.value)
}
</script>子组件
<template>
    <div class="child">
        <h1>子组件</h1>
        <h6>msg:{{ $attrs.msg }}</h6>
        <button @click="$attrs.onXxx">count:{{ $attrs.count }}</button>
    </div>
</template>
<script setup lang="ts">
import { useAttrs } from 'vue'
let $attrs = useAttrs();
</script>useAttrs和defineProps的区别
- 和使用 defineProps 接收属性时相比,useAttrs 的优先级要低
- useAttrs:方法能够接收到所有属性,包括未在组件中声明的属性,但不能够对接收到的属性进行类型校验和默认值设置。
- defineProps:方法只能接收到在组件中声明的属性,但能够对接收到的属性进行类型校验和默认值设置,使得组件能够更加健壮。
ref和$parent
ref用于:父->子 $parent用于:子->父
父组件
<template>
    <div class="parent">
        <h1>父组件</h1>
        <p> 工资: {{ money }}</p>
        <button @click="handle">给孩子生活费</button>
        <Child ref="son" />
    </div>
</template>
<script setup>
import Child from './Child.vue'
import { ref } from 'vue'
const money = ref(1000)
defineExpose({ money })
let son = ref()
const handle = () => {
    money.value -= 100
    son.value.money += 100
}
</script>子组件
<template>
    <div class="child">
        <h1>子组件</h1>
        <p> 生活费: {{ money }}</p>
        <button @click="handle($parent)">要生活费</button>
    </div>
</template>
<script setup lang="ts">
import { ref } from 'vue'
const money = ref(10)
defineExpose({ money })
const handle = ($parent: any) => {
    money.value += 100
    $parent.money -= 100
}
</script>provide和inject
常用于祖孙组件之间通信
父组件
<template>
    <div class="parent">
        <h1>父组件</h1>
        <p>{{ count }}</p>
        <Child />
    </div>
</template>
<script setup>
import Child from './Child.vue'
import { ref, provide } from 'vue'
const count = ref(123)
provide('count', count)
</script>子组件
<template>
    <div class="child">
        <h1>子组件</h1>
        <GrandChild></GrandChild>
    </div>
</template>
<script setup lang="ts">
import GrandChild from './GrandChild.vue';
</script>孙组件
<template>
    <div class="grandchild">
        <h1>孙组件</h1>
        <button @click="handler">count:{{ count }}</button>
    </div>
</template>
<script setup lang="ts">
import { inject } from 'vue'
let count = inject('count', 0)
const handler = () => {
    count.value += 1
};
</script>状态管理-pinia
任意组件之间通信
store
import { createPinia } from 'pinia';
let store = createPinia();
export default store;store modules
import { defineStore } from "pinia";
import { computed, ref } from "vue";
const useCountStore = defineStore('info', () => {
    const count = ref(0)
    const double = computed(() => {
        return count.value * 2
    })
    const increment = (num: number) => {
        count.value += num
    }
    return {
        count, double, increment
    }
});
export default useCountStore;父组件
<template>
    <div class="parent">
        <h1>父组件</h1>
        <p>count:{{ count }},double:{{ double }}</p>
        <div style="display: flex;">
            <Child1 />
            <Child2 />
        </div>
    </div>
</template>
<script setup>
import Child1 from './Child1.vue'
import Child2 from './Child2.vue';
import useCountStore from '@/store/modules/count';
import { storeToRefs } from 'pinia';
const countStore = useCountStore()
const { count, double } = storeToRefs(countStore)
</script>子组件1
<template>
    <div class="child">
        <h1>子组件</h1>
        <button @click="handler">count:{{ count }},double:{{ double }}</button>
    </div>
</template>
<script setup lang="ts">
import useCountStore from '@/store/modules/count';
import { storeToRefs } from 'pinia';
const countStore = useCountStore()
const { count, double } = storeToRefs(countStore)
const handler = () => {
    countStore.increment(1)
};
</script>子组件2
<template>
    <div class="child">
        <h1>子组件</h1>
        <button @click="handler">count:{{ count }},double:{{ double }}</button>
    </div>
</template>
<script setup lang="ts">
import useCountStore from '@/store/modules/count';
import { storeToRefs } from 'pinia';
const countStore = useCountStore()
const { count, double } = storeToRefs(countStore)
const handler = () => {
    countStore.increment(2)
};
</script>插槽-slot
实现父与子之间通信,不仅仅是数据,还可以包含页面结构
父组件
<template>
    <div class="parent">
        <h1>父组件</h1>
        <Child>
            <!-- 默认插槽 -->
            <h3>父组件定义的内容</h3>
            <!-- 具名插槽 -->
            <template v-slot:header>
                <h2>header</h2>
            </template>
            <!-- 作用域插槽 -->
            <template v-slot:footer="{ data }">
                <li v-for="item in data">{{ item }}</li>
            </template>
        </Child>
    </div>
</template>
<script setup>
import Child from './Child.vue'
</script>子组件
<template>
    <div class="child">
        <h1>子组件</h1>
        <h1>默认插槽</h1>
        <slot></slot>
        <h1>==========</h1>
        <h1>具名插槽</h1>
        <slot name="header"></slot>
        <h1>==========</h1>
        <h1>作用域插槽</h1>
        <slot name="footer" :data="['a', 'b', 'c']"></slot>
        <h1>==========</h1>
    </div>
</template>
<script setup lang="ts">
</script>