现代WEB主题色切换方案

随着 WEB 技术的发展,我们见证了网站和应用程序界面的主题色变得越来越多样化。这种多样性反映了用户对于个性化和视觉吸引力的需求,因此单一的主题色已经不再足够满足用户的多样化需求。为了更好地服务于不同用户群体的喜好和使用场景,网站和应用程序开始提供不同的预设主题以供用户切换。

这些预设主题主要涵盖了浅色主题深色主题跟随系统这三种基本的主题。深色主题为那些偏爱低调、沉稳的用户提供了舒适的浏览体验,尤其适用于夜间或低光环境下。而浅色主题则适合那些喜欢明亮、活泼色彩的用户,让他们在白天或明亮环境下享受清新的界面。跟随系统主题则根据用户的偏好或系统设置自动调整主题色,为用户提供了更智能、更个性化的体验。

🤔 稍提一嘴,深色主题还可以节省设备的电量。

浅色与深色主题切换

首先设置 color-schema 属性,告诉浏览器当前应用所支持的应用配色方案。

css 复制代码
:root {
  color-scheme: light dark;
}

然后,通过prefers-color-scheme 属性设置不同主题色下的样式。

prefers-color-scheme 用于检测用户是否有将系统的主题色设置为亮色或者暗色。通过这个属性,可以让网页根据用户系统的颜色模式自动调整显示效果。

这对于用户体验和可访问性都有很多好处。例如,如果用户喜欢在暗光环境下浏览网页,他们可以将系统设置为暗色模式,而网页可以根据这一设置自动切换到暗色主题,从而减少眼睛的疲劳和提高可读性。

此外,对于一些用户来说,暗色模式也是一种辅助功能,可以帮助他们更好地使用网页,比如对于光敏性癫痫患者来说,暗色模式可以减少光引发的癫痫发作的可能性。

css 复制代码
/* 在暗色模式下应用的样式 */
@media (prefers-color-scheme: dark) {
  body {
    background-color: #000;
    color: #fff;
  }
}

/* 在亮色模式下应用的样式 */
@media (prefers-color-scheme: light) {
  body {
    background-color: #fff;
    color: #000;
  }
}

跟随系统主题

通过媒体查询 + prefers-color-scheme 的方案,使得应用可以在浅色主题和深色主题之间切换,但此方案存在一些弊端,无法做到跟随系统主题的变化(毕竟系统主题由系统决定👀)。

如果要实现主题跟着操作系统主题的变化而变化,应该怎么做呢?🤔

(🤣既然 css不行,那就直接上 js )

随着查阅 MDN 相关的文档,可以使用 window 对象上的 matchMedia 方法监听当前是否匹配prefers-color-scheme的某一种情况。

js 复制代码
  const useDark = window.matchMedia('(prefers-color-scheme: dark)')
  useDark.addEventListener('change', function (evt) {
    const isDarkMode = evt.matches
    // 通过 isDarkMode 动态改变 document 的 class
    if (isDarkMode) { 
        document.documentElement.classList.add('dark')
    } else {
      document.documentElement.classList.remove('dark')
    }
  })

主题色切换优化

现代WEB网页开发大多采用的是组件化的开发方式,当系统的主题配色方案改变后,难以保证所有的组件过渡都能平滑的切换。

在 Chrome 111 中,推出了 View Transitions API。使得此 API 可以使得网页在 新旧快照之间进行一个平滑的过渡。

具体使用方法如下 :

ts 复制代码
function useViewTransition(event: MouseEvent) {
  const x = event.clientX
  const y = event.clientY
  const endRadius = Math.hypot(
    Math.max(x, innerWidth - x),
    Math.max(y, innerHeight - y),
  )

  let _isDark: boolean
  // startViewTransition 调用的时候 会获取当前页面的状态 ::view-transition-old(root)
  // 通过 startViewTransition 的回调函数,获取最新的页面的状态 ::view-transition-new(root)
  // 旧视图的动画效果从 opacity: 1 到 opacity: 0,而新视图则从 opacity: 0 过渡到 opacity: 1,从而形成淡入淡出。
  const transition = document.startViewTransition(() => {
    const root = document.documentElement
    _isDark = root.classList.contains('dark')
    const methodField = _isDark ? 'remove' : 'add'
    root.classList[methodField]('dark')
  })
  // 两者过渡的快照准备完成时,使用 Web Animations API 进行 动画过渡
  transition.ready.then(() => {
    const clipPath = [
      `circle(0px at ${x}px ${y}px)`,
      `circle(${endRadius}px at ${x}px ${y}px)`,
    ]

    document.documentElement.animate(
      {
        clipPath: _isDark ? [...clipPath].reverse() : clipPath,
      },
      {
        duration: 500,
        easing: 'ease-in',
        pseudoElement: _isDark // 运用于伪元素的过渡
          ? '::view-transition-old(root)'
          : '::view-transition-new(root)',
      },
    )
  })
}

除此之外,还需修改 view-transition css的默认行为,如下所示:

css 复制代码
::view-transition-old(root),
::view-transition-new(root) {
 /* 关闭默认的淡入淡出的效果 */
  animation: none;
  mix-blend-mode: normal;
}

.dark::view-transition-old(root) {
  z-index: 1;
}
.dark::view-transition-new(root) {
  z-index: 999;
}

::view-transition-old(root) {
  z-index: 999;
}
::view-transition-new(root) {
  z-index: 1;
}

参考资料

相关推荐
m0_7482561427 分钟前
前端 MYTED单篇TED词汇学习功能优化
前端·学习
小白学前端6662 小时前
React Router 深入指南:从入门到进阶
前端·react.js·react
web130933203982 小时前
前端下载后端文件流,文件可以下载,但是打不开,显示“文件已损坏”的问题分析与解决方案
前端
outstanding木槿2 小时前
react+antd的Table组件编辑单元格
前端·javascript·react.js·前端框架
好名字08213 小时前
前端取Content-Disposition中的filename字段与解码(vue)
前端·javascript·vue.js·前端框架
隐形喷火龙3 小时前
element ui--下拉根据拼音首字母过滤
前端·vue.js·ui
m0_748241123 小时前
Selenium之Web元素定位
前端·selenium·测试工具
风无雨3 小时前
react杂乱笔记(一)
前端·笔记·react.js
前端小魔女3 小时前
2024-我赚到自媒体第一桶金
前端·rust
鑫~阳3 小时前
快速建站(网站如何在自己的电脑里跑起来) 详细步骤 一
前端·内容管理系统cms