Vue3 实现图片的帧动画方案分享

一、纯 JS (有性能问题会卡顿)

html 复制代码
      <div class="robot-box">
        <img ref="robotImgRef" src="/Greet/Greet_00000.png" alt="" />
      </div>
ts 复制代码
const delay = 30;
const imgCount = 109;
let i = 0;
const robotImgRef = ref<HTMLImageElement | null>(null);
const animation = (i) => {
  if (i === imgCount) {
    i = 0;
    animation(i);
    return;
  }

  let path;
  if (Number(i) < 10) {
    path = '0000' + i;
  } else if (Number(i) < 100) {
    path = '000' + i;
  } else if (Number(i) < 1000) {
    path = '00' + i;
  }

  robotImgRef.value.src = `/Greet/Greet_${path}.png`;

  robotImgRef.value.onload = () => {
    setTimeout(() => {
      animation(i + 1);
    }, delay);
  };
};

onMounted(() => {
  nextTick(() => {
    animation(0);
  });
});

二、window.requestAnimationFrame()

window.requestAnimationFrame------MDN

window.requestAnimationFrame() 告诉浏览器------你希望执行一个动画,并且要求浏览器在下次重绘之前调用指定的回调函数更新动画。该方法需要传入一个回调函数作为参数,该回调函数会在浏览器下一次重绘之前执行。

ts 复制代码
onMounted(() => {
  nextTick(() => {
    rAF(animation, 60);
  });
});
ts 复制代码
const imgCount = 109;
let i = 0;
const robotImgRef = ref<HTMLImageElement | null>(null);
const animation = () => {
  if (i === imgCount) {
    // 重新开始
    i = 0;
    return;
  }
  let path;
  if (Number(i) < 10) {
    path = '0000' + i;
  } else if (Number(i) < 100) {
    path = '000' + i;
  } else if (Number(i) < 1000) {
    path = '00' + i;
  }
  if (robotImgRef.value) {
    robotImgRef.value.src = `/Greet/Greet_${path}.png`;
  }
  i++;
};
/**
 * @param {number} [fps] - 帧率
 * @param {function} callback - 动画逻辑
 * @return {function} cancelRFA  - 取消播放
 */

function rAF(callback, fps = 24) {
  let last = Date.now();
  let delta = last;
  const interval = 1000 / fps;
  let cancel = false;
  function draw() {
    if (cancel) return;
    requestAnimationFrame(draw);
    let now = Date.now();
    delta = now - last;
    if (delta > interval) {
      /* 这里不能简单last = now,否则出现细微时间差问题。例如fps= 10,每帧100ms,
        而现在每16ms(60fps)执行一次draw。16*7=112>100,需要7次才实际绘制一帧。这个情况下,
        实际10帧需要 112*10=1120ms>1000ms 才绘制完成。
        当 now - (delta % interval) 时,会将时间修正 delta % interval = 12ms,
        实际10帧需要 112+(112-12)*9=1012ms 绘制完成。
     */
      last = now - (delta % interval);
      callback();
    }
  }
  draw();

  return function cancelRAF() {
    cancel = true;
  };
}

三、CSS 实现

请看这篇博客:实现序列帧图片动画的3种方式

四、叫 UI 换 GIF 图片

我最后采用的是叫 UI 换 GIF 了😁

只有麻烦下 UI 了哈哈哈哈

感谢

实现序列帧图片动画的3种方式
利用 requestAnimationFrame API 流畅播放序列帧动画并控制帧率FPS

相关推荐
岁月宁静1 天前
RAG 文档摄入全链路,从原理到生产落地
vue.js·人工智能·python
#麻辣小龙虾#1 天前
基于vue3.0开发一款【固废与废气运维管理系统】(支持源码)
前端·vue.js·vue3
一 乐1 天前
家政服务管理系统|基于springboot + vue家政服务管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·家政服务管理系统
丷丩1 天前
MapLibre GL JS第47课:添加动画图标
javascript·gis·动画·mapbox·maplibre
独泪了无痕1 天前
Vue3中防御XSS攻击的“特效药”-DOMPurify
前端·vue.js·安全
云水一下1 天前
Vue.js从零到精通系列(五):全局状态管理——Pinia 核心与实践
前端·javascript·vue.js
老马聊技术1 天前
AI对话功能之SpringBoot整合Vue3
vue.js·人工智能·spring boot·后端
英勇无比的消炎药1 天前
一站式汇总TinyVue工具案例与真实落地经验
vue.js·前端框架
梵得儿SHI2 天前
Vue 项目实战与性能优化全攻略:从代码、渲染到首屏,一站式解决卡顿慢加载
前端·vue.js·性能优化·vite·前端面试·前端优化·首屏优化
一 乐2 天前
幼儿园管理系统|基于springboot + vue幼儿园管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·幼儿园管理系统