gpt4 book ai didi

javascript - 生成特定时间文件输入选中的视频文件的缩略图/快照

转载 作者:可可西里 更新时间:2023-11-01 02:17:21 27 4
gpt4 key购买 nike

如何获取通过 <input type="file"> 选择的视频文件的快照在视频中的特定时间在后台静默(即没有可见元素、闪烁、声音等)?

最佳答案

主要有四个步骤:

  1. 创建 <canvas><video>元素。
  2. 加载src URL.createObjectURL 生成的视频文件进入<video>元素并等待它加载通过监听被触发的特定事件
  3. 将视频时间设置为您想要拍摄快照的时间点并监听其他事件
  4. 使用 Canvas 抓取图像。

第 1 步 - 创建元素

这很简单:只需创建一个 <canvas>和一个<video>元素并将它们附加到 <body> (或任何地方,这并不重要):

var canvasElem = $( '<canvas class="snapshot-generator"></canvas>' ).appendTo(document.body)[0];
var $video = $( '<video muted class="snapshot-generator"></video>' ).appendTo(document.body);

请注意视频元素具有属性 muted .不要放置任何其他属性,如 autoplaycontrols .另请注意,它们都有类 snapshot-generator .这样我们就可以为它们设置样式,使它们不碍事:

.snapshot-generator {
display: block;
height: 1px;
left: 0;
object-fit: contain;
position: fixed;
top: 0;
width: 1px;
z-index: -1;
}

一些浏览器将它们设置为 display: none ,但其他浏览器将有严重的问题,除非它们被呈现在页面上,所以我们只是把它们做得很小,这样它们基本上是不可见的。 (不要将它们移出视口(viewport),否则您可能会在页面上看到一些难看的滚动条。)

第 2 步 - 加载视频

这里是事情开始变得棘手的地方。您需要收听事件以了解何时继续。不同的浏览器会触发不同的事件、不同的时间和不同的顺序,所以我会为你省力。在视频准备好之前,必须始终至少触发三个事件;他们是:

  • 加载元数据
  • 加载数据
  • 暂停

为这些事件设置事件处理程序并跟踪触发了多少事件。一旦所有三个都开火,你就可以继续了。请记住,由于其中一些事件可能会多次触发,您只想处理触发的每种类型的第一个事件,并丢弃后续触发。我使用了 jQuery 的 .one。 ,它会处理这个问题。

var step_2_events_fired = 0;
$video.one('loadedmetadata loadeddata suspend', function() {
if (++step_2_events_fired == 3) {
// Ready for next step
}
}).prop('src', insert_source_here);

源应该只是通过 URL.createObjectURL(file) 创建的对象 URL , 其中file是文件对象。

第 3 步 - 设置时间

这个阶段和之前的类似:设置时间然后监听一个事件。在我们的内部 if从前面的代码块:

$video.one('seeked', function() {
// Ready for next step
}).prop('currentTime', insert_time_here_in_seconds);

幸运的是这次只有一个事件,所以它非常清晰简洁。终于……

第 4 步 - 抓取快照

这部分只是使用<canvas>元素来抓取屏幕截图。在我们的内部 seeked事件处理程序:

canvas_elem.height = this.videoHeight;
canvas_elem.width = this.videoWidth;
canvas_elem.getContext('2d').drawImage(this, 0, 0);
var snapshot = canvas_elem.toDataURL();

// Remove elements as they are no longer needed
$video.remove();
$(canvas_elem).remove();

Canvas 需要匹配视频的尺寸(不是 <video> 元素)才能获得正确的图像。此外,我们正在设置 Canvas 的内部 .height.width属性,不是 Canvas 高度/宽度 CSS 样式值。

snapshot的值是一个数据URI,基本上就是一个以data:image/jpeg;base64开头的字符串然后是 base64 数据。

我们最终的 JS 代码应该是这样的:

var step_2_events_fired = 0;
$video.one('loadedmetadata loadeddata suspend', function() {
if (++step_2_events_fired == 3) {
$video.one('seeked', function() {
canvas_elem.height = this.videoHeight;
canvas_elem.width = this.videoWidth;
canvas_elem.getContext('2d').drawImage(this, 0, 0);
var snapshot = canvas_elem.toDataURL();

// Delete the elements as they are no longer needed
$video.remove();
$(canvas_elem).remove();
}).prop('currentTime', insert_time_here_in_seconds);
}
}).prop('src', insert_source_here);

庆祝!

你的图像是 base64 格式的!将此发送到您的服务器,将其作为 src<img>元素,或其他任何东西。

比如可以解码成二进制,直接写入文件(先去掉前缀),就会变成JPEG图片文件。

您还可以使用它在上传视频时提供视频预览。如果你把它写成 src<img> ,使用完整的数据 URI (不要删除前缀)

关于javascript - 生成特定时间文件输入选中的视频文件的缩略图/快照,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36883037/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com