Compare commits
19 commits
webextUpda
...
main
Author | SHA1 | Date | |
---|---|---|---|
![]() |
aca51cc11c | ||
![]() |
b684ce7016 | ||
![]() |
115d411218 | ||
![]() |
aec2aa5fb0 | ||
![]() |
69ee83bbf6 | ||
![]() |
9434147b48 | ||
![]() |
d82341ce4a | ||
![]() |
60a6666222 | ||
![]() |
f1a24ed6fb | ||
![]() |
0372abdc33 | ||
![]() |
366a50c8b6 | ||
![]() |
e96b275e02 | ||
![]() |
65243e2c06 | ||
![]() |
ab3e1ce4d8 | ||
![]() |
5194fcad0e | ||
![]() |
b6a1bff9e8 | ||
![]() |
5ae2047b2c | ||
![]() |
a60f5bb1be | ||
![]() |
c8c4e0f0c5 |
13 changed files with 185 additions and 119 deletions
|
@ -1,6 +1,6 @@
|
||||||
module.exports = {
|
module.exports = {
|
||||||
"parserOptions": {
|
"parserOptions": {
|
||||||
"ecmaVersion": 2018
|
"ecmaVersion": 2021
|
||||||
},
|
},
|
||||||
"env": {
|
"env": {
|
||||||
"browser": true,
|
"browser": true,
|
||||||
|
|
2
.github/workflows/builds.yaml
vendored
2
.github/workflows/builds.yaml
vendored
|
@ -26,7 +26,7 @@ jobs:
|
||||||
./bin/build-addon.sh nightly.xpi
|
./bin/build-addon.sh nightly.xpi
|
||||||
|
|
||||||
- name: Uploading
|
- name: Uploading
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: ${{matrix.config.name}} Build
|
name: ${{matrix.config.name}} Build
|
||||||
path: src/web-ext-artifacts
|
path: src/web-ext-artifacts
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
"name": "testpilot-containers",
|
"name": "testpilot-containers",
|
||||||
"title": "Multi-Account Containers",
|
"title": "Multi-Account Containers",
|
||||||
"description": "Containers helps you keep all the parts of your online life contained in different tabs. Custom labels and color-coded tabs help keep different activities — like online shopping, travel planning, or checking work email — separate.",
|
"description": "Containers helps you keep all the parts of your online life contained in different tabs. Custom labels and color-coded tabs help keep different activities — like online shopping, travel planning, or checking work email — separate.",
|
||||||
"version": "8.2.0",
|
"version": "8.3.0",
|
||||||
"author": "Andrea Marchesini, Luke Crouch, Lesley Norton, Kendall Werts, Maxx Crawford, Jonathan Kingston",
|
"author": "Andrea Marchesini, Luke Crouch, Lesley Norton, Kendall Werts, Maxx Crawford, Jonathan Kingston",
|
||||||
"bugs": {
|
"bugs": {
|
||||||
"url": "https://github.com/mozilla/multi-account-containers/issues"
|
"url": "https://github.com/mozilla/multi-account-containers/issues"
|
||||||
|
|
|
@ -110,7 +110,6 @@
|
||||||
--usercontext-bg-hover-color: #f0f0fa;
|
--usercontext-bg-hover-color: #f0f0fa;
|
||||||
--usercontext-bg-focus-color: #e0e0e6;
|
--usercontext-bg-focus-color: #e0e0e6;
|
||||||
--usercontext-bg-active-color: #cfcfd8;
|
--usercontext-bg-active-color: #cfcfd8;
|
||||||
--moz-vpn-tout-shadow: 0 0 7px 0 #9498a25e;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[data-theme="dark"] {
|
[data-theme="dark"] {
|
||||||
|
@ -180,7 +179,6 @@
|
||||||
--usercontext-bg-active-color: #5b5b66;
|
--usercontext-bg-active-color: #5b5b66;
|
||||||
--usercontext-bg-focus-color: #2b2a33;
|
--usercontext-bg-focus-color: #2b2a33;
|
||||||
--usercontext-bg-hover-color: #52525e;
|
--usercontext-bg-hover-color: #52525e;
|
||||||
--moz-vpn-tout-shadow: 0 0 7px 0 #7478825e;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* General Rules and Resets */
|
/* General Rules and Resets */
|
||||||
|
@ -635,37 +633,8 @@ input:checked:hover:focus + .slider {
|
||||||
background-color: var(--button-bg-active-color-primary);
|
background-color: var(--button-bg-active-color-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Mozilla VPN tout */
|
|
||||||
|
|
||||||
#moz-vpn-tout {
|
|
||||||
opacity: 0;
|
|
||||||
background-color: var(--panel-bg-color);
|
|
||||||
visibility: visible;
|
|
||||||
max-block-size: 500px;
|
|
||||||
position: absolute;
|
|
||||||
inset-block-end: var(--footerHeight);
|
|
||||||
inset-inline-start: 0;
|
|
||||||
inset-inline-end: 0;
|
|
||||||
box-shadow: var(--moz-vpn-tout-shadow);
|
|
||||||
animation: appear 0.2s ease-out 0.5s forwards;
|
|
||||||
transition: opacity 0.1s ease-in-out, max-height 0.3s ease-in-out;
|
|
||||||
}
|
|
||||||
|
|
||||||
@keyframes appear {
|
|
||||||
0% {
|
|
||||||
opacity: 0;
|
|
||||||
transform: translateY(10%);
|
|
||||||
}
|
|
||||||
|
|
||||||
100% {
|
|
||||||
opacity: 1;
|
|
||||||
transform: translateY(0%);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Mozilla VPN Controller UI in Container Management Panel */
|
/* Mozilla VPN Controller UI in Container Management Panel */
|
||||||
|
|
||||||
.moz-vpn-content,
|
|
||||||
.moz-vpn-controller-content {
|
.moz-vpn-controller-content {
|
||||||
display: flex;
|
display: flex;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
@ -1053,7 +1022,6 @@ input.proxies {
|
||||||
|
|
||||||
/* Mozilla VPN Server list */
|
/* Mozilla VPN Server list */
|
||||||
|
|
||||||
.moz-vpn-logo,
|
|
||||||
.moz-vpn-logotype {
|
.moz-vpn-logotype {
|
||||||
color: var(--text-color-primary);
|
color: var(--text-color-primary);
|
||||||
background-image: var(--logoMozillaVpn);
|
background-image: var(--logoMozillaVpn);
|
||||||
|
|
|
@ -326,7 +326,8 @@ window.assignManager = {
|
||||||
options.url,
|
options.url,
|
||||||
tab.index + 1,
|
tab.index + 1,
|
||||||
tab.active,
|
tab.active,
|
||||||
openTabId
|
openTabId,
|
||||||
|
tab.groupId
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
this.reloadPageInContainer(
|
this.reloadPageInContainer(
|
||||||
|
@ -336,7 +337,8 @@ window.assignManager = {
|
||||||
tab.index + 1,
|
tab.index + 1,
|
||||||
tab.active,
|
tab.active,
|
||||||
siteSettings.neverAsk,
|
siteSettings.neverAsk,
|
||||||
openTabId
|
openTabId,
|
||||||
|
tab.groupId
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
this.calculateContextMenu(tab);
|
this.calculateContextMenu(tab);
|
||||||
|
@ -481,9 +483,7 @@ window.assignManager = {
|
||||||
},
|
},
|
||||||
|
|
||||||
contextualIdentityRemoved(changeInfo) {
|
contextualIdentityRemoved(changeInfo) {
|
||||||
browser.contextMenus.remove(
|
this.removeMenuItem(changeInfo.contextualIdentity.cookieStoreId);
|
||||||
changeInfo.contextualIdentity.cookieStoreId
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
async _onClickedHandler(info, tab) {
|
async _onClickedHandler(info, tab) {
|
||||||
|
@ -670,11 +670,11 @@ window.assignManager = {
|
||||||
// See: https://bugzilla.mozilla.org/show_bug.cgi?id=1215376#c16
|
// See: https://bugzilla.mozilla.org/show_bug.cgi?id=1215376#c16
|
||||||
// We also can't change for always private mode
|
// We also can't change for always private mode
|
||||||
// See: https://bugzilla.mozilla.org/show_bug.cgi?id=1352102
|
// See: https://bugzilla.mozilla.org/show_bug.cgi?id=1352102
|
||||||
browser.contextMenus.remove(this.MENU_ASSIGN_ID);
|
this.removeMenuItem(this.MENU_ASSIGN_ID);
|
||||||
browser.contextMenus.remove(this.MENU_REMOVE_ID);
|
this.removeMenuItem(this.MENU_REMOVE_ID);
|
||||||
browser.contextMenus.remove(this.MENU_SEPARATOR_ID);
|
this.removeMenuItem(this.MENU_SEPARATOR_ID);
|
||||||
browser.contextMenus.remove(this.MENU_HIDE_ID);
|
this.removeMenuItem(this.MENU_HIDE_ID);
|
||||||
browser.contextMenus.remove(this.MENU_MOVE_ID);
|
this.removeMenuItem(this.MENU_MOVE_ID);
|
||||||
},
|
},
|
||||||
|
|
||||||
async calculateContextMenu(tab) {
|
async calculateContextMenu(tab) {
|
||||||
|
@ -727,7 +727,15 @@ window.assignManager = {
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
reloadPageInDefaultContainer(url, index, active, openerTabId) {
|
/**
|
||||||
|
* @param {string} url
|
||||||
|
* @param {number} index
|
||||||
|
* @param {boolean} active
|
||||||
|
* @param {number} [openerTabId]
|
||||||
|
* @param {number} [groupId]
|
||||||
|
* @returns {void}
|
||||||
|
*/
|
||||||
|
reloadPageInDefaultContainer(url, index, active, openerTabId, groupId) {
|
||||||
// To create a new tab in the default container, it is easiest just to omit the
|
// To create a new tab in the default container, it is easiest just to omit the
|
||||||
// cookieStoreId entirely.
|
// cookieStoreId entirely.
|
||||||
//
|
//
|
||||||
|
@ -746,16 +754,58 @@ window.assignManager = {
|
||||||
// does not automatically return to the original opener tab. To get this desired behaviour,
|
// does not automatically return to the original opener tab. To get this desired behaviour,
|
||||||
// we MUST specify the openerTabId when creating the new tab.
|
// we MUST specify the openerTabId when creating the new tab.
|
||||||
const cookieStoreId = "firefox-default";
|
const cookieStoreId = "firefox-default";
|
||||||
browser.tabs.create({url, cookieStoreId, index, active, openerTabId});
|
this.createTabWrapper(url, cookieStoreId, index, active, openerTabId, groupId);
|
||||||
},
|
},
|
||||||
|
|
||||||
reloadPageInContainer(url, currentUserContextId, userContextId, index, active, neverAsk = false, openerTabId = null) {
|
|
||||||
|
/**
|
||||||
|
* Wraps around `browser.tabs.create` and `browser.tabs.group` to create a
|
||||||
|
* tab and ensure that it ends up in the requested tab group, if applicable.
|
||||||
|
*
|
||||||
|
* @param {string} url
|
||||||
|
* @param {string} cookieStoreId
|
||||||
|
* @param {number} index
|
||||||
|
* @param {boolean} active
|
||||||
|
* @param {number} openerTabId
|
||||||
|
* @param {number} [groupId] Tab group ID
|
||||||
|
* @returns {Promise<Tab>}
|
||||||
|
*/
|
||||||
|
async createTabWrapper(url, cookieStoreId, index, active, openerTabId, groupId) {
|
||||||
|
const newTab = await browser.tabs.create({
|
||||||
|
url,
|
||||||
|
cookieStoreId,
|
||||||
|
index,
|
||||||
|
active,
|
||||||
|
openerTabId,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (groupId >= 0) {
|
||||||
|
// If the original tab was in a tab group, make sure that the reopened tab
|
||||||
|
// stays in the same tab group.
|
||||||
|
await browser.tabs.group({ groupId, tabIds: newTab.id });
|
||||||
|
}
|
||||||
|
|
||||||
|
return newTab;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {string} url
|
||||||
|
* @param {string} currentUserContextId
|
||||||
|
* @param {string} userContextId
|
||||||
|
* @param {number} index
|
||||||
|
* @param {boolean} active
|
||||||
|
* @param {boolean} [neverAsk=false]
|
||||||
|
* @param {number} [openerTabId=null]
|
||||||
|
* @param {number} [groupId]
|
||||||
|
* @returns {Promise<Tab>}
|
||||||
|
*/
|
||||||
|
reloadPageInContainer(url, currentUserContextId, userContextId, index, active, neverAsk = false, openerTabId = null, groupId = undefined) {
|
||||||
const cookieStoreId = backgroundLogic.cookieStoreId(userContextId);
|
const cookieStoreId = backgroundLogic.cookieStoreId(userContextId);
|
||||||
const loadPage = browser.runtime.getURL("confirm-page.html");
|
const loadPage = browser.runtime.getURL("confirm-page.html");
|
||||||
// False represents assignment is not permitted
|
// False represents assignment is not permitted
|
||||||
// If the user has explicitly checked "Never Ask Again" on the warning page we will send them straight there
|
// If the user has explicitly checked "Never Ask Again" on the warning page we will send them straight there
|
||||||
if (neverAsk) {
|
if (neverAsk) {
|
||||||
return browser.tabs.create({url, cookieStoreId, index, active, openerTabId});
|
return this.createTabWrapper(url, cookieStoreId, index, active, openerTabId, groupId);
|
||||||
} else {
|
} else {
|
||||||
let confirmUrl = `${loadPage}?url=${this.encodeURLProperty(url)}&cookieStoreId=${cookieStoreId}`;
|
let confirmUrl = `${loadPage}?url=${this.encodeURLProperty(url)}&cookieStoreId=${cookieStoreId}`;
|
||||||
let currentCookieStoreId;
|
let currentCookieStoreId;
|
||||||
|
@ -763,13 +813,14 @@ window.assignManager = {
|
||||||
currentCookieStoreId = backgroundLogic.cookieStoreId(currentUserContextId);
|
currentCookieStoreId = backgroundLogic.cookieStoreId(currentUserContextId);
|
||||||
confirmUrl += `¤tCookieStoreId=${currentCookieStoreId}`;
|
confirmUrl += `¤tCookieStoreId=${currentCookieStoreId}`;
|
||||||
}
|
}
|
||||||
return browser.tabs.create({
|
return this.createTabWrapper(
|
||||||
url: confirmUrl,
|
confirmUrl,
|
||||||
cookieStoreId: currentCookieStoreId,
|
currentCookieStoreId,
|
||||||
openerTabId,
|
|
||||||
index,
|
index,
|
||||||
active
|
active,
|
||||||
}).then(() => {
|
openerTabId,
|
||||||
|
groupId
|
||||||
|
).then(() => {
|
||||||
// We don't want to sync this URL ever nor clutter the users history
|
// We don't want to sync this URL ever nor clutter the users history
|
||||||
browser.history.deleteUrl({url: confirmUrl});
|
browser.history.deleteUrl({url: confirmUrl});
|
||||||
}).catch((e) => {
|
}).catch((e) => {
|
||||||
|
@ -797,12 +848,19 @@ window.assignManager = {
|
||||||
},
|
},
|
||||||
|
|
||||||
async removeBookmarksMenu() {
|
async removeBookmarksMenu() {
|
||||||
browser.contextMenus.remove(this.OPEN_IN_CONTAINER);
|
this.removeMenuItem(this.OPEN_IN_CONTAINER);
|
||||||
const identities = await browser.contextualIdentities.query({});
|
const identities = await browser.contextualIdentities.query({});
|
||||||
for (const identity of identities) {
|
for (const identity of identities) {
|
||||||
browser.contextMenus.remove(identity.cookieStoreId);
|
this.removeMenuItem(identity.cookieStoreId);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
removeMenuItem(menuItemId) {
|
||||||
|
// Callers do not check whether the menu exists before attempting to remove
|
||||||
|
// it. contextMenus.remove rejects when the menu does not exist, so we need
|
||||||
|
// to catch and swallow the error to avoid logspam.
|
||||||
|
browser.contextMenus.remove(menuItemId).catch(() => {});
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
assignManager.init();
|
assignManager.init();
|
||||||
|
|
|
@ -35,10 +35,39 @@ const backgroundLogic = {
|
||||||
browser.permissions.onRemoved.addListener(permissions => this.resetPermissions(permissions));
|
browser.permissions.onRemoved.addListener(permissions => this.resetPermissions(permissions));
|
||||||
|
|
||||||
// Update Translation in Manifest
|
// Update Translation in Manifest
|
||||||
browser.runtime.onInstalled.addListener(this.updateTranslationInManifest);
|
browser.runtime.onInstalled.addListener((details) => {
|
||||||
|
this.updateTranslationInManifest();
|
||||||
|
this._undoDefault820SortTabsKeyboardShortcut(details);
|
||||||
|
});
|
||||||
browser.runtime.onStartup.addListener(this.updateTranslationInManifest);
|
browser.runtime.onStartup.addListener(this.updateTranslationInManifest);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* One-time migration after updating from v8.2.0:
|
||||||
|
* Unset the default keyboard shortcut (Ctrl+Comma) for the `sort_tabs`
|
||||||
|
* command if it was set in v8.2.0 of this addon. If the user remapped
|
||||||
|
* a different shortcut manually, retain their shortcut. Users who used
|
||||||
|
* the default keyboard shortcut will need to manually set a shortcut.
|
||||||
|
* See https://support.mozilla.org/en-US/kb/manage-extension-shortcuts-firefox
|
||||||
|
*
|
||||||
|
* @param {{reason: runtime.OnInstalledReason, previousVersion?: string}} details
|
||||||
|
*/
|
||||||
|
async _undoDefault820SortTabsKeyboardShortcut(details) {
|
||||||
|
if (details.reason === "update" && details.previousVersion === "8.2.0") {
|
||||||
|
const commands = await browser.commands.getAll();
|
||||||
|
const sortTabsCommand = commands.find(command => command.name === "sort_tabs");
|
||||||
|
if (sortTabsCommand) {
|
||||||
|
const previouslySuggestedKeys = [
|
||||||
|
"Ctrl+Comma", // "default"
|
||||||
|
"MacCtrl+Comma", // "mac"
|
||||||
|
];
|
||||||
|
if (previouslySuggestedKeys.includes(sortTabsCommand.shortcut)) {
|
||||||
|
browser.commands.reset("sort_tabs");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
updateTranslationInManifest() {
|
updateTranslationInManifest() {
|
||||||
for (let index = 0; index < 10; index++) {
|
for (let index = 0; index < 10; index++) {
|
||||||
const ajustedIndex = index + 1; // We want to start from 1 instead of 0 in the UI.
|
const ajustedIndex = index + 1; // We want to start from 1 instead of 0 in the UI.
|
||||||
|
@ -343,7 +372,13 @@ const backgroundLogic = {
|
||||||
let pos = 0;
|
let pos = 0;
|
||||||
|
|
||||||
// Let's collect UCIs/tabs for this window.
|
// Let's collect UCIs/tabs for this window.
|
||||||
|
/** @type {Map<string, {order: string, tabs: Tab[]}>} */
|
||||||
const map = new Map;
|
const map = new Map;
|
||||||
|
|
||||||
|
const lastTab = tabs.at(-1);
|
||||||
|
/** @type {boolean} */
|
||||||
|
let lastTabIsInTabGroup = !!lastTab && lastTab.groupId >= 0;
|
||||||
|
|
||||||
for (const tab of tabs) {
|
for (const tab of tabs) {
|
||||||
if (pinnedTabs && !tab.pinned) {
|
if (pinnedTabs && !tab.pinned) {
|
||||||
// We don't have, or we already handled all the pinned tabs.
|
// We don't have, or we already handled all the pinned tabs.
|
||||||
|
@ -356,6 +391,11 @@ const backgroundLogic = {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (tab.groupId >= 0) {
|
||||||
|
// Skip over tabs in tab groups until it's possible to handle them better.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (!map.has(tab.cookieStoreId)) {
|
if (!map.has(tab.cookieStoreId)) {
|
||||||
const userContextId = backgroundLogic.getUserContextIdFromCookieStoreId(tab.cookieStoreId);
|
const userContextId = backgroundLogic.getUserContextIdFromCookieStoreId(tab.cookieStoreId);
|
||||||
map.set(tab.cookieStoreId, { order: userContextId, tabs: [] });
|
map.set(tab.cookieStoreId, { order: userContextId, tabs: [] });
|
||||||
|
@ -377,15 +417,25 @@ const backgroundLogic = {
|
||||||
const sortMap = new Map([...map.entries()].sort((a, b) => a[1].order > b[1].order));
|
const sortMap = new Map([...map.entries()].sort((a, b) => a[1].order > b[1].order));
|
||||||
|
|
||||||
// Let's move tabs.
|
// Let's move tabs.
|
||||||
sortMap.forEach(obj => {
|
for (const { tabs } of sortMap.values()) {
|
||||||
for (const tab of obj.tabs) {
|
for (const tab of tabs) {
|
||||||
++pos;
|
++pos;
|
||||||
browser.tabs.move(tab.id, {
|
browser.tabs.move(tab.id, {
|
||||||
windowId: windowObj.id,
|
windowId: windowObj.id,
|
||||||
index: pos
|
index: pinnedTabs ? pos : -1
|
||||||
});
|
});
|
||||||
|
// Pinned tabs are never grouped and always inserted in the front.
|
||||||
|
if (!pinnedTabs && lastTabIsInTabGroup && browser.tabs.ungroup) {
|
||||||
|
// If the last item in the tab strip is a grouped tab, moving a tab
|
||||||
|
// to its position will also add it to the tab group. Since this code
|
||||||
|
// is only sorting ungrouped tabs, this forcibly ungroups the first
|
||||||
|
// tab to be moved. All subsequent iterations will only be moving
|
||||||
|
// ungrouped tabs to the position of other ungrouped tabs.
|
||||||
|
lastTabIsInTabGroup = false;
|
||||||
|
browser.tabs.ungroup(tab.id);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
async hideTabs(options) {
|
async hideTabs(options) {
|
||||||
|
|
|
@ -91,7 +91,9 @@ const messageHandler = {
|
||||||
m.newUserContextId,
|
m.newUserContextId,
|
||||||
m.tabIndex,
|
m.tabIndex,
|
||||||
m.active,
|
m.active,
|
||||||
true
|
true,
|
||||||
|
null,
|
||||||
|
m.groupId
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
case "assignAndReloadInContainer":
|
case "assignAndReloadInContainer":
|
||||||
|
@ -101,7 +103,9 @@ const messageHandler = {
|
||||||
m.newUserContextId,
|
m.newUserContextId,
|
||||||
m.tabIndex,
|
m.tabIndex,
|
||||||
m.active,
|
m.active,
|
||||||
true
|
true,
|
||||||
|
null,
|
||||||
|
m.groupId
|
||||||
);
|
);
|
||||||
// m.tabId is used for where to place the in content message
|
// m.tabId is used for where to place the in content message
|
||||||
// m.url is the assignment to be removed/added
|
// m.url is the assignment to be removed/added
|
||||||
|
|
|
@ -69,11 +69,15 @@ function confirmSubmit(redirectUrl, cookieStoreId) {
|
||||||
openInContainer(redirectUrl, cookieStoreId);
|
openInContainer(redirectUrl, cookieStoreId);
|
||||||
}
|
}
|
||||||
|
|
||||||
function getCurrentTab() {
|
/**
|
||||||
return browser.tabs.query({
|
* @returns {Promise<Tab>}
|
||||||
|
*/
|
||||||
|
async function getCurrentTab() {
|
||||||
|
const tabs = await browser.tabs.query({
|
||||||
active: true,
|
active: true,
|
||||||
windowId: browser.windows.WINDOW_ID_CURRENT
|
windowId: browser.windows.WINDOW_ID_CURRENT
|
||||||
});
|
});
|
||||||
|
return tabs[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
async function denySubmit(redirectUrl, currentCookieStoreId) {
|
async function denySubmit(redirectUrl, currentCookieStoreId) {
|
||||||
|
@ -93,7 +97,7 @@ async function denySubmit(redirectUrl, currentCookieStoreId) {
|
||||||
|
|
||||||
await browser.runtime.sendMessage({
|
await browser.runtime.sendMessage({
|
||||||
method: "exemptContainerAssignment",
|
method: "exemptContainerAssignment",
|
||||||
tabId: tab[0].id,
|
tabId: tab.id,
|
||||||
pageUrl: redirectUrl
|
pageUrl: redirectUrl
|
||||||
});
|
});
|
||||||
document.location.replace(redirectUrl);
|
document.location.replace(redirectUrl);
|
||||||
|
@ -103,12 +107,15 @@ load();
|
||||||
|
|
||||||
async function openInContainer(redirectUrl, cookieStoreId) {
|
async function openInContainer(redirectUrl, cookieStoreId) {
|
||||||
const tab = await getCurrentTab();
|
const tab = await getCurrentTab();
|
||||||
await browser.tabs.create({
|
const reopenedTab = await browser.tabs.create({
|
||||||
index: tab[0].index + 1,
|
index: tab.index + 1,
|
||||||
cookieStoreId,
|
cookieStoreId,
|
||||||
url: redirectUrl
|
url: redirectUrl
|
||||||
});
|
});
|
||||||
if (tab.length > 0) {
|
if (tab.groupId >= 0) {
|
||||||
browser.tabs.remove(tab[0].id);
|
// If the original tab was in a tab group, make sure that the reopened tab
|
||||||
|
// stays in the same tab group.
|
||||||
|
await browser.tabs.group({ groupId: tab.groupId, tabIds: reopenedTab.id });
|
||||||
}
|
}
|
||||||
|
await browser.tabs.remove(tab.id);
|
||||||
}
|
}
|
||||||
|
|
|
@ -767,7 +767,6 @@ Logic.registerPanel(P_CONTAINERS_LIST, {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const mozillaVpnToutName = "moz-tout-main-panel";
|
|
||||||
const mozillaVpnPermissionsWarningDotName = "moz-permissions-warning-dot";
|
const mozillaVpnPermissionsWarningDotName = "moz-permissions-warning-dot";
|
||||||
|
|
||||||
let { mozillaVpnHiddenToutsList } = await browser.storage.local.get("mozillaVpnHiddenToutsList");
|
let { mozillaVpnHiddenToutsList } = await browser.storage.local.get("mozillaVpnHiddenToutsList");
|
||||||
|
@ -776,31 +775,6 @@ Logic.registerPanel(P_CONTAINERS_LIST, {
|
||||||
mozillaVpnHiddenToutsList = [];
|
mozillaVpnHiddenToutsList = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Decide whether to show Mozilla VPN tout
|
|
||||||
const mozVpnTout = document.getElementById("moz-vpn-tout");
|
|
||||||
const mozillaVpnInstalled = await browser.runtime.sendMessage({ method: "MozillaVPN_getInstallationStatus" });
|
|
||||||
const mozillaVpnToutShouldBeHidden = mozillaVpnHiddenToutsList.find(tout => tout.name === mozillaVpnToutName);
|
|
||||||
if (mozillaVpnInstalled || mozillaVpnToutShouldBeHidden) {
|
|
||||||
mozVpnTout.remove();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add handlers if tout is visible
|
|
||||||
const mozVpnDismissTout = document.querySelector(".dismiss-moz-vpn-tout");
|
|
||||||
if (mozVpnDismissTout) {
|
|
||||||
Utils.addEnterHandler((mozVpnDismissTout), async() => {
|
|
||||||
mozVpnTout.remove();
|
|
||||||
mozillaVpnHiddenToutsList.push({
|
|
||||||
name: mozillaVpnToutName
|
|
||||||
});
|
|
||||||
await browser.storage.local.set({ mozillaVpnHiddenToutsList });
|
|
||||||
});
|
|
||||||
|
|
||||||
Utils.addEnterHandler(document.querySelector("#moz-vpn-learn-more"), () => {
|
|
||||||
MozillaVPN.handleMozillaCtaClick("mac-main-panel-btn");
|
|
||||||
window.close();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Badge Options icon if both nativeMessaging and/or proxy permissions are disabled
|
// Badge Options icon if both nativeMessaging and/or proxy permissions are disabled
|
||||||
const bothMozillaVpnPermissionsEnabled = await MozillaVPN.bothPermissionsEnabled();
|
const bothMozillaVpnPermissionsEnabled = await MozillaVPN.bothPermissionsEnabled();
|
||||||
const warningDotShouldBeHidden = mozillaVpnHiddenToutsList.find(tout => tout.name === mozillaVpnPermissionsWarningDotName);
|
const warningDotShouldBeHidden = mozillaVpnHiddenToutsList.find(tout => tout.name === mozillaVpnPermissionsWarningDotName);
|
||||||
|
@ -1306,7 +1280,8 @@ Logic.registerPanel(REOPEN_IN_CONTAINER_PICKER, {
|
||||||
false,
|
false,
|
||||||
newUserContextId,
|
newUserContextId,
|
||||||
currentTab.index + 1,
|
currentTab.index + 1,
|
||||||
currentTab.active
|
currentTab.active,
|
||||||
|
currentTab.groupId
|
||||||
);
|
);
|
||||||
window.close();
|
window.close();
|
||||||
};
|
};
|
||||||
|
@ -1336,7 +1311,8 @@ Logic.registerPanel(REOPEN_IN_CONTAINER_PICKER, {
|
||||||
false,
|
false,
|
||||||
0,
|
0,
|
||||||
currentTab.index + 1,
|
currentTab.index + 1,
|
||||||
currentTab.active
|
currentTab.active,
|
||||||
|
currentTab.groupId
|
||||||
);
|
);
|
||||||
window.close();
|
window.close();
|
||||||
});
|
});
|
||||||
|
|
|
@ -94,6 +94,9 @@ const Utils = {
|
||||||
return result.join("");
|
return result.join("");
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {Promise<Tab|false>}
|
||||||
|
*/
|
||||||
async currentTab() {
|
async currentTab() {
|
||||||
const activeTabs = await browser.tabs.query({ active: true, windowId: browser.windows.WINDOW_ID_CURRENT });
|
const activeTabs = await browser.tabs.query({ active: true, windowId: browser.windows.WINDOW_ID_CURRENT });
|
||||||
if (activeTabs.length > 0) {
|
if (activeTabs.length > 0) {
|
||||||
|
@ -146,14 +149,24 @@ const Utils = {
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
async reloadInContainer(url, currentUserContextId, newUserContextId, tabIndex, active) {
|
/**
|
||||||
|
* @param {string} url
|
||||||
|
* @param {string} currentUserContextId
|
||||||
|
* @param {string} newUserContextId
|
||||||
|
* @param {number} tabIndex
|
||||||
|
* @param {boolean} active
|
||||||
|
* @param {number} [groupId]
|
||||||
|
* @returns {Promise<any>}
|
||||||
|
*/
|
||||||
|
async reloadInContainer(url, currentUserContextId, newUserContextId, tabIndex, active, groupId = undefined) {
|
||||||
return await browser.runtime.sendMessage({
|
return await browser.runtime.sendMessage({
|
||||||
method: "reloadInContainer",
|
method: "reloadInContainer",
|
||||||
url,
|
url,
|
||||||
currentUserContextId,
|
currentUserContextId,
|
||||||
newUserContextId,
|
newUserContextId,
|
||||||
tabIndex,
|
tabIndex,
|
||||||
active
|
active,
|
||||||
|
groupId
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -167,7 +180,8 @@ const Utils = {
|
||||||
currentUserContextId: false,
|
currentUserContextId: false,
|
||||||
newUserContextId: assignedUserContextId,
|
newUserContextId: assignedUserContextId,
|
||||||
tabIndex: currentTab.index +1,
|
tabIndex: currentTab.index +1,
|
||||||
active:currentTab.active
|
active: currentTab.active,
|
||||||
|
groupId: currentTab.groupId
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
await Utils.setOrRemoveAssignment(
|
await Utils.setOrRemoveAssignment(
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
{
|
{
|
||||||
"manifest_version": 2,
|
"manifest_version": 2,
|
||||||
"name": "Firefox Multi-Account Containers",
|
"name": "Firefox Multi-Account Containers",
|
||||||
"version": "8.2.0",
|
"version": "8.3.0",
|
||||||
"incognito": "not_allowed",
|
"incognito": "not_allowed",
|
||||||
"description": "__MSG_extensionDescription__",
|
"description": "__MSG_extensionDescription__",
|
||||||
"icons": {
|
"icons": {
|
||||||
|
@ -45,10 +45,6 @@
|
||||||
"description": "__MSG_openContainerPanel__"
|
"description": "__MSG_openContainerPanel__"
|
||||||
},
|
},
|
||||||
"sort_tabs": {
|
"sort_tabs": {
|
||||||
"suggested_key": {
|
|
||||||
"default": "Ctrl+Comma",
|
|
||||||
"mac": "MacCtrl+Comma"
|
|
||||||
},
|
|
||||||
"description": "__MSG_sortTabsByContainer__"
|
"description": "__MSG_sortTabsByContainer__"
|
||||||
},
|
},
|
||||||
"open_container_0": {
|
"open_container_0": {
|
||||||
|
|
|
@ -192,18 +192,7 @@
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
<div id="moz-vpn-tout" class="moz-vpn-content expanded">
|
|
||||||
<div class="flx-row button-wrapper">
|
|
||||||
<h4 class="moz-vpn-logo">Mozilla VPN</h4>
|
|
||||||
<button class="controller dismiss-moz-vpn-tout" tab-index="0"></button>
|
|
||||||
</div>
|
|
||||||
<div class="collapsible-content flx-col controller-collapsible-content">
|
|
||||||
<div class="flx-row flx-space-between">
|
|
||||||
<span class="moz-vpn-subtitle" data-i18n-message-id="integrateContainers"></span>
|
|
||||||
</div>
|
|
||||||
<button id="moz-vpn-learn-more" class="moz-vpn-cta primary-cta" data-i18n-message-id="getMozillaVpn"></button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<v-padding-hack-footer></v-padding-hack-footer> <!--prevents last container from getting covered up by the 'manage containers button' when list is long-->
|
<v-padding-hack-footer></v-padding-hack-footer> <!--prevents last container from getting covered up by the 'manage containers button' when list is long-->
|
||||||
<div class="bottom-btn keyboard-nav controller" id="manage-containers-link" tabindex="0" data-i18n-message-id="manageContainers"></div>
|
<div class="bottom-btn keyboard-nav controller" id="manage-containers-link" tabindex="0" data-i18n-message-id="manageContainers"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -32,6 +32,10 @@ const buildDom = async ({background = {}, popup = {}}) => {
|
||||||
window.crypto = {
|
window.crypto = {
|
||||||
getRandomValues: arr => crypto.randomBytes(arr.length),
|
getRandomValues: arr => crypto.randomBytes(arr.length),
|
||||||
};
|
};
|
||||||
|
// By default, the mock contextMenus.remove() returns undefined;
|
||||||
|
// Let it return a Promise instead, so that .then() calls chained to
|
||||||
|
// it (in src/js/background/assignManager.js) do not fail.
|
||||||
|
window.browser.contextMenus.remove.resolves();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
Loading…
Add table
Reference in a new issue