Vue3——组件传值 & v-model & 异步组件 & teleport组件

一、父子组件传值

父组件通过props向子组件传值,子组件通过emit触发自定义事件传递新值给父组件。
props:setup函数中第一个参数props用于接收父组件传递进来的参数。注意:props参数中,只会接收props选项中接收的参数 。
context参数:setup函数中的第二个参数是一个上下文对象context。context参数里面有三个对象:attrs,emit,slots。

  • attrs:用于获取没有采用props选项接收的参数。
  • emit:用于触发自定义事件。
  • slots:返回的是插槽里面的虚拟DOM信息。

虚拟DOM:就是vue实例所能识别的一个DOM信息对象。

父组件

<div class="parent">
  <h2>父组件</h2>
  <div>姓名:{{ name }}</div>
  <div>年龄:{{ age }}</div>
  <div>年龄:{{ sex }}</div>
  <Son1 :name="name" :age="age" :sex="sex" @updateData="updateData">
      <!-- 这里是具名插槽 -->
      <template v-slot:top>
          <div class="box">这里顶部</div>
          <div class="box">
              <button>这里顶部</button>
          </div>
      </template>
      <!-- #是v-slot:的简写 -->
      <template #bottom="scope">
          <div class="box">这里底部
              <ul>
                  <li>{{scope.car.name}}</li>
                  <li>{{scope.car.price}}</li>
              </ul>
          </div>
      </template>
  </Son1>
</div>
import {ref} from 'vue'
import Son1 from "./Son1.vue";
export default {
  name: "Parent",
  setup() {
      let name = ref('张三')
      let age = ref(20)
      let sex = ref('男')
      let updateData = (e)=>{   
          name.value = e.myName
          age.value = e.myAge
          sex.value = e.mySex
      }
      return {
          name,
          age,
          sex,
          updateData
      }
  },
  /* data() {
      return {
          name:'张三',
          age:20
      }
  },
  methods: {
      updateData(e){
          this.name = e.myName
          this.age = e.myAge
      }
  }, */
  components: {
    Son1,
  },
};

子组件

<div class="son1">
  <!-- 插槽,定义多个插槽时,需要给插槽定义名称:具名插槽 -->
  <slot name="top"></slot>
  <h2>Son1</h2>
  <div>姓名:{{ myName }}</div>
  <div>年龄:{{ myAge }}</div>
  <div>性别:{{ mySex }}</div>
  <div><button @click="updateData">修改数据</button></div>
  <!-- 可以通过插槽传递一份数据给插槽的使用者,这样的插槽称为作用域插槽 -->
  <slot name="bottom" :car="car"></slot>
</div>
import {ref,reactive} from 'vue';
export default {
  name: "Son1",
  //接收父组件的传值
  props: ["name", "age"],
  // setup的第一个参数,用于获取父组件的传值
  // 注意:props选项接收了几个参数,setup函数的第一个参数就只能获取几个参数。
  // setup的第二个参数,是一个上下文对象;它里面一个方法是emit,用于触发自定义事件
  // props选项没有接收的传值,在setup里面通过上下文对象的attrs属性接收
  setup(props,{emit,attrs,slots}) {
      // slots对象返回的是插槽里面的虚拟DOM信息
      console.log(slots.top());
      // 中转props里面的数据,因为props是只读的
      let myName = ref(props.name)
      let myAge = ref(props.age)
      let mySex = ref(attrs.sex)
      let car = reactive({
        name:'奔驰',
        price:20
      })
      let updateData = ()=>{
          myName.value = '李四'
          myAge.value = 30
          mySex.value = '女'
          // 触发自定义事件
          emit('updateData',{myName:myName.value,myAge:myAge.value,mySex:mySex.value})
      }
      return {
          myName,
          myAge,
          mySex,
          updateData,
          car
      }
  }
  /* data() {
      return {
          myName:this.name,
          myAge:this.age
      }
  },
  methods: {
      updateData(){
          this.myName = '李四'
          this.myAge = 30
          this.$emit('updateData',{myName:this.myName,myAge:this.myAge})
      }
  }, */
};

