Compare commits

..

1 commit

Author SHA1 Message Date
Danny Colin
4410245b0d Revert "Fix failing test install by replacing $(npm bin) with npx" 2023-02-22 12:18:22 -05:00
32 changed files with 219 additions and 565 deletions

View file

@ -1,6 +1,6 @@
module.exports = { module.exports = {
"parserOptions": { "parserOptions": {
"ecmaVersion": 2021 "ecmaVersion": 2018
}, },
"env": { "env": {
"browser": true, "browser": true,

View file

@ -9,7 +9,7 @@ body:
options: options:
- label: "I updated to the latest version of Multi-Account Container and tested if I can reproduce the issue" - label: "I updated to the latest version of Multi-Account Container and tested if I can reproduce the issue"
required: true required: true
- label: "I searched for existing reports to see if it hasn't already been reported" - label: "I searched for existing reports to see if it hasn\'t already been reported"
required: true required: true
- type: textarea - type: textarea
id: step_to_reproduce id: step_to_reproduce

View file

@ -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@v4 uses: actions/upload-artifact@v3
with: with:
name: ${{matrix.config.name}} Build name: ${{matrix.config.name}} Build
path: src/web-ext-artifacts path: src/web-ext-artifacts

View file

@ -32,23 +32,7 @@ repository like any other. Before editing files in this folder, you need to:
You can then [open a pull request][pr] on [the l10n repository][l10n]. You can then [open a pull request][pr] on [the l10n repository][l10n].
## Tips for contributing
1. Choose [an issue][issues] that you would like to work on.
2. Fork the repository and follow the instructions for setting it up locally.
3. Run the add-on locally and try reproducing the issue.
4. Debug add-ons by clicking the “Settings” icon in about:addons, and then clicking “Debug Add-ons”
5. Click “Inspect” on the MAC add-on to open developer tools for the popup extension (see [this documentation][extension-doc] for more information)
6. Once you have a fix ready, commit your changes with the following commit message template: “Fix #<insert issue id #>: <short description>
7. Push your changes and open a pull request for review.
If you run into an issue, you can always ask the other community members in the [discussions board][discussions].
<!-- Please keep the list in alphabetical order -->
[discussions]: https://github.com/mozilla/multi-account-containers/discussions
[extension-doc]: https://extensionworkshop.com/documentation/develop/debugging/
[fork]: https://docs.github.com/en/get-started/quickstart/fork-a-repo [fork]: https://docs.github.com/en/get-started/quickstart/fork-a-repo
[issues]: https://github.com/mozilla/multi-account-containers/issues
[l10n]: https://github.com/mozilla-l10n/multi-account-containers-l10n/ [l10n]: https://github.com/mozilla-l10n/multi-account-containers-l10n/
[pr]: https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-pull-requests [pr]: https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-pull-requests
[web-ext]: https://developer.mozilla.org/en-US/Add-ons/WebExtensions/Getting_started_with_web-ext [web-ext]: https://developer.mozilla.org/en-US/Add-ons/WebExtensions/Getting_started_with_web-ext

View file

@ -13,7 +13,7 @@ Everyone is welcome to contribute to Multi-Account Containers. To learn how
to contribute a patch to Multi-Account Container, please to contribute a patch to Multi-Account Container, please
[read our contributing guide][contributing]. [read our contributing guide][contributing].
You can also chat with us on [our Matrix room][matrix] or ask in [our discussions board][discussions]. You can also chat with us on [our Matrix room][matrix] or [our forum][forum].
This repository is governed by Mozilla's code of conduct and etiquette This repository is governed by Mozilla's code of conduct and etiquette
guidelines. For more details, [please read the Mozilla Community Participation Guidelines][cpg]. guidelines. For more details, [please read the Mozilla Community Participation Guidelines][cpg].
@ -29,5 +29,4 @@ file, You can obtain one at https://mozilla.org/MPL/2.0/.
[cpg]: https://www.mozilla.org/about/governance/policies/participation/ [cpg]: https://www.mozilla.org/about/governance/policies/participation/
[enduser]: https://support.mozilla.org/en-US/kb/containers [enduser]: https://support.mozilla.org/en-US/kb/containers
[forum]: https://discourse.mozilla.org/c/containers/223 [forum]: https://discourse.mozilla.org/c/containers/223
[discussions]: https://github.com/mozilla/multi-account-containers/discussions
[matrix]: https://matrix.to/#/#containers:mozilla.org [matrix]: https://matrix.to/#/#containers:mozilla.org

View file

@ -1,4 +1,4 @@
#!/bin/env bash #!/bin/bash
# This Source Code Form is subject to the terms of the Mozilla Public # This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this # License, v. 2.0. If a copy of the MPL was not distributed with this
@ -32,4 +32,4 @@ rm -rf $TMPDIR/src/_locales/.github || die
print G "done." print G "done."
print Y "Running the test..." print Y "Running the test..."
npx addons-linter $TMPDIR/src || die $(npm bin)/addons-linter $TMPDIR/src || die

View file

@ -1,4 +1,4 @@
#!/bin/env bash #!/bin/bash
# This Source Code Form is subject to the terms of the Mozilla Public # This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this # License, v. 2.0. If a copy of the MPL was not distributed with this
@ -23,4 +23,4 @@ if [[ $# -gt 0 ]]; then
EXTRA_PARAMS="--filename $1" EXTRA_PARAMS="--filename $1"
fi fi
npx web-ext build --overwrite-dest $EXTRA_PARAMS || die $(npm bin)/web-ext build --overwrite-dest $EXTRA_PARAMS || die

View file

@ -1,4 +1,4 @@
#!/bin/env bash #!/bin/bash
# This Source Code Form is subject to the terms of the Mozilla Public # This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this # License, v. 2.0. If a copy of the MPL was not distributed with this

18
package-lock.json generated
View file

@ -1,6 +1,6 @@
{ {
"name": "testpilot-containers", "name": "testpilot-containers",
"version": "8.1.3", "version": "8.1.1",
"lockfileVersion": 2, "lockfileVersion": 2,
"requires": true, "requires": true,
"packages": { "packages": {
@ -2226,14 +2226,14 @@
} }
}, },
"node_modules/cacheable-request": { "node_modules/cacheable-request": {
"version": "10.2.7", "version": "10.2.3",
"resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-10.2.7.tgz", "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-10.2.3.tgz",
"integrity": "sha512-I4SA6mKgDxcxVbSt/UmIkb9Ny8qSkg6ReBHtAAXnZHk7KOSx5g3DTiAOaYzcHCs6oOdHn+bip9T48E6tMvK9hw==", "integrity": "sha512-6BehRBOs7iurNjAYN9iPazTwFDaMQavJO8W1MEm3s2pH8q/tkPTtLDRUZaweWK87WFGf2Y5wLAlaCJlR5kOz3w==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"@types/http-cache-semantics": "^4.0.1", "@types/http-cache-semantics": "^4.0.1",
"get-stream": "^6.0.1", "get-stream": "^6.0.1",
"http-cache-semantics": "^4.1.1", "http-cache-semantics": "^4.1.0",
"keyv": "^4.5.2", "keyv": "^4.5.2",
"mimic-response": "^4.0.0", "mimic-response": "^4.0.0",
"normalize-url": "^8.0.0", "normalize-url": "^8.0.0",
@ -15922,14 +15922,14 @@
"dev": true "dev": true
}, },
"cacheable-request": { "cacheable-request": {
"version": "10.2.7", "version": "10.2.3",
"resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-10.2.7.tgz", "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-10.2.3.tgz",
"integrity": "sha512-I4SA6mKgDxcxVbSt/UmIkb9Ny8qSkg6ReBHtAAXnZHk7KOSx5g3DTiAOaYzcHCs6oOdHn+bip9T48E6tMvK9hw==", "integrity": "sha512-6BehRBOs7iurNjAYN9iPazTwFDaMQavJO8W1MEm3s2pH8q/tkPTtLDRUZaweWK87WFGf2Y5wLAlaCJlR5kOz3w==",
"dev": true, "dev": true,
"requires": { "requires": {
"@types/http-cache-semantics": "^4.0.1", "@types/http-cache-semantics": "^4.0.1",
"get-stream": "^6.0.1", "get-stream": "^6.0.1",
"http-cache-semantics": "^4.1.1", "http-cache-semantics": "^4.1.0",
"keyv": "^4.5.2", "keyv": "^4.5.2",
"mimic-response": "^4.0.0", "mimic-response": "^4.0.0",
"normalize-url": "^8.0.0", "normalize-url": "^8.0.0",

View file

@ -2,14 +2,14 @@
"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.3.0", "version": "8.1.1",
"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"
}, },
"dependencies": {}, "dependencies": {},
"devDependencies": { "devDependencies": {
"addons-linter": "^5.28.0", "addons-linter": "^3.23.0",
"ajv": "^6.6.3", "ajv": "^6.6.3",
"chai": "^4.2.0", "chai": "^4.2.0",
"eslint": "^7.32.0", "eslint": "^7.32.0",

@ -1 +1 @@
Subproject commit bdaa01291b7367a5e815470fd263ea36c862fe32 Subproject commit fa5fb497bd4e5c9b7d69407ee593b853cf839009

View file

@ -110,6 +110,7 @@
--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"] {
@ -179,6 +180,7 @@
--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 */
@ -226,7 +228,6 @@ body {
/* Hack for menu icons to use a light color without affecting container icons */ /* Hack for menu icons to use a light color without affecting container icons */
[data-theme="light"] img.delete-assignment, [data-theme="light"] img.delete-assignment,
[data-theme="dark"] img.reset-assignment,
[data-theme="dark"] .trash-button, [data-theme="dark"] .trash-button,
[data-theme="dark"] img.menu-icon, [data-theme="dark"] img.menu-icon,
[data-theme="dark"] .menu-icon > img, [data-theme="dark"] .menu-icon > img,
@ -235,7 +236,6 @@ body {
filter: invert(1); filter: invert(1);
} }
[data-theme="dark"] img.clear-storage-icon,
[data-theme="dark"] img.delete-assignment, [data-theme="dark"] img.delete-assignment,
[data-theme="dark"] #edit-sites-assigned .menu-icon, [data-theme="dark"] #edit-sites-assigned .menu-icon,
[data-theme="dark"] #container-info-table .menu-icon { [data-theme="dark"] #container-info-table .menu-icon {
@ -285,33 +285,9 @@ table {
display: none !important; display: none !important;
} }
.popup-notification-card {
opacity: 0;
pointer-events: none;
transition: opacity 2s;
border-radius: 4px;
font-size: 12px;
position: absolute;
inset-block-start: 0;
inset-inline-start: 0;
padding-block: 8px;
padding-inline: 8px;
margin-block: 8px;
margin-inline: 8px;
inline-size: calc(100vw - 25px);
background-color: var(--button-bg-active-color-secondary);
z-index: 3;
}
.is-shown {
pointer-events: auto;
opacity: 1;
transition: opacity 0s;
}
/* effect borrowed from tabs in firefox, ensure that the element flexes to the full width */ /* effect borrowed from tabs in firefox, ensure that the element flexes to the full width */
.truncate-text { .truncate-text {
inline-size: calc(100vw - 100px); inline-size: calc(100vw - 80px);
overflow: hidden; overflow: hidden;
position: relative; position: relative;
white-space: nowrap; white-space: nowrap;
@ -633,8 +609,37 @@ 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;
@ -1022,6 +1027,7 @@ 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);
@ -1499,9 +1505,8 @@ input[type=text] {
min-block-size: 500px; min-block-size: 500px;
} }
.delete-container-panel, .delete-container-panel {
.clear-container-storage-panel { min-block-size: 300px;
min-block-size: 500px;
} }
.panel.onboarding, .panel.onboarding,
@ -1789,14 +1794,12 @@ manage things like container crud */
margin-inline-end: 0; margin-inline-end: 0;
} }
.delete-container-confirm, .delete-container-confirm {
.clear-container-storage-confirm {
padding-inline-end: 20px; padding-inline-end: 20px;
padding-inline-start: 20px; padding-inline-start: 20px;
} }
.delete-container-confirm-title, .delete-container-confirm-title {
.clear-container-storage-confirm-title {
color: var(--text-color-primary); color: var(--text-color-primary);
font-size: var(--font-size-heading); font-size: var(--font-size-heading);
} }
@ -2170,11 +2173,6 @@ hr {
text-align: center; text-align: center;
} }
.confirmation-destructive-ok-btn {
background-color: var(--button-destructive-bg-color);
color: var(--button-destructive-text-color);
}
.delete-btn { .delete-btn {
background-color: var(--button-destructive-bg-color); background-color: var(--button-destructive-bg-color);
block-size: 32px; block-size: 32px;
@ -2305,8 +2303,7 @@ input {
font-weight: bolder; font-weight: bolder;
} }
.delete-warning, .delete-warning {
.clear-container-storage-warning {
padding-block-end: 8px; padding-block-end: 8px;
padding-block-start: 8px; padding-block-start: 8px;
padding-inline-end: 0; padding-inline-end: 0;
@ -2317,8 +2314,7 @@ input {
* rules grouped together at the beginning of the file * rules grouped together at the beginning of the file
*/ */
/* stylelint-disable no-descending-specificity */ /* stylelint-disable no-descending-specificity */
.trash-button, .trash-button {
.reset-button {
display: inline-block; display: inline-block;
block-size: 20px; block-size: 20px;
inline-size: 20px; inline-size: 20px;
@ -2327,21 +2323,11 @@ input {
text-align: center; text-align: center;
} }
.reset-button { tr > td > .trash-button {
margin-inline-end: 12px;
}
.tooltip-wrapper:hover .site-settings-tooltip {
display: block;
}
tr > td > .trash-button,
tr > td > .reset-button {
display: none; display: none;
} }
tr:hover > td > .trash-button, tr:hover > td > .trash-button {
tr:hover > td > .reset-button {
display: block; display: block;
} }

Binary file not shown.

After

Width:  |  Height:  |  Size: 342 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 578 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1 KiB

View file

@ -0,0 +1,9 @@
<!-- This Source Code Form is subject to the terms of the Mozilla Public
- License, v. 2.0. If a copy of the MPL was not distributed with this
- file, You can obtain one at http://mozilla.org/MPL/2.0/. --><svg data-name="Flat (For Export)" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16">
<style>rect,path {fill: rgba(249, 249, 250, 0.8);}</style>
<rect x="1" y="1" width="6" height="6" rx="1"/>
<path d="M14.75 3H13V1.25A0.25 0.25 0 0 0 12.75 1h-1.5A0.25 0.25 0 0 0 11 1.25V3H9.25A0.25 0.25 0 0 0 9 3.25v1.5A0.25 0.25 0 0 0 9.25 5H11v1.75A0.25 0.25 0 0 0 11.25 7h1.5A0.25 0.25 0 0 0 13 6.75V5h1.75A0.25 0.25 0 0 0 15 4.75v-1.5A0.25 0.25 0 0 0 14.75 3z" fill-rule="evenodd"/>
<rect x="1" y="9" width="6" height="6" rx="1"/>
<rect x="9" y="9" width="6" height="6" rx="1"/>
</svg>

After

Width:  |  Height:  |  Size: 801 B

View file

@ -1,13 +1,7 @@
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 16 16"> <svg data-name="Flat (For Export)" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16">
<style> <style>rect,path {fill: rgba(24, 25, 26, 01);}</style>
:root { color-scheme: light dark; }
rect, path { fill: rgb(24, 25, 26); }
@media (prefers-color-scheme: dark) {
rect, path { fill: rgba(249, 249, 250, 0.8); }
}
</style>
<rect x="1" y="1" width="6" height="6" rx="1"/> <rect x="1" y="1" width="6" height="6" rx="1"/>
<path d="M14.75 3H13V1.25A0.25 0.25 0 0 0 12.75 1h-1.5A0.25 0.25 0 0 0 11 1.25V3H9.25A0.25 0.25 0 0 0 9 3.25v1.5A0.25 0.25 0 0 0 9.25 5H11v1.75A0.25 0.25 0 0 0 11.25 7h1.5A0.25 0.25 0 0 0 13 6.75V5h1.75A0.25 0.25 0 0 0 15 4.75v-1.5A0.25 0.25 0 0 0 14.75 3z" fill-rule="evenodd"/> <path d="M14.75 3H13V1.25A0.25 0.25 0 0 0 12.75 1h-1.5A0.25 0.25 0 0 0 11 1.25V3H9.25A0.25 0.25 0 0 0 9 3.25v1.5A0.25 0.25 0 0 0 9.25 5H11v1.75A0.25 0.25 0 0 0 11.25 7h1.5A0.25 0.25 0 0 0 13 6.75V5h1.75A0.25 0.25 0 0 0 15 4.75v-1.5A0.25 0.25 0 0 0 14.75 3z" fill-rule="evenodd"/>
<rect x="1" y="9" width="6" height="6" rx="1"/> <rect x="1" y="9" width="6" height="6" rx="1"/>
<rect x="9" y="9" width="6" height="6" rx="1"/> <rect x="9" y="9" width="6" height="6" rx="1"/>
</svg> </svg>

Before

Width:  |  Height:  |  Size: 727 B

After

Width:  |  Height:  |  Size: 586 B

View file

@ -4,7 +4,6 @@
- file, You can obtain one at http://mozilla.org/MPL/2.0/. --> - file, You can obtain one at http://mozilla.org/MPL/2.0/. -->
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="32" height="32" viewBox="0 0 32 32" style="enable-background:new 0 0 32 32;"> <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="32" height="32" viewBox="0 0 32 32" style="enable-background:new 0 0 32 32;">
<style> <style>
:root { color-scheme: light dark; }
path, circle, g { path, circle, g {
fill: menutext; fill: menutext;
} }

Before

Width:  |  Height:  |  Size: 11 KiB

After

Width:  |  Height:  |  Size: 11 KiB

View file

@ -61,9 +61,8 @@ window.assignManager = {
this.area.get([siteStoreKey]).then((storageResponse) => { this.area.get([siteStoreKey]).then((storageResponse) => {
if (storageResponse && siteStoreKey in storageResponse) { if (storageResponse && siteStoreKey in storageResponse) {
resolve(storageResponse[siteStoreKey]); resolve(storageResponse[siteStoreKey]);
} else {
resolve(null);
} }
resolve(null);
}).catch((e) => { }).catch((e) => {
reject(e); reject(e);
}); });
@ -166,17 +165,11 @@ window.assignManager = {
_neverAsk(m) { _neverAsk(m) {
const pageUrl = m.pageUrl; const pageUrl = m.pageUrl;
if (m.neverAsk === true) { if (m.neverAsk === true) {
if (m.defaultContainer === true) {
this.storageArea.remove(pageUrl);
return;
}
// If we have existing data and for some reason it hasn't been // If we have existing data and for some reason it hasn't been
// deleted etc lets update it // deleted etc lets update it
this.storageArea.get(pageUrl).then((siteSettings) => { this.storageArea.get(pageUrl).then((siteSettings) => {
if (siteSettings) { if (siteSettings) {
siteSettings.neverAsk = true; siteSettings.neverAsk = true;
siteSettings.userContextId = backgroundLogic.getUserContextIdFromCookieStoreId(m.cookieStoreId);
this.storageArea.set(pageUrl, siteSettings); this.storageArea.set(pageUrl, siteSettings);
} }
}).catch((e) => { }).catch((e) => {
@ -326,8 +319,7 @@ 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(
@ -337,8 +329,7 @@ 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);
@ -483,7 +474,9 @@ window.assignManager = {
}, },
contextualIdentityRemoved(changeInfo) { contextualIdentityRemoved(changeInfo) {
this.removeMenuItem(changeInfo.contextualIdentity.cookieStoreId); browser.contextMenus.remove(
changeInfo.contextualIdentity.cookieStoreId
);
}, },
async _onClickedHandler(info, tab) { async _onClickedHandler(info, tab) {
@ -578,16 +571,6 @@ window.assignManager = {
return true; return true;
}, },
async _resetCookiesForSite(hostname, cookieStoreId) {
const hostNameTruncated = hostname.replace(/^www\./, ""); // Remove "www." from the hostname
await browser.browsingData.removeCookies({
cookieStoreId: cookieStoreId,
hostnames: [hostNameTruncated] // This does not remove cookies from associated domains. To remove all cookies, we have a container storage removal option.
});
return true;
},
async _setOrRemoveAssignment(tabId, pageUrl, userContextId, remove) { async _setOrRemoveAssignment(tabId, pageUrl, userContextId, remove) {
let actionName; let actionName;
// https://github.com/mozilla/testpilot-containers/issues/626 // https://github.com/mozilla/testpilot-containers/issues/626
@ -639,7 +622,7 @@ window.assignManager = {
}, },
async _maybeRemoveSiteIsolation(userContextId) { async _maybeRemoveSiteIsolation(userContextId) {
const assignments = await this.storageArea.getAssignedSites(userContextId); const assignments = await this.storageArea.getByContainer(userContextId);
const hasAssignments = assignments && Object.keys(assignments).length > 0; const hasAssignments = assignments && Object.keys(assignments).length > 0;
if (hasAssignments) { if (hasAssignments) {
return; return;
@ -670,11 +653,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
this.removeMenuItem(this.MENU_ASSIGN_ID); browser.contextMenus.remove(this.MENU_ASSIGN_ID);
this.removeMenuItem(this.MENU_REMOVE_ID); browser.contextMenus.remove(this.MENU_REMOVE_ID);
this.removeMenuItem(this.MENU_SEPARATOR_ID); browser.contextMenus.remove(this.MENU_SEPARATOR_ID);
this.removeMenuItem(this.MENU_HIDE_ID); browser.contextMenus.remove(this.MENU_HIDE_ID);
this.removeMenuItem(this.MENU_MOVE_ID); browser.contextMenus.remove(this.MENU_MOVE_ID);
}, },
async calculateContextMenu(tab) { async calculateContextMenu(tab) {
@ -695,7 +678,7 @@ window.assignManager = {
} }
browser.contextMenus.create({ browser.contextMenus.create({
id: menuId, id: menuId,
title: browser.i18n.getMessage("alwaysOpenSiteInContainer"), title: "Always Open in This Container",
checked, checked,
type: "checkbox", type: "checkbox",
contexts: ["all"], contexts: ["all"],
@ -709,13 +692,13 @@ window.assignManager = {
browser.contextMenus.create({ browser.contextMenus.create({
id: this.MENU_HIDE_ID, id: this.MENU_HIDE_ID,
title: browser.i18n.getMessage("hideThisContainer"), title: "Hide This Container",
contexts: ["all"], contexts: ["all"],
}); });
browser.contextMenus.create({ browser.contextMenus.create({
id: this.MENU_MOVE_ID, id: this.MENU_MOVE_ID,
title: browser.i18n.getMessage("moveTabsToANewWindow"), title: "Move Tabs to a New Window",
contexts: ["all"], contexts: ["all"],
}); });
}, },
@ -727,15 +710,7 @@ 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.
// //
@ -754,58 +729,16 @@ 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";
this.createTabWrapper(url, cookieStoreId, index, active, openerTabId, groupId); browser.tabs.create({url, cookieStoreId, index, active, openerTabId});
}, },
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 this.createTabWrapper(url, cookieStoreId, index, active, openerTabId, groupId); return browser.tabs.create({url, cookieStoreId, index, active, openerTabId});
} else { } else {
let confirmUrl = `${loadPage}?url=${this.encodeURLProperty(url)}&cookieStoreId=${cookieStoreId}`; let confirmUrl = `${loadPage}?url=${this.encodeURLProperty(url)}&cookieStoreId=${cookieStoreId}`;
let currentCookieStoreId; let currentCookieStoreId;
@ -813,14 +746,13 @@ window.assignManager = {
currentCookieStoreId = backgroundLogic.cookieStoreId(currentUserContextId); currentCookieStoreId = backgroundLogic.cookieStoreId(currentUserContextId);
confirmUrl += `&currentCookieStoreId=${currentCookieStoreId}`; confirmUrl += `&currentCookieStoreId=${currentCookieStoreId}`;
} }
return this.createTabWrapper( return browser.tabs.create({
confirmUrl, url: confirmUrl,
currentCookieStoreId, cookieStoreId: currentCookieStoreId,
index,
active,
openerTabId, openerTabId,
groupId index,
).then(() => { active
}).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) => {
@ -832,7 +764,7 @@ window.assignManager = {
async initBookmarksMenu() { async initBookmarksMenu() {
browser.contextMenus.create({ browser.contextMenus.create({
id: this.OPEN_IN_CONTAINER, id: this.OPEN_IN_CONTAINER,
title: browser.i18n.getMessage("openBookmarkInContainerTab"), title: "Open Bookmark in Container Tab",
contexts: ["bookmark"], contexts: ["bookmark"],
}); });
@ -848,19 +780,12 @@ window.assignManager = {
}, },
async removeBookmarksMenu() { async removeBookmarksMenu() {
this.removeMenuItem(this.OPEN_IN_CONTAINER); browser.contextMenus.remove(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) {
this.removeMenuItem(identity.cookieStoreId); browser.contextMenus.remove(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();

View file

@ -14,13 +14,7 @@ const backgroundLogic = {
NUMBER_OF_KEYBOARD_SHORTCUTS: 10, NUMBER_OF_KEYBOARD_SHORTCUTS: 10,
unhideQueue: [], unhideQueue: [],
init() { init() {
browser.commands.onCommand.addListener(function (command) { browser.commands.onCommand.addListener(function (command) {
if (command === "sort_tabs") {
backgroundLogic.sortTabs();
return;
}
for (let i=0; i < backgroundLogic.NUMBER_OF_KEYBOARD_SHORTCUTS; i++) { for (let i=0; i < backgroundLogic.NUMBER_OF_KEYBOARD_SHORTCUTS; i++) {
const key = "open_container_" + i; const key = "open_container_" + i;
const cookieStoreId = identityState.keyboardShortcut[key]; const cookieStoreId = identityState.keyboardShortcut[key];
@ -33,49 +27,6 @@ const backgroundLogic = {
browser.permissions.onAdded.addListener(permissions => this.resetPermissions(permissions)); browser.permissions.onAdded.addListener(permissions => this.resetPermissions(permissions));
browser.permissions.onRemoved.addListener(permissions => this.resetPermissions(permissions)); browser.permissions.onRemoved.addListener(permissions => this.resetPermissions(permissions));
// Update Translation in Manifest
browser.runtime.onInstalled.addListener((details) => {
this.updateTranslationInManifest();
this._undoDefault820SortTabsKeyboardShortcut(details);
});
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() {
for (let index = 0; index < 10; index++) {
const ajustedIndex = index + 1; // We want to start from 1 instead of 0 in the UI.
browser.commands.update({
name: `open_container_${index}`,
description: browser.i18n.getMessage("containerShortcut", `${ajustedIndex}`)
});
}
}, },
resetPermissions(permissions) { resetPermissions(permissions) {
@ -104,19 +55,6 @@ const backgroundLogic = {
return extensionInfo; return extensionInfo;
}, },
// Remove container data (cookies, localStorage and cache)
async deleteContainerDataOnly(userContextId) {
await browser.browsingData.removeCookies({
cookieStoreId: this.cookieStoreId(userContextId)
});
await browser.browsingData.removeLocalStorage({
cookieStoreId: this.cookieStoreId(userContextId)
});
return {done: true, userContextId};
},
getUserContextIdFromCookieStoreId(cookieStoreId) { getUserContextIdFromCookieStoreId(cookieStoreId) {
if (!cookieStoreId) { if (!cookieStoreId) {
return false; return false;
@ -185,8 +123,7 @@ const backgroundLogic = {
// We can't open these we just have to throw them away // We can't open these we just have to throw them away
if (protocol === "about:" if (protocol === "about:"
|| protocol === "chrome:" || protocol === "chrome:"
|| protocol === "moz-extension:" || protocol === "moz-extension:") {
|| protocol === "file:") {
return false; return false;
} }
return true; return true;
@ -372,13 +309,7 @@ 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.
@ -391,11 +322,6 @@ 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: [] });
@ -417,25 +343,15 @@ 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.
for (const { tabs } of sortMap.values()) { sortMap.forEach(obj => {
for (const tab of tabs) { for (const tab of obj.tabs) {
++pos; ++pos;
browser.tabs.move(tab.id, { browser.tabs.move(tab.id, {
windowId: windowObj.id, windowId: windowObj.id,
index: pinnedTabs ? pos : -1 index: pos
}); });
// 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) {

View file

@ -23,9 +23,6 @@ const messageHandler = {
case "deleteContainer": case "deleteContainer":
response = backgroundLogic.deleteContainer(m.message.userContextId); response = backgroundLogic.deleteContainer(m.message.userContextId);
break; break;
case "deleteContainerDataOnly":
response = backgroundLogic.deleteContainerDataOnly(m.message.userContextId);
break;
case "createOrUpdateContainer": case "createOrUpdateContainer":
response = backgroundLogic.createOrUpdateContainer(m.message); response = backgroundLogic.createOrUpdateContainer(m.message);
break; break;
@ -48,9 +45,6 @@ const messageHandler = {
// m.url is the assignment to be removed/added // m.url is the assignment to be removed/added
response = assignManager._setOrRemoveAssignment(m.tabId, m.url, m.userContextId, m.value); response = assignManager._setOrRemoveAssignment(m.tabId, m.url, m.userContextId, m.value);
break; break;
case "resetCookiesForSite":
response = assignManager._resetCookiesForSite(m.pageUrl, m.cookieStoreId);
break;
case "sortTabs": case "sortTabs":
backgroundLogic.sortTabs(); backgroundLogic.sortTabs();
break; break;
@ -91,21 +85,17 @@ 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":
tab = await assignManager.reloadPageInContainer( tab = await assignManager.reloadPageInContainer(
m.url, m.url,
m.currentUserContextId, m.currentUserContextId,
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
@ -230,9 +220,7 @@ const messageHandler = {
// if it's a container tab wait for it to complete and // if it's a container tab wait for it to complete and
// unhide other tabs from this container // unhide other tabs from this container
if (tab.cookieStoreId.startsWith("firefox-container")) { if (tab.cookieStoreId.startsWith("firefox-container")) {
browser.tabs.onUpdated.addListener(this.tabUpdateHandler, { browser.tabs.onUpdated.addListener(this.tabUpdateHandler);
properties: ["status"]
});
} }
} }
} }

View file

@ -7,16 +7,14 @@ async function load() {
redirectUrlElement.textContent = redirectUrl; redirectUrlElement.textContent = redirectUrl;
appendFavicon(redirectUrl, redirectUrlElement); appendFavicon(redirectUrl, redirectUrlElement);
// Option for staying on the previous container
document.getElementById("deny").addEventListener("click", (e) => { document.getElementById("deny").addEventListener("click", (e) => {
e.preventDefault(); e.preventDefault();
denySubmit(redirectUrl, currentCookieStoreId); denySubmit(redirectUrl);
}); });
// Option for going to the default container (no container)
document.getElementById("deny-no-container").addEventListener("click", (e) => { document.getElementById("deny-no-container").addEventListener("click", (e) => {
e.preventDefault(); e.preventDefault();
denySubmit(redirectUrl, currentCookieStoreId); denySubmit(redirectUrl);
}); });
const container = await browser.contextualIdentities.get(cookieStoreId); const container = await browser.contextualIdentities.get(cookieStoreId);
@ -29,7 +27,6 @@ async function load() {
el.textContent = browser.i18n.getMessage(elementData.messageId, containerName); el.textContent = browser.i18n.getMessage(elementData.messageId, containerName);
}); });
// Option for going to newly selected container
document.getElementById("confirm").addEventListener("click", (e) => { document.getElementById("confirm").addEventListener("click", (e) => {
e.preventDefault(); e.preventDefault();
confirmSubmit(redirectUrl, cookieStoreId); confirmSubmit(redirectUrl, cookieStoreId);
@ -62,42 +59,24 @@ function confirmSubmit(redirectUrl, cookieStoreId) {
browser.runtime.sendMessage({ browser.runtime.sendMessage({
method: "neverAsk", method: "neverAsk",
neverAsk: true, neverAsk: true,
cookieStoreId: cookieStoreId,
pageUrl: redirectUrl pageUrl: redirectUrl
}); });
} }
openInContainer(redirectUrl, cookieStoreId); openInContainer(redirectUrl, cookieStoreId);
} }
/** function getCurrentTab() {
* @returns {Promise<Tab>} return browser.tabs.query({
*/
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) {
const tab = await getCurrentTab(); const tab = await getCurrentTab();
const currentContainer = currentCookieStoreId ? await browser.contextualIdentities.get(currentCookieStoreId) : null;
const neverAsk = document.getElementById("never-ask").checked;
if (neverAsk) {
await browser.runtime.sendMessage({
method: "neverAsk",
neverAsk: true,
cookieStoreId: currentCookieStoreId,
pageUrl: redirectUrl,
defaultContainer: !currentContainer
});
}
await browser.runtime.sendMessage({ await browser.runtime.sendMessage({
method: "exemptContainerAssignment", method: "exemptContainerAssignment",
tabId: tab.id, tabId: tab[0].id,
pageUrl: redirectUrl pageUrl: redirectUrl
}); });
document.location.replace(redirectUrl); document.location.replace(redirectUrl);
@ -107,15 +86,12 @@ load();
async function openInContainer(redirectUrl, cookieStoreId) { async function openInContainer(redirectUrl, cookieStoreId) {
const tab = await getCurrentTab(); const tab = await getCurrentTab();
const reopenedTab = await browser.tabs.create({ await browser.tabs.create({
index: tab.index + 1, index: tab[0].index + 1,
cookieStoreId, cookieStoreId,
url: redirectUrl url: redirectUrl
}); });
if (tab.groupId >= 0) { if (tab.length > 0) {
// If the original tab was in a tab group, make sure that the reopened tab browser.tabs.remove(tab[0].id);
// stays in the same tab group.
await browser.tabs.group({ groupId: tab.groupId, tabIds: reopenedTab.id });
} }
await browser.tabs.remove(tab.id);
} }

View file

@ -24,12 +24,11 @@ async function addMessage(message) {
divElement.innerText = message.text; divElement.innerText = message.text;
const imageElement = document.createElement("img"); const imageElement = document.createElement("img");
const imagePath = browser.runtime.getURL("/img/multiaccountcontainer-16.svg"); const imagePath = browser.runtime.getURL("/img/container-site-d-24.png");
const response = await fetch(imagePath); const response = await fetch(imagePath);
const blob = await response.blob(); const blob = await response.blob();
const objectUrl = URL.createObjectURL(blob); const objectUrl = URL.createObjectURL(blob);
imageElement.src = objectUrl; imageElement.src = objectUrl;
imageElement.width = imageElement.height = 24;
divElement.prepend(imageElement); divElement.prepend(imageElement);
document.body.appendChild(divElement); document.body.appendChild(divElement);

View file

@ -32,7 +32,6 @@ const P_CONTAINER_EDIT = "containerEdit";
const P_CONTAINER_DELETE = "containerDelete"; const P_CONTAINER_DELETE = "containerDelete";
const P_CONTAINERS_ACHIEVEMENT = "containersAchievement"; const P_CONTAINERS_ACHIEVEMENT = "containersAchievement";
const P_CONTAINER_ASSIGNMENTS = "containerAssignments"; const P_CONTAINER_ASSIGNMENTS = "containerAssignments";
const P_CLEAR_CONTAINER_STORAGE = "clearContainerStorage";
const P_MOZILLA_VPN_SERVER_LIST = "moz-vpn-server-list"; const P_MOZILLA_VPN_SERVER_LIST = "moz-vpn-server-list";
const P_ADVANCED_PROXY_SETTINGS = "advanced-proxy-settings-panel"; const P_ADVANCED_PROXY_SETTINGS = "advanced-proxy-settings-panel";
@ -123,19 +122,6 @@ const Logic = {
}, },
notify(i18nOpts) {
const notificationCards = document.querySelectorAll(".popup-notification-card");
const text = browser.i18n.getMessage(i18nOpts.messageId, i18nOpts.placeholders);
notificationCards.forEach(notificationCard => {
notificationCard.textContent = text;
notificationCard.classList.add("is-shown");
setTimeout(() => {
notificationCard.classList.remove("is-shown");
}, 2000);
});
},
async showAchievementOrContainersListPanel() { async showAchievementOrContainersListPanel() {
// Do we need to show an achievement panel? // Do we need to show an achievement panel?
let showAchievements = false; let showAchievements = false;
@ -225,11 +211,6 @@ const Logic = {
async saveContainerOrder(rows) { async saveContainerOrder(rows) {
const containerOrder = {}; const containerOrder = {};
rows.forEach((node, index) => { rows.forEach((node, index) => {
if (typeof browser.contextualIdentities.move === "function") {
browser.contextualIdentities.move(
node.dataset.containerId, index);
}
return containerOrder[node.dataset.containerId] = index; return containerOrder[node.dataset.containerId] = index;
}); });
await browser.storage.local.set({ await browser.storage.local.set({
@ -415,11 +396,7 @@ const Logic = {
}, },
shortcutListener(e){ shortcutListener(e){
function openTopContainers() { function openNewContainerTab(identity) {
const identities = Logic.identities();
const key = e.code.substring(5);
const identity = e.code === "Digit0" ? identities[9] : identities[key - 1];
try { try {
browser.tabs.create({ browser.tabs.create({
cookieStoreId: identity.cookieStoreId cookieStoreId: identity.cookieStoreId
@ -429,34 +406,12 @@ const Logic = {
window.close(); window.close();
} }
} }
const identities = Logic.identities();
// We monitor if the search input is focused so we can disable opening if ((e.keyCode >= 49 && e.keyCode <= 57) &&
// containers by typing a digit between 0-9 while the popup is open. Logic._currentPanel === "containersList") {
const searchInput = document.getElementById("search-terms"); const identity = identities[e.keyCode - 49];
let isSearchInputFocused = false; if (identity) {
openNewContainerTab(identity);
if (document.activeElement === searchInput) {
isSearchInputFocused = true;
}
if (Logic._currentPanel === "containersList" && !isSearchInputFocused) {
switch(e.code) {
case "Digit0":
case "Digit1":
case "Digit2":
case "Digit3":
case "Digit4":
case "Digit5":
case "Digit6":
case "Digit7":
case "Digit8":
case "Digit9":
openTopContainers();
break;
case "Slash":
document.getElementById("search-terms").focus();
e.preventDefault();
break;
} }
} }
}, },
@ -685,7 +640,7 @@ Logic.registerPanel(P_ONBOARDING_7, {
// Let's move to the containers list panel. // Let's move to the containers list panel.
Utils.addEnterHandler(document.querySelector("#sign-in"), async () => { Utils.addEnterHandler(document.querySelector("#sign-in"), async () => {
browser.tabs.create({ browser.tabs.create({
url: "https://accounts.firefox.com/?service=sync&action=email&context=fx_desktop_v3&entrypoint=multi-account-containers&utm_source=addon&utm_medium=panel&utm_campaign=container-sync&brand=mozilla", url: "https://accounts.firefox.com/?service=sync&action=email&context=fx_desktop_v3&entrypoint=multi-account-containers&utm_source=addon&utm_medium=panel&utm_campaign=container-sync",
}); });
await Logic.setOnboardingStage(7); await Logic.setOnboardingStage(7);
Logic.showPanel(P_ONBOARDING_8); Logic.showPanel(P_ONBOARDING_8);
@ -767,6 +722,7 @@ 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");
@ -775,6 +731,31 @@ 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);
@ -959,7 +940,6 @@ Logic.registerPanel(P_CONTAINER_INFO, {
Utils.alwaysOpenInContainer(identity); Utils.alwaysOpenInContainer(identity);
window.close(); window.close();
}); });
// Show or not the has-tabs section. // Show or not the has-tabs section.
for (let trHasTabs of document.getElementsByClassName("container-info-has-tabs")) { // eslint-disable-line prefer-const for (let trHasTabs of document.getElementsByClassName("container-info-has-tabs")) { // eslint-disable-line prefer-const
trHasTabs.style.display = !identity.hasHiddenTabs && !identity.hasOpenTabs ? "none" : ""; trHasTabs.style.display = !identity.hasHiddenTabs && !identity.hasOpenTabs ? "none" : "";
@ -983,13 +963,6 @@ Logic.registerPanel(P_CONTAINER_INFO, {
Utils.addEnterHandler(manageContainer, async () => { Utils.addEnterHandler(manageContainer, async () => {
Logic.showPanel(P_CONTAINER_EDIT, identity); Logic.showPanel(P_CONTAINER_EDIT, identity);
}); });
const clearContainerStorageButton = document.getElementById("clear-container-storage-info");
Utils.addEnterHandler(clearContainerStorageButton, async () => {
const granted = await browser.permissions.request({ permissions: ["browsingData"] });
if (granted) {
Logic.showPanel(P_CLEAR_CONTAINER_STORAGE, identity);
}
});
return this.buildOpenTabTable(tabs); return this.buildOpenTabTable(tabs);
}, },
@ -1280,8 +1253,7 @@ 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();
}; };
@ -1311,8 +1283,7 @@ 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();
}); });
@ -1453,14 +1424,11 @@ Logic.registerPanel(P_CONTAINER_ASSIGNMENTS, {
/* As we don't have the full or correct path the best we can assume is the path is HTTPS and then replace with a broken icon later if it doesn't load. /* As we don't have the full or correct path the best we can assume is the path is HTTPS and then replace with a broken icon later if it doesn't load.
This is pending a better solution for favicons from web extensions */ This is pending a better solution for favicons from web extensions */
const assumedUrl = `https://${site.hostname}/favicon.ico`; const assumedUrl = `https://${site.hostname}/favicon.ico`;
const resetSiteCookiesInfo = browser.i18n.getMessage("clearSiteCookiesTooltipInfo");
const deleteSiteInfo = browser.i18n.getMessage("deleteSiteTooltipInfo");
trElement.innerHTML = Utils.escaped` trElement.innerHTML = Utils.escaped`
<td> <td>
<div class="favicon"></div> <div class="favicon"></div>
<span title="${site.hostname}" class="menu-text truncate-text">${site.hostname}</span> <span title="${site.hostname}" class="menu-text truncate-text">${site.hostname}</span>
<img title="${resetSiteCookiesInfo}" class="reset-button reset-assignment" src="/img/refresh-16.svg" /> <img class="trash-button delete-assignment" src="/img/container-delete.svg" />
<img title="${deleteSiteInfo}" class="trash-button delete-assignment" src="/img/container-delete.svg" />
</td>`; </td>`;
trElement.getElementsByClassName("favicon")[0].appendChild(Utils.createFavIconElement(assumedUrl)); trElement.getElementsByClassName("favicon")[0].appendChild(Utils.createFavIconElement(assumedUrl));
const deleteButton = trElement.querySelector(".trash-button"); const deleteButton = trElement.querySelector(".trash-button");
@ -1472,20 +1440,6 @@ Logic.registerPanel(P_CONTAINER_ASSIGNMENTS, {
delete assignments[siteKey]; delete assignments[siteKey];
this.showAssignedContainers(assignments); this.showAssignedContainers(assignments);
}); });
const resetButton = trElement.querySelector(".reset-button");
Utils.addEnterHandler(resetButton, async () => {
const cookieStoreId = Logic.currentCookieStoreId();
const granted = await browser.permissions.request({ permissions: ["browsingData"] });
if (!granted) {
return;
}
const result = await Utils.resetCookiesForSite(site.hostname, cookieStoreId);
if (result === true) {
Logic.notify({messageId: "cookiesClearedSuccess", placeholders: [site.hostname]});
} else {
Logic.notify({messageId: "cookiesCouldNotBeCleared", placeholders: [site.hostname]});
}
});
trElement.classList.add("menu-item", "hover-highlight", "keyboard-nav"); trElement.classList.add("menu-item", "hover-highlight", "keyboard-nav");
tableElement.appendChild(trElement); tableElement.appendChild(trElement);
}); });
@ -2261,47 +2215,6 @@ Logic.registerPanel(P_MOZILLA_VPN_SERVER_LIST, {
} }
}); });
// P_CLEAR_CONTAINER_STORAGE: Page for confirming container storage removal.
// ----------------------------------------------------------------------------
Logic.registerPanel(P_CLEAR_CONTAINER_STORAGE, {
panelSelector: "#clear-container-storage-panel",
// This method is called when the object is registered.
initialize() {
Utils.addEnterHandler(document.querySelector("#clear-container-storage-cancel-link"), () => {
const identity = Logic.currentIdentity();
Logic.showPanel(P_CONTAINER_INFO, identity, false, false);
});
Utils.addEnterHandler(document.querySelector("#close-clear-container-storage-panel"), () => {
const identity = Logic.currentIdentity();
Logic.showPanel(P_CONTAINER_INFO, identity, false, false);
});
Utils.addEnterHandler(document.querySelector("#clear-container-storage-ok-link"), async () => {
const identity = Logic.currentIdentity();
const userContextId = Utils.userContextId(identity.cookieStoreId);
const result = await browser.runtime.sendMessage({
method: "deleteContainerDataOnly",
message: { userContextId }
});
if (result.done === true) {
Logic.notify({messageId: "storageWasClearedConfirmation", placeholders: [identity.name]});
}
Logic.showPanel(P_CONTAINER_INFO, identity, false, false);
});
},
// This method is called when the panel is shown.
prepare() {
const identity = Logic.currentIdentity();
// Populating the panel: name, icon, and warning message
document.getElementById("container-clear-storage-title").textContent = identity.name;
return Promise.resolve(null);
},
});
// P_CONTAINER_DELETE: Delete a container. // P_CONTAINER_DELETE: Delete a container.
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------

View file

@ -94,9 +94,6 @@ 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) {
@ -141,32 +138,14 @@ const Utils = {
}); });
}, },
resetCookiesForSite(pageUrl, cookieStoreId) { async reloadInContainer(url, currentUserContextId, newUserContextId, tabIndex, active) {
return browser.runtime.sendMessage({
method: "resetCookiesForSite",
pageUrl,
cookieStoreId,
});
},
/**
* @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
}); });
}, },
@ -180,8 +159,7 @@ 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(

View file

@ -1,12 +1,12 @@
{ {
"manifest_version": 2, "manifest_version": 2,
"name": "Firefox Multi-Account Containers", "name": "Firefox Multi-Account Containers",
"version": "8.3.0", "version": "8.1.1",
"incognito": "not_allowed", "incognito": "not_allowed",
"description": "__MSG_extensionDescription__", "description": "__MSG_extensionDescription__",
"icons": { "icons": {
"48": "img/multiaccountcontainer-16.svg", "48": "img/container-site-d-48.png",
"96": "img/multiaccountcontainer-16.svg" "96": "img/container-site-d-96.png"
}, },
"homepage_url": "https://github.com/mozilla/multi-account-containers#readme", "homepage_url": "https://github.com/mozilla/multi-account-containers#readme",
"permissions": [ "permissions": [
@ -26,7 +26,6 @@
], ],
"optional_permissions": [ "optional_permissions": [
"bookmarks", "bookmarks",
"browsingData",
"nativeMessaging", "nativeMessaging",
"proxy" "proxy"
], ],
@ -42,81 +41,77 @@
"default": "Ctrl+Period", "default": "Ctrl+Period",
"mac": "MacCtrl+Period" "mac": "MacCtrl+Period"
}, },
"description": "__MSG_openContainerPanel__" "description": "Open containers panel"
},
"sort_tabs": {
"description": "__MSG_sortTabsByContainer__"
}, },
"open_container_0": { "open_container_0": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+1" "default": "Ctrl+Shift+1"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 1"
}, },
"open_container_1": { "open_container_1": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+2" "default": "Ctrl+Shift+2"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 2"
}, },
"open_container_2": { "open_container_2": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+3" "default": "Ctrl+Shift+3"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 3"
}, },
"open_container_3": { "open_container_3": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+4" "default": "Ctrl+Shift+4"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 4"
}, },
"open_container_4": { "open_container_4": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+5" "default": "Ctrl+Shift+5"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 5"
}, },
"open_container_5": { "open_container_5": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+6" "default": "Ctrl+Shift+6"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 6"
}, },
"open_container_6": { "open_container_6": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+7" "default": "Ctrl+Shift+7"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 7"
}, },
"open_container_7": { "open_container_7": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+8" "default": "Ctrl+Shift+8"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 8"
}, },
"open_container_8": { "open_container_8": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+9" "default": "Ctrl+Shift+9"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 9"
}, },
"open_container_9": { "open_container_9": {
"suggested_key": { "suggested_key": {
"default": "Ctrl+Shift+0" "default": "Ctrl+Shift+0"
}, },
"description": "__MSG_containerShortcut__" "description": "Container Shortcut 10"
} }
}, },
"browser_action": { "browser_action": {
"browser_style": true, "browser_style": true,
"default_icon": "img/multiaccountcontainer-16.svg", "default_icon": "img/multiaccountcontainer-16.svg",
"default_title": "Firefox Multi-Account Containers", "default_title": "Multi-Account Containers",
"default_popup": "popup.html", "default_popup": "popup.html",
"default_area": "navbar",
"theme_icons": [ "theme_icons": [
{ {
"light": "img/multiaccountcontainer-16.svg", "light": "img/multiaccountcontainer-16-dark.svg",
"dark": "img/multiaccountcontainer-16.svg", "dark": "img/multiaccountcontainer-16.svg",
"size": 32 "size": 32
} }
@ -125,7 +120,7 @@
"page_action": { "page_action": {
"browser_style": true, "browser_style": true,
"default_icon": "img/container-openin-16.svg", "default_icon": "img/container-openin-16.svg",
"default_title": "__MSG_alwaysOpenSiteInContainer__", "default_title": "Always open this in a Container",
"default_popup": "pageActionPopup.html", "default_popup": "pageActionPopup.html",
"pinned": false, "pinned": false,
"show_matches": ["*://*/*"] "show_matches": ["*://*/*"]
@ -149,7 +144,7 @@
], ],
"default_locale": "en", "default_locale": "en",
"web_accessible_resources": [ "web_accessible_resources": [
"/img/multiaccountcontainer-16.svg" "/img/container-site-d-24.png"
], ],
"options_ui": { "options_ui": {
"page": "options.html", "page": "options.html",

View file

@ -40,7 +40,7 @@
</div> </div>
</div> </div>
</div> </div>
<h3 data-i18n-message-id="sync"></h3> <h3 data-i18n-message-id="firefoxAccountsSync"></h3>
<div class="settings-group"> <div class="settings-group">
<label> <label>
<input type="checkbox" id="syncCheck"> <input type="checkbox" id="syncCheck">

View file

@ -1,7 +1,7 @@
<html> <html>
<head> <head>
<meta http-equiv="content-type" content="text/html; charset=utf-8"> <meta http-equiv="content-type" content="text/html; charset=utf-8">
<title>Firefox Multi-Account Containers</title> <title>Multi-Account Containers</title>
<script type="text/javascript" src="./js/i18n.js"></script> <script type="text/javascript" src="./js/i18n.js"></script>
<link rel="stylesheet" type="text/css" href="css/popup.css"> <link rel="stylesheet" type="text/css" href="css/popup.css">

View file

@ -1,7 +1,7 @@
<html data-theme="auto"> <html data-theme="auto">
<head> <head>
<meta http-equiv="content-type" content="text/html; charset=utf-8"> <meta http-equiv="content-type" content="text/html; charset=utf-8">
<title>Firefox Multi-Account Containers</title> <title>Multi-Account Containers</title>
<script type="text/javascript" src="./js/i18n.js"></script> <script type="text/javascript" src="./js/i18n.js"></script>
<link rel="stylesheet" href="./css/popup.css"> <link rel="stylesheet" href="./css/popup.css">
</head> </head>
@ -44,7 +44,7 @@
<div class="panel onboarding onboarding-panel-6 hide" id="onboarding-panel-6"> <div class="panel onboarding onboarding-panel-6 hide" id="onboarding-panel-6">
<img class="onboarding-img" alt="" src="/img/Sync.svg" /> <img class="onboarding-img" alt="" src="/img/Sync.svg" />
<h3 class="onboarding-title" data-i18n-message-id="onboarding-6-header"></h3> <h3 class="onboarding-title" data-i18n-message-id="onboarding-6-header"></h3>
<p data-i18n-message-id="onboarding-6-description-2"></p> <p data-i18n-message-id="onboarding-6-description"></p>
<div class="half-button-wrapper"> <div class="half-button-wrapper">
<a href="#" id="no-sync" class="half-onboarding-button grey-button keyboard-nav" tabindex="0" data-i18n-message-id="notNow"></a> <a href="#" id="no-sync" class="half-onboarding-button grey-button keyboard-nav" tabindex="0" data-i18n-message-id="notNow"></a>
<a href="#" id="start-sync-button" class="half-onboarding-button keyboard-nav" tabindex="0" data-i18n-message-id="startSyncing"></a> <a href="#" id="start-sync-button" class="half-onboarding-button keyboard-nav" tabindex="0" data-i18n-message-id="startSyncing"></a>
@ -53,8 +53,8 @@
<div class="panel onboarding onboarding-panel-7 hide" id="onboarding-panel-7"> <div class="panel onboarding onboarding-panel-7 hide" id="onboarding-panel-7">
<img class="onboarding-img" alt="" src="/img/Account.svg" /> <img class="onboarding-img" alt="" src="/img/Account.svg" />
<h3 class="onboarding-title" data-i18n-message-id="onboarding-7-header-2"></h3> <h3 class="onboarding-title" data-i18n-message-id="onboarding-7-header"></h3>
<p data-i18n-message-id="onboarding-7-description-2"></p> <p data-i18n-message-id="onboarding-7-description"></p>
<div class="half-button-wrapper"> <div class="half-button-wrapper">
<a href="#" id="no-sign-in" class="half-onboarding-button grey-button keyboard-nav" tabindex="0" data-i18n-message-id="notNow"></a> <a href="#" id="no-sign-in" class="half-onboarding-button grey-button keyboard-nav" tabindex="0" data-i18n-message-id="notNow"></a>
<a href="#" id="sign-in" class="half-onboarding-button keyboard-nav" tabindex="0" data-i18n-message-id="signIn"></a> <a href="#" id="sign-in" class="half-onboarding-button keyboard-nav" tabindex="0" data-i18n-message-id="signIn"></a>
@ -107,8 +107,7 @@
</div> </div>
<div class="panel menu-panel container-panel hide" id="container-panel"> <div class="panel menu-panel container-panel hide" id="container-panel">
<span class="popup-notification-card"></span> <h3 class="title">Multi-Account Containers</h3>
<h3 class="title">Firefox Multi-Account Containers</h3>
<a href="#" class="info-icon" id="info-icon" tabindex="10"> <a href="#" class="info-icon" id="info-icon" tabindex="10">
<img data-i18n-attribute-message-id="info" data-i18n-attribute="alt" alt="" src="/img/info.svg" / > <img data-i18n-attribute-message-id="info" data-i18n-attribute="alt" alt="" src="/img/info.svg" / >
</a> </a>
@ -192,14 +191,24 @@
</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>
<div class="hide panel menu-panel container-info-panel" id="container-info-panel" tabindex="-1"> <div class="hide panel menu-panel container-info-panel" id="container-info-panel" tabindex="-1">
<span class="popup-notification-card"></span>
<h3 class="title" id="container-info-title" data-i18n-attribute-message-id="personal"></h3> <h3 class="title" id="container-info-title" data-i18n-attribute-message-id="personal"></h3>
<button class="btn-return arrow-left controller keyboard-nav-back" id="close-container-info-panel" tabindex="0"></button> <button class="btn-return arrow-left controller keyboard-nav-back" id="close-container-info-panel" tabindex="0"></button>
<hr> <hr>
@ -236,14 +245,6 @@
</span> </span>
</td> </td>
</tr> </tr>
<tr class="menu-item hover-highlight keyboard-nav" id="clear-container-storage" tabindex="0">
<td>
<img class="menu-icon clear-storage-icon" alt="" src="img/container-delete.svg" />
<span class="menu-text" id="clear-container-storage-info" data-i18n-message-id="clearContainerStorage"></span>
<span class="menu-arrow">
</span>
</td>
</tr>
</table> </table>
<hr> <hr>
<div class="sub-header-wrapper"> <div class="sub-header-wrapper">
@ -266,9 +267,8 @@
<div class="panel menu-panel container-picker-panel hide" id="container-picker-panel"> <div class="panel menu-panel container-picker-panel hide" id="container-picker-panel">
<span class="popup-notification-card"></span>
<h3 class="title" id="picker-title"> <h3 class="title" id="picker-title">
Firefox Multi-Account Containers Multi-Account Containers
</h3> </h3>
<button class="btn-return arrow-left controller keyboard-nav-back" id="close-container-picker-panel" tabindex="0"></button> <button class="btn-return arrow-left controller keyboard-nav-back" id="close-container-picker-panel" tabindex="0"></button>
<hr> <hr>
@ -291,7 +291,6 @@
</div> </div>
<div class="panel menu-panel edit-container-panel hide" id="edit-container-panel"> <div class="panel menu-panel edit-container-panel hide" id="edit-container-panel">
<span class="popup-notification-card"></span>
<h3 class="title" id="container-edit-title" data-i18n-message-id="default"></h3> <h3 class="title" id="container-edit-title" data-i18n-message-id="default"></h3>
<button class="btn-return arrow-left controller" id="close-container-edit-panel"></button> <button class="btn-return arrow-left controller" id="close-container-edit-panel"></button>
<hr> <hr>
@ -382,7 +381,6 @@
</div> </div>
<div class="panel menu-panel edit-container-assignments hide" id="edit-container-assignments"> <div class="panel menu-panel edit-container-assignments hide" id="edit-container-assignments">
<span class="popup-notification-card"></span>
<h3 class="title" id="edit-assignments-title" data-i18n-message-id="default"></h3> <h3 class="title" id="edit-assignments-title" data-i18n-message-id="default"></h3>
<button class="btn-return arrow-left controller" id="close-container-assignment-panel"></button> <button class="btn-return arrow-left controller" id="close-container-assignment-panel"></button>
<hr> <hr>
@ -412,23 +410,7 @@
</div> </div>
<div class="panel-footer"> <div class="panel-footer">
<a href="#" class="button expanded secondary footer-button cancel-button" data-i18n-message-id="cancel" id="delete-container-cancel-link"></a> <a href="#" class="button expanded secondary footer-button cancel-button" data-i18n-message-id="cancel" id="delete-container-cancel-link"></a>
<a href="#" class="button expanded confirmation-destructive-ok-btn footer-button alert-text" data-i18n-message-id="ok" id="delete-container-ok-link"></a> <a href="#" class="button expanded primary footer-button" data-i18n-message-id="ok" id="delete-container-ok-link"></a>
</div>
</div>
<div class="hide panel clear-container-storage-panel" id="clear-container-storage-panel">
<h3 class="title" id="container-clear-storage-title" data-i18n-message-id="default">
</h3>
<button class="btn-return arrow-left controller" id="close-clear-container-storage-panel"></button>
<hr>
<div class="panel-content clear-container-storage-confirm">
<h4 class="clear-container-storage-confirm-title" data-i18n-message-id="clearContainerStoragePanelTitle"></h4>
<p class="clear-container-storage-warning" data-i18n-message-id="clearContainerStorageConfirmation"></p>
</div>
<div class="panel-footer">
<a href="#" class="button expanded secondary footer-button cancel-button" data-i18n-message-id="cancel" id="clear-container-storage-cancel-link"></a>
<a href="#" class="button expanded confirmation-destructive-ok-btn footer-button alert-text" data-i18n-message-id="ok" id="clear-container-storage-ok-link"></a>
</div> </div>
</div> </div>

View file

@ -32,10 +32,6 @@ 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();
} }
} }
}; };

15
web-ext-config.js Normal file
View file

@ -0,0 +1,15 @@
module.exports = {
sourceDir: "./src",
build: {
overwriteDest: true,
},
run: {
pref: [
"ui.popup.disable_autohide=true",
"extensions.manifestV3.enabled=false",
"xpinstall.signatures.required=false",
"ui.systemUsesDarkTheme=1"
],
browserConsole: true
},
};