先看bootstrap-tab.js的结构

var Tab = function ( element ) {} //构造器
Tab.prototype ={} //构造器的原型
$.fn.tab = function ( option ){} //jQuery原型上自定义的方法
$.fn.tab.Constructor = Tab //重写jQuery原型上的自定义方法的构造器名
$(function () {}) //初始化

HTML结构

<ul class="nav nav-tabs">
<li class="active"><a href="#home" data-toggle="tab" >首页</a></li>
<li><a href="#profile" data-toggle="tab">介绍</a></li>
<li><a href="#messages" data-toggle="tab">消息</a></li>
<li><a href="#settings" data-toggle="tab">设置</a></li> </ul> <div class="tab-content">
<div class="tab-pane active" id="home">1</div>
<div class="tab-pane" id="profile">2</div>
<div class="tab-pane" id="messages">3</div>
<div class="tab-pane" id="settings">4</div>
</div>

先从初始化开始

/*
* 初始化
* 给拥有data-toggle='tab'属性的标签绑定click事件
* */
$(function () {
$('body').on('click.tab.data-api', '[data-toggle="tab"], [data-toggle="pill"]', function (e) {
e.preventDefault()
$(this).tab('show')//向jQuery原型方法tab传入参数show,应该执行show方法。this为拥有data-toggle属性的a标签对象
})
})

让body监听a标签的click事件,并且阻止其冒泡,调用了jQuery的原型方法tab。

 /*
* jQuery原型上自定义的方法
* */
$.fn.tab = function ( option ) {
return this.each(function () {
var $this = $(this)
, data = $this.data('tab')
if (!data) $this.data('tab', (data = new Tab(this)))//实例化构造器
if (typeof option == 'string') data[option]()//执行option的方法
})
}

这个结构跟之前的几个插件差不多,不过最后它是执行了show方法的,进入构造器

var Tab = function ( element ) {
this.element = $(element)
}

将对象封装成构造器实例的一个属性。接下来我们执行show方法

show: function () {
var $this = this.element
, $ul = $this.closest('ul:not(.dropdown-menu)')//找到最近的不是dropdown类的ul元素
, selector = $this.attr('data-target')
, previous
, $target
//获取与a标签对应的内容id
if (!selector) {
selector = $this.attr('href')
selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
}
//如果一直点击自己,这句之后的代码将不执行
if ( $this.parent('li').hasClass('active') ) return previous = $ul.find('.active a').last()[0]//获得拥有active类的原生a标签对象,即上一次高亮的节点 $this.trigger({
type: 'show'
, relatedTarget: previous
}) $target = $(selector)//获得内容节点的jQueryDOM对象 this.activate($this.parent('li'), $ul)//tab页切换 this.activate($target, $target.parent(), function () {
$this.trigger({
type: 'shown'
, relatedTarget: previous
}) })

根据标签a的data-target属性或是其href找到对应的id的dom节点。最后我们执行了两次activate方法,其实大概猜的出来,一个方法控制a标签的高亮显示,另一个控制dom节点的显示与隐藏

/*
* 控制显示与隐藏
* 参数主要形式是这样的,一个是可以拥有active类的元素,另一个是其父类。
* 先找到父类下来所有拥有active类的节点,删除它们的active,并将当前节点,也就是第一个参数加上active类。
*
* */
, activate: function ( element, container, callback) { var $active = container.find('> .active')
, transition = callback
&& $.support.transition//需要引入其他js文件,这里没有引入。
&& $active.hasClass('fade')
console.log($active[0])
function next() {
$active
.removeClass('active')
.find('> .dropdown-menu > .active')
.removeClass('active') element.addClass('active') if (transition) {
element[0].offsetWidth // reflow for transition
element.addClass('in')
} else {
element.removeClass('fade')
} if ( element.parent('.dropdown-menu') ) {
element.closest('li.dropdown').addClass('active')
} callback && callback()
} transition ?
$active.one($.support.transition.end, next) :
next() $active.removeClass('in')
}

逻辑比较简单,一般遇到一个显示其他同类不显示的情况,基本都是这一套方法,增删class来达到要求。但对于jQuery操作css完成class切换与javascript原生方法相比,哪个效率会更好一点呢?也希望有经验的园友给予解答。

另外是部分css,表现了两块active类的作用

/*tab页切换时的active作用*/
.nav-tabs > .active > a,
.nav-tabs > .active > a:hover,
.nav-tabs > .active > a:focus {
color: #555555;
cursor: default;
background-color: #ffffff;
border: 1px solid #ddd;
border-bottom-color: transparent;//将底部边框变透明
} /*dom*/ .tab-content > .tab-pane,
.pill-content > .pill-pane {
display: none; //原来为隐藏
} .tab-content > .active,
.pill-content > .active {
display: block; //加入active ,变为显示
}

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

