vue-create-api 让 Vue 组件通过 API 方式调用的插件

发布于 2021-07-15 22:41:28 字数 7704 浏览 2244 评论 0

安装

通过 npm

$ npm install vue-create-api

通过 cdn

<script src="https://unpkg.com/vue-create-api/dist/vue-create-api.min.js"></script>

使用

import CreateAPI from 'vue-create-api'

Vue.use(CreateAPI)

// 也可以传递一个配置项

Vue.use(CreateAPI, {
  componentPrefix: 'cube-'
  apiPrefix: '$create-'
})

// 之后会在 Vue 构造器下添加一个 createAPI 方法

import Dialog from './components/dialog.vue'

// 调用 createAPI 生成对应 API,并挂载到 Vue.prototype 和 Dialog 对象上

Vue.createAPI(Dialog, true)

// 之后便可以在普通的 js 文件中使用,但是 $props 不具有响应式

Dialog.$create({
  $props: {
    title: 'Hello',
    content: 'I am from pure JS'
  }
}).show()

// 在 vue 组件中可以通过 this 调用

this.$createDialog({
  $props: {
    title: 'Hello',
    content: 'I am from a vue component'
  },
}).show()
// typescript 使用方式
import CreateAPI from 'vue-create-api'

Vue.use(CreateAPI)

Vue.createAPI(Dialog, events, single)

this.$createDialog({
  $props: {
    title: 'Hello',
    content: 'I am from a vue component'
  }
}).show()
// 自定义 d.ts 扩展通过api创建的方法
import Vue, { VueConstructor } from 'vue'
import { createFunction } from 'vue-create-api';

export declare class UIComponent extends Vue {
  show ():void
  hide ():void
}

declare module 'vue/types/vue' {
  interface Vue {
    /** create Dialog instance */
    $createDialog: createFunction<UIComponent>
  }
}

提示

使用 typescript 时,terser-webpack-plugin(vue-cli3.x)或 uglifyjs(vue-cli2.x)增加 { keep_fnames: true }

构造器配置项

键名描述默认值
componentPrefix组件名前缀,最终生成的 API 会忽略该前缀-
apiPrefix为生成的 API 添加统一前缀$create

方法

Vue.createAPI(Component, [single])

参数

{Function | Object} Component Vue 组件必须要有组件名 name

{Boolean} [single] 是否采用单例模式实例化组件

使用

调用该方法会在 Vue.prototype 上添加名为 $create{camelize(Component.name)} 的方法, 之后在其他 vue 组件中可以通过 const instance = this.$createAaBb(config, [renderFn, single]) 实例化该组件。组件实例化后对应的 DOM 元素会添加到 body 中。

const instance = this.$createAaBb(config, renderFn, single) 参数:

名称描述类型可选值默认值
config配置参数Object{}-
renderFn可选参数,用于生成子 VNode 节点,通常用于处理插槽Function-function (createElement) {...}
single可选参数, 决定实例化是否采用单例模式。在没有传递 renderFn 时,可以直接作为第二个参数传入。Booleantrue/false调用 createAPI 时传入的 single 值

配置项  config

你可以在 config 中配置 $props$events, $props 中的属性会被 watch,从而支持响应式更新.

属性描述类型可选值默认值
$props组件的 PropObject-{
title: 'title',
content: 'my content',
open: false
}
$events组件的事件回调Object-{
click: 'clickHandler',
select: this.selectHandler
}

$props 示例, { [key]: [propKey] }:

{
  title: 'title',
  content: 'my content',
  open: false
}

title, contentopen 是传递给组件的 Prop 键名, 而 Prop 对应的值采用下面的步骤得到:

  1. 如果 propKey 不是一个字符串, 则直接取 propKey 作为该 Prop 值。
  2. 如果 propKey 是一个字符串,但该字符串并没有作为属性名存在于调用 $createAaBb() 的组件中,则直接取 propKey 这个字符串作为该 Prop 值。
  3. 如果 propKey 是一个字符串,且作为属性名存在于调用 $createAaBb() 的组件中, 则会取该实例对应的属性值作为该 Prop 值。 同时会 watch 该属性,做到响应式更新。

