第十章:动态组件,插槽,自定义指令

简介: 第十章:动态组件,插槽,自定义指令

第十章:动态组件,插槽,自定义指令

本章任务

  1. 动态组件
  2. 插槽
  3. 自定义指令
  4. vant UI 组件库
  5. axios的配置

本章内容

  • 动态组件
  • 自定义指令

一、动态组件

1.1 什么是动态组件

概念:动态组件就是可以动态改变的组件,之前我们引入组件,都是把组件固定写在某一个位置显示,没有办法切换组件显示,动态组件可以帮我们完成这个目的。vue中提供了一个组件来动态的完成组件的切换, 不需要我们自己去封装。

1.2 动态组件的使用

  1. 父组件中导入并注册子组件
//导入需要动态切换的 两个子组件 
  import com01 from '../components/Com01.vue'
  import com02 from '../components/Com02.vue'
  export default{
    name:"about",
        //注册子组件
    components:{
      com01,com02
    },
  }
  1. 在父组件中通过内置组件的:is属性,使用子组件
直接写is相当于直接写死 一个组件的名称,可以使用v-bind 的:is属性来完成动态的组件切换效果。
<template>
  <div class="about">
    <h1>This is an about page</h1>
  <button  @click="changeCom()">切换组件</button>
  <component :is="comName"></component>
  </div>
</template>

存在问题:此时虽然可以动态切换组件,但是存在问题,组件的切换其实就是组件的销毁并重新创建,此时会导致再次切换回来无法保证原来的组件中的数据信息。可以通过生命周期函数验证。

com01.vue组件

<template>
  <div class="box">
    <h1>这里是组件01</h1>
    <p>{{num}}</p>
    <button type="button" @click="add">点击增加</button>
  </div>
</template>
<script>
  export default{
    data(){
      return{
        num:0
      }
    },
        methods:{
      add(){
        this.num+=1;
      }
    }
  }
</script>
<style lang="less" scoped="scoped">
  h1{
    background-color: blue;
    color: white;
  }
</style>

com02.vue组件:

<template>
  <div class="box">
    <h1>这里是组件02</h1>
    <p>{{num}}</p>
    <button type="button" @click="add">点击增加</button>
  </div>
</template>
<script>
  export default{
    data(){
      return{
        num:0
      }
    },
        methods:{
      add(){
        this.num+=1;
      }
    }
  }
</script>
<style lang="less" scoped="scoped">
  h1{
    background-color: red;
    color: white;
  }
</style>

about.vue父组件:

<template>
  <div class="about">
    <h1>This is an about page</h1>
  <button  @click="changeCom()">切换组件</button>
  <component :is="comName"></component>
  </div>
</template>
<script>
  import com01 from '../components/Com01.vue'
  import com02 from '../components/Com02.vue'
  export default{
    name:"about",
    components:{
      com01,com02
    },
    data(){
      return {
        comName:'com01',
        comList:['com01','com02']
      }
    },
    methods:{
      changeCom(){
        let i = this.comList.findIndex(item=>{
          return item == this.comName
        })
        if(i==0){
          //console.log(this.comList[1])
          this.comName = this.comList[1]
        }
        if(i==1){
          this.comName = this.comList[0]
        }
      }
    }
  }
</script>
<style type="text/css">
</style>

1.3 使用 keep-alive 解决上述问题

keep-alive可以保证在组件切换时 不被销毁,处于缓存状态,从而保证数据的完整性。

代码实现:使用包括起来就可以了。

<keep-alive>
    <component :is="comName"></component>
  </keep-alive>

1.4 keep-alive的生命周期函数和属性props

1.4.1 生命周期函数

keep-alive缓存组件有两个生命周期钩子函数,分别是:

  • activated 组件被激活时调用
  • deactivated 组件被缓存时调用
<template>
  <div class="box">
    <h1>这里是组件01</h1>
    <p>{{num}}</p>
    <button type="button" @click="add">点击增加</button>
  </div>
</template>
<script>
  export default{
    data(){
      return{
        num:0
      }
    },
    methods:{
      add(){
        this.num+=1;
      }
    },
        //定义keep-alive组件的生命周期函数
    activated() {
      console.log('com01组件被激活了');
    },
    deactivated() {
      console.log('com01组件被缓存了');
    }
  }
</script>
<style lang="less" scoped="scoped">
  h1{
    background-color: blue;
    color: white;
  }
