Skip to content

Latest commit

 

History

History
259 lines (185 loc) · 10.1 KB

explainer.md

File metadata and controls

259 lines (185 loc) · 10.1 KB

Shortcuts Explainer

Authors:

Introduction

A number of operating systems offer a means by which authors add menu items to the app launcher icon itself. These provide quick access to key tasks for an app. Typically, these are exposed via a right click or similar context menu-triggering action. The list of shortcut actions often contains a mixture of static and dynamic shortcuts.1 Generally these number around 4–5, depending on the platform—or in the case of Android, the launcher—used. Adding these to the web platform provides an equivalent capability to native apps on many platforms.

Why are Shortcuts needed?

Shortcuts will provide another mechanism for websites (especially progressive web apps) to facilitate re-engagement with key tasks.

Goals for v1

  • Enable authors to provide a list of shortcuts to specific URLs within their website in a standardized fashion
  • Enable users to access key tasks easily

Non-goals for v1

  • Enable authors to access or mutate Shortcuts via JavaScript
  • Enable authors to directly trigger JavaScript events from Shortcuts

Use Cases

Most websites (and, by extension, progressive web apps) have a handful of key tasks that users need to do with relative frequency. Providing quick access to those tasks from anywhere the app icon is displayed will improve users’ productivity as well as increase their engagement with the app. It will also reduce the number of clicks needed to access numerous locations and features including, but not limited to:

  • Top-level navigation items (e.g., home, timeline, orders),
  • Search,
  • Data entry tasks (e.g., compose an email or tweet, add a receipt), and
  • Activities (e.g., start a chat with the 4 most popular contacts in your contacts).

Examples

PlayerFM "Shortcuts" in Android

This menu provides access to the following app functionality:

  • Play Later - a list of podcasts the user flagged for later listening;
  • Subscriptions - the user’s entire subscription catalog;
  • Search; and
  • Discover - browse available & promoted podcasts.

Slack’s "Dock Menu" in macOS

This menu provides a list of Slack “workspaces” the user is signed into, followed by a link to “Sign into another workspace.”

Twitter’s "Jump List" in Windows 10

This menu provides direct access to:

  • Compose a new tweet;
  • Search (a.k.a. Explore);
  • Access the user’s notifications; and
  • Access the user’s direct messages

Customer validation

  • Twitter is actively using Windows’ "Jump Lists" feature in their PWA via WinRT (see the above example).
  • Starbucks has also expressed interest in increasing re-engagement with their PWA by providing quick access to options like:
    • Pay with your Starbucks card,
    • Place an order, and
    • Find the closest store.

API Proposal

We are proposing that a new member be added to the Web App Manifest: shortcuts.

dictionary WebAppManifest {
  …
  sequence<ShortcutInfo> shortcuts;
}

This member would take an array of ShortcutInfo objects defined thusly:

dictionary ShortcutInfo {
  DOMString name;
  DOMString short_name;
  DOMString description;
  sequence icons;
  dictionary data;
}

name

Provides a human-readable label for the shortcut action when displayed to the user.

short_name

Optional. Provides an abbreviated, human-readable label for the shortcut action. It is intended to be used where there is insufficient space to display the name. In its absense, systems looking for this key should fall back to name.

description

Optional. Provides the shortcut action’s purpose.

uri

The URL that loads when a user activates the shortcut. This URL must exist within the navigation scope (scope) defined in the manifest. If the uri is a relative URL, the base URL will be the URL of the manifest.

icons

The path to one or more ImageResources. If the ImageResource’s src is a relative URL, the base URL will be the URL of the manifest.

data

Optional. Configuration data to be passed into the target url as part of a query string.

Example

To replicate the shortcut menu from PlayerFM, an author could do the following:

