Composition API(三)

2022-05-09  本文已影响0人  Imkata

认识h函数

Vue推荐在绝大数情况下使用模板来创建你的HTML,然而一些特殊的场景,你真的需要JavaScript的完全编程的能力,这个时候你可以使用渲染函数,它比模板更接近编译器。

前面我们讲解过VNode和VDOM的改变,Vue在生成真实的DOM之前,会将我们的节点转换成VNode,而VNode组合在一起形成一颗树结构,就是虚拟DOM(VDOM)。事实上,我们之前编写的 template 中的HTML最终也是使用渲染函数生成对应的VNode,那么,如果你想充分的利用JavaScript的编程能力,我们可以自己来编写 createVNode 函数,生成对应的VNode。

那么我们应该怎么来做呢?使用 h() 函数:
h() 函数是一个用于创建 vnode 的一个函数,其实更准确的命名是createVNode() 函数,但是为了简便在Vue中将之简化为 h() 函数。

可能你会懵逼了,render()函数、h()函数,createVNode()函数什么关系?
render()函数是我们在组件中编写的一个选项,它要求我们返回一个VNode对象,我们利用h()函数来返回一个VNode对象,而h()函数更准确的命名是createVNode()函数。

h()函数如何使用呢?

h()函数如何使用呢?它接受三个参数:

注意:如果没有props,最好将null作为第二个参数传入,将children作为第三个参数传入,以防产生歧义。

h函数的基本使用

h函数可以在两个地方使用:

<script>
  // 先导入h函数
  import { h } from 'vue';

  export default {
    //使用render函数后,上面的template就可以删掉了
    render() {
      // 参数一:h2标签  参数二:带有title的class  参数三:内容是Hello Render
      return h("h2", {class: "title"}, "Hello Render")
    }
  }
</script>

<style scoped>
</style>

h函数计数器案例

<script>
  import { h } from 'vue';

  export default {
    data() {
      return {
        counter: 0
      }
    },
    render() {
      return h("div", {class: "app"}, [
        h("h2", null, `当前计数: ${this.counter}`),
        h("button", {
          onClick: () => this.counter++
        }, "+1"),
        h("button", {
          onClick: () => this.counter--
        }, "-1"),
      ])
    }
  }
</script>

<style scoped>
</style>

如果使用setup,代码如下:

<script>
  import { ref, h } from 'vue';

  export default {
    setup() {
      // setup函数可以代替data选项
      const counter = ref(0);
      
      // setup函数还可以代替render函数
      return () => {
        return h("div", {class: "app"}, [
          h("h2", null, `当前计数: ${counter.value}`),
          h("button", {
            onClick: () => counter.value++
          }, "+1"),
          h("button", {
            onClick: () => counter.value--
          }, "-1"),
        ])
      }
    }
  }
</script>

<style scoped>
</style>

h函数中组件和插槽的使用

前面我们使用h函数第一个参数传的是元素标签的名字,其实还可以传组件。

App.vue组件代码:

<script>
  import { h } from 'vue';
  import HelloWorld from './HelloWorld.vue';

  export default {
    render() {
      return h("div", null, [
        h(HelloWorld, null, {
          default: props => h("span", null, `app传入到HelloWorld中的内容: ${props.name}`)
        })
      ])
    }
  }
</script>

<style scoped>
</style>

HelloWorld.vue组件代码:

<script>
  import { h } from "vue";

  export default {
    render() {
      // HelloWorld组件也是通过render函数创建出来的
      return h("div", null, [
        h("h2", null, "Hello World"),
        this.$slots.default ? this.$slots.default({name: "coderwhy"}): h("span", null, "我是HelloWorld的插槽默认值")
      ])
    }
  }
</script>

<style lang="scss" scoped>
</style>

jsx的babel配置

上面的代码显然难以阅读,如果我们又想充分利用JavaScript的编程能力又不想写上面那些代码,我们可以使用jsx,jsx和React很像。

如果我们希望在项目中使用jsx,那么我们需要添加对jsx的支持。jsx我们通常会通过Babel来进行转换(React编写的jsx就是通过babel转换的),对于Vue来说,我们只需要在Babel中配置对应的插件即可。

安装Babel支持Vue的jsx插件:

npm install @vue/babel-plugin-jsx -D

在babel.config.js配置文件中配置插件:

编写jsx代码:

jsx计数器案例

