一句配置让你的小程序自动适应Pad端

序言

相信开发过小程序的朋友,都用过rpx这个单位,最近项目中在开发一款小程序,而在测试阶段发现,在Pad端的元素显示过大了,完全不适应Pad端,而业务上由于Pad端的使用者还是占多数的,所以需要实现Pad端的响应式。

问题

项目中都是使用rpx这个单位来布局的,用过这个小程序rpx单位都知道该特性:

简单来说,就是rpx将布局分成了750份,根据设置rpx的多少来布局,这样对应我们手机750的设计就很好的匹配适应,而在iPad端上,屏幕分辨率却是1536*2048,这样导致同样在手机端显示的文字大小,在iPad端会呈现2倍的效果。

方案

针对上述问题,我们很容易想到,那是不是在iPad端时,让元素的显示的大小减去一半就行了,比如

scss 复制代码
.rule {
    font-size: 20rpx;
}

// => iPad
.rule {
    font-size: 10rpx;
}

而判断iPad端的方式有两种:

  1. 根据UA是否带有pad字样进行判断;
  2. 根据媒体查询进行判断;

针对第一种,如果需要根据UA进行判断,就需要js进行判断再插入对应的className,这样不单每个页面都带重复写,还影响小程序性能,这样太过繁琐,所以我们采用了第二种方案。

通过媒体查询,我们在大于750px(pad端的dpr是2,所以这里是750)的情况,将rpx整体缩小一倍,如:

scss 复制代码
.rule {
    font-size: 20rpx;
}

// => iPad
@media (min-width: 750px) { 
    .rule {
        font-size: 10rpx;
    }
}

思考

虽然问题得到解决,但是每个页面组件都单独写媒体查询,依然太过复杂,那有没有根据简单的方法?

这里我想到了利用 postcss-plugin 在编译时对样式进行修改,大概思路如下

  1. copy一份原有的样式,将所有rpx对应缩小一半;
  2. 再用@media进行包裹,并插入到原样式后面(小程序媒体查询没有权重,所以要插入原样式之后,否则插入前面会更合理);

发布

基于想法,进行实现, @hsuna/postcss-media-query-transform 便孕育而生:

安装

shell 复制代码
npm i -D @hsuna/postcss-media-query-transform
yarn add -D @hsuna/postcss-media-query-transform
pnpm i -D @hsuna/postcss-media-query-transform

使用

  • postcss.config.js使用
js 复制代码
// postcss.config.js
module.exports = {
  plugins: [
    // for example
    // require('autoprefixer'),
    require("postcss-media-query-transform")({
      mediaQuery: { query: "(min-width: 750px)", scale: 0.5 },
      propList: ["*"],
      transformUnit: "rpx",
      insert: "after",
    }),
  ],
};

输入/输出

scss 复制代码
// input
.rule {
  margin: 0 0 20px;
  font-size: 2rpx;
  line-height: 1.2;
  letter-spacing: 0.0625px;
}

// output
.rule {
  margin: 0 0 20px;
  font-size: 2rpx;
  line-height: 1.2;
  letter-spacing: 0.0625px;
}

@media (min-width: 400px) {
  .rule {
    letter-spacing: 0.03125px;
  }
}

参数说明

  • 默认
js 复制代码
const defaultOptions = {
  mediaQuery: { query: "(min-width: 400px)", scale: 0.5 }, // 媒体查询及比例,允许多个
  unitPrecision: 5, // 保留位数
  selectorBlackList: [], // 不允许转化的选择器
  propList: ["font", "font-size", "line-height", "letter-spacing"], // 需要转化的样式 
  transformUnit: "px", // px | rpx | rem 需要转化的单位
  insert: "before", // before | after 需要拷贝的样式插入原样式的位置,默认之前
  exclude: [/node_modules/i], // 过滤的文件
  disabled: false, // 是否禁用该插件
};
  • 多个媒体查询
js 复制代码
const defaultOptions = {
  mediaQuery: [
      { query: "(min-width: 200px)", scale: 2 }, // 超过 200 缩放
      { query: "(min-width: 400px)", scale: 0.5 }
  ],
};
  • 小程序pad端自适应
js 复制代码
const defaultOptions = {
  mediaQuery: { query: "(min-width: 750px)", scale: 0.5 }, // 750px以上缩小一倍像素
  propList: ["*"], // 应用在所有像素上
  transformUnit: "rpx", // 使用在rpx单位
  insert: "after", // 小程序媒体查询没有权重,所以要插入原样式之后
};

插件其他效果

  • 对应转化后如果和原本样式一致的话,比如类似display:block;或者padding: 0;,那插件会将对这部分样式舍弃;
  • 针对@keyframes,如果所有过程都没有转化,则舍弃该@keyframes,否则保留所有@keyframes
  • 如果样式原本就包裹在@media,则舍弃;

最后

虽然需求项目由于工期,我们还是采用了手写的方法去处理,但是后续的项目我们还是采用了该插件,目前线上效果挺好的~

相关推荐
A_aspectJ1 小时前
【Bootstrap V4系列】学习入门教程之 组件-输入组(Input group)
前端·css·学习·bootstrap·html
兆。1 小时前
电子商城后台管理平台-Flask Vue项目开发
前端·vue.js·后端·python·flask
互联网搬砖老肖2 小时前
Web 架构之负载均衡全解析
前端·架构·负载均衡
sunbyte2 小时前
Tailwind CSS v4 主题化实践入门(自定义 Theme + 主题模式切换)✨
前端·javascript·css·tailwindcss
风之舞_yjf3 小时前
Vue基础(8)_监视属性、深度监视、监视的简写形式
javascript·vue.js·ecmascript
湛海不过深蓝3 小时前
【css】css统一设置变量
前端·css
DONSEE广东东信智能读卡器3 小时前
蓝牙身份证阅读器使用Uniapp调用二次开发demo
javascript·uni-app·蓝牙·身份证阅读器
Codingwiz_Joy3 小时前
Day28 -js开发01 -JS三个实例:文件上传 & 登录验证 & 购物商城 & ---逻辑漏洞复现 及 判断js的payload思路
开发语言·javascript·安全·安全性测试
程序员的世界你不懂4 小时前
tomcat6性能优化
前端·性能优化·firefox
爱吃巧克力的程序媛4 小时前
QML ProgressBar控件详解
前端