rem实现移动端自适应页面

一、把px转换成rem方案

1、cssrem插件

2、css预处理器

3、rem-unit插件

4、px2rem插件

rem就是相对于根元素的font-size来做计算,设置好根结点字体大小,子节点用rem做单位,实现自适应。

二、动态改变字体大小方案

1、css方案之媒体查询

设置html的font-size

@media screen and (min- 320px) {
    html {font-size: 14px;}
}
@media screen and (min- 360px) {
    html {font-size: 16px;}
}
@media screen and (min- 400px) {
    html {font-size: 18px;}
}
@media screen and (min- 440px) {
    html {font-size: 20px;}
}
@media screen and (min- 480px) {
    html {font-size: 22px;}
}
@media screen and (min- 640px) {
    html {font-size: 28px;}
}

2、js方案A

window.onload=function() {
    function reCalc() {
         var windowWidth = document.documentElement.clientWidth || window.innerWidth || document.body.clientWidth;
         // windowWidth = windowWidth > 750 ? 750 : windowWidth;
         var rootSize = 28 * (windowWidth / 375);
         var htmlNode = document.getElementsByTagName("html")[0];
         htmlNode.style.fontSize = rootSize + 'px';
    }
    reCalc();
    window.addEventListener('resize', reCalc, false);
}

3、js方案B

!(function(doc, win) {
  var docEl = doc.documentElement,
  resizeEvt = 'orientationchange' in window ? 'orientationchange' : 'resize',
  reCalc = function() {
    var clientWidth = docEl.clientWidth;
    if (!clientWidth) return;
    docEl.style.fontSize = 28 * (clientWidth / 375) + 'px';
  };
  if (!doc.addEventListener) return;
  win.addEventListener(resizeEvt, reCalc, false);
  doc.addEventListener('DOMContentLoaded', reCalc, false);
})(document, window);

4、js方案C

设备像素比 = 物理像素 / 设备独立像素

A、设备像素比
通过window.devicePixelRatio来获取设备像素比,可以通过-webkit-device-pixel-ratio,-webkit-min-device-pixel-ratio和 -webkit-max-device-pixel-ratio进行媒体查询,对不同DPR的设备,做一些样式适配。
B、物理像素
肉眼能够看到的大小,往往不真实。
C、设备独立像素
一般是个绝对值,比如css像素。
如果css边框为1px,放在iphone的retina屏(设备像素比=2)下,会以为看到的是2px。

在此手淘方案中,为了在一些设备中,保持文本大小相同,文本大小依然用px为单位,宽高,内外边距要用rem为单位。安装pxtorem插件,写一个div,即可生成不同dpr的div样式。

div { 
   1rem; 
  height: 0.4rem; 
  font-size: 12px; /* 默认写上dpr为1的fontSize */
} 
[data-dpr="2"] div { 
  font-size: 24px; 
} 
[data-dpr="3"] div { 
  font-size: 36px; 
}

该js手淘适配方案,模拟viewport,只对iOS设备进行dpr的判断,对于Android系列,始终认为其dpr为1。该手淘方案在iframe中有bug。

  • 动态改写<meta>标签;
  • 给<html>元素添加data-dpr属性,并且动态改写data-dpr的值;
  • 给<html>元素添加font-size属性,并且动态改写font-size的值。
