WebRtc直播流的播放

WebRtc直播流的播放

后端服务已经使用ZLMediaKit框架提供了直播流地址

一、WebRtc原生播放器播放直播流

1、使用ZLMediaKit框架提供的ZLMRTCClient.js

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>播放示例</title> <!-- 引入 zlmrtcclient.js 1.0.2 --> <script src="http://117.40.242.70:35080/ZLMRTCClient.js"></script> </head> <body> <!-- 视频播放元素 --> <video autoplay controls></video> <script> // 1. 获取视频元素 const videoElement = document.getElementById('videoPlayer'); // 2. 配置参数(严格对应 1.0.2 版本的 defaults 结构) const config = { element: videoElement, // 绑定视频播放元素(必填) debug: true, // 开启调试日志,方便排查问题 zlmsdpUrl: 'http://117.40.242.70:35080/index/api/webrtc?app=live&stream=test1&type=play', // 后端提供的流地址(必填) simulcast: false, // 不启用 simulcast(早期版本可能不支持) useCamera: false, // 播放时不需要启用本地摄像头 audioEnable: true, // 启用音频 videoEnable: true, // 启用视频 recvOnly: true, // 仅接收流(播放模式) resolution: { w: 800, h: 450 }, // 视频分辨率 usedatachannel: false // 不使用数据通道 }; // 3. 创建实例(1.0.2 版本可能通过构造函数直接初始化,无需 connect 方法) const rtcClient = new ZLMRTCClient.Endpoint(config); // 4. 监听事件(1.0.2 版本的事件名可能与早期定义一致) // 监听错误事件 rtcClient.on('error', (err) => { console.error('ZLMRTCClient 错误:', err); }); // 监听 WebRTC 不支持事件 rtcClient.on('webrtcNotSupport', () => { alert('当前浏览器不支持 WebRTC,请更换浏览器'); }); var remoteStream = null; // 监听远程流添加事件(核心:获取到流后播放) rtcClient.on('WEBRTC_ON_REMOTE_STREAMS', (event) => { console.log('收到远程流,开始播放'); // 1.0.2 版本可能自动绑定到 element,若未自动绑定则手动设置 if (videoElement.srcObject !== event.streams) { const stream = event.streams[0]; remoteStream = stream; videoElement.srcObject = stream; } }); // 监听连接状态变化 rtcClient.on('WEBRTC_ON_CONNECTION_STATE_CHANGE', (state) => { console.log('连接状态:', state); // 可能的状态:connecting/connected/disconnected }); function stop() { if (!rtcClient) return; // 停止视频播放 videoElement.pause(); videoElement.srcObject = null; // 停止媒体流轨道 if (remoteStream) { remoteStream.getTracks().forEach(track => track.stop()); remoteStream = null; } // 断开连接并清理实例 if (rtcClient.disconnect) { rtcClient.disconnect(); } else if (rtcClient.close) { rtcClient.close(); } rtcClient = null; } </script> </body> </html>

ZLMRTCClient.js的版本不同,实例化对象rtcClient对象不同,这里使用的1.0.2是

new ZLMRTCClient.Endpoint(config)

