现代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;
}

参考资料

相关推荐
亚里士多没有德7752 分钟前
强制删除了windows自带的edge浏览器,重装不了怎么办【已解决】
前端·edge
micro2010146 分钟前
Microsoft Edge 离线安装包制作或获取方法和下载地址分享
前端·edge
.生产的驴10 分钟前
Electron Vue框架环境搭建 Vue3环境搭建
java·前端·vue.js·spring boot·后端·electron·ecmascript
awonw13 分钟前
[前端][easyui]easyui select 默认值
前端·javascript·easyui
九圣残炎34 分钟前
【Vue】vue-admin-template项目搭建
前端·vue.js·arcgis
柏箱1 小时前
使用JavaScript写一个网页端的四则运算器
前端·javascript·css
TU^1 小时前
C语言习题~day16
c语言·前端·算法
学习使我快乐014 小时前
JS进阶 3——深入面向对象、原型
开发语言·前端·javascript
bobostudio19954 小时前
TypeScript 设计模式之【策略模式】
前端·javascript·设计模式·typescript·策略模式