Bootstrap滚动监控器
前面的话
滚动监听插件是用来根据滚动条所处的位置来自动更新导航项的。滚动导航条下面的区域并关注导航项的变化,下拉菜单中的条目也会自动高亮显示。本文将详细介绍Bootstrap滚动监控器
基本用法
滚动监听插件是根据滚动的位置自动更新导航条中相应的导航项的,该插件可自动检测到达哪个位置了,然后在需要高亮的菜单父元素上加了一个active样式
如果导航里有下拉菜单,并且滚动区域的内容到达下拉菜单子项所对应的区域,除了子菜单高亮之外,子菜单的父元素(dropdown按钮)也会高亮
在平时使用的过程中,滚动监听一般有两种用法,一种是固定一个元素的高度,进行滚动,然后对相应的菜单进行高亮显示;另外一种是对整个页面(body)进行滚动监听。两种方式的用法一样,都需要有如下3个步骤:
1、设置滚动容器,即在所要监听的元素上设置data-target="#selector" data-spy="scroll"属性
2、设置菜单链接容器,该容器的id(或样式)和data-target属性所对应的选择符要一致
3、在菜单容器内,必须有.nav样式的元素,并且在其内容有li元素,li内包含的a元素也是可以侦测高亮的菜单链接,即符合.nav li > a这种选择符的条件
4、无论何种实现方式,滚动监听都需要被监听的组件是 position: relative;
即相对定位方式
【固定元素高度】
<div id="myNavbar" class="navbar navbar-default navbar-fixed-top" role="navigation" style="position:relative">
<ul class="nav navbar-nav">
<li><a href="#html" tabindex="-1">HTML</a></li>
<li><a href="#css" tabindex="-1">CSS</a></li>
<li><a href="#javascript" tabindex="-1">javascript</a></li>
</ul>
</div> <div data-spy="scroll" data-target="#myNavbar" style="margin-top:150px;height:250px;overflow:auto;position:relative">
<h4 id="html">Html</h4>
<p>Html内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p>
<h4 id="css">CSS</h4>
<p>CSS内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p>
<h4 id="javascript">javascript</h4>
<p>javascript内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p>
</div>
【body元素】
<body data-spy="scroll" data-target="#myNavbar" style="height:300px;position:relative"> <div id="myNavbar" class="navbar navbar-default navbar-fixed-top" role="navigation">
<ul class="nav navbar-nav">
<li><a href="#html" tabindex="-1">HTML</a></li>
<li><a href="#css" tabindex="-1">CSS</a></li>
<li><a href="#javascript" tabindex="-1">javascript</a></li>
</ul>
</div> <h4 id="html" style="margin-top:150px">Html</h4>
<p>Html内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p>
<h4 id="css">CSS</h4>
<p>CSS内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p>
<h4 id="javascript">javascript</h4>
<p>javascript内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p> </body>
JS调用
在Bootstrap框架中,使用JavaScript方法触发滚动监控器相对来说较为简单,只需要指定两个容器的名称即可
<div id="myNavbar" class="navbar navbar-default navbar-fixed-top" role="navigation">
<ul class="nav navbar-nav">
<li><a href="#html" tabindex="-1">HTML</a></li>
<li><a href="#css" tabindex="-1">CSS</a></li>
<li><a href="#javascript" tabindex="-1">javascript</a></li>
</ul>
</div>
<div id="scrollspy" style="margin-top:150px;height:250px;overflow:auto;position:relative">
<h4 id="html">Html</h4>
<p>Html内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p>
<h4 id="css">CSS</h4>
<p>CSS内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p>
<h4 id="javascript">javascript</h4>
<p>javascript内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p>
</div>
<script>
$('#scrollspy').scrollspy({ target: '#myNavbar' })
</script>
方法
当使用滚动监听插件的同时在 DOM 中添加或删除元素后,需要像下面这样调用此刷新( refresh) 方法
$('[data-spy="scroll"]').each(function () {
var $spy = $(this).scrollspy('refresh')
})
要注意的是,这种refresh方法只对声明式用法有效。如果使用的是JS触发,并且需要刷新DOM,则需要重新应用该插件;或者从data-scrollspy属性上获取该实例,然后再调用refresh方法
【参数】
可以通过 data 属性或 JavaScript 传递参数。对于 data 属性,其名称是将参数名附着到 data-
后面组成,例如 data-offset=""
滚动监控提供了一个offset参数,此参数默认值为10。默认情况下,滚动内容距离滚动容器10px以内的话,就高亮显示所对应的菜单项
【事件】
滚动监控也支持事件的订阅和触发功能,目前只支持一个activate事件
activate.bs.scrollspy 每当一个新条目被激活后都将由滚动监听插件触发此事件。
<div id="myNavbar" class="navbar navbar-default navbar-fixed-top" role="navigation">
<ul class="nav navbar-nav">
<li><a href="#html" tabindex="-1">HTML</a></li>
<li><a href="#css" tabindex="-1">CSS</a></li>
<li><a href="#javascript" tabindex="-1">javascript</a></li>
</ul>
</div>
<div id="scrollspy" data-spy="scroll" data-target="#myNavbar" data-offset="0" style="margin-top:150px;height:250px;overflow:auto;position;relative">
<h4 id="html">Html</h4>
<p>Html内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p>
<h4 id="css">CSS</h4>
<p>CSS内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p>
<h4 id="javascript">javascript</h4>
<p>javascript内容</p>
<br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p><br><p>...</p>
</div>
<script>
$(function(){
$("#myNavbar").on('activate.bs.scrollspy',function(e){
$(e.target).siblings().css('outline','none')
.end().css('outline','1px solid black');
})
})
</script>
JS源码
【1】IIFE
使用立即调用函数,防止插件内代码外泄,从而形成一个闭环,并且只能从jQuery的fn里进行扩展
+function ($) {
//使用es5严格模式
'use strict';
//
}(window.jQuery);
【2】初始设置
function ScrollSpy(element, options) {
this.$body = $(document.body)
//判断滚动容器是否是body,如果是则使用window,如果不是则使用该元素本身
this.$scrollElement = $(element).is(document.body) ? $(window) : $(element)
//将默认值和传进来的options参数合并,后者优先级高
this.options = $.extend({}, ScrollSpy.DEFAULTS, options)
//如果option里设置了target,即data-target有值,则优先使用
//如果没有,则查找通过.nav样式的子元素,即.nav样式内的li子元素内的a链接,作为菜单容器
this.selector = (this.options.target || '') + ' .nav li > a'
this.offsets = []
this.targets = []
//高亮显示的菜单
this.activeTarget = null
this.scrollHeight = 0
//给滚动容器绑定滚动事件
this.$scrollElement.on('scroll.bs.scrollspy', $.proxy(this.process, this))
//计算当前页面内所有滚动容器内的id集合和每个id元素距离浏览器顶部的像素距离
this.refresh()
//开始正式处理
this.process()
}
//版本是3.3.7
ScrollSpy.VERSION = '3.3.7'
//默认值为offset:10
ScrollSpy.DEFAULTS = {
offset: 10
}
【3】插件核心代码
//获取滚动容器的滚动高度
ScrollSpy.prototype.getScrollHeight = function () {
//获取特定滚动容器的滚动高度,如果没有则获取body元素的滚动高度
return this.$scrollElement[0].scrollHeight || Math.max(this.$body[0].scrollHeight, document.documentElement.scrollHeight)
} ScrollSpy.prototype.refresh = function () {
var that = this
var offsetMethod = 'offset'
var offsetBase = 0 this.offsets = []
this.targets = []
this.scrollHeight = this.getScrollHeight() if (!$.isWindow(this.$scrollElement[0])) {
offsetMethod = 'position'
offsetBase = this.$scrollElement.scrollTop()
} this.$body
.find(this.selector)
.map(function () {
var $el = $(this)
var href = $el.data('target') || $el.attr('href')
var $href = /^#./.test(href) && $(href)
//返回一个二维数组,每个滚动容器内的id对象到页面顶部的距离以及高亮菜单容器里所对应的href值
return ($href
&& $href.length
&& $href.is(':visible')
&& [[$href[offsetMethod]().top + offsetBase, href]]) || null
})
.sort(function (a, b) { return a[0] - b[0] })
.each(function () {
//收集所有的偏移值,也就是距离top的距离
that.offsets.push(this[0])
//收集菜单容器里的所有href值,也就是滚动容器里的id值
that.targets.push(this[1])
})
} ScrollSpy.prototype.process = function () {
//获取滚动容器的scrollTop,再加上设置的offset值
var scrollTop = this.$scrollElement.scrollTop() + this.options.offset
//获取滚动高度
var scrollHeight = this.getScrollHeight()
//最大滚动=总scrollheight + 设置的offset值 - 设置高度height
var maxScroll = this.options.offset + scrollHeight - this.$scrollElement.height()
var offsets = this.offsets
var targets = this.targets
var activeTarget = this.activeTarget
var i
if (this.scrollHeight != scrollHeight) {
this.refresh()
}
//如果超过了最大滚动,说明已经滚动到底了
if (scrollTop >= maxScroll) {
//如果最后一个元素还没有高亮,则设置最后一个元素高亮
return activeTarget != (i = targets[targets.length - 1]) && this.activate(i)
} if (activeTarget && scrollTop < offsets[0]) {
this.activeTarget = null
return this.clear()
}
//倒序遍历所有元素的offset
for (i = offsets.length; i--;) {
//如果i元素不等于当前高亮元素
activeTarget != targets[i]
//滚动高度 大于 i元素的offsets
&& scrollTop >= offsets[i]
//i+1元素不存在,或者i+1元素大于滚动高度
&& (offsets[i + 1] === undefined || scrollTop < offsets[i + 1])
//则设置i为高亮元素
&& this.activate(targets[i])
}
} //设置高亮菜单元素
ScrollSpy.prototype.activate = function (target) {
//赋值实例属性
this.activeTarget = target this.clear()
//查找菜单中符合[data-target+"#' + 所高亮元素的id + '"]属性的元素
//或者href值是#' + 所高亮元素的id + '的话,也可以
var selector = this.selector +
'[data-target="' + target + '"],' +
this.selector + '[href="' + target + '"]'
//查找父元素li,然后添加active高亮样式
var active = $(selector)
.parents('li')
.addClass('active')
//如果li元素的父元素有dropdown-menu样式,则表示是一个dropdown下拉菜单
if (active.parent('.dropdown-menu').length) {
active = active
.closest('li.dropdown')
//则需要给dropdown的li元素也加上active高亮样式
.addClass('active')
}
//触发自定义高亮事件
active.trigger('activate.bs.scrollspy')
} //删除其他高亮元素的active样式
ScrollSpy.prototype.clear = function () {
$(this.selector)
.parentsUntil(this.options.target, '.active')
.removeClass('active')
}
【4】jQuery插件定义
function Plugin(option) {
//根据选择器,遍历所有符合规则的元素
return this.each(function () {
var $this = $(this)
//获取自定义属性bs.scrollspy的值
var data = $this.data('bs.scrollspy')
//如果option参数是对象,则作为ScrollSpy的参数传入
var options = typeof option == 'object' && option
//如果值不存在,则将ScrollSpy实例设置为bs.scrollSpy值
if (!data) $this.data('bs.scrollspy', (data = new ScrollSpy(this, options)))
//如果option传递了string,则表示要执行某个方法
if (typeof option == 'string') data[option]()
})
} var old = $.fn.scrollspy
//保留其他库的$.fn.scrollspy代码(如果定义的话),以便在noConflict之后可以继续使用该老代码
$.fn.scrollspy = Plugin
//重设插件构造器,可以通过该属性获取插件的真实类函数
$.fn.scrollspy.Constructor = ScrollSpy
【5】防冲突处理
$.fn.scrollspy.noConflict = function () {
//恢复以前的旧代码
$.fn.scrollspy = old
//将$.fn.scrollspy.noConflict()设置为Bootstrap的Scrollspy插件
return this
}
【6】绑定触发事件
$(window).on('load.bs.scrollspy.data-api', function () {
//遍历所有符合条件的滚动容器
$('[data-spy="scroll"]').each(function () {
var $spy = $(this)
//执行scrollspy插件,并传入滚动容器上设置的自定义参数(data-开头)
Plugin.call($spy, $spy.data())
})
})
Bootstrap滚动监控器的更多相关文章
- Bootstrap滚动监听(Scrollspy)插件
Bootstrap滚动监听(Scrollspy)插件,即自动更新导航插件,会根据滚动条的位置自动更新对应的导航目标
- Bootstrap滚动监听
滚动监听(Scrollspy)插件,即自动更新导航插件,会根据滚动条的位置自动更新对应的导航目标.其基本的实现是随着您的滚动,基于滚动条的位置向导航栏添加 .active class. <!DO ...
- bootstrap滚动监视原理实现
最近在公司实习,刚好写了一个静态的网页,用到了滚动监视,就自己写了个监视,话不多说直接进入正题 $(function () { var $root = $("html,body") ...
- bootstrap 滚动监听 标签页 和 工具提示
标签 <div class="container"> <h4>Tabs</h4> <ul id="myTab ...
- Bootstrap学习目录
前面的话 Bootstrap与CSS的关系,类似于javascript与jQuery的关系,原理与应用的关系.只是jQuery不再火爆,而Bootstrap依然火热,它在github有着超过100万的 ...
- boostrap插件
第一章:模态弹出框 一.导入JavaScript插件 Bootstrap的JavaScript插件可以单独导入到页面中,也可以一次性导入到页面中.因为在Bootstrap中的JavaScript插件都 ...
- Bootstrap_Javascript_滚动监视器
滚动监控器是Bootstrap提供的非常实用的JavaScript插件,被广泛应用到Web开发中.其表现形式是: 1.当用户鼠标滚动时,滚动条的位置会自动更新导航条中相应的导航项. 2.用户拖动滚动条 ...
- bootstrap 支持的JavaScript插件
一次性导入: Bootstrap提供了一个单一的文件,这个文件包含了Bootstrap的所有JavaScript插件,即bootstrap.js(压缩版本:bootstrap.min.js). 具体使 ...
- Bootstrap 教程
Bootstrap,来自 Twitter,是基于 HTML.CSS.JAVASCRIPT 的简介灵活的流行前段框架及交互组件集. 内容列表 Bootstrap 教程 Bootstrap 教程 Boot ...
随机推荐
- 5G的真正价值
导读 5G时代已经到来,它究竟会给我们的工作和生活带来什么样的改变?5G的爆点,究竟会出现在哪里?我们又该如何把握5G带来的发展机遇? 进入2019年之后,5G的商用步伐在不断加快.全球各地,都在忙着 ...
- 利用python scapy包进行抓包发包与ARP扫描
小技巧 通过在交互式的python解释器下,可以通过help()函数查看函数或模块的用途. dir() 函数不带参数时,返回当前范围内的变量.方法和定义的类型列表:带参数时,返回参数的属性.方法列表 ...
- ESP8266开发综合篇(SDK开发-视频教程总揽)
为了解决基础教程简单入门但不实用,项目方案非常实用但比较难的问题,开始推出8266开发综合篇 综合篇涉及到AT,LUA,SDK,LUA(sdk)开发,LUA和SDK开发会同步进行,后期再整理AT指令的 ...
- GIT 工作区和暂存区
工作区和暂存区 Git和其他版本控制系统如SVN的一个不同之处就是有暂存区的概念. 先来看名词解释. 工作区(Working Directory) 就是你在电脑里能看到的目录,比如我的studygit ...
- [04] 高级映射 association和collection
之前我们提到的映射,都是简单的字段和对象属性一对一,假设对象的属性也是一个对象,即涉及到两个表的关联,此时应该如何进行映射处理? 先看两张表,author 和 book: 业务上对应关系为,一个 ...
- WebApi系列~HttpClient的性能隐患 - 转
最近在进行开发过程中,基于都是接口开发,A站接口访问B接口接口来请求数据,而在这个过程中我们使用的是HttpClient这个框架,当然也是微软自己的框架,性能当前没有问题,但如果你直接使用官方的写法, ...
- 控制反转IOC与依赖注入DI - 理论篇
学无止境,精益求精 十年河东十年河西,莫欺少年穷 昨天是五一小长假归来上班的第一天,身体疲劳,毫无工作热情.于是就看看新闻,喝喝茶,荒废了一天 也就在昨天,康美同事张晶童鞋让我学习下IOC的理论及实现 ...
- SERDES关键技术总结
转自https://www.cnblogs.com/liujinggang/p/10125727.html 一.SERDES介绍 随着大数据的兴起以及信息技术的快速发展,数据传输对总线带宽的要求越来越 ...
- redis调优的实战经验
本文根据redis的info命令查看redis的内存使用情况以及state状态,来观察redis的运行情况以及需要作出的相应优化. info 1.memory used_memory:13409011 ...
- 总目录(Catalog)
总目录(Catalog) C#高级编程(C# advanced programming) 1.并发编程(Concurrent programming)(8) ...... 数据结构与算法(Data s ...