二、使用开源的播放器easyPlayer

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>演示</title> <script src="./js/EasyPlayer-pro.js"></script> </head> <style> .player_item { position: relative; padding-bottom: 56%; background-color: black; margin-bottom: 20px; } .player_box { position: absolute; top: 0; bottom: 0; right: 0; left: 0; } .inputs { -webkit-appearance: none; background-color: #fff; background-image: none; border-radius: 4px; border: 1px solid #dcdfe6; box-sizing: border-box; color: #606266; display: inline-block; font-size: inherit; height: 36px; line-height: 36px; outline: none; padding: 0 15px; transition: border-color .2s cubic-bezier(.645, .045, .355, 1); width: 100%; max-width: 600px; margin-right: 16px; margin-bottom: 16px; margin-top: 16px; } .btn-item { cursor: pointer; display: inline-block; padding: 6px 12px; margin-right: 15px; border-radius: 4px; border: 1px #ccc solid; } .df { display: flex; align-items: center; margin-bottom: 16px; } .box { padding-top: 10px; margin: auto; max-width: 800px; } </style> <body> <div> <br> <h2>EasyPlayerPro案例演示</h2> <br> <div> <div></div> </div> <div> <input type="checkbox" /><span>音频</span> </div> <input type="text" value="ws://localhost:6230/ws_flv/live/stream_1_0.flv"> <div> <div>播放</div> <div>重播</div> <div>静音</div> <div>注销</div> </div> </div> <script> window.onload = function () { var playerInfo = null var config = { stretch: true, hasAudio: true, } playCreate() var input = document.getElementById('input'); var audio = document.getElementById('hasAudio'); var player = document.getElementById('onPlayer'); var replay = document.getElementById('onReplay'); var mute = document.getElementById('onMute'); var stop = document.getElementById('onStop'); if (config.hasAudio) audio.checked = true audio.onclick = () => { if (audio.checked == true) { config.hasAudio = true } else { config.hasAudio = false } } player.onclick = () => { onPlayer() } replay.onclick = () => { onReplay() } mute.onclick = () => { onMute() } stop.onclick = () => { onStop() } function playCreate() { var container = document.getElementById('player_box'); playerInfo = new EasyPlayerPro(container, config); } function onPlayer() { playerInfo && playerInfo.play(input.value).then(() => { }).catch((e) => { console.error(e); }); } function onMute() { if (!playerInfo) return playerInfo.setMute(true) } function onReplay() { onDestroy().then(() => { playCreate(); onPlayer() }); } function onDestroy() { return new Promise((resolve, reject) => { if (playerInfo) { playerInfo.destroy() playerInfo = null } setTimeout(() => { resolve(); }, 100); }) } function onStop() { onDestroy().then(() => { playCreate(); }); } } </script> </body> </html>

播放效果好,具有截图、视频录制等功能

Read more

前端浏览器指纹:原理、实现与应用实践

前端浏览器指纹:原理、实现与应用实践 * 前言 * 一、技术原理 * 二、指纹插件 * 三、实际应用场景 * 四、指纹防护方法 * 五、法律和隐私合规问题 前言 浏览器指纹(Browser Fingerprinting)是一种通过收集用户浏览器环境的各种特征信息,生成唯一标识符的技术。与传统的Cookie不同,浏览器指纹具有无感知、难清除、跨会话追踪的特点,即使清除Cookie、使用隐私模式,指纹依然可以识别同一用户。 一、技术原理 1. 基础特征收集 浏览器指纹主要依赖以下可采集的特征维度: 特征类别具体指标采集方法HTTP头信息User-Agent、Accept-Language、Accept-Encodingnavigator.userAgent等屏幕属性屏幕分辨率、色彩深度、像素比screen.width/height时区与语言时区、系统语言Intl.DateTimeFormat插件检测已安装插件列表navigator.plugins字体检测系统字体列表navigator.fontsCanvas指纹图形渲染差异Canvas绘制测试W

用 Vue 3 重构 Dify 聊天前端(上篇):项目搭建与基础架构

用 Vue 3 重构 Dify 聊天前端(上篇):项目搭建与基础架构

本系列教程将带你从零开始,用 Vue 3 + TypeScript 复刻一个类似 Dify 的 AI 聊天前端。上篇聚焦项目搭建、类型设计、路由认证、HTTP 封装和状态管理。 项目简介 背景 Dify 是一个开源的 LLM 应用开发平台,提供了对话式 AI 的后端服务。在实际项目中,我们往往需要自建前端来对接Dify后端 API或LLM后端服务,实现定制化的聊天界面。 本项目的目标:用 Vue 3 构建一个生产级的 AI 聊天前端,具备以下能力: * SSE 流式输出(打字机效果) * Markdown 渲染 + 代码高亮 * 用户认证 * 文件/图片上传 * 聊天会话历史管理 * 工作流执行可视化 * Agent 思考过程展示 * 移动端响应式适配

工业协议驱动热插拔:基于 WebAssembly 的运行时动态加载架构实战 (Rust/Go 示例)

工业协议驱动热插拔:基于 WebAssembly 的运行时动态加载架构实战 (Rust/Go 示例)

一、 场景痛点:为了改一个驱动,重启了整条产线 在最近的一个半导体封装厂项目中,我们遇到了典型的“单体架构”瓶颈: * 现状:网关核心程序是用 C++ 写的一个巨大单体(Monolith),集成了西门子、三菱、欧姆龙等 20 种协议驱动。 * 事故:现场新进了一台国产贴片机,使用非标的 TCP 协议。 * 代价: 1. 研发团队花了 3 天修改 C++ 代码,增加新协议。 2. 重新编译整个固件,进行 OTA 升级。 3. 最致命的是:升级需要重启网关进程。就在重启的那 1 分钟里,其他正在运行的 50 台设备的关键生产数据断连了,导致 MES 系统误判报警,整条产线急停。 架构师指令:

WebVOWL 本体可视化工具完整部署手册

WebVOWL 本体可视化工具完整部署手册 【免费下载链接】WebVOWLVisualizing ontologies on the Web 项目地址: https://gitcode.com/gh_mirrors/we/WebVOWL 概述简介 WebVOWL 是一款专业的网络本体可视化工具,能够将复杂的 RDF 和 OWL 数据转换为直观的图形化展示。该工具采用现代化的 Web 技术栈,为语义网研究和本体工程提供了强大的可视化支持。 环境要求与前置准备 在开始部署之前,请确保您的系统满足以下基本要求: 系统环境要求: * Node.js 运行环境(推荐最新稳定版本) * 基本的命令行操作知识 * 现代浏览器支持(Chrome、Firefox、Safari、Edge) 软件版本确认: 通过命令行输入以下命令检查当前环境: node --version npm --version 完整部署流程 第一步: