记录---前端实现画中画超简单,让网页飞出浏览器

news/2024/12/2 19:36:00/文章来源:https://www.cnblogs.com/smileZAZ/p/18582546

🧑‍💻 写在开头

点赞 + 收藏 === 学会🤣🤣🤣

Document Picture-in-Picture 介绍

    今天,我来介绍一个非常酷的前端功能:文档画中画 (Document Picture-in-Picture, 本文简称 PiP)。你有没有想过,网页上的任何内容能悬浮在桌面上?😏

🎬 视频流媒体的画中画功能

        你可能已经在视频平台(如腾讯视频哔哩哔哩等网页)见过这种效果:视频播放时,可以点击画中画后。无论你切换页面,它都始终显示在屏幕的最上层,非常适合上班偷偷看电视💻

 

在今天的教程中,不仅仅是视频,我将教你如何将任何 HTML 内容放入画中画模式,无论是动态内容、文本、图片,还是纯炫酷的 div,统统都能“飞”起来。✨

        一个如此有趣的功能,在网上却很少有详细的教程来介绍这个功能的使用。于是我决定写一篇详细的教程来教大家如何实现画中画 (建议收藏)😁

体验网址:Treasure-Navigation

 

📖 Document Picture-in-Picture 详细教程

🛠 HTML 基本代码结构

    首先,我们随便写一个简单的 HTML 页面,后续的 JS 和样式都会基于它实现。

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document Picture-in-Picture API 示例</title><style>#pipContent {width: 600px;height: 300px;background: pink;font-size: 20px;}</style>
</head><body><div id="container"><div id="pipContent">这是一个将要放入画中画的 div 元素!</div><button id="clickBtn">切换画中画</button></div><script>// 在这里写你的 JavaScript 代码</script>
</body>
</html>

  

1️. 请求 PiP 窗口

    PiP 的核心方法是 window.documentPictureInPicture.requestWindow。它是一个 异步方法,返回一个新创建的 window 对象。
    PIP 窗口可以将其看作一个新的网页,但它始终悬浮在屏幕上方。

document.getElementById("clickBtn").addEventListener("click", async function () {// 获取将要放入 PiP 窗口的 DOM 元素const pipContent = document.getElementById("pipContent");// 请求创建一个 PiP 窗口const pipWindow = await window.documentPictureInPicture.requestWindow({width: 200,  // 设置窗口的宽度height: 300  // 设置窗口的高度});// 将原始元素添加到 PiP 窗口中pipWindow.document.body.appendChild(pipContent);
});

演示:

 

👏 现在,我们已经成功创建了一个画中画窗口! 这段代码展示了如何将网页中的元素放入一个新的画中画窗口,并让它悬浮在最上面。非常简单吧

关闭PIP窗口

可以直接点右上角关闭PIP窗口,如果我们想在代码中实现关闭,直接调用window上的api就可以了

window.documentPictureInPicture.window.close();

  

2️. 检查是否支持 PiP 功能

    一切不能兼容浏览器的功能介绍都是耍流氓,我们需要检查浏览器是否支持PIIP功能。 实际就是检查documentPictureInPicture属性是否存在于window上 🔧

if ('documentPictureInPicture' in window) {console.log("🚀 浏览器支持 PiP 功能!");
} else {console.warn("⚠️ 当前浏览器不支持 PiP 功能,更新浏览器或者换台电脑吧!");
}

  如果是只需要将视频实现画中画功能,视频画中画 (Picture-in-Picture) 的兼容性会好一点,但是它只能将元素放入画中画窗口。它与本文介绍的 文档画中画(Document Picture-in-Picture) 使用方法也是十分相似的。


3️. 设置 PiP 样式

    我们会发现刚刚创建的画中画没有样式,一点都不美观。那是因为我们只放入了dom元素,没有添加css样式。

3.1. 全局样式同步

假设网页中的所有样式如下:

<head><style>#pipContent {width: 600px;height: 300px;background: pink;font-size: 20px;}</style><link rel="stylesheet" type="text/css" href="https://abc.css">
</head>

  为了方便,我们可以直接把之前的网页的css样式全部赋值给画中画

