Tabbed application mode for PWAs

Work on more than one document at a time with tabs in your Progressive Web App

In the world of computing, the desktop metaphor is an interface metaphor that is a set of unifying concepts used by graphical user interfaces (GUI) to help users interact more easily with the computer. In keeping with the desktop metaphor, GUI tabs are modeled after traditional card tabs inserted in books, paper files, or card indexes. A tabbed document interface (TDI) or tab is a graphical control element that allows multiple documents or panels to be contained within a single window, using tabs as a navigational widget for switching between sets of documents.

Progressive Web Apps can run in various display modes determined by the display property in the web app manifest. Examples are fullscreen, standalone, minimal-ui, and browser. These display modes follow a well-defined fallback chain ("fullscreen""standalone""minimal-ui""browser"). If a browser does not support a given mode, it falls back to the next display mode in the chain. Via the "display_override" property, developers can specify their own fallback chain if they need to.

What is tabbed application mode

Something that has been missing from the platform so far is a way to let PWA developers offer their users a tabbed document interface, for example, to enable editing different files in the same PWA window. Tabbed application mode closes this gap.

Suggested use cases for tabbed application mode

Examples of sites that may use tabbed application mode include:

  • Productivity apps that let the user edit more than one document (or file) at the same time.
  • Communication apps that let the user have conversations in different rooms per tab.
  • Reading apps that open article links in new in-app tabs.

Differences to developer-built tabs

Having documents in separate browser tabs comes with resource isolation for free, which is not possible using the web today. Developer-built tabs would not scale acceptably to hundreds of tabs like browser tabs do. Browser affordances such as navigation history, "Copy this page URL", "Cast this tab" or "Open this page in a web browser" would be applied to the developer-built tabbed interface page, but not the selected document page.

Differences to "display": "browser"

The current "display": "browser" already has a specific meaning:

Opens the web application using the platform-specific convention for opening hyperlinks in the user agent (e.g., in a browser tab or a new window).

While browsers can do whatever they want regarding UI, it would clearly be a pretty big subversion of developer expectations if "display": "browser" suddenly meant "run in a separate application-specific window with no browser affordances, but a tabbed document interface".

Setting "display": "browser" is effectively the way you opt out of being put into an application window.

Current status

Step Status
1. Create explainer Completed
2. Create initial draft of specification Not started
3. Gather feedback & iterate on design In progress
4. Origin trial In progress
5. Launch Not started

Using tabbed application mode

To use tabbed application mode, developers need to opt their apps in by setting a specific "display_override" mode value in the web app manifest.

{
  "display": "standalone",
  "display_override": ["tabbed"]
}

Next, the property "tab_strip" can optionally be used to fine-tune the tab behavior. It has two allowed sub-properties, "home_tab" and "new_tab_button". If the "tab_strip" property is not present, the particular properties' "auto" values are used. The browser determines what values to use for "auto".

Home tab

The home tab is a pinned tab that, if enabled for an app, should always be present when the app is open. This tab should never navigate. Links clicked from this tab should open in a new app tab. Apps can choose to customize the URL this tab is locked to and the icon displayed on the tab.

The allowed values for "home_tab" are:

  • "auto" to let the browser determine what to do.
  • "absent" to tell the browser to not show a home tab.
  • An object with two sub-properties:
    • "url" with the allowed values of "auto" or a URL to lock the home tab to.
    • "icons" with the allowed values of "auto" or an array of icons as in the main "icons" property.

New tab button

The new tab button, if present, should open a new tab at a URL that is within the scope of the app. The app may choose to set a custom URL and icon for this button. Browsers can decide how to handle dragging these tabs around to create new windows or combine with browser tabs.

The allowed values for "new_tab_button" are:

  • "auto" to let the browser determine what to do.
  • "absent" to tell the browser to not show a new tab button.
  • An object with two sub-properties:
    • "url" with the allowed values of "auto" or an in-scope URL to open new tabs on.
    • "icons" with the allowed values of "auto" or an array of icons as in the main "icons" property.

Complete example

A complete example to configure the behavior of an app with a tabbed interface may look as follows:

{
  "display_override": ["tabbed"],
  "tab_strip": {
    "home_tab": {
      "url": "./home/",
      "icons": [
        {
          "src": "./home.svg",
          "sizes": "any",
          "type": "image/svg+xml"
        }
      ]
    },
    "new_tab_button": {
      "url": "./new-tab/",
      "icons": [
        {
          "src": "./new-tab.svg",
          "sizes": "any",
          "type": "image/svg+xml"
        }
      ]
    }
  }
}

Detecting tabbed application mode

Apps can detect whether they are running in tabbed application mode by checking the display-mode CSS media feature in either CSS or JavaScript:

@media (display-mode: tabbed) {
  /* Styles to apply in tabbed application mode. */
}
const tabbedApplicationModeEnabled = window.matchMedia('(display-mode: tabbed)').matches;

Interaction with the Launch Handler API

The Launch Handler API lets sites redirect app launches into existing app windows to prevent duplicate windows from being opened. When a tabbed app sets "client_mode": "navigate-new", app launches will open a new tab in an existing app window.

Demo

You can try tabbed application mode by setting a browser flag:

  1. Set the #enable-desktop-pwas-tab-strip flag.
  2. Install the app tabbed-application-mode.glitch.me (source code).
  3. Click the different links on the different tabs.

Screenshot of the tabbed application mode demo at tabbed-application-mode.glitch.me.

Feedback

The Chrome team wants to hear about your experiences with tabbed application mode.

Tell us about the API design

Is there something about tabbed application mode that does not work like you expected? Comment on the web app manifest Issue that we have created.

Report a problem with the implementation

Did you find a bug with Chrome's implementation? File a bug at new.crbug.com. Be sure to include as much detail as you can, simple instructions for reproducing, and enter UI>Browser>WebAppInstalls in the Components box. Glitch works great for sharing quick and easy reproduction cases.

Show support for the API

Are you planning to use tabbed application mode? Your public support helps the Chrome team prioritize features and shows other browser vendors how critical it is to support them.

Send a tweet to @ChromiumDev using the hashtag #TabbedApplicationMode and let us know where and how you are using it.

Acknowledgements

Tabbed application mode was explored by Matt Giuca. The experimental implementation in Chrome was the work of Alan Cutter. This article was reviewed by Joe Medley. Hero image by Till Niermann on Wikimedia Commons.