二、祖孙组件传值

祖级组件通过provide将指定的数据添加为依赖数据,让后代组件可以直接使用。孙代组件通过inject注入祖级组件中设置为依赖的数据。

祖级组件

import {ref,provide} from 'vue'
import Son from './components/Son.vue'
export default {
  name: 'App',
  setup() {
    let name = ref('张三')
    let age = ref(20)
    // 通过provide()方法,定义依赖数据,从此它的子组件,就可以获取这些数据了
    provide('name',name)
    provide('age',age)
    return {
      name,
      age
    }
  },
  components: {
    Son
  }
  /* data() {
    return {
      name:'张三',
      age:20
    }
  },
  methods: {
    // 修改数据的方法
    updateData(name,age){
      this.name = name
      this.age = age
    }
  },
  // 定义依赖数据
  provide(){
    return {
      name:this.name,
      age:this.age,
      updateData:this.updateData
    }
  }, */
}

孙级组件

<div class="subSon">
  <h2>SubSon</h2>
  <ul>
      <li>姓名:{{name}}</li>
      <li>年龄:{{age}}</li>
      <li>
          <button @click="update">修改信息</button>
      </li>
  </ul>
</div>
import {inject} from 'vue'
export default {
  name: "SubSon",
  setup() {
      // inject()方法,用于注入父级中依赖的数据
      let name = inject('name')
      let age = inject('age')
      let update = ()=>{
          name.value = '李四'
          age.value = 30
      }
      return {
          name,
          age,
          update
      }
  }
  /* inject:['name','age','updateData'],
  data() {
      return {
          myName:this.name,
          myAge:this.age
      }
  },
  methods: {
      update(){
          this.myName = '李四'
          this.myAge = 30
          this.updateData('李四',30)
      }
  }, */
};

三、v-model

在Vue3中,父组件中可以通过v-model指令实现对多个数据的双向绑定。注意:vue3取消了sync修饰符,它将v-model指令和sync修饰符进行了合并。
子组件中,自定义事件名称必须命名为update:属性名,就可以实现对父组件中指定属性的双向绑定。

父级组件

<div class="app">
  <h2>App</h2>
  <!-- 在vue3中子定义组件时,v-model可以使用多次,实现对多个数据的双向绑定-->
  <Son3 v-model:planeName="planeName"  v-model:planePrice="planePrice" v-model:planeAddress="planeAddress" />
