欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Angular指令封装jQuery日期时间插件datetimepicker实现双向绑定示例

程序员文章站 2022-06-07 23:29:26
00.混乱的前端界 angular1.x确实是个学习成本很高的框架,刚开始实习那会儿,前端啥也不懂,工头说用angular,我们这群小弟也只能硬着头皮学。在这之前,前端的...

00.混乱的前端界

angular1.x确实是个学习成本很高的框架,刚开始实习那会儿,前端啥也不懂,工头说用angular,我们这群小弟也只能硬着头皮学。在这之前,前端的东西大部分都用的jquery,而angular正好是和jquery的思维是相反的,开发过程中遇到了不少坑。而angular团队也放弃了1.x开始开发和react神似的2.0版本,唉,真是沧海桑田啊。

01.angular vs jquery

angular模块化和解耦的思路确实值得一学,但是相对于成熟的jquery插件库,angular就显得寒酸了不少,比如,angular-ui中日期控件是这样的:

Angular指令封装jQuery日期时间插件datetimepicker实现双向绑定示例

丑的不要不要的,还不能选时间,相比之下jquery就有很多优秀的控件了比如这样的:

Angular指令封装jQuery日期时间插件datetimepicker实现双向绑定示例

此插件传送门:

那么问题来了,控件一般是直接像这样$("#xx").val("xx")直接塞值进<input />标签的,这不会触发ng-change事件,ng-model也不会被更新,于是笔者写了个angular适配指令,来实现这个控件的双向绑定,对于其他jquery插件,也可以用类似的思路来进行适配。

10.干货

下面是一个demo,比较两者的不同,注意右边ng-bind的属性使用adapter是会同步变化的↓

demo传送门:http://xiazai.jb51.net/201701/yuanma/angular.datetimepicker_jb51.rar

angular.module("directives",[]).directive("datetimepicker",function(){
  return {
    restrict: "ea",  //指令作用范围是element或attribute
    require : "ngmodel", //控制器是指令标签对应的ngmodel
    link: function (scope, element, attrs, ctrl) {

      var unregister = scope.$watch(function(){        //关键点,下面详述

        $(element).append("<input id='date-"+attrs.dateid+"' style='border:none;width:100%' value='"+ctrl.$modelvalue+"'>");
 //template用不好,于是用这个笨办法加上input标签

        element.on('change', function() { //注册onchange事件,设置viewvalue
          scope.$apply(function() {
            ctrl.$setviewvalue($("#date-"+attrs.dateid).val());
          });
        });

        element.on('click',function(){  //click触发日期框
          $("#date-"+attrs.dateid).datetimepicker({ 
            format : attrs.format || 'y/m/d h:i',  //格式
            onclose : function(){          //关闭日期框时手动触发change事件
              element.change();
            }
          });
        });

        element.click();    //第一次绑定事件,模拟一次click,否则肯能要点两下才会出日期框

        return ctrl.$modelvalue;
      }, initialize);

      function initialize(value){ //下面再说
        ctrl.$setviewvalue(value);
        unregister();
      }
    }
  }
});

写这个指令过程中遇到了一个大坑,查了很久才明白,angular初始化一个ngmodel的时候,是会先给它的value置为nan,初始化必须要先调用$watch()来监测真正值被设置的时候,然后调用上面的initialize方法来设置view值。否则在controller中设置的初始值会变成nan。

11.不足之处

原插件是有很多可选项的,我只实现了一个最基本的format,有其他需求的自行改代码吧。可以利用第三个attrs参数获取属性,然后调用原插件的配置方法,来实现更复杂的逻辑。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。