JS编写插件

瞎逼逼

做个原生js插件装逼一下

再逼逼

部分经常重复的代码抽象出来,写到一个单独的文件中为以后再次使用。再看一下我们的业务逻辑是否可以为团队服务。

插件不是随手就写成的,而是根据自己业务逻辑进行抽象。没有放之四海而皆准的插件,只有对插件,之所以叫做插件,那么就是开箱即用,或者我们只要添加一些配置参数就可以达到我们需要的结果。如果都符合了这些情况,我们才去考虑做一个插件。

插件封装的条件

  1. 插件自身的作用域与用户当前的作用域相互独立,也就是插件内部的私有变量不能影响使用者的环境变量;
  2. 插件需具备默认设置参数;
  3. 件除了具备已实现的基本功能外,需提供部分API,使用者可以通过该API修改插件功能的默认参数,从而实现用户自定义插件效果;
  4. 插件支持链式调用;
  5. 插件需提供监听入口,及针对指定元素进行监听,使得该元素与插件响应达到插件效果。

插件的外包装

用函数包装

插件就是封装在一个闭包中的函数集

  function add(n1, n2){
     return n1 + n2;
  }
  • 团队开发中可能存在的问题

命名冲突,全局污染

用全局对象包装

  • 为了解决这种全局变量污染的问题。这时我们定义一个js对象来接收我们这些工具函数
   let plugin = {
      add: function(n1, n2) {},   //加
      sub: function(n1, n2) {},   //减
      mul: function(n1, n2) {}    //乘
   }
  • 调用: plugin.add(1, 2)

上面的方式,在一定程度上已经解决了全部污染的问题。在团队协作中只要约定好命名规则了,告知其它同学即可。当然不排除别人重新定义并赋值这个值,这时的解决方案

   if(!plugin){  //这里的if条件可以用: (typeof plugin == 'undefined')
        var plugin = {
            // 逻辑
        }
   }

   // 也可以这样写
   var plugin;
   if(!plugin){
        plugin = {
            // ...
        }
   }
  • 备注:为什么可以在此声明plugin变量?实际上js的解释执行,会把所有声明都提前。如果一个变量已经声明过,后面如果不是在函数内声明的,则没有影响的。所以,就算在别的地方声明 var plugin,也可以在这里再次声明一次
    如何判断Javascript对象是否存在

利用闭包包装

上面基本实现了插件的基本功能,不过我们的插件,是定义在全局域里面的。js变量的调用,从全局作用域上查找的速度比私有作用域里面慢的多。所以,我们最好将插件逻辑写在一个私有作用域里面。实现私有作用域,最好的办法就是使用闭包。可以把插件当做一个函数,插件内部的变量及函数的私有变量,为了在调用插件后依旧能使用其功能,闭包的作用就是延长函数(插件)内部变量的生命周期,使得函数可以重复调用,而不影响用户自身作用域。

   ;(function(global, undefined) {
        var plugin = {
            add: function(n1, n2) {... }
            ...
        }
        //最后将插件对象暴露给全局对象
        'plugin' in global && global.plugin = plugin;
   })(window);
  1. 在定义插件之前添加一个分号,可以解决js合并时可能会产生的错误问题;
  2. undefined在老一辈的浏览器是不被支持的,直接使用会报错,js框架要考虑到兼容性,因此增加一个形参undefined,就算有人把外面的undefined定义看,里面的undefined依然不受影响;
  3. 把window对象作为参数传入,是避免了函数执行的时候到外部去查找
  • 直接传window对象进去,还是不妥当,插件不一定用于浏览器上,所以我们不传参数,直接取当前的全局this对象作为顶级对象用。
  ;(function(global, undefined) {
    "use strict"  //使用严格模式检查,使语法更规范
    var _global;
    var plugin = {
        add: function(n1, n2) { ... }
        ...
    }
    //最后将插件对象暴露给全局对象
    _global = (function() {return this || (0, eval)('this');}());
    !('plugin' in _global) && (_global.plugin = plugin);
  }());
  • 如此,我们不需要传入任何参数,并且解决了插件对环境的依事性。如此我们的插件可以在任何宿主环境上运行了。

上面的代码段中有段奇怪的表达式:(0, eval)('this'),实际上(0,eval)是一个表达式,这个表达式执行之后的结果就是eval这一句相当于执行eval('this')的意思,详细解释看此篇:(0,eval)('this')释义或者看一下这篇(0,eval)('this')

立即自执行函数,有两种写法:

   // 写法一
   (function() {})
  //写法二
  (function(){} ())
  • 两种写法没区别,都是正确的写法,第二张更像一个整体

知识点
js里面()括号就是将代码结构变成表达式,被包在()里面的变成表达式之后,则就会立即执行,js中的代码表达式有很多种


    void function(){...}();

    // 或者
    !function foo(){...}();

    //或者
    +function foot(){...}();

当然,我们不推荐你这么用,而且乱用可能会产生一些歧义。

使用模块化的规范包装

大型开发,多人开发,会产生多个文件,每个人负责一个小功能,那么如何才能将所有人开发的代码集合起来呢?要实现协同开发插件,需具备如下条件

  • 每个功能互相之间的依赖必须明确,则必须严格按照依赖的顺序进行合并或者加载
  • 每个子功能分别都要是一个闭包,并且将公共的接口暴露到共享域也及即使一个被主函数暴露的公共对象

关键如何实现,有多种办法,最笨的办法就是按顺序加载js

    <script type="text/javascript" src="part1.js"></script>
    <script type="text/javascript" src="part2.js"></script>
    <script type="text/javascript" src="part3.js"></script>
    ...
    <script type="text/javascript" src="main.js"></script>

但是不推荐这么做,这样做与我们所追求的插件的封装性相背。
加载器,比如requireseajs,或者也可以类似Node的方式进行加载,不过在浏览器端,我们还得利用打包器实现模块加载,比如browserify

为了实现插件的模块化并且让我们的插件也是一个模块,我们就得让我们的插件也实现模块化的机制
实际上,只要判断是否存在加载器,我们就是使用加载器,如果不存在加载器。我们就使用顶级域对象

    if(typeof module !== "undefined" && module.exports) {
      module.exports = plugin;
    } else if (type define === "function" && define.amd) {
      define(function(){return plugin;});
    } else {
      _globals.plugin = plugin;
    }

这样我们的完整插件的样子应该是这样的:

    // plugin.js
    ;(function(undefined) {
        "use strict"
        var _global;
        var plugin = {
          add: function(n1, n2) { return n1 + n2; }, //加
          sub: function(n1, n2) { return n1 - n2; }, //减
          mul: function(n1, n2) { return n1 * n2; }, //乘
          div: function(n1, n2) { return n1 / n2; }, //除
          sur: function(n1, n2) { return n1 % n2; } //余
        }
        //最后将插件对象暴露给全局对象
        _global = (function(){return this || (0, eval)('this');}());
        if(typeof module !=="undefined" && module.exports) {
            module.exports = plugin;
        } else if(typeof define === "function" && define.amd) {
            define(function(){return plugin;});
        } else {
            !('plugin' in _global) && (_global.plugin = plugin);
        }
    }())

    // 引入插件之后,则可以直接使用plugin对象
    with(plugin) {
      console.log(add(2, 1)) //3
      console.log(sub(2, 1)) //1
      console.log(mul(2,1)) // 2
      console.log(div(2,1)) // 2
      console.log(sur(2,1)) // 0
    }

插件的API

插件的默认参数

我们知道,函数是可以设置默认参数,而不管我们是否传有参数,我们都应该返回一个值以告诉用户我做了怎样的处理

    function add(param) {
      var args = !!param ?  Array.prototype.slice.call(arguments) : [];
      return args.reduce(function(pre, cur){
        return pre + cur;
      }, 0);
    }

    console.log(add()) //不传参,结果输出0,则这里已经设置了默认了参数的空数组
    console.log(add(1, 2, 3, 4, 5)) //传参,结果输出15

则作为一个健壮的js插件,我们应该吧一些基本的状态参数添加到我们需要的插件上去。
假设还是上面的加减乘除余的需求,我们如何实现插件的默认参数呢? 道理是一样的

    //plugin.js
    ;(function(undefined) {
      "use strict"
      var _global;

      function result(args, fn) {
        var argsArr = Array.prototype.slice.call(args);
        if(argsArr.length > 0) {
          return argsArr.reduce(fn);
        } else {
          return 0;
        }
      }

      var plugin = {
          add: function() {
              return result(arguments, function(pre, cur) {
                  return pre + cur;
              });
          },
          sub: function() {
              return result(arguments, function(pre, cur) {
                  return pre - cur;
              });
          },
          mul: function() {
              return result(arguments, function(pre, cur) {
                  return pre * cur;
              });
          },
          div: function() {
              return result(arguments, function(pre, cur) {
                  return pre / cur;
              });
          },
          sur: function() {
              return result(arguments, function(pre, cur) {
                  return pre % cur;
              });
          }
      }

      // 将插件对象暴露给全局对象
      _global = (function(){ return this || (0, eval)('this'); }());
      if(typeof module !=="undefined" && module.exports) {
          module.exports = plugin;
      } else if (typeof define === "function" && define.amd) {
          define(function() {return plugin;});
      } else {
          !('plugin' in _global) && (_global.plugin = plugin);
      }

    }());

    //输出结果
    with(plugin){
        console.log(add());
        console.log(add(2, 1)); //3
    }

