掌控滚动操作 - 自定义下拉刷新和溢出效果

Eric Bidelman
Majid Valipour

要点

借助 CSS overscroll-behavior 属性,开发者可以覆盖在到达内容顶部/底部时浏览器的默认溢出滚动行为。使用场景包括在移动设备上停用下拉刷新功能、移除滚动发光和橡皮筋效果,以及阻止网页内容在模态/叠加层下方时滚动。

背景

滚动边界和滚动链

Chrome Android 上的滚动链。

滚动是与页面交互的最基本方式之一,但由于浏览器奇怪的默认行为,某些用户体验模式可能难以处理。例如,应用抽屉中包含大量用户可能必须滚动浏览的项目。当它们到达底部时,溢出容器会停止滚动,因为没有更多内容可供消费。换言之,用户到达“滚动边界”。但请注意,如果用户继续滚动,会发生什么情况。抽屉式导航栏后面的内容开始滚动!滚动操作被父级容器取代;在本示例中,主页面本身。

原来,这种行为称为“滚动链”,即浏览器在滚动内容时的默认行为。通常,默认设置非常好,但有时并不可取,甚至出乎意料。某些应用可能需要在用户遇到滚动边界时提供不同的用户体验。

下拉刷新效果

下拉刷新是一种在 Facebook 和 Twitter 等移动应用中常用的直观手势。下拉社交信息流并发布即可创建包含最新帖子的新空间。事实上,这种特定的用户体验变得如此受欢迎,以至于 Android 版 Chrome 等移动浏览器也采用了相同的效果。在页面顶部向下滑动可刷新整个页面:

Twitter 在其 PWA 中刷新 Feed 时的自定义拉取刷新
Chrome Android 的原生下拉刷新操作
会刷新整个网页。

对于 Twitter PWA 等情况,停用原生拉取刷新操作可能是合理的。原因何在?在此应用中,您可能不希望用户意外刷新页面。此外,您可能还会看到双重刷新动画!或者,您也可以自定义浏览器的操作,使其更贴近网站的品牌形象。遗憾的是,此类自定义很难实现。开发者最终会编写不必要的 JavaScript,添加非被动触摸监听器(阻止滚动),或将整个页面固定在 100vw/vh <div>(以防止页面溢出)。这些权宜解决方法对滚动性能有详尽记录的负面影响。

我们可以做得更好!

隆重推出 overscroll-behavior

overscroll-behavior 属性是一项新的 CSS 功能,可控制滚动容器(包括页面本身)时发生的行为。您可以使用它来取消滚动链、停用/自定义下拉刷新操作、停用 iOS 上的橡皮筋效果(当 Safari 实现 overscroll-behavior 时)等。最好的一点是,使用 overscroll-behavior 不会对网页性能产生不利影响,比如简介中提到的技巧!

该属性采用三个可能的值:

  1. auto - 默认值。在该元素上发起的滚动可能会传播到祖先元素。
  2. contains - 用于防止滚动链。滚动不会传播到祖先实体,但会显示节点内的局部效应。例如,Android 上的滚动发光效果或 iOS 上的橡皮筋效果,在用户到达滚动边界时通知用户。注意:对 html 元素使用 overscroll-behavior: contain 可防止滚动导航操作。
  3. none - 与 contain 相同,但这也会阻止节点本身内的滚动效果(例如,Android 滚动发光或 iOS 橡皮筋)。

让我们通过一些示例来了解如何使用 overscroll-behavior

防止滚动操作转义固定位置元素

Chatbox 场景

聊天窗口下方的内容也会滚动 :(

假设有一个位于页面底部的固定位置聊天框。这样做的目的是,聊天框是一个独立的组件,并且与其后面的内容独立滚动。不过,由于滚动链的缘故,只要用户点击聊天记录中的最后一条消息,文档就会开始滚动。

对于此应用,更合适的做法是将源自 Chat 的滚动操作留在聊天中。为此,我们可以将 overscroll-behavior: contain 添加到保存聊天消息的元素:

#chat .msgs {
  overflow: auto;
  overscroll-behavior: contain;
  height: 300px;
}

从本质上讲,这是在聊天框的滚动上下文和主页面之间建立逻辑分隔。最终结果是,当用户浏览到聊天记录的顶部/底部时,主页面会保持原位。从聊天框中开始的滚动不会传播。

网页重叠式广告场景

“滚动下”场景的另一种变化是,当您看到内容在固定位置叠加层后面滚动时。送上惊喜大奖 overscroll-behavior来啦!浏览器会尽力提供帮助,但最终导致网站看起来有漏洞

示例 - 模态窗口,带和不带 overscroll-behavior: contain

之前:网页内容在叠加层下方滚动。
之后:网页内容不会在叠加层下方滚动。

停用下拉刷新

只需一行 CSS 即可关闭下拉刷新操作。只需避免在定义视口的整个元素上产生滚动链即可。在大多数情况下,就是 <html><body>

body {
  /* Disables pull-to-refresh but allows overscroll glow effects. */
  overscroll-behavior-y: contain;
}

通过这一简单添加,我们修复了聊天框演示中的双重下拉刷新动画,并且可以改为实现使用更简洁的加载动画的自定义效果。刷新收件箱时,整个收件箱也会变得模糊:

之前
之后

以下是完整代码的代码段:

<style>
  body.refreshing #inbox {
    filter: blur(1px);
    touch-action: none; /* prevent scrolling */
  }
  body.refreshing .refresher {
    transform: translate3d(0,150%,0) scale(1);
    z-index: 1;
  }
  .refresher {
    --refresh-width: 55px;
    pointer-events: none;
    width: var(--refresh-width);
    height: var(--refresh-width);
    border-radius: 50%;
    position: absolute;
    transition: all 300ms cubic-bezier(0,0,0.2,1);
    will-change: transform, opacity;
    ...
  }
</style>

<div class="refresher">
  <div class="loading-bar"></div>
  <div class="loading-bar"></div>
  <div class="loading-bar"></div>
  <div class="loading-bar"></div>
</div>

<section id="inbox"><!-- msgs --></section>

<script>
  let _startY;
  const inbox = document.querySelector('#inbox');

  inbox.addEventListener('touchstart', e => {
    _startY = e.touches[0].pageY;
  }, {passive: true});

  inbox.addEventListener('touchmove', e => {
    const y = e.touches[0].pageY;
    // Activate custom pull-to-refresh effects when at the top of the container
    // and user is scrolling up.
    if (document.scrollingElement.scrollTop === 0 && y > _startY &&
        !document.body.classList.contains('refreshing')) {
      // refresh inbox.
    }
  }, {passive: true});
</script>

停用滚动发光和橡皮筋效果

如需在到达滚动边界时停用弹跳效果,请使用 overscroll-behavior-y: none

body {
  /* Disables pull-to-refresh and overscroll glow effect.
     Still keeps swipe navigations. */
  overscroll-behavior-y: none;
}
之前:达到滚动边界时会显示发光。
之后:停用发光功能。

完整演示

综上所述,完整的聊天框演示使用 overscroll-behavior 创建自定义下拉刷新动画,并禁止滚动离开聊天框 widget。这样可以提供最佳用户体验,如果没有 CSS overscroll-behavior,很难实现这样的体验。

观看演示 | 来源