iPhone 键盘弹起时 position 为 sticky/fixed 时失效

本贴最后更新于 1665 天前,其中的信息可能已经渤澥桑田

2020-05-13

描述

将一个元素设置为 positon: sticky 时,该元素会相对于他最近的滚动祖先进行定位。如:屏幕滚动到横向二级导航的位置时,该二级导航需固定在屏幕顶部。

将一个元素设置为 positon: fixed 时,该元素相对于屏幕的位置将永远保持不变。如:回到顶部按钮。

在没有唤出键盘的情况下,以上属性都可以正常使用。但在 iPhone 中,当键盘被唤出时,使用以上属性的元素相对于屏幕的位置并不会被重新计算,因此就会被弹出的键盘顶到了屏幕外。

解决方案

  • 使用 JavaScript API visualViewport 监听可视窗口的 scrollresize 事件
  • 使用 JavaScript API getBoundingClientRect 获取定位元素所偏移的位置
  • 根据偏移位置使用 CSS 属性 transform 将其重新定位到所需位置
let pendingUpdate = false;
const viewportHandler = (event: Event) => {
    if (pendingUpdate) {
        return;
    }
    pendingUpdate = true;

    requestAnimationFrame(() => {
        pendingUpdate = false;
        const layoutViewport = vditor.toolbar.element;
        layoutViewport.style.transform = "none";
        if (layoutViewport.getBoundingClientRect().top < 0) {
            layoutViewport.style.transform = `translate(0, ${-layoutViewport.getBoundingClientRect().top}px)`;
        }
    });
};
window.visualViewport.addEventListener("scroll", viewportHandler);
window.visualViewport.addEventListener("resize", viewportHandler);

示例

Vditor 工具栏

说明

  • visualViewport:提供查询或修改 visual viewport 属性的机制。visual viewport 为屏幕的视觉部分,不包含键盘屏幕、缩放区域之外的区域及任何其他不随页面尺寸缩放的元素。
  • requestAnimationFrame:确保在下一次渲染之前进行更新
  • endingUpdate:防止 resizescroll 同时触发时发生被调用多次
  • getBoundingClientRect:返回元素大小及其相对于可视区域的位置
  • transform:对元素进行旋转、缩放、倾斜或平移

返回总目录

每天 30 秒系列之小八哥

  • Bug

    Bug 本意是指臭虫、缺陷、损坏、犯贫、窃听器、小虫等。现在人们把在程序中一些缺陷或问题统称为 bug(漏洞)。

    76 引用 • 1737 回帖 • 2 关注
  • 30Seconds

    📙 前端知识精选集,包含 HTML、CSS、JavaScript、React、Node、安全等方面,每天仅需 30 秒。

    • 精选常见面试题,帮助您准备下一次面试
    • 精选常见交互,帮助您拥有简洁酷炫的站点
    • 精选有用的 React 片段,帮助你获取最佳实践
    • 精选常见代码集,帮助您提高打码效率
    • 整理前端界的最新资讯,邀您一同探索新世界
    488 引用 • 384 回帖 • 10 关注
  • iPhone
    12 引用 • 75 回帖 • 1 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
Vanessa
我们终此一生,就是要摆脱他人的期待,找到真正的自己。 红河