Merge pull request #1537 from kendallcorner/bookmark-context
Adds a bookmark context menu for opening bookmarks in container tabs #323
This commit is contained in:
commit
8dd9927539
7 changed files with 165 additions and 8 deletions
|
@ -4,7 +4,7 @@ const assignManager = {
|
||||||
MENU_SEPARATOR_ID: "separator",
|
MENU_SEPARATOR_ID: "separator",
|
||||||
MENU_HIDE_ID: "hide-container",
|
MENU_HIDE_ID: "hide-container",
|
||||||
MENU_MOVE_ID: "move-to-new-window-container",
|
MENU_MOVE_ID: "move-to-new-window-container",
|
||||||
|
OPEN_IN_CONTAINER: "open-bookmark-in-container-tab",
|
||||||
storageArea: {
|
storageArea: {
|
||||||
area: browser.storage.local,
|
area: browser.storage.local,
|
||||||
exemptedTabs: {},
|
exemptedTabs: {},
|
||||||
|
@ -220,7 +220,7 @@ const assignManager = {
|
||||||
|
|
||||||
init() {
|
init() {
|
||||||
browser.contextMenus.onClicked.addListener((info, tab) => {
|
browser.contextMenus.onClicked.addListener((info, tab) => {
|
||||||
this._onClickedHandler(info, tab);
|
info.bookmarkId ? this._onClickedBookmark(info) : this._onClickedHandler(info, tab);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Before a request is handled by the browser we decide if we should route through a different container
|
// Before a request is handled by the browser we decide if we should route through a different container
|
||||||
|
@ -240,7 +240,39 @@ const assignManager = {
|
||||||
delete this.canceledRequests[options.tabId];
|
delete this.canceledRequests[options.tabId];
|
||||||
}
|
}
|
||||||
},{urls: ["<all_urls>"], types: ["main_frame"]});
|
},{urls: ["<all_urls>"], types: ["main_frame"]});
|
||||||
|
this.getPermissions();
|
||||||
|
},
|
||||||
|
|
||||||
|
async getPermissions() {
|
||||||
|
const {permissions} = await browser.permissions.getAll();
|
||||||
|
permissions.includes("bookmarks") ? this.makeBookmarksMenu() : browser.contextMenus.remove(this.OPEN_IN_CONTAINER);
|
||||||
|
},
|
||||||
|
|
||||||
|
makeBookmarksMenu() {
|
||||||
|
this.initBookmarksMenu();
|
||||||
|
browser.contextualIdentities.onCreated.addListener(this.contextualIdentityCreated);
|
||||||
|
browser.contextualIdentities.onUpdated.addListener(this.contextualIdentityUpdated);
|
||||||
|
browser.contextualIdentities.onRemoved.addListener(this.contextualIdentityRemoved);
|
||||||
|
},
|
||||||
|
|
||||||
|
contextualIdentityCreated(changeInfo) {
|
||||||
|
browser.contextMenus.create({
|
||||||
|
parentId: assignManager.OPEN_IN_CONTAINER,
|
||||||
|
id: changeInfo.contextualIdentity.cookieStoreId,
|
||||||
|
title: changeInfo.contextualIdentity.name,
|
||||||
|
icons: { "16": `img/usercontext.svg#${changeInfo.contextualIdentity.icon}` }
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
contextualIdentityUpdated(changeInfo) {
|
||||||
|
browser.contextMenus.update(changeInfo.contextualIdentity.cookieStoreId, {
|
||||||
|
title: changeInfo.contextualIdentity.name,
|
||||||
|
icons: { "16": `img/usercontext.svg#${changeInfo.contextualIdentity.icon}` }
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
contextualIdentityRemoved(changeInfo) {
|
||||||
|
browser.contextMenus.remove(changeInfo.contextualIdentity.cookieStoreId);
|
||||||
},
|
},
|
||||||
|
|
||||||
async _onClickedHandler(info, tab) {
|
async _onClickedHandler(info, tab) {
|
||||||
|
@ -274,6 +306,38 @@ const assignManager = {
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
async _onClickedBookmark(info) {
|
||||||
|
|
||||||
|
async function _getBookmarksFromInfo(info) {
|
||||||
|
const [bookmarkTreeNode] = await browser.bookmarks.get(info.bookmarkId);
|
||||||
|
if (bookmarkTreeNode.type === "folder") {
|
||||||
|
return await browser.bookmarks.getChildren(bookmarkTreeNode.id);
|
||||||
|
}
|
||||||
|
return [bookmarkTreeNode];
|
||||||
|
}
|
||||||
|
|
||||||
|
const bookmarks = await _getBookmarksFromInfo(info);
|
||||||
|
for (const bookmark of bookmarks) {
|
||||||
|
// Some checks on the urls from https://github.com/Rob--W/bookmark-container-tab/ thanks!
|
||||||
|
if ( !/^(javascript|place):/i.test(bookmark.url) && bookmark.type !== "folder") {
|
||||||
|
const openInReaderMode = bookmark.url.startsWith("about:reader");
|
||||||
|
if(openInReaderMode) {
|
||||||
|
try {
|
||||||
|
const parsed = new URL(bookmark.url);
|
||||||
|
bookmark.url = parsed.searchParams.get("url") + parsed.hash; // can't believe const lets me do this ...
|
||||||
|
} catch (err) {
|
||||||
|
return err.message;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
browser.tabs.create({
|
||||||
|
cookieStoreId: info.menuItemId,
|
||||||
|
url: bookmark.url,
|
||||||
|
openInReaderMode: openInReaderMode
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
deleteContainer(userContextId) {
|
deleteContainer(userContextId) {
|
||||||
this.storageArea.deleteContainer(userContextId);
|
this.storageArea.deleteContainer(userContextId);
|
||||||
|
@ -439,6 +503,24 @@ const assignManager = {
|
||||||
throw e;
|
throw e;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async initBookmarksMenu() {
|
||||||
|
browser.contextMenus.create({
|
||||||
|
id: this.OPEN_IN_CONTAINER,
|
||||||
|
title: "Open Bookmark in Container Tab",
|
||||||
|
contexts: ["bookmark"],
|
||||||
|
});
|
||||||
|
|
||||||
|
const identities = await browser.contextualIdentities.query({});
|
||||||
|
for (const identity of identities) {
|
||||||
|
browser.contextMenus.create({
|
||||||
|
parentId: this.OPEN_IN_CONTAINER,
|
||||||
|
id: identity.cookieStoreId,
|
||||||
|
title: identity.name,
|
||||||
|
icons: { "16": `img/usercontext.svg#${identity.icon}` }
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -46,9 +46,6 @@ const backgroundLogic = {
|
||||||
donePromise = browser.contextualIdentities.create(options.params);
|
donePromise = browser.contextualIdentities.create(options.params);
|
||||||
}
|
}
|
||||||
await donePromise;
|
await donePromise;
|
||||||
browser.runtime.sendMessage({
|
|
||||||
method: "refreshNeeded"
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
|
|
||||||
async openNewTab(options) {
|
async openNewTab(options) {
|
||||||
|
@ -329,4 +326,4 @@ const backgroundLogic = {
|
||||||
cookieStoreId(userContextId) {
|
cookieStoreId(userContextId) {
|
||||||
return `firefox-container-${userContextId}`;
|
return `firefox-container-${userContextId}`;
|
||||||
}
|
}
|
||||||
};
|
};
|
|
@ -10,6 +10,9 @@ const messageHandler = {
|
||||||
let response;
|
let response;
|
||||||
|
|
||||||
switch (m.method) {
|
switch (m.method) {
|
||||||
|
case "resetBookmarksContext":
|
||||||
|
response = assignManager.getPermissions();
|
||||||
|
break;
|
||||||
case "deleteContainer":
|
case "deleteContainer":
|
||||||
response = backgroundLogic.deleteContainer(m.message.userContextId);
|
response = backgroundLogic.deleteContainer(m.message.userContextId);
|
||||||
break;
|
break;
|
||||||
|
|
41
src/js/options.js
Normal file
41
src/js/options.js
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
|
||||||
|
function requestPermissions() {
|
||||||
|
const checkbox = document.querySelector("#bookmarksPermissions");
|
||||||
|
if (checkbox.checked) {
|
||||||
|
browser.permissions.request({permissions: ["bookmarks"]}).
|
||||||
|
then((response) => {
|
||||||
|
if (response) {
|
||||||
|
browser.runtime.sendMessage({ method: "resetBookmarksContext" });
|
||||||
|
} else {
|
||||||
|
checkbox.checked = false;
|
||||||
|
}
|
||||||
|
}).
|
||||||
|
catch((err) => {
|
||||||
|
return err.message;
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
browser.permissions.remove({permissions: ["bookmarks"]}).
|
||||||
|
then(() => {
|
||||||
|
browser.runtime.sendMessage({ method: "resetBookmarksContext" });
|
||||||
|
}).
|
||||||
|
catch((err) => {
|
||||||
|
return err.message;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function restoreOptions() {
|
||||||
|
browser.permissions.getAll()
|
||||||
|
.then((permissions) => {
|
||||||
|
if (permissions.permissions.includes("bookmarks")) {
|
||||||
|
document.querySelector("#bookmarksPermissions").checked = true;
|
||||||
|
}
|
||||||
|
}).
|
||||||
|
catch((err) => {
|
||||||
|
return err.message;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", restoreOptions);
|
||||||
|
document.querySelector("#bookmarksPermissions").addEventListener( "change", requestPermissions);
|
|
@ -29,6 +29,9 @@
|
||||||
"webRequestBlocking",
|
"webRequestBlocking",
|
||||||
"webRequest"
|
"webRequest"
|
||||||
],
|
],
|
||||||
|
"optional_permissions": [
|
||||||
|
"bookmarks"
|
||||||
|
],
|
||||||
"commands": {
|
"commands": {
|
||||||
"_execute_browser_action": {
|
"_execute_browser_action": {
|
||||||
"suggested_key": {
|
"suggested_key": {
|
||||||
|
@ -70,5 +73,8 @@
|
||||||
],
|
],
|
||||||
"web_accessible_resources": [
|
"web_accessible_resources": [
|
||||||
"/img/container-site-d-24.png"
|
"/img/container-site-d-24.png"
|
||||||
]
|
],
|
||||||
|
"options_ui": {
|
||||||
|
"page": "options.html"
|
||||||
|
}
|
||||||
}
|
}
|
16
src/options.html
Normal file
16
src/options.html
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<form>
|
||||||
|
<input type="checkbox" id="bookmarksPermissions"/>
|
||||||
|
<label>Enable Bookmark Menus</label>
|
||||||
|
<p>This setting allows you to open a bookmark or folder of bookmarks in a container.</p>
|
||||||
|
</form>
|
||||||
|
<script src="js/options.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -58,7 +58,16 @@ module.exports = () => {
|
||||||
contextualIdentities: {
|
contextualIdentities: {
|
||||||
create: sinon.stub(),
|
create: sinon.stub(),
|
||||||
get: sinon.stub(),
|
get: sinon.stub(),
|
||||||
query: sinon.stub().resolves([])
|
query: sinon.stub().resolves([]),
|
||||||
|
onCreated: {
|
||||||
|
addListener: sinon.stub()
|
||||||
|
},
|
||||||
|
onUpdated: {
|
||||||
|
addListener: sinon.stub()
|
||||||
|
},
|
||||||
|
onRemoved: {
|
||||||
|
addListener: sinon.stub()
|
||||||
|
}
|
||||||
},
|
},
|
||||||
contextMenus: {
|
contextMenus: {
|
||||||
create: sinon.stub(),
|
create: sinon.stub(),
|
||||||
|
@ -82,6 +91,9 @@ module.exports = () => {
|
||||||
},
|
},
|
||||||
extension: {
|
extension: {
|
||||||
getURL: sinon.stub().returns("moz-extension://multi-account-containers/confirm-page.html")
|
getURL: sinon.stub().returns("moz-extension://multi-account-containers/confirm-page.html")
|
||||||
|
},
|
||||||
|
permissions: {
|
||||||
|
getAll: sinon.stub().returns({"permissions": ["bookmarks"]})
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue