H5移动端适配方案rem/vw

news2025/2/25 4:33:40

前言

在讲解适配方案之前,我们聊聊为什么要做适配?

因为视口、逻辑像素、分辨率这些变量的不同,在每个设备展示都不一样,需要一种方案来统一。

物理像素:也就是分辨率,一个物理像素是显示器上的最小的物理显示单元,所以分辨率越高,屏幕可展示的像素点越多,也就更清晰。

逻辑像素:也叫设备独立像素,可以认为是计算机坐标系统中的一个点,这一个点代表可以由程序使用的虚拟像素(css像素),然后由相关系统转换为物理像素。

设备像素比:简称dpr(Device pixel ratio),定义了物理像素和设备独立像素的对应关系,设备像素比 = 物理像素 / 逻辑像素。

在移动端布局的时候,经常会有移动端如何展示1px的问题?

一般设计稿都是以750px宽度为基准,当设计师给出1px的数据的时候,对于视觉视口宽度为375px的手机,dpr如果为2,逻辑像素应该是0.5px,但移动端css最低只支持显示1px大小,这时候边框宽度就会是设计稿而言就会略粗一点。

视口

通常与浏览器窗口相同,但不包括浏览器的UI、菜单栏。

文档的 Element.clientWidth 是指一个文档使用 CSS pixels 单位表示的内部宽度,包括其 padding(不包括 border、margin 或垂直滚动条)。这就是视口的宽度。-----《MND文档》

移动设备的视口的默认值为 980px。

对于各种不同形状,不同设备像素比移动设备,其浏览器的视口(窗口中显示网页信息的区域)不一定与渲染页面大小相同。

移动设备的视口的默认值为 980px,一般情况下都要比这些设备的屏幕尺寸要大。

为了让页面能够全部展示,这些浏览器在渲染时会对页面进行缩放。

比如在一个宽 375px 的移动设备显示一个视觉视口宽为 980px 的页面,移动设备浏览器会对这个页面进行缩放直至其视觉视口宽度为 375px(具体取决于浏览器实现)。

但直接缩放页面会导致页面字体变小,使得缩放后的页面显示效果都不会很理想。
  在这里插入图片描述
在这里插入图片描述
在移动端,为了让页面展示更多的内容,视口有以下区分。

布局视口:移动设备视口的默认值是980px,比屏幕宽度大,目的是能够展示更多那些PC端的网页内容;

视觉视口:屏幕宽度;

理想视口:布局视口=视觉视口;

上文说到移动设备布局视口的默认值是980px,由于视觉视口只有屏幕宽度,比如375px,那么我们看到的页面就会有一定的缩放,这样的效果是不太好的。

那么如何创建一个理想视口,让移动端浏览器使用屏幕宽度作为视口替换默认的 980px 宽度视口呢?

我们可以在 HTML 的头部添加以下标签:

<meta name="viewport" content="width=device-width">

这里 width 属性指的是布局视口宽度,现在布局视口的宽度被设置为了设备的屏幕宽度,也就是布局视口=视觉视口。

即文档视口宽度大小与设备宽度大小 100% 对应(转换为 CSS 像素值相同),现在理想视口就实现了。

其他属性释义:

initial-scale(初始缩放比例)。

maximum-scale(最大缩放比例)。

minimum-scale(最小缩放比例)。

user-scalable(是否允许用户进行缩放操作)。

这些属性的默认值已经很好了,开发时可以忽略掉这些属性。
  
  
移动端适配方案

1、弹性盒模型布局。

2、流式布局(百分比布局)。

3、rem布局。

4、vw/vh。

5、响应式布局(媒体查询)。

常用适配方案

1、rem & vw。

2、rem & 媒体查询。

下面重点讲解常用适配方案。
  
rem适配

rem这个单位代表根元素(通常为 元素)的 font-size 大小。当用在根元素的 font-size 上面时,它代表了它的初始值。–《MDN》
原理与实现:

通俗来讲,rem是一个相对单位,根元素的font-size的大小为一个rem单位。

rem适配指的是,通过设置根元素的font-size的大小,后续子元素就可以使用rem为单位,实际CSS pixels 单位需要在这个基础上乘以根元素的font-size。

eg. 设置根元素font-size为16px,设置其中一个子元素宽度为20rem,那这个子元素实际的宽度为20*16px=320px
在这里插入图片描述
 一般移动端设计稿会以750px为基准,但由于每个移动设备的宽度不同,我们需要去按照这个基准去按比例适配。

// 设计稿宽度
const WIDTH = 750
// 设计稿中某元素的宽度
const uiWidth = 100px 
// 屏幕宽度与设计稿宽度比例
const scale = document.documentElement.clientWidth / WIDTH
// 屏幕中的实际宽度
const realWidth = uiWidth * scale 
// 以375px屏幕宽为例
// realWidth = 100 * (375/750)= 50px

按照这个思路,为了计算方便,我们可以使用rem来进行适配,将以上代码可以改写为:

// 根据屏幕宽度与设计稿宽度比例 设置rem单位长度
const htmlFontSize = (document.documentElement.clientWidth / WIDTH ) * 100 + 'px'
// (375/750)*100= 50px
// 实际大小其实就是
let realWidth = uiWidth * htmlFontSize / 100
// 100*50/100 = 50px
// 得出最终公式
realWidth = uiWidth / 100 + 'rem'

最后,需要监听页面大小改变,rem适配配置就完成了。

function setRem() {
    const WIDTH = 750
    // 当前页面宽度相对于 750 宽的缩放比例,可根据自己需要修改
    const scale = document.documentElement.clientWidth / WIDTH;
    document.documentElement.style.fontSize = scale + 'px';
}
setRem();
window.onresize = setRem;

优点:

兼容性高,可以通过动态设置根元素的font-size来适配不同的屏幕。

缺点:

1、需要使用js,更适用于手机端,在大屏设备(如ipad)上,元素的尺寸大,可展示的内容不足。

2、移动端1px问题需要另外处理。
  
针对大屏改进方案( flexible 方案)

动态设置 rem 实现在不同尺寸的设备上界面展示效果一致。

1、限制 rem 的最大值。
2、通过媒体查询限制内容最大宽度。

插件使用postcss-pxtorem。

1、下载postcss postcss-pxtorem;

npm install postcss postcss-pxtorem --save-dev

2、js动态设置;

<script>
    const baseSize = 37.5  //跟配置中rootValue的值是一致的     
    // 设置 rem 函数     
    function setRem() {
        // 当前页面宽度相对于 375 宽的缩放比例,可根据自己需要修改。         
        const scale = document .documentElement.clientWidth / 375         
        // 设置页面根节点字体大小 最高为两倍图 即设计稿为750     
        document.documentElement.style.fontSize = (baseSize * Math.min(scale, 2)) + 'px'
    }
    // 初始化     
    setRem()
    // 改变窗口大小时重新设置 rem     
    window.onresize = function () {
        setRem()
    }
</script>

3、写好index.css之后执行以下脚本,可以编译出一个已经进行rem转换的文件main-rem.css。

var fs = require ( 'fs' ) ;
var postcss = require ( 'postcss' ) ;
var pxtorem = require ( 'postcss-pxtorem' ) ;
var css = fs.readFileSync( 'index.css' , 'utf8' ) ;
var options = {  rootValue : 37.5 ,  propList : [ '*' ] } ;
var processedCss = postcss(pxtorem( options )).process( css ).css;
fs.writeFile('main-rem.css', processedCss, function (err) {
  if (err) {
    throw err;
  }
  console.log('Rem file written.');
});

效果:

/*index.css*/
body {
    margin: 0;
    padding: 0;
}
#app {
    width: 375px;
    height: 200px;
    background-color: brown;
    color: #fff;
    font-size: 16px;
}

在这里插入图片描述
vw适配(推荐)

原理与实现:

vw/vh是将屏幕直接分为100等份,1vw是1%的屏幕宽度,1vh是1%的屏幕高度。

优点:

比较方便,只需要css就可以实现适配。

缺点:

1、vw、vh都是固定的,无法改变,和rem适配一样,对于大屏幕而言展示的内容较少。

2、兼容性问题(ie11不支持,少数低版本手机不支持),但目前兼容性越来越好。

方案升级:vw+rem适配方案:

前文谈到,vw的缺点是无法改变,rem的缺点是需要使用js来设置,而且它们都有各自的优点,那么可以将这两者巧妙的结合来解决这个问题。

html {
    /*针对于750px的设计稿*/
    font-size: calc(100*100vw/750)
}
div {
    /*100px*/
    width: 1rem;
}
/*媒体查询:适配大屏幕*/
@media screen and (min-width: 450px) {
  html {
    font-size: calc(50 * 100vw / 750);
  }
}

流式布局

流式布局也就是百分比布局。

比较简单粗暴,由于参考对象是父元素,尤其是嵌套层级比较深的时候,计算百分比的时候有时会有些许误差。

