子元素scroll父元素容器不跟随滚动JS实现

一、开场暖身

网上常见蹲来蹲去的小段子,比方说:“李代沫蹲,李代沫蹲,李代沫蹲完黄海波蹲;黄海波蹲,黄海波蹲,黄海波蹲完宁财神蹲;宁财神蹲,宁财神蹲,宁财神蹲完张耀扬蹲;张耀扬蹲,张耀扬蹲,张耀扬蹲完郭美美蹲;郭美美蹲,郭美美蹲,郭美美蹲完……”。应该源自“萝卜蹲,萝卜蹲,萝卜蹲完苹果蹲……”。

在网页中,滚动条的滚动行为也是类似的调调,如果页面出现多个内嵌滚动条,则行为表现是:子元素滚,子元素滚,子元素滚完父元素滚;父元素滚,父元素滚,父元素滚完容器滚……

比方说下面:


在妹子脸上滚,先是妹子滚,妹子滚完主页面滚,对吧~

//zxx: 别问为什么不使用张含韵,因为张妹子照片是横的,滚动空间小,晓得伐~

这是浏览器的默认行为,如果我们遇到了一个需求:子元素滚,子元素滚完,就完了,父元素不需要滚了。那该如何实现呢?

在PC端,OK,本文介绍的方法,值适用于PC端,移动端,咳咳,我15年就没做过移动端项目,不好意思,手生,我也没去研究。

补充于翌日
移动端的处理,可以参见@hacke2的这篇文章:“在移动端上使用原生滑屏解决方案”

二、阻止浏览器默认行为的特定套路

哈,本文标题有些拗口,实际上用一句话概括就是:如何阻止浏览器的默认滚动行为。

基本上,好像印象中就没有例外的,阻止浏览器的默认行为,就一条(假设事件对象参数是event):event.preventDefault().

这是标准规范使用方法。但是,对于老IE浏览器,event.returnValue = false. 如果你使用jQuery等框架,直接上面的event.preventDefault()就可以,库已经帮你搞定了兼容细节处理。

OK,回到本文。阻止默认滚动,也是类似,关键是找到准确的事件。

第一反应是scroll事件,不知道是不是我测试的方法不对,结果没鸟用;其实想想也可以理解,scroll事件要触发,尼玛必须已经滚动了哈~

后来,发现要从滚动事件的源头处理起来。在PC端,绝大多数滚动都是鼠标滚动触发的(上下快捷键也可以滚动页面,但一般人不知道),因此,我们可以从鼠标滚轮事件入手。

三、鼠标滚轮事件

JS基础知识的啦,mousewheel事件:

dom.onmousewheel = function() {
    // 嘿嘿嘿
};

IE, Chrome都认识,但是FireFox浏览器,要使用DOMMouseScroll, 具体知识呢我之前有写过文章分析过:“JS滚轮事件(mousewheel/DOMMouseScroll)了解”。现在回过头看看这篇文章,内容和点都挺好。但是,当时正好在学习模块化开发,以及JavaScript语言模式,所以,提供的代码,科科,不是拿来主义的调调,所以这篇文章没有火啊~

扯远了,总之呢,我们对鼠标滚动这个事件,进行event.preventDefault(),页面就像齿轮卡壳了一样,滚不动了!

四、原理爬上来

找到了关键钥匙,现在就要开门了。

子元素可以滚,父元素不能滚。

我们可以对子元素写上鼠标滚轮事件,对吧,的那个子元素滚动到边界的时候,我们立马插一刀event.preventDefault()。干掉整个页面的滚动,世界一下子安静了,时间好像突然静止了一般,好像很不错的样子哦!

于是,寡人我屁颠屁颠搞起代码(粗糙示意):

if (direction == "up" && scrollTop == 0) {
   event.preventDefault()
}

翻译下就是:哥哥我往上滚,当滚到头的时候,页面滚动歇菜。

Chrome一测试,喔噢,好棒,鼓掌!  FireFox一测试,喔噢,好棒too,鼓掌again!  IE一测试,喔噢,好…………尼玛,滚蛋了~ 滚动高度直接跳过了0,直接把父元素给滚了。 

靠,什么鬼?不兼容,搞不定,怎么办?

五、临界手动翻滚

就是说,我们不要到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: 你问我什么不加高页面造一个滚动条?唉,舍不得把底部的广告刻意藏在滚动条之外~

六、抛砖引玉

前文也提到,页面滚动条滚动的事件源很多,不仅仅是鼠标滚动,上下键,End键, Home键等都有滚动定位行为。因此,大家要想100%全方位封杀滚动行为,仅仅上面的鼠标滚动代码是不够的,但是,关键钥匙已经给大家了,大家可以依次,按照自己的项目需求进行进一步深入拓展。

不过,我个人觉得,上面mousewheel处理已经足够了,什么键盘触发滚动,让他自己去玩耍吧,还是别折腾了,吃力不讨好。

