Skip to main content

Element-UI 官网主题切换动画

基本效果

首先我们写一个 html 文件,写一个按钮,以及简单的背景颜色切换,来模拟主题的切换。

document.startViewTransition

想要实现过渡效果,需要先用到一个 JavaScript 的原生方法 document.startViewTransition

这个方法是用来实现动画过渡效果的:

通过调用 API,让浏览器为新旧两种不同视图分别捕获并建立了快照 (即 ::view-transition-old(root)旧快照::view-transition-new(root) 新快照)。

而后新旧两快照在 ::view-transition-image-pair(root)容器 中完成转场动画的过渡,动画结束后则删除其相关伪元素(快照和容器)。

过渡动画效果

我们尝试在初始 html 中使用下 document.startViewTransition,现在去切换主题颜色,发现有过渡效果了:

圆形扩散过渡动画

接下来实现圆形过渡的效果,其实这个动画最终是展示 ::view-transition-new(root) 这个伪元素。

所以我们主要让这个伪元素有原型扩散的过渡动画。其实很简单,只需要将伪元素的半径,从 0 -> 100% 即可:

并且我们需要取消掉 document.startViewTransition 默认的动画效果,不然它会导致我们自定义的动画效果无效:

最终得到圆形扩散的效果:

完整代码

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
<style>
:root {
/* 默认亮主题 */
--bg-color: #fff;
background-color: var(--bg-color);
}
:root.dark {
/* 暗主题 */
--bg-color: #000;
}
::view-transition-new(root),
::view-transition-old(root) {
/* 关闭默认动画 */
animation: none;
}
</style>
</head>
<body>
<button id="themeButton">切换主题</button>
<script>
const themeButton = document.getElementById('themeButton')
themeButton.addEventListener('click', e => {
// 执行切换主题的操作
const transition = document.startViewTransition(() => {
// 动画过渡切换主题色
document.documentElement.classList.toggle('dark')
})

// document.startViewTransition 的 ready 返回一个 Promise
transition.ready.then(() => {
// 获取鼠标的坐标
const { clientX, clientY } = e

// 计算最大半径
const radius = Math.hypot(Math.max(clientX, innerWidth - clientX), Math.max(clientY, innerHeight - clientY))

// 圆形动画扩散开始
document.documentElement.animate(
{
clipPath: [`circle(0% at ${clientX}px ${clientY}px)`, `circle(${radius}px at ${clientX}px ${clientY}px)`]
},
// 设置时间,已经目标伪元素
{
duration: 300,
pseudoElement: '::view-transition-new(root)'
}
)
})
})
</script>
</body>
</html>