跳出彈窗頁面禁止滾動(PC端和手機端)


pc端如何實現

1.當彈窗顯示時,為body元素添加屬性:overflow:hidden, 當關閉彈窗時移除該屬性即可
2.在彈窗的div上設置 @scroll.stop.prevent

3.前端頁面彈框遮罩禁止頁面滾動

<div @scroll.stop.prevent>
 你要顯示的內容
</div>

3.出現彈窗時,為body元素添加position:fixed,這樣主頁面就禁止滑動,同時很好地解決了彈窗穿透的問題。
若彈窗為獨立組件,可以采用如下代碼:

beforeMount() {
    // 獲取原來的scrollTop 並將body的top修改為對應的值
    this.prevBodyStyle_scrollTop = document.body.scrollTop || document.documentElement.scrollTop
    this.prevBodyStyle_top = window.getComputedStyle(document.body, null).getPropertyValue('top')
    document.body.style.top = `-${this.prevBodyStyle_scrollTop}px`
    // 獲取原來body的position 為了解決iOS上光標漂移的問題 將position修改為fixed
    this.prevBodyStyle_position = window.getComputedStyle(document.body, null).getPropertyValue('position')
    document.body.style.position = 'fixed'
    // 為了避免width空值的情況
    this.prevBodyStyle_width = window.getComputedStyle(document.body, null).getPropertyValue('width')
    document.body.style.width = '100%'
  },

  beforeDestroy() {
    document.body.style.top = this.prevBodyStyle_top || '0px'
    document.body.style.position = this.prevBodyStyle_position
    document.body.style.width = this.prevBodyStyle_width || '100%'
    document.body.scrollTop = document.documentElement.scrollTop = this.prevBodyStyle_scrollTop || 0
  },

備注: 如果彈窗為一個獨立的組件, 那么需要使用v-if來控制彈窗是否顯示,不可使用v-show(因為使用v-show,會在主頁面剛生成的同時生成該組件,導致position=fixed生效,在彈窗關閉的情況下頁面也禁止滑動)

移動端如何實現

1.在彈窗的最外層div上添加@touchmove.stop.prevent (適合彈窗內容不需要滾動的情況下)

<div @touchmove.stop.prevent>
 你要顯示的內容
</div>

存在問題: 雖然可以阻止滑動,但是雙擊的時候主頁面還是會跳動

2.同PC端第三種方法
3.通過addEventListener解決

mounted() {
        document.body.addEventListener('touchmove', this.p, {passive: false})  
    },
beforeDestroy () {
      document.body.removeEventListener('touchmove', this.p)
    },
methods: {
    p (e) {
         e.preventDefault()
          e.stopPropagation()
      } 
}

划重點:addEventListener的第三個參數 {passive: false}

先說說錯誤的代碼,網上千篇一律的都是怎么成功的,納悶了,反正我沒有成功,在手機端和chrome瀏覽器等依然可以正常滾屏:

document.body.addEventListener("touchstart",function(e){
                    e.stopPropagation();
                    e.preventDefault();
                },false);

錯誤代碼為什么不行呢,而為什么使用{passive: false}就生效了呢?
摘自MDN的解釋:

passive: Boolean,表示 listener 永遠不會調用preventDefault()。如果 listener 仍然調用了這個函數,客戶端將會忽略它並拋出一個控制台警告。

MDN對於上述錯誤現象解釋地很清楚:

根據規范,passive 選項的默認值始終為false。但是,這引入了處理某些觸摸事件(以及其他)的事件監聽器在嘗試處理滾動時阻止瀏覽器的主線程的可能性,從而導致滾動處理期間性能可能大大降低。
為防止出現此問題,某些瀏覽器(特別是Chrome和Firefox)已將touchstart和touchmove事件的passive選項的默認值更改為true文檔級節點 Window,Document和Document.body。這可以防止調用事件監聽器,因此在用戶滾動時無法阻止頁面呈現。
var elem = document.getElementById('elem'); 
elem.addEventListener('touchmove', function listener() { /* do something */ }, { passive: true });
添加passive:true參數后,touchmove事件不會阻塞頁面的滾動(同樣適用於鼠標的滾輪事件)。

所以,我們可以通過將passive的值顯式設置為false來覆蓋此行為。

另外:您無需擔心基本scroll 事件的passive值。由於無法取消,因此事件監聽器無法阻止頁面呈現。(也是在PC端監聽scroll無效的原因)

參考鏈接

前兩種方法:https://blog.csdn.net/Chelle1...
第三種:https://blog.csdn.net/yuhk231...
vue prevent方法: https://www.cnblogs.com/Eden-...
mdn關於addEventListener: https://developer.mozilla.org...

原文地址:https://segmentfault.com/a/1190000017217663


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM