bootstrap插件学习-bootstrap.tooltip.js

先看bootstrap-tooltip.js的结构

var Tooltip = function ( element, options ){} // 构造器
Tooltip.prototype ={} //构造器的原型
$.fn.tooltip = function ( option ) {} //jQuery原型上自定义的方法
$.fn.tooltip.Constructor = Tooltip //重置jQuery原型方法tooltip的构造器名
$.fn.tooltip.defaults ={} // 默认参数

因为tooltip插件的使用比较多,调用者比较杂,源码中没有给出初始化的步骤,在看源码的过程中,我们手动添加初始化。

<p class="muted">
    “这是我的第一次英文访问,很抱歉它不够严谨,但是我不得不这么做,不只因为采访时间限制,更因为我面对的是卡梅隆,这个人喜爱挑战、从无畏惧,他也希望别人如此,他可以原谅不完美,但他无法接受一个人不去努力接近自己的极限。” ——
    <a rel="tooltip" href="#" data-original-title="柴静始终站在离新闻最近的地方,她以她的犀利和敏锐、坚定与坚持,最终历练成为一名优秀的新闻工作者。 ">柴静</a>
    《看见》专访
    <a id="a1" rel="tooltip" href="#" data-original-title="1954年8月16日生于加拿大的著名电影导演,擅长拍摄动作片以及科幻电影。">卡梅隆</a>
</p>
$("#a1").tooltip();

在script标签部分加入以上代码,我们可以鼠标移入'卡梅隆'时显示提示框了。

下面开始,进入jQuery原型上的自定义方法tooltip中

/*
  * jQuery原型上自定义的方法
  * */
  $.fn.tooltip = function ( option ) {
    return this.each(function () {
      var $this = $(this)
        , data = $this.data('tooltip')
        , options = typeof option == 'object' && option
      if (!data) $this.data('tooltip', (data = new Tooltip(this, options)))//实例化构造器
      if (typeof option == 'string') data[option]()//支持传入方法名参数,执行该方法。
    })
  }

跟之前的几个插件类似,进入构造器中。

/*
  * 构造器
  * */
  var Tooltip = function ( element, options ) {
    this.init('tooltip', element, options)//实例化直接调用原型的init方法
  }

进入原型上的init方法

init: function ( type, element, options ) {
      var eventIn
        , eventOut

      this.type = type
      this.$element = $(element)
      this.options = this.getOptions(options)
      this.enabled = true      if (this.options.trigger != 'manual') { //选择事件名
        eventIn  = this.options.trigger == 'hover' ? 'mouseenter' : 'focus'
        eventOut = this.options.trigger == 'hover' ? 'mouseleave' : 'blur'
        //绑定事件

        this.$element.on(eventIn, this.options.selector, $.proxy(this.enter, this))//其中有false,达到阻止冒泡的功能
        this.$element.on(eventOut, this.options.selector, $.proxy(this.leave, this))
      }

      this.options.selector ?
        (this._options = $.extend({}, this.options, { trigger: 'manual', selector: '' })) :
        this.fixTitle()//没有默认参数执行
    }

init方法一般执行初始化的步骤,其中的getOptions方法

/*
    * 添加默认项
    * */
  , getOptions: function ( options ) {
      options = $.extend({}, $.fn[this.type].defaults, options, this.$element.data())
      if (options.delay && typeof options.delay == 'number') {
        options.delay = {
          show: options.delay
        , hide: options.delay
        }
      }

      return options
    }

这里我们看到init方法为点击标签绑定了两种事件mouseenter和mouseleave,它们与mouseover和mouseout的区别:

不论鼠标指针穿过被选元素或其子元素,都会触发 mouseover 事件。

只有在鼠标指针穿过被选元素时,才会触发 mouseenter 事件。

绑定完事件之后,如果是无参数时,我们执行fixTitle方法

/*
    * 将被点击标签的title属性值转给data-original-title属性,最后删除title属性
    * */
  , fixTitle: function () {
      var $e = this.$element
      if ($e.attr('title') || typeof($e.attr('data-original-title')) != 'string') {
        $e.attr('data-original-title', $e.attr('title') || '').removeAttr('title')
      }
    }

为data-original-title属性赋值,为以后将该值加如提示框中做准备。

当鼠标移入时,触发mouseenter事件,进入绑定好的enter方法中

/*
    * 目的调用show方法
    * */
  , enter: function ( e ) {
      /*
      * 以下方法执行了jQuery原型上的tooltip方法。通过each遍历,获取data对象中的tooltip属性:即该对象的jQuery对象
      * 感觉这个写法蛋疼
      * */
      var self = $(e.currentTarget)[this.type](this._options).data(this.type)//e.currentTarget获取当前点击对象

      if (!self.options.delay || !self.options.delay.show) {
        self.show()//鼠标移上去执行show方法
      } else {
        self.hoverState = 'in'
        setTimeout(function() {
          if (self.hoverState == 'in') {
            self.show()
          }
        }, self.options.delay.show)
      }
    }

纵观这个方法,其主要目的是为了调用show方法。