</style>
1.4.2 keep-alive属性
  • include - 字符串或正则表达式。只有名称匹配的组件会被缓存。
  • exclude - 字符串或正则表达式。任何名称匹配的组件都不会被缓存。
  • max - 数字。最多可以缓存多少组件实例。最多可以缓存多少组件实例。一旦这个数字达到了,在新实例被创建之前,已缓存组件中最久没有被访问的实例会被销毁掉。
<keep-alive include="com01,com02">
    <component :is="comName"></component>
  </keep-alive>

**注意:**include和exclude两个属性不能同时使用。

二、插槽 slot

2.1 slot简介 v-slot

slot是插槽(槽口)的意思,其目的在于让组件的 可扩展性 更强,用来混合父组件的内容与子组件自己的模板

#例如:需要在组件中插入一些内容
<alert-box>
  Something bad happened.
  ul
  li
</alert-box>
#定义组件模版时
<template id="ab">
    <div class="demo-alert-box">
      //调用组件时,插入的内容会显示在模版中的这个位置
      <slot></slot>
    </div>   
</template>
Vue.component('alert-box', {
  template:'#ab'
})

2.2 使用场景和分类

组件化开发中,虽然组件是一样的,但是在不同的使用场景组件的一部分内容需要有不同的内容显示

2.2.1 分类
  • 匿名slot
  • 具名slot
2.2.2 匿名插槽

匿名插槽:从字面意思来理解是就是没有名字的插槽,特点就是可以放任何你放的东西。

**注意:**此时的插槽不是真的没有名字,而是名字默认为 ‘default’ ====>

<template id="mycomponent">
    <div>
        <p>头部区域</p>
        <slot>如果没有分发内容,则显示默认提示</slot>
        <p>底部区域</p>
    </div>
</template>
<mycomponent>
    <span>可以防止任何内容</span>
</mycomponent>
2.2.3 具名插槽

假设我们的电脑主板上的各种插槽,有插CPU的,有插显卡的,有插内存的,有插硬盘的,所以假设有个组件是computer,我们不可能把显卡插到内存的位置上,具名slot也就是每个slot都有名字,不能随意替换,对应插入

vue2.6之前的版本:
  <div id="app">
        <first>
            <!-- 这里的内容就相当于插头 slot就连接组件中的插槽 -->
            <ul slot="usb">
                <li>第1个li标签</li>
                <li>第2个li标签</li>
                <li>第3个li标签</li>
            </ul>
            <ul>
                <li>你好嘿嘿额1</li>
                <li>你好嘿嘿额2</li>
            </ul>
        </first>
    </div>
在组件模板中定义slot有名字的插槽 但是在使用时 也要带上插槽的名字。
    <template id="tem">
        <div>
            <p>你好哈哈哈哈</p>
            <!-- 定义插槽 name属性定义插槽的名字 -->
            <slot name="usb"></slot>
            <slot></slot>
        </div>
    </template>
vue2.6之后的版本:
<!--子组件模版-->
<template id="mycomponent">
    <div class="container">
        <header>
            <slot name="header"></slot>
        </header>
        <main>
            <slot></slot>
        </main>
        <footer>
            <slot name="footer"></slot>
        </footer>
    </div>
</template>
<!--调用子组件 向具名插槽提供内容的时候,我们可以在一个 <template> 元素上使用 v-slot 指令,并以 v-slot 的参数的形式提供其名称:
缩写:v-slot:header  ===     #header
-->
<mycomponent>
    <template v-slot:header>
        <h1>Here might be a page title</h1>
    </template>
    <p>A paragraph for the main content.</p>
    <p>And another one.</p>
  v-slot的简写形式就是#
    <template #footer>
        <p>Here's some contact info</p>
    </template>
</mycomponent>
2.2.4 作用域插槽:借助插槽的子组件向父组件传值:

**场景:**子组件的数据需要到父组件,父组件使用子组件的数据进行一些列运算 再发送给子组件。

如果使用父子组件的通信:需要借助 自定义事件监听,props[]做数据传递,麻烦。

此时可以使用插槽来实现 子向父 再到 父向子做数据传递。

#com01.vue子组件中
<template>
  <div class="box">
    <h1>
      这里是组件01
      <slot name="aaa" msg='你好哈哈哈'></slot>
    </h1>
  </div>
