多监控m3u8视频流,怎么获取每个监控的封面图(纯前端)
文章目录
- 1.背景
- 2.问题分析
- 3.解决方案
- 3.1解决思路
- 3.2解决过程
- 3.2.1 封装播放组件
- 3.2.2 隐形的视频div
- 3.2.3 截取封面图
- 3.3 结束
1.背景
有这样一个需求:
- 给你一个监控列表,每页展示多个监控(至少12个,m3u8格式);
- 且展示每个监控的第一帧画面的封面图,但是后端没给你返回封面图;
- 点击监控时可以查看具体的监控弹框,支持全屏;
那么、你会怎么办呢?
后端说:后端获取封面图会比较麻烦,还要单独引入一个模块,所以丢给前端来做喽
备注:本例仅为思路分析,具体代码需要具体实现
实现效果如下:

2.问题分析
- 首先,有的vue2的视频流播放组件,支持多个视频流同时播放,但那是定制的4宫格或9宫格,不够自由;
- 或者,有的支持多视频流播放,但是本人暂未找到,其实也可以自定义实现多流播放;
- 其次,即使有这种组件,但是毕竟是网页端,如果一直播放,性能肯定会扛不住,毕竟一直接收视频片段,这个是巨量的。
- 所以,还是老老实实地想办法获取第一帧封面图吧!
备注:前端用的vue2
下图为多视频流播放的示例图:

但是这样会出现一个问题,那就是你会获取无止境的m3u8推流,不久,你的浏览器就会内存溢出(内存将不断增大,下图非只是初始时的截图):

作为对比,看下只用封面截图的网页资源占用:

