From 6cb40c0c6f7bb9fd51009ba1edd3bacdce35afaf Mon Sep 17 00:00:00 2001 From: skyclouds2001 <95597335+skyclouds2001@users.noreply.github.com> Date: Mon, 23 Sep 2024 11:06:41 +0800 Subject: [PATCH] add --- .../using/index.md | 199 ++++++++++++++++++ 1 file changed, 199 insertions(+) create mode 100644 files/zh-cn/web/api/document_picture-in-picture_api/using/index.md diff --git a/files/zh-cn/web/api/document_picture-in-picture_api/using/index.md b/files/zh-cn/web/api/document_picture-in-picture_api/using/index.md new file mode 100644 index 00000000000000..6a45bc1b827697 --- /dev/null +++ b/files/zh-cn/web/api/document_picture-in-picture_api/using/index.md @@ -0,0 +1,199 @@ +--- +title: 使用 Document Picture-in-Picture API +slug: Web/API/Document_Picture-in-Picture_API/Using +l10n: + sourceCommit: d0b23f3f26637aa405ee9ee0a0892fc6e9b742ef +--- + +{{SeeCompatTable}}{{DefaultAPISidebar("Document Picture-in-Picture API")}}{{securecontext_header}} + +本指南提供了 {{domxref("Document Picture-in-Picture API", "Document Picture-in-Picture API", "", "nocode")}} 典型用法的详细指导。 + +> [!NOTE] +> 你可以在 [Document Picture-in-Picture API 示例](https://mdn.github.io/dom-examples/document-picture-in-picture/)中看到特色演示(另请参阅完整的 [源代码](https://github.com/mdn/dom-examples/tree/main/document-picture-in-picture))。 + +## 示例 HTML + +以下 HTML 设置了一个基本的视频播放器。 + +```html +
+

+ 视频播放器当前位于单独的画中画窗口中。 +

+
+ + + + +
+

+ Document Picture-in-Picture API 不可用 +

+ +

+
+
+
+``` + +## 特征检测 + +要检查是否支持 Document Picture-in-Picture API,你可以测试 `window` 上的 `documentPictureInPicture` 属性是否可用: + +```js +if ("documentPictureInPicture" in window) { + document.querySelector(".no-picture-in-picture").remove(); + + const togglePipButton = document.createElement("button"); + togglePipButton.textContent = "切换画中画"; + togglePipButton.addEventListener("click", togglePictureInPicture, false); + + document.getElementById("controlbar").appendChild(togglePipButton); +} +``` + +如果可用,我们将删除“Document Picture-in-Picture API 不可用”消息,并添加 {{htmlelement("button")}} 元素以在文档画中画窗口中打开视频播放器。 + +## 打开画中画窗口 + +以下 JavaScript 调用 {{domxref("DocumentPictureInPicture.requestWindow", "window.documentPictureInPicture.requestWindow()")}} 打开一个空白的画中画窗口。返回的 {{jsxref("Promise")}} 用画中画 {{domxref("Window")}} 对象实现。使用 {{domxref("Element.append()")}} 将视频播放器移动到该窗口,然后我们显示消息通知用户它已被移动。 + +`requestWindow()` 的 `width` 和 `height` 选项将画中画窗口设置为所需大小。如果选项值太大或太小而无法适应用户友好的窗口大小,浏览器可能会限制选项值。 + +```js +async function togglePictureInPicture() { + 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); + } +}); + +// ... +``` + +## 画中画模式下目标样式 + +{{cssxref("@media/display-mode", "display-mode")}} [媒体特性](/zh-CN/docs/Web/CSS/@media#媒体特性) 的 `picture-in-picture` 值允许开发人员根据文档是否以画中画模式显示来将 CSS 应用于文档。基本用法如下: + +```css +@media (display-mode: picture-in-picture) { + body { + background: red; + } +} +``` + +仅当以画中画模式显示时,此代码片段才会将文档 `` 的背景变为红色。 + +在[我们的演示](https://mdn.github.io/dom-examples/document-picture-in-picture/)中,我们将 `display-mode: picture-in-picture` 值与 {{cssxref("@media/prefers-color-scheme", "prefers-color-scheme")}} 媒体功能相结合,以根据用户的配色方案偏好创建仅当应用以画中画模式显示时才应用的明暗配色方案。 + +```css +@media (display-mode: picture-in-picture) and (prefers-color-scheme: light) { + body { + background: antiquewhite; + } +} + +@media (display-mode: picture-in-picture) and (prefers-color-scheme: dark) { + body { + background: #333; + } + + a { + color: antiquewhite; + } +} +``` + +## 处理画中画窗口关闭时的情况 + +当第二次按下按钮时,切换画中画窗口再次关闭的代码如下所示: + +```js +inPipMessage.style.display = "none"; +playerContainer.append(videoPlayer); +window.documentPictureInPicture.window.close(); +``` + +在这里,我们恢复 DOM 更改——隐藏消息并将视频播放器放回主应用窗口中的播放器容器中。我们还使用 {{domxref("Window.close()")}} 方法以编程方式关闭画中画窗口。 + +但是,你还需要考虑用户通过按下窗口本身上浏览器提供的关闭(X)按钮来关闭画中画窗口的情况。你可以通过使用 [`pagehide`](/zh-CN/docs/Web/API/Window/pagehide_event) 事件检测窗口何时关闭来处理这种情况: + +```js +pipWindow.addEventListener("pagehide", (event) => { + inPipMessage.style.display = "none"; + playerContainer.append(videoPlayer); +}); +``` + +## 当网站进入画中画模式时监听 + +监听 `DocumentPictureInPicture` 实例上的 {{domxref("DocumentPictureInPicture.enter_event", "enter")}} 事件,了解画中画窗口何时打开。 + +在我们的演示中,我们使用 `enter` 事件向画中画窗口添加静音切换按钮: + +```js +documentPictureInPicture.addEventListener("enter", (event) => { + const pipWindow = event.window; + console.log("视频播放器已进入 pip 窗口"); + + const pipMuteButton = pipWindow.document.createElement("button"); + pipMuteButton.textContent = "已静音"; + pipMuteButton.addEventListener("click", () => { + const pipVideo = pipWindow.document.querySelector("#video"); + if (!pipVideo.muted) { + pipVideo.muted = true; + pipMuteButton.textContent = "未静音"; + } else { + pipVideo.muted = false; + pipMuteButton.textContent = "已静音"; + } + }); + + pipWindow.document.body.append(pipMuteButton); +}); +``` + +> [!NOTE] +> {{domxref("DocumentPictureInPictureEvent")}} 事件对象包含一个 `window` 属性,用于访问画中画窗口。 + +## 访问元素并处理事件 + +你可以通过多种不同的方式访问画中画窗口中的元素: + +- {{domxref("DocumentPictureInPicture.requestWindow()")}} 方法返回的 {{domxref("Window")}} 实例,如上所示。 +- 通过 {{domxref("DocumentPictureInPictureEvent")}} 事件对象的 `window` 属性(在 {{domxref("DocumentPictureInPicture.enter_event", "enter")}} 事件上),如上所示。 +- 通过 {{domxref("DocumentPictureInPicture.window")}} 属性: + +```js +const pipWindow = window.documentPictureInPicture.window; +if (pipWindow) { + // 使画中画窗口中播放的视频静音。 + const pipVideo = pipWindow.document.querySelector("#video"); + pipVideo.muted = true; +} +``` + +一旦获得了对画中画 `window` 实例的引用,你就可以操作 DOM(例如创建按钮)并响应用户输入事件(例如 [`click`](/zh-CN/docs/Web/API/Element/click_event) 事件),就像在常规浏览器窗口上下文中正常执行一样。