</template>
#About.vue父组件中
<template >
  <!--使用子组件-->
  <com01>
      <template v-slot:aaa='obj'>
      这是插槽插入的内容
      {{obj.msg}}
    </template>
    </com01>      
</template>
观察发现,在com01.vue子组件中的插槽内 定义了属性msg='你好哈哈哈',在about.vue父组件中的<template>标签中通过v-slot:aaa='obj'指令中 可以通过obj来获取到 msg的属性的数据。同理,此时的msg是直接写死的内容,可以通过:msg把这个数据变成动态传递的。

**使用场景:**作用域插槽一般在封装组件时比较常用,在封装好的组件中,通过插槽预留数据位置,在使用组件时,只需要传入需要的数据就可以了。

课下改版购物车:把加减按钮单独提取成一个组件。

vant:组件库https://vant-contrib.gitee.io/vant/#/zh-CN

三、自定义指令

3.1 什么是自定义指令

概念: 在vue中内部提供的有很多v-系列的指令,例如 v-if v-model 等,都有特定的作用,但是如果我们有一些特殊的需求时,vue也支持我们可以自定义指令来完成一些功能的设置。

我们自己封装一个函数,给这个函数链接一个对应的名字 把这个名字当作指令来使用。

**分类:**主要有两种

  • 全局自定义指令
  • 局部(私有)自定义指令

3.2 为什么要用自定义指令

问题一:vue的内置指令干了什么?

就是DOM操作,vue做DOM操作的方式 跟我们之前学习 原生js 自己写的代码是一样的。

问题二:自定义指令是干啥的?

让我们自己操作DOM,按照我们自己的需求 就跟我们当年写原生js一样。 借助函数

vue的指令底层实际上就是调用函数,通过v-系列的指令调用对应的函数。

3.3 自定义指令的使用

完成案例:当输入框加载完成时,让输入框得到焦点。

全局自定义指令:

// 注册一个全局自定义指令 `v-focus`
Vue.directive('focus', {
  // 当被绑定的元素插入到 DOM 中时……
  inserted: function (el) {
    // 聚焦元素
    el.focus()
  }
})

局部自定义指令:

//所有的局部自定义指令放在这里。 只能再当前组件内部使用这个指令
directives: {
  focus: {
    // 指令的定义
    inserted: function (el) {
      el.focus()
    }
  }
}
指令的钩子函数以及参数对象:

一个指令定义对象可以提供如下几个钩子函数 (均为可选):

  • bind:只调用一次,指令第一次绑定到元素时调用。在这里可以进行一次性的初始化设置。
  • inserted:被绑定元素插入父节点时调用 (仅保证父节点存在,但不一定已被插入文档中)。
  • update:所在组件的 VNode 更新时调用,但是可能发生在其子 VNode 更新之前。指令的值可能发生了改变,也可能没有。但是你可以通过比较更新前后的值来忽略不必要的模板更新 (详细的钩子函数参数见下)。
    当数据发生更新时 就会调用。

在使用自定义指令时 如果我们想传递一些参数使用,此时在自定义指令的bind(el,binding)函数中可以进行参数接收

el:指令所绑定的元素,可以用来直接操作 DOM。

binding:一个对象,包含以下 property:

  • name:指令名,不包括 v- 前缀。 当有多个单词组成指令名字时,推荐使用‘ - ’ 连接多个单词
  • value:指令的绑定值,例如:v-my-directive="1 + 1" 中,绑定值为 2
  • oldValue:指令绑定的前一个值,仅在 updatecomponentUpdated 钩子中可用。无论值是否改变都可用。
  • expression:字符串形式的指令表达式。例如 v-my-directive="1 + 1" 中,表达式为 "1 + 1"
  • arg:传给指令的参数,可选。例如 v-my-directive:foo 中,参数为 "foo"
  • modifiers:一个包含修饰符的对象。例如:v-my-directive.foo.bar 中,修饰符对象为 { foo: true, bar: true }
  1. 对象式自定义指令

在vue实例中添加节点 directives 此节点专门用来定义私有自定义指令:

<template>
  <div class="box">
    <h1 v-aaa>
      这里是组件01
    </h1>
  </div>
</template>
//在directives属性下  可以定义私有自定义指令
    directives:{
      //此处的aaa就是自定义指令的名字
      aaa:{
        //当指令绑定到标签上时  会立即触发 bind事件
        bind(el){
          el.style.aaa = "green"
        }
      }
    }
