饿了么面试官:实现一下 Element-UI 官网的主题切换动画!

boyanx6个月前技术教程15

最近看到 ElementPlus 官网上的切换主题方式非常有趣,这是一个过渡的动画效果

所以在网上查了一番,找到基本的实现方法

实现

基本效果

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

可以看到实现了最简单的主题切换效果

document.startViewTransition

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

这个方法是用来做动画过渡效果的

通过调用 API,让浏览器为新旧两种不同视图分别捕获并建立了快照 (即 ::view-transition-old(root)旧快照 和 ::view-transition-new(root) 新快照),而后新旧两快照在 ::view-transition-image-pair(root) 容器中完成转场动画的过渡。动画结束后则删除其相关伪元素 (快照和容器)

过渡动画效果

我们可以应用一下这个 API

现在去切换主题颜色,发现有过渡效果了~

圆形扩散过渡动画

接下来实现圆形过渡的效果,其实这个动画最终是展示::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>
标签: animate css

相关文章

如何使用CSS实现旋转地球动画效果

旋转地球功能实现主要借助于CSS动画效果完成,通过移动地图背景图层,云彩图层等,在视觉上呈现出旋转地球效果。旋转地球最终实现效果如下图所示:设计思路与核心技术旋转地球效果实现主要借助于animatio...

仅使用一个 DIV 配合 CSS 实现饼状图

#头条创作挑战赛#本文同步本人掘金平台的原创翻译:https://juejin.cn/post/7053763392590315557本文为译文「意译」完整的代码请滑到文末。我们只使用一个div,仅...

极客Web前端开发资源大荟萃每周精选#017

1. css3 代码生成玫瑰 七夕快乐css3 代码生成的玫瑰花 送给大家 七夕节快乐!代码链接:http://www.gbtags.com/gb/rtreplayerpreview/1364.htm...

带你看好玩的CSS-霓虹灯按钮

对于前端开发人员来说,css是我们再熟悉不过的朋友的,它就相当于是我们页面的衣服,页面好不好看,就看我们css运用的是否炉火纯青。css学起来简单,但是我们要把它“修炼”到出神入化境界,那这可不是一丁...

基于 Vue3 Element Plus 的中后台管理系统模板

Pure Admin 是一个开源的前端中后台管理系统模板,基于Vue3、Element-Plus,支持移动端、国际化、多主题设置,支持前端静态路由、后端动态路由配置,旨在为开发人员提供一个易于使用、高...

html的web框架,Angular.js 1.3.19发布

近日,Angular.js 1.3.19 发布下载,更新内容如下:Bug 修复$http:propagate status -1 for timed out requests (f13055a0,#4...

发表评论    

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。