前端实现视频/直播预览
有一个需求:后端返回视频的预览地址,不仅要支持这个视频的预览,还需要设置视频封面。
这里有两种情况:
- 如果是类似.mp4,.mov等格式的视频可以选用原生 video 进行视频展示,并且原生的 video 也支持全屏、音量调节、倍速调节、视频下载、封面设置等等...
- 如果是类似.m3u8等格式的直播不推荐使用 video,可以使用 Plyr 等实现, 这个我会放在后面讲。(一开始我使用的是 ckPlayer, 但它在 ios 系统中无法播放带有一些携带 token 等参数的视频地址,因此换了 Plyr)
一、.mp4,.mov 视频预览
<video width="100%" controls :src="video_url" :poster="poster_img" class="video-player"/>
- src为视频地址,poster为封面图地址, controls为显示浏览器默认的视频控制栏(播放/暂停,音量调节,进度调节,倍速控制,全屏播放,视频下载...);
- 如果不加 controls,用户无法控制视频。
样式方面不过多赘述,可根据实际情况进行调整:
.video-player {max-width: 100%;max-height: 100%;object-fit: contain;
}
如果需要支持更多格式的视频播放并进行具体的类型控制,可以用下面的写法:
<video width="100%" controls :poster="posterImage" class="video-player"><!-- 支持 MP4 格式 --><source :src="video_url" type="video/mp4"><!-- 支持 MOV 格式 --><source :src="video_url.replace('.mp4', '.mov')" type="video/quicktime"><!-- 支持 WebM 格式 --><source :src="video_url.replace('.mp4', '.webm')" type="video/webm"><!-- 支持 Ogg 格式 --><source :src="video_url.replace('.mp4', '.ogv')" type="video/ogg">暂不支持播放
</video>
注意:有些 .mp4 视频是 HEVC 编码的(如下图),这类视频在谷歌浏览器中可以播放,但在 Edge 等的浏览器会无法播放,这与内核浏览器的视频格式支持有关,不是 video 标签的问题,如果要做兼容性处理,可以进行转码、使用播放插件,或推荐使用谷歌浏览器访问。
二、直播预览
这里模拟一个场景:在某段时间内进行直播,直播开始前和直播结束后将播放器替换为封面图。
页面结构:
<div v-show="showPlayer" class="live-header-video"><video id="live-header-video" playsinline :poster="cover_url"></video>
</div><img v-show="!showPlayer" :src="cover_url" alt="">
注意:这里不要写成自闭合的<video /> !!!
- v-show="showPlayer":仅在满足直播时间条件时显示视频容器
- id="live-header-video":用于 JS 中获取 video DOM
- playsinline:在移动设备上避免自动全屏播放
- :poster:设置视频封面
变量定义:
import Plyr from 'plyr';
import Hls from 'hls.js';
import 'plyr/dist/plyr.css'const showPlayer = ref(false)
const header_url = ref('')
const live_start_time = ref('')
const live_end_time = ref('')
const cover_url = ref('')
let videoPlayer = null
let div_video = null
- showPlayer:控制是否显示播放器
- header_url:视频地址
- cover_url:封面地址
- live_start_time和live_end_time:直播开始和结束时间
- videoPlayer:保存 Plyr 播放器实例
- div_video:video DOM 元素,用于绑定 Hls 或 Plyr
核心函数:
const setVideoPlayer = () =>{if (header_url.value){div_video = document.getElementById('live-header-video')videoPlayer = new Plyr(div_video, {controls: ['play-large','play','mute','volume','captions','pip','airplay','fullscreen'],settings: [],fullscreen: {enabled: true,fallback: true,iosNative: true}})const urlObj = new URL(header_url.value);if (urlObj.pathname.endsWith('.m3u8')) {if (Hls.isSupported()) {const hls = new Hls();hls.loadSource(header_url.value);hls.attachMedia(div_video);videoPlayer.play()hls.on(Hls.Events.MANIFEST_PARSED, () => {console.log('HLS 流准备就绪');});hls.on(Hls.Events.ERROR, (event, data) => {console.error('HLS 错误:', data);});}else if (div_video.canPlayType('application/vnd.apple.mpegurl')) {console.log("不支持Hls.isSupported")div_video.src = header_url.value;}else {console.error('不支持 HLS 播放');}}else {div_video.src = header_url.valuediv_video.load();div_video.addEventListener('loadeddata', () => {videoPlayer.play();});}if(live_start_time.value && live_end_time.value) {const startDate = new Date(live_start_time.value.replace(" ", "T"));const endDate = new Date(live_end_time.value.replace(" ", "T"));const currentDate = new Date();showPlayer.value = currentDate >= startDate && currentDate <= endDate;}}
}
代码解析:
div_video = document.getElementById('live-header-video')
videoPlayer = new Plyr(div_video, { ... })
- 获取 video 元素并初始化 Plyr
- Plyr 是视频播放器库,初始化后控制按钮等都由它管理
if (urlObj.pathname.endsWith('.m3u8')) {
- 判断是否是 .m3u8 流地址
如果是 .m3u8:
- 使用 Hls.js 加载并绑定流:
if (Hls.isSupported()) {const hls = new Hls();hls.loadSource(header_url.value);hls.attachMedia(div_video);videoPlayer.play()...
}
- 若不支持 Hls.js,但浏览器原生支持 .m3u8(如 Safari):
else if (div_video.canPlayType('application/vnd.apple.mpegurl')) {div_video.src = header_url.value;
}
- 否则提示不支持 HLS 播放。
- 如果是普通视频文件(非 .m3u8)
div_video.src = header_url.value
div_video.load();
div_video.addEventListener('loadeddata', () => {videoPlayer.play();
});
- 当视频数据加载完成后自动播放
if(live_start_time.value && live_end_time.value) {const startDate = new Date(live_start_time.value.replace(" ", "T"));const endDate = new Date(live_end_time.value.replace(" ", "T"));const currentDate = new Date();showPlayer.value = currentDate >= startDate && currentDate <= endDate;
}
- 判断是否在直播时间段:时间格式转为 Date 对象后进行比较,控制 showPlayer 的值,从而决定是否显示播放器。
相关文章:

前端实现视频/直播预览
有一个需求:后端返回视频的预览地址,不仅要支持这个视频的预览,还需要设置视频封面。 这里有两种情况: 如果是类似.mp4,.mov等格式的视频可以选用原生 video 进行视频展示,并且原生的 video 也支持全屏、…...

React源码阅读-fiber核心构建原理
React源码阅读(2)-fiber核心构建原理 好的,我明白了。您提供的文本主要介绍了 React 源码中 Fiber 核心的构建原理,涵盖了从执行上下文到构建、提交、调度等关键阶段,以及相关的代码实现。 您提出的关联问题也很重要,它们深入探讨…...

视频监控管理平台EasyCVR与V4分析网关对接后告警照片的清理优化方案
一、问题概述 在安防监控、设备运维等场景中,用户将视频监控管理平台EasyCVR与V4网关通过http推送方式协同工作时,硬件盒子上传的告警图片持续累积,导致EasyCVR服务器存储空间耗尽,影响系统正常运行与告警功能使用。 二、解决方…...
基于 BGE 模型与 Flask 的智能问答系统开发实践
基于 BGE 模型与 Flask 的智能问答系统开发实践 一、前言 在人工智能快速发展的今天,智能问答系统成为了提升信息检索效率和用户体验的重要工具。本文将详细介绍如何利用 BGE(Base General Embedding)模型、Faiss 向量检索库以及 Flask 框架…...

机器学习:决策树和剪枝
本文目录: 一、决策树基本知识(一)概念(二)决策树建立过程 二、决策树生成(一)ID3决策树:基于信息增益构建的决策树。(二)C4.5决策树(三ÿ…...

vscode自定义主题语法及流程
vscode c/c 主题 DIY 启用自己的主题(最后步骤) 重启生效 手把手教你制作 在C:\Users\jlh.vscode\extensions下自己创建一个文件夹 里面有两个文件一个文件夹 package.json: {"name":"theme-jlh","displayName":"%displayName%&qu…...

vue中加载Cesium地图(天地图、高德地图)
目录 1、将下载的Cesium包移动至public下 2、首先需要将Cesium.js和widgets.css文件引入到 3、 新建Cesium.js文件,方便在全局使用 4、新建cesium.vue文件,展示三维地图 1、将下载的Cesium包移动至public下 npm install cesium后 2、…...

SpringBoot整合RocketMQ与客户端注意事项
SpringBoot整合RocketMQ 引入依赖(5.3.0比较稳定) <dependencies><dependency><groupId>org.apache.rocketmq</groupId><artifactId>rocketmq-spring-boot-starter</artifactId><version>2.3.1</version&…...
Github 2025-06-04 C开源项目日报 Top7
根据Github Trendings的统计,今日(2025-06-04统计)共有7个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量C项目7C++项目1Assembly项目1jq:轻量灵活的命令行JSON处理器 创建周期:4207 天开发语言:C协议类型:OtherStar数量:27698 个Fork数量:1538 …...
大二下期末
一.Numpy(Numerical Python) Numpy库是Python用于科学计算的基础包,也是大量Python数学和科学计算包的基础。不少数据处理和分析包都是在Numpy的基础上开发的,如后面介绍的Pandas包。 Numpy的核心基础是ndarray(N-di…...
LeetCode 热题 100 74. 搜索二维矩阵
LeetCode 热题 100 | 74. 搜索二维矩阵 大家好,今天我们来解决一道经典的算法题——搜索二维矩阵。这道题在 LeetCode 上被标记为中等难度,要求我们在一个满足特定条件的二维矩阵中查找一个目标值。如果目标值在矩阵中,返回 true;…...
解决 VSCode 中无法识别 Node.js 的问题
当 VSCode 无法识别 Node.js 时,通常会出现以下症状: 代码提示缺失require 等 Node.js API 被标记为错误调试功能无法正常工作终端无法运行 Node.js 命令 常见原因及解决方案 1. Node.js 未安装或未正确配置 解决方法: 确保已安…...

Mysql的卸载与安装
确保卸载干净mysql 不然在进行mysal安装时候会出现不一的页面和问题 1、卸载 在应用页面将查询到的mysql相关应用卸载 2、到c盘下将残留的软件包进行数据删除 3、删除programData下的mysql数据 4、检查系统中的mysql是否存在 cmd中执行 sc deleted mysql80 5、删除注册表中的…...
ES101系列09 | 运维、监控与性能优化
本篇文章主要讲解 ElasticSearch 中 DevOps 与性能优化的内容,包括集群部署最佳实践、容量规划、读写性能优化和缓存、熔断器等。 集群部署最佳实践 在生产环境中建议设置单一角色的节点。 Dedicated master eligible nodes:负责集群状态的管理。使用…...
Java常用的判空方法
文章目录 Java常用的判空方法JDK 自带的判空方法1. 使用 或 ! 运算符2. 使用 equals 方法3. Objects.isNull / Objects.nonNull4. Objects.equals4. JDK8 中的 Optional 第三方工具包1. Apache Commons Lang32. Google Guava3. Lombok 注解4. Vavr(函数式风格&…...

Excel处理控件Aspose.Cells教程:使用 C# 在 Excel 中创建组合图表
可视化项目时间线对于有效规划和跟踪至关重要。在本篇教程中,您将学习如何使用 C# 在 Excel 中创建组合图。只需几行代码,即可自动生成动态、美观的组合图。无论您是在构建项目管理工具还是处理内部报告,本指南都将向您展示如何将任务数据转换…...

【多线程初阶】阻塞队列 生产者消费者模型
文章目录 一、阻塞队列二、生产者消费者模型(一)概念(二)生产者消费者的两个重要优势(阻塞队列的运用)1) 解耦合(不一定是两个线程之间,也可以是两个服务器之间)2) 削峰填谷 (三)生产者消费者模型付出的代价 三、标准库中的阻塞队列(一)观察模型的运行效果(二)观察阻塞效果1) 队…...

《100天精通Python——基础篇 2025 第5天:巩固核心知识,选择题实战演练基础语法》
目录 一、踏上Python之旅二、Python输入与输出三、变量与基本数据类型四、运算符五、流程控制 一、踏上Python之旅 1.想要输出 I Love Python,应该使用()函数。 A.printf() B.print() C.println() D.Print() 在Python中想要在屏幕中输出内容,应该使用print()函数…...

机器人夹爪的选型与ROS通讯——机器人抓取系统基础系列(六)
文章目录 前言一、夹爪的选型1.1 任务需求分析1.2 软体夹爪的选型 二、夹爪的ROS通讯2.1 夹爪的通信方式介绍2.2 串口助手测试2.3 ROS通讯节点实现 总结Reference: 前言 本文将介绍夹爪的选型方法和通讯方式。以鞋子这类操作对象为例,将详细阐述了对应的夹爪选型过…...

第二十八章 RTC——实时时钟
第二十八章 RTC——实时时钟 目录 第二十八章 RTC——实时时钟 1 RTC实时时钟简介 2 RTC外设框图剖析 3 UNIX时间戳 4 与RTC控制相关的库函数 4.1 等待时钟同步和操作完成 4.2 使能备份域涉及RTC配置 4.3 设置RTC时钟分频 4.4 设置、获取RTC计数器及闹钟 5 实时时…...

使用 DuckLake 和 DuckDB 构建 S3 数据湖实战指南
本文介绍了由 DuckDB 和 DuckLake 组成的轻量级数据湖方案,旨在解决传统数据湖(如HadoopHive)元数据管理复杂、查询性能低及厂商锁定等问题。该方案为中小规模数据湖场景提供了简单、高性能且无厂商锁定的替代选择。 1. 什么是 DuckLake 和 D…...

大语言模型提示词(LLM Prompt)工程系统性学习指南:从理论基础到实战应用的完整体系
文章目录 前言:为什么提示词工程成为AI时代的核心技能一、提示词的本质探源:认知科学与逻辑学的理论基础1.1 认知科学视角下的提示词本质信息处理理论的深层机制图式理论的实际应用认知负荷理论的优化策略 1.2 逻辑学框架下的提示词架构形式逻辑的三段论…...

如何基于Mihomo Party http端口配置git与bash命令行代理
如何基于Mihomo Party http端口配置git与bash命令行代理 1. 确定Mihomo Party http端口配置 点击内核设置后即可查看 默认7892端口,开启允许局域网连接 2. 配置git代理 配置本机代理可以使用 127.0.0.1 配置局域网内其它机代理需要使用本机的非回环地址 IP&am…...
CMake 为 Debug 版本的库或可执行文件添加 d 后缀
在使用 CMake 构建项目时,我们经常需要区分 Debug 和 Release 构建版本。一个常见的做法是为 Debug 版本的库或可执行文件添加后缀(如 d),例如 libmylibd.so 或 myappd.exe。 本文将介绍几种在 CMake 中实现为 Debug 版本自动添加 d 后缀的方法。 方法一:使用 CMAKE_DEBU…...
Linux 特殊权限位详解:SetUID, SetGID, Sticky Bit
Linux 特殊权限位详解:SetUID, SetGID, Sticky Bit 在Linux权限系统中,除了基本的读、写(w)、执行(x)权限外,还有三个特殊权限位:SetUID、SetGID和Sticky Bit。这些权限位提供了更精细的权限控制机制,尤其在需要临时提升权限或管理共享资源时非常有用。 一、SetUID (s位…...

埃文科技智能数据引擎产品入选《中国网络安全细分领域产品名录》
嘶吼安全产业研究院发布《中国网络安全细分领域产品名录》,埃文科技智能数据引擎产品成功入选数据分级分类产品名录。 在数字化转型加速的今天,网络安全已成为企业生存与发展的核心基石,为了解这一蓬勃发展的产业格局,嘶吼安全产业…...
使用VTK还是OpenGL集成到qt程序里哪个好?
在Qt程序中集成VTK与OpenGL:选择哪个更好? 在Qt程序中实现三维可视化时,开发者常常面临一个选择:是使用VTK(Visualization Toolkit)还是OpenGL(Open Graphics Library)。这两种技术…...
Java-IO流之打印流详解
Java-IO流之打印流详解 一、打印流概述1.1 什么是打印流1.2 打印流的特点1.3 打印流的应用场景 二、PrintStream详解2.1 基本概念2.2 构造函数2.3 核心方法2.4 使用示例 三、PrintWriter详解3.1 基本概念3.2 构造函数3.3 核心方法3.4 使用示例 四、PrintStream与PrintWriter的比…...
高效图像处理:使用 Pillow 进行格式转换与优化
高效图像处理:使用 Pillow 进行格式转换与优化 1. 背景引入 在图像处理应用中,格式转换、裁剪、压缩等操作是常见需求。Python 的 Pillow 库基于 PIL(Python Imaging Library),提供 轻量、强大 的图像处理能力,广泛用于 Web 开发、数据分析、机器学习 等领域。 本文将…...
Github 2025-06-06 Java开源项目日报Top10
根据Github Trendings的统计,今日(2025-06-06统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Java项目10TypeScript项目1Java实现的算法集合:使用Gitpod.io进行编辑和贡献 创建周期:2883 天开发语言:Java协议类型:MIT LicenseStar数量…...