我们先来看一个最简单的例子:

(function($){

  $.fn.extend({     //把此插件添加到jQuery的原型上

    pluginName:function(){   //插件的名字

      return this.each(function(){     //遍历匹配元素的集合

        //插件要实现的功能

      });

    }

  });

})(jQuery);   //传入jQuery对象

由于jQuery是集化操作($("div")会选择多个div元素进行操作),而我们的插件编写应该一个元素对应一个配置对象,我们可以使用$.extend({},defaults,options)来为每个元素分配独立的配置对象。其中,defaults为默认配置对象,以防我们什么都没传时,插件也能正常运作。如果传入的配置对象options比较复杂,也就是说options的属性也是一个对象,那么我们可以使用深拷贝,$.extend(true,{},defaults,options)。

当然,你可以在上面例子中的each中做一个元素对应一个配置对象的操作,但是如果操作很多,那么在each中就会有很多代码,维护不方便。

因此,我们需要使用面向对象的写法。请看例子:

(function($){

  var Plugin = function(element, options){

  };

  Plugin.defaults = {    //默认配置

  };

  Plugin.prototype = {};

  $.fn.extend({     //把此插件添加到jQuery的原型上

    pluginName:function(options){   //插件的名字

      return this.each(function(){     //遍历匹配元素的集合

        var ui = $._data(this, "pluginname");    //看此元素在jQuey缓存系统中是否存有pluginname属性,它的值是一个Plugin实例对象

        if(!ui){

          var opts = $.extend(true,{}, Plugin.defaults , options);    //这里的默认配置对象一般写在Plugin实例构造中使用

          ui = new Plugin(this,opts);

          $._data(this,"pluginname", ui);    //一个元素对应一个Plugin实例对象,插件的功能,全部在Plugin对象中,因此,我们只需要改写Plugin构造函数以及它的原型,就能实现此插件的功能。

        }       

      });

    }

  });

})(jQuery);   //传入jQuery对象

Bootstrap在上面的基础上,进行了三大改造,第一:插件的无冲突处理。第二:将内部的对象构造器(类)放到了$.fn.pluginName.Constructor上,方便我们来扩展这个内部类。第三:利用事件代理自动初始化实例,目的是让用户只需要引入此插件,并按照规定写HTML,就能让此HTML实现插件的功能。举个例子:

(function($){

//------------------------------------------------------------------------内部的对象构造器(内部类)

  var Dropdown= function(element, options){

    $(element).on("click",this.toggle);    //当$("div").dropdowm({....})时,就会给div绑定一个click事件,如果点击了div,就会触发toggle方法。

  };

  Dropdown.defaults = {    //默认配置

  };

  Dropdown.prototype = {

    constructor:Dropdown,

    toggle:function(){

      ......

    }

  };

//--------------------------------------------------------------

  var old = $.fn.dropdown;    //因为需要重写此方法名,因此先把原始的保存在old变量中,当调用$.fn.dropdown.noConflict(),就会把它还原。

  $.fn.extend({     //把此插件添加到jQuery的原型上

    dropdown :function(options){   //插件的名字

      var args = [].slice.call(arguments,1);

      return this.each(function(){     //遍历匹配元素的集合

        var ui = $._data(this, "dropdown");    //看此元素在jQuey缓存系统中是否存有dropdown属性,它的值是一个Dropdown实例对象

        if(!ui){

          var opts = $.extend(true,{}, Dropdown.defaults , options);   //这里的默认配置对象,一般在Dropdown实例对象构造中使用。

          ui = new Dropdown(this,opts);

          $._data(this,"dropdown", ui);    //一个元素对应一个Dropdown实例对象,插件的功能,全部在Dropdown对象中,因此,我们只需要改写Dropdown构造函数以及它的原型,就能实现此插件的功能。

        }    

        if (typeof options == 'string' && typeof ui[options] == 'function') {  //这里添加的这段代码,是针对jQuery插件的,当我们实例化此插件后,比如:$("div").dropdown({....})后,如果想调用此插件的方法,那么,可以用$("div").dropdown(方法名, arg1,arg2....)
          ui[options].apply(ui, args);
        }    

      });

    }

  });

  $.fn.dropdown.Constructor = Dropdown;   //把内部类暴露出来,我们可以通过$.fn.dropdown.Constructor来访问,并且扩展它

  $.fn.dropdown.noConflict = function(){

    $.fn.dropdown = old;

    return this;

  }

  $(document).on("click", ".dropdown",Dropdown.prototype.toggle);    //加载完此插件后,在HTML的元素中只要有class = dropdowm,那么点击此元素,就会执行toggle方法。 

})(jQuery);   //传入jQuery对象

如果想做jQuery插件,此思想一定要掌握,面试会问,一次看不懂,多看几次就懂了,或者去看下公司里面其他人写的插件,就很容易懂了。如果大家没有很好的例子,那可以看一下我写的那个日历插件:http://www.cnblogs.com/chaojidan/p/4140725.html。在这个插件中,我是直接使用了全局变量来定义了CCalendar,这样会污染我们的全局环境,因此,我们只需要在整个代码的外面,添加一个立即执行的匿名函数,同时包装到jQuery中就行了。

