原文链接:http://caibaojian.com/flexible-js.html

本文讲的通过flexible.js实现了rem自适应,有了flexible.js,我们就不必再为移动端各种设备兼容烦恼,flexible.js是如何通过rem实现自适应的呢?一起来看看:·

通过rem与px的换算,你可以把设计稿从px转到rem。再也不用为各种设备横行而担忧。

rem是相对于根元素<html>,这样就意味着,我们只需要在根元素确定一个px字号,则可以来算出元素的宽高。1rem=16px(浏览器html的像素,可以设定这个基准值),假如浏览器的html设为64px,则下面的元素则1rem=64px来运算

阿里团队开源的一个库。使用flexible.js轻松搞定各种不同的移动端设备兼容自适应问题。

实现方法:

通过js来调整html的字体大小,而在页面中的制作稿则统一使用rem这个单位来制作。关键代码如下:

  1. ;(function(win, lib) {
  2. var doc = win.document;
  3. var docEl = doc.documentElement;
  4. var metaEl = doc.querySelector('meta[name="viewport"]');
  5. var flexibleEl = doc.querySelector('meta[name="flexible"]');
  6. var dpr = 0;
  7. var scale = 0;
  8. var tid;
  9. var flexible = lib.flexible || (lib.flexible = {});
  10.  
  11. if (metaEl) {
  12. console.warn('将根据已有的meta标签来设置缩放比例');
  13. var match = metaEl.getAttribute('content').match(/initial\-scale=([\d\.]+)/);
  14. if (match) {
  15. scale = parseFloat(match[1]);
  16. dpr = parseInt(1 / scale);
  17. }
  18. } else if (flexibleEl) {
  19. var content = flexibleEl.getAttribute('content');
  20. if (content) {
  21. var initialDpr = content.match(/initial\-dpr=([\d\.]+)/);
  22. var maximumDpr = content.match(/maximum\-dpr=([\d\.]+)/);
  23. if (initialDpr) {
  24. dpr = parseFloat(initialDpr[1]);
  25. scale = parseFloat((1 / dpr).toFixed(2));
  26. }
  27. if (maximumDpr) {
  28. dpr = parseFloat(maximumDpr[1]);
  29. scale = parseFloat((1 / dpr).toFixed(2));
  30. }
  31. }
  32. }
  33.  
  34. if (!dpr && !scale) {
  35. var isAndroid = win.navigator.appVersion.match(/android/gi);
  36. var isIPhone = win.navigator.appVersion.match(/iphone/gi);
  37. var devicePixelRatio = win.devicePixelRatio;
  38. if (isIPhone) {
  39. // iOS下,对于2和3的屏,用2倍的方案,其余的用1倍方案
  40. if (devicePixelRatio >= 3 && (!dpr || dpr >= 3)) {
  41. dpr = 3;
  42. } else if (devicePixelRatio >= 2 && (!dpr || dpr >= 2)){
  43. dpr = 2;
  44. } else {
  45. dpr = 1;
  46. }
  47. } else {
  48. // 其他设备下,仍旧使用1倍的方案
  49. dpr = 1;
  50. }
  51. scale = 1 / dpr;
  52. }
  53.  
  54. docEl.setAttribute('data-dpr', dpr);
  55. if (!metaEl) {
  56. metaEl = doc.createElement('meta');
  57. metaEl.setAttribute('name', 'viewport');
  58. metaEl.setAttribute('content', 'initial-scale=' + scale + ', maximum-scale=' + scale + ', minimum-scale=' + scale + ', user-scalable=no');
  59. if (docEl.firstElementChild) {
  60. docEl.firstElementChild.appendChild(metaEl);
  61. } else {
  62. var wrap = doc.createElement('div');
  63. wrap.appendChild(metaEl);
  64. doc.write(wrap.innerHTML);
  65. }
  66. }
  67.  
  68. function refreshRem(){
  69. var width = docEl.getBoundingClientRect().width;
  70. if (width / dpr > 540) {
  71. width = 540 * dpr;
  72. }
  73. var rem = width / 10;
  74. docEl.style.fontSize = rem + 'px';
  75. flexible.rem = win.rem = rem;
  76. }
  77.  
  78. win.addEventListener('resize', function() {
  79. clearTimeout(tid);
  80. tid = setTimeout(refreshRem, 300);
  81. }, false);
  82. win.addEventListener('pageshow', function(e) {
  83. if (e.persisted) {
  84. clearTimeout(tid);
  85. tid = setTimeout(refreshRem, 300);
  86. }
  87. }, false);
  88.  
  89. if (doc.readyState === 'complete') {
  90. doc.body.style.fontSize = 12 * dpr + 'px';
  91. } else {
  92. doc.addEventListener('DOMContentLoaded', function(e) {
  93. doc.body.style.fontSize = 12 * dpr + 'px';
  94. }, false);
  95. }
  96.  
  97. refreshRem();
  98.  
  99. flexible.dpr = win.dpr = dpr;
  100. flexible.refreshRem = refreshRem;
  101. flexible.rem2px = function(d) {
  102. var val = parseFloat(d) * this.rem;
  103. if (typeof d === 'string' && d.match(/rem$/)) {
  104. val += 'px';
  105. }
  106. return val;
  107. }
  108. flexible.px2rem = function(d) {
  109. var val = parseFloat(d) / this.rem;
  110. if (typeof d === 'string' && d.match(/px$/)) {
  111. val += 'rem';
  112. }
  113. return val;
  114. }
  115.  
  116. })(window, window['lib'] || (window['lib'] = {}));

