利用Vue.js指令实现全选功能


Posted in Javascript onSeptember 08, 2016

因为刚开始接触vue不久,全选的实现参考了知乎上的实现方法:

     1、从服务器拿到数据,为每个item设置checked属性

     2、计算选中的数量selectCount,如果选中的数量与selectItems的数量相等,则全选selectAll选中

     3、点全选时,将每个item的checked属性置为true,反选时置为false,

     4、每次selectItems的属性发生变化时,都将checked的为true的item放入数组checkedGroups中

下面为实现代码:

//全选
 data: function() {
  return {
   selectItems: [], // 从服务器拿到的数据
  }
 },
 computed: {
  // 全选checkbox绑定的model
  selectAll: {
   get: function() {
    return this.selectCount == this.selectItems.length;
   },
   set: function(value) {
    this.selectItems.forEach(function(item) {
     item.checked = value;
    });
    return value;
   }
  },
  //选中的数量
  selectCount: {
   get: function() {
    var i = 0;
    this.selectItems.forEach(function(item) {
     if (item.checked) {
      i++;
     }
    });
    return i;
   }
  },
  //选中的数组
  checkedGroups: {
   get: function() {
    var checkedGroups = [];
    this.selectItems.forEach(function(item) {
     if (item.checked) {
      checkedGroups.push(item);
     }
    });
    return checkedGroups;
   }
  }
 }

这种方法用起来不太方便,首先是很难复用,每次要用到的时候都需要写一次computed,其次是selectAll、checkedGroups、selectItems都已经固定,不太灵活。

所以在这次项目中,我用vue的指令重新实现了全选的功能,directive的思路其实跟computed差不多,先上代码:

export default {
 'check-all': {
  twoWay: true,
  params: ['checkData'],
  bind() {
   /**
    - 如果所有的列表的checked属性都为true,则选中全选框,否则不选中全选框
    */
   this.vm.$watch(this.params.checkData, (checkData) => {
    if (checkData.every((item) => item.checked)) {
     this.set(true);
    } else {
     this.set(false);
    }
   }, { deep: true });
  },
  // checkAll发生更改时
  update(checkAll) {
   /**
    - 如果全选框被选中,则将列表的所有checked属性转为true,否则转为false
    */
   if (checkAll) {
    this.vm[this.params.checkData].forEach((item) => {
     item.checked = true;
    });
   } else {
    this.vm[this.params.checkData].forEach((item) => {
     item.checked = false;
    });
   }
  },
 },
};

调用:

<input type="checkbox" v-model="checkAll" v-check-all="checkAll" check-data="checkData">
  <ul>
   <li v-for="item in checkData">
    <input type="checkbox" v-model="item.checked">
    {{item.text}}
   </li>
  </ul>

先说说这样用的优点:

1、方便使用,在需要用的地方,写上v-check-all指令和check-data就可以

2、全选的model和数组名可以定制,用什么名字都可以,全选的model不想叫checkAll叫checkAllData也可以,数组不想叫checkData叫dataFromServer也可以。

在指令中,指定twoWay为true,就可以用this.set(value)来设置checkAll的值,用params接收绑定指令元素上的属性值checkData,也就是需要操作的数组。

this.vm获取使用指令的上下文,调用上下文的$watch来监听checkData的变化,如果checkData全部选中,则设置checkAll为true,否则设置checkAll为false。

当指令值(checkAll)发生变化,如果为true,则将checkData的checked属性都设为true,否则为false。至此,一个全选的指令就完成了。

在做这个全选指令的时候,本来想用paramWatchers来监听checkData的变化的,但是发觉checkData变动时,并不会触发paramWatchers的回调,后来看了一下源码才发现,paramWatchers其实也是调用了$watch,但是不支持深度检测:

Directive.prototype._setupParamWatcher = function (key, expression) {
 var self = this;
 var called = false;
 var unwatch = (this._scope || this.vm).$watch(expression, function (val, oldVal) {
 self.params[key] = val;
 // since we are in immediate mode,
 // only call the param change callbacks if this is not the first update.
 if (called) {
  var cb = self.paramWatchers && self.paramWatchers[key];
  if (cb) {
  cb.call(self, val, oldVal);
  }
 } else {
  called = true;
 }
 }, {
 immediate: true,
 user: false
 });(this._paramUnwatchFns || (this._paramUnwatchFns = [])).push(unwatch);
};

总结