$events 示例,{ [eventName]: [eventValue] }

{
  click: 'clickHandler',
  select: this.selectHandler
}

clickselect 是事件名, 同时对应的事件回调会通过下面的步骤确定:

  1. 如果 eventValue 不是一个字符串, 那么直接取 eventValue 作为事件回调.
  2. 如果 eventValue 是一个字符串, 那么会取调用 $createAaBb 的组件中以 eventValue 作为属性名的值,当做事件回调.

同时,config 中可以设置 Vue 支持的所有的配置值,但是必须要加 $。比如:

this.$createAaBb({
  $attrs: {
    id: 'id'
  },
  $class: {
    'my-class': true
  }
})

返回值 instance:

instance 是一个实例化的 Vue 组件。

实例上会包含一个 remove 方法。你可以调用 remove 方法去销毁该组件,同时原本添加到 body 下的 DOM 元素也会删除。

如果调用 $createAaBb 的组件销毁了,那么该组件也会自动销毁。

示例:首先我们先创建一个 Hello.vue 组件:

<template>
  <div @click="clickHandler">
    {{content}}
    <slot name="other"></slot>
  </div>
</template>

<script type="text/ecmascript-6">
  export default {
    name: 'hello',
    props: {
      content: {
        type: String,
        default: 'Hello'
      }
    },
    methods: {
      clickHandler(e) {
        this.$emit('click', e)
      }
    }
  }
</script>

然后我们通过调用 createAPI,得到一个可以命令式创建该组件的 API 。

  import Vue from 'vue'
  import Hello from './Hello.vue'
  import CreateAPI from 'vue-create-api'
  Vue.use(CreateAPI)

  // 得到 this.$createHello API,它会添加到 Vue 原型上
  Vue.createAPI(Hello, true)

  // 实例化 Vue
  new Vue({
    el: '#app',
    render: function (h) {
      return h('button', {
        on: {
          click: this.showHello
        }
      }, ['Show Hello'])
    },
    methods: {
      showHello() {
        const instance = this.$createHello({
          $props: {
            content: 'My Hello Content',
          },
          $events: {
            click() {
              console.log('Hello component clicked.')
              instance.remove()
            }
          }
        }, /* renderFn */ (createElement) => {
          return [
            createElement('p', {
              slot: 'other'
            }, 'other content')
          ]
        })
      }
    }
  })

在该示例中,我们创建了一个 Hello 组件,利用 createAPI() 你可以在其他组件中调用 API 去创建该组件。可以看到,在 showHello() 方法中,通过 this.$createHello(config, renderFn) 可以实例化 Hello 组件。

如何在普通 js 文件中或者全局调用

由于使用 createAPI() 生成实例化组件的 API 时,会将该 API 添加到 Vue 原型上,因此在 Vue 实例中,可以直接通过 this.$createHello(config, renderFn) 创建组件。而如果在普通 JS 中,可以通过组件自身的 $create 来进行实例化了,因为我们同样将该 API 添加到了组件自身上,比如:

import Vue from 'vue'
import Hello from './Hello.vue'
import CreateAPI from 'vue-create-api'
Vue.use(CreateAPI)

// 得到 Vue.prototype.$createHello 和 Hello.create API
Vue.createAPI(Hello, true)

Hello.$create(config, renderFn)

注意:当我们在普通 JS 文件中使用时,无法让 Prop 响应式更新。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

JSmiles

生命进入颠沛而奔忙的本质状态,并将以不断告别和相遇的陈旧方式继续下去。

文章
评论
84963 人气
更多

推荐作者

微信用户

文章 0 评论 0

小情绪

文章 0 评论 0

ゞ记忆︶ㄣ

文章 0 评论 0

笨死的猪

文章 0 评论 0

彭明超

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文