🎶 Sym - 一款用 Java 实现的现代化社区(论坛/BBS/社交网络/博客)平台

📕 思源笔记 - 一款桌面端笔记应用,支持 Windows、Mac 和 Linux

🎸 Solo - B3log 分布式社区的博客端节点,欢迎加入下一代社区网络

♏ Vditor - 一款浏览器端的 Markdown 编辑器

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

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 秒系列之小八哥


欢迎注册黑客派社区,开启你的博客之旅。让学习和分享成为一种习惯!

留下你的脚步