实际上,插件都有自己的默认参数,就以我们最为常见的表单验证插件为例 validate.js

插件的钩子

设计一下插件,参数或者其逻辑肯定不是写死的,我们得像函数一样,得让用户提供自己的参数去实现用户的需求。则我们的插件需要提供一个修改默认参数的入口。[API]

我们常将容易被修改和变动的方法或属性统称为钩子(Hook),方法则直接叫钩子函数
实际上,我们知道插件可以像一条绳子上挂东西,也可以拿掉挂的东西,那么一个插件,实际上就是个形象上的。不过我们上面的所有钩子都是挂在对象上的,用于实现链并不是很理想。

插件的链式调用(利用当前对象)

插件并非都是链式调用的,有时候,我们只是用钩子来实现一个计算并返回结果,取得运算结果就可以了。但是有些时候,我们用钩子并不需要其返回结果。我们只利用其实现我们的业务逻辑,为了代码简洁与方便,我们常常将插件的调用按链式的方式进行调用。
最常见的jquery的链式调用如下:

  $(<id>).show().css('color','red').width(100).height(100)...

将链式调用运用到我们的插件中,将业务结构改为:

    var plugin = {
        add: function(n1, n2) { return this; },
        sub: function(n1, n2) { return this; },
        ...    
    }

我们只要将插件的当前对象this直接返回,则在下一个方法中,同样可以引用插件对象plugin的其它钩子方法,然后调用得时候就可以使用链式了

    plugin.add().sub().mul().div()  

显然这样做并没有什么意义。我们这里的每一个钩子函数都只是用来计算并且获取返回值而已。而链式调用本身的意义是用来处理业务逻辑的。

插件的链式调用(利用原型链)

Js中,万物皆对象,所有对象都是继承自原型。JS在创建对象(不论是普通对象还是函数对象)的时候,都有一个叫proto的内容属性,用于指向创建它的函数对象的原型对象prototype。

更改插件将plugin写成一个构造函数,我们将插件名换为Calculate避免因为Plugin大写的时候与window对象中的API冲突。

    function Calculate() {}
    Calculate.prototype.add = function() {return this;}
    Calculate.prototype.sub = function() {return this;}
    Calculate.prototype.mul = function() {return this;}

假设我们的插件是对初始化参数进行运算并输出结果,则

    // plugins.js
    ;(function(undefined) {
        "use strict"
        var _global;
        
        function result(args, type) {
            var argsArr = Array.prototype.slice.call(args);
            if(argsArr.length == 0) return 0;
            switch(type) {
                case 1:  return argsArr.reduce(function(p, c) {return p + c;});
                case 2:  return argsArr.reduce(function(p, c) {return p - c;});
                case 3:  return argsArr.reduce(function(p, c) {return p * c;});
                case 4:  return argsArr.reduce(function(p, c) {return p / c;});
                case 5:  return argsArr.reduce(function(p, c) {return p % c;});
                default: return 0;
          }
      }

      function Calculate() {}
      Calculate.prototype.add = function() { console.log(result(arguments,1)); return this; }
      Calculate.prototype.sub = function() { console.log(result(arguments,2)); return this; }
      Calculate.prototype.mul = function() { console.log(result(arguments,3)); return this; }
      Calculate.prototype.div = function() { console.log(result(arguments,4)); return this; }
      Calculate.prototype.sur = function() { console.log(result(arguments,5)); return this; }

    // 最后将插件对象暴露给全局对象
   _global = (function(){ return this || (0, eval)('this'); }());
   if (typeof module !== "undefined" && module.exports) {
         module.exports = Calculate;
    } else if (typeof define === "function" && define.amd) {
        define(function(){return Calculate;});
    } else {
        !('Calculate' in _global) && (_global.Calculate = Calculate);
    }
  }());