所以,上述方案作废!
由于上述方案作废,所以此处不做多视频流同时播放的代码示例了。
3.解决方案
3.1解决思路
- 在监控列表页放一个隐藏的div,div中放的是视频播放组件;
- 然后循环播放,在播放的时候截取封面,用的
html2canvas; - 将截取的封面处理进监控列表的数组中;
- 销毁隐藏的视频流播放的div;
3.2解决过程
3.2.1 封装播放组件
首先,你要有一个支持m3u8播放的vue组件,本例用的是hls.min.js,将这个js文件放在/public/js/hls.min.js下。
然后封装的组件示例如下hlsVideo.vue:
<!-- 视频巡检专用,其他组件勿用 -->
<template><div ><videoid="hls"class="img-responsive video-js vjs-default-skin"controlsref="videoRef"preload="auto":autoplay="autoplay":muted="muted":style="{ width: width, height: height }"></video></div>
</template><script>
let hlsPlayer
export default {props: {// 监控地址videoUrl: { type: String, default: '' },// 监控宽高width: { type: String, default: '100%' },height: { type: String, default: '100%' },// 自动播放autoplay: { type: Boolean, default: true },// 是否静音,默认静音muted: { type: Boolean, default: true }},watch: {videoUrl: {deep: true,handler(val) {if ( hlsPlayer != null ) {hlsPlayer.destroy()}this.handlePlayer()}}},mounted() {this.handlePlayer()},beforeDestroy() {if ( hlsPlayer!= null ) {hlsPlayer.destroy()hlsPlayer = null}},methods: {// emit 发送播放状态handlePlayer() {if(this.videoUrl) {const video = this.$refs.videoRefhlsPlayer = new Hls();hlsPlayer.loadSource(this.videoUrl);hlsPlayer.attachMedia(video);//将视频元素绑定到此 HLS 对象hlsPlayer.on(Hls.Events.MANIFEST_PARSED, () => {video.play();this.$emit("playing", true);});hlsPlayer.on(Hls.Events.ERROR, (eventName, data) => {if ( data.fatal && data.type !== "networkError" ) {video.pause();this.$emit("playing", false);}});} else {if ( hlsPlayer != null ) {hlsPlayer.destroy()}}},}
}
</script>
<style lang="scss" scoped>
::v-deep {// 隐藏视频监控进度条video::-webkit-media-controls-timeline {display: none;}// 隐藏视频监控剩余时间video::-webkit-media-controls-time-remaining-display {display: none;}
}
</style>
3.2.2 隐形的视频div
有了视频播放组件,我们只需要引入即可:
import HlsVideo from "@/componentsUser/m3u8Video/hlsVideo.vue";
然后在页面中写一个div元素来放置该组件:
<!-- 隐藏区域,用于截取封面图 --><div class="hide-video"><hls-videov-if="tempVideoUrl":video-url="tempVideoUrl"id="hiddenVideo"ref="hiddenRef":autoplay="true"@playing="changePlayStatus"/></div>
再加上样式,让我们看不见它,让它自己截图玩去吧,CSS样式如下:
.hide-video {width: 300px;height: 180px;position: absolute;top: 0;left: 0;z-index: -2;
}
同时在data中声明以下变量:
tempVideoUrl: null, // 隐藏的m3u8播放路径tempVideoIndex: 0, // 隐藏播放的当前下标,用户循环monitorList: [], // 监控列表
3.2.3 截取封面图
获取监控列表,然后进行截图处理
请确保已引入如下插件:
import html2canvas from ‘html2canvas’;
获取监控列表,也就是正常的接口请求:
// 获取监控列表getList() {this.loading = truelistDevice(this.queryParams).then(res => {const tempArr = res.rows.map(item => {return {deviceCode: item.deviceCode,type1Name: item.type1Name,type2Name: item.type2Name,name: item.name,deviceLocation: item.deviceLocation,status: item.status,videoUrl: null, // 监控地址,接口未返回,下面单独的接口循环获取,最好还是让后端返回吧posterImg: null, // 默认封面为空,下面循环截取}})this.total = res.totalthis.loading = falsethis.isFirstLoading = falsethis.requestAllVideoUrl(tempArr)})},
由于监控列表未返回m3u8的地址,所以此处需要再次循环请求获取地址:
你们最好让你们的后端把地址也返回,不然前端循环获取不太好!
/** 循环请求,获取监控地址,todo:并且截取封面图 */requestAllVideoUrl(tempArr) {for(let i = 0; i < tempArr.length; i++) {const id = tempArr[i].deviceCode// 离线(值为2)的直接空值,省略请求if(!!id && tempArr[i].status != 2) {getVideo({channelId: id,}).then(res => {tempArr[i].videoUrl = res.data || null})}}this.monitorList = tempArr// 初始设置第一个监控地址给隐藏的视频模仿组件,然后下面才可以循环截图let timer2 = setInterval(() => {if(tempArr[0].videoUrl) {this.tempVideoUrl = tempArr[0].videoUrlclearInterval(timer2)}}, 100)},
我们在上面的hlsVideo.vue组件中写了:
this.$emit("playing", true);
this.$emit("playing", false);
目的是当一个m3u8播放成功后,就告诉组件:我播放完了,你可以截图了,并且可以切换下一个m3u8地址了。
具体的截图代码如下:
// todo:视频改变播放状态,执行截图功能changePlayStatus(e) {if(e) {setTimeout(() => {this.$nextTick(() => {if(this.$refs.hiddenRef) {const video_width = 396const video_height = 180const dom = document.getElementById("hiddenVideo");html2canvas(dom, {windowWidth: video_width * 1.78,windowHeight: video_height * 1.78,x: 0,y: 0,useCORS: true, //跨域scale: 0.8,}).then(canvas => {// 截图的base64位图片const imgData = canvas.toDataURL('image/jpeg', 1.0);// console.log('index', this.tempVideoIndex, imgData)this.monitorList[this.tempVideoIndex].posterImg = imgDataif(this.tempVideoIndex < this.monitorList.length - 1) {this.tempVideoIndex++if(this.monitorList[this.tempVideoIndex].status != 0) {this.tempVideoUrl = this.monitorList[this.tempVideoIndex].videoUrl} else {this.tempVideoIndex++this.tempVideoUrl = this.monitorList[this.tempVideoIndex].videoUrl}} else {this.tempVideoUrl = null}})}})}, 500)}
3.3 结束
至此,截图完毕!截图就是最上面放的截图,此处不重复放了
- 也成功地把封面图赋值给监控列表了,M3u8也销毁了,
- 监控列表现在放的是图片,
- 也不存在内存溢出等性能问题了
本例仅做思路,具体代码具体对待
相关文章:
多监控m3u8视频流,怎么获取每个监控的封面图(纯前端)
文章目录 1.背景2.问题分析3.解决方案3.1解决思路3.2解决过程3.2.1 封装播放组件3.2.2 隐形的视频div3.2.3 截取封面图 3.3 结束 1.背景 有这样一个需求: 给你一个监控列表,每页展示多个监控(至少12个,m3u8格式)&…...
【机器学习实战入门项目】使用深度学习创建您自己的表情符号
深度学习项目入门——让你更接近数据科学的梦想 表情符号或头像是表示非语言暗示的方式。这些暗示已成为在线聊天、产品评论、品牌情感等的重要组成部分。这也促使数据科学领域越来越多的研究致力于表情驱动的故事讲述。 随着计算机视觉和深度学习的进步,现在可以…...
技术洞察:C++在后端开发中的前沿趋势与社会影响
文章目录 引言C在后端开发中的前沿趋势1. 高性能计算的需求2. 微服务架构的兴起3. 跨平台开发的便利性 跨领域技术融合与创新实践1. C与人工智能的结合2. C与区块链技术的融合 C对社会与人文的影响1. 提升生产力与创新能力2. 促进技术教育与人才培养3. 技术与人文的深度融合 结…...
【人工智能 | 大数据】基于人工智能的大数据分析方法
【作者主页】Francek Chen 【专栏介绍】 ⌈ ⌈ ⌈智能大数据分析 ⌋ ⌋ ⌋ 智能大数据分析是指利用先进的技术和算法对大规模数据进行深入分析和挖掘,以提取有价值的信息和洞察。它结合了大数据技术、人工智能(AI)、机器学习(ML&a…...
数字经济时代下的创新探索与实践:以“开源AI智能名片2+1链动模式S2B2C商城小程序源码”为核心
摘要:在数字经济蓬勃发展的今天,中国作为全球数字经济的领航者,正以前所未有的速度推进“数字中国”建设。本文旨在探讨“开源AI智能名片21链动模式S2B2C商城小程序源码”在数字经济背景下的应用潜力与实践价值,从多个维度分析其对…...
【English-Book】Go in Action目录页翻译中文
第8页 内容 前言 xi 序言 xiii 致谢 xiv 关于本书 xvi 关于封面插图 xix 1 介绍 Go 1 1.1 用 Go 解决现代编程挑战 2 开发速度 3 • 并发 3 • Go 的类型系统 5 内存管理 7 1.2 你好,Go 7 介绍 Go 玩具 8 1.3 总结 8 2 Go 快速入门 9 2.1 程序架构 10 2.2 主包 …...
js: 区分后端返回数字是否为null、‘-’ 或正常number类型数字。
问: 这是我的代码<CountTo v-if!isNaN(Number(item.num))> <span v-else>{{item.num}}</span> 我希望不是null的时候走countTo,是null的时候直接<span>{{item.num}}</span>显示 回答: 最终结果: …...
网络变压器的分类
网络变压器是局域网(LAN)中各级网络设备中必备的元件。它们的主要功能是传输数据,增强信号,并提供电气隔离,以防雷保护和匹配阻抗。网络变压器也被称为数据泵或网络隔离变压器。它们广泛应用于网络交换机、路由器、网卡、集线器等设备中。 网…...
SUCTF-SU_BBRE-好久不见21
哈哈哈哈哈哈,,,,纯汇编有大佬用工具反编译成伪代码吗。。。 题解: 由function2处逻辑,解rc4得到第一段flag We1com3ToReWorld,正常输入下执行完function0,程序结束,cong…...
Python 实现 NLP 的完整流程
💖 欢迎来到我的博客! 非常高兴能在这里与您相遇。在这里,您不仅能获得有趣的技术分享,还能感受到轻松愉快的氛围。无论您是编程新手,还是资深开发者,都能在这里找到属于您的知识宝藏,学习和成长…...
穷举vs暴搜vs深搜vs回溯vs剪枝系列一>N 皇后
题目: 解析: 1.决策树: 代码设计: 根据决策树剪枝设计: 代码: class Solution {private List<List<String>> ret;private char[][] path;private boolean[] checkdig1,checkdig2,checkco…...
JEL分类号
JEL分类系统,是美国经济学会“经济文献杂志”(《经济文献杂志》)所创立的对经济学文献的主题分类系统,并被现代西方经济学界广泛采用。 该分类方法主要采用开头的一个英文字母与随后的两位阿拉伯数字一起对经济学各部类进行“辞书式”编码分类。 https:…...
设计和优化用于 AR、HUD 和高级显示系统的表面浮雕光栅
表面浮雕光栅是许多光学系统中的关键组件,在控制增强现实 (AR) 显示器、平视显示器 (HUD) 和其他先进光子器件中的光传播方面发挥着关键作用。作为在这个领域工作的工程师和设计师,您了解针对特定应用优化这…...
【今日分享】人工智能加速发现能源新材料的结构与性能
人工智能与材料国际学术会议(ICAIM)workshop9是由来自宁夏大学材料与新能源学院副院长王海龙教授及马薇副教授、杜鑫老师组成,他们将以“人工智能加速发现新能源新材料的结构与性能”为主题开展研讨工作,欢迎对该主题感兴趣的专家学者携稿加入。 loadin…...
Boost Asio TCP异步服务端和客户端
服务端 消息分两次发送,第一次发送head,第二次发送body。接收也是先接收head,然后通过head结构中的body长度字段再接收body。 TcpServer.h #pragma once #include <atomic> #include <vector> #include <unordered_set> #…...
1.7 ChatGPT:引领AI对话革命的致胜之道
ChatGPT:引领AI对话革命的致胜之道 随着人工智能(AI)技术的迅猛发展,特别是在自然语言处理(NLP)领域,OpenAI 的 ChatGPT 已经成为了举世瞩目的技术突破。从普通的自动化客服到深入的创作与协作,ChatGPT 通过其卓越的语言理解和生成能力,改变了人们与计算机交互的方式…...
WPS数据分析000001
目录 一、表格的新建、保存、协作和分享 新建 保存 协作 二、认识WPS表格界面 三、认识WPS表格选项卡 开始选项卡 插入选项卡 页面布局选项卡 公式选项卡 数据选项卡 审阅选项卡 视图选项卡 会员专享选项卡 一、表格的新建、保存、协作和分享 新建 ctrlN------…...
电脑风扇声音大怎么办? 原因及解决方法
电脑风扇是电脑的重要组件之一,它的作用是为电脑的各个部件提供冷却,防止电脑过热。然而,有时候我们会发现电脑风扇的声音特别大,不仅影响我们的使用体验,也可能是电脑出现了一些问题。那么,电脑风扇声音大…...
高效实现 Markdown 转 PDF 的跨平台指南20250117
高效实现 Markdown 转 PDF 的跨平台指南 引言 Markdown 文件以其轻量化和灵活性受到开发者和技术写作者的青睐,但如何将其转换为易于分享和打印的 PDF 格式,是一个常见需求。本文整合了 macOS、Windows 和 Linux 三大平台的转换方法,并探讨…...
Spark Streaming的核心功能及其示例PySpark代码
Spark Streaming是Apache Spark中用于实时流数据处理的模块。以下是一些常见功能的实用PySpark代码示例: 基础流处理:从TCP套接字读取数据并统计单词数量 from pyspark import SparkContext from pyspark.streaming import StreamingContext# 创建Spar…...
SkyWalking 10.2.0 SWCK 配置过程
SkyWalking 10.2.0 & SWCK 配置过程 skywalking oap-server & ui 使用Docker安装在K8S集群以外,K8S集群中的微服务使用initContainer按命名空间将skywalking-java-agent注入到业务容器中。 SWCK有整套的解决方案,全安装在K8S群集中。 具体可参…...
python打卡day49
知识点回顾: 通道注意力模块复习空间注意力模块CBAM的定义 作业:尝试对今天的模型检查参数数目,并用tensorboard查看训练过程 import torch import torch.nn as nn# 定义通道注意力 class ChannelAttention(nn.Module):def __init__(self,…...
ubuntu搭建nfs服务centos挂载访问
在Ubuntu上设置NFS服务器 在Ubuntu上,你可以使用apt包管理器来安装NFS服务器。打开终端并运行: sudo apt update sudo apt install nfs-kernel-server创建共享目录 创建一个目录用于共享,例如/shared: sudo mkdir /shared sud…...
遍历 Map 类型集合的方法汇总
1 方法一 先用方法 keySet() 获取集合中的所有键。再通过 gey(key) 方法用对应键获取值 import java.util.HashMap; import java.util.Set;public class Test {public static void main(String[] args) {HashMap hashMap new HashMap();hashMap.put("语文",99);has…...
STM32+rt-thread判断是否联网
一、根据NETDEV_FLAG_INTERNET_UP位判断 static bool is_conncected(void) {struct netdev *dev RT_NULL;dev netdev_get_first_by_flags(NETDEV_FLAG_INTERNET_UP);if (dev RT_NULL){printf("wait netdev internet up...");return false;}else{printf("loc…...
电脑插入多块移动硬盘后经常出现卡顿和蓝屏
当电脑在插入多块移动硬盘后频繁出现卡顿和蓝屏问题时,可能涉及硬件资源冲突、驱动兼容性、供电不足或系统设置等多方面原因。以下是逐步排查和解决方案: 1. 检查电源供电问题 问题原因:多块移动硬盘同时运行可能导致USB接口供电不足&#x…...
React19源码系列之 事件插件系统
事件类别 事件类型 定义 文档 Event Event 接口表示在 EventTarget 上出现的事件。 Event - Web API | MDN UIEvent UIEvent 接口表示简单的用户界面事件。 UIEvent - Web API | MDN KeyboardEvent KeyboardEvent 对象描述了用户与键盘的交互。 KeyboardEvent - Web…...
项目部署到Linux上时遇到的错误(Redis,MySQL,无法正确连接,地址占用问题)
Redis无法正确连接 在运行jar包时出现了这样的错误 查询得知问题核心在于Redis连接失败,具体原因是客户端发送了密码认证请求,但Redis服务器未设置密码 1.为Redis设置密码(匹配客户端配置) 步骤: 1).修…...
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列,以便知晓哪些列包含有价值的数据,…...
九天毕昇深度学习平台 | 如何安装库?
pip install 库名 -i https://pypi.tuna.tsinghua.edu.cn/simple --user 举个例子: 报错 ModuleNotFoundError: No module named torch 那么我需要安装 torch pip install torch -i https://pypi.tuna.tsinghua.edu.cn/simple --user pip install 库名&#x…...