哟,写完了,抬头一看,一张截图都没有,这可不行,风水不能断,搞一张。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索浏览器
, 鼠标滚动事件
, 事件
, 滚动
, 元素
, event
, 子页面滚怎么滚动
, preventdefault
, js阻止滚动条
, js滚轮事件兼容性
, js滚轮事件
, js阻止默认事件
, js默认行为
滚动事件
jquery导航栏跟随滚动、div跟随滚动条移动、js div跟随滚动条滑动、div跟随页面滚动、固定头部跟随屏幕滚动,以便于您获取更多的相关知识。

时间: 2025-01-02 09:19:46

子元素scroll父元素容器不跟随滚动JS实现的相关文章

js 阻止子元素响应父元素的onmouseout事件具体实现

 本文为大家介绍下js阻止子元素响应父元素的onmouseout事件,具体实现如下,感兴趣的朋友可以参考下  代码如下: $(".target-menu").mouseout(function(e){  evt = window.event || e;  var obj = evt.toElement || evt.relatedTarget;  var pa = this;  if(pa.contains(obj)) return false;  $(this).hide();  }

Javascript中找到子元素在父元素内相对位置的代码_javascript技巧

经过自己一晚上尝试,貌似找到了一个方法. 现在脑袋还糊涂,先记下来,以后再分析. 复制代码 代码如下: // 找到子元素在父元素中的相对位置 function getElementTop(element){ var el = (typeof element == "string") ? document.getElementById(element) : element; if (el.parentNode === null || el.style.display == 'none')

高效的获取当前元素是父元素的第几个子元素_javascript技巧

例如处理事件的时候,有时候需要知道当前点击的是第几个子节点,而HTML DOM本身并没有直接提供相应的属性,需要自己来计算. 从一个索引序号,很容易得到该索引对应的子节点或者子元素,直接用parentNode.childNodes[index] 或 parentNode.children[index] 就行. 但反过来,已知一个节点或元素对象,要知道它的索引序号则没有那么直接了. 一些特殊的元素,HTML DOM有对应的属性表示其索引序号,主要是表格的TD 和 TR 元素. 表格单元格TD元素有

阻止子元素继承父元素事件具体思路及实现_javascript技巧

复制代码 代码如下: <div id="p_box" onclick="a()"> <div id=p_cont> </div> </div> #p_box包括#p_cont,当点击#p_box区域任何位置时(包括#p_cont),都会触发a事件.想要阻止点击#p_cont区域时触发a事件,需要在#p_cont区域内加入阻止事件冒泡的代码. 变成 复制代码 代码如下: <div id="p_box&qu

js与jquery获取父元素,删除子元素的两种不同方法

 本篇文章主要是对js与jquery获取父元素,删除子元素的两种不同方法进行了介绍,需要的朋友可以过来参考下,希望对大家有所帮助 var obj=document.getElementById("id");得到的是dom对象,对该对象进行操作的时候使用js方法   var obj=$("#id");得到的是jquery对象,对该对象进行操作的时候使用jquery方法   1.对于上面获得的对象进行遍历   (1).js方法  for(vat i=0;j<obj

css父元素与子元素之间的margin-top问题

父元素的盒子包含一个子元素盒子,给子元素盒子一个垂直外边距margin-top,父元素盒子也会往下走margin-top的值,而子元素和父元素的边距则没有发生变化.   html代码: <div class="box1"> <div class="box2"></div> </div> css样式: .box1{height:200px;width:200px;background:gray;} .box2{heigh

Jquery搜索父元素操作方法

 这篇文章主要介绍了Jquery搜索父元素操作方法,实例分析了parents.cloest.parent.parentsUtil及offsetParent方法的使用技巧,需要的朋友可以参考下     本文实例讲述了Jquery搜索父元素操作方法.分享给大家供大家参考.具体分析如下: 1. parents()方法 格式: 代码如下: parents([selector]) 用于获取当前匹配元素集合中每个元素的祖先元素,根据需要还可以使用一个选择器进行筛选. 如: 代码如下: $("p")

Jquery搜索父元素操作方法_jquery

本文实例讲述了Jquery搜索父元素操作方法.分享给大家供大家参考.具体分析如下: 1. parents()方法 格式: 复制代码 代码如下: parents([selector])用于获取当前匹配元素集合中每个元素的祖先元素,根据需要还可以使用一个选择器进行筛选. 如: 复制代码 代码如下: $("p").parents().css("border", "1px solid blue); 2. cloest方法 格式: 复制代码 代码如下: closes

html中label宽度设置、非替换元素和替换元素

<label ></label> 单独对label设置一个width:100px的属性石不起作用的,和float:left或者display:inline-block配合的话 都可以设置上  参考:css position, display, float 内联元素.块级元素  在 CSS 中,任何元素都可以float浮动.浮动元素会生成一个块级框,而不论它本身是何种元素.如果浮动非替换元素,则要指定一个明确的宽度:否则,它们会尽可能地窄. 元素是文档结构的基础,在css里面,每个元