A PWA outside of the browser manages its own window. Learn about the APIs and capabilities for managing a window within the operating system.
Running in your own window, managed by your PWA, has all the advantages and responsibilities of any window in that operating system, such as:
- The ability to move and resize the window in multi-window operating systems, such as Windows or ChromeOS.
- Sharing the screen with other app windows, as in iPadOS split mode or Android split-screen mode.
- Appearing in docks, taskbars, and in the alt-tab menu on desktop, and multi-task window lists on mobile devices.
- The ability to minimize, move the window across screens and desktops, and close it at any time.
Move and resize the window
Your PWA window can be of any size and positioned anywhere on the screen on desktop operating systems. By default, when the user opens the PWA for the first time after installation, the PWA gets a default window size of a percentage of the current screen, with a maximum resolution of 1920x1080 positioned at the top-left corner of the screen.
The user can move and resize the window, and the browser will remember the last preference. The next time the user opens the app, the window will retain the size and position from the previous usage.
There is no way to define your PWA's preferred size and position within the
manifest. You can only reposition and resize the window using the JavaScript
API. From your code, you can move and resize your own PWA window using the
moveTo(x, y)
and
resizeTo(x, y)
functions of the window
object.
For example, you can resize and move your PWA window when the PWA loads using:
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);
}
});
You can query the current screen size and position using the window.screen
object; you can detect when the window is resized using the resize
event from
the window
object. There is no event for capturing the window move, so your
option is to query the position frequently.
Instead of moving and resizing the window absolutely, you can move relatively
and resize using moveBy()
and resizeBy()
.
Browse other sites
If you want to send the user to an external site that is out-of-scope for your
PWA, you can do so with a standard <a href>
HTML element. Use location.href
or open a new window on compatible platforms.
When you visit a URL that is out of the scope of your manifest, the browser engine of your PWA renders an in-app browser within the context of your window.
Some features of the in-app browsers are:
- They appear on top of your content.
- They have a static address bar showing the current origin, the window's title,
and a menu. Typically, they're themed with the
theme_color
of your manifest. - From the contextual menu, you can open that URL in the browser.
- Users can close the browser or go back.
While the in-app browser is on the screen, your PWA is waiting in the background, as if another window is obscuring it.
![](https://web.developers.google.cn/static/learn/pwa/windows/image/an-app-browser-an-iphon-dddc327f709aa.png?authuser=1)
![](https://web.developers.google.cn/static/learn/pwa/windows/image/an-app-browser-android-3d73f1c909f0b.png?authuser=1)
Authorization flows
Many web authentication and authorization flows require redirecting the user to a different URL on a different origin to acquire a token that returns to your PWA's origin, such as with OAuth 2.0.
In these cases, the in-app browser follows this process:
- The user opens your PWA and clicks login.
- Your PWA redirects the user to a URL that is out of the scope of the PWA so that the rendering engine opens an in-app browser within your PWA.
- The user can cancel the in-app browser and go back to your PWA at any time.
- The user logs into the in-app browser. The authentication server redirects the user to your PWA origin, sending the token as an argument.
- The in-app browser closes itself when it detects a URL that is part of the scope of the PWA.
- The engine redirects the main PWA window navigation to the URL that the authentication server went to while in the in-app browser.
- Your PWA gets the token, stores the token, and renders the PWA.
Force a browser's navigation
If you want to force open the browser with a URL and not an in-app browser, you
can use the _blank
target of <a href>
elements. This works only on desktop
PWAs. On mobile devices, there's no option to open a browser with a URL.
function openBrowser(url) {
window.open("url", "_blank", "");
}
Open new windows
On desktop, users can open more than one window of the same PWA. Each window has
a different navigation for the same start_url
, as if you were opening two
browser tabs of the same URL.
From the menu in the PWA, users can select File then New window. From
your PWA code, you can open a new window with the
open()
function.
function openNewWindow() {
window.open("/", "new-window", "width=600,height=600");
}
When you call open()
within a PWA window on iOS or iPadOS, it returns null
and doesn't open a window. Opening new windows on Android creates a new in-app
browser for the URL, even if the URL is within the scope of your PWA, that
typically doesn't trigger an external browsing experience.
Window title
The <title>
element was primarily used for SEO purposes as the space within a browser tab is limited. When you move from the browser to your window in a PWA, all the title bar space is available to you.
You can define the contents of the title bar:
- Statically in your HTML
<title>
element. - Dynamically changing the
document.title
string property at any time.
On desktop PWAs, the title is essential, and it's used in the title bar of the window and sometimes in the task manager or multi-task selection. If you have a single-page application, you may want to update your title on every route.
Tabbed mode
Tabbed mode is an experimental capability that lets your PWA have a tab-based design, similar to a web browser. In this case, the user can open several tabs in the same PWA, but all tied together in the same operating system window.
You can read more about this experimental capability at Tabbed application mode for PWA.
Window controls overlay
We've mentioned that you can change the window's title by defining the value of
the <title>
element or the document.title
property. But it's always a string
value. What if we could design the title bar with HTML, CSS, and images?
Window Controls Overlay, an experimental capability in Microsoft Edge and Google
Chrome for desktop PWAs, may help.
You can read more about this capability at Customize the window controls overlay of your PWA's title bar.
Window management
With multiple screens, users want to use all the space available to them. For example:
- Multi-window graphics editors, such as Gimp, can place various editing tools in accurately positioned windows.
- Virtual trading desks can show market trends in multiple windows any of which can be viewed in fullscreen mode.
- Slideshow apps can show speaker notes on the internal primary screen and the presentation on an external projector.
The Window Management API allows PWAs to do just that and more.
Getting screen details
The Window Management API adds a new method, window.getScreenDetails()
, that returns an object with screens as an immutable array of attached screens. There's also a live object accessible from ScreenDetails.currentScreen
, corresponding to the current window.screen
.
The returned object also fires a screenschange
event when the screens
array changes. (This does not happen when attributes on individual screens are changed.) Individual screens, either window.screen
or a screen in the screens
array, also fire a change
event when their attributes 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;
});
If the user or the operating system moves your PWA's window from one screen to
another, a currentscreenchange
event is also fired from the screen
details object.
Screen wake lock
Imagine this. You're in the kitchen following a recipe on your tablet. You've just finished prepping your ingredients. Your hands are a mess, and you turn back to your device to read the next step. Disaster! The screen's gone black! The Screen Wake Lock API is here for you and lets a PWA prevent screens from dimming, sleeping, or locking, allowing users to stop, start, leave, and return without worry.
// 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();
Virtual keyboard
Touch-based devices, such as phones and tablets, offer a virtual on-screen keyboard so the user can type when form elements of your PWA are in focus.
With the VirtualKeyboard API, your PWA can have
more control of the keyboard on compatible platforms using the
navigator.virtualKeyboard
interface.
- Show and hide the virtual keyboard with
navigator.virtualKeyboard.show()
andnavigator.virtualKeyboard.hide()
. - Tell the browser that you are closing the virtual keyboard yourself by setting
navigator.virtualKeyboard.overlaysContent
equal totrue
. - Know when the keyboard appears and disappears with the
geometrychange
event. - Set the virtual keyboard policy on editing host elements by setting
contenteditable
toauto
ormanual
, with thevirtualkeyboardpolicy
HTML attribute. A policy lets you decide if you want the virtual keyboard to be handled automatically by the browser (auto
) or handled by your script (manual
). - Use CSS environmental variables to get information about the virtual
keyboard appearance, such as
keyboard-inset-height
andkeyboard-inset-top
.
Read more about this API in Full control with the VirtualKeyboard API.