This proposal introduces a Multi-Screen Window Placement API enhancement that would allow web applications to initiate a multi-screen experience from a single user activation.
Multi-Screen Window Placement is a new permission-gated web platform API that provides web applications with information about the device's connected screens, allows them to open and position windows or request fullscreen on any of those screens.
The HTML living standard specifies Tracking User Activation, to gate some web platform APIs on user interaction signals. The transient user activation signal (e.g. from a mouse click) is required and consumed when web application scripts open a popup window via Window.open(), and when they request fullscreen via Element.requestFullscreen().
Web applications wishing to offer compelling multi-screen experiences are currently hindered by the single-screen paradigms incorporated in existing algorithms of standard specifications.
A Multi-Screen Window Placement API partner requested the ability to display multi-screen content when the user clicks a button. See Issue 1233970: Window Placement: Pop-ups are blocked when fullscreen is requested.
Specifically, they would like to place fullscreen content on a specific screen and place a new popup window on a separate specific screen, but each of these consume the click’s activation signal, and the second script request is blocked (unless the user grants popups without gestures via a broad ‘Popups and Redirects’ content setting).
The aim of this proposal is enable better experiences for web application users with multiple screens. Here are some use cases that inform the goal below:
- Slideshow app presents fullscreen slides and opens a speaker notes window
- Financial app launches a fullscreen dashboard and a stock tracker window
- Medical app shows a fullscreen x-ray image and opens an image list window
- Creativity app enters a fullscreen preview and opens a separate control window
The goal of this document is to explore tractable and incremental solutions for the stated multi-screen web application use cases, while minimizing any facets of the capability that would be prone to abuse. The following specific goals and proposed solutions make incremental extensions of existing window placement APIs to support initiating a multi-screen experience from a single user gesture.
- Extend
Element.requestFullscreen()
,Window.open()
, andTracking User Activation
algorithms and concepts to conditionally permit a fullscreen request and a popup window request from a single user-activation.
An eventual end-goal (that is not explored here) is to enable scripts to enter
fullscreen on N screens and open M popup windows on M screens from a single user
activation, where N and M are disjoint and potentially empty sets, as long as
the site has the multi-screen window-management
permission.
Allow sites with the window-management
permission to
place fullscreen content on a specific screen
and
place a new popup window on a separate specific screen,
from a single user gesture, when the device has multiple screens and fullscreen
targets a specific screen.
This fulfills the API partner’s use case in the most narrow manner possible, and limits the potential for accidental misuse or abuse with the following set of mitigations:
- Require the window-management permission and devices with multiple displays
- These are needed to show a popup and fullscreen on separate displays
- Allow opening a popup after requesting fullscreen on a specific display
- This ordering prevents basic misuse (showing a popup under fullscreen)
- Pre-existing protections prevent placing popups over fullscreen
- Chrome exits fullscreen (see ForSecurityDropFullscreen)
- Firefox exits fullscreen
- Safari opens the popup in a separate fullscreen space
- Do not automatically activate popups opened from a fullscreen frame
- This leaves input focus on the fullscreen window, for “Press [Esc] to exit”
- Activate popups once the opener exits fullscreen
- This avoids creation of persistent popunders (e.g. Chrome’s PopunderPreventer)
This proposal leverages the flexibility of implementation-specific behavior (blocking popups, postponing window activation), while enabling a path towards defining standardized algorithms. It is partly inspired by existing standardized precedent: requestFullscreen() waives its activation requirement on user generated orientation changes.
Note: the window-management
permission is requested earlier, as needed, by
Window.getScreenDetails()
,
before scripts can request fullscreen (or open a popup) on a specific screen.
So it is expected that scripts making these requests already have permission.
This feature can be used after the site successfully requests detailed screen information by requesting fullscreen on a specific screen of a multi-screen device, and then opening a popup window on another screen of the device, in a single event listener:
initiateMultiScreenExperienceButton.addEventListener('click', async () => {
// Find the primary screen, show some content fullscreen there.
const primaryScreen = screenDetails.screens.find(s => s.isPrimary);
await document.documentElement.requestFullscreen({screen : primaryScreen});
// Find a different screen, fill its available area with a new window.
const otherScreen = screenDetails.screens.find(s => s !== primaryScreen);
window.open(url, '_blank', `left=${otherScreen.availLeft},` +
`top=${otherScreen.availTop},` +
`width=${otherScreen.availWidth},` +
`height=${otherScreen.availHeight}`);
});
This proposal can be brought about as algorithmic changes to existing specs:
- Tracking User Activation
Element.requestFullscreen()
- The rules for choosing a browsing context
- Invoked by
Window.open()
's window open steps
- Invoked by
Loosely, an internal slot can be added to the window context, activated when a fullscreen request is made that targets a specific screen of a multi-screen device, checked and consumed in the rules for choosing a browsing context, when the document requests openining a new popup window.
The Window Management Working Draft Spec incorporates these changes:
- Usage Overview: 1.2.6. Initiate multi-screen experiences
- 3.2.3. Window.open() method definition changes
- 3.5.1. Element.requestFullscreen() method definition changes
There is no obvious way for a web application to detect support for the proposed
feature, in advance of attempting its use. Scripts can attempt to use this
functionality and check the return value of the call to Window.open()
, to
determine whether the window was created successfully or not.
It may be possible to expose signals for this transient token akin to the proposed JS API for querying User Activation.
See the existing algorithm for requesting new browsing contexts, invoked from the window.open steps, which may permit or block popup window requests based on the transient activation signal, and the user agent’s configuration.
This feature enables sites to perform two facets of the Window Management API with a single user activation; i.e. 1.2.4. Place fullscreen content on a specific screen
and 1.2.5. Place windows on a specific screen
are combined to 1.2.6. Initiate multi-screen experiences
. This may exacerbate some documented Window Management Security Considerations.
A notable security consideration is the placement of the popup window in front of, or behind, the fullscreen window, which could be abused by malicious sites to place content in a deceptive or surreptitious fashion. This was a foremost consideration during development of the current proposal, which documents the inherent and added mitigations against accidental misuse or abuse.
Another notable concern is that the user agent's fullscreen message (e.g. Firefox's "<origin> is now full screen [Exit Full Screen (Esc)]" or Chrome's "Press [Esc] to exit full screen") might go unnoticed when the site first places fullscreen content on a less keenly observed screen, especially if the site simultaneously places a companion popup window on a more keenly observed screen with the same transient user activation signal. This exacerbates the ability for a malicious site to spoof the OS, browser, or other sites for phishing attacks. To mitigate this, the user agent could show a similar message when the fullscreen window seems to regain the user's attention; i.e. when it receives an input event (cursor hover, touch down, keyboard event), when the window is activated (or focused or becomes key), when the companion popup window is closed, and/or other events (perhaps only after some time has passed since the window entered fullscreen on that display).
User agents already offer controls to permit any number of script-initiated popups without any transient user activation requirement whatsoever (e.g. Chrome's "chrome://settings/content/popups" and Firefox's "Block pop-up windows" setting). This proposal fulfills web developer requirements to initiate multi-screen experiences, without asking users to grant those existing overly-permissive controls, by introducing a tightly scoped capability enhancement.
This feature does not expose any information to sites, and there are no privacy considerations to note beyond those already documented in the Window Management Privacy Considerations section.
- Permit N
Window.open()
calls on a device with N screens- This doesn't satisfy the goal of also entering fullscreen
- Introduce a new API to support multi-screen fullscreen
- This is a desirable long-term path, but it is not nearly as tractable
- Allow a target-screen fullscreen request after opening a cross-screen popup
- While similar to the active proposal, this has fewer inherent protections against accidental misuse or abuse regarding popunders