stickUp插件用于实现固定菜单栏效果,原理很简单,说白了就是监听document的scroll事件,滚动到特定值时,将特定元素的position设置为fixed,核心代码如下:
 
  $(document).on('scroll', function() {
varscroll = parseInt($(document).scrollTop());
if (menuSize != null) {
for (var i = 0; i < menuSize; i++) {
contentTop[i] = $('#' + content[i] + '').offset().top; function bottomView(i) {
contentView = $('#' + content[i] + '').height() * .4;
testView = contentTop[i] - contentView;
//console.log(varscroll);
if (varscroll > testView) {
$('.' + itemClass).removeClass(itemHover);
$('.' + itemClass + ':eq(' + i + ')').addClass(itemHover);
} else if (varscroll < 50) {
$('.' + itemClass).removeClass(itemHover);
$('.' + itemClass + ':eq(0)').addClass(itemHover);
}
}
if (scrollDir == 'down' && varscroll > contentTop[i] - 50 && varscroll < contentTop[i] + 50) {
$('.' + itemClass).removeClass(itemHover);
$('.' + itemClass + ':eq(' + i + ')').addClass(itemHover);
}
if (scrollDir == 'up') {
bottomView(i);
}
}
} if (vartop < varscroll + topMargin) {
$('.stuckMenu').addClass('isStuck');
$('.stuckMenu').next().closest('div').css({
'margin-top': stickyHeight + stickyMarginB + currentMarginT + 'px'
}, 10);
$('.stuckMenu').css("position", "fixed");
$('.isStuck').css({
top: '0px'
}, 10, function() { });
}; if (varscroll + topMargin < vartop) {
$('.stuckMenu').removeClass('isStuck');
$('.stuckMenu').next().closest('div').css({
'margin-top': currentMarginT + 'px'
}, 10);
$('.stuckMenu').css("position", "relative");
}; });
  但是,在实际使用过程中,还是发现诸多不便,
  • 它只支持最后一次调用(因为是使用闭包实现的变量存储,但每次init都基于同样一套变量)
  • 存在一些未经声明的变量调用(即全局变量,这可是很不好的编程习惯啊)
  • 存在一些不必要的函数声明,导致一些不必要的性能损耗(比如上述代码中的bottomView函数)
  • 不支持回调函数,无法支持比较复杂的应用
  • 单页网站时,在页面滚动到parts参数指定的块时,会给对应的菜单块加itemHover类(请参考http://lirancohen.github.io/stickUp/),parts是一个额外指定的参数,而parts中每一个id对应的菜单项又是基于parts参数的顺序的,这是一种不稳定结构(语文老师死得早,凑合着看吧
    stickUp原项目在github中很久没有更新了,所以我决定fork出一个分支,然后自己重构stickUp插件,我的项目地址是:https://github.com/VanMess/stickUp,有兴趣的童鞋可以clone下来看看,核心文件的代码只有150多行,结构也比较清晰,大家可以看看,有什么问题请联系我,交流交流。。。。当然,如果哪位大神能提出一些意见就更好了。
      新的stickUp代码主要分三个部分:Context类、Context._init_ 初始化函数、Context.onScroll 滚动处理函数。
    Context是一个上下文数据结构,用于记录每次调用的上下文信息,这样就解决了上面的第1个问题,代码如下:
    
 var Context = function() {},
_ctxList = {},
lastScrollTop = 0;
Context.prototype = {
dataProperty: 'data-menu',
selector: '',
itemClass: '',
itemHover: '',
jqDom: null,
menuItems: [],
region: 'top',
height: 0,
parentMarginTop: 0,
top: 0,
marginTop: 0,
marginBottom: 0,
beforeStick: null,
afterStick: null,
beforeUnstick: null,
afterUnstick: null
};
  具体每一项的含义、用法,建议大家可以看看源码。
    Context._init_ 是一个初始化函数,一个工厂,接受一个option参数,并将之转换为一个Context实例,注意,这里使用了_ctxList 变量来存储历史以来所有的上下文信息,代码如下:
 
  最后,是Context.prototype.onScroll 类,用于处理页面滚动事件,是整个stickUp的核心所在,代码如下:
 
onScroll: function(scrollDir, varscroll) {
var contentView = null,
testView = null,
_me = this; // 计算并给适当元素添加 itemHover 类
if ( !! _me.menuItems && _me.menuItems.length > 0) {
var offset = null,
contentTop = 0,
tmp_menuTarget = null;
for (var i = 0; i < _me.menuItems.length; i++) {
tmp_menuTarget = $('#' + $(_me.menuItems[i]).attr(_me.dataProperty));
offset = tmp_menuTarget.offset();
contentTop = !! offset ? offset.top : 0; // 之前這裡定義了一個bottomView
// 会在每次执行这个地方的时候都去创建一个函数
// 实际上是很没必要的性能损耗,所以这里将代码移动下面
if (scrollDir == 'down' &&
varscroll > contentTop - 50 &&
varscroll < contentTop + 50) {
_me.jqDom.find('.' + _me.itemClass).removeClass(_me.itemHover);
_me.jqDom.find('.' + _me.itemClass + ':eq(' + i + ')').addClass(_me.itemHover);
}
if (scrollDir == 'up') {
// 这里就是原来的bottomView代码
contentView = tmp_menuTarget.height() * 0.4;
testView = contentTop - contentView;
if (varscroll > testView) {
_me.jqDom.find('.' + _me.itemClass).removeClass(_me.itemHover);
_me.jqDom.find('.' + _me.itemClass + ':eq(' + i + ')').addClass(_me.itemHover);
} else if (varscroll < 50) {
_me.jqDom.find('.' + _me.itemClass).removeClass(_me.itemHover);
_me.jqDom.find('.' + _me.itemClass + ':eq(0)').addClass(_me.itemHover);
}
}
}
} // 固定菜单栏目,使之固定(fixed)
if (_me.top < varscroll + _me.marginTop) {
if ( !! _me.beforeStick) _me.beforeStick.call(_me);
_me.jqDom.addClass('isStuck');
if ( !! _me.afterStick) _me.afterStick.call(_me);
_me.jqDom.next().closest('div').css({
'margin-top': _me.height + _me.marginBottom + _me.parentMarginTop + 'px'
}, 10);
_me.jqDom.css("position", "fixed");
_me.jqDom.css({
top: '0px'
}, 10);
}; // 菜單欄目,使之不固定(relative)
if (varscroll + _me.marginTop < _me.top) {
if ( !! _me.beforeUnstick) _me.beforeUnstick.call(_me);
_me.jqDom.removeClass('isStuck');
if ( !! _me.afterUnstick) _me.afterUnstick.call(_me);
_me.jqDom.next().closest('div').css({
'margin-top': _me.parentMarginTop + 'px'
}, 10);
_me.jqDom.css("position", "relative");
};
}
后记:一直想做一个自己的开源项目,不过还没想清楚要做什么,所以想着先拿别人的来重构、优化,这次使用stickUp是看中它的小(下次想修改百度的ECharts)。stickUp还不是很成熟,但根据我的测试在多个浏览器下都不存在大问题,欢迎大家使用,有什么问题请尽管联系我
另外,看在我辛苦的份上,就麻烦大家留个言鼓励鼓励吧,谢谢
最后附上源码:https://github.com/VanMess/stickUp
 
 

jQuery.stickUp插件重构的更多相关文章

  1. PHP结合jQuery.autocomplete插件实现输入自动完成提示的功能

    XHTML 首先将jquery库和相关ui插件,以及css导入.一共引用三个 jquery ,jquery_ui.js,jquery-ui.css 三个文件,不同版本小哥可能稍有差异(最后注意ajax ...

  2. 深入学习jQuery自定义插件

    原文地址:jQuery自定义插件学习 1.定义插件的方法 对象级别的插件扩展,即为jQuery类的实例增加方法, 调用:$(选择器).函数名(参数);      $(‘#id’).myPlugin(o ...

  3. [jQuery]jQuery DataTables插件自定义Ajax分页实现

    前言 昨天在博客园的博问上帮一位园友解决了一个问题,我觉得有必要记录一下,万一有人也遇上了呢. 问题描述 园友是做前端的,产品经理要求他使用jQuery DataTables插件显示一个列表,要实现分 ...

  4. 使用jQuery.form插件,实现完美的表单异步提交

    传送门:异步编程系列目录…… 时间真快,转眼一个月快结束了,一个月没写博客了!手开始生了,怎么开始呢…… 示例下载:使用jQuery.form插件,实现完美的表单异步提交.rar 月份的尾巴,今天的主 ...

  5. 为jQuery写插件

    很多场合,我们都会调用jQuery的插件去完成某个功能,比如slider. 如下图,做一个div,通过“$( "#slider" ).slider();”的方式直接将div变成sl ...

  6. bootstrap-简洁实用的jQuery手风琴插件

    前端 <html lang="zh"> <head> <meta charset="UTF-8"> <meta htt ...

  7. 推荐15款响应式的 jQuery Lightbox 插件

    利用现代 Web 技术,网络变得越来越轻巧与.模态框是突出展现内容的重要形式,能够让用户聚焦到重要的内容上去.在这个列表中,我们编制了15款响应式的 jQuery 灯箱库,这将有助于开发人员创建和设计 ...

  8. Chocolat.js – 响应式的 jQuery Lightbox 插件

    Chocolat.js 使您能够显示一个或多个图像在同一页面上.给用户展示一组图片缩略图,可以显示全页或块.Chocolat.js 可以很好地处理所有主要的浏览器.它在下面这些浏览器测试通过:IE7+ ...

  9. 让网站动起来!12款优秀的 jQuery 动画插件推荐

    如今,大多数设计师和开发人员被要客户要求开发动态的网站.创造视觉震撼和醒目的动态网站是艰巨的任务,因为它需要大量的努力和创造力.在网络上有大量的工具和插件可用于创建网站动画.许多开发人员正在使用 HT ...

随机推荐

  1. ajaxFileUpload文件上传

    一.简介 ajaxFileUpload是一种异步的文件上传控件,通过ajax进行文件上传,并获取上传处理结果.在很多时候我们需要使用到文件上传的功能,但是不需要使用那些强大的上传插件.此时就可以使用a ...

  2. html页面通过特殊链接:打电话,发短信,发邮件详细教程

    采用url href链接的方式,实现在Safari  ios,Android 浏览器,webos浏览器,塞班浏览器,IE,Operamini等主流浏览器,进行拨打电话功能. 1. 拨打电话 在电话号码 ...

  3. BusyBox Init

    嵌入式系统内核启动后的第一个程序就是init,一般位于/sbin/init(一般是符号链接到/bin/busybox), 但有些也会直接放在根目录下如linuxrc,busybox的init不支持多级 ...

  4. NetBIOS

    NetBIOS是Network Basic Input/Output System的缩写,严格来说它不是一个网络协议,而是一套API,为局域网内应用程序通信提供会话层(OSI七层参考模型)的支持. N ...

  5. [转]Javascript中的自执行函数表达式

    [转]Javascript中的自执行函数表达式 本文转载自:http://www.ghugo.com/javascript-auto-run-function/ 以下是正文: Posted on 20 ...

  6. 如何实现ZBrush中部分模型的选择和隐藏

    在ZBrush中制作雕刻比较庞大细节又很丰富模型的时候,有时你可能只想显示模型的某些部分,有些部分挡住了视线想要暂时隐藏.ZBrush®软件中有一个选项功能使这项操作变得相当简单,像其他功能一样,使用 ...

  7. C语言异或运算在程序设计中的妙用

    异或运算符∧也称XOR运算符.它的规则是若参加运算的两个二进位同号,则结果为0(假):异号则为1(真).即0∧0=0,0∧1=1,1∧1=0. 性质: (1).与1异或会翻转 (2).与0异或保持不变 ...

  8. Doxygen Tool For Unity

    一.准备阶段 在之前的一系列文章中,我尝试了不同方法为Unity的C#生成脚本手册(帮助文档) 使用Doxygen生成C#帮助文档 为Unity项目生成文档(一)为Unity项目生成文档(二) 建议的 ...

  9. 为Unity项目生成文档(一)

    VS生成chm帮助文档 VS代码中使用Xml注释,并通过Sandcastle生成chm文档的文章,这几篇值得分享: 使用.NET中的XML注释(一) -- XML注释标签讲解 使用.NET中的XML注 ...

  10. JavaEE

    1.  为什么需要JavaEE 我们编写的JSP代码中,由于大量的显示代码和业务逻辑混淆在一起,彼此嵌套,不利于程序的维护和扩展.当业务需求发生变化的时候,对于程序员和美工都是一个很重的负担. 为了程 ...