调用插件

    var plugin = new Calculate();
    plugin
        .add(2, 1)
        .sub(2, 1);
   // 结果
   // 3
   // 1

编写UI组件

一般情况,如果一个js仅仅是处理一个逻辑,我们称之为插件,但如果与dom和css有关系并且具备一定的交互性,一般叫组件。
利用原型链,可以将一些UI层面的业务代码封装在一个小组件,并利用js实现组件的交互性

需求:

  1. 实现一个弹层,此弹层可以显示一些文字提示性的信息;
  2. 弹层右上角必须有一个关闭按扭,点击之后弹层消失;
  3. 弹层底部必有一个“确定”按扭,然后根据需求,可以配置多一个“取消”按扭;
  4. 点击“确定”按扭之后,可以触发一个事件;
  5. 点击关闭/“取消”按扭后,可以触发一个事件。

html 代码

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>index</title>
    <link rel="stylesheet" type="text/css" href="index.css">
</head>
<body>
    <div class="mydialog">
        <span class="close">×</span>
        <div class="mydialog-cont">
            <div class="cont">hello world!</div>
        </div>
        <div class="footer">
            <span class="btn">确定</span>
            <span class="btn">取消</span>
        </div>
    </div>
    <script src="index.js"></script>
</body>
</html>

css

* {
    padding: 0;
    margin: 0;
}

.mydialog {
    background: #fff;
    box-shadow: 0 1px 10px 0 rgba(0, 0, 0, 0.3);
    overflow: hidden;
    width: 300px;
    height: 180px;
    border: 1px solid #dcdcdc;
    position: absolute;
    top: 0;
    right: 0;
    bottom: 0;
    left: 0;
    margin: auto;
}

.close {
    position: absolute;
    right: 5px;
    top: 5px;
    width: 16px;
    height: 16px;
    line-height: 16px;
    text-align: center;
    font-size: 18px;
    cursor: pointer;
}

.mydialog-cont {
    padding: 0 0 50px;
    display: table;
    width: 100%;
    height: 100%;
}

.mydialog-cont .cont {
    display: table-cell;
    text-align: center;
    vertical-align: middle;
    width: 100%;
    height: 100%;
}

.footer {
    display: table;
    table-layout: fixed;
    width: 100%;
    position: absolute;
    bottom: 0;
    left: 0;
    border-top: 1px solid #dcdcdc;
}

.footer .btn {
    display: table-cell;
    width: 50%;
    height: 50px;
    line-height: 50px;
    text-align: center;
    cursor: pointer;
}

.footer .btn:last-child {
    display: table-cell;
    width: 50%;
    height: 50px;
    line-height: 50px;
    text-align: center;
    cursor: pointer;
    border-left: 1px solid #dcdcdc;
}

插件

    function MyDialog() {} // 组件对象
    MyDialog.prototype = {
        constructor: this,
        _initial: function() {},
        _parseTpl: function() {},
        _parseToDom:function(){},
        show:function(){},
        hide:function(){},
        css:function(){},
        ...
   }

然后就可以将插件的功能都写上,不过中间的业务逻辑,需要自己一步一步研究。

  1. 对象合并函数
// 对象合并
function extend(o, n, override) {
    for(var key in n) {
        if(n,hasOwnProperty(key) && (!o.hasOwnProperty(key) || override)) {
              o[key] = n[key];
       }
    }
    return o;
}
  1. 自定义模版引擎解释函数