原文链接:http://caibaojian.com/flexible-js.html

从上面的代码,主要是改变了dpx和document的font-size大小。大小为docEl.getBoundingClientRect().width / 10 + 'px';

假设我们的设计稿宽是640的,则html的字体大小则被设为64px.则相当于1rem=64px。

假如一个元素的宽是160px,在平时,我们可以采用百分比可以做到自适应,假如使用响应式的话,可能需要设置多个,比如在320px,输出80px,而在640px输出160px等。·

而采用以上rem的方法,则只需要输出2.5rem就能实现统一,如下表格:

设备宽度 320px 360px 414px 640px
Html字体大小 32px 36px 41.4px 64px
实际输出 1rem 1rem 1rem 1rem
设计稿缩放大小 80px 90px 103.5px 160px
实际输出 2.5rem 2.5rem 2.5rem 2.5rem

以上的2.5rem是怎么得出的呢?

160/64(1rem的基数为64px)=2.5rem;按照官方的说法(640px举例)

Flexible会将视觉稿分成100份(主要为了以后能更好的兼容vhvw),而每一份被称为一个单位a。同时1rem单位被认定为10a。针对我们这份视觉稿可以计算出:(设计稿为750px为例)

  1. 1a = 7.5px
  2. 1rem = 75px

那么我们这个示例的稿子就分成了10a,也就是整个宽度为10rem<html>对应的font-size75px

这样一来,对于视觉稿上的元素尺寸换算,只需要原始的px值除以rem基准值即可。例如此例视觉稿中的图片,其尺寸是176px * 176px,转换成为2.346667rem * 2.346667rem

另外大漠还写了一篇详细的文章:使用Flexible实现手淘H5页面的终端适配 里面介绍了一个如何快速转换rem为px的几种方法,感兴趣的童鞋可以去看看。

也可以看这篇文章:Sass函数功能——rem转px

另外在使用这个来处理自适应的另一个坑就是css sprite,作者的建议是使用svg,或者icon font.或者base64等其他方案。

另外就是在dpr=2时,小图片可能会出现模糊,建议以最大的图片来切图。

字体建议使用px

在作者的观点中,是建议描述性的字体使用px,如果有slogan之类大于48px的,可以使用rem,由于使用rem在iPhone5和iPhone6中字体不同,可能出现13px和15px,点阵字体。
显然,我们在iPhone3G和iPhone4的Retina屏下面,希望看到的文本字号是相同的。也就是说,我们不希望文本在Retina屏幕下变小,另外,我们希望在大屏手机上看到更多文本,以及,现在绝大多数的字体文件都自带一些点阵尺寸,通常是16px24px,所以我们不希望出现13px15px这样的奇葩尺寸

如此一来,就决定了在制作H5的页面中,rem并不适合用到段落文本上。所以在Flexible整个适配方案中,考虑文本还是使用px作为单位。只不过使用[data-dpr]属性来区分不同dpr下的文本字号大小。

  1. div {
  2. width: 1rem;
  3. height: 0.4rem;
  4. font-size: 12px; // 默认写上dpr为1的fontSize
  5. }
  6. [data-dpr="2"] div {
  7. font-size: 24px;
  8. }
  9. [data-dpr="3"] div {
  10. font-size: 36px;
  11. }

  

为了能更好的利于开发,在实际开发中,我们可以定制一个font-dpr()这样的Sass混合宏:

  1. @mixin font-dpr($font-size){
  2. font-size: $font-size;
  3.  
  4. [data-dpr="2"] & {
  5. font-size: $font-size * 2;
  6. }
  7.  
  8. [data-dpr="3"] & {
  9. font-size: $font-size * 3;
  10. }
  11. }

  

有了这样的混合宏之后,在开发中可以直接这样使用:

  1. //code from http://caibaojian.com/flexible-js.html
  1. @include font-dpr(16px);

  

当然这只是针对于描述性的文本,比如说段落文本。但有的时候文本的字号也需要分场景的,比如在项目中有一个slogan,业务方希望这个slogan能根据不同的终端适配。针对这样的场景,完全可以使用rem给slogan做计量单位。

