Working with the Tabs API
选项卡允许用户在其浏览器窗口中打开多个网页,然后在这些网页之间切换。使用 Tabs API,你可以使用和操作这些选项卡来创建实用程序,为用户提供使用选项卡或提供扩展功能的新方法。
在这篇 how-to 文章中我们将看到:
- 使用 Tabs API 所需的权限。
- 使用
tabs.query
发现有关标签及其属性的更多信息。 - 创建、复制、移动、更新、重新加载和删除选项卡。
- 操纵标签的缩放大小。
- 操纵选项卡的 CSS。
最后,我们通过查看 API 提供的其他一些其他功能。
备注: 其他地方有一些 Tab API 功能。这些是可用于使用脚本操作选项卡内容的方法tabs.connect
(en-US),tabs.sendMessage
(en-US)和tabs.executeScript
(en-US)。如果你需要有关这些方法的更多信息,请参阅概念文章内容脚本和操作指南修改网页。
权限和选项卡 API
对于大多数 Tabs API 函数,你不需要任何权限; 但是,有一些例外:
- 需要
"tabs"
权限才能访问 Tab 对象的Tab.url
,Tab.title
和Tab.favIconUrl
属性。在 Firefox 中,你还需要"tabs"
来按 URL 执行查询。 tabs.executeScript
(en-US)或tabs.insertCSS
需要主机权限。
以下是你可以在扩展程序的 manifest.json 文件中请求"tabs"
权限的方法:
json
"permissions": [
"<all_urls>",
"tabs"
],
此请求允许你在用户访问的所有网站上使用所有标签 API 功能。还有一种替代方法可以请求权限使用tabs.executeScript
(en-US)或tabs.insertCSS
,其中你不需要主机权限,形式为"activeTab"
。此权限与<all_urls>
提供与"tabs"
相同的权限,但有两个限制:
- 用户必须通过其浏览器或页面操作,上下文菜单或快捷键与扩展进行交互。
- 它仅在活动选项卡中授予权限。
这种方法的好处是用户不会收到权限警告,也就是说你的扩展程序可以“访问所有网站的数据”。这是因为<all_urls>
权限使扩展能够在任何选项卡中随时执行脚本,而"activeTab"
仅限于允许扩展在当前选项卡中执行用户请求的操作。
Discovering more about tabs and their properties
There will be occasions when you want to get a list of all the tabs in all the browser windows. Other times you might want to find a subset of tabs that match some specific criteria, such as those opened from a specific tab or displaying pages from a particular domain. And once you have your list of tabs, you'll probably want to know more about their properties.
This is where tabs.query
comes in. Used alone to get all tabs or taking the queryInfo
object—to specify query criteria such as whether the tab is active, in the current window, or one or more of 17 criteria—tabs.query
returns an array of tabs.Tab
objects containing information about the tabs.
Where you want information about the current tab only, you can get a tabs.Tab
object for that tab using tabs.getCurrent
(en-US). If you have a tab’s ID, you can get its tabs.Tab
object using tabs.get
(en-US).
备注:
- In Chrome, the user can select multiple tabs in a window, and the Tabs API sees these as highlighted tabs.
- In Firefox, the user can't select multiple tabs, so "highlighted" and "active" are synonymous.
How to example
To see how tabs.query
and tabs.Tab
are used, let’s walk through how the tabs-tabs-tabs example adds the list of "switch to tabs" to its toolbar button popup.
manifest.json
Here is the manifest.json:
json
{
"browser_action": {
"browser_style": true,
"default_title": "Tabs, tabs, tabs",
"default_popup": "tabs.html"
},
"description": "A list of methods you can perform on a tab.",
"homepage_url": "https://github.com/mdn/webextensions-examples/tree/main/tabs-tabs-tabs",
"manifest_version": 2,
"name": "Tabs, tabs, tabs",
"permissions": ["tabs"],
"version": "1.0"
}
备注: Note the following:
- tabs.html is defined as the
default_popup
inbrowser_action
. It is displayed whenever the user clicks the extension's toolbar icon. - Permissions includes tabs. This is needed to support the tab list feature, as the extension reads the title of the tabs for display in the popup.
tabs.html
tabs.html defines the content of the extension's popup:
html
<!doctype html>
<html>
<head>
<meta charset="utf-8" />
<link rel="stylesheet" href="tabs.css" />
</head>
<body>
<div class="panel">
<div class="panel-section panel-section-header">
<div class="text-section-header">Tabs-tabs-tabs</div>
</div>
<a href="#" id="tabs-move-beginning"
>Move active tab to the beginning of the window</a
><br />
… Define the other menu items …
<div class="switch-tabs">
<p>Switch to tab</p>
<div id="tabs-list"></div>
</div>
</div>
<script src="tabs.js"></script>
</body>
</html>
Here is a summary of the above does:
- The menu items are declared.
- An empty
div
with the IDtabs-list
is declared to contain the list of tabs. - tabs.js is called.
tabs.js
In tabs.js, we’ll see how the list of tabs is built and added to the popup.
Creating the popup
First, an event handler is added to execute listTabs()
when tabs.html is loaded:
js
document.addEventListener("DOMContentLoaded", listTabs);
The first thing that listTabs()
does is to call getCurrentWindowTabs()
. This is where tabs.query
is used to get a tabs.Tab
object for the tabs in the current window:
js
function getCurrentWindowTabs() {
return browser.tabs.query({ currentWindow: true });
}
Now, listTabs()
is ready to create the content for the popup.
To start with:
- Grab the
tabs-list
div
. - Create a document fragment (into which the list will be built).
- Set counters.
- Clear the content of the
tabs-list
div
.
js
function listTabs() {
getCurrentWindowTabs().then((tabs) => {
let tabsList = document.getElementById('tabs-list');
let currentTabs = document.createDocumentFragment();
let limit = 5;
let counter = 0;
tabsList.textContent = '';
Next, we'll create the links for each tab:
- Loops through the first 5 items from the
tabs.Tab
object. - For each item, add a hyperlink to the document fragment.
- The link’s label—that is, its text—is set using the tab’s title (or the ID, if it has no title).
- The link’s address is set using the tab’s ID.
js
for (let tab of tabs) { if (!tab.active && counter <= limit) { let tabLink = document.createElement("a"); tabLink.textContent = tab.title || tab.id; tabLink.setAttribute("href", tab.id); tabLink.classList.add("switch-tabs"); currentTabs.appendChild(tabLink); } counter += 1; }
Finally, the document fragment is written to the tabs-list
div:
js
tabsList.appendChild(currentTabs);
});
}
Working with the active tab
Another related example feature is the "Alert active tab" info option that dumps all the tabs.Tab
object properties for the active tab into an alert:
js
else if (e.target.id === "tabs-alertinfo") {
callOnActiveTab((tab) => {
let props = "";
for (let item in tab) {
props += `${ item } = ${ tab[item] } \n`;
}
alert(props);
});
}
Where callOnActiveTab()
finds the active tab object by looping through the tabs.Tab
objects looking for the item with active set:
js
document.addEventListener("click", function(e) {
function callOnActiveTab(callback) {
getCurrentWindowTabs().then((tabs) => {
for (var tab of tabs) {
if (tab.active) {
callback(tab, tabs);
}
}
});
}
}
Creating, duplicating, moving, updating, reloading, and removing tabs
Having gathered information about the tabs you'll most likely want to do something with them—either to offer users features for manipulating and managing tabs or to implement functionality in your extension.
The following functions are available:
- create a new tab (
tabs.create
(en-US)). - duplicate a tab (
tabs.duplicate
(en-US)). - remove a tab (
tabs.remove
(en-US)). - move a tab (
tabs.move
(en-US)). - update the tab's URL—effectively browse to a new page—(
tabs.update
(en-US)). - reload the tab's page (
tabs.reload
(en-US)).
备注: These functions all require the ID (or IDs) of the tab they are manipulating:
Whereas the following functions will act on the active tab (if no tab ID is provided):
How to example
The tabs-tabs-tabs example exercises all of these features except for updating a tab's URL The way in which these APIs are used is similar, so we'll look at one of the more involved implementations, that of the "Move active tab to the beginning of the window list" option.
But first, here is a demonstration of the feature in action:
manifest.json
None of the functions require a permission to operate, so there are no features in the manifest.json file that need to be highlighted.
tabs.html
tabs.html defines the "menu" displayed in the popup, which includes the "Move active tab to the beginning of the window list" option, with a series of <a>
tags grouped by a visual separator. Each menu item is given an ID, which is used in tabs.js to determine which menu item is being requested.
html
<a href="#" id="tabs-move-beginning"
>Move active tab to the beginning of the window</a
><br />
<a href="#" id="tabs-move-end">Move active tab to the end of the window</a
><br />
<div class="panel-section-separator"></div>
<a href="#" id="tabs-duplicate">Duplicate active tab</a><br />
<a href="#" id="tabs-reload">Reload active tab</a><br />
<a href="#" id="tabs-alertinfo">Alert active tab info</a><br />
tabs.js
To implement the "menu" defined in tabs.html, tabs.js includes a listener for clicks in tabs.html:
js
document.addEventListener("click", function(e) {
function callOnActiveTab(callback) {
getCurrentWindowTabs().then((tabs) => {
for (var tab of tabs) {
if (tab.active) {
callback(tab, tabs);
}
}
});
}
}
A series of if
statements then look to match the ID of the item clicked.
This code snippet is for the "Move active tab to the beginning of the window list" option:
js
if (e.target.id === "tabs-move-beginning") {
callOnActiveTab((tab, tabs) => {
var index = 0;
if (!tab.pinned) {
index = firstUnpinnedTab(tabs);
}
console.log(`moving ${tab.id} to ${index}`);
browser.tabs.move([tab.id], { index });
});
}
It's worth noting the use of console.log()
. This enables you to output information to the debugger console, which can be useful when resolving issues found during development.
The move code first calls callOnActiveTab()
which in turn calls getCurrentWindowTabs()
to get a tabs.Tab
object containing the active window’s tabs. It then loops through the object to find and return the active tab object:
js
function callOnActiveTab(callback) {
getCurrentWindowTabs().then((tabs) => {
for (var tab of tabs) {
if (tab.active) {
callback(tab, tabs);
}
}
});
}
Pinned tabs
A feature of tabs is that the user can pin tabs in a window. Pinned tabs are placed at the start of the tab list and cannot be moved. This means that the earliest position a tab can move to is the first position after any pinned tabs. So, firstUnpinnedTab()
is called to find the position of the first unpinned tab by looping through the tabs
object:
js
function firstUnpinnedTab(tabs) {
for (var tab of tabs) {
if (!tab.pinned) {
return tab.index;
}
}
}
We now have everything needed to move the tab: the active tab object from which we can get the tab ID and the position the tab is to be moved to. So, we can implement the move:
js
browser.tabs.move([tab.id], { index });
The remaining functions to duplicate, reload, create, and remove tabs are implemented similarly.
Manipulating a tab's zoom level
The next set of functions enable you to get (tabs.getZoom
(en-US)) and set (tabs.setZoom
(en-US)) the zoom level within a tab. You can also retrieve the zoom settings (tabs.getZoomSettings
(en-US)) but, at the time of writing, the ability to set the settings (tabs.setZoomSettings
(en-US)) wasn’t available in Firefox.
The level of zoom can be between 30% and 300% (represented as decimals 0.3
to 3
).
In Firefox the default zoom settings are:
- default zoom level: 100%.
- zoom mode: automatic (so the browser manages how zoom levels are set).
- scope of zoom changes:
"per-origin"
, meaning that when you visit a site again, it takes the zoom level set in your last visit.
How to example
The tabs-tabs-tabs example includes three demonstrations of the zoom feature: zoom in, zoom out, and reset zoom. Here is the feature in action:
Let's take a look at how the zoom in is implemented.
manifest.json
None of the zoom functions require permissions, so there are no features in the manifest.json file that need to be highlighted.
tabs.html
We have already discussed how the tabs.html defines the options for this extension, nothing new or unique is done to provide the zoom options.
tabs.js
tabs.js starts by defining several constants used in the zoom code:
js
const ZOOM_INCREMENT = 0.2;
const MAX_ZOOM = 3;
const MIN_ZOOM = 0.3;
const DEFAULT_ZOOM = 1;
It then uses the same listener we discussed earlier so it can act on clicks in tabs.html.
For the zoom in feature, this runs:
js
else if (e.target.id === "tabs-add-zoom") {
callOnActiveTab((tab) => {
var gettingZoom = browser.tabs.getZoom(tab.id);
gettingZoom.then((zoomFactor) => {
//the maximum zoomFactor is 3, it can't go higher
if (zoomFactor >= MAX_ZOOM) {
alert("Tab zoom factor is already at max!");
} else {
var newZoomFactor = zoomFactor + ZOOM_INCREMENT;
//if the newZoomFactor is set to higher than the max accepted
//it won't change, and will never alert that it's at maximum
newZoomFactor = newZoomFactor > MAX_ZOOM ? MAX_ZOOM : newZoomFactor;
browser.tabs.setZoom(tab.id, newZoomFactor);
}
});
});
}
This code uses callOnActiveTab()
to get the details of the active tab, then tabs.getZoom
(en-US) gets the tab’s current zoom factor. The current zoom is compared to the defined maximum (MAX_ZOOM
) and an alert issued if the tab is already at the maximum zoom. Otherwise, the zoom level is incremented but limited to the maximum zoom, then the zoom is set with tabs.getZoom
(en-US).
Manipulating a tab's CSS
Another significant capability offered by the Tabs API is the ability to manipulate the CSS within a tab—add new CSS to a tab (tabs.insertCSS
) or remove CSS from a tab (tabs.removeCSS
(en-US)).
This can be useful, for example, if you want to highlight certain page elements or change the default layout of the page.
How to example
The apply-css example uses these features to add a red border to the web page in the active tab. Here is the feature in action:
Let's walk through how it's set up.
manifest.json
To use the CSS features you need either:
"tabs"
permission and host permission or"activeTab"
permission.
The latter is the most useful, as it allows an extension to use tabs.insertCSS
and tabs.removeCSS
(en-US) in the active tab when run from the extension’s browser or page action, context menu, or a shortcut.
json
{
"description": "Adds a page action to toggle applying CSS to pages.",
"manifest_version": 2,
"name": "apply-css",
"version": "1.0",
"homepage_url": "https://github.com/mdn/webextensions-examples/tree/main/apply-css",
"background": {
"scripts": ["background.js"]
},
"page_action": {
"default_icon": "icons/off.svg"
},
"permissions": ["activeTab", "tabs"]
}
You will note that "tabs"
permission is requested in addition to "activeTab"
. This additional permission is needed to enable the extension’s script to access the tab’s URL, the importance of which we’ll see in a moment.
The other main features in the manifest.json file are the definition of:
- a background script, which starts running as soon as the extension is loaded.
- a "page action", which defines an icon to be added to the browser’s address bar.
background.js
On startup, background.js sets some constants to define the CSS to be applied, titles for the "page action", and a list of protocols the extension will work in:
js
const CSS = "body { border: 20px solid red; }";
const TITLE_APPLY = "Apply CSS";
const TITLE_REMOVE = "Remove CSS";
const APPLICABLE_PROTOCOLS = ["http:", "https:"];
When first loaded, the extension uses tabs.query
to get a list of all the tabs in the current browser window. It then loops through the tabs calling initializePageAction()
.
js
var gettingAllTabs = browser.tabs.query({});
gettingAllTabs.then((tabs) => {
for (let tab of tabs) {
initializePageAction(tab);
}
});
initializePageAction
uses protocolIsApplicable()
to determine whether the active tab's URL is one the CSS can be applied to:
js
function protocolIsApplicable(url) {
var anchor = document.createElement("a");
anchor.href = url;
return APPLICABLE_PROTOCOLS.includes(anchor.protocol);
}
Then, if the example can act on the tab, initializePageAction()
sets the tab's pageAction
(navigation bar) icon and title to use the “off” versions before making the pageAction
visible:
js
function initializePageAction(tab) {
if (protocolIsApplicable(tab.url)) {
browser.pageAction.setIcon({ tabId: tab.id, path: "icons/off.svg" });
browser.pageAction.setTitle({ tabId: tab.id, title: TITLE_APPLY });
browser.pageAction.show(tab.id);
}
}
Next, a listener on pageAction.onClicked
waits for the pageAction
icon to be clicked, and calls toggleCSS
when it is.
js
browser.pageAction.onClicked.addListener(toggleCSS);
toggleCSS()
gets the title of the pageAction
and then takes the action described:
- For "Apply CSS":
- toggles the
pageAction
icon and title to the "remove" versions. - applies the CSS using
tabs.insertCSS
.
- toggles the
- For "Remove CSS":
- toggles the
pageAction
icon and title to the "apply" versions. - removes the CSS using
tabs.removeCSS
(en-US).
- toggles the
js
function toggleCSS(tab) {
function gotTitle(title) {
if (title === TITLE_APPLY) {
browser.pageAction.setIcon({ tabId: tab.id, path: "icons/on.svg" });
browser.pageAction.setTitle({ tabId: tab.id, title: TITLE_REMOVE });
browser.tabs.insertCSS({ code: CSS });
} else {
browser.pageAction.setIcon({ tabId: tab.id, path: "icons/off.svg" });
browser.pageAction.setTitle({ tabId: tab.id, title: TITLE_APPLY });
browser.tabs.removeCSS({ code: CSS });
}
}
var gettingTitle = browser.pageAction.getTitle({ tabId: tab.id });
gettingTitle.then(gotTitle);
}
Finally, to ensure that the pageAction
is valid after each update to the tab, a listener on tabs.onUpdated
(en-US) calls initializePageAction()
each time the tab is updated to check that the tab is still using a protocol to which the CSS can be applied.
js
browser.tabs.onUpdated.addListener((id, changeInfo, tab) => {
initializePageAction(tab);
});
Some other interesting abilities
There are a couple of other Tabs API features that don't fit into one of the earlier sections:
- Capture the visible tab content with
tabs.captureVisibleTab
(en-US). - Detect the primary language of the content in a tab using
tabs.detectLanguage
(en-US). This could be used, for example, to match the language in your extension’s UI with that of the page it’s running in.
Learn more
If you want to learn more about the Tabs API, check out:
- Tabs API reference
- Example extensions (many of which use the Tabs API)