vue-cli工具build测试与生产包对css处理的不同

前言

项目上线遇到了css样式问题,经过排查发现是 main.css 中声明了两个 :root{},导致后一个覆盖了前一个。然而测试环境并未出现这个问题,样式效果为两个 :root {} 合并。

package.json 如下:

原因

build:test 打包结果:


build:prod 打包结果:

dist/index.html 观察到的不同点:

  • 测试环境 (build:test) 没有 static/css 文件,观察 index.htmlCSS 是直接内联方式加载的。
  • 生产环境 (build:prod) 生成了 static/css,即 CSS 被单独提取到了 CSS 文件中。

原因如下:

  1. CSS 代码优化策略
    • Vue CLI 默认在 production 环境使用 cssnano 进行 CSS 代码优化(压缩、合并、去重)。
    • cssnano 优化了 :root{} 变量,它的策略是 合并相同的变量,如果变量相同会合并,如果有不同值,后面的会覆盖前面的。
  2. CSS 提取方式不同
    • 测试环境可能是 inline 或通过 JS 注入 style,在合并多个 :root{} 时,浏览器会自动合并。
    • 生产环境中 CSS 被单独提取到 static/css,并可能经过 cssnano 的优化,导致 :root{} 选择器的行为不同。
  3. PostCSS 处理机制
    • Vue CLI 使用 postcss 处理 CSS,它可能在 production 环境启用了 cssnanomergeRules 规则,导致 :root{} 合并方式发生变化。

解决方式

一、测试环境开启CSS提取确保与生产环境一致

javascript 复制代码
module.exports = {
  css: {
    extract: process.env.NODE_ENV !== 'development'
  }
};

再次执行 npm run build:test

二、手动将多个:root合并

css 复制代码
:root {
  --color-primary: #409eff;
  --color-secondary: #f56c6c;
}

:root {
  --color-success: #67c23a;
  --color-warning: #e6a23c;
}

手动合并为

css 复制代码
:root {
  --color-primary: #409eff;
  --color-secondary: #f56c6c;
  --color-success: #67c23a;
  --color-warning: #e6a23c;
}

三、手动禁用 cssnano 相关优化

javascript 复制代码
module.exports = {
  css: {
    extract: true, // 确保生产环境也提取 CSS
    loaderOptions: {
      postcss: {
        plugins: [
          require('cssnano')({
            preset: ['default', { mergeRules: false, mergeLonghand: false }]
          })
        ]
      }
    }
  }
};
相关推荐
excel9 分钟前
迭代器与生成器全面理解
前端
可口码农20 分钟前
MixOne:Electron Remote模块的现代化继任者
java·前端·electron
大聪明了26 分钟前
uniapp vue3 使用 pinia
javascript·vue.js·uni-app
发如雪-ty27 分钟前
Bash常用操作总结
前端·chrome
冲!!37 分钟前
使用nvm查看/安装node版本
前端·node.js·node·nvm
LilyCoder1 小时前
HTML5二十四节气网站源码
前端·javascript·html·html5
nyf_unknown1 小时前
(vue)将文件夹打成tar包, Git Bash(推荐)具体使用
vue.js·git·bash
Bruce_Liuxiaowei1 小时前
跨站脚本攻击(XSS)高级绕过技术与防御方案
前端·网络安全·xss
EF@蛐蛐堂2 小时前
【vue3】v-model 的 “新玩法“
前端·javascript·vue.js
两个月菜鸟2 小时前
vue+微信小程序 五角星
前端·vue.js·微信小程序