(function($){

//------------------------------------------------------------------------内部的对象构造器(内部类)

  // 日历插件代码

//-------------------------------------------------------------- 

  $.fn.ccalendar = function (options) {
    return this.each(function () {
      var $this = $(this),
      data = $this.data('CCalendar'),      //这个就相当于$._data(this, "CCalendar")
      if (!data) {

        data = new CCalendar(this, options);
        $this.data('CCalendar', data );  //这个就相当于$._data(this,"CCalendar“, data);
      }
    });
  };
  $.fn.ccalendar.Constructor = CCalendar;

})(jQuery)

这样一包装之后,插件代码中的方法和变量,都变成局部的了,就不会污染全局环境了,但是在使用这个日历插件时,你就需要这样调用了:

$("input").ccalendar({   .....   });

如果大家使用的是sea.js这种模块化开发的模式:

那么只需要:

define(function(require, exports, module){

  var $ = require("./jQuery.js");

  //然后再把上面的立即执行函数中的代码放到这里。

  module.exports=CCalendar;

})

这时,你既可以用jQuery的调用方式:$("input").ccalendar({   .....   });

也可以使用sea.js的调用方式:var callendar = require("./ccalendar.js");   var date = new callendar(element, { ..... })

加油!

第四十三课:jQuery插件化的更多相关文章

  1. NeHe OpenGL教程 第四十三课:FreeType库

    转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线 ...

  2. UEditor的jQuery插件化

    UEditor本身并不依赖jQuery,但如果在项目中同时使用两者的话,可能会希望使用jQuery语法创建和获取编辑器实例.为此,需要为jQuery编写插件,代码如下: (function ($) { ...

  3. wangEditor的jQuery插件化

    wangEditor是一款优秀的Web富文本编辑器.这篇随笔中讲述的wangEditor版本是2.1.22,由于它依赖于jQuery(作者打算在第三版中取消对jQuery的依赖),那么如果能使用$(& ...

  4. UEditor的jQuery插件化 -转

    UEditor本身并不依赖jQuery,但如果在项目中同时使用两者的话,可能会希望使用jQuery语法创建和获取编辑器实例.为此,需要为jQuery编写插件,代码如下: (function ($) { ...

  5. 潭州课堂25班:Ph201805201 django 项目 第四十三课 后台 用户管理前后功能实现 (课堂笔记)

    用户的展示,编辑,删除, 把用户显示出来,用户名,员工(是,否), 超级用户(是, 否) 活跃状态,(非活跃示为删除) 在前台要显示该用户所属的用户组,在前台代码中是调用类的属性,所以在 user 的 ...

  6. python第四十三课——封装性

    1.面向对象的三大特性:封装性.继承性.多态性 封装: 封装使用的领悟: 1).生活层面:食品.快递.计算机.明星... 2).计算机层面: ①.模块.类.函数... ②.属性数据的封装与隐藏 权限修 ...

  7. Android 插件化开发(四):插件化实现方案

    在经过上面铺垫后,我们可以尝试整体实现一下插件化了.这里我们先介绍一下最简单的实现插件化的方案. 一.最简单的插件化实现方案 最简单的插件化实现方案,对四大组件都是适用的,技术面涉及如下: 1). 合 ...

  8. Gradle 1.12用户指南翻译——第四十三章. 构建公告插件

    本文由CSDN博客貌似掉线翻译,其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Githu ...

  9. 【技术分享会】 @第四期 JQuery插件

    本讲内容 JavaScript JQuery JQuery插件 实例 JavaScript 前端开发工程师必须掌握的三种技能 描述内容的HTML 描述网页样式的CSS 描述网页行为的JavaScrip ...

随机推荐

  1. Android 系统架构

    Android 系统从下至上分为四层:Linux 内核.Android 核心库及Android 运行时环境(Android Runtime). 应用程序框架以及应用程序等. Linux 内核(Linu ...

  2. Merge compare columns when null

    Key words: merge compare columns when we contact merge sql in ETL, When we update some columns we sh ...

  3. FPGA 设计技巧

    1.  资源共享的应用限制在同一个module里 这样 综合工具才能最大限度地发挥其资源共 享综合作用 2.  尽可能将Critical path上所有相关逻辑放在同一个module里 这样 综合工具 ...

  4. html之大白

    <!doctype html> <html> <head> <meta charset="utf-8"> <title> ...

  5. windows下如何安装jira

    ---恢复内容开始--- 准备工作: 1.安装jdk,详细不在介绍 2.建立jira帐号:https://id.atlassian.com/login?application=mac&cont ...

  6. <转>如何进行code review

    转自: http://pm.readthedocs.org/zh_CN/latest/codereview/howto.html 如何进行code review? code reivew是保障代码质量 ...

  7. Socurce Insight 快捷键

    1. 高亮当前选中的的 变量 Shift + F8

  8. You are note Hk

    直接打开是forbidden  最后一句话提示 于是改包 第二次还是修改包 Mozilla/5.0 (MSIE 7.0; Windows NT 6.0;.NET CLR 9.9)

  9. andorid 自定义seekbar

    效果如图: <?xml version="1.0" encoding="utf-8"?> <resources> <style n ...

  10. 错误异常 (1)Android Studio错误提示:Gradle project sync failed. Basic functionality (eg. editing, debugging) will not work properly

    [已解决]Android Studio错误提示:Gradle project sync failed. Basic functionality (eg. editing, debugging) wil ...