网站在苹果 Safari 进行适配遇到的问题

在网站进行移动端 Web 适配开发中,弹窗和导航栏弹出等常常会出现一些问题,如果是奇奇怪怪的客户严格要求的话,那么就会有下面这些情况:

  • 打开弹窗后页面自动放大,视图区被放大到看不全
  • 打开对话框打开后背景仍然能滚动
  • 导航栏弹窗后,背后内容可滚动,影响体验
  • 点击聚焦输入框,导致视图放大

下面通过 Nuxtjs3 示例,分析问题原因并提供一些解决方案,有错误或者有其他想法可以评论提出!


1. 问题分析

在 iOS Safari 中,可能开发者都看到了已经对 body 进行超出隐藏了,但是就会出现穿透滑动滚动现象(查了下资料说是 iOS Safari 对这个属性无效,实际大家可以自己去看看);

弹窗中有输入框或效果切换时,浏览器会触发自动缩放行为,导致全页视图放大一丢丢,视图溢出和乱移。


2. 解决思路

视图放大问题

通过 meta viewport 可以限制浏览器的缩放操作(也就是 html 原生放在 head 上的 meta):

/* by 01130.hk - online tools website : 01130.hk/zh/generatebtcwallets.html */
useHead({
  meta: [
    { name: 'viewport', content: 'width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no' }
  ]
})

禁滚 body & 禁滑

当弹窗或菜单打开时,通过动态修改 body CSS 来防止背景滚动以及监听触碰滑动(可单独拎出):

/* by 01130.hk - online tools website : 01130.hk/zh/generatebtcwallets.html */
const preventDefault = (e: Event) => {
  e.preventDefault()
}

const disableBodyScroll = () => {
  document.body.style.overflow = 'hidden';
  document.body.style.position = 'fixed';
  document.body.style.width = '100%';
  document.body.style.top = `-${window.scrollY}px`;
  document.addEventListener('touchmove', preventDefault, { passive: false })
};

const enableBodyScroll = () => {
  const scrollY = document.body.style.top;
  document.body.style.overflow = '';
  document.body.style.position = '';
  document.body.style.width = '';
  document.body.style.top = '';
  if (scrollY) {
    window.scrollTo(0, parseInt(scrollY || '0') * -1);
  }
  document.removeEventListener('touchmove', preventDefault)
};

通过记录当前滚动位置,关闭弹窗后恢复滚动状态,可以防止页面上下移动。

注:这里是自定义的弹窗组件,如果禁滑的话要在对应弹窗内部加上单防穿透(@touchmove.stop),避免内部超出不可触摸滚动,原理如下(可以看看官方属性进行理解):

当用户在页面上滑动时:
[触摸点]
↓ 触发 touchmove
[某个 div]
↓ 冒泡
[父元素]
↓ 冒泡
[body]
↓ 冒泡
[html]
↓ 冒泡
[document] ← 在这里被拦截并 preventDefault()
↓ 冒泡
[window]

当弹窗内部滑动时:
[弹窗内的滚动区域]
↓ 触发 touchmove
↓ 遇到 @touchmove.stop
停止冒泡(stopPropagation)
不会到达 document,因此不会被 preventDefault
保留默认滚动行为

Vue 中通过 watch 监听

watch(visible, (newVal) => {
  if (newVal) {
    disableBodyScroll();
  } else {
    enableBodyScroll();
  }
});

当 visible 为 true 时禁滚,其他情况恢复滚动。


3. 实现例子

客服对话窗 (Chat Window)

watch(visible, (newVal) => {
  if (newVal) {
    minimized.value = false;
    loadMessagesFromStorage();
    disableBodyScroll();
  } else {
    enableBodyScroll();
  }
});

每次弹窗打开,禁滚 body;关闭后恢复滚动。

通过 Teleport 把对话窗注入 body,保证层级级次上在最上方,避免被其他元素遮挡或发生 z-index 第一问题。

顶部导航栏 (Navbar)

移动端打开横向菜单后,同样需要禁滚 body:

const toggleMobileMenu = () => {
  mobileMenuOpen.value = !mobileMenuOpen.value;
  if (mobileMenuOpen.value) {
    disableBodyScroll();
  } else {
    enableBodyScroll();
  }
};

5. 总结

移动端弹窗和横向菜单的滚动漏洞,在经典 WebApp 中很容易被忽视。使用上述方法,可以在 Vue3 / Nuxt3 项目中简单且稳定地处理这些问题,但是并没有分开详细介绍,只给大家提供思路,因为现在 AI 较多,大家参考完进行头脑风暴比较好。

如需查看全部实现代码,可以留言,会把弹窗代码分享下。

代码转载自:https://pan.quark.cn/s/7f503284aed9 Hibernate的核心组件总数达到五个,具体包括:Session、SessionFactory、Transaction、Query以及Configuration。 这五个核心组件在各类开发项目中都具有普遍的应用性。 借助这些组件,不仅可以高效地进行持久化对象的读取与存储,还能够实现事务管理功能。 接下来将通过图形化的方式,逐一阐述这五个核心组件的具体细节。 依据所提供的文件内容,可以总结出以下几个关键知识点:### 1. SSH框架详细架构图尽管标题提及“SSH框架详细架构图”,但在描述部分并未直接呈现关于SSH的详细内容,而是转向介绍了Hibernate的核心接口。 然而,在此我们可以简要概述SSH框架(涵盖Spring、Struts、Hibernate)的核心理念及其在Java开发中的具体作用。 #### Spring框架- **定义**:Spring框架是一个开源架构,其设计目标在于简化企业级应用的开发流程。 - **特点**: - **分层结构**:该框架允许开发者根据实际需求选择性地采纳部分组件,而非强制使用全部功能。 - **可复用性**:Spring框架支持创建可在不同开发环境中重复利用的业务逻辑和数据访问组件。 - **核心构成**: - **核心容器**:该部分包含了Spring框架的基础功能,其核心在于`BeanFactory`,该组件通过工厂模式运作,并借助控制反转(IoC)理念,将配置和依赖管理与具体的应用代码进行有效分离。 - **Spring上下文**:提供一个配置文件,其中整合了诸如JNDI、EJB、邮件服务、国际化支持等企业级服务。 - **Spring AO...
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值