width/height:参考父元素
margin/padding:参考父元素的width
border-radius/background-size/tarnsform: 参考自身宽高

响应式布局

通过媒体查询@media来适配不同屏幕大小,具体可以参考MDN文档

// 用户设备的最小宽度为 680px
@media (min-width: 680px), screen { … }
// min-和max-可以测试一个在两个值之间的宽度
@media (min-width: 30em) and (max-width: 50em) { … }

总结

目前比较推荐的移动适配方案是rem&vw,rem&媒体查询等,了解其中的适配原理对于前端而言是非常重要。

虽然有很多插件可供使用,但是熟知其中奥秘也能给我们的工作带来很大的帮助。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/408319.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

项目无法识别 ?. 语法,导致报错Module parse failed: Unexpected token

一、第一种情况是如果你需要引入的组件库源码的过程中&#xff0c;比如你要import一个封装好的组件这种场景&#xff1a; 首先找到报这个问题的地方 我这里是因为es语法问题 如果你也是npm安装了一个包&#xff0c;然后在导入的时候报这个语法错&#xff0c;那你可以接着往下…

【前端趋势】Vue、Vite作者尤雨溪前端趋势2022主题演讲

欢迎来到我的博客 &#x1f4d4;博主是一名大学在读本科生&#xff0c;主要学习方向是前端。 &#x1f36d;目前已经更新了【Vue】、【React–从基础到实战】、【TypeScript】等等系列专栏 &#x1f308;博客主页&#x1f449;codeMak1r.的博客 最新专栏【React–从基础到实战】…

Vant UI 中 van-collapse 下拉折叠面板如何默认展开第一项

问题重现 最近在做项目时&#xff0c;有一个需求是让 van-collapse 下拉折叠面板的第一项默认是展开状态&#xff1b;我使用到了 Vant UI 组件&#xff0c;首先按照官网文档的写法直接引入&#xff1a; <!--下拉折叠面板--><van-collapse class"collapse" …

vue路由的安装和使用

TOCVue 路由 vue-router 1 安装vue-router npm i vue-router -s 2 创建路由模块 // src/router/index.js就是当前项目的路由模块 // 1 导入Vue 和VueRouter的包 import Vue from vue import VueRouter from vue-router // 导入路由要用到的组件 import Home from ../component…

【TypeScript介绍】一文带你初步了解TypeScript

前言 博主主页&#x1f449;&#x1f3fb;蜡笔雏田学代码 专栏链接&#x1f449;&#x1f3fb;【TypeScript专栏】 今天开始学习学习TypeScript这门语言&#xff01; 感兴趣的小伙伴一起来看看吧~&#x1f91e; 文章目录TypeScript介绍1. TypeScript是什么2. TypeScript为什么要…

大文件上传

大文件上传原理 当我们在做文件上传的功能时&#xff0c;如果上传的文件过大&#xff0c;可能会导致长传时间特别长&#xff0c;且上传失败后需要整个文件全部重新上传。因此&#xff0c;我们需要前后端配合来解决这个问题。 最常用的解决方案就是 —— 切片上传。 这次我们…

图文详解uni-app PC端宽屏适配方案

图文详解uni-app PC端宽屏适配方案 随着互联网的发展&#xff0c;iOS、Android、H5以及各种小程序快应用层出不穷&#xff0c;随之而来的问题就是如何用一套代码&#xff0c;适配多个平台&#xff0c;其中uni-app表现优异&#xff0c;已经实现了手机端全覆盖&#xff0c;支持i…

【JavaScript速成之路】JavaScript对象

&#x1f4c3;个人主页&#xff1a;「小杨」的csdn博客 &#x1f525;系列专栏&#xff1a;【JavaScript速成之路】 &#x1f433;希望大家多多支持&#x1f970;一起进步呀&#xff01; 文章目录前言1&#xff0c;初识对象1.1&#xff0c;对象概念1.2&#xff0c;对象创建1.2.…

手撕前端面试题【JavaScript】

前端的那些基本标签&#x1f353;&#x1f353; 浅拷贝&#x1f353;&#x1f353;简易深拷贝&#x1f353;&#x1f353;深拷贝&#x1f353;&#x1f353;什么是深拷贝&#xff0c;什么是浅拷贝&#xff0c;二者有什么区别&#xff0c;平时改怎么区分呢&#xff1f;&#x1f…

vue3.0本地引入vue.js

第一步获取vue版本 Vue的各种版本介绍&#xff1a; cjs&#xff08;两个版本都是完整版&#xff0c;包含编译器&#xff09; vue.cjs.js vue.cjs.prod.js&#xff08;开发版&#xff0c;代码进行了压缩&#xff09; global&#xff08;这四个版本都可以在浏览器中直接通过…

Vue中 provide、inject 详解及使用

传送门&#xff1a;Vue中 子组件向父组件传值 及 .sync 修饰符 详解 传送门&#xff1a;Vue中 状态管理器&#xff08;vuex&#xff09;详解及应用场景 传送门&#xff1a;Vue中 $ attrs、$ listeners 详解及使用 传送门&#xff1a;Vue中 事件总线&#xff08;eventBus&#x…

New Bing申请与使用教程

&#xff08;1&#xff09;安装 Edge 浏览器插件&#xff1a; 由于在 CN 访问 Bing 国际版会自动跳转到 CN 版&#xff0c;因此需要先重定向浏览器的 Header&#xff0c;先在 Edge 浏览器安装插件&#xff1a;Header Editor&#xff1a; 添加如下配置&#xff08;其中&#xf…

① 尚品汇的后台管理系统【尚硅谷】【Vue】

后台管理系统项目简介 什么是后台管理系统项目&#xff1f; 在前端领域当中&#xff0c;开发后台管理系统项目&#xff0c;并非是Java&#xff0c;PHP等后台语言项目 在前面课程当中&#xff0c;我们已经开发了一个项目【尚品汇电商平台项目】&#xff0c;这个项目主要针对的是…

HTML5+CSS3模拟实现《亮剑》平安县城一役精彩微信群聊-谁开的炮?

转眼从2005年到现在&#xff0c;《亮剑》已经播出多年&#xff0c;但热度依然不减&#xff0c;而且每次重温我都会看出不一样的意蕴&#xff0c;今天&#xff0c;我就用HTML5CSS3模拟实现《亮剑》平安县城一役精彩微信群聊-谁开的炮&#xff1f; 目录 1. 实现思路 2. 素材介绍…

零基础JavaScript学习【第四期】

博主有话说:不知不觉这一已经第四期了,感谢大家的支持。 个人空间:GUIDM的个人空间 专栏内容:零基础JavaScript学习 基础还是关键。 欢迎大家的一键三连。 往期回顾: 前情回顾:第一期https://blog.csdn.net/m0_61901625/article/details/124772807?spm=1001.2014.3001.…

Spring-Security实现登录接口

目录简介原理初探实现思路登录接口实现准备工作编码实现效果展示简介 Security 是 Spring 家族中的一个安全管理框架。相比与另外一个安全框架Shiro&#xff0c;它提供了更丰富的功能&#xff0c;社区资源也比Shiro丰富。 具体介绍和入门看springSecurity入门 原理初探 在实…

CSS 实现六边形柱状图

前言 &#x1f44f;CSS 实现六边形柱状图 速速来Get吧~ &#x1f947;文末分享源代码。记得点赞关注收藏&#xff01; 1.实现效果 2.实现步骤 定义全局css变量&#xff0c;柱状宽度为–w&#xff0c;最大高度为–h&#xff0c;柱形整体为渐变色&#xff0c;定义上部分颜色为…

VUE-webpack超详细教程

一.认识webpack 官网解释&#xff1a;从本质上来讲&#xff0c;webpack是一个现代的JavaScript应用的静态模块打包工具。 从模块化和打包两个角度来理解&#xff1a; 模块化&#xff1a; webpack其中一个核心就是让我们可能进行模块化开发&#xff0c;并且会帮助我们处理模块…

【uni-app导入vant】

uni-app导入vant一、为什么要写这篇文章二、 安装vant三、 进行全局导入Vant组件四、 导入Vant全局公共样式五、 在页面中使用Vant组件一、为什么要写这篇文章 相信有很多人在进行小程序转为uni-app的时候都会遇到转换后vant组件不能用的问题&#xff0c;为了解决这个问题本人…

前端开发:颜色代码速查表【英文颜色、HEX(十六进制)格式、RGB格式、十进制】

&#x1f48c; 作者简介 &#x1f4d6; 个人介绍&#xff1a;小伙伴们&#xff0c;大家好&#xff01;我是水香木鱼&#xff0c;【前端领域创作者】&#x1f61c;&#x1f4dc; CSDN主页&#xff1a;水香木鱼&#x1f4d1; 个人博客&#xff1a;陈春波&#x1f3a8; 系列专栏&…