HTML5 移动端自适应方案与踩坑

2023-12-18 16:48

本文主要是介绍HTML5 移动端自适应方案与踩坑,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文链接:blog.sjfkai.com/2019/01/29/…

转载请注明出处

最近刚接触前端开发,接手了一个移动端H5项目。着实体会掉了前端的坑之多,和H5移动端的坑之多多。

如今项目告一段落,在这里做一总结

屏幕自适应方案

介绍方案之前,首先还是交代一下项目背景与需求,毕竟一切方案也不能脱离实际需求。

需求与背景

  • 设备宽度 > 800px, body宽度为800px
  • 320px < 设备宽度 < 800px, 宽度根据设备宽度自适应
  • 设备宽度 > 320px, body宽度为320px
  • 大部分字体不随宽度变化而缩放
  • 设计图宽度:1080px

自适应方案

关于自适应方案,google一搜就会有很多结果,但是总的来说个人认为最有用的还是手淘的大漠写的一系列文章,后面会给出原文链接。总的来说主流的方案有remvh两种。

REM(flexible)

rem(font size of the root element)是指相对于根元素的字体大小的单位。简单的说它就是一个相对单位。看到rem大家一定会想起em单位,em(font size of the element)是指相对于父元素的字体大小的单位。它们之间其实很相似,只不过一个计算的规则是依赖根元素一个是依赖父元素计算。

所以简而言之,就是根据屏幕宽度设置 html 标签的 font-size。 再在布局时使用 rem 单位来布局,就可以达到自适应的目的。

使用此方案,可以借助手淘的开源项目lib-flexible。它可以自定帮你设置html 标签的 font-size等。将1rem设置为屏幕的1/10

关于 rem 方案,大漠老师在使用Flexible实现手淘H5页面的终端适配中进行了详细的介绍。建议大家阅读一下。

如你所见,大漠老师也在近期对文正进行了更新,建议大家使用更方便的 vw 方案。

VW

vw 是视口宽度的1/100,用 vw 来做自适应再合适不过了。

比如如果你的设计图是 750px 的宽度。 对于 75px 的元素就可以设置为 1vw。 这样在宽度为 375px 的手机上的表现就是37.5px

当然,如果我们把每个 px 标注都手动转换的话,那也是很大的工作量, postcss-px-to-viewport可以自定帮你转换为 vw。 你只需要在配置时指定设计图宽度就可以了。

同样,强烈建议你去阅读以下大漠老师关于 vw 布局的文章 再聊移动端页面的适配

我的方案 vw + rem

vw 虽好,可惜却无法满足我的需求。因为 vw 是整个视口宽度的1%,如果单纯采用 vw 方案,是无法限制 body 最大、最小宽度的。

于是我便采用了 vw + rem。 如果屏幕宽度在需要自适应的宽度之内,则将html 标签的 font-size设置为 10vw。如果屏幕宽度超过最大或最小限制的话。则将html 标签的 font-size设置为固定值。类似于lib-flexible,我们将1rem设为了屏幕的1/10

具体 css 如下:

html {height: 100%;font-size: 10vw;
}
body {font-size: 16px;width: 100%;height: 100%;margin: 0 auto;
}@media screen and (max-width: 320px) {html{font-size: 32px;}body{min-width: 320px;}
}
@media screen and (min-width: 800px) {html{font-size: 80px;}body{max-width: 800px;}
}
复制代码

当然,这样在布局时,我们就需要使用rem单位来布局了。 当然设计图标注pxrem单位同样有现成的工具。博主使用的是postcss-pxtorem。

最终的效果:

 

效果图

 

 

以上介绍的适配方案,基本上就可以满足大部分的需求了。 下面我们来聊一聊我都遇到了哪些坑。

小数像素问题

由于我们的方案,所有元素根据屏幕宽度来自适应。因而很难保证转换后的像素为整数像素。

在未接触前端,或者说H5开发之前并没有认真考虑过小数像素的问题,最初以为就是在可现实的精度上四舍五入。真正开发时发现并不是这样的。

