纯前端:提取视频某一帧显示在页面上

用户在发帖等场景中上传视频,需要提取视频帧作为封面来暗示视频上传成功;本文采用HTML video + canvas的方法提取视频帧第0秒作为封面。

1. 获取上传的文件

前端使用<input type="file">上传文件,当然也可以使用各大UI的uploader组件如移动端的van-uploader。其底层都是<input type="file">

用户上传文件后,可以通过inputchange事件获取上传后的File对象,File继承自Blob对象

js 复制代码
const handleChange = (e) => {
    const file = e.target.files[0]
}

2. File对象转换

看这张图:

首先需要明确我们需要将File转换成什么类型。方法中需要创建一个Video DOM对象,将文件加载到video.src中,所以需要将File转化为ObjectURL

js 复制代码
// file是上面获取到的File对象
const url = URL.createObjectURL(file)

然后创建一个Video对象,这里不需要将它插入到HTML中,使用video.load()也可以触发Video的各种加载事件

js 复制代码
const video = document.createElement('video')
video.src = url
video.load()

3. 在seeked事件中提取视频帧

video事件中loadeddata在视频加载完成后触发,seeked在移动视频时间位置时触发。理论上在loadeddata后就可以提取视频帧,但是实际开发中可能会遇到第一帧纯黑的情况。与系统内置软件展示的不一样。所以可以在video准备好后将视频移动到第0秒,在seeked事件中提取

js 复制代码
//loadeddata后移动视频位置到第0秒
video.addEventListener('loadeddata', loadeddataHandler, false)
const loadeddataHandler = () => {
    video.currentTime = 0
}

移动位置后就可以在seeked中捕获移动完成的事件,canvas的context可以使用drawImage方法直接获取当前位置的视频帧。所以先从HTML中获取canvas的context

js 复制代码
// 获取canvas的context
const canvas = document.getElementByTag("canvas")
const context = canvas.getContext("2d")

使用context.drawImage提取将video画到canvas中。其中参数的意义:video DOM对象,截取位置左上角x,截取位置左上角y,截取的宽度,截取的长度

js 复制代码
canvas.width = video.videoWidth
canvas.height = video.videoHeight
context.drawImage(video, 0,  0, video.videoWidth, video.videoHeight)

drawImage在seeked事件中进行,截取完成后记得移除video事件,video=null释放内存

js 复制代码
video.addEventListener('seeked', seekedHandler, false)
const seekedHandler = async () => {
    canvas.width = video.videoWidth
    canvas.height = video.videoHeight
    context.drawImage(video, 0,  0, video.videoWidth, video.videoHeight)

    video.removeEventListener('loadeddata', loadeddataHandler, false)
    video.removeEventListener('seeked', seekedHandler, false)
}

至此,我们就提取了封面并回显在用户界面中 最后记得将创建的Object URL释放掉,防止内存泄漏URL.revokeObjectURL(url)

从canvas中保存封面

获取到的视频帧不但要回显,还需要上传到服务器。

将canvas图像转化成File对象可以遵循以下步骤: canvas -> base64(data url) -> Blob -> File

js 复制代码
// 首先使用canvas.toDataURL('image/png')将canvas转化为base64图片
url = canvas.value!.toDataURL('image/png')

// 通过fetch方法,获取文件成blob
const res = await fetch(url)

// 将Blob转化成File
new File([await res.blob()], 'cover', { type: 'image/png' })

总结

通过HTML video加载用户上传视频文件,然后通过loadeddata, seek等事件定位视频帧,通过canvas的context.drawImage绘制video的方式可以提取视频帧并回显

canvas -> base64(data url) -> Blob -> File可以获取到视频的File对象上传到服务器。

相关推荐
爱喝水的小周1 小时前
AJAX vs axios vs fetch
前端·javascript·ajax
Jinxiansen02111 小时前
unplugin-vue-components 最佳实践手册
前端·javascript·vue.js
几道之旅1 小时前
介绍electron
前端·javascript·electron
周胡杰1 小时前
鸿蒙arkts使用关系型数据库,使用DB Browser for SQLite连接和查看数据库数据?使用TaskPool进行频繁数据库操作
前端·数据库·华为·harmonyos·鸿蒙·鸿蒙系统
31535669131 小时前
ClipReader:一个剪贴板英语单词阅读器
前端·后端
玲小珑1 小时前
Next.js 教程系列(十一)数据缓存策略与 Next.js 运行时
前端·next.js
qiyue771 小时前
AI编程专栏(三)- 实战无手写代码,Monorepo结构框架开发
前端·ai编程
断竿散人2 小时前
JavaScript 异常捕获完全指南(下):前端框架与生产监控实战
前端·javascript·前端框架
Danny_FD2 小时前
Vue2 + Vuex 实现页面跳转时的状态监听与处理
前端
小飞悟2 小时前
别再只会用 px 了!移动端适配必须掌握的 CSS 单位
前端·css·设计