百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

如何使用客户端 JavaScript 将视频剪辑转换为 GIF 文件

zhezhongyun 2025-03-30 23:12 7 浏览

作为一个程序员,一个热爱技术写作的人,把自己的知识经验通过文字呈现在读者面前,并且让读者通过文字,就能将知识进行学以致用,其实,是非常有调整性的一件事情,因为,它需要我与读者保持有效沟通,既不浪费读者的时间,也不浪费自己的时间,同时还要保证读者能够读有所获。

由于技术文章的质量很大程度上取决于其内容的易消化程度,因此,我倾向于利用以图形交换格式 (GIF) 格式编码的屏幕截图,以尽可能减少额外的文本描述块。例如,在更倾向于 Web 开发内容的文章中,例如:

在这种情况下,我总是发现嵌入动画 GIF 来演示我创建的开源工具的任何用法是有益的:

鉴于 Microsoft Powerpoint 的内置屏幕录制功能很容易访问,虽然,可以轻松完成自己的屏幕录制以输出屏幕截图,但我觉得更有益的探索领域将是 - 易于访问转换视频的工具 将屏幕截图(.mp4、.avi 等)转换为 .GIF 图像文件。

— 注意:按照惯例,如果要渲染动画短片(≤ 30 秒),则 GIF 文件更合适。

使用 GIFEncoder.js 将视频构建到 GIF Maker

为了构建浏览器实用程序,如下所示:

这 3 个 JavaScript 插件是必需的:

  • GIFEncoder.js
  • LZWEncoder.js
  • NeuQuant.js

仅供参考:这些插件最初是由 GitHub 用户 Kevin Kwok(创建者)从 GitHub repo jsgif 中检索到的。

视频到 GIF 转换概述

先决条件:包含在上述3个文件中+ b64.js如下:

<script type="text/javascript" src="LZWEncoder.js"></script>
<script type="text/javascript" src="NeuQuant.js"></script>
<script type="text/javascript" src="GIFEncoder.js"></script>
<script type="text/javascript" src="b64.js"></script>

技术实施——总共 4 个步骤

步骤(1):上传视频片段(≤30秒)

因此,在 HTML 代码中,包含一个简单的用户输入界面:


将事件处理程序(onchange)标记到上面并继续在 JavaScript 中初始化 FileReader 实例:

inputVideoClipFile.onchange = function(uploadFle) {
  let file = inputVideoClipFile.files[0];
let fileName=file.name;
  let fileSize=(file.size/1024).toFixed(2);
  let fileType=file.type;
let fileredr = new FileReader();
fileredr.onload = function (fle) {
    var b64Str=fle.target.result;
}; // end file-reader onload
  fileredr.readAsDataURL(file);
};

请注意,新的 FileReader() 实例调用 readAsDataURL,因此分配给 b64Str 的视频文件内容被读取为 Base64 字符串。

从文件对象中检索视频文件的信息以供稍后显示。

步骤(2):处理视频二进制数据并提取帧,有两个主要部分需要考虑:

Part I. 通过创建 DOM 元素并在 JavaScript 中分配相应的属性来预览视频内容以供显示.

// rendered as  in HTML code
var videoObj = document.createElement('video');
var displayedHeight=500;
if(videoObj.canPlayType(fileType)) {
  videoObj.setAttribute('id','inputVideo');
  videoObj.setAttribute('src', b64Str);
  videoObj.setAttribute('height', displayedHeight);
}

注意b64Str是前面步骤中FileReader()读取的视频文件数据。

第二部分,帧提取——每个视频帧都指的是剪辑在唯一时间戳的图像快照。

由于 GIF 文件是通过合并一组连续的图像创建的,因此,对于视频的每一次时间图形更新,都需要提取一个嵌入了图像数据的帧,用于后续的 GIF 创建过程。

虽然不能直接从 DOM 元素 中提取每个视频帧所需的图像数据,但可以将 中的预览内容渲染到 元素上 帧图像数据提取。

接下来,在 JavaScript 中创建一个 元素并为其分配相应的属性(类似于 ):

var vidHeight=videoObj.videoHeight;
var vidWidth=videoObj.videoWidth;
var bitmap = document.createElement('canvas');
bitmap.setAttribute('id', 'bitmap');
bitmap.setAttribute('width', vidWidth);
bitmap.setAttribute('height', vidHeight);