比如下面这个例子,同样的像素值表现就不一样:在线实例

 

 

 

IOSmacOS设备最小像素好像支持到了0.5px,所以上面的例子在苹果设备上表现并不是很明显。

但是毕竟大部分设备还是Androidwindows系统。

那么,到底浏览器是如何处理小数像素的呢? rem 产生的小数像素问题 这篇文章给出了答案:

浏览器在渲染时所做的舍入处理只是应用在元素的渲染尺寸上,其真实占据的空间依旧是原始大小。
也就是说如果一个元素尺寸是 0.625px,那么其渲染尺寸应该是 1px,空出的 0.375px 空间由其临近的元素填充;同样道理,如果一个元素尺寸是 0.375px,其渲染尺寸就应该是 0,但是其会占据临近元素 0.375px 的空间。
复制代码

那么在我们的方案里会出现什么问题呢?

  1. 缩放到低于1px的元素会时隐时现
  2. 两个同样宽度的元素因为各自周围的元素宽度不同,导致两元素相差1px
  3. 宽高相同的正方形,长宽不相等了
  4. border-radius: 50% 画的圆不圆了

对于第一个问题,一般都会出现在标注为1px的地方。所以大部分的插件 postcss-pxtorem 或者 postcss-px-to-viewport 都提供了最小转换像素的选项。 我们只要指定最小转换像素,对于比较小的像素(如:1px),就不转换为remvw了。当然1px在视网膜屏同样存在过粗的问题,我们在之后会讨论。

对于剩下的几个问题,目前本人也没找到特别好的办法,毕竟很多地方相差1px是可以接受的。只有一些比较小的元素会表现的比较明显,本人的解决办法是不通过插件自动转换为remvw,而是通过js根据设备宽度,计算出该元素在该设备下实际的px。取整后动态地设置到元素的style上。这样就不会出现上述问题了。

如果各位有更好的解决方案的话。欢迎留言讨论。

1px问题

由于上面小数像素的问题,我们并没有对1px的元素进行转换,所以对于750px的设计图上1px的细线,在屏幕宽度为375pxiphone6上依旧为1px,按比例应该是0.5px。所以设计同学会问:“为什么这条细线变粗了?” 我们也很无奈啊,因为0.5px显不出来啊……

但是转念一想,对于DPR=2甚至更高的设备,1px是由多个物理像素渲染的,其实是可以显示更细的线的。那么这样才能画出更细的线呢?

大漠老师又出场了,《再谈Retina下1px的解决方案》中给出了几种方案:

  1. viewport放大为device-widthdpr倍数,然后缩小1/dpr倍显示
  2. border-image设为一个一半透明一半显示的图片,以达到将边框一分为二的目的
  3. 同样是上面的原理,但是使用svg绘制图片
  4. 媒体查询配合伪元素,为伪元素设置1px的边框,然后缩小1/dpr倍显示

以上方案各有各的特点,2、3两个方案画出来的其实是0.5px,而1、4两个方案画出来的更接近物理像素的1px

cursor:pointer 元素点击背景变色的问题

对于添加了 cursor:pointer 属性的元素,在移动端点击时,背景会高亮。

为元素添加 -webkit-tap-highlight-color: transparent; 属性可以隐藏背景高亮。

Android浏览器下line-height垂直居中偏离的问题

我们常用的垂直居中方式就是使用line-height,但是这种方法在Android设备下并不能完全居中。

具体原因是因为Android中文字体排版的问题,可以参考 知乎:Android浏览器下line-height垂直居中为什么会偏离?

通过设置字体,确实能够解决一部分偏离的问题。但仍然会出现一些略微偏离的情况,据说与行高奇数偶数有关。不过已经不太容易分辨了,如果还是不能接受的话建议通过设置上下padding的方式进行垂直居中,再根据具体情况进行微调。

参考文章

  • 使用Flexible实现手淘H5页面的终端适配
  • 再聊移动端页面的适配
  • rem 产生的小数像素问题
  • 再谈Retina下1px的解决方案
  • Android浏览器下line-height垂直居中为什么会偏离?


作者:sjfkai
链接:https://juejin.im/post/5c8870c1e51d453ce668cc7f
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

这篇关于HTML5 移动端自适应方案与踩坑的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/509127

相关文章

Vue3 的 shallowRef 和 shallowReactive:优化性能

大家对 Vue3 的 ref 和 reactive 都很熟悉,那么对 shallowRef 和 shallowReactive 是否了解呢? 在编程和数据结构中,“shallow”(浅层)通常指对数据结构的最外层进行操作,而不递归地处理其内部或嵌套的数据。这种处理方式关注的是数据结构的第一层属性或元素,而忽略更深层次的嵌套内容。 1. 浅层与深层的对比 1.1 浅层(Shallow) 定义

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

无人叉车3d激光slam多房间建图定位异常处理方案-墙体画线地图切分方案

墙体画线地图切分方案 针对问题:墙体两侧特征混淆误匹配,导致建图和定位偏差,表现为过门跳变、外月台走歪等 ·解决思路:预期的根治方案IGICP需要较长时间完成上线,先使用切分地图的工程化方案,即墙体两侧切分为不同地图,在某一侧只使用该侧地图进行定位 方案思路 切分原理:切分地图基于关键帧位置,而非点云。 理论基础:光照是直线的,一帧点云必定只能照射到墙的一侧,无法同时照到两侧实践考虑:关

【 html+css 绚丽Loading 】000046 三才归元阵

前言:哈喽,大家好,今天给大家分享html+css 绚丽Loading!并提供具体代码帮助大家深入理解,彻底掌握!创作不易,如果能帮助到大家或者给大家一些灵感和启发,欢迎收藏+关注哦 💕 目录 📚一、效果📚二、信息💡1.简介:💡2.外观描述:💡3.使用方式:💡4.战斗方式:💡5.提升:💡6.传说: 📚三、源代码,上代码,可以直接复制使用🎥效果🗂️目录✍️

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

高效+灵活,万博智云全球发布AWS无代理跨云容灾方案!

摘要 近日,万博智云推出了基于AWS的无代理跨云容灾解决方案,并与拉丁美洲,中东,亚洲的合作伙伴面向全球开展了联合发布。这一方案以AWS应用环境为基础,将HyperBDR平台的高效、灵活和成本效益优势与无代理功能相结合,为全球企业带来实现了更便捷、经济的数据保护。 一、全球联合发布 9月2日,万博智云CEO Michael Wong在线上平台发布AWS无代理跨云容灾解决方案的阐述视频,介绍了

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

计算机毕业设计 大学志愿填报系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点赞 👍 收藏 ⭐评论 📝 🍅 文末获取源码联系 👇🏻 精彩专栏推荐订阅 👇🏻 不然下次找不到哟~Java毕业设计项目~热门选题推荐《1000套》 目录 1.技术选型 2.开发工具 3.功能

Vue3项目开发——新闻发布管理系统(六)

文章目录 八、首页设计开发1、页面设计2、登录访问拦截实现3、用户基本信息显示①封装用户基本信息获取接口②用户基本信息存储③用户基本信息调用④用户基本信息动态渲染 4、退出功能实现①注册点击事件②添加退出功能③数据清理 5、代码下载 八、首页设计开发 登录成功后,系统就进入了首页。接下来,也就进行首页的开发了。 1、页面设计 系统页面主要分为三部分,左侧为系统的菜单栏,右侧

JavaFX应用更新检测功能(在线自动更新方案)

JavaFX开发的桌面应用属于C端,一般来说需要版本检测和自动更新功能,这里记录一下一种版本检测和自动更新的方法。 1. 整体方案 JavaFX.应用版本检测、自动更新主要涉及一下步骤: 读取本地应用版本拉取远程版本并比较两个版本如果需要升级,那么拉取更新历史弹出升级控制窗口用户选择升级时,拉取升级包解压,重启应用用户选择忽略时,本地版本标志为忽略版本用户选择取消时,隐藏升级控制窗口 2.