移动端自适应解决方法(原理是rem布局)

2024-01-23 19:04

本文主要是介绍移动端自适应解决方法(原理是rem布局),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

adaptive

点击跳转官方Git地址
此博客参考笔记

也可以直接复制下方的js文件直接使用

index.js

var adaptive = {};
(function (win, lib) {var doc = win.document;var docEl = doc.documentElement;// 设备像素比var devicePixelRatio = win.devicePixelRatio;// 我们设置的布局视口与理想视口的像素比var dpr = 1; // viewport缩放值var scale = 1; // 设置viewportfunction setViewport() {// 判断IOSvar isIPhone = /iphone/gi.test(win.navigator.appVersion);if (lib.scaleType === 2 && isIPhone || lib.scaleType === 3) {dpr = devicePixelRatio;}// window对象上增加一个属性,提供对外的布局视口与理想视口的值win.devicePixelRatioValue = dpr;// viewport缩放值,布局视口缩放后刚好显示成理想视口的宽度,页面就不会过长或过短了scale = 1 / dpr;// 获取已有的viewportvar hasMetaEl = doc.querySelector('meta[name="viewport"]');var metaStr = 'initial-scale=' + scale + ', maximum-scale=' + scale + ', minimum-scale=' + scale + ', user-scalable=no';if (dpr === 1) {metaStr = 'width=device-width, '.concat(metaStr);}if (!isIPhone && dpr !== 1) {metaStr = metaStr.concat(', target-densitydpi=device-dpi');}// 如果有,改变之if (hasMetaEl) {hasMetaEl.setAttribute('content', metaStr);}// 如果没有,添加之else {var metaEl = doc.createElement('meta');metaEl.setAttribute('name', 'viewport');metaEl.setAttribute('content', metaStr);if (docEl.firstElementChild) {docEl.firstElementChild.appendChild(metaEl);}else {var containDiv = doc.createElement('div');containDiv.appendChild(metaEl);docEl.appendChild(containDiv);}}}var newBase = 100;lib.errDpr = 1;function setRem() {// 布局视口// var layoutView = docEl.clientWidth; 也可以 获取布局视口的宽度var layoutView;if (lib.maxWidth) {layoutView = Math.min(docEl.getBoundingClientRect().width, lib.maxWidth * dpr);}else {layoutView = docEl.getBoundingClientRect().width;}// 为了计算方便,我们规定 1rem === 100px设计图像素,我们切图的时候就能快速转换// 有人问,为什么不让1rem === 1px设计像素呢?// 设计图一般是640或者750px// 布局视口一般是320到1440// 计算一个值,使layout的总宽度为 (desinWidth/100) rem// 那么有计算公式:layoutView / newBase = desinWidth / 100// newBase = 100 * layoutView / desinWidth// newBase = 介于50到200之间// 如果 1rem === 1px 设计像素,newBase就介于0.5到2之间,由于很多浏览器有最小12px限制,这个时候就不能自适应了newBase = 100 * layoutView / lib.desinWidth * (lib.errDpr || 1);docEl.style.fontSize = newBase + 'px';// rem基准值改变后,手动reflow一下,避免旋转手机后页面自适应问题doc.body&&(doc.body.style.fontSize = lib.baseFont / 100 + 'rem');// 重新设置rem后的回调方法lib.setRemCallback&&lib.setRemCallback();lib.newBase = newBase;}var tid;lib.desinWidth = 750;lib.baseFont = 28;// 局部刷新的时候部分chrome版本字体过大的问题lib.reflow = function() {docEl.clientWidth;};// 检查安卓下rem值是否显示正确function checkRem() {if (/android/ig.test(win.navigator.appVersion)) {var hideDiv = document.createElement('p');hideDiv.style.height = '1px';hideDiv.style.width = '2.5rem';hideDiv.style.visibility = 'hidden';document.body.appendChild(hideDiv);var now = hideDiv.offsetWidth;var right = lib.newBase * 2.5; if (Math.abs(right / now - 1) > 0.05) {lib.errDpr = right / now;setRem();}document.body.removeChild(hideDiv);}}lib.init = function () {// resize的时候重新设置rem基准值// 触发orientationchange 事件时也会触发resize,故不需要再添加此事件了win.addEventListener('resize', function () {clearTimeout(tid);tid = setTimeout(setRem, 300);}, false);// 浏览器缓存中读取时也需要重新设置rem基准值win.addEventListener('pageshow', function (e) {if (e.persisted) {clearTimeout(tid);tid = setTimeout(setRem, 300);}}, false);// 设置body上的字体大小if (doc.readyState === 'complete') {doc.body.style.fontSize = lib.baseFont / 100 + 'rem';checkRem();}else {doc.addEventListener('DOMContentLoaded', function (e) {doc.body.style.fontSize = lib.baseFont / 100 + 'rem';checkRem();}, false);}setViewport();// 设置rem值setRem();// html节点设置布局视口与理想视口的像素比docEl.setAttribute('data-dpr', dpr);};// 有些html元素只能以px为单位,所以需要提供一个接口,把rem单位换算成pxlib.remToPx = function (remValue) {return remValue * newBase;};
})(window, adaptive);
if (typeof module != 'undefined' && module.exports) {module.exports = adaptive;
} else if (typeof define == 'function' && define.amd) {define(function() {return adaptive;});
} else {window.adaptive = adaptive;
}

