Maison > interface Web > Voir.js > Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

PHPz
Libérer: 2023-05-15 09:28:05
avant
2998 Les gens l'ont consulté

Résumé de cet article

  • Les composants parent et enfant peuvent communiquer via des événements

  • Envoi d'événements et surveillance des rappels comportant des paramètres

  • Utilisez la section emis de le composant pour organiser les événements du composantemits板块 整理组件事件

  • 使用 组件emits板块的 Object形式 校验外传的参数值

  • 结合$emitv-bindv-model 实现 父子组件通信(数据双向绑定)

  • 结合$emitv-bindv-model 实现 父子组件通信(多个字段的应用案例)

  • 自己设置修饰符

    • 试验this.modelModifiers的作用

  • 下面在子组件的点击回调handleClick()中,通过this.modelModifiers.[自己设置修饰符名]实现自己设置修饰符逻辑

  • 插槽【slot】【传组件示例】

  • 注意,slot标签上是无法直接增加事件(修饰符)的,如有需要,可以在<slot>外层包裹一层标签,再加上事件

  • 插槽【传 字符串示例】

  • 插槽【传 自己设置子组件 示例】

  • 插槽作用域问题

  • 插槽 UI默认值

  • 插槽的灵活拆分与应用【具名插槽】

  • v-slot指令的简写

  • 普通的v-for例子 进行 列表渲染

  • v-for结合v-bindv-slot<slot></slot>做列表渲染

  • 使用解构概念进行简写

  • 动态组件

    • 常规的利用双向绑定特性,通过点击事件切换UI的写法

    • 动态组件写法

  • 异步组件

父子组件可通过事件 进行通信

前面的笔记 —— 《Vue3 | 组件的定义及复用性、局部组件、全局组件、组件间传值及其校验、单项数据流、Non-props属性》,单向数据流的概念,
即子组件无法修改来自父组件的数据字段,

假如确要修改,可以使用下面说的方式进行通信:
首先,在子组件的UI点击回调方法中,调用this.$emit('【自己设置事件名】')
向外发送一个事件

接着各级父组件会收到这个事件,
则在父组件中 调用 子组件标签处,
以 @【事件名】= "回调方法名"的形式,监听该事件以及配置回调方法
回调方法中就可 对 子组件用意修改 的 父组件数据字段 进行修改;

注意,
触发事件的命名,用驼峰命名法(如下heHeDa);
监听事件的命名,用横杆间隔法(如下he-he-da)

代码:

