最近作項目常常在 chrome 的控制檯看到以下提示:css
Unable to preventDefault inside passive event listener due to target being treated as passive. See https://www.chromestatus.com/features/5093566007214080
因而 Google 了一番,找到這篇文章,有了詳細解釋。Making touch scrolling fast by defaultgit
簡而言之:github
因爲瀏覽器必需要在執行事件處理函數以後,才能知道有沒有掉用過 preventDefault() ,這就致使了瀏覽器不能及時響應滾動,略有延遲。
因此爲了讓頁面滾動的效果如絲般順滑,從 chrome56 開始,在 window、document 和 body 上註冊的 touchstart 和 touchmove 事件處理函數,會默認爲是 passive: true。瀏覽器忽略 preventDefault() 就能夠第一時間滾動了。 舉例: wnidow.addEventListener('touchmove', func) 效果和下面一句同樣 wnidow.addEventListener('touchmove', func, { passive: true })
這就致使了一個問題:web
若是在以上這 3 個元素的 touchstart 和 touchmove 事件處理函數中調用 e.preventDefault() ,會被瀏覽器忽略掉,並不會阻止默認行爲。
測試:chrome
body {
margin: 0; height: 2000px; background: linear-gradient(to bottom, red, green); } // 在 chrome56 中,照樣滾動,並且控制檯會有提示,blablabla window.addEventListener('touchmove', e => e.preventDefault())
那麼如何解決這個問題呢?不讓控制檯提示,並且 preventDefault() 有效果呢?
兩個方案:
一、註冊處理函數時,用以下方式,明確聲明爲不是被動的
window.addEventListener('touchmove', func, { passive: false })typescript
二、應用 CSS 屬性 touch-action: none;
這樣任何觸摸事件都不會產生默認行爲,可是 touch 事件照樣觸發。
touch-action 還有不少選項,詳細請參考touch-actionsegmentfault
[注]將來可能全部的元素的 touchstart touchmove 事件處理函數都會默認爲 passive: true瀏覽器