本文地址(轉載):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
處理已經足夠了,什麼鍵盤觸發滾動,讓他本身去玩耍吧,仍是別折騰了,吃力不討好。
喲,寫完了,擡頭一看,一張截圖都沒有,這可不行,風水不能斷,搞一張。
恩,不錯,真正的無可挑剔的「截」圖。