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

前端私教训练营:1v1私教,终身辅导计划,帮你拿到满意的 offer 已帮助数百位同学拿到了中大厂 offer

前言

大家好,我是林三心,用最通俗易懂的话讲最难的知识点是我的座右铭,基础是进阶的前提是我的初心~

最近看到 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-colorvar(--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)`,
              ],
            },
            // 设置时间,已经目标伪元素
            {
              duration300,
              pseudoElement"::view-transition-new(root)",
            }
          );
        });
      });
    
</script>
  </body>
</html>

结语

我是林三心,感谢您的阅读~

1v1私教、找工作、全程陪跑、终身服务

推荐一个非常靠谱的 Sunday 老师目前在做一个【1v1前端私教训练营】主打的就是:1v1 私教,终身辅导,帮大家拿到满意的 offer 点击这里即可查看详情

可直接加 Sunday老师 微信沟通,回复【资料】可直接领取 Sunday独家整理 大厂面试题、大厂真实面经

相关推荐

  • 即插即用的开源WINDOWS优化工具
  • 快速搭建Linux系统、一键清理Windows系统,两个实用开源项目
  • 基于Neo4j与LangChain实现GraphRAG:具象化理解其关键模块
  • 传统RAG及GraphRAG最佳实践探索及落地中的知识库建设、现实坑点:老刘说NLP技术社区第24、25讲精彩回放
  • 解密字符串的底层结构,它是怎么实现的?
  • 我的 3 个项目
  • 3分钟带你秒懂线程池设计机制
  • Kaggle竞赛Top5分享融合绝学。
  • [开源]MIT开源协议,一款 Vue UI 组件库,60多个开箱即用高质量组件
  • Spring Boot集成groovy快速入门Demo
  • ECMAScript 2024新特性
  • 工作中如何体现一个人的技术深度?
  • 微软开源了可视化工具SandDance
  • 北京内推 | ​百度搜索策略部视频搜索团队招聘多模态方向算法实习生
  • CVPR最佳论文候选:NeRF新突破!用启发式引导分割去除瞬态干扰物
  • 探索LLM安全漏洞新视角:植入后门的RAG会对大模型推理带来哪些风险?
  • WizardLM新作!ArenaLearning: 通过模拟LLM竞技场来构建大规模数据飞轮
  • 上位机干了3年!还不知道串口调试助手???
  • 29.3K Star震惊!!!Windows一个常用工具开源了,程序员玩疯了
  • 疯狂!CVPR 超越 Science!全球第二大