以上就是这篇文章的全部内容,有不对的地方,还请大家多多指教。希望这篇文章的内容对大家能有所帮助。

Javascript 相关文章推荐
Javascript客户端脚本的设计和应用
Aug 21 Javascript
javascript CSS画图之基础篇
Jul 29 Javascript
基于jQuery的倒计时插件代码
May 07 Javascript
调试Node.JS的辅助工具(NodeWatcher)
Jan 04 Javascript
JS正则表达式比较常见用法
Jan 26 Javascript
layui框架中layer父子页面交互的方法分析
Nov 15 Javascript
JavaScript重复元素处理方法分析【统计个数、计算、去重复等】
Dec 14 Javascript
layui 表格的属性的显示转换方法
Aug 14 Javascript
vue  directive定义全局和局部指令及指令简写
Nov 20 Javascript
在Vue mounted方法中使用data变量详解
Nov 05 Javascript
vue实现把接口单独存放在一个文件方式
Aug 13 Javascript
Vue-router中hash模式与history模式的区别详解
Dec 15 Vue.js
AngularJS 实现JavaScript 动画效果详解
Sep 08 #Javascript
javascript使用 concat 方法对数组进行合并的方法
Sep 08 #Javascript
jQuery插入节点和移动节点用法示例(insertAfter、insertBefore方法)
Sep 08 #Javascript
jQuery删除节点用法示例(remove方法)
Sep 08 #Javascript
jQuery复制节点用法示例(clone方法)
Sep 08 #Javascript
jQuery替换节点用法示例(使用replaceWith方法)
Sep 08 #Javascript
jQuery元素属性操作实例(设置、获取及删除元素属性)
Sep 08 #Javascript
You might like
php ckeditor上传图片文件名乱码解决方法
2013/11/15 PHP
php对二维数组进行相关操作(排序、转换、去空白等)
2015/11/04 PHP
php版微信公众平台实现预约提交后发送email的方法
2016/09/26 PHP
PHP共享内存使用与信号控制实例分析
2018/05/09 PHP
php 提交表单 关闭layer弹窗iframe的实例讲解
2018/08/20 PHP
IE event.srcElement和FF event.target 功能比较
2010/03/01 Javascript
JS实现跟随鼠标立体翻转图片的方法
2015/05/04 Javascript
JavaScript驾驭网页-CSS与DOM
2016/03/24 Javascript
基于javascript实现九宫格大转盘效果
2020/05/28 Javascript
详解jQuery中的deferred对象的使用(一)
2016/05/27 Javascript
轮播图组件js代码
2016/08/08 Javascript
详解XMLHttpRequest(一)同步请求和异步请求
2016/09/14 Javascript
jQuery实现自动调用和触发某个事件的方法
2016/11/18 Javascript
bootstrap常用组件之头部导航实现代码
2017/04/20 Javascript
BootStrap Fileinput上传插件使用实例代码
2017/07/28 Javascript
webpack-dev-server远程访问配置方法
2018/02/22 Javascript
Layui组件Table绑定行点击事件和获取行数据的方法
2018/08/19 Javascript
详解vue+axios给开发环境和生产环境配置不同的接口地址
2019/08/16 Javascript
layui-table表复选框勾选的所有行数据获取的例子
2019/09/13 Javascript
微信小程序自定义胶囊样式
2020/12/27 Javascript
[20:39]DOTA2-DPC中国联赛 正赛开幕式 1月18日
2021/03/11 DOTA
使用Python制作获取网站目录的图形化程序
2015/05/04 Python
详解Python的三种可变参数
2019/05/08 Python
PyQt5 如何让界面和逻辑分离的方法
2020/03/24 Python
利用CSS3实现的文字定时向上滚动
2016/08/29 HTML / CSS
语文教育专业应届生求职信
2013/11/23 职场文书
个人自我评价范文
2014/02/05 职场文书
竞选班委演讲稿
2014/04/28 职场文书
庆祝教师节标语
2014/10/09 职场文书
2014年宣传部工作总结
2014/11/12 职场文书
2014年后勤管理工作总结
2014/12/01 职场文书
关于幸福的感言
2015/08/03 职场文书
《我和小伙伴》教学反思
2016/02/20 职场文书
Python还能这么玩之用Python修改了班花的开机密码
2021/06/04 Python
MySQL 发生同步延迟时Seconds_Behind_Master还为0的原因
2021/06/21 MySQL
JS精髓原型链继承及构造函数继承问题纠正
2022/06/16 Javascript