先看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()), {
width: 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对象点就能点出上述的方法了,很方便。

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

05-08 08:35
查看更多