如何处理 iOS 客户端内 Webview H5 中后台播放的音视频问题
目录
- 问题描述
- Page Visibility API 的应用
- 什么是 Page Visibility API?
- 使用 Page Visibility API 暂停音视频
- 完整解决方案
- 1. 监听媒体的播放和暂停事件
- 2. 防止自动播放
- 3. 结合 Intersection Observer 进行媒体控制
- 4. 手动处理应用生命周期中的事件
问题描述
在 iOS 客户端中,通过 WebView 打开 H5 页面时,如果页面中包含视频或音频内容,WebView 默认情况下会继续在后台播放这些媒体。即使用户切换到了其他应用,媒体也不会停止播放,这可能导致以下几个问题:
-
用户体验差:用户可能没有意识到音频或视频在继续播放,产生意外的声音输出。
-
电量和数据流量的消耗:后台播放会消耗不必要的设备电量和流量,影响用户的电池续航和流量使用。
-
隐私问题:后台播放的音频或视频内容可能是用户不希望被别人听到的,影响用户隐私。
为了解决这些问题,我们可以从 H5 页面(前端代码)的角度采取相应的措施,确保在 WebView 进入后台时,音视频能够停止播放。
Page Visibility API 的应用
在 H5 页面中,我们可以使用 Page Visibility API 来监听页面的可见性变化,从而做出相应的处理。Page Visibility API 允许开发者检测页面当前是否在浏览器的可视区域中,当页面不可见时,触发相应的逻辑来暂停音视频的播放。
什么是 Page Visibility API?
Page Visibility API 是一种网页的可见性检测机制,通过 document.visibilityState 和 visibilitychange 事件,开发者可以得知页面当前的状态是否为可见。这个 API 在处理 WebView 的后台行为时非常有用。
document.visibilityState:用于获取页面当前的可见状态,值可以是 visible(可见)或 hidden(隐藏)。
visibilitychange 事件:当页面的可见性状态发生变化时(例如切换到后台),这个事件就会被触发。
使用 Page Visibility API 暂停音视频
以下是一个简单的代码示例,展示如何使用 Page Visibility API 来暂停页面中的所有音频和视频:
// 监听页面的可见性变化
document.addEventListener('visibilitychange', function() {if (document.visibilityState === 'hidden') {// 获取页面中所有的视频和音频元素var videos = document.querySelectorAll('video');var audios = document.querySelectorAll('audio');// 遍历所有视频并暂停播放videos.forEach(function(video) {video.pause();});// 遍历所有音频并暂停播放audios.forEach(function(audio) {audio.pause();});} else if (document.visibilityState === 'visible') {// 可选:页面重新可见时恢复播放var videos = document.querySelectorAll('video[data-autoplay-resume="true"]');var audios = document.querySelectorAll('audio[data-autoplay-resume="true"]');videos.forEach(function(video) {video.play();});audios.forEach(function(audio) {audio.play();});}
});
代码解析:
-
监听 visibilitychange 事件:当页面的可见性发生变化时,会触发 visibilitychange 事件。
-
判断页面是否不可见:通过 document.visibilityState 判断页面是否为 hidden。
-
暂停视频和音频播放:当页面不可见时,获取页面中的所有 video 和 audio 元素,调用它们的 pause() 方法,确保音视频在后台不会继续播放。
-
页面重新可见时的自动恢复播放:当页面重新可见时,可选地恢复之前暂停的音视频播放,具体由自定义属性 data-autoplay-resume 标记需要恢复的媒体元素。
-
这种方法对于前端开发者来说简单且有效,无需与原生应用代码交互即可实现后台播放控制。
完整解决方案
尽管 Page Visibility API 是一种非常好的解决方法,但它有时可能并不完全可靠,尤其是在一些边界场景下(如不同的浏览器环境,或者由于其他因素导致的未检测到页面的可见性变化)。为了应对这些潜在的问题,我们可以进一步采取以下措施:
1. 监听媒体的播放和暂停事件
为了增强控制的可靠性,我们可以监听页面中媒体元素的 play 和 pause 事件。当页面进入后台时,强制暂停所有的音视频播放。
// 强制控制音视频播放
function handleMediaPlayback() {var videos = document.querySelectorAll('video');var audios = document.querySelectorAll('audio');videos.forEach(function(video) {video.addEventListener('play', function() {if (document.visibilityState === 'hidden') {video.pause();}});});audios.forEach(function(audio) {audio.addEventListener('play', function() {if (document.visibilityState === 'hidden') {audio.pause();}});});
}document.addEventListener('DOMContentLoaded', handleMediaPlayback);
通过监听媒体播放事件,我们可以更精确地控制音视频的播放,确保用户在切换页面或 WebView 进入后台时不会继续播放。
2. 防止自动播放
在前端代码中,尽量避免设置音视频的自动播放属性。在 iOS WebView 中,如果设置了 autoplay,即使页面进入后台,媒体可能仍会继续播放。因此,应将 autoplay 属性移除:
<video controls><source src="example.mp4" type="video/mp4">您的浏览器不支持 HTML5 视频。
</video>
确保视频元素没有 autoplay 属性,这样可以避免不必要的自动播放。
3. 结合 Intersection Observer 进行媒体控制
除了 Page Visibility API 之外,还可以使用 Intersection Observer API 来检测视频或音频元素是否在视口内,如果它们不可见,则暂停播放。
let observer = new IntersectionObserver((entries) => {entries.forEach(entry => {if (!entry.isIntersecting) {entry.target.pause();}});
});// 观察所有视频和音频元素
let mediaElements = document.querySelectorAll('video, audio');
mediaElements.forEach((element) => {observer.observe(element);
});
这种方法可以进一步提高音视频控制的灵活性,确保用户在滚动页面或切换标签时,视频和音频不会继续播放。
4. 手动处理应用生命周期中的事件
对于一些场景,我们可以直接与移动应用开发团队合作,通过消息传递的方式,当应用进入后台时,通知 WebView 执行 JavaScript 代码来暂停媒体播放。iOS 开发者可以通过 WebKit 的 evaluateJavaScript 方法执行前端提供的暂停脚本。
let js = "document.querySelectorAll('video, audio').forEach(media => media.pause());"
webView.evaluateJavaScript(js, completionHandler: nil)
相关文章:

如何处理 iOS 客户端内 Webview H5 中后台播放的音视频问题
目录 问题描述Page Visibility API 的应用什么是 Page Visibility API?使用 Page Visibility API 暂停音视频完整解决方案1. 监听媒体的播放和暂停事件2. 防止自动播放3. 结合 Intersection Observer 进行媒体控制4. 手动处理应用生命周期中的事件 问题描述 在 iOS…...

C++的一些模版
1、不限制次数的输入数据 vector<int> nums;int num;while (cin >> num) {nums.push_back(num);if (cin.get() \n) break;}2、取模模版 template<int kcz> struct ModInt { #define T (*this)int x;ModInt() : x(0) {}ModInt(int y) : x(y > 0 ? y : y…...

spring boot整合https协议
整体目录 1. 生成SSL证书 首先,使用keytool生成一个自签名证书。打开命令行工具并运行以下命令: keytool -genkeypair -alias myserver -keyalg RSA -keysize 2048 -keystore keystore.jks -validity 365 这将创建一个名为keystore.jks的文件…...

服务器开机即占用大量内存,解决
1.服务器开机两分钟不到,内存使用飙升 [rootlocalhost ~]# top #查看是否有了明显的内存占用程序 2.上述未果,查看是否有违规的开机自启项 [rootlocalhost ~]# chkconfig --list 3.上述无果,查看开启启动加载项内容 上网搜后ÿ…...

Keil uvision的edition
0 Preface/Foreword 0.1 参考网址 https://zhuanlan.zhihu.com/p/456069876 1 Keil版本介绍 版本介绍: Keil Lite(免费版):最多32KB代码,无法使用中间件Keil Essential(基础版):没…...

[每周一更]-(第123期):模拟面试|消息队列面试思路解析
文章目录 22|消息队列:消息队列可以用来解决什么问题?1. 你用过消息队列吗?主要用来解决什么问题?异步、削峰和解耦你能各举一个例子吗?2. 你用的是哪个消息队列?为什么使用它而不用别的消息队列?3. 为什么你一定要用消息队列?不用行不行?不用有什么缺点?4. 在对接多…...

游戏引擎学习第12天
视频参考:https://www.bilibili.com/video/BV1yom9YnEWY 这节没讲什么东西,主要是改了一下音频的代码 后面有介绍一些alloc 和malloc,VirtualAlloc 的东西 _alloca 函数(或 alloca)分配的是栈内存,它的特点是: 生命周…...

深入理解Flutter生命周期函数之StatefulWidget(一)
目录 前言 1.为什么需要生命周期函数 2.开发过程中常用的生命周期函数 1.initState() 2.didChangeDependencies() 3.build() 4.didUpdateWidget() 5.setState() 6.deactivate() 7.dispose() 3.Flutter生命周期总结 1.调用顺序 2.函数调用时机以及主要作用 4.生…...

413: Quick Sort
解法: #include <bits/stdc.h> using namespace std; const int N1e55; int a[N]; int n;int main(int argc, char** argv) {cin>>n;for (int i0;i<n;i) cin>>a[i];sort(a,an);for (int i0;i<n;i) cout<<a[i]<<" "…...

vue之axios根据某个接口创建实例,并设置headers和超时时间,捕捉异常
import axiosNew from axios;//给axios起个别名//创建常量实例 const instanceNew axiosNew.create({//axios中请求配置有baseURL选项,表示请求URL的公共部分,url baseUrl requestUrlbaseURL: baseURL,//设置超时时间为20秒timeout: 20000,headers: {…...

Pandas数据透视表:交叉分析与聚合计算
大家好,在数据分析中,数据透视表(Pivot Table)是一种强大的工具,用于交叉分析和聚合计算。Pandas库中的数据透视表功能,使我们能够在多维数据中快速生成汇总表、统计特定维度的聚合数据,帮助揭示…...

软件设计师考试大纲
文章目录 一 、考 试 说 明1. 考试目标2. 考试要求3. 考试科目设置 二、考 试 范 围考试科目1:计算机与软件工程知识1. 计算机系统基础知识1.1计算机内数据的表示及运算1.2 其他数学基础知识1.3 计算机硬件基础知识1.3.1 计算机系统的组成、体系结构分类及特性1.3.2 存储系统1.…...

一文说清C++类型转换操作符(cast operator)
一 前言 大家在编程时,一定会遇到要做类型转换的应用场景。 但是,C风格的类型转换太强大,太危险,它允许将一个给定类型转换成我们想要的任何其他类型。 所以在C中,提供了一些更安全和更明确的类型转换操作符ÿ…...

MOSFET电路栅源极GS之间并联电容后,MOS炸管原因分析
1、前言 在介绍,在进行MOSFET相关的电路设计时,可能会遇到MOSFET误导通的问题,为了解决此问题,我们提出了两种方法,一种是增大MOSFET栅极串联电阻的阻值,另外一种是在MOSFET栅-源极之间并联一个电容&#…...

gitHub常用操作
gitHub常用操作 1、把项目拉下来2、添加上游仓库3、进入分支4、从上游仓库拉取更新 1、把项目拉下来 在对应项目的右上角点击fork,fork下来:将远程仓库复制到个人仓库 在创建好的分支文件夹下使用 git clone自己远程仓库下的http地址(fork…...

[项目代码] YOLOv5 铁路工人安全帽安全背心识别 [目标检测]
YOLOv5是一种单阶段(one-stage)检测算法,它将目标检测问题转化为一个回归问题,能够在一次前向传播过程中同时完成目标的分类和定位任务。相较于两阶段检测算法(如Faster R-CNN),YOLOv5具有更高的…...

Java 垃圾回收机制(GC)概览
简介 Java垃圾收集、堆和运行时编译器默认选择 jdk1.9开始,默认使用G1收集器,GC Threads的最大数量受堆大小和可用CPU资源限制初始堆大小为物理内存的1/64最大堆大小为物理内存的1/4分层编译器,同时使用C1和C2 JVM 垃圾收集器可以为配置优…...

Kafka节点服役和退役
1 服役新节点 1)新节点准备 (1)关闭 bigdata03,进行一个快照,并右键执行克隆操作。 (2)开启 bigdata04,并修改 IP 地址。 vi /etc/sysconfig/network-scripts/ifcfg-ens33修改完…...

Git如何简单使用
文章目录 GitGitlabGitLab和GitHub有什么区别?Gitlab简单使用Gitlab常用指令Git Git是一个分布式版本控制系统。 它用于记录文件的修改历史,方便多人协作开发软件等项目。例如一个软件开发团队,成员们会频繁修改代码,Git可以追踪每个人的修改内容、时间等信息。 主要功能…...

酒水分销积分商城小程序开发方案php+uniapp
酒水分销积分商城小程序开发,开发语言后端php,前端uniapp。核心功能模块:酒水商城、积分商城、二级分销、抽奖、优惠券。可以二开或定制。协助部署搭建。...

MTU-内核态(数据链路层或网络接口上能够传输的最大数据包大小)
MTU(最大传输单元,Maximum Transmission Unit)是网络中用于表示数据链路层或网络接口上能够传输的最大数据包大小。 1. 工作原理 MTU 决定了一个数据包(包括头部和数据部分)的最大长度。它影响到数据的传输ÿ…...

React的基础API介绍(一)
目录 useEffect1. 替代生命周期方法2. 副作用管理3. 依赖项数组4. 多次使用5. 与闭包配合6. 支持异步操作7. 减少样板代码 注意事项useEffetct是如何拿到变量count最新的值?1. 每次渲染都会创建新的函数作用域2. 闭包捕获最新的状态值3. useEffect 的执行时机 useLa…...

【Electron】总结:如何创建Electron+Element Plus的项目
我将结合官网手册与AI问到的信息,直接给出步骤,与命令。 一、准备环境 首先在C盘Users,你的登录的账号名文件夹下,编辑.npmrc文件。添加镜像地址。 如果使用了yarn,则是.yarnrc。可以全部都配置。 npm install -g …...

从依托指标字典到 NoETL 自动化指标平台,指标口径一致性管理的进阶
今天,我们一起来梳理和盘点下不同代际指标平台如何实现指标口径一致性管理: 第一代:指标口径登记与管理 第一代指标平台聚焦于指标口径的登记与管理,依托指标字典实现企业指标口径的有效检索与管理功能。 此阶段,业…...

嵌入式面试题练习 - 2024/11/15
欢迎找我进行职业规划,超值的自我投资 -> 嵌入式软件工程师一对一指导 1.设有定义char *p[]{"Shanghai","Beijing","Honkong"};则结果为j字符的表达式是() A *p[1] 3 B *(p[1] 3) C *(p[3] 1) D p[3] […...

分析http话术异常挂断原因
用户反馈在与机器人通话时,自己明明有说话,但是通话还是被挂断了,想知道原因。 分析日志 我们根据用户提供的freeswitch日志分析:发现是因为超时导致话术执行hangup动作,结束了通话。 从这一行向上分析日志ÿ…...

云岚到家 秒杀抢购
目录 秒杀抢购业务特点 常用技术方案 抢券 抢券界面 进行抢券 我的优惠券列表 活动查询 系统设计 活动查询分析 活动查询界面显示了哪些数据? 面向高并发如何提高活动查询性能? 如何保证缓存一致性? 数据流 Redis数据结构设计 如…...

【WPF】Prism库学习(一)
Prism介绍 1. Prism框架概述: Prism是一个用于构建松耦合、可维护和可测试的XAML应用程序的框架。它支持WPF、.NET MAUI、Uno Platform和Xamarin Forms等多个平台。对于每个平台,Prism都有单独的发布版本,并且它们在不同的时间线上独立开发。…...

0 -vscode搭建python环境教程参考(windows)
引用一篇非常详细的vscode搭建python环境教程 链接:vscode安装以及配置Python基本环境 以下是VSCode和PyCharm的对比 个人更建议使用VSCode Visual Studio Code (VSCode) Visual Studio Code 是由微软开发的一款免费、开源的轻量级代码编辑器。它支持多种编程语…...

Uniapp 引入 Android aar 包 和 Android 离线打包
需求: 原生安卓 apk 要求嵌入到 uniapp 中,并通过 uniapp 前端调起 app 的相关组件。 下面手把手教你,从 apk 到 aar,以及打包冲突到如何运行,期间我所遇到的问题都会 一 一 进行说明,相关版本以我文章内为…...