0%

前端如何实现视觉设计稿

在这篇文章中将和大家探讨一下关于前端在移动端开发如何去实现视觉设计稿。探讨过后,在大家的实际工作中或许能帮助解决一些问题。

前端工程师需要明白的「像素」

一般设计稿是640px或者750px(现在最流行),但是iPhone 5不是320px宽吗,iPhone 6不是375px宽吗?
这里需要理解一下基础概念: 设备像素(device pixel),CSS像素(css pixel)以及设备像素比(device pixel ratio)。

  • 设备像素(device pixel):
    设备像素设是物理概念,指的是设备中使用的物理像素。
    比如iPhone 5的分辨率640 x 1136px

  • CSS像素(css pixel):
    CSS像素是Web编程的概念,指的是CSS样式代码中使用的逻辑像素。
    在CSS规范中,长度单位可以分为两类,绝对(absolute)单位以及相对(relative)单位。px是一个相对单位,相对的是设备像素(device pixel)。

  • 设备像素比(device pixel ratio):
    window.devicePixelRatio,是设备上物理像素和设备独立像素(device-independent pixels (dips))的比例。
    公式表示就是window.devicePixelRatio = 物理像素 / dips

垂直手机屏幕下,使用<meta name="viewport" content="width=device-width">,iphone5屏幕物理像素640像素,独立像素还是320像素,因此,window.devicePixelRatio等于2。

Image text

比如iPhone 5,6使用的是Retina视网膜屏幕(2倍屏),6plus是3倍屏,使用2px x 2px的device pixel 代表1px x 1px css pixel,所以设备像素数为640 x 1136px(5),750 x 1134(6),而CSS逻辑像素数为320 x 568px(5),375 x 667(6);5,6的window.devicePixelRatio=2,6plus 为3。

Image text

H5适配:rem方案

rem:是CSS3新增的一个相对单位,相对于html标签font-size的大小为基础的。而font-size的大小可以动态根据手机屏幕宽度document.documentElement.clientWidth来设置,从而达到自适应屏幕的目的。

我这里找了一下小米网易拉勾网手淘 以及糯米,大同小异。

小米官网

Image text

设计稿是720px的,即5英寸屏幕的安卓手机(720 x 1280px)。
对于页面缩放和横竖屏事件进行监听,改变html根元素字体clientWidth/720/100
如图是这样计算的375/(720/100) = 52.0833

网易

Image text

Image text

iphone6 : 375/7.5=50, 则知道设计稿应该是基于iphone6来的,所以它的设计稿竖直放时的横向分辨率为750px,为了计算方便,取一个100px的font-size为参照,那么body元素的宽度就可以设置为width: 7.5rem,于是html的font-size=deviceWidth / 7.5。布局时,设计图标注的尺寸除以100得到css中的尺寸。并且布局中的font-size也可用rem单位。

拉勾网

Image text
Image text

1
2
3
4
html {
font-size: 65.5%;
}

设置html根元素字体为65.5%,对应px单位则为10.48px,则列表里时间信息字体设置为1rem = 10.48px,chrome在-webkit-text-size-adjust: 100%;情况下小于12px的一律显示为12px

拉勾网页面列表部分是px为单位,字体是rem,底部bar是使用百分百来控制宽高间距。

之前网上讨论的比较多的是

1
2
3
4
5
6
body {
font-size: 62.5%;
}
p {
font-size: 1.2em;
}

1em = 16px * 62.5% = 10px,em的初始值为1em = 16px,而为了方便计算, 换算一下10 / 16(16px是chrome浏览器默认字体大小)。缺点是进行任何元素设置,都有可能需要知道他父元素的大小,比较繁琐低效。

手淘

Image text

(1)动态设置viewport的scale

1
2
var scale = 1 / devicePixelRatio;
document.querySelector('meta[name="viewport"]').setAttribute('content','initial-scale=' + scale + ', maximum-scale=' + scale + ', minimum-scale=' + scale + ', user-scalable=no');

(2)动态计算html的font-size

1
document.documentElement.style.fontSize = document.documentElement.clientWidth / 10 + 'px';

(3)布局的时候,各元素的css尺寸=设计稿标注尺寸/设计稿横向分辨率/10

设计稿是750的,所以html的font-size就是75,如果某个元素是150px的宽,换算成rem就是150 / 75 = 2rem

整个手淘设计师和前端开发的适配协作基本思路是:

  • 选择一种尺寸作为设计和开发基准
  • 定义一套适配规则,自动适配剩下的两种尺寸(介于iphone6的小屏和大屏)
  • 特殊适配效果给出设计效果

Image text

手淘推出了一套移动端适配的方案——flexible方案

总结来说:

  • 动态读取设备宽度并结合设备的像素比
  • 动态改变html的font-size大小 & 页面缩放比例
  • 影响以rem为单位的元素的最终呈现

糯米

  • 糯米某活动H5
    Image text

设计稿是640的,html字体设置为屏幕宽度,以iphone6为例,font-size: 375px; 如果某个元素是150px的宽,换算成rem就是150/640rem


px方案:css尺寸为对应设计稿/2

设计稿是750的。
Image text
Image text

优点:简单粗暴,所有css尺寸均为设计稿尺寸直接除2,开发快速简单;
缺点:可能出现一排放不下的情况,需要针对小屏幕如5及以下做单独适配

vw方案

  • 糯米WAP
    利用CSS3中新增单位vw,配合百分比来做响应式开发。
单位 释义 说明
px 相对于显示器屏幕分辨率 -
em 相对于父元素字体大小 -
rem 相对于根元素字体大小 css3
vw 相对于视窗的宽度 css3
vh 相对于视窗的高度 css3

vw 相对于视窗的宽度:视窗宽度是100vw
如果视区宽度是100vm, 则1vm是视区宽度的1/100, 也就是1%,类似于width: 1%。
那iphone6来说,document.documentElement.clientWidth=375, 则豆腐块宽度为 375/100*30=112.5px

Image text
Image text


混合: rem px vw 百分百等单位混用

  • rem & 百分比%

    1
    2
    3
    4
    5
    6
    7
    8
    9
    body {
    padding-bottom: 14.0625%;
    }

    a.link {
    width: 30vw;
    height: 23vw;
    }

    略,同上糯米WAP

  • rem & vw

    1
    2
    3
    4
    html {  
    font-size: 4.375vw;
    }

    这里假设设计稿640px,则设置根元素font-size为4.375vw,根据屏幕宽度自适应,在视窗宽度为 320px 的时候,正好是 14px (14 / 320 = 0.04375)。 达到页面默认字体大小14px的目的(其他大小也ok)。好了,现在页面上所有以 rem 为单位的属性值都会随着屏幕的宽度变化而变化,达到自适应的目的。(自适应不用js动态设置根元素大小

1
2
3
4
p {  
font-size: 1rem; /* 设计稿上为 14px */
padding-top: 2rem: /* 设计稿上为 28px */
}

总结

在移动端页面开发中,视觉童鞋一般会用750px(iphone 6)来出设计稿,然后要求FE童鞋能够做到页面是自适应屏幕的,这种情况下就可以用rem或者vm等相对单位来做适配,愉快和视觉童鞋一起玩耍啦。