<script>
  import HelloWorld from './HelloWorld.vue';

  export default {
    data() {
      return {
        counter: 0
      }
    },

    render() {
      const increment = () => this.counter++;
      const decrement = () => this.counter--;

      return (
        <div>
          <h2>当前计数: {this.counter}</h2>
          <button onClick={increment}>+1</button>
          <button onClick={decrement}>-1</button>
          // 还可以使用组件
          <HelloWorld>
          </HelloWorld>
        </div>
      )
    }
  } 
</script>

<style lang="scss" scoped>
</style>

jsx组件的使用

认识自定义指令

在Vue的模板语法中我们学习过各种各样的指令:v-show、v-for、v-model等等,除了使用这些指令之外,Vue也允许我们来自定义自己的指令。

通常在某些情况下,你需要对DOM元素进行底层操作,这个时候就会用到自定义指令。(注意,在Vue中,代码的复用和抽象主要还是通过组件)。

自定义指令分为两种:
自定义局部指令:组件中通过 directives 选项,只能在当前组件中使用;
自定义全局指令:app的 directive 方法,可以在任意组件中被使用;

比如我们来做一个非常简单的案例:当某个元素挂载完成后可以自定获取焦点。
方式一:如果我们使用默认的实现方式;
方式二:自定义一个 v-focus 的局部指令;
方式三:自定义一个 v-focus 的全局指令;

实现方式一:聚焦的默认实现

实现方式二:局部自定义指令

自定义一个 v-focus 的局部指令,这个自定义指令实现非常简单,我们只需要在组件选项中使用 directives 即可,它是一个对象,在对象中编写我们自定义指令的名称(注意:这里不需要加v-),自定义指令有一个生命周期,是在组件挂载后调用 mounted,我们可以在其中完成操作。

<template>
  <div>
    <input type="text" v-focus>
  </div>
</template>

<script>
  export default {
    // 局部指令
    directives: {
      focus: {
        // 参数一:当前el元素
        // 参数二:绑定的一个对象,传入的参数和修饰符就在这里面
        // 参数三:虚拟节点
        // 参数四:上一个虚拟节点
        mounted(el, bindings, vnode, preVnode) {
          //console.log("focus mounted");
          el.focus();
        }
      }
    }
  }
</script>

<style scoped>
</style>

方式三:自定义全局指令

自定义一个全局的v-focus指令可以让我们在任何地方直接使用。

import { createApp } from 'vue'

const app = createApp(App);

app.directive("focus", {
  mounted(el, bindings, vnode, preVnode) {
    //console.log("focus mounted");
    el.focus();
  }
})

app.mount('#app');

指令的生命周期、参数、修饰符

一个指令定义的对象,Vue提供了如下几个钩子函数,在每个钩子函数中都有四个参数:el, bindings, vnode, preVnode 详情如下:

created:在绑定元素的 attribute 或事件监听器被应用之前调用;
beforeMount:当指令第一次绑定到元素并且在挂载父组件之前调用;
mounted:在绑定元素的父组件被挂载后调用;
beforeUpdate:在更新包含组件的 VNode 之前调用;
updated:在包含组件的 VNode 及其子组件的 VNode 更新后调用;
beforeUnmount:在卸载绑定元素的父组件之前调用;
unmounted:当指令与元素解除绑定且父组件已卸载时,只调用一次;

<template>
  <div>
    <button v-if="counter < 2" v-why.aaaa.bbbb="'coderwhy'" @click="increment">当前计数: {{counter}}</button>
  </div>
</template>

<script>
  import { ref } from "vue";

  export default {
    // 指令生命周期
    directives: {
      why: {
        // 参数一:当前el元素
        // 参数二:绑定的一个对象,传入的参数和修饰符就在这里面
        // 参数三:虚拟节点
        // 参数四:上一个虚拟节点
        created(el, bindings, vnode, preVnode) {
          console.log("why created", el, bindings, vnode, preVnode);
          //获取传递的参数,也就是'coderwhy'
          console.log(bindings.value);
          //获取修饰符,modifiers是个对象 {aaa: true, bbb: true}
          console.log(bindings.modifiers);
        },
        beforeMount() {
          console.log("why beforeMount");
        },
        mounted() {
          console.log("why mounted");
        },
        beforeUpdate() {
          console.log("why beforeUpdate");
        },
        updated() {
          console.log("why updated");
        },
        beforeUnmount() {
          console.log("why beforeUnmount");
        },
        unmounted() {
          console.log("why unmounted");
        }
      }
    },
    setup() {
      const counter = ref(0);
      const increment = () => counter.value++;

      return {
        counter,
        increment
      }
    }
  }
</script>

<style scoped>
</style>

自定义指令练习:时间戳的显示需求