/*
    * 显示提示框
    * */
  , show: function () {
      var $tip
        , inside
        , pos
        , actualWidth
        , actualHeight
        , placement
        , tp

      if (this.hasContent() && this.enabled) {
        $tip = this.tip()//获取提示框的jQuery对象
        this.setContent()//给提示框赋值,初始化提示框的样式

        if (this.options.animation) {
          $tip.addClass('fade')//提示框拥有运动效果,加入fade类
        }

        placement = typeof this.options.placement == 'function' ?
          this.options.placement.call(this, $tip[0], this.$element[0]) :
          this.options.placement//控制提示框的显示方位,默认为top

        inside = /in/.test(placement)//是否是in

        $tip
          .remove()
          .css({ top: 0, left: 0, display: 'block' })
          .appendTo(inside ? this.$element : document.body)
        /*先将提示框从文档中删除,获取其返回对象,即本身,加入样式,最后再重新插回文档中,这么写主要考虑in的情况
        * 符合一种操作规范,如果添加样式,操作dom过于复杂,可以先将节点从dom中取出,经过一系列装饰后,再加入dom中,第一
        * 便于浏览器渲染,二来也符合dom操作规范
        * */
        pos = this.getPosition(inside)//获取被点击对象的位置和本身尺寸
        //获取提示框的宽度和高度
        actualWidth = $tip[0].offsetWidth
        actualHeight = $tip[0].offsetHeight
        //我们拿top举例,在a标签上方,left方向,对着a标签居中。
        switch (inside ? placement.split(' ')[1] : placement) {
          case 'bottom':
            tp = {top: pos.top + pos.height, left: pos.left + pos.width / 2 - actualWidth / 2}
            break
          case 'top':
            tp = {top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2}
            break
          case 'left':
            tp = {top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth}
            break
          case 'right':
            tp = {top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width}
            break
        }

        $tip
          .css(tp)
          .addClass(placement)//配合内部的div实现小三角效果
          .addClass('in')//半透明效果
      }
    }

show方法是这个插件的核心方法,与show相关的几个方法,tip()方法,获得提示框模版,setContent()为提示框赋值,hasContent()中调用getTitle()方法获取被点击标签的获取data-original-title属性的值。逻辑不复杂,插件相关的方法调用比较多,耐心点看。

/*
    * 调用getTitle方法
    * */
  , hasContent: function () {
      return this.getTitle()
    }
/*
    * 获取data-original-title属性的值
    * */
  , getTitle: function () {
      var title
        , $e = this.$element
        , o = this.options

      title = $e.attr('data-original-title')
        || (typeof o.title == 'function' ? o.title.call($e[0]) :  o.title)
      title = (title || '').toString().replace(/(^s*|s*$)/, "")//将title中的|去除,并且将|两边的文字合并

      return title
    }

tip方法

/*
    * 返回提示框模版的jQuery对象
    * */
  , tip: function () {
      return this.$tip = this.$tip || $(this.options.template)
    }

为提示框赋内容

/*
* 往提示框中添加title信息,初始化提示框的样式
* */
  , setContent: function () {
      var $tip = this.tip()
      $tip.find('.tooltip-inner').html(this.getTitle())//添加内容信息
      $tip.removeClass('fade in top bottom left right')
    }

调用getPosition方法,获得对象的位置和尺寸

/*
    * 获取对象的高度和宽度,和偏移值(left和top)
    * */
  , getPosition: function (inside) {
      return $.extend({}, (inside ? {top: 0, left: 0} : this.$element.offset()), {
         this.$element[0].offsetWidth
      , height: this.$element[0].offsetHeight
      })
    }

offsetWidth和offsetHeight是javascript元素的属性。表示内容高度+内边距+边框,内容宽度+内边距+边框。

对于提示框的显示,默认在被点击标签的上方。相对于被点击标签居中。

当鼠标移开时,触发原型上的leave方法

/*
    * 调用hide方法
    * */
  , leave: function ( e ) {
      /*
      * 与enter中一样
      * */
      var self = $(e.currentTarget)[this.type](this._options).data(this.type)

      if (!self.options.delay || !self.options.delay.hide) {
        self.hide()
      } else {
        self.hoverState = 'out'
        setTimeout(function() {
          if (self.hoverState == 'out') {
            self.hide()
          }
        }, self.options.delay.hide)
      }
    }

进入核心方法hide

hide: function () {
      var that = this
        , $tip = this.tip()

      $tip.removeClass('in')
      /*
      * 定义了提示框消失的特效,要显示这个特效,需要引入其他js,这先不讨论
      * */
      function removeWithAnimation() {
        var timeout = setTimeout(function () {
          $tip.off($.support.transition.end).remove()
        }, 500)

        $tip.one($.support.transition.end, function () {
          clearTimeout(timeout)
          $tip.remove()
        })
      }

      $.support.transition && this.$tip.hasClass('fade') ?
        removeWithAnimation() :
        $tip.remove()//直接删除提示框
    }

我们没有引入相关js时,提示框直接删除了。

有兴趣的朋友可以研究一下bootstrap.tooltip插件对于提示框样式的编写,也挺不错的。最后插件还提供了几个方法,便于我们以后扩展。

/*
    * 开启提示框功能
    * */
  , enable: function () {
      this.enabled = true
    }

/*
    * 禁用提示框功能
    * */
  , disable: function () {
      this.enabled = false
    }
    /*开启/禁用切换*/
  , toggleEnabled: function () {
      this.enabled = !this.enabled
    }
    /*提示框显示隐藏切换*/
  , toggle: function () {
      this[this.tip().hasClass('in') ? 'hide' : 'show']()
    }

validate: function () {
      if (!this.$element[0].parentNode) {
        this.hide()
        this.$element = null
        this.options = null
      }
    }

使用的时候,直接用jQuery取得一个dom对象点就能点出上述的方法了,很方便。

 内容不多,时间刚好,以上是我的一点读码体会,如有错误,请指出,大家共通学习。