bootstrap插件学习-bootstrap.tab.js的更多相关文章

  1. bootstrap插件学习-bootstrap.dropdown.js

    bootstrap插件学习-bootstrap.dropdown.js 先看bootstrap.dropdown.js的结构 var toggle = '[data-toggle="drop ...

  2. bootstrap插件学习-bootstrap.modal.js

    bootstrap插件学习-bootstrap.modal.js 先从bootstrap.modal.js的结构看起. function($){ var Modal = function(){} // ...

  3. bootstrap插件学习-bootstrap.tab.js(读码)

    先看bootstrap-tab.js的结构 var Tab = function ( element ) {} //构造器 Tab.prototype ={} //构造器的原型 $.fn.tab = ...

  4. bootstrap插件学习-bootstrap.typehead.js

    先看bootstrap.typehead.js的结构 var Typeahead = function ( element, options ){} //构造器 Typeahead.prototype ...

  5. bootstrap插件学习-bootstrap.carousel.js

    先看bootstrap.carousel.js的结构 var Carousel = function (element, options){} //构造器 Carousel.prototype = { ...

  6. bootstrap插件学习-bootstrap.collapse.js

    先看bootstrap.collapse.js的结构 var Collapse = function ( element, options ){} // 构造器 Collapse.prototype ...

  7. bootstrap插件学习-bootstrap.alert.js

    我们先看bootstrap.alert.js的结构 var dismiss = '[data-dismiss="alert"]' //自定义属性 Alert = function ...

  8. bootstrap插件学习-bootstrap.button.js

    先看bootstrap.button.js的结构 var Button = function ( element, options ){} //构造器 Button.prototype = {} // ...

  9. bootstrap插件学习-bootstrap.popover.js

    先看bootstrap.popover.js的结构 var Popover = function ( element, options ){} //构造器 Popover.prototype = {} ...

随机推荐

  1. Java并发包中Lock的实现原理

    1. Lock 的简介及使用 Lock是java 1.5中引入的线程同步工具,它主要用于多线程下共享资源的控制.本质上Lock仅仅是一个接口(位于源码包中的java\util\concurrent\l ...

  2. ThinkPHP执行原生sql,实现一些复杂的业务需求

    1)事情起因:写php的同事做了社区消息接口,主要返回几个方面的消息,如我的主贴的点赞.我的层帖的点赞.我的主贴的评论.我的评论的评论, 数据因为关联了5张以上的表,返回的格式不一: 如原来的thin ...

  3. Revit API 楼板开洞

    start [Transaction(TransactionMode.Manual)] [Regeneration(RegenerationOption.Manual)] , , ) *  / , - ...

  4. ZZmsvcprt.lib(MSVCP90.dll) : error LNK2005:已经在libcpmtd.lib(xmutex.obj) 中定义 .的分析解决办法 (转)

    很久没有写程式设计入门知识的相关文章了,这篇文章要来谈谈程式库 (Library) 连结,以及关于 MSVC 与 CRT 之间的种种恩怨情仇. 如果你使用的作业系统是 Linux.Mac 或其他非 W ...

  5. atmega32u4制作arduino leonardo最小系统

    转载请注明:@小五义http://www.cnblogs.com/xiaowuyiQQ群:64770604 一.leonardo最小系统 关于leonardo这里不再介绍,直接上最小系统原理图,该系统 ...

  6. Scala 深入浅出实战经典 第42讲:scala 泛型类,泛型函数,泛型在spark中的广泛应用

    王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-64讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 ...

  7. Python将文本生成二维码

    #coding:utf-8 ''' Python生成二维码 v1.0 主要将文本生成二维码图片 测试一:将文本生成白底黑字的二维码图片 测试二:将文本生成带logo的二维码图片 ''' __autho ...

  8. iOS开发——项目实战总结&经典错误一

    经典错误一 No architectures to compile for (ONLY_ACTIVE_ARCH=YES, active arch=armv7, VA 运行报错 出现的原因:armv7s ...

  9. Xcode报错:“Your build settings specify a provisioning profile with the UUID..... however, no such provisioning profile was found”

    运行环境: Xcode5 & 5.0及以上版本 对工程进行Archive打包的时候出现如下错误   问题描述: Code Sign error: No matching provisionin ...

  10. OBS---环境配置之#include <D3DX10.h>报错

    一.先贴错误 因为这个笔记主要记录我如何整好这个OBS源码环境的,给需要的童鞋一个参考 1.1.#include <D3DX10.h>  报错 没有这个 解决方案:把2,3先解决了就水到渠 ...