loading效果很常见,常见到我们任何一个项目中,都可以见到他的身影。今天就以loading作为切入口,唠叨一下vuejs的插件的写法。

  看vuejs官方文档关于插件的说明,关于使用插件和写插件,Vue插件基本上都躲不开以下几种方案: 

  1. 添加全局方法或者属性,如: vue-custom-element

  2. 添加全局资源:指令/过滤器/过渡等,如 vue-touch

  3. 通过全局 mixin 方法添加一些组件选项,如: vue-router

  4. 添加 Vue 实例方法,通过把它们添加到 Vue.prototype 上实现。

  5. 一个库,提供自己的 API,同时提供上面提到的一个或多个功能,如 vue-router

  我们以loading效果为例,添加一个全局自定义指令的方式写一个插件。使用方式如: 

<div v-loading="loading"></div>

  依赖loading值,展示或者隐藏loading效果。loading值默认为false,不展示loading效果。

  loading效果的模板其实很简单,甚至可以用唯一一个div标签完成,这里我们不强求怎么写loading模板,我们将模板定义在Loading.vue文件内。

  在loading.js文件内真正写插件:

  首先看一下Vue.use这个方法:

安装 Vue.js 插件。如果插件是一个对象,必须提供 install 方法。如果插件是一个函数,它会被作为 install 方法。install 方法调用时,会将 Vue 作为参数传入。

  也就是说我们在loading.js文件中export一个对象出去的时候,如果该对象是普通的对象,则需要在对象中添加一个install的方法,而如果导出一个函数对象,可以直接导出该函数。

  就以当前的loading举例说明:

  如果导出一个对象:

export default {
vm: null,
install: Vue => {
Vue.directive(Loading.name, {
// 真正的实现逻辑
})
},
...
}

  而如果导出是一个函数,我们可以这么写:

export default Vue => {
Vue.directive(Loading.name, {
// 真正的实现逻辑
})
}

  仔细观察一下,发现我们可以将install方法的函数体直接导出当做use的参数。

  我们这里就以对象方式导出,主要目的是留一个vm属性,用于保存生成的loading节点,方便删除该节点。

  在刚刚写这个插件的时候,不是插件不会写,恰恰是自定义指令的钩子函数使用不当,导致问题的产生。我们需要依赖指令的binding.value值来展示或者隐藏loading效果。基本上尝试了所有的钩子函数后,发现只有update是在所在的组件 VNode 更新时调用,也就是说binding.value的值发生变化,在update内部更新dom。

update (el, binding) {
const ele = window.getComputedStyle(el, null)
// 添加判断,如果当前元素的高度小于60 则loading固定在整个屏幕中间 否则固定在当前元素中间
LoadingComp.vm.$el.style.position = !binding.modifiers.fixed && (parseInt(ele['height']) < 60 ? 'fixed' : 'absolute')
binding.value ? el.appendChild(LoadingComp.vm.$el) : el.removeChild(LoadingComp.vm.$el)
}

  需要稍作解释的时候,我们有时候绑定的v-loading的dom节点高度在完全没有数据的时候,可能完全没有高度可言,那么这里就将loading的position设置为fixed,固定在屏幕中央。这里可以自行定义是否需要该判断条件。

 binding.value ? el.appendChild(LoadingComp.vm.$el) : el.removeChild(LoadingComp.vm.$el)

  这么一句,就是添加和删除当前的loading节点,有点类似于v-if的处理效果。

  这里还缺少生成loading的DOM节点这一过程,是因为我认为其他的过程只要生成一次就可以了,而不会随着binding.value的值变化发生任何变化。所以需要在钩子函数inserted内实现: 

inserted (el, binding) {
const elem = window.getComputedStyle(el, null)
el.style.position = elem['position'] === 'static' && 'relative'
LoadingComp.vm = new LoadingConstructor().$mount()
// 如果带修饰符fixed loading固定在屏幕中间
let val = binding.modifiers.fixed && 'fixed'
LoadingComp.vm.$el.style.position = val && val
}

  主要实现的是,当前绑定v-loading的元素是否有定位,如果没有定位属性,则添加position:"relative"属性。并同时生成loading对象,注意这里仅仅是生成,并没有挂载到文档中。

  添加的一行注释,主要多实现了一个功能,就是如果用户就非得相对整个屏幕固定loading,那么可以使用v-loading.fixed="loading"来达成这一效果。

  这里其实可以引申出来,如果loading需要遮罩层的效果的话,也可以以类似的方法实现。

  看看完整的代码吧!

import Vue from 'vue'
import Loading from './Loading.vue'
const LoadingConstructor = Vue.extend(Loading)
const LoadingComp = {
vm: null,
install: Vue => {
Vue.directive(Loading.name, {
inserted (el, binding) {
const elem = window.getComputedStyle(el, null)
el.style.position = elem['position'] === 'static' && 'relative'
LoadingComp.vm = new LoadingConstructor().$mount()
// 如果带修饰符fixed loading固定在屏幕中间
let val = binding.modifiers.fixed && 'fixed'
LoadingComp.vm.$el.style.position = val && val
},
update (el, binding) {
const ele = window.getComputedStyle(el, null)
// 添加判断,如果当前元素的高度小于60 则loading固定在整个屏幕中间 否则固定在当前元素中间
LoadingComp.vm.$el.style.position = !binding.modifiers.fixed && (parseInt(ele['height']) < 60 ? 'fixed' : 'absolute')
binding.value ? el.appendChild(LoadingComp.vm.$el) : el.removeChild(LoadingComp.vm.$el)
}
})
}
} export default LoadingComp

  