在开发中,大多数情况下从服务器获取到的都是时间戳,我们需要将时间戳转换成具体格式化的时间来展示。在Vue2中我们可以通过过滤器来完成,Vue3中删除了过滤器,在Vue3中我们可以通过计算属性(computed)或者自定义一个方法(methods)来完成,其实我们还可以通过一个自定义的指令来完成。

我们来实现一个可以自动对时间格式化的指令v-format-time,这里我封装了一个函数,在首页中我们只需要调用这个函数并且传入app即可。

我们创建directives文件夹,format-time.js是每个指令的具体代码,index.js是注册每个指令,这样我们在main.js中直接导入,然后传入app即可,文件目录如下:

format-time.js代码如下:

// 需要通过npm install dayjs
import dayjs from 'dayjs';

export default function(app) {
  app.directive("format-time", {
    // 1. 先设置时间格式参数
    created(el, bindings) {
      // 设置默认的时间格式
      // 这里我们把值设置到bindings上,目的就是为了在bindings里面能取到
      // 如果我们不设置到bindings上就没法传给mounted了
      bindings.formatString = "YYYY-MM-DD HH:mm:ss";
      //如果自定义指令传递了时间格式参数,就使用传递的参数
      if (bindings.value) {
        bindings.formatString = bindings.value;
      }
    },
    // 2. 再转换
    mounted(el, bindings) {
      //获取显示的文字
      const textContent = el.textContent;
      //转成int
      let timestamp = parseInt(textContent);
      //如果10位就是秒,转成毫秒
      if (textContent.length === 10) {
        timestamp = timestamp * 1000
      }
      //format后面的参数使用传递的参数
      el.textContent = dayjs(timestamp).format(bindings.formatString);
    }
  })
}

index.js文件代码如下:

import registerFormatTime from './format-time';

export default function registerDirectives(app) {
  registerFormatTime(app);
}

main.js文件代码如下:

import { createApp } from 'vue'
import App from './03_自定义指令/App.vue'
import registerDirectives from './directives'

const app = createApp(App);

registerDirectives(app);

app.mount('#app');

这样我们就将main.js里面注册全局组件的代码抽出来了,下次如果需要注册新的全局组件,只需在index.js里面添加一行代码即可。

认识Teleport

在组件化开发中,我们封装一个组件A,在另外一个组件B中使用,那么组件A中template的元素,会被挂载到组件B中template的某个位置,最终我们的应用程序会形成一颗DOM树结构。但是某些情况下,我们希望组件不是挂载在这个组件树上的,可能是移动到Vue app之外的其他位置,比如移动到body元素上,或者我们有其他的div#app之外的元素上,这个时候我们就可以通过teleport来完成。

Teleport是什么呢?
它是一个Vue提供的内置组件,类似于react的Portals,teleport翻译过来是心灵传输、远距离运输的意思,它有两个属性:

我们来看下面代码的效果:

和组件结合使用

当然,teleport也可以和组件结合一起来使用,我们可以在 teleport 中使用组件,并且也可以给他传入一些数据。

多个teleport

如果我们将多个teleport应用到同一个目标上(to的值相同),那么这些目标会进行合并:

实现效果如下:

认识Vue插件

通常我们向Vue全局添加一些功能时,会采用插件的模式,它有两种编写方式:
① 对象类型:一个对象,但是必须包含一个 install 的函数,该函数会在安装插件时执行;
② 函数类型:一个function,这个函数会在安装插件时自动执行;

插件可以完成的功能没有限制,比如下面的几种都是可以的:

插件的编写方式

对象类型的写法,plugins_object.js文件代码:

export default {
  install(app) {
    // 全局配置添加一个name
    app.config.globalProperties.$name = "coderwhy"
  }
}

函数类型的写法,plugins_function.js文件代码:

export default function(app) {
  console.log(app);
}

安装插件:

import { createApp } from 'vue'
import App from './03_自定义指令/App.vue'

const app = createApp(App);

//安装插件,其实内部就是调用install方法
app.use(pluginObject);
app.use(pluginFunction);

app.mount('#app');

使用插件:

如果在Vue2中,直接使用this.$xxx就可以拿到全局配置的属性了:

mounted() {
  console.log(this.$name);
},

如果在Vue3中,因为在setup中我们不能使用this,所以我们可以通过如下方式获取this.$xxx:

import { getCurrentInstance } from "vue";

setup() {
  const instance = getCurrentInstance();
  console.log(instance.appContext.config.globalProperties.$name);
},
上一篇 下一篇

猜你喜欢

热点阅读