浏览器外的 PWA 会管理自己的窗口。在本章中,您将了解用于在操作系统中管理窗口的 API 和功能。
PWA 窗口
在由 PWA 管理的自有窗口中运行,具有该操作系统中任何窗口的所有优势和责任,例如:
- 在多窗口操作系统(例如 Windows 或 ChromeOS)上调整窗口大小和移动窗口。
- 与其他应用窗口共享屏幕,例如在 iPadOS 分屏模式或 Android 分屏模式下。
- 在桌面设备上的底板、任务栏和 Alt+Tab 菜单中显示,在移动设备上的多任务窗口列表中显示。
- 能够最小化窗口、在屏幕和桌面之间移动窗口,以及随时关闭窗口。
移动和调整窗口大小
在桌面操作系统上,您的 PWA 窗口可以是任意大小,并且可以放置在屏幕上的任意位置。默认情况下,当用户在安装后首次打开 PWA 时,PWA 的默认窗口大小为当前屏幕的百分比,最大分辨率为 1920x1080,位于屏幕的左上角。
用户可以移动和调整窗口大小,浏览器会记住上次的偏好设置,因此当用户下次打开应用时,窗口会保留上次使用时的大小和位置。
您无法在清单中定义 PWA 的首选大小和位置。您只能使用 JavaScript API 调整窗口的位置和大小。您可以在代码中使用 window
对象的 moveTo(x, y)
和 resizeTo(x, y)
函数移动和调整自己的 PWA 窗口的大小。
例如,您可以在 PWA 加载时使用以下方法调整 PWA 窗口的大小并移动该窗口:
document.addEventListener("DOMContentLoaded", event => {
// we can move only if we are not in a browser's tab
isBrowser = matchMedia("(display-mode: browser)").matches;
if (!isBrowser) {
window.moveTo(16, 16);
window.resizeTo(800, 600);
}
});
您可以使用 window.screen
对象查询当前屏幕尺寸和位置;您可以使用 window
对象中的 resize
事件检测窗口何时调整大小。没有用于捕获窗口移动的事件,因此您可以选择频繁查询位置。
浏览其他网站
如果您想将用户定向到超出 PWA 范围的外部网站,可以使用标准 <a href>
HTML 元素、使用 location.href
或在兼容平台上打开新窗口来实现此目的。
目前,在所有浏览器中,如果您已安装 PWA,当您浏览到超出清单范围的网址时,PWA 的浏览器引擎将在窗口上下文中呈现应用内浏览器。
应用内浏览器的一些功能包括:
- 它们会显示在内容上方。
- 它们有一个静态网址栏,用于显示当前来源、窗口标题和菜单。通常,它们采用清单的
theme_color
主题。 - 您可以在上下文菜单中,在浏览器中打开该网址。
- 用户可以关闭浏览器或返回。
授权流程
许多 Web 身份验证和授权流程都涉及将用户重定向到其他来源中的其他网址,以获取将返回到 PWA 来源的令牌,例如使用 OAuth 2.0。
在这些情况下,应用内浏览器会遵循以下流程:
- 用户打开您的 PWA 并点击“登录”。
- 您的 PWA 会将用户重定向到超出 PWA 范围的网址,以便呈现引擎在 PWA 中打开应用内浏览器。
- 用户可以随时取消应用内浏览器并返回您的 PWA。
- 用户登录应用内浏览器。身份验证服务器会将用户重定向到您的 PWA 源,并将令牌作为参数发送。
- 当应用内浏览器检测到网址属于 PWA 的范围时,会自行关闭。
- 引擎会将主要 PWA 窗口导航重定向到身份验证服务器在应用内浏览器中访问的网址。
- 您的 PWA 会获取令牌、存储令牌并呈现 PWA。
强制浏览器导航
如果您想使用网址(而非应用内浏览器)强制打开浏览器,可以使用 <a href>
元素的 _blank
目标。此功能仅适用于桌面版 PWA;在移动设备上,没有使用网址打开浏览器的选项。
function openBrowser(url) {
window.open("url", "_blank", "");
}
打开新窗口
在桌面设备上,用户可以打开同一 PWA 的多个窗口。每个窗口都会是指向同一 start_url
的不同导航,就像您打开了两个指向同一网址的浏览器标签页一样。在 PWA 的菜单中,用户可以依次选择“File”(文件)和“New window”(新窗口),而在 PWA 代码中,您可以使用 open()
函数打开新窗口。如需了解详情,请参阅文档。
function openNewWindow() {
window.open("/", "new-window", "width=600,height=600");
}
在 iOS 或 iPadOS 上的 PWA 窗口中调用 open()
会返回 null
,并且不会打开窗口。在 Android 设备上打开新窗口会为网址创建新的应用内浏览器,即使网址在 PWA 的范围内,通常也不会触发外部浏览体验。
窗口标题
由于浏览器标签页中的空间有限,因此 <title>
元素主要用于 SEO 目的。当您从浏览器切换到 PWA 窗口时,您可以使用标题栏的所有空间。
您可以定义标题栏的内容:
- 在 HTML
<title>
元素中以静态方式。 - 随时动态更改
document.title
字符串属性。
在桌面 PWA 中,标题至关重要,它会显示在窗口的标题栏中,有时还会显示在任务管理器或多任务选择中。如果您使用的是单页应用,则可能需要更新每个路线的标题。
标签页模式
一种名为标签页式模式的实验性功能可让您的 PWA 采用类似于网络浏览器的标签页式设计。在这种情况下,用户可以从同一 PWA 打开多个标签页,但所有标签页都绑定在同一操作系统窗口中,如以下视频所示:
如需详细了解此实验性功能,请参阅 PWA 的标签页式应用模式。
窗口控件叠加层
我们之前提到,您可以通过定义 <title>
元素或 document.title
属性的值来更改窗口的标题。但它始终是字符串值。如果我们可以使用 HTML、CSS 和图片随意设计标题栏,那该多好!
这时,窗口控件叠加层就派上用场了。这是 Microsoft Edge 和 Google Chrome 针对桌面版 PWA 推出的一项全新实验性功能。
如需详细了解此功能,请参阅自定义 PWA 标题栏的窗口控件叠加层。
窗口管理
在多屏设备上,用户会希望使用所有可用的空间。例如:
- 类似于 Gimp 的多窗口图形编辑器可以将各种编辑工具放置在精确定位的窗口中。
- 虚拟交易台可以在多个窗口中显示市场趋势,其中任一窗口都可以以全屏模式查看。
- 幻灯片应用可以在内部主屏幕上显示演讲者备注,并在外部投影仪上显示演示文稿。
借助 Window Management API,PWA 可以做到这一点,还可以执行更多操作。
获取屏幕详情
Window Management API 添加了一个新方法 window.getScreenDetails()
,该方法会返回一个将屏幕作为附加屏幕的不可变数组的对象。还有一个可从 ScreenDetails.currentScreen
访问的实时对象,对应于当前的 window.screen
。
当 screens
数组发生变化时,返回的对象还会触发 screenschange
事件。(更改各个屏幕上的属性时不会出现这种情况。)当各个界面(window.screen
或 screens
数组中的界面)的属性发生变化时,也会触发 change
事件。
// Request an object with a screen objects
const screenDetails = await window.getScreenDetails();
screenDetails.screens[0].isPrimary; // e.g. true
screenDetails.screens[0].isInternal; // e.g. true
screenDetails.screens[0].label; // e.g. 'Samsung Electric Company 28"'
// Access the live object corresponding to the current `window.screen`.
// The object is updated on cross-screen window placements or device changes.
screenDetails.currentScreen;
screenDetails.addEventListener('screenschange', function() {
// NOTE: Does not fire on changes to attributes of individual screens.
const screenCount = screenDetails.screens.length;
const currentScreen screenDetails.currentScreen.id;
});
如果用户或操作系统将 PWA 的窗口从一个屏幕移到另一个屏幕,系统还会从屏幕详情对象触发 currentscreenchange
事件。
屏幕唤醒锁定
假设您在厨房里,按照平板电脑上的食谱做饭。您刚刚准备好了食材。您的双手很脏,因此您又转回设备,查看后续步骤。糟糕!屏幕变黑了!Screen Wake Lock API 可让 PWA 防止屏幕变暗、进入休眠状态或锁定,让用户可以放心地停止、启动、离开和返回。
// Request a screen wake lock
const wakeLock = await navigator.wakeLock.request();
// Listen for wake lock release
wakeLock.addEventListener('release', () => {
console.log(`Screen Wake Lock released: ${wakeLock.released}`);
});
// Manually release the wake lock
wakeLock.release();
虚拟键盘
触控设备(例如手机和平板电脑)提供虚拟屏幕键盘,以便用户在 PWA 的表单元素获得焦点时输入内容。
借助 VirtualKeyboard API,您的 PWA 现在可以使用 navigator.virtualKeyboard
接口在兼容平台上更好地控制键盘,包括:
- 使用
navigator.virtualKeyboard.show()
和navigator.virtualKeyboard.hide()
函数显示和隐藏虚拟键盘。 - 通过将
navigator.virtualKeyboard.overlaysContent
设为true
,告知浏览器您将自行关闭虚拟键盘。 - 通过
navigator.virtualKeyboard
的事件geometrychange
了解键盘何时显示和消失。 - 使用
virtualkeyboardpolicy
HTML 属性在修改宿主元素(使用contenteditable
)时设置虚拟键盘政策。通过政策,您可以决定是让浏览器使用auto
值自动处理虚拟键盘,还是让脚本使用manual
值处理虚拟键盘。 - 使用 CSS 环境变量获取有关虚拟键盘外观的信息,例如
keyboard-inset-height
和keyboard-inset-top
。
如需详细了解此 API,请参阅使用 VirtualKeyboard API 实现完全控制。