!function(win, lib) {
  var timer,
      doc     = win.document,
      docElem = doc.documentElement,

      vpMeta   = doc.querySelector('meta[name="viewport"]'),
      flexMeta = doc.querySelector('meta[name="flexible"]'),

      dpr   = 0,
      scale = 0,

      flexible = lib.flexible || (lib.flexible = {});

  // 设置了 viewport meta
  if (vpMeta) {
    console.warn("将根据已有的meta标签来设置缩放比例");
    var initial = vpMeta.getAttribute("content").match(/initial-scale=([d.]+)/);

    if (initial) {
        scale = parseFloat(initial[1]); // 已设置的 initialScale
        dpr = parseInt(1 / scale);      // 设备像素比 devicePixelRatio
    }
  }
  // 设置了 flexible Meta
  else if (flexMeta) {
    var flexMetaContent = flexMeta.getAttribute("content");
    if (flexMetaContent) {
      var initial = flexMetaContent.match(/initial-dpr=([d.]+)/),
          maximum = flexMetaContent.match(/maximum-dpr=([d.]+)/);

      if (initial) {
        dpr = parseFloat(initial[1]);
        scale = parseFloat((1 / dpr).toFixed(2));
      }

      if (maximum) {
        dpr = parseFloat(maximum[1]);
        scale = parseFloat((1 / dpr).toFixed(2));
      }
    }
  }
  // viewport 或 flexible
  // meta 均未设置
  if (!dpr && !scale) {
    var isAndroid = win.navigator.appVersion.match(/android/gi);
    var isIPhone = win.navigator.appVersion.match(/iphone/gi);
    var devicePixelRatio = win.devicePixelRatio;
    if (isIPhone) {
        // iOS下,对于2和3的屏,用2倍的方案,其余的用1倍方案
        if (devicePixelRatio >= 3 && (!dpr || dpr >= 3)) {                
            dpr = 3;
        } else if (devicePixelRatio >= 2 && (!dpr || dpr >= 2)){
            dpr = 2;
        } else {
            dpr = 1;
        }
    } else {
        // 其他设备下,仍旧使用1倍的方案
        dpr = 1;
    }
    scale = 1 / dpr;
  }
  docElem.setAttribute("data-dpr", dpr);
  // 插入 viewport meta
  if (!vpMeta) {
    vpMeta = doc.createElement("meta");
    vpMeta.setAttribute("name", "viewport");
    vpMeta.setAttribute("content",
        "initial-scale=" + scale + ", maximum-scale=" + scale + ", minimum-scale=" + scale + ", user-scalable=no");
    if (docElem.firstElementChild) {
      docElem.firstElementChild.appendChild(vpMeta)
    } else {
      var wrap = doc.createElement("div");
      wrap.appendChild(vpMeta);
      doc.write(wrap.innerHTML);
    }
  }
  function setFontSize() {
    var winWidth = docElem.getBoundingClientRect().width;
    if (winWidth / dpr > 540) {
        (winWidth = 540 * dpr);
    }
    // 根节点 fontSize 根据宽度决定
    var baseSize = winWidth / 10;
    docElem.style.fontSize = baseSize + "px";
    flexible.rem = win.rem = baseSize;
  }
  // 调整窗口时重置
  win.addEventListener("resize", function() {
    clearTimeout(timer);
    timer = setTimeout(setFontSize, 300);
  }, false);
  // 方向改变时重置
  win.addEventListener("orientationchange", function() {
    clearTimeout(timer);
    timer = setTimeout(setFontSize, 300);
  }, false);
  // 从浏览器读取缓存时触发,pageshow每次加载页面时触发。
  win.addEventListener("pageshow", function(e) {
    if (e.persisted) {
      clearTimeout(timer);
      timer = setTimeout(setFontSize, 300);
    }
  }, false);
  // 设置基准字体
  if ("complete" === doc.readyState) {
    doc.body.style.fontSize = 12 * dpr + "px";
  } else {
    doc.addEventListener("DOMContentLoaded", function() {
      doc.body.style.fontSize = 12 * dpr + "px";
    }, false);
  }
  setFontSize();
  flexible.dpr = win.dpr = dpr;
  flexible.refreshRem = setFontSize;
  flexible.rem2px = function(d) {
    var c = parseFloat(d) * this.rem;
    if (typeof d === "string" && d.match(/rem$/)) {
        c += "px";
    }
    return c;
  };
  flexible.px2rem = function(d) {
    var c = parseFloat(d) / this.rem;    
    if (typeof d === "string" && d.match(/px$/)) {
        c += "rem";
    }
    return c;
  }
}(window, window.lib || (window.lib = {}));

对于不同dpr,通过上面的js运行会产生不同的meta。

<!-- dpr = 1-->
<meta name="viewport" content="initial-scale=scale,maximum-scale=scale,minimum-scale=scale,user-scalable=no"> 
<!-- dpr = 2-->
<meta name="viewport" content="initial-scale=0.5,maximum-scale=0.5,minimum-scale=0.5,user-scalable=no">
<!-- dpr = 3-->
<meta name="viewport" content="initial-scale=0.3333333333,maximum-scale=0.3333333333,minimum-scale=0.3333333333,user-scalable=no">
原文地址:https://www.cnblogs.com/camille666/p/rem_autoadapt.html