</div>
import {ref,provide} from 'vue'
import Son3 from './components/Son3.vue'
export default {
  name: 'App',
  setup() {
    //定义飞机的相关数据
    let planeName = ref('波音747')
    let planePrice = ref(100)
    let planeAddress = ref('美国')

    return {
      // 返回飞机相关信息
      planeName,
      planePrice,
      planeAddress
    }
  },
  components: {
    Son3
  }

子级组件

<div class="son3">
  <h2>Son3</h2>
  <ul>
    <li>飞机名称:{{ planeName }}</li>
    <li>飞机价格:{{ planePrice }}</li>
    <li>飞机产地:{{ planeAddress }}</li>
    <li>
      <button @click="updatePlaneName">修改飞机名称</button>
    </li>
    <li>
      <button @click="updatePlanePrice">修改飞机价格</button>
    </li>
    <li>
      <button @click="updatePlaneAddress">修改飞机产地</button>
    </li>
  </ul>
</div>
export default {
  name: "Son3",
  //接收父组件传递过来的数据
  props: ["planeName", "planePrice", "planeAddress"],
  setup(props, { emit }) {
    let updatePlaneName = () => {
      // 注意:事件方法必须是update:prop,如果父组件中采用的是v-model:prop
      // 此时,父组件就可以实现对prop的双向数据绑定。
      emit("update:planeName", "长城1号");
    };
    let updatePlanePrice = () => {
      emit("update:planePrice", 200);
    };
    let updatePlaneAddress = () => {
      emit("update:planeAddress", "中国");
    };
    return {
      updatePlaneName,
      updatePlanePrice,
      updatePlaneAddress,
    };
  },
};

四、异步组件

suspense内置组件:用于在渲染异步组件时,添加Loading效果。
使用 <suspense></suspense> 包裹所有异步组件相关代码。
<suspense></suspense> 下 <template #default></template> 插槽包裹异步组件。
<suspense></suspense> 下 <template #fallback></template> 插槽包裹渲染异步组件之前的内容。
注意:异步加载的组件可以用suspense,也可以不用。不用suspense组件,会失去异步的作用;但是,如果组件中setup的返回值是一个Promise对象,该组件必须要用suspense。

定义组件

<div class="son4">
  <h2>Son4</h2>
  <ul>
    <li>商品名称:{{ goodsName }}</li>
    <li>商品价格:{{ goodsPrice }}</li>
    <li>
        <input type="text" v-model="goodsName">
    </li>
  </ul>
</div>
import { ref } from "vue";
export default {
  name: "Son4",
  setup() {
    let goodsName = ref("小米电视");
    let goodsPrice = ref(2000);
    //setup方法的返回值,可以是一个Promise对象
    return new Promise((resolve,reject)=>{
        setTimeout(() => {
            resolve({
                goodsName,
                goodsPrice,
                show
            })
        }, 2000);
    })

    // return {
    //     goodsName,
    //     goodsPrice,
    // }
  },
};

使用

<div class="app">
  <h2>App</h2>
  <!-- suspense内置组件,用于在渲染异步组件时,添加Loading效果 -->
  <suspense>
    <!-- default插槽里面放置异步组件 -->
    <template #default>
      <Son4/>
    </template>
    <!-- fallback插槽里面放置组件没有加载完成时显示的内容 -->
    <template #fallback>
      Loading...
    </template>
  </suspense>
</div>
// defineAsyncComponent组合式API,用于定义异步组件
import {defineAsyncComponent} from 'vue'
// 异步导入组件
const Son4 = defineAsyncComponent(()=>import('./components/Son4.vue'))

五、teleport组件

Vue 3.0 新增了一个内置组件 teleport ,主要是为了解决以下场景:有时组件模板的一部分逻辑上属于该组件,而从技术角度来看,最好将模板的这一部分移动到 DOM 中 Vue app 之外的其他位置。
teleport组件:瞬移。通过to属性确定里面的元素移动到哪。to属性的属性值是指定的选择器。

例如:在下面的案例中,box盒子逻辑上属于son4盒子,但是为了使box盒子设置的相对定位,不受其他父级元素的影响,我们将box盒子瞬移到body下。

<div class="son4">
  <h2>Son4</h2>
  <button @click="show=true">显示</button>
  <!-- teleport组件:瞬移。通过to属性确定里面的元素移动到哪。
  to属性的属性值是指定的选择器 -->
  <teleport to="body">
      <div v-show="show" class="box">
          <ul>
              <li>商品名称:{{ goodsName }}</li>
              <li>商品价格:{{ goodsPrice }}</li>
          </ul>
          <button @click="show=false">关闭</button>
      </div>
  </teleport>
</div>
.box{
    width: 200px;
    height: 200px;
    border: 1px solid black;
    padding: 10px;
    background: lightblue;
    /* 绝对定位 */
    position: absolute;
    top: 0;
    left: 0;
    right: 0;
    bottom: 0;
    margin: auto;
}

使用teleport瞬移后,box盒子的结构是在body下,与#app平级。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,542评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,596评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,021评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,682评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,792评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,985评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,107评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,845评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,299评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,612评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,747评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,441评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,072评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,828评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,069评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,545评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,658评论 2 350

推荐阅读更多精彩内容