-
Notifications
You must be signed in to change notification settings - Fork 6
/
background.js
220 lines (202 loc) · 6.89 KB
/
background.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
"use strict";
const BOOKMARK_MENU_ITEM_ID = "bookmark-menu-item";
createRootMenuItems();
let lastIds = [];
let lastIsVisible = true;
browser.menus.onClicked.addListener(async (info) => {
let {urls} = await getBookmarkUrls(info.bookmarkId);
if (!urls.length) {
// E.g. when the bookmark folder is empty.
throw new Error("No URLs found in the bookmark");
}
// TODO: show URL in tab when a URL cannot be opened by extensions?
// - file:-URLs - https://bugzil.la/1266960
// - data:-URLs - https://bugzil.la/1317166
// - about:-URLs (other than about:blank)
// The number of tabs that are actively loaded. If the number of tabs to
// open exceed this number, they will lazily be loaded instead.
let discardThreshold = 10;
let i = -1;
for (let url of urls) {
++i;
let openInReaderMode = url.startsWith("about:reader");
if (openInReaderMode) {
try {
let parsed = new URL(url);
url = parsed.searchParams.get("url") + parsed.hash;
} catch (e) {
console.warn(`Failed to parse: ${url} ${e}`);
}
}
let createProperties = {
active: i === 0,
url,
cookieStoreId: info.menuItemId,
openInReaderMode,
};
if (i + 1 >= discardThreshold) {
createProperties.discarded = true;
}
try {
browser.tabs.create(createProperties);
} catch (e) {
// discard supported since Firefox 63 - https://bugzil.la/1378647
if (e.message.includes("Unexpected property \"discarded\"")) {
// Don't try to discard another tab if not supported.
discardThreshold = Infinity;
delete createProperties.discarded;
browser.tabs.create(createProperties);
}
}
}
});
browser.menus.onShown.addListener(async (info) => {
if (info.contexts.includes("bookmark")) {
if (!await shouldEnableMenu(info.bookmarkId)) {
toggleRootMenu(false);
browser.menus.refresh();
return;
}
toggleRootMenu(true);
await Promise.all([
updateRootMenuItem(info.bookmarkId),
updateBookmarkMenuItems(),
]);
browser.menus.refresh();
}
});
browser.runtime.getBrowserInfo().then(({version}) => {
if (/^6[01]\./.test(version)) {
// Firefox 60 and 61 have a bug, where onShown does not work - https://bugzil.la/1473720
// As a work-around, update the menus when the identities change:
browser.contextualIdentities.onCreated.addListener(updateBookmarkMenuItems);
browser.contextualIdentities.onUpdated.removeListener(updateBookmarkMenuItems);
browser.contextualIdentities.onRemoved.hasListener(updateBookmarkMenuItems);
updateBookmarkMenuItems();
}
});
async function shouldEnableMenu(bookmarkId) {
if (bookmarkId === "toolbar_____") {
// The submenu disappears on hover, so let's disable the menu - https://bugzil.la/1489545
return false;
}
let incognito;
try {
({incognito} = await browser.windows.getCurrent());
} catch (e) {
console.warn(`Cannot identify current window: ${e}`);
}
if (incognito) {
// Cannot open container tabs in private browsing mode.
return false;
}
return true;
}
function toggleRootMenu(visible) {
if (lastIsVisible === visible) {
return; // No change.
}
lastIsVisible = visible;
if (!toggleRootMenu._visibleNotSupported) {
try {
// "visible" is supported in Firefox 63+ - https://bugzil.la/1482529
browser.menus.update(BOOKMARK_MENU_ITEM_ID, {visible});
return; // Not thrown? .visible is supported.
} catch (e) {
toggleRootMenu._visibleNotSupported = true;
}
}
if (visible) {
createRootMenuItems();
} else {
lastIds.length = 0;
browser.menus.removeAll();
}
}
function createRootMenuItems() {
browser.menus.create({
id: BOOKMARK_MENU_ITEM_ID,
contexts: ["bookmark"],
title: browser.i18n.getMessage("open_one_in_container_tab"),
});
browser.menus.create({
id: "firefox-default",
parentId: BOOKMARK_MENU_ITEM_ID,
contexts: ["bookmark"],
title: browser.i18n.getMessage("no_container"),
});
browser.menus.create({
id: "separator-after-no-container",
type: "separator",
parentId: BOOKMARK_MENU_ITEM_ID,
contexts: ["bookmark"],
});
}
async function getBookmarkUrls(bookmarkId) {
let [bookmark] = await browser.bookmarks.get(bookmarkId);
let isFolder = bookmark.type === "folder";
let urls = [];
if (isFolder) {
let bookmarks = await browser.bookmarks.getChildren(bookmarkId);
urls = bookmarks.map(b => b.url).filter(url => url);
} else if (bookmark.url) {
urls.push(bookmark.url);
}
// Filter URLs that we certainly don't want to open.
// - Bookmarklets (javascript:) aren't supported.
// - "Most Visited" has type "bookmark" with url = "place:sort=8&maxResults=10".
urls = urls.filter(url => !/^(javascript|place):/i.test(url));
return {urls, isFolder};
}
async function updateRootMenuItem(bookmarkId) {
let urls, isFolder;
try {
({urls, isFolder} = await getBookmarkUrls(bookmarkId));
} catch (e) {
urls = [];
isFolder = false;
}
let title;
if (isFolder) {
title = browser.i18n.getMessage("open_all_in_container_tab");
} else {
title = browser.i18n.getMessage("open_one_in_container_tab");
}
browser.menus.update(BOOKMARK_MENU_ITEM_ID, {
title,
enabled: urls.length > 0,
});
}
async function updateBookmarkMenuItems() {
let cids = await browser.contextualIdentities.query({});
if (cids.length === lastIds.length &&
lastIds.every((id, i) => id === cids[i].cookieStoreId)) {
// No change.
return;
}
// Keep all menu items at the start that haven't been changed,
// remove all others and then append new ones.
let firstI = lastIds.findIndex((id, i) => !cids[i] || cids[i].cookieStoreId !== id);
if (firstI !== -1) {
for (let i = firstI; i < lastIds.length; ++i) {
browser.menus.remove(lastIds[i]);
}
lastIds.length = firstI;
cids = cids.slice(firstI);
} else if (lastIds.length < cids.length) {
cids = cids.slice(lastIds.length);
}
// Append new ones.
for (let cid of cids) {
browser.menus.create({
id: cid.cookieStoreId,
parentId: BOOKMARK_MENU_ITEM_ID,
contexts: ["bookmark"],
title: cid.name,
icons: {
16: `icons/${cid.icon}.svg#${cid.color}`,
},
});
lastIds.push(cid.cookieStoreId);
}
}