vue2.0插件--loading的更多相关文章

  1. vue2.0插件

    1.better-scroll 参考网址:https://ustbhuangyi.github.io/better-scroll/doc/zh-hans/ better-scroll 是什么 firs ...

  2. 干货分享:vue2.0做移动端开发用到的相关插件和经验总结(2)

    最近一直在做移动端微信公众号项目的开发,也是我首次用vue来开发移动端项目,前期积累的移动端开发经验较少.经过这个项目的锻炼,加深了对vue相关知识点的理解和运用,同时,在项目中所涉及到的微信api( ...

  3. vue2.0做移动端开发用到的相关插件和经验总结

    最近一直在做移动端微信公众号项目的开发,也是我首次用vue来开发移动端项目,前期积累的移动端开发经验较少.经过这个项目的锻炼,加深了对vue相关知识点的理解和运用,同时,在项目中所涉及到的微信api( ...

  4. vue-swiper 基于Vue2.0开发 轻量、高性能轮播插件

    vue-swiper 基于 Vue2.0 开发,基本满足大部分功能 轻量.高性能轮播插件.目前支持 无缝衔接自动轮播.无限轮播.手势轮播 没有引入第三方库,原生 js 封装,打包之后只有 8.2KB ...

  5. vue2.0实现倒计时的插件(时间戳 刷新 跳转 都不影响)

    我发现好多倒计时的插件,刷新都会变成从头再来,于是自己用vue2.0写了一个,测试通过,直接上代码 如下是组件代码: <template> <span :endTime=" ...

  6. vue2.0 移动端,下拉刷新,上拉加载更多 插件

    本人正在基于 vue2.0 + webpack + es6 搭建前端架构,整理了部分插件,下面这个是下拉更新 上拉更多的,挺好用的,分享给大家. 直接上代码,不懂的多看几遍,下面我换会告诉大家如何使用 ...

  7. Vue2.0 分页插件pagination使用详细说明

    Vue2.0 分页pagination使用 插件下载地址:Vue_Pagination 插件描述:基于jQuery的分页插件大家都用过很多了吧,今天分享一下基于Vue的分页插件pagination.j ...

  8. vue2.0做移动端开发用到的相关插件和经验总结1.0

    最近在用vue2.0做微信公众号相关的前端开发,经过这次开发实践,现将项目中用到的相关比较实用的插件及遇到的相关问题进行整理,希望和大家共同交流...... cssrem:一个CSS值转REM的VSC ...

  9. vue2.0 移动端,下拉刷新,上拉加载更多插件,修改版

    在[实现丰盛]的插件基础修改[vue2.0 移动端,下拉刷新,上拉加载更多 插件], 1.修改加载到尾页面,返回顶部刷新数据,无法继续加重下一页 2.修改加载完成文字提示 原文链接:http://ww ...

随机推荐

  1. C语言中的二维数组

    1.二维数组的定义和引用 一. 数据类型 数组名[常量表达式1][常量表达式2]; (1)假如有个二维数组array[n][m],则行下标的取值范围0~n-1 (2)列下标的取值范围0~m-1 (3) ...

  2. 最基础的 swift 语言

    import Foundation //打印函数 print("Hello, World!") //不用加分号, 字符串就是"", 不用加@ print(&qu ...

  3. abaqus python库变强变大233333333333333

    有没有小伙伴想在 至于怎么安装pip 度小娘一位大神提供了办法  https://jingyan.baidu.com/article/7e4409533f32092fc0e2ef24.html 如有需 ...

  4. CentOS7下MySQL5.7安装配置方法图文教程(YUM)

    安装环境:CentOS7 64位,MySQL5.7 1.配置YUM源 在MySQL官网中下载YUM源rpm安装包:http://dev.mysql.com/downloads/repo/yum/ # ...

  5. 项目工程中的WebRoot与WebContent有什么区别?

    [1] 在MyEclipse中创建web项目后,web程序的根目录文件夹是WebRoot,而创建dynam web project时候,web程序的根 目录文件夹是WebContent,他们之间没有本 ...

  6. 闲记 单元格与单元格之间的边 ///字体属性都是font开头,除了颜色属性 ///文本属性都是text开的,除了设置行高。

    这两天一直在做网页没有什么太大的问题,期间也考了一场试,对答案的时候老师讲了一些小知识,因此来记录一下. 单元格与单元格之间的边距(cellspaling) list-type-image可以使用图像 ...

  7. java_集合类_简

    Collection 来源于Java.util包,实用常用的数据结构,字面意思就是容器 主要方法 boolean add(Object o)添加对象到集合 boolean remove(Object ...

  8. Android L2TP Client Setup

    原文链接:http://www.softether.org/4-docs/2-howto/9.L2TPIPsec_Setup_Guide_for_SoftEther_VPN_Server/3.Andr ...

  9. js入门 关于js属性及其数据类型(详解)

    1. js的本质就是处理数据.数据来自于后台的数据库. 所以变量就起到一个临时存储数据的作用. ECMAScript制定了js的数据类型. 数据类型有哪些? 1. 字符串   String 2. 数字 ...

  10. saltstack实战1-salt-syndic

    Syndic建立在中心Master和Minions之间, 并允许多层分级Syndic, 使Salt拓扑可以变得更为灵活. 那么Syndic是如何工作的? 当前有哪些优势和局限哪? Syndic的优势和 ...