User actions
Some WebExtension APIs perform functions that usually occur as a result of a user action. Following the principle of "no surprises", these APIs can only be called from inside the handler for a user action (also referred to as user gestures). These user actions are:
- Clicking the extension's browser action or page action.
- Selecting a context menu item defined by the extension.
- Activating a keyboard shortcut defined by the extension (this is only treated as a user action from Firefox 63).
- Clicking a button on a page bundled with the extension.
- Clicking an extension suggestion in the address bar (omnibox) (this is only treated as a user action from Firefox 142).
The APIs enabled by a user action are:
- The action.openPopup,browserAction.openPopup, andpageAction.openPopupAPIs that open an extension's browser or page action popup. Users do the same by clicking the browser or page action.
- The sidebarAction.open,sidebarAction.close, andsidebarAction.toggleAPIs open and close an extension's sidebar. The user does this from some part of the browser's built-in user interface, such as the View > Sidebar menu.
- The downloads.openAPI opens a downloaded file. The user does this from some part of the browser's built-in user interface, such as the Tools > Downloads menu.
- The management.setEnabledAPI. The user can turn off a theme extension from the extension's Add-on Manager page.
- The permissions.requestAPI. The user can grant permissions from the extension's Add-on Manager permissions and data tab.
For example:
function handleClick() {
  browser.sidebarAction.open();
}
browser.browserAction.onClicked.addListener(handleClick);
In addition to enabling the APIs, these actions also enable the "activeTab" permission. This permission grants extra privileges for the tab visible when the user action takes place.
User interactions in normal web pages aren't treated as user actions. For example, consider a button on a normal web page that uses a content script. This content script added a click handler for the button that sends a message to the extension's background page. When a user clicks the button, the background page message handler is not considered to be handling a user action.
Also, if a user input handler waits on a promise, then its status as a user input handler is lost. For example:
async function handleClick() {
  let result = await someAsyncFunction();
  // this fails, because the handler lost its "user action handler" status
  browser.sidebarAction.open();
}
browser.browserAction.onClicked.addListener(handleClick);