前面一片文章说了vue2.0过滤器,其实自定义指令跟过滤器非常相似,单就定义方式而言,其与过滤器完全一致,分为局部指令,和全局指令。不过就是filter改为directive的区别。

过滤器一般用于对一些常见文本的格式化,而自定义指令主要是为了对底层DOM实现操作,虽然说vue主要是以数据驱动视图,但是总有一些只能操作DOM的应用场景,例如最常见的:拖拽。此时,就是自定义指令大显身手的时候了。

在移动端,常见这么一个需求,以微信为例,在微信主屏左右滑动时候,可以随着滑动法相更改工具栏,也就是随着滑动,可以在“微信”、“通讯录”、“发现”、“我”之间随意切换。
这类需求,必须是操作DOM实现的,而无法通过数据来驱动,所以这时候,不管是你使用插件,还是自己定义,必然都是通过自定义指令来实现左右滑动,而切换工具栏的。

为了方便操作,以tab切换为例:

  <ul class="tabs">
<router-link to="/home" tag="li"><span>首页</span></router-link>
<router-link to="/product" tag="li"><span>产品</span></router-link>
<router-link to="/about" tag="li"><span>关于我们</span></router-link>
</ul>

  在router-view上左右滑动,切换路由:

<router-view v-swipe-horizontal v-animate-css="'bounceInLeft'" appear></router-view>

  我是利用Vue Animate Css来为路由切换添加了一个动画的效果,这里暂不关注。这里需要关注的就是v-swipe-horizontal,“v-”是vue中约定的各种指令的写法,所以其实真正的指令是swipe-horizontal,那么怎么定义swipe-horizontal呢?

  全局定义: 