在index.html使用

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="./index.js"></script><script>window['adaptive'].desinWidth = 750// 设计图宽度window['adaptive'].baseFont = 28// 没有缩放时的字体大小window['adaptive'].maxWidth = 750// 页面最大宽度 默认540window['adaptive'].init()// 调用初始化方法</script><style>div {width: 0.3rem;height: 0.3rem;background-color: red;}</style>
</head><body><div>123</div>
</body></html>

hotcss

点击跳转官网

index.js文件

;(function (window, document) {'use strict'//给hotcss开辟个命名空间,别问我为什么,我要给你准备你会用到的方法,免得用到的时候还要自己写。var hotcss = {};(function () {//根据devicePixelRatio自定计算scale//可以有效解决移动端1px这个世纪难题。var viewportEl = document.querySelector('meta[name="viewport"]'),hotcssEl = document.querySelector('meta[name="hotcss"]'),dpr = window.devicePixelRatio || 1,maxWidth = 540,designWidth = 0dpr = dpr >= 3 ? 3 : dpr >= 2 ? 2 : 1//允许通过自定义name为hotcss的meta头,通过initial-dpr来强制定义页面缩放if (hotcssEl) {var hotcssCon = hotcssEl.getAttribute('content')if (hotcssCon) {var initialDprMatch = hotcssCon.match(/initial\-dpr=([\d\.]+)/)if (initialDprMatch) {dpr = parseFloat(initialDprMatch[1])}var maxWidthMatch = hotcssCon.match(/max\-width=([\d\.]+)/)if (maxWidthMatch) {maxWidth = parseFloat(maxWidthMatch[1])}var designWidthMatch = hotcssCon.match(/design\-width=([\d\.]+)/)if (designWidthMatch) {designWidth = parseFloat(designWidthMatch[1])}}}document.documentElement.setAttribute('data-dpr', dpr)hotcss.dpr = dprdocument.documentElement.setAttribute('max-width', maxWidth)hotcss.maxWidth = maxWidthif (designWidth) {document.documentElement.setAttribute('design-width', designWidth)}hotcss.designWidth = designWidth // 保证px2rem 和 rem2px 不传第二个参数时, 获取hotcss.designWidth是undefined导致的NaNvar scale = 1 / dpr,content ='width=device-width, initial-scale=' +scale +', minimum-scale=' +scale +', maximum-scale=' +scale +', user-scalable=no'if (viewportEl) {viewportEl.setAttribute('content', content)} else {viewportEl = document.createElement('meta')viewportEl.setAttribute('name', 'viewport')viewportEl.setAttribute('content', content)document.head.appendChild(viewportEl)}})()hotcss.px2rem = function (px, designWidth) {//预判你将会在JS中用到尺寸,特提供一个方法助你在JS中将px转为rem。就是这么贴心。if (!designWidth) {//如果你在JS中大量用到此方法,建议直接定义 hotcss.designWidth 来定义设计图尺寸;//否则可以在第二个参数告诉我你的设计图是多大。designWidth = parseInt(hotcss.designWidth, 10)}return (parseInt(px, 10) * 320) / designWidth / 20}hotcss.rem2px = function (rem, designWidth) {//新增一个rem2px的方法。用法和px2rem一致。if (!designWidth) {designWidth = parseInt(hotcss.designWidth, 10)}//rem可能为小数,这里不再做处理了return (rem * 20 * designWidth) / 320}hotcss.mresize = function () {//对,这个就是核心方法了,给HTML设置font-size。var innerWidth =document.documentElement.getBoundingClientRect().width ||window.innerWidthif (hotcss.maxWidth && innerWidth / hotcss.dpr > hotcss.maxWidth) {innerWidth = hotcss.maxWidth * hotcss.dpr}if (!innerWidth) {return false}document.documentElement.style.fontSize = (innerWidth * 20) / 320 + 'px'hotcss.callback && hotcss.callback()}hotcss.mresize()//直接调用一次window.addEventListener('resize',function () {clearTimeout(hotcss.tid)hotcss.tid = setTimeout(hotcss.mresize, 33)},false)//绑定resize的时候调用window.addEventListener('load', hotcss.mresize, false)//防止不明原因的bug。load之后再调用一次。setTimeout(function () {hotcss.mresize()//防止某些机型怪异现象,异步再调用一次}, 333)window.hotcss = hotcss//命名空间暴露给你,控制权交给你,想怎么调怎么调。
})(window, document)