function templateEngine(html, data) {
    var re = /<%([^%>]+)?%>/g,
        reExp = /(^( )?(if|for|else|switch|case|break|{|}))(.*)?/g,
        code = 'var r=[];\n',
        cursor = 0;
    var match;
    var add = function(line, js) {
        js ? (code += line.match(reExp) ? line + '\n' : 'r.push(' + line + ');\n') :
            (code += line != '' ? 'r.push("' + line.replace(/"/g, '\\"') + '");\n' : '');
        return add;
    }
    while (match = re.exec(html)) {
        add(html.slice(cursor, match.index))(match[1], true);
        cursor = match.index + match[0].length;
    }
    add(html.substr(cursor, html.length - cursor));
    code += 'return r.join("");';
    return new Function(code.replace(/[\r\t\n]/g, '')).apply(data);
}
  1. 查找class获取dom函数
// 通过class查找dom
if(!('getElementsByClass' in HTMLElement)){
    HTMLElement.prototype.getElementsByClass = function(n, tar){
        var el = [],
            _el = (!!tar ? tar : this).getElementsByTagName('*');
        for (var i=0; i<_el.length; i++ ) {
            if (!!_el[i].className && (typeof _el[i].className == 'string') && _el[i].className.indexOf(n) > -1 ) {
                el[el.length] = _el[i];
            }
        }
        return el;
    };
    ((typeof HTMLDocument !== 'undefined') ? HTMLDocument : Document).prototype.getElementsByClass = HTMLElement.prototype.getElementsByClass;
}

结合工具函数,再去实现每一个钩子函数具体逻辑结构:

// plugin.js
;(function(undefined) {
    "use strict"
    var _global;

    ...

    // 插件构造函数 - 返回数组结构
    function MyDialog(opt){
        this._initial(opt);
    }
    MyDialog.prototype = {
        constructor: this,
        _initial: function(opt) {
            // 默认参数
            var def = {
                ok: true,
                ok_txt: '确定',
                cancel: false,
                cancel_txt: '取消',
                confirm: function(){},
                close: function(){},
                content: '',
                tmpId: null
            };
            this.def = extend(def,opt,true);
            this.tpl = this._parseTpl(this.def.tmpId);
            this.dom = this._parseToDom(this.tpl)[0];
            this.hasDom = false;
        },
        _parseTpl: function(tmpId) { // 将模板转为字符串
            var data = this.def;
            var tplStr = document.getElementById(tmpId).innerHTML.trim();
            return templateEngine(tplStr,data);
        },
        _parseToDom: function(str) { // 将字符串转为dom
            var div = document.createElement('div');
            if(typeof str == 'string') {
                div.innerHTML = str;
            }
            return div.childNodes;
        },
        show: function(callback){
            var _this = this;
            if(this.hasDom) return ;
            document.body.appendChild(this.dom);
            this.hasDom = true;
            document.getElementsByClass('close',this.dom)[0].onclick = function(){
                _this.hide();
            };
            document.getElementsByClass('btn-ok',this.dom)[0].onclick = function(){
                _this.hide();
            };
            if(this.def.cancel){
                document.getElementsByClass('btn-cancel',this.dom)[0].onclick = function(){
                    _this.hide();
                };
            }
            callback && callback();
            return this;
        },
        hide: function(callback){
            document.body.removeChild(this.dom);
            this.hasDom = false;
            callback && callback();
            return this;
        },
        modifyTpl: function(template){
            if(!!template) {
                if(typeof template == 'string'){
                    this.tpl = template;
                } else if(typeof template == 'function'){
                    this.tpl = template();
                } else {
                    return this;
                }
            }
            // this.tpl = this._parseTpl(this.def.tmpId);
            this.dom = this._parseToDom(this.tpl)[0];
            return this;
        },
        css: function(styleObj){
            for(var prop in styleObj){
                var attr = prop.replace(/[A-Z]/g,function(word){
                    return '-' + word.toLowerCase();
                });
                this.dom.style[attr] = styleObj[prop];
            }
            return this;
        },
        width: function(val){
            this.dom.style.width = val + 'px';
            return this;
        },
        height: function(val){
            this.dom.style.height = val + 'px';
            return this;
        }
    }

    _global = (function(){ return this || (0, eval)('this'); }());
    if (typeof module !== "undefined" && module.exports) {
        module.exports = MyDialog;
    } else if (typeof define === "function" && define.amd) {
        define(function(){return MyDialog;});
    } else {
        !('MyDialog' in _global) && (_global.MyDialog = MyDialog);
    }
}());

到这一步,我们的插件已经达到了基础需求了。我们可以在页面这样调用:

<script type="text/template" id="dialogTpl">
    <div class="mydialog">
        <span class="close">×</span>
        <div class="mydialog-cont">
            <div class="cont"><% this.content %></div>
        </div>
        <div class="footer">
            <% if(this.cancel){ %>
            <span class="btn btn-ok"><% this.ok_txt %></span>
            <span class="btn btn-cancel"><% this.cancel_txt %></span>
            <% } else{ %>
            <span class="btn btn-ok" style="width: 100%"><% this.ok_txt %></span>
            <% } %>
        </div>
    </div>
</script>
<script src="index.js"></script>
<script>
    var mydialog = new MyDialog({
        tmpId: 'dialogTpl',
        cancel: true,
        content: 'hello world!'
    });
    mydialog.show();
</script>

插件的监听

弹出框插件我们已经实现了基本的显示与隐藏的功能。不过我们在怎么时候弹出,弹出之后可能进行一些操作,实际上还是需要进行一些可控的操作。就好像我们进行事件绑定一样,只有用户点击了按扭,才响应具体的事件。那么,我们的插件,应该也要像事件绑定一样,只有执行了某些操作的时候,调用相应的事件响应。
这种js的设计模式,被称为 订阅/发布模式,也被叫做观察者模式。我们插件中的也需要用到观察者模式,比如,在打开弹窗之前,我们需要先进行弹窗的内容更新,执行一些判断逻辑等,然后执行完成之后才显示出弹窗。在关闭弹窗之后,我们需要执行关闭之后的一些逻辑,处理业务等。这时候我们需要像平时绑定事件一样,给插件做一些“事件”绑定回调方法。
我们jquery对dom的事件响应是这样的:
$(<dom>).on("click",function(){})

我们照上面的方式设计了对应的插件响应式这样的:
mydialog.on('show',function(){})

我们需要实现一个事件机制,以到达监听的事件效果。关于自定义事件监听。参考漫谈js自定义事件、DOM/伪DOM自定义事件

最终插件

// plugin.js
;(function(undefined) {
    "use strict"
    var _global;

    // 工具函数
    // 对象合并
    function extend(o,n,override) {
        for(var key in n){
            if(n.hasOwnProperty(key) && (!o.hasOwnProperty(key) || override)){
                o[key]=n[key];
            }
        }
        return o;
    }
    // 自定义模板引擎
    function templateEngine(html, data) {
        var re = /<%([^%>]+)?%>/g,
            reExp = /(^( )?(if|for|else|switch|case|break|{|}))(.*)?/g,
            code = 'var r=[];\n',
            cursor = 0;
        var match;
        var add = function(line, js) {
            js ? (code += line.match(reExp) ? line + '\n' : 'r.push(' + line + ');\n') :
                (code += line != '' ? 'r.push("' + line.replace(/"/g, '\\"') + '");\n' : '');
            return add;
        }
        while (match = re.exec(html)) {
            add(html.slice(cursor, match.index))(match[1], true);
            cursor = match.index + match[0].length;
        }
        add(html.substr(cursor, html.length - cursor));
        code += 'return r.join("");';
        return new Function(code.replace(/[\r\t\n]/g, '')).apply(data);
    }
    // 通过class查找dom
    if(!('getElementsByClass' in HTMLElement)){
        HTMLElement.prototype.getElementsByClass = function(n){
            var el = [],
                _el = this.getElementsByTagName('*');
            for (var i=0; i<_el.length; i++ ) {
                if (!!_el[i].className && (typeof _el[i].className == 'string') && _el[i].className.indexOf(n) > -1 ) {
                    el[el.length] = _el[i];
                }
            }
            return el;
        };
        ((typeof HTMLDocument !== 'undefined') ? HTMLDocument : Document).prototype.getElementsByClass = HTMLElement.prototype.getElementsByClass;
    }

    // 插件构造函数 - 返回数组结构
    function MyDialog(opt){
        this._initial(opt);
    }
    MyDialog.prototype = {
        constructor: this,
        _initial: function(opt) {
            // 默认参数
            var def = {
                ok: true,
                ok_txt: '确定',
                cancel: false,
                cancel_txt: '取消',
                confirm: function(){},
                close: function(){},
                content: '',
                tmpId: null
            };
            this.def = extend(def,opt,true); //配置参数
            this.tpl = this._parseTpl(this.def.tmpId); //模板字符串
            this.dom = this._parseToDom(this.tpl)[0]; //存放在实例中的节点
            this.hasDom = false; //检查dom树中dialog的节点是否存在
            this.listeners = []; //自定义事件,用于监听插件的用户交互
            this.handlers = {};
        },
        _parseTpl: function(tmpId) { // 将模板转为字符串
            var data = this.def;
            var tplStr = document.getElementById(tmpId).innerHTML.trim();
            return templateEngine(tplStr,data);
        },
        _parseToDom: function(str) { // 将字符串转为dom
            var div = document.createElement('div');
            if(typeof str == 'string') {
                div.innerHTML = str;
            }
            return div.childNodes;
        },
        show: function(callback){
            var _this = this;
            if(this.hasDom) return ;
            if(this.listeners.indexOf('show') > -1) {
                if(!this.emit({type:'show',target: this.dom})) return ;
            }
            document.body.appendChild(this.dom);
            this.hasDom = true;
            this.dom.getElementsByClass('close')[0].onclick = function(){
                _this.hide();
                if(_this.listeners.indexOf('close') > -1) {
                    _this.emit({type:'close',target: _this.dom})
                }
                !!_this.def.close && _this.def.close.call(this,_this.dom);
            };
            this.dom.getElementsByClass('btn-ok')[0].onclick = function(){
                _this.hide();
                if(_this.listeners.indexOf('confirm') > -1) {
                    _this.emit({type:'confirm',target: _this.dom})
                }
                !!_this.def.confirm && _this.def.confirm.call(this,_this.dom);
            };
            if(this.def.cancel){
                this.dom.getElementsByClass('btn-cancel')[0].onclick = function(){
                    _this.hide();
                    if(_this.listeners.indexOf('cancel') > -1) {
                        _this.emit({type:'cancel',target: _this.dom})
                    }
                };
            }
            callback && callback();
            if(this.listeners.indexOf('shown') > -1) {
                this.emit({type:'shown',target: this.dom})
            }
            return this;
        },
        hide: function(callback){
            if(this.listeners.indexOf('hide') > -1) {
                if(!this.emit({type:'hide',target: this.dom})) return ;
            }
            document.body.removeChild(this.dom);
            this.hasDom = false;
            callback && callback();
            if(this.listeners.indexOf('hidden') > -1) {
                this.emit({type:'hidden',target: this.dom})
            }
            return this;
        },
        modifyTpl: function(template){
            if(!!template) {
                if(typeof template == 'string'){
                    this.tpl = template;
                } else if(typeof template == 'function'){
                    this.tpl = template();
                } else {
                    return this;
                }
            }
            this.dom = this._parseToDom(this.tpl)[0];
            return this;
        },
        css: function(styleObj){
            for(var prop in styleObj){
                var attr = prop.replace(/[A-Z]/g,function(word){
                    return '-' + word.toLowerCase();
                });
                this.dom.style[attr] = styleObj[prop];
            }
            return this;
        },
        width: function(val){
            this.dom.style.width = val + 'px';
            return this;
        },
        height: function(val){
            this.dom.style.height = val + 'px';
            return this;
        },
        on: function(type, handler){
            // type: show, shown, hide, hidden, close, confirm
            if(typeof this.handlers[type] === 'undefined') {
                this.handlers[type] = [];
            }
            this.listeners.push(type);
            this.handlers[type].push(handler);
            return this;
        },
        off: function(type, handler){
            if(this.handlers[type] instanceof Array) {
                var handlers = this.handlers[type];
                for(var i = 0, len = handlers.length; i < len; i++) {
                    if(handlers[i] === handler) {
                        break;
                    }
                }
                this.listeners.splice(i, 1);
                handlers.splice(i, 1);
                return this;
            }
        },
        emit: function(event){
            if(!event.target) {
                event.target = this;
            }
            if(this.handlers[event.type] instanceof Array) {
                var handlers = this.handlers[event.type];
                for(var i = 0, len = handlers.length; i < len; i++) {
                    handlers[i](event);
                    return true;
                }
            }
            return false;
        }
    }

    // 最后将插件对象暴露给全局对象
    _global = (function(){ return this || (0, eval)('this'); }());
    if (typeof module !== "undefined" && module.exports) {
        module.exports = MyDialog;
    } else if (typeof define === "function" && define.amd) {
        define(function(){return MyDialog;});
    } else {
        !('MyDialog' in _global) && (_global.MyDialog = MyDialog);
    }
}());

调用

var mydialog = new MyDialog({
    tmpId: 'dialogTpl',
    cancel: true,
    content: 'hello world!'
});
mydialog.on('confirm',function(ev){
    console.log('you click confirm!');
    // 写你的确定之后的逻辑代码...
});
document.getElementById('test').onclick = function(){
    mydialog.show();
}

案例Demo

参考资料

  1. 原生JavaScript插件编写指南
  2. js原型链
最新动态关注公众号:小宝说事
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,874评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,102评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,676评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,911评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,937评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,935评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,860评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,660评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,113评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,363评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,506评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,238评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,861评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,486评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,674评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,513评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,426评论 2 352