注意:在上边代码中 aaa就是指令的名字  调用时 要加上v-   v-aaa就是此指令的全名
  在指令对象中要添加一个bind()函数  只要把指令添加到标签上,那么就可以自动触发 这个bind函数,括号中的el就是触发当前指令的元素DOM对象。

传递参数的自定义指令使用

<template>
  <div class="box">
    <h1 v-aaa="'blue'">
      这里是组件01
    </h1>
  </div>
</template>
//在directives属性下  可以定义私有自定义指令
    directives:{
      //此处的color就是自定义指令的名字
      color:{
        //当指令绑定到标签上时  会立即触发 bind事件
        //可以直接从binding的value属性中取到 指令的数据
        bind(el,binding){
          el.style.color = binding.value
        }
      }
    }

此时存在问题: 如果我们不是通过指令直接赋值,而是通过组件中的data数据来给指令进行赋值,此时,如果我们在指令绑定之后,再次修改了该data数据的值,此时 这个结果不会同步更新到DOM结构中。

使用update()方法 解决这个问题,该方法会在DOM结构更新时调用一次,就可以保证指令中的数据发生更新后 也可以把数据的结果同步更新到DOM结构中。

<template>
  <div class="box">
    <h1 v-color='yanse'>
      这里是组件01
      <button type="button" @click="yanse='pink'">点击按钮改变颜色</button>
    </h1>
  </div>
</template>
<script>
  export default{
    data(){
      return{
        num:0,
        yanse:'red'
      }
    },
    //在directives属性下  可以定义私有自定义指令
    directives:{
      //此处的color就是自定义指令的名字
      color:{
        //当指令绑定到标签上时  会立即触发 bind事件
        bind(el,binding){
          el.style.color = binding.value
          console.log('触发了bind函数')
        },
        update(el,binding){
          el.style.color = binding.value
          console.log('触发了update函数')
        }
      },
    }
  }
</script>
<style lang="less" scoped="scoped">
  h1{
    background-color: blue;
    color: white;
  }
</style>
  1. 函数式自定义指令

在很多时候,你可能想在 bindupdate 时触发相同行为,而不关心其它的钩子。比如这样写:

//全局指令中
Vue.directive('color-swatch', function (el, binding) {
  el.style.backgroundColor = binding.value
})
//局部指令中
<script>
  export default{
    data(){
      return{
        num:0,
        yanse:'red'
      }
    },
    //在directives属性下  可以定义私有自定义指令
    directives:{
      //此处的color就是自定义指令的名字
      color(el,binding){
        el.style.color = binding.value
      },
    }
  }
</script>
此时我们的bind和update函数触发的行为是完全一样的 我们可以把指令对象简写成一个函数使用。

注意:

  • 指令的名字 如果是多个单词 推荐使用 ‘ - ’ 连接。
  • 指令中的内置函数中的this指向window对象。

3.4 全局自定义指令

注册一个全局自定义指令 Vue.directive(参数一,参数二)
参数一:指的是指令的名字
参数二:指的是指令的配置  可以是对象,也可以是函数
Vue.directive('focus', {
  // 当被绑定的元素插入到 DOM 中时……
  inserted: function (el) {
    // 聚焦元素
    el.focus()
  }
})

四、项目中axios的相关配置

**问题一:**在以后的vue-cli项目开发中我们很多时候都会发送大量的请求,按照以往的方式 我们会在每一个用到axios的vue文件中导入axios文件,这样比较麻烦 所以我们可以把axios挂载到vue 实例的原型对象上。

main.js文件:

//先在main.js文件中导入axios模块
import axios from 'axios'
//把axios模块挂载到vue的原型对象上,这样只要是vue的实例对象就都可以使用这个axios模块了,调用的时候 直接使用
//this.$http.get()  此时的$http<====>axios
Vue.prototype.$http = axios

问题二: 重复的导入axios是一个问题,其次每一次发送请求时 都要填写完整的路径也是一个问题,不利于后期的维护,所以我们可以在挂载axios之前 先对axios进行全局的路径配置:

//1.先在main.js文件中导入axios模块
import axios from 'axios'
//2.挂载之前先配置axios的BaseUrl路径
axios.defaults.baseURL = 'http://127.0.0.1:8080/'(请求的根路径)
//3.把axios模块挂载到vue的原型对象上,这样只要是vue的实例对象就都可以使用这个axios模块了,调用的时候 直接使用
//this.$http.get()  此时的$http<====>axios
Vue.prototype.$http = axios