index.html使用

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="./index.js"></script><style>div {width: 0.3rem;height: 0.3rem;background-color: red;}</style>
</head><body><div>123</div>
</body></html>

具体效果打开浏览器终端看根元素HTML的font-size大小

这篇关于移动端自适应解决方法(原理是rem布局)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java 方法重载Overload常见误区及注意事项

《Java方法重载Overload常见误区及注意事项》Java方法重载允许同一类中同名方法通过参数类型、数量、顺序差异实现功能扩展,提升代码灵活性,核心条件为参数列表不同,不涉及返回类型、访问修饰符... 目录Java 方法重载(Overload)详解一、方法重载的核心条件二、构成方法重载的具体情况三、不构

SQL中如何添加数据(常见方法及示例)

《SQL中如何添加数据(常见方法及示例)》SQL全称为StructuredQueryLanguage,是一种用于管理关系数据库的标准编程语言,下面给大家介绍SQL中如何添加数据,感兴趣的朋友一起看看吧... 目录在mysql中,有多种方法可以添加数据。以下是一些常见的方法及其示例。1. 使用INSERT I

Python中反转字符串的常见方法小结

《Python中反转字符串的常见方法小结》在Python中,字符串对象没有内置的反转方法,然而,在实际开发中,我们经常会遇到需要反转字符串的场景,比如处理回文字符串、文本加密等,因此,掌握如何在Pyt... 目录python中反转字符串的方法技术背景实现步骤1. 使用切片2. 使用 reversed() 函

Python中将嵌套列表扁平化的多种实现方法

《Python中将嵌套列表扁平化的多种实现方法》在Python编程中,我们常常会遇到需要将嵌套列表(即列表中包含列表)转换为一个一维的扁平列表的需求,本文将给大家介绍了多种实现这一目标的方法,需要的朋... 目录python中将嵌套列表扁平化的方法技术背景实现步骤1. 使用嵌套列表推导式2. 使用itert

Python使用pip工具实现包自动更新的多种方法

《Python使用pip工具实现包自动更新的多种方法》本文深入探讨了使用Python的pip工具实现包自动更新的各种方法和技术,我们将从基础概念开始,逐步介绍手动更新方法、自动化脚本编写、结合CI/C... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

在Linux中改变echo输出颜色的实现方法

《在Linux中改变echo输出颜色的实现方法》在Linux系统的命令行环境下,为了使输出信息更加清晰、突出,便于用户快速识别和区分不同类型的信息,常常需要改变echo命令的输出颜色,所以本文给大家介... 目python录在linux中改变echo输出颜色的方法技术背景实现步骤使用ANSI转义码使用tpu

Conda与Python venv虚拟环境的区别与使用方法详解

《Conda与Pythonvenv虚拟环境的区别与使用方法详解》随着Python社区的成长,虚拟环境的概念和技术也在不断发展,:本文主要介绍Conda与Pythonvenv虚拟环境的区别与使用... 目录前言一、Conda 与 python venv 的核心区别1. Conda 的特点2. Python v

Spring Boot中WebSocket常用使用方法详解

《SpringBoot中WebSocket常用使用方法详解》本文从WebSocket的基础概念出发,详细介绍了SpringBoot集成WebSocket的步骤,并重点讲解了常用的使用方法,包括简单消... 目录一、WebSocket基础概念1.1 什么是WebSocket1.2 WebSocket与HTTP

SQL Server配置管理器无法打开的四种解决方法

《SQLServer配置管理器无法打开的四种解决方法》本文总结了SQLServer配置管理器无法打开的四种解决方法,文中通过图文示例介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录方法一:桌面图标进入方法二:运行窗口进入检查版本号对照表php方法三:查找文件路径方法四:检查 S

MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景)

《MyBatis-Plus中nested()与and()方法详解(最佳实践场景)》在MyBatis-Plus的条件构造器中,nested()和and()都是用于构建复杂查询条件的关键方法,但... 目录MyBATis-Plus 中nested()与and()方法详解一、核心区别对比二、方法详解1.and()