// 1. document.styleSheets获取所有的css样式信息
[...document.styleSheets].forEach((styleSheet) => {try {// 转成字符串方便赋值const cssRules = [...styleSheet.cssRules].map((rule) => rule.cssText).join('');// 创建style标签const style = document.createElement('style');// 设置为之前页面中的css信息style.textContent = cssRules;console.log('style', style);// 把style标签放到画中画的<head><head/>标签中pipWindow.document.head.appendChild(style);} catch (e) {// 通过 link 引入样式,如果有跨域,访问styleSheet.cssRules时会报错。没有跨域则不会报错const link = document.createElement('link');/*** rel = stylesheet 导入样式表* type: 对应的格式* media: 媒体查询(如 screen and (max-width: 600px))*  href: 外部样式表的 URL*/link.rel = 'stylesheet';link.type = styleSheet.type;link.media = styleSheet.media;link.href = styleSheet.href ?? '';console.log('error: link', link);pipWindow.document.head.appendChild(link);}
});

演示:

 

3.2. 使用 link 引入外部 CSS 文件

向其他普通html文件一样,可以通过link标签引入特定css文件:

创建 pip.css 文件:

#pipContent {width: 600px;height: 300px;background: skyblue;
}

js引用:

// 其他不变
const link = document.createElement('link');
link.rel = 'stylesheet';
link.href = './pip.css';  // 引入外部 CSS 文件
pipWindow.document.head.appendChild(link);
pipWindow.document.body.appendChild(pipContent);
演示:

 

3.3. 媒体查询的支持

可以设置媒体查询 @media (display-mode: picture-in-picture)。在普通页面中会自动忽略样式,在画中画模式会自动渲染样式

<style>#pipContent {width: 600px;height: 300px;background: pink;font-size: 20px;}<!-- 普通网页中会忽略 -->@media (display-mode: picture-in-picture) {#pipContent {background: lightgreen;}}
</style>

  

在普通页面中显示为粉色,在画中画自动变为浅绿色

演示:

 

4️. 监听进入和退出 PiP 模式的事件

我们还可以为 PiP 窗口 添加事件监听,监控画中画模式的 进入 和 退出。这样,你就可以在用户操作时,做出相应的反馈,比如显示提示或执行其他操作。

// 进入 PIP 事件
documentPictureInPicture.addEventListener("enter", (event) => {console.log("已进入 PIP 窗口");
});const pipWindow = await window.documentPictureInPicture.requestWindow({width: 200,height: 300
});
// 退出 PIP 事件
pipWindow.addEventListener("pagehide", (event) => {console.log("已退出 PIP 窗口");
});

  

演示

 

5️. 监听 PiP 焦点和失焦事件

const pipWindow = await window.documentPictureInPicture.requestWindow({width: 200,height: 300
});pipWindow.addEventListener('focus', () => {console.log("PiP 窗口进入了焦点状态");
});pipWindow.addEventListener('blur', () => {console.log("PiP 窗口失去了焦点");
});

演示

 

6. 克隆节点画中画

我们会发现我们把原始元素传入到PIP窗口后,原来窗口中的元素就不见了。
我们可以把原始元素克隆后再传入给PIP窗口,这样原始窗口中的元素就不会消失了

const pipContent = document.getElementById("pipContent");
const pipWindow = await window.documentPictureInPicture.requestWindow({width: 200,height: 300
});
// 核心代码:pipContent.cloneNode(true)
pipWindow.document.body.appendChild(pipContent.cloneNode(true));

演示

 

PIP 完整示例代码

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document Picture-in-Picture API 示例</title><style>#pipContent {width: 600px;height: 300px;background: pink;font-size: 20px;}</style>
</head>
<body><div id="container"><div id="pipContent">这是一个将要放入画中画的 div 元素!</div><button id="clickBtn">切换画中画</button></div><script>// 检查是否支持 PiP 功能if ('documentPictureInPicture' in window) {console.log("🚀 浏览器支持 PiP 功能!");} else {console.warn("⚠️ 当前浏览器不支持 PiP 功能,更新浏览器或者换台电脑吧!");}// 请求 PiP 窗口document.getElementById("clickBtn").addEventListener("click", async function () {const pipContent = document.getElementById("pipContent");// 请求创建一个 PiP 窗口const pipWindow = await window.documentPictureInPicture.requestWindow({width: 200,  // 设置窗口的宽度height: 300  // 设置窗口的高度});// 将原始元素克隆并添加到 PiP 窗口中pipWindow.document.body.appendChild(pipContent.cloneNode(true));// 设置 PiP 样式同步[...document.styleSheets].forEach((styleSheet) => {try {const cssRules = [...styleSheet.cssRules].map((rule) => rule.cssText).join('');const style = document.createElement('style');style.textContent = cssRules;pipWindow.document.head.appendChild(style);} catch (e) {const link = document.createElement('link');link.rel = 'stylesheet';link.type = styleSheet.type;link.media = styleSheet.media;link.href = styleSheet.href ?? '';pipWindow.document.head.appendChild(link);}});// 监听进入和退出 PiP 模式的事件pipWindow.addEventListener("pagehide", (event) => {console.log("已退出 PIP 窗口");});pipWindow.addEventListener('focus', () => {console.log("PiP 窗口进入了焦点状态");});pipWindow.addEventListener('blur', () => {console.log("PiP 窗口失去了焦点");});});// 关闭 PiP 窗口// pipWindow.close();  // 可以手动调用关闭窗口</script>
</body>
</html>

  

