前言
2020年清明小假期,虽然新型冠状病毒疫情得到了很好的控制,但是大家还是尽量避免出门,在家待着不能浪费时间,让我们重新思考一下移动端适配方案吧。
随着Web技术的革新,移动端适配方案也在不断的变化,网上有很多关于移动端适配的文章,说什么rem
布局已经过时,vm
适配才是最好的适配方案。有这种理解的同学是错误的,任何适配方案都有它的优缺点,要结合自己的使用场景来进行选择。
文章先讲一下几种常见的适配方案,然后再看看几个大厂(包括拍拍贷、小红书、微博、美团、B站、搜狐、京东、网易、饿了么、携程、大众点评、知乎、腾讯、陆金所)的移动端页面都采用了什么样的适配方案,最后讨论下各个适配方法的适用场景和优缺点,如果有不对之处,希望能得到大佬们的指正。
移动端适配的重新思考
移动端适配就是用rem或vw?
答:并不是所有场景都适合用用rem
或vw
进行适配。
-
vw
和rem
适配的本质是等比例缩放,让页面在不同屏幕尺寸下有类似于矢量图片缩放的效果,保证了页面元素之间的尺寸缩放比例和位置。 - 这两张适配方案适合视觉组件种类比较多,视觉设计对元素位置的相对关系依赖较强的移动端页面,基本上大部分页面都可以用着两种方案进行适配。
- 但对于文本内容较多,我们希望引导用户沉浸在更多的内容而不是更大的内容的,这种等比例缩放的方案并不能满足要求,我推荐直接使用px结合flex等布局方式进行适配。
rem该抛弃了,使用vw不香么?
答:vw
适配不是万能的,最好与rem
配合使用
当初之所以使用rem的方案流行开来正是因为在那时
viewport units
的浏览器支持程度不甚理想(IOS 8+, Android 4.4+
参见viewport units
的caniuse)。而相比较之下rem
就好多了(IOS 4.1+, Android 2.1+
参见caniuse),所以对于vw
,在当时的大环境下前端想说爱你不容易。随着前端技术的革新,最主要是各大浏览器厂商的给力,除Opera Mini全版本和IE低版本不支持之外,其他的浏览器基本上都已经支持vw了,开始有人或者有团队在探讨论在实际项目中的使用。虽然大漠老师在《再聊移动端页面的适配》一文中提出的
vw
方案中使用viewport-units-buggyfill
库进行兼容的做法,我个人更是不建议,由于这个库使用了css content
属性进行兼容处理,官方文档中就指出了对部分浏览器的img
标签有影响 ,需要全局引入一条css
规则。且对于需要正常使用content
的情况(如:图标字体)也会引起不可避免的冲突,另外也不支持伪元素的兼容。所以从我个人的角度来说,如果你一定要问我使用怎样的vw
适配方案,我会推荐给你上述两种vw + rem
的方案。虽然采用
vw
适配后的页面效果很好,但是它是利用视口单位实现的布局,依赖视口大小而自动缩放,无论视口过大还是过小,它也随着视口过大或者过小,失去了最大最小宽度的限制。
移动端适配方案
1. rem适配
rem
适配的本质是布局等比例的缩放,通过动态设置html
的font-size
来改变rem
的大小。
viewport 配置
上面把scale
设置成固定1倍的视口的大小,也可以根据dpr
的值缩放viewport
,如下:
//下面是根据设备dpr设置viewport
var dpr = window.devicePixelRatio || 1
var scale = 1 / dpr
viewport.setAttribute(
"content",
"width=device-width" +
",initial-scale=" +
scale +
", maximum-scale=" +
scale +
", minimum-scale=" +
scale +
", user-scalable=no"
)
有几点注意:
-
viewport
标签只对移动端浏览器有效,对PC
端浏览器是无效的。 - 当缩放比例为
100%
时,逻辑像素 = CSS 像素宽度 = 理想视口的宽度 = 布局视口的宽度
。 - 单独设置
initial-scale
或width
都会有兼容性问题,所以设置布局视口为理想视口的最佳方法是同时设置这两个属性。 - 即使设置了
user-scalable = no
,在Android Chrome
浏览器中也可以强制启用手动缩放。
设置 rem 基准值
核心代码为:
// set 1rem = 逻辑像素(设备独立像素) / 10
function setRemUnit () {
var rem = document.documentElement.clientWidth / 10
// 375/10 = 37.5
docEl.style.fontSize = rem + 'px'
}
setRemUnit()
- 将html节点的font-size设置为页面clientWidth(布局视口)的1/10,即:
1rem
= 布局视口的1/10, - 在
iphone6
下:docEl.clientWidth
=设备独立像素(逻辑像素)= 布局视口宽度 = 理想窗口宽度 = 375。此时:1rem = 375/10 +px = 37.5px
postcss-pxtorem将单位转化为 rem
module.exports = {
plugins: {
'autoprefixer': {
browsers: ['Android >= 4.0', 'iOS >= 7']
},
'postcss-pxtorem': {
rootValue: 37.5,
propList: ['*', '!font-size'],
selectorBlackList: ['van-circle__layer', 'ignore'],
}
}
}
-
rootValue
是转换px
的基准值,参考设备iPhone6
,设备宽度375px
规则:基准值=当前设备宽度的1/10 - 基准值设置代码中,在
iPhone6
设备设置的html—>font-size
也为37.5px
- 但是设计稿尺寸
750px
大小,所以在设计稿量取的尺寸使用时候需要除以2
rem布局的缺点
在响应式布局中,必须通过js来动态控制根元素font-size
的大小,也就是说css样式和js代码有一定的耦合性,且必须将改变font-size的代码放在css
样式之前。
2. vw适配
vw
是基于Viewport视窗的长度单位,这里的视窗(Viewport
)指的就是浏览器可视化的区域,而这个可视区域是window.innerWidth/window.innerHeight
的大小。用下图简单的来示意一下:
在CSS Values and Units Module Level 3中和Viewport相关的单位有四个,分别为vw
、vh
、vmin
和vmax
。
-
vw
:是Viewport's width的简写,1vw
等于window.innerWidth
的1%
-
vh
:和vw
类似,是Viewport's height的简写,1vh
等于window.innerHeihgt
的1%
-
vmin
:vmin
的值是当前vw
和vh
中较小的值 -
vmax
:vmax
的值是当前vw
和vh
中较大的值
如果设计稿用750px
宽度的,100vw = 750px
,即1vw = 7.5px
。那么我们可以根据设计图上的px
值直接转换成对应的vw
值。如果不想自己计算,我们可以使用PostCSS的插件postcss-px-to-viewport,让我们可以直接在代码中写px
。
{
loader: 'postcss-loader',
options: {
plugins: ()=>[
require('autoprefixer')({
browsers: ['last 5 versions']
}),
require('postcss-px-to-viewport')({
viewportWidth: 375, //视口宽度(数字)
viewportHeight: 1334, //视口高度(数字)
unitPrecision: 3, //设置的保留小数位数(数字)
viewportUnit: 'vw', //设置要转换的单位(字符串)
selectorBlackList: ['.ignore', '.hairlines'], //不需要进行转换的类名(数组)
minPixelValue: 1, //设置要替换的最小像素值(数字)
mediaQuery: false//允许在媒体查询中转换px(true/false)
})
]
}
3. 搭配vw和rem
- 给根元素大小设置随着视口变化而变化的vw单位,这样就可以实现动态改变其大小。
- 限制根元素字体大小的最大最小值,配合body加上最大宽度和最小宽度。
// rem 单位换算:定为 75px 只是方便运算,750px-75px、640-64px、1080px-108px,如此类推
$vm_fontsize: 75; // iPhone 6尺寸的根元素大小基准值
@function rem($px) {
@return ($px / $vm_fontsize ) * 1rem;
}
// 根元素大小使用 vw 单位
$vm_design: 750;
html {
font-size: ($vm_fontsize / ($vm_design / 2)) * 100vw;
// 同时,通过Media Queries 限制根元素最大最小值
@media screen and (max-width: 320px) {
font-size: 64px;
}
@media screen and (min-width: 540px) {
font-size: 108px;
}
}
// body 也增加最大最小宽度限制,避免默认100%宽度的 block 元素跟随 body 而过大过小
body {
max-width: 540px;
min-width: 320px;
}
4. px 适配
就像开篇提到的,并不是说移动端就一定要使用相对长度单位,传统的响应式布局依然是很好的选择,尤其在新闻,社区等可阅读内容较多的场景直接使用px
单位可以营造更好地体验。px方案可以让大屏幕手机展示出更多的内容,更符合人们的阅读习惯。
互联网大厂的适配调研
1. rem适配例子
1.1 固定1倍vieport
注:下面描述的rem
与px
的对应关系是在设备独立像素为375px(iPhone6/7/8)
情况下。
拍拍贷m站首页
1rem = 20px
- 最大基准值为
40px
- 限制页面宽度
750px
小红书
1rem = 50px
- 最大基准值为
60px
- 字体和页面都进行缩放
- 配合
media query
,限制body的最大宽度
@media screen and (min-width: 768px)
body {
width: 450PX!important;
}
微博
- 字体和页面都进行缩放
- 基准值是根据
media query
生成的
1.2 可缩放vieport
注:下面描述的rem与px的对应关系是在设备独立像素为375px(iPhone6/7/8)、viewport scale 0.5
的情况下。
美团
1rem = 100px
B站主站
1rem = 46.875px
搜狐
1rem = 75px
2. vm适配例子
拍拍贷借款页
- 不限制页面宽度
- 无兼容性处理,个人不推荐
3. vm+rem适配例子
京东
- 固定
vieport
,元素布局上使用rem
单位 -
html
元素的font-size
使用vw + px fallback
的形式 - 当页面超过一定宽度时,根据
media query
设置font-size
为px
,优先级高于vw
。 - 限制页面宽度为
1080px
网易
- 固定
viewport
,元素布局上使用rem
单位 -
html
元素的font-size
使用vw + px fallback
的形式 - 使用
media query
设置根元素font-size
中px
的值 - 当页面超过一定宽度时,
px
单位的优先级高于vw
- 限制布局宽度为
768px
饿了么
- 对
viewport
进行了缩放 -
html
元素的font-size
依然由px指定 - 具体元素的布局上使用
vw + rem fallbak
的形式 - 没有限制布局宽度
-
css
构建过程需要插件支持,可参考这个插件:pandaGao/stylus-px-to-relative-unit
4. px 方案例子
携程
- 固定
1倍vieport
- 布局方案:
px+flex+百分比
- 设置
body
的最大宽度为max-width: 540px;
大众点评
- 元素较丰富,采用
px+flex
布局,适配效果很好
知乎
- 追求阅读体验的场景,使用
px
布局。
腾讯
- 首页主要内容是新闻,为了更好的阅读体验,使用
px
布局。
陆金所
-
:root {font-size:10px;}
,并没有根据屏幕的大小来设置不同的font-size
- 存在问题:布局页面设成
1rem
时候,在chrome
浏览器上任然12px
并不是10px
- 布局中虽然用了
rem
单位,但其实还是绝对单位方案 - 可能希望用户在大屏手机上能看到更多内容吧
总结
- 新闻,社区等可阅读内容较多的场景:
px+flex+百分比
- 对视觉组件种类较多,视觉设计对元素位置的相对关系依赖较强的移动端页面:
vw + rem
以上只是自己的拙见以及自己这一两年有关于移动端适配的一些探索,如果有不对之处,还请各路大神指正。