// 注册一个全局自定义指令 `v-focus`
Vue.directive('swipe-horizontal', {
bind: function (el, binding, vnode, oldVnode) {
// 各种逻辑操做,这个指令中就是滑动的逻辑 }
})

  局部定义:

  

swipeHorizontal:{
bind(el, binding, vnode){
/**
* 当前这种做法就完全依赖于路由配置的顺序需要和tab选项卡保持一致,
* 并且不能随意添加其他位置的路由了
* 如果通过为每一个路由对象的index使用数组的find方法查找的话,
* 则需要为路由对象添加index属性
*/
/* // 利用路由配置数组自身的顺序进行判断
let routes = vnode.context.$router.options.routes,
xc = routes.find((item, index)=>{
return vnode.context.$route.path == item.path
}),
currentRouteIndex = routes.indexOf(xc),
nextRoute = null,
nextIndex = null;
*/
// 利用路有对象自定义的index属性判断
let routes = vnode.context.$router.options.routes,
currentRoute = vnode.context.$route.path,
currentRouteIndex = routes.find((item)=>{
return currentRoute == item.path
}).index,
nextRoute = null,
nextIndex = null;
var arr = routes.map((item)=>{
return item.index
})
var max = Math.max(...arr) el.ontouchstart = (ev)=>{
let touch = ev.touches[0];
let disX = touch.clientX - el.offsetLeft;
let x = disX;
document.ontouchmove = (ev)=>{
let touch = ev.touches[0];
x = touch.clientX ;
}
document.ontouchend = () => {
document.ontouchmove = null;
document.ontouchend = null;
console.log("移动距离: ", x, disX)
/**
* 需要利用(routes.length-1)的原因是 路由配置最后一个对象配置了路由重定向
*/
// 利用当前的路由对象在整个路由中的序号,即路由配置的数组的顺序
/* if (x - disX > 20) {
nextIndex = (currentRouteIndex + 1) >= (routes.length-1) ? 0 : (currentRouteIndex + 1);
} else if(disX - x > 20) {
nextIndex = (currentRouteIndex - 1) < 0 ? routes.length -2 : (currentRouteIndex - 1);
} */
nextRoute = routes[nextIndex]
if (x - disX > 20) {
nextIndex = (currentRouteIndex + 1) > max ? 0 : (currentRouteIndex + 1);
} else if(disX - x > 20) {
nextIndex = (currentRouteIndex - 1) < 0 ? max :(currentRouteIndex - 1);
}
if (nextIndex != null) {
// 利用路有对象自定义的index属性判断
nextRoute = routes.find((item)=>{
return nextIndex == item.index
})
//在自定义的指令中,只有通过vnode.context才能获取到vue对象
vnode.context.$router.replace(nextRoute)
}
}

  我这里提供了两种确定路由的模式,主要是确定路由路径的需要。

  一种依赖路由数组的顺序,但是这种情况下,路由数组的顺序就必须与tab选项卡视图展示的数据保持一致。

  另外一种是为每一个路由添加一个index属性,以确定当前tab选项卡展示的顺序。

  自定义指令中,主要逻辑是需要存在于自定义指令的钩子函数中的,而钩子函数又分为bind,inserted,update,componentUpdated,unbind五种,我不知道有多少人能够用到所有的钩子函数,从我的角度来说,常用的钩子函数bind或者inserted,两者选择其一就可以了。

  而钩子函数中的参数:

  

真的已经很清楚了。

既然使用自定义指令了,那么尽量不要去修改vue的数据,但是如果非得修改,还是可以通过vnode.context获取到当前的Vue对象的,到这一步,当前这个示例,剩下的也就是一些路由操作,以及基本的数组操作了。

其实,主要的就是分清楚什么情况下使用自定义指令,自定义指令的钩子函数中el提供了当前操作的dom,binding提供了各类参数,vnode.context提供了当前的vue对象,然后就是逻辑操作了。各类需求不一致,逻辑就不可能相同,所以也没什么好说的了。

  

vue2.0自定义指令的更多相关文章

  1. vue2.0自定义指令的使用方法

    感觉2.0好坑啊,自定义指令和1.0完全不一样,并且文档写得也不太清晰,下面是我写得一个demo,希望帮助大家更好地理解自定义指令 <!DOCTYPE html> <html lan ...

  2. Vue2.0 - 自定义指令 vue-directive

    Vue.directive('指令',function(el,binding,vnode){ el.style='color:'+binding.value;}); el : 指令所绑定的元素,可以用 ...

  3. vue2.0 自定义指令

    Vue指令 Vue的指令以v-开头,作用在HTML元素上,将指令绑定在元素上,给绑定的元素添加一些特殊行为. 例如: <h1 v-if="yes">Yes</h1 ...

  4. vue2.0 自定义指令详解

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  5. CKEditor5 + vue2.0 自定义图片上传、highlight、字体等用法

    因业务需求,要在 vue2.0 的项目里使用富文本编辑器,经过调研多个编辑器,CKEditor5 支持 vue,遂采用.因 CKEditor5 文档比较少,此处记录下引用和一些基本用法. CKEdit ...

  6. vue3.0自定义指令(drectives)

    在大多数情况下,你都可以操作数据来修改视图,或者反之.但是还是避免不了偶尔要操作原生 DOM,这时候,你就能用到自定义指令. 举个例子,你想让页面的文本框自动聚焦,在没有学习自定义指令的时候,我们可能 ...

  7. vue2.0 v-model指令

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  8. vue2.0 自定义过滤器

    2.0中已经废弃了过滤器,需要我们自定义 <div id="app"> {{message|uppercase}} </div> //过滤器 Vue.fil ...

  9. vue2.0 自定义 提示框(Toast)组件

    1.自定义 提示框 组件 src / components / Toast / index.js /** * 自定义 提示框( Toast )组件 */ var Toast = {}; var sho ...

随机推荐

  1. jdk源码剖析三:锁Synchronized

    一.Synchronized作用 (1)确保线程互斥的访问同步代码 (2)保证共享变量的修改能够及时可见 (3)有效解决重排序问题.(Synchronized同步中的代码JVM不会轻易优化重排序) 二 ...

  2. linux 安装多个版本JDK,指定tomcat的jdk版本

    JDK的下载可以直接到官网下载,这里不再介绍 一.安装JDK 7 vi /etc/profile #set java environmentexport JAVA_HOME=/usr/java/jdk ...

  3. 多层级sql查询

    SELECT ( SELECT ) FROM t_app_user WHERE parent_user_id = t.user_id /*多层级,t.user_id */ ) AS directCou ...

  4. redis常用客户端命令

    sentinel操作: 1.查询主节点:redis-cli -h 172.16.4.220 -p 46379 SENTINEL get-master-addr-by-name mymaster 2.查 ...

  5. mac系统 Xcode打包ionic项目(iOS)

    一.环境搭建 1. 安装Node.js,使用node -v 查询版本号: 2. 安装ionic: $ sudo npm install -g cordova(可以指定版本,如cordova@7.0.1 ...

  6. matlab与示波器连接及电脑连接

    参考:http://blog.sina.com.cn/s/blog_4eff3a0e0100zb8h.html 最近进行了示波器的数据采集,MSO2014,openChoice软件+Tekvisa驱动 ...

  7. 放一个Dynamicinputs corresponding to Dynamicknobs的Node源码

    static const char* const CLASS = "AddInputsSol"; static const char* const HELP = "Add ...

  8. MySQL性能管理及架构设计

    第1章 实例和故事 1-1 什么决定了电商双11大促的成败 老板可能会说:"是我们的英明决策和运筹帷幄". 运营和产品可能会说:"是由于我们的活动策划和产品设计" ...

  9. BlockingQueue队列

    1.BlockingQueue定义的常用方法如下     抛出异常 特殊值 阻塞 超时 插入 add(e) offer(e) put(e) offer(e,time,unit) 移除 remove() ...

  10. C# 基于大整数类的RSA算法实现(公钥加密私钥解密,私钥加密公钥解密)

    但是C#自带的RSA算法类RSACryptoServiceProvider只支持公钥加密私钥解密,即数字证书的使用. 所以参考了一些网上的资料写了一个RSA的算法实现.算法实现是基于网上提供的一个大整 ...