移动端适配方案

移动端适配

方案 1:rem + html font-size

方案 2:vw

rem + html font-size

rem 是相对于 html 元素的 font-size 来设置的单位,通过在不同屏幕尺寸下动态修改 html 元素的 font-size 可达到适配效果

在开发中,我们只需要考虑两个问题:

  1. 针对不同的屏幕尺寸设置不同的 html font-size

  2. 将元素尺寸单位转为 rem

动态设置 html font-size

方法一:通过媒体查询设置 html font-size

思路:通过媒体查询根据不同屏幕尺寸设置不同的 html font-size

缺点:① 如果动态改变屏幕尺寸,不能实时更新、 ② 只能针对某个尺寸范围设置 html font-size

css 复制代码
@media screen and (min-width: 320px) {
    html {
        font-size: 20px;
    }
}

@media screen and (min-width: 375px) {
    html {
        font-size: 24px;
    }
}

@media screen and (min-width: 414px) {
    html {
        font-size: 28px;
    }
}

@media screen and (min-width: 480px) {
    html {
        font-size: 32px;
    }
}

.box {
    width: 5rem;
    height: 5rem;
    background-color: blue;
}

方法二:通过 JS 设置 html font-size

思路:通过监听屏幕尺寸的变化动态修改 html font-size

一般会将 html font-size 设置为屏幕宽度的 1/10,方便计算

js 复制代码
function setRemUnit() {
    const htmlEl = document.documentElement;
    const htmlFontSize = htmlEl.clientWidth / 10;
    htmlEl.style.fontSize = htmlFontSize + 'px';
}
setRemUnit();
window.addEventListener('resize', setRemUnit);

px 与 rem 的单位换算

假设原型图中屏幕宽度为 375px,现有一宽度为 100px 的 div。我们想将 100px 转成对应的 rem 值:

  1. 根元素 html 的 font-size = 375px / 10 = 37.5px (37.5px 即为 "基准字体大小")

  2. 元素的 rem 值 = 100px / 37.5px

手动计算

编写 scss 函数:

scss 复制代码
$baseFontSize: 37.5px; // 基准字体大小
@function pxToRem($pxValue) {
    @return $pxValue / $baseFontSize * 1rem;
}

使用 scss 函数编写样式:

scss 复制代码
.example {
    font-size: pxToRem(24px); // 将 24px 转换为对应的 rem 值
    margin: pxToRem(16px) pxToRem(8px); // 将 16px 和 8px 转换为对应的 rem 值
}

工程化自动计算

安装 postcss-pxtorem 依赖包、配置 webpack.config.js 文件:

bash 复制代码
npm install postcss-pxtorem -D
js 复制代码
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
const postcssPxToRem = require('postcss-pxtorem');

const stylesHandler = MiniCssExtractPlugin.loader;
const postcssLoader = {
    loader: 'postcss-loader',
    options: {
        postcssOptions: {
            plugins: [
                postcssPxToRem({
                    rootValue: 37.5, // 基准字体大小, 默认 16
                    propList: ['*'], // 转换的属性, 默认 ['font', 'font-size', 'line-height', 'letter-spacing']
                }),
            ],
        },
    },
};

const config = {
    // ...
    module: {
        // ...
        rules: [
            // ...
            {
                test: /\.css$/i,
                use: [stylesHandler, 'css-loader', postcssLoader],
            },
        ],
    },
};

正常编写样式,px 会自动转换为 rem:

css 复制代码
.example {
    font-size: 24px; /* 将转换为 0.64rem */
    margin: 16px 8px; /* 将转换为 0.43rem 0.21rem */
}

vw 单位

100vw 相当于整个视口的宽度 innerWidth,1vw 相当于视口宽度的 1%,将 px 转换为 vw 即可完成适配

vw 相对于 rem 的优势:不需要考虑 html font-size 的问题

事实上,rem 作为一种过渡的方案,它利用的也是 vw 的思想

px 与 vw 的单位转换

假设原型图中屏幕宽度为 375px,有一宽度为 100px 的 div。我们需要将 100px 转成对应的 vw 值:

  1. 1vw = 375px / 100 = 3.75px
  2. 元素的 vw 值 = 100px / 3.75px

手动计算

编写 scss 函数:

scss 复制代码
$baseWidth: 375px; // 设计稿宽度
@function pxToVw($pxValue) {
    @return $pxValue / $baseWidth * 100vw;
}

使用 scss 函数编写样式:

scss 复制代码
.example {
    width: pxToVw(100px); // 将 100px 转换为对应的 vw 值
    margin: pxToVw(16px) pxToVw(8px); // 将 16px 和 8px 转换为对应的 vw 值
}

工程化自动计算

安装 postcss-px-to-viewport-8-plugin 依赖包、配置 webpack.config.js 文件:

bash 复制代码
npm install postcss-px-to-viewport-8-plugin -D
js 复制代码
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
const postcssPxToViewport = require('postcss-px-to-viewport-8-plugin');

const stylesHandler = MiniCssExtractPlugin.loader;
const postcssLoader = {
    loader: 'postcss-loader',
    options: {
        postcssOptions: {
            plugins: [
                postcssPxToViewport({
                    viewportWidth: 375, // 设计稿宽度, 默认 320
                }),
            ],
        },
    },
};

const config = {
    // ...
    module: {
        // ...
        rules: [
            // ...
            {
                test: /\.css$/i,
                use: [stylesHandler, 'css-loader', postcssLoader],
            },
        ],
    },
};

正常编写样式,px 会自动转换为 vw:

css 复制代码
.example {
    width: 100px; /* 将转换为 26.67vw */
    margin: 16px 8px; /* 将转换为 4.27vw 2.13vw */
}
相关推荐
竹林8186 小时前
从 RPC 超时到批量签名:我用 @solana/web3.js 重构了一个 NFT 铸造页面,踩了这些坑
前端·javascript
aaaameliaaa6 小时前
计算斐波那契数(递归、迭代)(1,1,2,3,5.....)
c语言·开发语言·笔记·算法·排序算法
m0_547486666 小时前
《模式识别:使用MATLAB分析与实现》全套PPT课件
开发语言·matlab·模式识别
Tim_106 小时前
【C++】009、extern关键字
java·开发语言
工业HMI实战笔记6 小时前
工业HMI界面布局“1核2辅”黄金结构,适配90%场景
前端·ui·性能优化·自动化·交互
橘子星6 小时前
从零手写 RAG 语义检索:基于 Node.js 实现轻量级向量搜索
javascript·人工智能
林希_Rachel_傻希希6 小时前
web性能优化之————图片效果
前端·javascript·面试
橘子星6 小时前
基于 MCP 协议实现本地文件读取工具服务开发实践
javascript·人工智能
Darling噜啦啦6 小时前
前端存储与 this 指向完全指南:从 LocalStorage 实战到 call/apply/bind 深度解析
前端·javascript
sugar__salt6 小时前
手撕字符串算法:反转、回文、验证回文 Ⅱ 完整拆解
javascript·算法·面试·职场和发展