五、总结与作业

  • 移动端页面头部组件的封装:

s挂载到vue 实例的原型对象上。

main.js文件:

//先在main.js文件中导入axios模块
import axios from 'axios'
//把axios模块挂载到vue的原型对象上,这样只要是vue的实例对象就都可以使用这个axios模块了,调用的时候 直接使用
//this.$http.get()  此时的$http<====>axios
Vue.prototype.$http = axios

问题二: 重复的导入axios是一个问题,其次每一次发送请求时 都要填写完整的路径也是一个问题,不利于后期的维护,所以我们可以在挂载axios之前 先对axios进行全局的路径配置:

//1.先在main.js文件中导入axios模块
import axios from 'axios'
//2.挂载之前先配置axios的BaseUrl路径
axios.defaults.baseURL = 'http://127.0.0.1:8080/'(请求的根路径)
//3.把axios模块挂载到vue的原型对象上,这样只要是vue的实例对象就都可以使用这个axios模块了,调用的时候 直接使用
//this.$http.get()  此时的$http<====>axios
Vue.prototype.$http = axios

五、总结与作业

  • 移动端页面头部组件的封装:
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ULLtc3kJ-1655272887374)(assets/image-20220216150337599.png)]


目录
相关文章
|
8月前
|
设计模式 JavaScript 开发者
深度解析Vue中的插槽机制:打开组件设计的无限可能
深度解析Vue中的插槽机制:打开组件设计的无限可能
145 1
|
3月前
|
JavaScript 前端开发
VUE学习三:双向绑定指令(v-mode)、组件化开发(全局组件/局部组卷/组件通信)、组件化高级(slot插槽使用)
这篇文章是关于Vue.js框架中的v-model指令和组件化开发的详细教程,涵盖了从基础使用到高级功能的多个方面。
40 1
|
3月前
|
JavaScript 搜索推荐
Vue 插槽全攻略:重塑组件灵活性
【10月更文挑战第7天】 Vue 的插槽(Slots)是一个强大的特性,用于增强组件的灵活性和可扩展性。插槽允许父组件向子组件传递内容,实现组件的复用和个性化展示。主要包括默认插槽、具名插槽和作用域插槽三种类型,分别适用于不同场景。通过插槽,可以提高组件的复用性、实现灵活的布局,并促进团队协作。
38 1
|
8月前
|
前端开发 JavaScript 安全
【亮剑】探讨了在React TypeScript应用中如何通过道具(props)传递CSS样式,以实现模块化、主题化和动态样式
【4月更文挑战第30天】本文探讨了在React TypeScript应用中如何通过道具(props)传递CSS样式,以实现模块化、主题化和动态样式。文章分为三部分:首先解释了样式传递的必要性,包括模块化、主题化和动态样式以及TypeScript集成。接着介绍了内联样式的基本用法和最佳实践,展示了一个使用内联样式自定义按钮颜色的例子。最后,讨论了使用CSS模块和TypeScript接口处理复杂样式的方案,强调了它们在组织和重用样式方面的优势。结合TypeScript,确保了样式的正确性和可维护性,为开发者提供了灵活的样式管理策略。
86 0
|
8月前
|
JavaScript
Vue 插槽:让你的组件更具扩展性(下)
Vue 插槽:让你的组件更具扩展性(下)
Vue 插槽:让你的组件更具扩展性(下)
|
8月前
|
JavaScript 开发者 UED
Vue 插槽:让你的组件更具扩展性(上)
Vue 插槽:让你的组件更具扩展性(上)
Vue 插槽:让你的组件更具扩展性(上)
|
前端开发
前端学习笔记202305学习笔记第二十天-vue3.0-了解作用域插槽的作用2
前端学习笔记202305学习笔记第二十天-vue3.0-了解作用域插槽的作用2
54 0
|
前端开发
前端学习笔记202305学习笔记第二十天-vue3.0-了解作用域插槽的作用1
前端学习笔记202305学习笔记第二十天-vue3.0-了解作用域插槽的作用1
48 0
【Vue3 第十三章】动态组件 & 递归组件 & 组件别名
【Vue3 第十三章】动态组件 & 递归组件 & 组件别名
268 0
|
JavaScript
【Vue3 第十六章】非父子组件间传值
【Vue3 第十六章】非父子组件间传值
302 0