flexible.js 布局详解的更多相关文章

  1. Android 布局学习之——Layout(布局)详解二(常见布局和布局参数)

    [Android布局学习系列]   1.Android 布局学习之——Layout(布局)详解一   2.Android 布局学习之——Layout(布局)详解二(常见布局和布局参数)   3.And ...

  2. Android开发重点难点1:RelativeLayout(相对布局)详解

    前言 啦啦啦~博主又推出了一个新的系列啦~ 之前的Android开发系列主要以完成实验的过程为主,经常会综合许多知识来写,所以难免会有知识点的交杂,给人一种混乱的感觉. 所以博主推出“重点难点”系列, ...

  3. 【翻译】Anatomy of a Program in Memory—剖析内存中的一个程序(进程的虚拟存储器映像布局详解)

    [翻译]Anatomy of a Program in Memory—剖析内存中的一个程序(进程的虚拟存储器映像布局详解) . . .

  4. Android布局详解之一:FrameLayout

      原创文章,如有转载,请注明出处:http://blog.csdn.net/yihui823/article/details/6702273 FrameLayout是最简单的布局了.所有放在布局里的 ...

  5. Android 布局详解

    Android 布局详解 1.重用布局 当一个布局文件被多处使用时,最好<include>标签来重用布局. 例如:workspace_screen.xml的布局文件,在另一个布局文件中被重 ...

  6. js对象详解(JavaScript对象深度剖析,深度理解js对象)

    js对象详解(JavaScript对象深度剖析,深度理解js对象) 这算是酝酿很久的一篇文章了. JavaScript作为一个基于对象(没有类的概念)的语言,从入门到精通到放弃一直会被对象这个问题围绕 ...

  7. DevExpress控件GridControl中的布局详解 【转】

    DevExpress控件GridControl中的布局详解 [转] 2012-10-24 13:27:28|  分类: devexpress |  标签:devexpress  |举报|字号 订阅   ...

  8. 【转载】图说C++对象模型:对象内存布局详解

    原文: 图说C++对象模型:对象内存布局详解 正文 回到顶部 0.前言 文章较长,而且内容相对来说比较枯燥,希望对C++对象的内存布局.虚表指针.虚基类指针等有深入了解的朋友可以慢慢看.本文的结论都在 ...

  9. 弹性盒布局详解(display: flex;)

    弹性盒布局详解 弹性盒介绍 弹性盒的CSS属性 开启弹性盒 弹性容器的CSS属性 flex-direction设置弹性元素在弹性容器中的排列方向 主轴与侧轴(副轴) flex-wrap设置弹性容器空间 ...

随机推荐

  1. MongoDB自学------(5)MongoDB分片

    分片 在Mongodb里面存在另一种集群,就是分片技术,可以满足MongoDB数据量大量增长的需求. 当MongoDB存储海量的数据时,一台机器可能不足以存储数据,也可能不足以提供可接受的读写吞吐量. ...

  2. c#汉字转拼音首字母全拼支持多音字

    1.首先在NuGet安装pingyinConverter 2.下载-安装-引用ChineseChar.dll到项目中 官网了解:http://www.microsoft.com/zh-cn/downl ...

  3. 聊聊业务系统中投递消息到mq的几种方式

    背景 电商中有这样的一个场景: 下单成功之后送积分的操作,我们使用mq来实现 下单成功之后,投递一条消息到mq,积分系统消费消息,给用户增加积分 我们主要讨论一下,下单及投递消息到mq的操作,如何实现 ...

  4. ABAP ALV显示前排序合并及布局显示

    有时候会有用户要求显示出来的ALV立即就是升序或者降序,或者是上下同一个字段值一样的情况显示一次,如 变为 这个时候内表用SORT有时候会不好用,可以使用函数 REUSE_ALV_GRID_DISPL ...

  5. 练手WPF(二)——2048游戏的简易实现(上)

    1.创建游戏界面编辑MainWindow.xaml,修改代码如下: <Window.Resources> <Style TargetType="Label"> ...

  6. 初学Python常见异常错误,总有一处你会遇到!

    初学Python常见错误 忘记写冒号 误用= 错误 缩紧 变量没有定义 中英文输入法导致的错误 不同数据类型的拼接 索引位置问题 使用字典中不存在的键 忘了括号 漏传参数 缺失依赖库 使用了pytho ...

  7. JS基础语法---break关键字

    break关键字: 如果在循环中使用,遇到了break,则立刻跳出当前所在的循环       for (var i = 0; i < 10; i++) {         while (true ...

  8. 2019年上半年收集到的人工智能LSTM干货文章

    2019年上半年收集到的人工智能LSTM干货文章 门控神经网络:LSTM 和 GRU 简要说明 LSTM-CNN-Attention算法系列之一:LSTM提取时间特征 对时间序列分类的LSTM全卷积网 ...

  9. PHP将数组转字符串

    implode(',',$arr) //将数组转字符串 $arr = [ 'a'=>1, 'b'=>2, 'c'=>3, ]; $arr_string = implode(',',$ ...

  10. 基于Spark.NET和ML.NET Automated ML (自动学习)进行餐厅等级的检查预测

    简介 Apache Spark是一个开源.分布式.通用的分析引擎.多年来,它一直是大数据生态系统中对大型数据集进行批量和实时处理的主要工具.尽管对该平台的本地支持仅限于JVM语言集,但其他通常用于数据 ...