总结

🎉 你现在已经掌握了如何使用 Document Picture-in-Picture API 来悬浮任意 HTML 内容! 希望能带来更灵活的交互体验。✨

如果你有什么问题,或者对 PiP 功能有更多的想法,欢迎在评论区与我讨论!👇📬

本文转载于:https://juejin.cn/post/7441954981342036006

如果对您有所帮助,欢迎您点个关注,我会定时更新技术文档,大家一起讨论学习,一起进步。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/845579.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

洛谷题单指南-线段树-P4513 小白逛公园

原题链接:https://www.luogu.com.cn/problem/P4513 题意解读:给定序列a[n],支持两种操作:1.查询区间[l,r]内的最大子段和 2.将a[x]修改成s,输出其中每一个查询操作的结果。 解题思路:区间问题依然想到线段树,问题主要在于线段树的节点要维护哪些信息: 最直接的,肯定要…

LeetCode 2413[最小偶倍数]

LeetCode 2413[最小偶倍数]题目 链接 LeetCode 2413[最小偶倍数] 详情实例提示题解 思路 判断奇偶性 奇数乘以2并返回 偶数直接返回 代码 class Solution { public:int smallestEvenMultiple(int n) {if (0 == (n % 2))return n;return 2 * n;} };本文来自博客园,作者:EricsT…

power BI

工作一:实现地图 1、先启用地图功能2、导入数据 获取数据更改数据类型检测:加载出现在右侧

使用服务器docker搭建Pwn题目

一、docker的安装 1、安装前先卸载操作系统默认安装的docker sudo apt-get remove docker docker-engine docker.io containerd runc 2、安装必要支持 sudo apt install apt-transport-https ca-certificates curl software-properties-common gnupg lsb-release 3、添加gpg KE…

Windows系统下通过命令行获取进程指标

1.获取当前ProcessID。GetCurrentProcess2.执行cmd或PowerShell cmd:wmic process where "processid=15844" get /format:list PowerShell:Get-Process -id 15844 | Format-List * 作者:快雪 出处:http://www.cnblogs.com/kuaixue/ 本文版权归作者所有,欢迎转…

各层协议

原文链接:点我

技术框架中ORM概念和原理的学习

ORM概念和原理 ORM 概念 我们在介绍 MyBatis 时说到,MyBatis是一种半自动 ORM 实现。那何为 ORM,何为半自动? ORM(Object/Relation Mapping,对象/关系数据库映射)是一种描述对象与关系数据库之间映射的规范。 ORM 作用Java对象和关系数据库如同马和牛,简直是牛头不对马嘴…

在华为开发者空间,基于鲲鹏服务器快速开发打砖块小游戏

本案例将指导开发者如何在鲲鹏服务器部署并运行web小游戏。本文分享自华为云社区《基于鲲鹏服务器的打砖块小游戏部署》,作者: 开发者空间小蜜蜂。 1.1 案例介绍 鲲鹏服务器是基于鲲鹏处理器的新一代数据中心服务器,适用于大数据、分布式存储、高性能计算和数据库等应用。鲲…

5-文件上传漏洞

1、文件上传漏洞原理 1.1 一句话木马 <?php @eval($_POST[xu]); ?> 其中@表示忽略错误,eval()函数表示把传进去的字符串作为php代码执行 从http post里面拿到参数叫xu的value,然后作为代码去执行,并忽略错误 2、Webshell介绍 一句话木马、大马、小马、图片马都是web…

纯血鸿蒙进程加速,混合app开发迎来又一波新机会

对于开发者来说,未来相当一段时间,将来会为鸿蒙,安卓,IOS三个系统同时开发APP上架,其中面对全新的鸿蒙系统,百万APP在未来几年会重新开发上架。 鸿蒙SDK厂商及App开发团队,迎来新的市场机遇。时间不知不觉又来到了2024年最后一个月,又到了辞旧迎新,复盘今年,放眼明年…

实验五 继承和多态

实验一: 代码: publisher.hpp:1 #pragma once2 3 #include<iostream>4 #include<string>5 6 using std::cout;7 using std::endl;8 using std::string;9 10 class Publisher { 11 public: 12 Publisher(const string& s = ""); 13 14 public…

博客园-添加统计图

💖简介 通过WPS在线列表构建博客园每日相关数据统计图。 👉效果📖实现前往WPShttps://www.kdocs.cn/latest新建多维表格 创建表格视图新建仪表盘 新建卡片、折线图卡片配置示例折线图配置示例点击分享获取链接⭐链接配置在config中配置link{icon: <svg t="173313…