Vue3 大屏适配组件(Scale / Rem 双方案一键切换)

  • 一键切换「整体 Scale 缩放」「Rem 等分适配」
  • 窗口自动监听 resize
  • 适配设计稿 1920*1080
  • Vue3 全局直接引入用

一、新建组件 ScreenAdapter.vue

xml 复制代码
<template>
   <div class="screen-adapter" :class="{ 'scale-mode': type === 'scale' }">
    <!-- 大屏所有内容插槽 -->
    <slot />
  </div>
</template>

<script setup>
import { onMounted, onUnmounted } from 'vue';

const props = defineProps({
  // 适配模式:scale / rem
  type: {
    type: String,
    default: 'scale', // 默认用不变形的scale
    validator: val => ['scale', 'rem'].includes(val)
  },
  // 设计稿宽高
  designWidth: {
    type: Number,
    default: 1920
  },
  designHeight: {
    type: Number,
    default: 1080
  }
});

let resizeFn = null;

// 1. Scale 整体缩放适配
function initScale() {
  const wrap = document.querySelector('.screen-adapter');
  if (!wrap) return;

  const clientW = document.documentElement.clientWidth;
  const clientH = document.documentElement.clientHeight;

  const scaleW = clientW / props.designWidth;
  const scaleH = clientH / props.designHeight;
  const scale = Math.min(scaleW, scaleH);

  wrap.style.transform = `translate(-50%, -50%) scale(${scale})`;
}

// 2. Rem 等分适配 设计稿px直接写rem
function initRem() {
  const clientW = document.documentElement.clientWidth;
  // 把屏幕分成100份 1rem = 1%屏幕宽
  document.documentElement.style.fontSize = clientW / 100 + 'px';
}

// 初始化适配
function adapterInit() {
  if (props.type === 'scale') {
    initScale();
  } else {
    initRem();
  }
}

onMounted(() => {
  adapterInit();
  resizeFn = adapterInit;
  window.addEventListener('resize', resizeFn);
});

onUnmounted(() => {
  window.removeEventListener('resize', resizeFn);
});
</script>

<style scoped>
html,
body {
  margin: 0;
  height: 100%;
  overflow: hidden;
}

.screen-adapter {
  position: fixed;
  left: 50%;
  top: 50%;
  transform-origin: 0 0;
}

/* Scale模式固定画布大小 */
.scale-mode {
  width: 1920px;
  height: 1080px;
}
</style>

二、页面中使用

1. 用 Scale 模式(推荐交付、地图大屏、不变形)

xml 复制代码
<template>
  <!-- 只用包一层,内部全部按1920*1080写px即可 -->
  <ScreenAdapter type="scale">
    <!-- 你的大屏页面内容 -->
    <div class="title">智慧公路大屏</div>
  </ScreenAdapter>
</template>

<script setup>
import ScreenAdapter from '@/components/ScreenAdapter.vue';
</script>

<style>
.title {
  /* 直接写px,不用任何换算 */
  font-size: 36px;
  width: 400px;
  height: 80px;
  line-height: 80px;
  text-align: center;
}
</style>

2. 切换 Rem 模式(内部项目、铺满全屏)

ini 复制代码
<ScreenAdapter type="rem">
  内部大屏内容
</ScreenAdapter>

Rem 用法规则设计稿多少 px,直接写多少 rem:

  • 设计稿 200px → width: 200rem
  • 字体 28px → font-size: 28rem不用计算、不用除以任何数。

三、两种模式使用场景记住

  1. type="scale"
  • 对外交付、政府大屏、Mapbox/L7/GIS 地图
  • 优点:不变形、像素级还原、适配所有屏
  • 写法:全部写 px
  1. type="rem"
  • 内部后台、运维大屏、工期紧
  • 优点:全屏无留白、开发快
  • 写法:设计稿 px 直接换成 rem

四、关键注意点

  1. Scale 模式下所有弹窗、ECharts、地图都要包在插槽里面,不要单独挂 body
  2. 不需要再写任何额外适配 JS,组件已经封装好监听 resize
  3. 以后所有大屏页面,直接套这个组件,改个 type 就能切换适配方案
相关推荐
kyriewen6 小时前
别再 console.log 了:5 个 Chrome DevTools 调试技巧,用过就回不去了
前端·javascript·面试
IT_陈寒8 小时前
Python搞不定字符串编码?这破玩意坑我两小时!
前端·人工智能·后端
DigitalOcean9 小时前
Laravel 开发者已在 DigitalOcean 上开通超过 10 万台服务器
前端·laravel
星始流年9 小时前
从 Tool 到 Skill——基于 LangChain 的服务端Skill实现
前端·langchain·agent
李惟9 小时前
开源本地通信库,纯客户端 RPC,像聊天一样通信
前端
YAwu119 小时前
深入解析 React 炫彩鼠标跟随标题组件:从坐标定位到动画性能
前端·react.js
GuWenyue9 小时前
排序效率低?5分钟吃透快速排序,性能飙升至O(nlogn)
前端·javascript·面试
OpenTiny社区9 小时前
🎨 看完 GenUI SDK 源码我悟了!
前端·vue.js·github
叁两9 小时前
前端转型AI Agent该如何学习?(前置篇)
前端·人工智能·node.js
何时梦醒9 小时前
深入理解递归与快速排序 —— 从基础入门到手写实现
前端·javascript