请注意,vidWidth 和 vidHeight 是从 检索的。(这些是剪辑的原始尺寸。)

接下来的几行代码基于标记到每个元素的 'id' 属性来引用

const inputVideo=document.getElementById('inputVideo');
const bitmapCanvas=document.getElementById('bitmap');
const bitmapCtx = bitmapCanvas.getContext('2d');
inputVideo.muted = true;
inputVideo.loop = false;
inputVideo.autoplay=true;
const background = () => {
  bitmapCtx.fillStyle = '#FFFFFF';
  bitmapCtx.fillRect(0, 0, vidWidth, vidHeight);
};

由于视频设置为自动播放,当播放事件被发出时,这会触发 background() 的执行,它不仅将 GIF 背景填充为白色,而且还设置了绘制到 canvas 元素上的每一帧的尺寸:

const step = async() => {
  let bgStatus=await background();
  await new Promise(resolve => {
    bitmapCtx.drawImage(inputVideo, 0, 0, vidWidth, vidHeight);
    frameB64Str = bitmapCanvas.toDataURL();
    resolve();
  });
  window.requestAnimationFrame(step);
};
inputVideo.addEventListener('play', () => {
  step();
  window.requestAnimationFrame(step);
});

window.requestAnimationFrame(step) 接受一个回调函数(即step())来处理每一帧的图像数据。

bitmapCtx.drawImage() 继续将每个图像快照渲染到画布上,以便 bitmapCanvas.toDataURL() 返回每个快照的 base64 编码图像。

步骤(3):依次合并所有图像快照

最后实现了JS插件GIFEncoder.js。使用与上面相同的代码片段,以下粗体代码行指的是 GIFEncoder 编码器捕获嵌入在 中的每个数据帧的实例。

const encoder = new GIFEncoder(vidWidth, vidHeight);
encoder.setRepeat(0);
encoder.setDelay(500);
const step = async() => {
  let bgStatus=await background();
  await new Promise(resolve => {
    bitmapCtx.drawImage(inputVideo, 0, 0, vidWidth, vidHeight);
    frameB64Str = bitmapCanvas.toDataURL();
    encoder.addFrame(bitmapCtx);
    resolve();
  });
  window.requestAnimationFrame(step);
};
inputVideo.addEventListener('play', () => {
  encoder.start();
  step();
  window.requestAnimationFrame(step);
});

当上传的视频最终播放完整时长时,应在 GIFEncoder 调用方法 finish() 的地方发出结束事件:

inputVideo.addEventListener('ended', () => {
  encoder.finish();
});

步骤 (4):通过 GIFEncoder 创建 GIF

要从编码器中提取所有帧的合并版本(即 GIF 输出),需要实现以下 JavaScript 代码片段:

var fileType='image/gif';
var readableStream=encoder.stream();
var binary_gif=readableStream.getData();
var b64Str='data:'+fileType+';base64,'+encode64(binary_gif);

encode64() 是 b64.js 中的一种方法,用于将 GIFEncoder 捕获的流数据转换为 Base64 格式。

b64Str 通过合并 GIFEncoder 中存在的所有帧来引用为 GIF 文件编码的数据。因此,在 HTML 代码中,继续包含:${fileName}以预览输出 GIF 文件。

最后创建一个GIF文件的下载链接,如下:

let dwnlnk = document.createElement('a');
dwnlnk.download = fileName;
dwnlnk.innerHTML = ` Save`;
dwnlnk.className = 'btn btn-outline-dark';
dwnlnk.href = b64Str;

仅供参考:完整的放在我的 GitHub:video-to-GIFhttps://github.com/incubated-geek-cc/video-to-GIF

总结

虽然,有进一步探索 GIFEncoder.js 中输出 GIF 文件的可定制调整的空间,但本文介绍的是一种基本的轻量级方法,该方法可以完全在浏览器中和纯客户端将视频剪辑转换为 GIF 文件。

- End -

相关推荐

字体缩放(方式一)(字体缩放150%怎么做)

