Spotify 是全球最受欢迎的音频在线播放订阅服务,致力于不断改进用户消费音频和视频内容的方式。它拥有丰富的音乐、播客和有声读物库,每天在移动设备、PC 和其他平台上为数百万用户提供服务。
Spotify 最近为其桌面版和网页版播放器客户端发布了 Spotify Miniplayer。迷你播放器旨在提供一款始终显示在顶部的紧凑小窗口,以便用户随时随地访问 Spotify 的必要播放控件。用户一直在呼吁推出此功能,借助此功能,用户可以在不同的窗口和应用中无缝处理多任务,同时在 Spotify 上欣赏喜爱的音乐人、播放列表和播客。
下面将详细介绍迷你播放器的开发历程,从最初的“画布黑客”到基于新的 Document Picture-in-Picture API 构建的更高级且人性化的版本。
“画布黑客”
迷你播放器的初始迭代版本于 2019 年作为一项黑客项目在 Spotify 的 Web 播放器上发布。目标是使用浏览器的适用于 <video> 的画中画 (PiP) API,在始终位于顶部的窗口中显示专辑封面。不过,此 API 主要用于视频元素,无法显示专辑封面图片。Spotify 通过将专辑封面呈现到画布元素并使用 HTMLCanvasElement
captureStream()
方法获取实时 MediaStream 对象来规避此问题。然后,此流将用作用于 PiP API 的视频源。此方法基于 Google Chrome 的“音频播放列表”示例。
Spotify 将画布与 Media Session API 中设置的适当操作处理脚本结合使用,以控制画中画窗口中会显示哪些播放器控件。这样,用户就可以在一个浮动窗口中看到专辑封面和播放器控件,并在专注于其他任务时控制播放。
这样,Spotify 就可以拥有一个基本的迷你播放器。不过,这种方法存在以下几项限制:
- 不支持在 PiP 窗口中显示视频字幕。由于 Spotify 必须为所有视频显示字幕,因此在视频开始播放后,他们不得不立即关闭 PiP 窗口。
- 只有在本地播放时,播放器控件才会显示。Spotify 允许使用 Spotify Connect(和其他协议)进行远程播放,并希望用户也能控制此类播放
- 不支持自定义 PIP 窗口的外观和风格。Spotify 只能显示海报图片并使用 Chrome 提供的播放器控件,无法添加 Spotify 品牌或其他播放器控件。
由于无法控制界面,也无法在此处添加 Spotify 专用功能(例如为曲目点赞),因此他们认为这种方法不适合他们的桌面客户端。
文档画中画:迷你播放器的演变
2023 年初,Spotify 得知 Google Chrome 重新开始关注推出一项新 API,该 API 允许在 PiP 窗口中显示任意 HTML 内容,称为 Document Picture-in-Picture API。这对 Spotify 来说是一个令人兴奋的进展,因为这将使他们能够完全控制 PiP 窗口的外观。Spotify 在 Chrome 团队的源代码试用期间与其合作,开发了基于 Document Picture-in-Picture API 的全新迷你播放器。
借助 Document PiP API,您可以打开一个新的始终置顶窗口,并在其中附加元素。由于 Spotify Web 播放器是 React Web 应用,因此 Spotify 使用了 ReactDOM 的 createPortal()
方法,将自定义组件从主要应用渲染到 PiP 窗口,从而完全控制迷你播放器的外观和功能。
新的 Document Picture-in-Picture API 还解决了 Spotify 之前存在的问题:
- 画中画窗口中的视频是常规视频元素,完全支持字幕。
- 通过完全控制界面,即使使用 Spotify Connect 远程播放,也可以显示播放器控件。
- Spotify 能够融入自己的外观和风格以及播放器控件,从而提升用户体验。
- 他们能够为 Spotify 的桌面客户端引入对 Document PiP API 的支持,从而将迷你播放器引入到数百万桌面用户手中。
使用 React 创建画中画窗口
以下示例演示了如何在 React 中使用文档画中画功能,就像 Spotify 团队所做的那样。您将创建两个 React 组件:MyFeature
和 PiPContainer
。
MyFeature
组件负责管理画中画窗口。它会渲染一个用于切换画中画窗口的按钮,并渲染 PiPContainer
组件。它还会订阅画中画窗口的 "pagehide"
事件,以便在窗口关闭时更新状态。
const MyFeature = () => {
const [pipWindow, setPiPWindow] = useState<Window | null>(
documentPictureInPicture.window
);
const handleClick = useCallback(async () => {
if (pipWindow) {
pipWindow.close();
} else {
const newWindow = await documentPictureInPicture.requestWindow();
setPiPWindow(newWindow);
}
}, [pipWindow]);
useEffect(() => {
const handleWindowClose = (): void => {
setPiPWindow(null);
};
pipWindow?.addEventListener("pagehide", handleWindowClose);
return () => {
pipWindow?.removeEventListener("pagehide", handleWindowClose);
};
}, [pipWindow]);
return (
<>
<button onClick={handleClick}>
{pipWindow ? "Close PiP Window" : "Open PiP Window"}
</button>
<PiPContainer pipWindow={pipWindow}>Hello World 👋!</PiPContainer>
</>
);
};
PiPContainer
组件使用 ReactDOM 的 createPortal() 方法将内容呈现到画中画窗口中。
type Props = PropsWithChildren<{
pipWindow: Window | null;
}>;
const PiPContainer = ({ pipWindow, children }: Props) => {
useEffect(() => {
if (pipWindow) {
cloneStyles(window.document, pipWindow.document);
}
}, [pipWindow]);
return pipWindow ? createPortal(children, pipWindow.document.body) : null;
};
后续步骤
随着 Spotify 不断进步和创新,他们将继续致力于改进迷你播放器,并计划进一步优化其功能和用户体验。虽然他们还无法承诺会推出哪些具体功能,但对迷你播放器的未来发展充满期待。
Document Picture-in-Picture API 提供了灵活性和控制功能,可让您创建更直观、更易用的迷你播放器。我们希望其他浏览器供应商能注意到此 API 提供的机会,并考虑添加对此 API 的支持。这样一来,无论用户选择哪种浏览器,Spotify 都能为其提供一致且更出色的体验。
致谢
感谢 Spotify 中参与构建迷你播放器的所有人。
Spotify 还想感谢 Google Chrome 团队的协作,以及在 Document Picture-in-Picture API 方面采纳了 Spotify 的反馈。