{
  "name": "PlayerFM",
  "start_url": "https://app.playerfm.com",
  
  "shortcuts": [
    {
      "name": "Play Later",
      "description": "View the list of podcasts you saved for later",
      "uri": "/play-later",
      "icons": [
        {
          "src": "/icons/play-later.svg",
          "type": "image/svg+xml",
          "purpose": "any"
        }
      ]
    },
    {
      "name": "Subscriptions",
      "description": "View the list of podcasts you listen to",
      "uri": "/subscriptions",
      "icons": [
        {
          "src": "/icons/subscriptions.svg",
          "type": "image/svg+xml",
          "purpose": "any"
        }
      ]
    },
    {
      "name": "Search",
      "description": "Search for new podcasts to listen to",
      "uri": "/search",
      "icons": [
        {
          "src": "/icons/search.svg",
          "type": "image/svg+xml",
          "purpose": "any"
        }
      ]
    },
    {
      "name": "Discover",
      "description": "Browse for new podcasts to listen to",
      "uri": "/discover",
      "icons": [
        {
          "src": "/icons/discover.svg",
          "type": "image/svg+xml",
          "purpose": "any"
        }
      ]
    }  
  ]
}

Responding to shortcut activation

There is no need to add any special handler for the shortcut aside from having a URL (which is part of the general site architecture) to respond to the action.

Depending on how the site is built and its needs, the author might choose to respond to these shortcuts using the proposed ServiceWorker Launch Event:

self.addEventListener('launch', event => {
  // Only "/search" is handled specially. All other URLs navigate as normal.
  if (event.url !== new URL('search', self.registration.scope).href) {
    return;
  }

  event.handleLaunch((async () => {
    // Get an existing client.
    const client = (await clients.matchAll())[0];

    if (client) {
      client.focus();
      client.postMessage('search');
    } else {
      clients.openWindow(event.url);
    }
  })());
});

Existing Implementations

This section describes possible treatment on each major OS. User agents are free to implement however they like, but this should give an idea of what the API will look like in practice. In all cases, the menu items consist of text, a destination, and an icon.

Android

In Android, shortcut activation is triggered via a long press. Android allows the static shortcut list to be supplemented at runtime via Dynamic Shortcuts and allows users to rearrange and "pin" shortcut items. (Documentation)

iOS

In iOS, a hard press (3D Touch) exposes the application shortcuts (iOS 9.0+ on 3D touch-capable devices). iOS also enables a subtitle and has a hard limit of 4 items. (Documentation)

macOS/Windows/Linux GUI/etc.

In most OSes, this context menu is triggered by right clicking (or two finger tapping) on the icon. Windows calls these "Jump Lists" and allows each shortcut item to have a description (Documentation). In macOS these shortcuts are referred to as the "Dock Menu" (Documentation).

Electron

Electron enables shortcuts for macOS and Windows app icons as well. (Documentation)

Open Questions

  1. Should the icons member in the Shortcut object be optional?
  2. Given that color schemes are becoming a thing in CSS, it seems like the icon options should somehow support an indication of whether the icon itself is intended for use in a dark or light theme. Assuming there is interest in that, would it make sense to piggyback on the existing purpose member of an ImageResource or should there be a new member (e.g., color_scheme) added to it? We’ve drafted an addition to the ImageResource dictionary for this purpose.
  3. For more advanced use cases like minimal media player controls (e.g., Spotify, Netflix, etc.), should we provide options for custom shortcut layouts or control groups and JavaScript interaction with the Shortcuts menu? Should that be part of this spec or defined separately?
  4. Should we include a hard limit on Shortcuts? If no, should we allow for collections?
  5. Does the recommendation for per-language manifest files still stand? Should we look to enable multi-lingual supprot within a manifest as more text keys get added?

Considerations for v2

  • Enable authors to access and/or mutate Shortcuts via JavaScript (navigator.shortcuts?).
  • Enable authors to directly trigger JavaScript events from Shortcuts.
  • Enable a ServiceWorker to listen for and respond to Shortcut events (shortcutclick?).
  • Allow shortcuts to be aggregated into multiple sections.
  • Allow shortcuts to be designated for different contexts (e.g., all apps list vs. home screen/start menu vs. task bar).

1 In a survey of Android applications, static items are by far the most common.