本文地址(转载):http://www.zhangxinxu.com/wordpress/?p=5092html
1、开场暖身
网上常见蹲来蹲去的小段子,比方说:“李代沫蹲,李代沫蹲,李代沫蹲完黄海波蹲;黄海波蹲,黄海波蹲,黄海波蹲完宁财神蹲;宁财神蹲,宁财神蹲,宁财神蹲完张耀扬蹲;张耀扬蹲,张耀扬蹲,张耀扬蹲完郭美美蹲;郭美美蹲,郭美美蹲,郭美美蹲完……”。应该源自“萝卜蹲,萝卜蹲,萝卜蹲完苹果蹲……”。浏览器
在网页中,滚动条的滚动行为也是相似的调调,若是页面出现多个内嵌滚动条,则行为表现是:子元素滚,子元素滚,子元素滚完父元素滚;父元素滚,父元素滚,父元素滚完容器滚……框架
比方说下面:dom

在妹子脸上滚,先是妹子滚,妹子滚完主页面滚,对吧~模块化
//zxx: 别问为何不使用张含韵,由于张妹子照片是横的,滚动空间小,晓得伐~wordpress
这是浏览器的默认行为,若是咱们遇到了一个需求:子元素滚,子元素滚完,就完了,父元素不须要滚了。那该如何实现呢?学习
在PC端,OK,本文介绍的方法,值适用于PC端,移动端,咳咳,我15年就没作过移动端项目,很差意思,手生,我也没去研究。测试
补充于翌日
移动端的处理,能够参见@hacke2的这篇文章:“在移动端上使用原生滑屏解决方案”this
2、阻止浏览器默认行为的特定套路
哈,本文标题有些拗口,实际上用一句话归纳就是:如何阻止浏览器的默认滚动行为。spa
基本上,好像印象中就没有例外的,阻止浏览器的默认行为,就一条(假设事件对象参数是event):event.preventDefault()
.
这是标准规范使用方法。可是,对于老IE浏览器,event.returnValue = false
. 若是你使用jQuery等框架,直接上面的event.preventDefault()
就能够,库已经帮你搞定了兼容细节处理。
OK,回到本文。阻止默认滚动,也是相似,关键是找到准确的事件。
第一反应是scroll
事件,不知道是否是我测试的方法不对,结果没鸟用;其实想一想也能够理解,scroll事件要触发,尼玛必须已经滚动了哈~
后来,发现要从滚动事件的源头处理起来。在PC端,绝大多数滚动都是鼠标滚动触发的(上下快捷键也能够滚动页面,但通常人不知道),所以,咱们能够从鼠标滚轮事件入手。
3、鼠标滚轮事件
JS基础知识的啦,mousewheel
事件:
dom.onmousewheel = function() { // 嘿嘿嘿 };
IE, Chrome都认识,可是FireFox浏览器,要使用DOMMouseScroll, 具体知识呢我以前有写过文章分析过:“JS滚轮事件(mousewheel/DOMMouseScroll)了解”。如今回过头看看这篇文章,内容和点都挺好。可是,当时正好在学习模块化开发,以及JavaScript语言模式,因此,提供的代码,科科,不是拿来主义的调调,因此这篇文章没有火啊~
扯远了,总之呢,咱们对鼠标滚动这个事件,进行event.preventDefault()
,页面就像齿轮卡壳了同样,滚不动了!
4、原理爬上来
找到了关键钥匙,如今就要开门了。
子元素能够滚,父元素不能滚。
咱们能够对子元素写上鼠标滚轮事件,对吧,的那个子元素滚动到边界的时候,咱们立马插一刀event.preventDefault()
。干掉整个页面的滚动,世界一会儿安静了,时间好像忽然静止了通常,好像很不错的样子哦!
因而,寡人我屁颠屁颠搞起代码(粗糙示意):
if (direction == 'up' && scrollTop == 0) { event.preventDefault() }
翻译下就是:哥哥我往上滚,当滚到头的时候,页面滚动歇菜。
Chrome一测试,喔噢,好棒,鼓掌! FireFox一测试,喔噢,好棒too,鼓掌again!
IE一测试,喔噢,好…………尼玛,滚蛋了~ 滚动高度直接跳过了0,直接把父元素给滚了。
靠,什么鬼?不兼容,搞不定,怎么办?
5、临界手动翻滚
就是说,咱们不要到0或者最大滚动高度时候,再去阻止默认滚动,咱们要在到达边界的前一个滚动,就开始下手,手动滚动到边界,同时event.preventDefault()
阻止鼠标滚动行为。因而,IE浏览器也棒棒哒了!
说实话,从开头到如今,中文啪啪啪敲了这么多,其实毛线用都没有,从度娘或谷哥过来的同窗须要的不是什么神神叨叨的废话,须要的只是下面这段能够直接拿来主义的代码,好吧,拿去吧——子元素滚完就滚完的方法源代码:
$.fn.scrollUnique = function() { return $(this).each(function() { var eventType = 'mousewheel'; // 火狐是DOMMouseScroll事件 if (document.mozHidden !== undefined) { eventType = 'DOMMouseScroll'; } $(this).on(eventType, function(event) { // 一些数据 var scrollTop = this.scrollTop, scrollHeight = this.scrollHeight, height = this.clientHeight; var delta = (event.originalEvent.wheelDelta) ? event.originalEvent.wheelDelta : -(event.originalEvent.detail || 0); if ((delta > 0 && scrollTop <= delta) || (delta < 0 && scrollHeight - height - scrollTop <= -1 * delta)) { // IE浏览器下滚动会跨越边界直接影响父级滚动,所以,临界时候手动边界滚动定位 this.scrollTop = delta > 0? 0: scrollHeight; // 向上滚 || 向下滚 event.preventDefault(); } }); }); };
没错,依赖jQuery的一个扩展方法,上面代码只要拷贝到你页面的JS中,而后,你但愿哪一个元素滚动到底,父级不滚动,直接:
$().scrollUnique();
就能够了,而后就能够打卡下班了。
对了,有个demo, 您能够狠狠地点击这里:里面元素滚动到底外部容器不滚动demo
若是您的显示器竖屏,或者宽度1920的,会发现右侧没有大滚动条,则,麻烦你们手动高度改小,拉拉窗口啊,或者打开控制台之类的。
//zxx: 你问我什么不加高页面造一个滚动条?唉,舍不得把底部的广告刻意藏在滚动条以外~
6、抛砖引玉
前文也提到,页面滚动条滚动的事件源不少,不只仅是鼠标滚动,上下键,End键, Home键等都有滚动定位行为。所以,你们要想100%全方位封杀滚动行为,仅仅上面的鼠标滚动代码是不够的,可是,关键钥匙已经给你们了,你们能够依次,按照本身的项目需求进行进一步深刻拓展。
不过,我我的以为,上面mousewheel
处理已经足够了,什么键盘触发滚动,让他本身去玩耍吧,仍是别折腾了,吃力不讨好。
哟,写完了,抬头一看,一张截图都没有,这可不行,风水不能断,搞一张。
恩,不错,真正的无可挑剔的「截」图。