当前位置: 首页 > news >正文

前端实现 GIF 图片循环播放

前言

使用 img 加载 GIF 图片,内容只会播放一次,之后就会自动暂停;

通过定时器在一段时间后重新加载图片的方式,会导致浏览器内存不断增大,并且可能会有闪烁、卡顿的问题;

ImageDecoder

WebCodecs API 的 ImageDecoder 接口提供了一种对编码图像数据进行解包和解码的方法。

在这里插入图片描述

ImageDecoder 只能在 localhost 或者 https 的 tab 页中获取到

代码

  1. canvas 设置 widthheight,和 GIF 图片尺寸一致
    <canvas width="1310" height="954"></canvas>
    
    或许你也可以通过一下方法动态设置尺寸;
    const img = new Image();
    img.src = '图片路径';
    img.onload = function() {var width = this.width;   // 图片宽度var height = this.height; // 图片高度console.log(`图片尺寸为:${width} x ${height}`);// some code... // 为canvas 设置 width、height
    };
    
  2. 通过 fetch 获取 GIF 图片资源,传入 url
    fetch(url).then((response) => decodeImage(response.body));
    
  3. decodeImage 方法对图片解码
    imageDecoder = new ImageDecoder({ data: imageByteStream, type: "image/gif" });
    imageDecoder.decode({ frameIndex: imageIndex }).then(renderImage);
    
  4. renderImage 渲染每一帧的图片
     function renderImage (result) {const canvas = document.querySelector("canvas");const canvasContext = canvas.getContext("2d");// 绘制图片 canvasContext.drawImage(result.image, 0, 0);const track = imageDecoder.tracks.selectedTrack;if (imageDecoder.complete) {if (track.frameCount === 1) return;// 如果序号超出了帧数,重置为 0if (imageIndex + 1 >= track.frameCount) imageIndex = 0;}// 解码下一帧图片并渲染imageDecoder.decode({ frameIndex: ++imageIndex }).then((nextResult) =>setTimeout(() => {renderImage(nextResult);}, result.image.duration / 1000) // 可以调整播放速度).catch((e) => {if (e instanceof RangeError) {imageIndex = 0;imageDecoder.decode({ frameIndex: imageIndex }).then(renderImage);} else {throw e;}});}
    

完整代码

<template><div><canvas width="1310" height="954"></canvas></div>
</template><script>
export default {mounted() {this.play(require("@/assets/images/animate.gif"));},play(url) {let imageDecoder = null;let imageIndex = 0;function renderImage (result) {const canvas = document.querySelector("canvas");const canvasContext = canvas.getContext("2d");canvasContext.drawImage(result.image, 0, 0);const track = imageDecoder.tracks.selectedTrack;if (imageDecoder.complete) {if (track.frameCount === 1) return;if (imageIndex + 1 >= track.frameCount) imageIndex = 0;}imageDecoder.decode({ frameIndex: ++imageIndex }).then((nextResult) =>setTimeout(() => {renderImage(nextResult);}, result.image.duration / 1000)).catch((e) => {if (e instanceof RangeError) {imageIndex = 0;imageDecoder.decode({ frameIndex: imageIndex }).then(renderImage);} else {throw e;}});}function decodeImage (imageByteStream) {imageDecoder = new ImageDecoder({ data: imageByteStream, type: "image/gif" });imageDecoder.decode({ frameIndex: imageIndex }).then(renderImage);}fetch(url).then((response) => decodeImage(response.body));}
}
</script>

相关链接

  • 纯前端如何实现Gif暂停、倍速播放
  • MDN ImageDecoder
http://www.lryc.cn/news/535291.html

相关文章:

  • React - 事件绑定this
  • STM32系统架构介绍
  • Macbook Pro快速搭建Easysearch学习环境
  • 老游戏回顾:SWRacer
  • Firefox无法隐藏标题栏
  • vue基础(五)
  • MySQL的深度分页如何优化?
  • 深度学习每周学习总结R6(RNN实现阿尔茨海默病诊断)
  • Node.js 多模态图像描述服务 调用siliconflow:现代 JavaScript 实践
  • 机器学习数学基础:21.特征值与特征向量
  • 【目标检测json2txt】label从COCO格式json文件转YOLO格式txt文件
  • 强化学习之 PPO 算法:原理、实现与案例深度剖析
  • vue-点击生成动态值,动态渲染回显输入框
  • 高性能 :OpenAI Triton Open-source GPU programming Language LINUX 环境配置
  • TCP 端口号为何位于首部前四个字节?协议设计的智慧与启示
  • HTML之JavaScript函数声明
  • R 数组:高效数据处理的基础
  • git服务器搭建,gitea服务搭建,使用systemclt管理服务
  • Pdf手册阅读(1)--数字签名篇
  • 嵌入式WebRTC压缩至670K,目标将so动态库压缩至500K,.a静态库还可以更小
  • 百度高德地图坐标转换
  • ES 索引结构
  • HPM_SDK应用本地化——基于6750evkmini
  • 【deepseek-r1本地部署】
  • 查询语句来提取 detail 字段中包含 xxx 的 URL 里的 commodity/ 后面的数字串
  • 堆排序
  • 【MySQL】我在广州学Mysql 系列—— 数据备份与还原
  • 【LeetCode Hot100 双指针】移动零、盛最多水的容器、三数之和、接雨水
  • HTML应用指南:利用POST请求获取接入比亚迪业态的充电桩位置信息
  • Android车机DIY开发之软件篇(十二) AOSP12下载编译