<!DOCTYPE html><html><head>    <meta charset="UTF-8">    <meta name="viewport" content="width=device-width, initial-scale=1.0">    <title>Hello World! heheheheheheda</title>    <script src="https://unpkg.com/vue@next"></script></head><body>    <div id="heheApp"></div></body><script>    const app = Vue.createApp({        data() {            return {count: 1}        },        methods: {            handleItemEvent() {                this.count += 1;            }        },        template: `        <div>            <counter :count="count" @he-he-da="handleItemEvent"/>        </div>`    });    app.component(&#39;counter&#39;, {        props: [&#39;count&#39;],        methods: {            handleItemClick() {                this.$emit(&#39;heHeDa&#39;);            }        },        template:`        <div @click="handleItemClick">{{count}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script></html>
Copier après la connexion

运行,点击组件:Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

携带参数的事件 发送和监听回调

this.$emit()可以增加参数位,
父组件的监听回调中,
则可加形参位 用于接收参数(如handleItemEvent(param)中的 param);

代码:

<script>    const app = Vue.createApp({        data() {            return {count: 1}        },        methods: {            handleItemEvent(param) {                this.count += param;            }        },        template: `        <div>            <counter :count="count" @add-count="handleItemEvent"/>        </div>`    });    app.component(&#39;counter&#39;, {        props: [&#39;count&#39;],        methods: {            handleItemClick() {                this.$emit(&#39;addCount&#39;, 8);            }        },        template:`        <div @click="handleItemClick">{{count}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

运行,点击效果:

Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants
子组件需 发送多个参数 亦可,只需在this.$emit()按需增加参数位,
父组件的监听回调中,增加对应的形参 去接收就可:

<script>    const app = Vue.createApp({        data() {            return {count: 1}        },        methods: {            handleItemEvent(param1, param2, param3) {                this.count = this.count + param1 + param2 + param3;                console.log(this.count);            }        },        template: `        <div>            <counter :count="count" @add-count="handleItemEvent"/>        </div>`    });    app.component(&#39;counter&#39;, {        props: [&#39;count&#39;],        methods: {            handleItemClick() {                this.$emit(&#39;addCount&#39;, 8, 2, 6);            }        },        template:`        <div @click="handleItemClick">{{count}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

效果:Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants



当然 父组件 接收 子组件参数 后的 计算逻辑,
可以在 子组件传参 的时候 计算完成 再传给this.$emit()
父组件接收时,直接 受值就可(handleItemEvent(count)

🎜 utilisez la forme Object de la section emis du composant pour vérifier la valeur du paramètre transmise🎜🎜🎜🎜🎜🎜 combinée avec < code>$emit, v-bind et v-model réalisent la communication entre composants parent-enfant (liaison de données bidirectionnelle) 🎜🎜🎜🎜🎜🎜 combinés avec $emit, v-bind et v-model implémentent la communication des composants parent-enfant (cas d'application de plusieurs champs) 🎜🎜🎜🎜🎜 🎜Définissez les modificateurs par vous-même🎜🎜🎜🎜🎜🎜🎜 Testez l'effet de this.modelModifiers🎜🎜🎜🎜🎜🎜 ci-dessous dans le rappel de clic handleClick()</ code> du sous-composant, transmettez <code>this.modelModifiers. [Définissez le nom du modificateur par vous-même]Implémentez la logique de définition du modificateur par vous-même🎜🎜🎜🎜🎜🎜Slot [slot] [Exemple de passage du composant ]🎜🎜🎜🎜🎜🎜Notez que les événements ne peuvent pas être ajoutés directement à la balise slot (Modificateur), si nécessaire, vous pouvez envelopper une balise à l'extérieur de <slot> et ajouter l'événement 🎜🎜🎜🎜🎜🎜 slot [Exemple de chaîne de passage]🎜🎜🎜🎜🎜🎜 Slot [Exemple de définition de sous-composants par vous-même] 🎜🎜🎜🎜🎜🎜Problème de portée du slot🎜🎜🎜🎜🎜🎜Valeur par défaut de l'interface utilisateur du slot🎜🎜🎜 🎜🎜🎜Flexible fractionnement et application des slots [Slots nommés] 】🎜🎜🎜🎜🎜🎜L'abréviation de v-slot instruction🎜🎜🎜🎜🎜🎜Exemple v-for ordinaire pour la liste le rendu🎜🎜🎜🎜🎜🎜v-for combine v-bind, v-slot, <slot></code > Rendu de liste de tâches🎜🎜🎜🎜🎜 🎜Utilisez le concept de <code>déconstruction pour l'abréviation🎜🎜🎜🎜🎜🎜Composants dynamiques🎜🎜🎜🎜🎜🎜🎜Utilisez de manière conventionnelle la liaison bidirectionnelle fonctionnalité pour changer l'écriture de l'interface utilisateur via des événements de clic 🎜🎜🎜🎜🎜🎜Comment écrire des composants dynamiques🎜🎜🎜🎜🎜🎜Composants asynchrones🎜🎜🎜

Les composants parents et enfants peuvent communiquer via des événements

🎜🎜Notes précédentes - "Vue3 | Définition et réutilisabilité des composants, composants locaux, composants globaux, transfert de valeur entre composants et vérification, flux de données unique, attributs non-props", le concept de flux de données unidirectionnel,
Autrement dit, le composant enfant ne peut pas modifier les champs de données du composant parent.

Si vous souhaitez vraiment le modifier, vous pouvez communiquer dans le de la manière suivante :
Tout d'abord, dans la méthode de rappel de clic de l'interface utilisateur du composant enfant, appelez this.$emit('[Définissez vous-même le nom de l'événement]'),
envoyez un event;

puis chaque composant parent recevra cet événement,
puis appellera la balise du composant enfant dans le composant parent,
sous la forme de @[nom de l'événement] = "nom de la méthode de rappel", Écoutez cet événement et configurez la méthode de rappel
 ; La méthode de rappel peut modifier les champs de données du composant parent que le composant enfant a l'intention de modifier ;🎜🎜🎜🎜🎜Remarque :
Pour nommer l'événement déclencheur, utilisez la méthode de dénomination camel case. (heHeDa ci-dessous);
Pour nommer l'événement d'écoute, utilisez la méthode d'intervalle de barre (He-he-da ci-dessous). 🎜🎜🎜Code : 🎜
<script>    const app = Vue.createApp({        data() {            return {count: 1}        },        methods: {            handleItemEvent(count) {                this.count = count;                console.log(this.count);            }        },        template: `        <div>            <counter :count="count" @add-count="handleItemEvent"/>        </div>`    });    app.component(&#39;counter&#39;, {        props: [&#39;count&#39;],        methods: {            handleItemClick() {                this.$emit(&#39;addCount&#39;, this.count + 16);            }        },        template:`        <div @click="handleItemClick">{{count}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion
🎜Exécutez, cliquez sur le composant : Méthodes Vue3 pour la communication entre les composants parents et enfants et la liaison bidirectionnelle entre les composants🎜

Envoi d'événements et surveillance des rappels avec paramètres

🎜🎜this.$emit() peut ajouter des bits de paramètre,
Dans le rappel d'écoute du composant parent,
vous pouvez ajouter des paramètres formels pour recevoir des paramètres (tels que param dans handleItemEvent(param)< /code>) ;🎜🎜🎜Code : 🎜<div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:css;toolbar:false">&lt;script&gt; const app = Vue.createApp({ data() { return {count: 1} }, methods: { handleItemEvent(count) { this.count = count; console.log(this.count); } }, template: ` &lt;div&gt; &lt;counter :count=&quot;count&quot; @add-count=&quot;handleItemEvent&quot;/&gt; &lt;/div&gt;` }); app.component(&amp;#39;counter&amp;#39;, { props: [&amp;#39;count&amp;#39;], emits: [&amp;#39;hehehe&amp;#39;], methods: { handleItemClick() { this.$emit(&amp;#39;addCount&amp;#39;, this.count + 16); } }, template:` &lt;div @click=&quot;handleItemClick&quot;&gt;{{count}}&lt;/div&gt; ` }); const vm = app.mount(&amp;#39;#heheApp&amp;#39;);&lt;/script&gt;</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div>🎜🎜Exécuter, effet de clic : 🎜🎜🎜<img src="https://img.php.cn/upload/article/000/000/164/168411408827207.png " alt=" Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants " />🎜<br/> Les composants enfants peuvent également envoyer plusieurs paramètres, ajoutez-les simplement si nécessaire dans <code>this.$emit( ) Bit de paramètre,
Dans le rappel d'écoute du composant parent, ajoutez simplement le paramètre formel correspondant pour le recevoir : 🎜🎜
<script>    const app = Vue.createApp({        data() {            return {count: 1}        },        methods: {            handleItemEvent(count) {                this.count = count;                console.log(this.count);            }        },        template: `        <div>            <counter :count="count" @add-count="handleItemEvent"/>        </div>`    });    app.component(&#39;counter&#39;, {        props: [&#39;count&#39;],        emits: {            addCount: (count) => {                if (count < 0) {                    return true;                }                return false;            }        },        methods: {            handleItemClick() {                this.$emit(&#39;addCount&#39;, this.count + 16);            }        },        template:`        <div @click="handleItemClick">{{count}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion
Copier après la connexion
🎜Effet : Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants🎜🎜

🎜Of Bien sûr, la logique de calcul après que le composant parent a reçu les paramètres du composant enfant,
Il peut être calculé lorsque le sous-composant passe les paramètres, puis passé à this.$emit() !
Lorsque le composant parent le reçoit, acceptez simplement la valeur directement (handleItemEvent(count)); 🎜🎜
<script>    const app = Vue.createApp({        data() {            return {count: 1}        },        template: `            <counter v-model="count"/>`    });    app.component(&#39;counter&#39;, {        props: [&#39;modelValue&#39;],        methods: {            handleItemClick() {                this.$emit(&#39;update:modelValue&#39;, this.modelValue + 16);                console.log(vm.$data.count);            }        },        template:`        <div @click="handleItemClick">{{modelValue}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion
Copier après la connexion
🎜L'effet est le même que dans l'exemple précédent 🎜 ;

使用 组件的emits板块 整理组件事件

实际开发场景中,我们一个组件自己设置的触发事件可能会很多,
我们不可能一个一个去梳理核实,
这个时候即可以使用 组件的emits板块 来整理组件的事件;

可以把组件中 自己设置到的事件都写在这里,方便梳理,提高可读性
或者者把 想要定义的事件 写在这里,
如此一来,假如不记得编写对应的自己设置事件,
Vue系统会在运行时 给予警告

<script>    const app = Vue.createApp({        data() {            return {count: 1}        },        methods: {            handleItemEvent(count) {                this.count = count;                console.log(this.count);            }        },        template: `        <div>            <counter :count="count" @add-count="handleItemEvent"/>        </div>`    });    app.component(&#39;counter&#39;, {        props: [&#39;count&#39;],        emits: [&#39;hehehe&#39;],        methods: {            handleItemClick() {                this.$emit(&#39;addCount&#39;, this.count + 16);            }        },        template:`        <div @click="handleItemClick">{{count}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion
Copier après la connexion

假如不记得编写对应的自己设置事件,Vue系统会在运行时 给予警告

Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

使用 组件emits板块的 Object形式 校验外传的参数值

可以根据需要,使用 组件emits板块的 Object形式 校验外传的参数值,
如下,子组件的emits板块,
‘key’值定义对应的事件名,‘value’值定义一个校验函数,

返回true表示同意数值外传,
返回false表示不同意,会给出警告;

<script>    const app = Vue.createApp({        data() {            return {count: 1}        },        methods: {            handleItemEvent(count) {                this.count = count;                console.log(this.count);            }        },        template: `        <div>            <counter :count="count" @add-count="handleItemEvent"/>        </div>`    });    app.component(&#39;counter&#39;, {        props: [&#39;count&#39;],        emits: {            addCount: (count) => {                if (count < 0) {                    return true;                }                return false;            }        },        methods: {            handleItemClick() {                this.$emit(&#39;addCount&#39;, this.count + 16);            }        },        template:`        <div @click="handleItemClick">{{count}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion
Copier après la connexion

运行,点击效果:Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

结合$emitv-bindv-model 实现 父子组件通信(数据双向绑定)

v-model可以实现数据字段与DOM节点内容的双向绑定,
也可以实现数据字段与数据字段之间的双向绑定;
v-bind只能是实现单向数据流

若不自己设置承接的字段名,则需要用modelValue作为默认的承接字段名
同时,$emit()的一参默认为update:modelValue,二参为绑定的数据;

如下代码,
子组件 的承接变量modelValue 同父组件的count字段 双向绑定,
(实际上就是v-model的特性 —— 将 子组件的内容即modelValue 同 父组件的数据字段双向绑定)
而后显示在子组件的DOM中({{modelValue}}):

<script>    const app = Vue.createApp({        data() {            return {count: 1}        },        template: `            <counter v-model="count"/>`    });    app.component(&#39;counter&#39;, {        props: [&#39;modelValue&#39;],        methods: {            handleItemClick() {                this.$emit(&#39;update:modelValue&#39;, this.modelValue + 16);                console.log(vm.$data.count);            }        },        template:`        <div @click="handleItemClick">{{modelValue}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion
Copier après la connexion

效果:Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

当然也可以自己设置字段名
这种方式需要给v-model字段接一个字段名,
同时将这个字段名替代子组件中所有modelValue的位置:

<script>    const app = Vue.createApp({        data() {            return {count: 1}        },        template: `            <counter v-model:testField="count"/>`    });    app.component(&#39;counter&#39;, {        props: [&#39;testField&#39;],        methods: {            handleItemClick() {                this.$emit(&#39;update:testField&#39;, this.testField + 16);                console.log(vm.$data.count);            }        },        template:`        <div @click="handleItemClick">{{testField}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

实现效果与上例相同;

结合$emitv-bindv-model 实现 父子组件通信(多个字段的应用案例)

如下代码,
父组件的count与子组件承接的testField字段,
父组件的count1与子组件承接的testField1字段,
分别实现了双向绑定:

<script>    const app = Vue.createApp({        data() {            return {                count: 1,                count1: 1            }        },        template: `            <counter v-model:testField="count" v-model:testField1="count1"/>`    });    app.component(&#39;counter&#39;, {        props: [&#39;testField&#39;,&#39;testField1&#39;],        methods: {            handleItemClick() {                this.$emit(&#39;update:testField&#39;, this.testField + 16);                console.log("vm.$data.count", vm.$data.count);            },            handleItemClick1() {                this.$emit(&#39;update:testField1&#39;, this.testField1 + 8);                console.log("vm.$data.count1", vm.$data.count1);            }        },        template:`        <div @click="handleItemClick">{{testField}}</div>        <div @click="handleItemClick1">{{testField1}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

效果:Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

自己设置修饰符

机制:在父组件调用处,在v-model后 使用自己设置修饰符
实现修饰符逻辑的地方,如点击事件中,
通过this.modelModifiers.[自己设置修饰符名]返回的布尔值
判断客户能否使用了修饰符,
进而分别对使用与否做相应的解决;
另外&#39;modelModifiers&#39;板块中可以指定默认值(下代码指定为一个空对象{});

试验this.modelModifiers的作用

首先下面是一个空的解决,&#39;modelModifiers&#39;板块中指定默认值(下代码指定为一个空对象{}),
mounted函数中打印 子组件modelModifiers属性的内容,

代码如下,
运行后,可以见打印了一个对象{captalize: true}
正是我们传入的自己设置修饰符.captalize(这里未做解决)
【假如这里v-model不接修饰符,
console.log(this.modelModifiers);将打印一个空对象{}】:

<script>    const app = Vue.createApp({        data() {            return {                char: &#39;a&#39;            }        },        template: `            <counter v-model.captalize="char"/>`    });    app.component(&#39;counter&#39;, {        props: {            &#39;modelValue&#39;: String,            &#39;modelModifiers&#39;: {                default: () => ({})            }        },        mounted() {            console.log(this.modelModifiers);        },        methods: {            handleClick() {                this.$emit(&#39;update:modelValue&#39;, this.modelValue + &#39;h&#39;);                console.log("vm.$data.count", vm.$data.char);            }        },        template:`        <div @click="handleClick">{{modelValue}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

下面在子组件的点击回调handleClick()中,通过this.modelModifiers.[自己设置修饰符名]实现自己设置修饰符逻辑

实现效果即 点击之后使得对应的字符串 全变大写;

<script>    const app = Vue.createApp({        data() {            return {                testString: &#39;a&#39;            }        },        template: `            <counter v-model.heheda="testString"/>`    });    app.component(&#39;counter&#39;, {        props: {            &#39;modelValue&#39;: String,            &#39;modelModifiers&#39;: {                default: () => ({})            }        },        mounted() {            console.log(this.modelModifiers);        },        methods: {            handleClick() {                let newValue = this.modelValue + &#39;h&#39;;                if(this.modelModifiers.heheda) {                    newValue = newValue.toUpperCase();                }                this.$emit(&#39;update:modelValue&#39;, newValue);                console.log("vm.$data.count", vm.$data.testString);            }        },        template:`        <div @click="handleClick">{{modelValue}}</div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

效果:

Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

插槽【slot】【传组件示例】

使用关键 主要分两个部分:
自己设置子组件:
在需要 被父组件插入组件的位置,
使用<slot></slot>标签对临时占位;

父组件:
在调用子组件标签对时,
子组件标签对
写上 要替换子组件标签对<slot></slot>位置的组件

【slot】的出现,
方便父子组件之间数据的传递,
方便DOM的传递;

<!DOCTYPE html><html><head>    <meta charset="UTF-8">    <meta name="viewport" content="width=device-width, initial-scale=1.0">    <title>Hello World! heheheheheheda</title>    <script src="https://unpkg.com/vue@next"></script></head><body>    <div id="heheApp"></div></body><script>    const app = Vue.createApp({        template: `            <myform>                <div>提交</div>            </myform>            <myform>                <button>提交</button>            </myform>`    });    app.component(&#39;myform&#39;, {        template:`        <div>            <input />            <slot></slot>            <br><br>        </div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script></html>
Copier après la connexion

运行效果:Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

注意,slot标签上是无法直接增加事件(修饰符)的,如有需要,可以在<slot>外层包裹一层标签,再加上事件

<script>    const app = Vue.createApp({        template: `            <myform>                <div>提交</div>            </myform>            <myform>                <button>提交</button>            </myform>`    });    app.component(&#39;myform&#39;, {        methods: {            handleClick() {                console.log("heheda!!===");            }        },        template:`        <div>            <input />            <span @click="handleClick">                <slot></slot>            </span>                <br><br>        </div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

运行,点击提交文本或者按钮:Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

插槽【传 字符串示例】

<script>    const app = Vue.createApp({        template: `            <myform>                66666            </myform>            <myform>                88888            </myform>`    });    app.component(&#39;myform&#39;, {        methods: {            handleClick() {                console.log("heheda!!===");            }        },        template:`        <div>            <input />            <span @click="handleClick">                <slot></slot>            </span>                <br><br>        </div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

插槽【传 自己设置子组件 示例】

<script>    const app = Vue.createApp({        template: `            <myform>                <test />            </myform>            <myform>                88888            </myform>`    });    app.component(&#39;test&#39;, {        template: `<div>test component</div>`    })    app.component(&#39;myform&#39;, {        methods: {            handleClick() {                console.log("heheda!!===");            }        },        template:`        <div>            <input />            <span @click="handleClick">                <slot></slot>            </span>                <br><br>        </div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

运行:Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

插槽作用域问题

尽管,父组件中 往子组件标签间 插入的组件 会替换子组件的插槽位,
但是父组件中 往子组件标签间 插入的组件,
其所使用的数据字段,依然是父组件的,而非子组件

父组件的template中 调用的数据是 父组件中的 data;
子组件的template中 调用的数据是 子组件中的 data;

<script>    const app = Vue.createApp({        data() {            return {                text: &#39;提交&#39;            }        },        template: `            <myform>                <div>{{text}}</div>            </myform>            <myform>                <button>{{text}}</button>            </myform>`    });    app.component(&#39;myform&#39;, {        methods: {            handleClick() {                console.log("heheda!!===");            }        },        template:`        <div>            <input />            <span @click="handleClick">                <slot></slot>            </span>                <br><br>        </div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

插槽 UI默认值

可以在子组件的插槽<slot>标签间 编写默认值
假如父组件没有使用 组件 注入插槽
则对应位置 会显示默认值

<script>    const app = Vue.createApp({        data() {            return {                text: &#39;提交&#39;            }        },        template: `            <myform>                <div>{{text}}</div>            </myform>            <myform>                <button>{{text}}</button>            </myform>            <myform>            </myform>`    });    app.component(&#39;myform&#39;, {        methods: {            handleClick() {                console.log("heheda!!===");            }        },        template:`        <div>            <input />            <span @click="handleClick">                <slot>default value</slot>            </span>                <br><br>        </div>        `    });    const vm = app.mount(&#39;#heheApp&#39;);</script>
Copier après la connexion

效果:Méthodes de communication entre les composants parents et enfants de Vue3 et liaison bidirectionnelle entre les composants

插槽的灵活拆分与应用【具名插槽】

  • 使得插槽的 父组件注入部分 和 子组件占位部分,能够更加灵活的布局,

    可以通过v-slot:[插槽名]来对一组插槽命名,
    父组件定义之后 插槽名及其对应的组件之后,

    子组件只要要在要占位的地方,
    配合name属性 使用对应命名的<slot>标签,
    就可将对应的父组件插槽组件占用过来;

  • 父组件 的插槽注入部分的组件,
    需要用