通过元素宽度和字数计算得到缩放简单实现如下:/***字体最大为视觉要求大小(maxFontSize);超出缩小字体显示,最小为minFontSize;最小字体时超出部分使用圆点(...);*p...

网页世界隐藏的神秘代码语言,竟能这样改变布局

CSS基础:选择器与属性CSS(CascadingStyleSheets)是用于控制网页外观的一门样式表语言。它通过定义HTML元素的显示方式来增强网页的表现力。CSS的选择器允许开发者精确地定位...

CSS属性值计算过程详解(css属性用来定义元素计算)

在CSS中,即使某些属性没有显式声明,浏览器也会通过**属性值计算过程**为每个元素的所有属性赋予最终值。这一过程分为四个关键步骤,以下将逐一解析。1.确定声明值浏览器首先检查所有**直接应用**到...

软网推荐:找回调整Windows 10字号功能

之前的系统,从WindowsXP到早期版本的Windows10,均有字体大小调整功能,但从创意者版Windows10以来,取消了之前的设置选项,取而代之的是自定义缩放比例设置。使用这个功能调整过...

Excel中如何设置文本框属性,实例代码讲解

Excel不仅可以对数据进行处理,而且也可以图形化数据,直观显示数据表达的内容。本节介绍一个很重要的对象,Characters,字符对象,使用Characters对象可修改包含在全文本字符串中的任...

CSS 字体样式(css中字体)

本节我们来讲字体样式,之前我们学习HTML的时候学过一些用于字体加粗、倾斜的标签,但是使用标签来实现的效果肯定没有我们通过CSS中的样式来的方便。接下来我们会给大家介绍下面这几个属性的使用:通...

PC网站建设必备代码知识:HTML基础与应用技巧

在PC网站建设的相关课程里,代码扮演着至关重要的角色。只有熟练运用正确的代码,我们才能打造出功能完善、用户体验出色的PC网站。接下来,我会详细讲解在PC网站建设环节中必须了解的代码知识。HTML基础代...

让你大跌眼镜的疯狂 HTML 和 CSS 技巧

今天,分享一个让你大开眼界的技巧。通过使用这个技巧,你可以将整个网页变成一个CSS编辑器。没错,你从未见过这种方法。当我第一次尝试时,我完全被震惊到了。现在,让我们开始吧!步骤1首先,创建一个基础的...

jQuery EasyUI使用教程:创建一个链接按钮

jQueryEasyUI最新版下载>本教程主要为大家展示如何使用jQueryEasyUI创建一个链接按钮。通常情况下,使用“button/”元素来创建一个按钮;使用“a/”元素来创建链接按钮...

React 19 有哪些新特性?(react100)

如果你对React18还不熟悉,欢迎阅读之前的文章《React18全览[1]》最近React发布了V19RC版本,按照惯例,我们对React19的新特性进行一次深度的体验学习...

Java注解探秘:为什么@PostConstruct能解决你的初始化难题?

你是否在Spring项目中遇到过这样的困扰:明明依赖注入已经完成,但某些配置就是无法正常加载?手动调用初始化方法又容易引发空指针异常?这就是@PostConstruct注解大显身手的时候了!@Post...

AI驱动的表单自动填写(ai置入表格)

我们都同意,填写表格是一项枯燥且耗时的任务。如果我们可以创建一个可以为我们填写表格的AI助手,让我们将时间投入到更有建设性的任务中,那会怎样?AI助手将能够通过调用以表单字段为参数的函数来填写表...

从零到一:小程序设计新手如何快速上手?

开发环境搭建对于小程序设计新手而言,搭建合适的开发环境是首要任务。以小程序为例,其官方提供了功能强大的开发工具——开发者工具。首先,新手需前往官方开发者平台,在页面中找到“工具下载”板块,根据...

JavaSwingGUI从小白到大神-6(续)(java从小白到大牛怎么样)

接上一篇《JavaSwingGUI从小白到大神-6》,因本篇文章3万多字,头条一篇发不完,只能分开发。同事查询面板:CompanyFind.javapublicclassCompanyFind{...

C# winform界面假死(c#程序假死)

针对C#WinForm界面假死问题,以下是分步解决方案:1.使用异步编程(async/await)将耗时操作移至后台线程,保持UI线程响应。步骤:将事件处理函数标记为async。使用Task....