零 JS、无闪屏的暗黑模式需用prefers-color-scheme媒体查询配合:root CSS变量实现系统级自动适配,但手动切换必须结合localStorage和data-theme属性;data-theme须设在<html>上并优先读取localStorage,fallback至matchMedia检测,同时通过<meta name="color-scheme">同步浏览器UI颜色。
直接用 prefers-color-scheme 媒体查询 + :root CSS 变量就能实现零 JS、无闪屏的暗黑模式,但仅靠它无法响应用户手动切换——必须配合 localStorage 和 data-theme 属性才能真正可用。
prefers-color-scheme 做基础系统级适配这是最轻量的起点,浏览器原生支持,无需 JS 就能随系统设置自动生效。关键不是“写两套样式”,而是用媒体查询控制变量定义位置:
@media (prefers-color-scheme: dark) 内只改 :root 里的 CSS 变量值,比如 --bg: #121212,别在这里重写整个 body 样式background-color、color、border-color、box-shadow、svg fill)都必须用 var(--bg) 调用,硬编码值会绕过主题系统@layer 或预处理器输出中的该查询有解析 bug,建议单独提一层写localStorage 检测逻辑data-theme 比 class 切换更可靠用户点一次“?”按钮,就得记住选择、刷新不丢、不被系统偏好覆盖——class="dark" 容易和已有 class 冲突,而 data-theme="dark" 是语义明确、层级干净、SSR 友好的方案:
<html> 元素上:document.documentElement.dataset.theme = "dark",不是 body 或其他节点html 前缀:html[data-theme="dark"] { --bg: #1e1e1e; },否则可能被子元素样式意外覆盖localStorage.getItem("theme"),返回 null 时再 fallback 到 window.matchMedia("(prefers-color-scheme: dark)").matches,别把 null 直接赋给 dataset.theme(会导致 data-theme="null",CSS 匹配失败)localStorage.setItem("theme", "dark"),否则刷新即失效页面变暗了,但 Chrome 地址栏还是白的、iOS 输入框边框仍是浅灰——这是因为浏览器 UI 颜色由 <meta name="color-scheme"> 控制,它不会随 JS 自动更新:
立即学习“前端免费学习笔记(深入)”;
localStorage 读取主题,并设 <meta name="color-scheme" content="light dark">(注意:两个值都要写,不能只写 dark)document.querySelector('meta[name="color-scheme"]').content = "light dark"
transition 在暗黑模式里经常失效写了 transition: background-color .2s 却没动画?因为 CSS 变量变化本身不触发重绘,transition 只响应具体属性值的数值变更:
html[data-theme="dark"] 一加,所有 var(--bg) 就自动过渡——它们是批量重计算,非逐属性渐变body { background-color: var(--bg); })本身可被 transition,且确保 --bg 的变更能触发该属性重算(现代浏览器基本支持,但部分安卓 WebView 不稳定)body)用 opacity 或 transform 做极简过渡,或接受“无动画切换”——多数用户更在意一致性而非动效@media (prefers-color-scheme: dark) 里写 transition,它只控制变量定义,不控制渲染行为最容易被跳过的其实是初始化时机:JS 必须在 DOMContentLoaded 之前完成 data-theme 设置,否则页面会先按默认样式渲染一次,再闪一下变暗——内联一小段 script 放在 <head> 里执行是最稳的解法。