子元素scroll父元素容器不跟隨滾動JS實現

小tip: 子元素scroll父元素容器不跟隨滾動JS實現

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處理已經足夠了,什麼鍵盤觸發滾動,讓他本身去玩耍吧,仍是別折騰了,吃力不討好。

喲,寫完了,擡頭一看,一張截圖都沒有,這可不行,風水不能斷,搞一張。

 

恩,不錯,真正的無可挑剔的「截」圖。

相關文章
相關標籤/搜索