Compare commits
79 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| df43ffbf2e | |||
| ea7d80a01c | |||
| bd1e2107ae | |||
| 06cb95e9db | |||
| 769162d755 | |||
| 16deecda37 | |||
| 312b5f3e1c | |||
| a1018e2732 | |||
| c233ec3ada | |||
| 20c59fb26f | |||
| dd2788ee41 | |||
| b630fd8c4e | |||
| d456b98873 | |||
| 1f0c522773 | |||
| 2f6596259b | |||
| d432e316bd | |||
| 19a2a76f7c | |||
| eda79aaf05 | |||
| 37a2b67224 | |||
| fb5eb2c0db | |||
| b4d0115d22 | |||
| 02f9ea8ec9 | |||
| 50e4b2742f | |||
| 8b70aca184 | |||
| 0539c12ba2 | |||
| d6cb8f7707 | |||
| 89aa2ffe5b | |||
| f072ad478c | |||
| 91a92bd446 | |||
| f0274d1e45 | |||
| 0cf1e14731 | |||
| 1406ad34b4 | |||
| f377174bf2 | |||
| f6a59ab54e | |||
| 542161f8b4 | |||
| a5cbb48907 | |||
| 56c5838d2d | |||
| 35956f132a | |||
| b4ad47bf04 | |||
| c34c1c1e04 | |||
| 2908419671 | |||
| e296f438fa | |||
| 85ce6375e5 | |||
| dcc42e2a3a | |||
| eeefaaba1e | |||
| 00504ebbd9 | |||
| 0eb13f214d | |||
| adbf310a17 | |||
| d44d789e73 | |||
| bd7e33b11e | |||
| 6a5e48e8b3 | |||
| d3aa323a5a | |||
| aca51cc11c | |||
| b684ce7016 | |||
| 115d411218 | |||
| aec2aa5fb0 | |||
| 69ee83bbf6 | |||
| 9434147b48 | |||
| d82341ce4a | |||
| 60a6666222 | |||
| f1a24ed6fb | |||
| 0372abdc33 | |||
| 366a50c8b6 | |||
| e96b275e02 | |||
| 65243e2c06 | |||
| ab3e1ce4d8 | |||
| 5194fcad0e | |||
| b6a1bff9e8 | |||
| 5ae2047b2c | |||
| a60f5bb1be | |||
| c8c4e0f0c5 | |||
| 037a804725 | |||
| 6fcb828e1d | |||
| aa9bb41305 | |||
| 546ee7a098 | |||
| d8cff7ca41 | |||
| e6e7d5178e | |||
| 7767bb0c58 | |||
| 580fb5234b |
@@ -1,2 +0,0 @@
|
|||||||
lib/testpilot/*.js
|
|
||||||
coverage
|
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
module.exports = {
|
|
||||||
"parserOptions": {
|
|
||||||
"ecmaVersion": 2018
|
|
||||||
},
|
|
||||||
"env": {
|
|
||||||
"browser": true,
|
|
||||||
"es6": true,
|
|
||||||
"node": true,
|
|
||||||
"webextensions": true
|
|
||||||
},
|
|
||||||
"globals": {
|
|
||||||
"Utils": true,
|
|
||||||
"CustomizableUI": true,
|
|
||||||
"CustomizableWidgets": true,
|
|
||||||
"SessionStore": true,
|
|
||||||
"Services": true,
|
|
||||||
"Components": true,
|
|
||||||
"XPCOMUtils": true,
|
|
||||||
"OS": true,
|
|
||||||
"ADDON_UNINSTALL": true,
|
|
||||||
"ADDON_DISABLE": true,
|
|
||||||
"CONTAINER_ORDER_STORAGE_KEY": true,
|
|
||||||
"proxifiedContainers": true,
|
|
||||||
"MozillaVPN": true,
|
|
||||||
"MozillaVPN_Background": true
|
|
||||||
},
|
|
||||||
"plugins": [
|
|
||||||
"promise",
|
|
||||||
"no-unsanitized"
|
|
||||||
],
|
|
||||||
"extends": [
|
|
||||||
"eslint:recommended"
|
|
||||||
],
|
|
||||||
"root": true,
|
|
||||||
"rules": {
|
|
||||||
"promise/always-return": "off",
|
|
||||||
"promise/avoid-new": "off",
|
|
||||||
"promise/catch-or-return": "error",
|
|
||||||
"promise/no-callback-in-promise": "warn",
|
|
||||||
"promise/no-native": "off",
|
|
||||||
"promise/no-nesting": "warn",
|
|
||||||
"promise/no-promise-in-callback": "warn",
|
|
||||||
"promise/no-return-wrap": "error",
|
|
||||||
"promise/param-names": "error",
|
|
||||||
|
|
||||||
"no-unsanitized/method": [
|
|
||||||
"error"
|
|
||||||
],
|
|
||||||
"no-unsanitized/property": [
|
|
||||||
"error",
|
|
||||||
{
|
|
||||||
"escape": {
|
|
||||||
"taggedTemplates": ["Utils.escaped"]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
],
|
|
||||||
|
|
||||||
"eqeqeq": "error",
|
|
||||||
"indent": ["error", 2],
|
|
||||||
"linebreak-style": ["error", "unix"],
|
|
||||||
"no-throw-literal": "error",
|
|
||||||
"no-warning-comments": "warn",
|
|
||||||
"no-var": "error",
|
|
||||||
"prefer-const": "error",
|
|
||||||
"quotes": ["error", "double"],
|
|
||||||
"radix": "error",
|
|
||||||
"semi": ["error", "always"]
|
|
||||||
}
|
|
||||||
};
|
|
||||||
@@ -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
|
||||||
|
|||||||
+17
-1
@@ -32,7 +32,23 @@ 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
|
||||||
@@ -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 [our forum][forum].
|
You can also chat with us on [our Matrix room][matrix] or ask in [our discussions board][discussions].
|
||||||
|
|
||||||
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,4 +29,5 @@ 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
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/env bash
|
#!/usr/bin/env 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
|
||||||
|
|||||||
+1
-1
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/env bash
|
#!/usr/bin/env 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
|
||||||
|
|||||||
+1
-1
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/env bash
|
#!/usr/bin/env 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
|
||||||
|
|||||||
@@ -0,0 +1,102 @@
|
|||||||
|
const {
|
||||||
|
defineConfig,
|
||||||
|
globalIgnores,
|
||||||
|
} = require("eslint/config");
|
||||||
|
|
||||||
|
const globals = require("globals");
|
||||||
|
const promise = require("eslint-plugin-promise");
|
||||||
|
const noUnsanitized = require("eslint-plugin-no-unsanitized");
|
||||||
|
const js = require("@eslint/js");
|
||||||
|
|
||||||
|
module.exports = defineConfig([{
|
||||||
|
languageOptions: {
|
||||||
|
"ecmaVersion": 2021,
|
||||||
|
parserOptions: {},
|
||||||
|
|
||||||
|
globals: {
|
||||||
|
...globals.browser,
|
||||||
|
...globals.node,
|
||||||
|
...globals.webextensions,
|
||||||
|
"Utils": true,
|
||||||
|
"CustomizableUI": true,
|
||||||
|
"CustomizableWidgets": true,
|
||||||
|
"SessionStore": true,
|
||||||
|
"Services": true,
|
||||||
|
"Components": true,
|
||||||
|
"XPCOMUtils": true,
|
||||||
|
"OS": true,
|
||||||
|
"ADDON_UNINSTALL": true,
|
||||||
|
"ADDON_DISABLE": true,
|
||||||
|
"CONTAINER_ORDER_STORAGE_KEY": true,
|
||||||
|
"proxifiedContainers": true,
|
||||||
|
"MozillaVPN": true,
|
||||||
|
"MozillaVPN_Background": true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
plugins: {
|
||||||
|
js,
|
||||||
|
promise,
|
||||||
|
"no-unsanitized": noUnsanitized,
|
||||||
|
},
|
||||||
|
|
||||||
|
extends: ["js/recommended"],
|
||||||
|
|
||||||
|
"rules": {
|
||||||
|
"promise/always-return": "off",
|
||||||
|
"promise/avoid-new": "off",
|
||||||
|
"promise/catch-or-return": "error",
|
||||||
|
"promise/no-callback-in-promise": "warn",
|
||||||
|
"promise/no-native": "off",
|
||||||
|
"promise/no-nesting": "warn",
|
||||||
|
"promise/no-promise-in-callback": "warn",
|
||||||
|
"promise/no-return-wrap": "error",
|
||||||
|
"promise/param-names": "error",
|
||||||
|
"no-unsanitized/method": ["error"],
|
||||||
|
|
||||||
|
"no-unsanitized/property": ["error", {
|
||||||
|
"escape": {
|
||||||
|
"taggedTemplates": ["Utils.escaped"],
|
||||||
|
},
|
||||||
|
}],
|
||||||
|
|
||||||
|
"eqeqeq": "error",
|
||||||
|
"indent": ["error", 2],
|
||||||
|
"linebreak-style": ["error", "unix"],
|
||||||
|
"no-throw-literal": "error",
|
||||||
|
"no-warning-comments": "warn",
|
||||||
|
"no-var": "error",
|
||||||
|
"prefer-const": "error",
|
||||||
|
"quotes": ["error", "double"],
|
||||||
|
"radix": "error",
|
||||||
|
"semi": ["error", "always"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
files: ["test/**/*.js"],
|
||||||
|
languageOptions: {
|
||||||
|
globals: {
|
||||||
|
...globals.mocha,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"rules": {
|
||||||
|
"no-restricted-globals": ["error", "browser"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
files: ["src/js/**/*.js"],
|
||||||
|
languageOptions: {
|
||||||
|
globals: {
|
||||||
|
"assignManager": true,
|
||||||
|
"badge": true,
|
||||||
|
"backgroundLogic": true,
|
||||||
|
"identityState": true,
|
||||||
|
"messageHandler": true,
|
||||||
|
"sync": true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
globalIgnores(["lib/testpilot/*.js", "**/coverage"])]);
|
||||||
Generated
+2683
-14723
File diff suppressed because it is too large
Load Diff
+8
-7
@@ -2,19 +2,20 @@
|
|||||||
"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.7",
|
||||||
"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": {},
|
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@eslint/js": "^9.36.0",
|
||||||
"addons-linter": "^5.28.0",
|
"addons-linter": "^5.28.0",
|
||||||
"ajv": "^6.6.3",
|
"ajv": "^6.6.3",
|
||||||
"chai": "^4.2.0",
|
"chai": "^4.2.0",
|
||||||
"eslint": "^7.32.0",
|
"eslint": "^9.36.0",
|
||||||
"eslint-plugin-no-unsanitized": "^4.0.0",
|
"eslint-plugin-no-unsanitized": "^4.1.4",
|
||||||
"eslint-plugin-promise": "^5.2.0",
|
"eslint-plugin-promise": "^7.2.1",
|
||||||
|
"globals": "^16.4.0",
|
||||||
"htmllint-cli": "0.0.7",
|
"htmllint-cli": "0.0.7",
|
||||||
"json": ">=10.0.0",
|
"json": ">=10.0.0",
|
||||||
"mocha": "^10.1.0",
|
"mocha": "^10.1.0",
|
||||||
@@ -25,7 +26,7 @@
|
|||||||
"stylelint": "^13.5.0",
|
"stylelint": "^13.5.0",
|
||||||
"stylelint-config-standard": "^20.0.0",
|
"stylelint-config-standard": "^20.0.0",
|
||||||
"stylelint-order": "^4.0.0",
|
"stylelint-order": "^4.0.0",
|
||||||
"web-ext": "^7.5.0",
|
"web-ext": "^8.10.0",
|
||||||
"webextensions-jsdom": "^1.2.1"
|
"webextensions-jsdom": "^1.2.1"
|
||||||
},
|
},
|
||||||
"homepage": "https://github.com/mozilla/multi-account-containers#readme",
|
"homepage": "https://github.com/mozilla/multi-account-containers#readme",
|
||||||
@@ -43,7 +44,7 @@
|
|||||||
"lint:css": "stylelint src/css/*.css",
|
"lint:css": "stylelint src/css/*.css",
|
||||||
"lint:html": "htmllint *.html",
|
"lint:html": "htmllint *.html",
|
||||||
"lint:js": "eslint .",
|
"lint:js": "eslint .",
|
||||||
"package": "rm -rf src/web-ext-artifacts && npm run build && mv src/web-ext-artifacts/firefox_multi-account_containers-*.zip addon.xpi",
|
"package": "rm -rf web-ext-artifacts && npm run build && mv web-ext-artifacts/firefox_multi-account_containers-*.zip addon.xpi",
|
||||||
"restore-locales-github": "cd src/_locales && git restore .github/",
|
"restore-locales-github": "cd src/_locales && git restore .github/",
|
||||||
"remove-locales-github": "rm -rf src/_locales/.github",
|
"remove-locales-github": "rm -rf src/_locales/.github",
|
||||||
"test": "npm run lint && npm run coverage",
|
"test": "npm run lint && npm run coverage",
|
||||||
|
|||||||
+1
-1
Submodule src/_locales updated: bdaa01291b...2d46101815
+7
-33
@@ -28,6 +28,8 @@
|
|||||||
|
|
||||||
[data-theme="light"],
|
[data-theme="light"],
|
||||||
:root {
|
:root {
|
||||||
|
color-scheme: light;
|
||||||
|
|
||||||
--fontInter: "Inter", sans-serif;
|
--fontInter: "Inter", sans-serif;
|
||||||
--fontInterMedium: "Inter-Medium", sans-serif;
|
--fontInterMedium: "Inter-Medium", sans-serif;
|
||||||
--fontMetropolis: "Metropolis", sans-serif;
|
--fontMetropolis: "Metropolis", sans-serif;
|
||||||
@@ -110,10 +112,11 @@
|
|||||||
--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"] {
|
||||||
|
color-scheme: dark;
|
||||||
|
|
||||||
--iconCloseX: url("/img/close-light.svg");
|
--iconCloseX: url("/img/close-light.svg");
|
||||||
--iconGear: url("/img/gear-icon-light.svg");
|
--iconGear: url("/img/gear-icon-light.svg");
|
||||||
--iconArrowRight: url("/img/arrow-icon-right-light.svg");
|
--iconArrowRight: url("/img/arrow-icon-right-light.svg");
|
||||||
@@ -180,7 +183,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 */
|
||||||
@@ -240,7 +242,9 @@ body {
|
|||||||
[data-theme="dark"] img.clear-storage-icon,
|
[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,
|
||||||
|
[data-theme="dark"] #always-open .menu-icon,
|
||||||
|
[data-theme="dark"] #always-open-in .menu-icon {
|
||||||
filter: invert(0);
|
filter: invert(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -635,37 +639,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 +1028,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);
|
||||||
|
|||||||
@@ -2,8 +2,11 @@
|
|||||||
- 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
|
||||||
- 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 data-name="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16">
|
<svg data-name="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16">
|
||||||
<rect x="1" y="1" width="6" height="6" rx="1"/>
|
<style>
|
||||||
<rect x="1" y="9" width="6" height="6" rx="1"/>
|
:root { color-scheme: light dark; }
|
||||||
<rect x="9" y="9" width="6" height="6" rx="1"/>
|
</style>
|
||||||
<path fill="context-fill" fill-opacity="context-fill-opacity" d="M14.92 1.62a1 1 0 0 0-0.54-0.54A1 1 0 0 0 14 1h-4a1 1 0 0 0 0 2h1.59l-2.3 2.29a1 1 0 0 0 0 1.42 1 1 0 0 0 1.42 0L13 4.41V6a1 1 0 0 0 2 0V2a1 1 0 0 0-0.08-0.38z"/>
|
<rect fill="context-fill light-dark(black, white)" fill-opacity="context-fill-opacity" x="1" y="1" width="6" height="6" rx="1"/>
|
||||||
|
<rect fill="context-fill light-dark(black, white)" fill-opacity="context-fill-opacity" x="1" y="9" width="6" height="6" rx="1"/>
|
||||||
|
<rect fill="context-fill light-dark(black, white)" fill-opacity="context-fill-opacity" x="9" y="9" width="6" height="6" rx="1"/>
|
||||||
|
<path fill="context-fill light-dark(black, white)" fill-opacity="context-fill-opacity" d="M14.92 1.62a1 1 0 0 0-0.54-0.54A1 1 0 0 0 14 1h-4a1 1 0 0 0 0 2h1.59l-2.3 2.29a1 1 0 0 0 0 1.42 1 1 0 0 0 1.42 0L13 4.41V6a1 1 0 0 0 2 0V2a1 1 0 0 0-0.08-0.38z"/>
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 677 B After Width: | Height: | Size: 1006 B |
@@ -1,14 +0,0 @@
|
|||||||
module.exports = {
|
|
||||||
"extends": [
|
|
||||||
"../../.eslintrc.js"
|
|
||||||
],
|
|
||||||
"globals": {
|
|
||||||
"assignManager": true,
|
|
||||||
"badge": true,
|
|
||||||
"backgroundLogic": true,
|
|
||||||
"identityState": true,
|
|
||||||
"messageHandler": true,
|
|
||||||
"sync": true,
|
|
||||||
"Utils": true
|
|
||||||
}
|
|
||||||
};
|
|
||||||
@@ -61,8 +61,9 @@ 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);
|
||||||
});
|
});
|
||||||
@@ -76,7 +77,6 @@ window.assignManager = {
|
|||||||
this.setExempted(pageUrlorUrlKey, tabId);
|
this.setExempted(pageUrlorUrlKey, tabId);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
// eslint-disable-next-line require-atomic-updates
|
|
||||||
data.identityMacAddonUUID =
|
data.identityMacAddonUUID =
|
||||||
await identityState.lookupMACaddonUUID(data.userContextId);
|
await identityState.lookupMACaddonUUID(data.userContextId);
|
||||||
await this.area.set({
|
await this.area.set({
|
||||||
@@ -232,7 +232,7 @@ window.assignManager = {
|
|||||||
try {
|
try {
|
||||||
container = await browser.contextualIdentities
|
container = await browser.contextualIdentities
|
||||||
.get(backgroundLogic.cookieStoreId(siteSettings.userContextId));
|
.get(backgroundLogic.cookieStoreId(siteSettings.userContextId));
|
||||||
} catch (e) {
|
} catch {
|
||||||
container = false;
|
container = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -325,7 +325,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(
|
||||||
@@ -335,7 +336,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);
|
||||||
@@ -480,9 +482,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) {
|
||||||
@@ -638,7 +638,7 @@ window.assignManager = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
async _maybeRemoveSiteIsolation(userContextId) {
|
async _maybeRemoveSiteIsolation(userContextId) {
|
||||||
const assignments = await this.storageArea.getByContainer(userContextId);
|
const assignments = await this.storageArea.getAssignedSites(userContextId);
|
||||||
const hasAssignments = assignments && Object.keys(assignments).length > 0;
|
const hasAssignments = assignments && Object.keys(assignments).length > 0;
|
||||||
if (hasAssignments) {
|
if (hasAssignments) {
|
||||||
return;
|
return;
|
||||||
@@ -669,11 +669,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) {
|
||||||
@@ -726,7 +726,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.
|
||||||
//
|
//
|
||||||
@@ -745,16 +753,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;
|
||||||
@@ -762,13 +812,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) => {
|
||||||
@@ -796,12 +847,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();
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
* 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/. */
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
|
/* global MAC_CONSTANTS */
|
||||||
|
|
||||||
const DEFAULT_TAB = "about:newtab";
|
const DEFAULT_TAB = "about:newtab";
|
||||||
|
|
||||||
const backgroundLogic = {
|
const backgroundLogic = {
|
||||||
@@ -11,22 +13,33 @@ const backgroundLogic = {
|
|||||||
"about:home",
|
"about:home",
|
||||||
"about:blank"
|
"about:blank"
|
||||||
]),
|
]),
|
||||||
NUMBER_OF_KEYBOARD_SHORTCUTS: 10,
|
NUMBER_OF_KEYBOARD_SHORTCUTS: MAC_CONSTANTS.NUMBER_OF_KEYBOARD_SHORTCUTS,
|
||||||
unhideQueue: [],
|
unhideQueue: [],
|
||||||
init() {
|
|
||||||
|
|
||||||
browser.commands.onCommand.addListener(function (command) {
|
init() {
|
||||||
|
browser.commands.onCommand.addListener(async function (command) {
|
||||||
if (command === "sort_tabs") {
|
if (command === "sort_tabs") {
|
||||||
backgroundLogic.sortTabs();
|
backgroundLogic.sortTabs();
|
||||||
return;
|
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 = MAC_CONSTANTS.OPEN_CONTAINER_PREFIX + i;
|
||||||
const cookieStoreId = identityState.keyboardShortcut[key];
|
const reopenKey = MAC_CONSTANTS.REOPEN_IN_CONTAINER_PREFIX + i;
|
||||||
if (command === key) {
|
if (command === key) {
|
||||||
if (cookieStoreId === "none") return;
|
const cookieStoreId = identityState.keyboardShortcut[key];
|
||||||
browser.tabs.create({cookieStoreId});
|
if (cookieStoreId && cookieStoreId !== "none") {
|
||||||
|
browser.tabs.create({cookieStoreId});
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (command === reopenKey) {
|
||||||
|
const cookieStoreId = identityState.keyboardShortcut[reopenKey];
|
||||||
|
if (cookieStoreId && cookieStoreId !== "none") {
|
||||||
|
backgroundLogic.reopenInContainer(cookieStoreId);
|
||||||
|
}
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -35,16 +48,86 @@ 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);
|
||||||
|
this._removeSurveyAchievement();
|
||||||
|
});
|
||||||
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");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async reopenInContainer(cookieStoreId) {
|
||||||
|
const currentTab = await browser.tabs.query({ active: true, currentWindow: true });
|
||||||
|
|
||||||
|
if (currentTab.length > 0) {
|
||||||
|
const tab = currentTab[0];
|
||||||
|
|
||||||
|
let url = tab.url;
|
||||||
|
if (this.NEW_TAB_PAGES.has(url) || !this.isPermissibleURL(url)) {
|
||||||
|
url = undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
browser.tabs.create({
|
||||||
|
url,
|
||||||
|
cookieStoreId,
|
||||||
|
index: tab.index + 1,
|
||||||
|
active: tab.active
|
||||||
|
});
|
||||||
|
|
||||||
|
browser.tabs.remove(tab.id);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We left an achievement entry in storage during a user research study in
|
||||||
|
* version 8.3.1. This method removes that entry to prevent broken logic in
|
||||||
|
* the achievement views.
|
||||||
|
*/
|
||||||
|
async _removeSurveyAchievement() {
|
||||||
|
const achievementsStorage = await browser.storage.local.get({ achievements: [] });
|
||||||
|
const achievements = achievementsStorage.achievements;
|
||||||
|
const filtered = achievements.filter(a => a.name !== "survey");
|
||||||
|
if (filtered.length !== achievements.length) {
|
||||||
|
await browser.storage.local.set({achievements: filtered});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
updateTranslationInManifest() {
|
updateTranslationInManifest() {
|
||||||
for (let index = 0; index < 10; index++) {
|
for (let index = 0; index < MAC_CONSTANTS.NUMBER_OF_KEYBOARD_SHORTCUTS; index++) {
|
||||||
const ajustedIndex = index + 1; // We want to start from 1 instead of 0 in the UI.
|
const adjustedIndex = index + 1; // We want to start from 1 instead of 0 in the UI.
|
||||||
browser.commands.update({
|
browser.commands.update({
|
||||||
name: `open_container_${index}`,
|
name: `${MAC_CONSTANTS.OPEN_CONTAINER_PREFIX}${index}`,
|
||||||
description: browser.i18n.getMessage("containerShortcut", `${ajustedIndex}`)
|
description: browser.i18n.getMessage("containerShortcut", `${adjustedIndex}`)
|
||||||
|
});
|
||||||
|
browser.commands.update({
|
||||||
|
name: `${MAC_CONSTANTS.REOPEN_IN_CONTAINER_PREFIX}${index}`,
|
||||||
|
description: browser.i18n.getMessage("reopenInContainerShortcut", `${adjustedIndex}`)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -210,7 +293,7 @@ const backgroundLogic = {
|
|||||||
containerState.isIsolated = "locked";
|
containerState.isIsolated = "locked";
|
||||||
}
|
}
|
||||||
return await identityState.storageArea.set(cookieStoreId, containerState);
|
return await identityState.storageArea.set(cookieStoreId, containerState);
|
||||||
} catch (error) {
|
} catch {
|
||||||
// console.error(`No container: ${cookieStoreId}`);
|
// console.error(`No container: ${cookieStoreId}`);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -343,7 +426,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 +445,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 +471,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) {
|
||||||
|
|||||||
@@ -0,0 +1,10 @@
|
|||||||
|
/* 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/. */
|
||||||
|
|
||||||
|
// Shared constants for background scripts
|
||||||
|
window.MAC_CONSTANTS = {
|
||||||
|
OPEN_CONTAINER_PREFIX: "open_container_",
|
||||||
|
REOPEN_IN_CONTAINER_PREFIX: "reopen_in_container_",
|
||||||
|
NUMBER_OF_KEYBOARD_SHORTCUTS: 10,
|
||||||
|
};
|
||||||
@@ -1,3 +1,8 @@
|
|||||||
|
/* 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/. */
|
||||||
|
|
||||||
|
/* global MAC_CONSTANTS */
|
||||||
window.identityState = {
|
window.identityState = {
|
||||||
keyboardShortcut: {},
|
keyboardShortcut: {},
|
||||||
storageArea: {
|
storageArea: {
|
||||||
@@ -50,18 +55,22 @@ window.identityState = {
|
|||||||
|
|
||||||
async loadKeyboardShortcuts () {
|
async loadKeyboardShortcuts () {
|
||||||
const identities = await browser.contextualIdentities.query({});
|
const identities = await browser.contextualIdentities.query({});
|
||||||
for (let i=0; i < backgroundLogic.NUMBER_OF_KEYBOARD_SHORTCUTS; i++) {
|
for (let i=0; i < MAC_CONSTANTS.NUMBER_OF_KEYBOARD_SHORTCUTS; i++) {
|
||||||
const key = "open_container_" + i;
|
const openKey = MAC_CONSTANTS.OPEN_CONTAINER_PREFIX + i;
|
||||||
const storageObject = await this.area.get(key);
|
const reopenKey = MAC_CONSTANTS.REOPEN_IN_CONTAINER_PREFIX + i;
|
||||||
if (storageObject[key]){
|
|
||||||
identityState.keyboardShortcut[key] = storageObject[key];
|
for (const key of [openKey, reopenKey]) {
|
||||||
continue;
|
const storageObject = await this.area.get(key);
|
||||||
|
|
||||||
|
if (storageObject[key]){
|
||||||
|
identityState.keyboardShortcut[key] = storageObject[key];
|
||||||
|
} else if (identities[i]) {
|
||||||
|
identityState.keyboardShortcut[key] = identities[i].cookieStoreId;
|
||||||
|
} else {
|
||||||
|
identityState.keyboardShortcut[key] = "none";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (identities[i]) {
|
|
||||||
identityState.keyboardShortcut[key] = identities[i].cookieStoreId;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
identityState.keyboardShortcut[key] = "none";
|
|
||||||
}
|
}
|
||||||
return identityState.keyboardShortcut;
|
return identityState.keyboardShortcut;
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -15,6 +15,7 @@
|
|||||||
-->
|
-->
|
||||||
<script type="text/javascript" src="../utils.js"></script>
|
<script type="text/javascript" src="../utils.js"></script>
|
||||||
<script type="text/javascript" src="../proxified-containers.js"></script>
|
<script type="text/javascript" src="../proxified-containers.js"></script>
|
||||||
|
<script type="text/javascript" src="constants.js"></script>
|
||||||
<script type="text/javascript" src="backgroundLogic.js"></script>
|
<script type="text/javascript" src="backgroundLogic.js"></script>
|
||||||
<script type="text/javascript" src="mozillaVpnBackground.js"></script>
|
<script type="text/javascript" src="mozillaVpnBackground.js"></script>
|
||||||
<script type="text/javascript" src="assignManager.js"></script>
|
<script type="text/javascript" src="assignManager.js"></script>
|
||||||
|
|||||||
@@ -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
|
||||||
@@ -137,7 +141,6 @@ const messageHandler = {
|
|||||||
if (!extensionInfo.permissions.includes("contextualIdentities")) {
|
if (!extensionInfo.permissions.includes("contextualIdentities")) {
|
||||||
throw new Error("Missing contextualIdentities permission");
|
throw new Error("Missing contextualIdentities permission");
|
||||||
}
|
}
|
||||||
// eslint-disable-next-line require-atomic-updates
|
|
||||||
externalExtensionAllowed[sender.id] = true;
|
externalExtensionAllowed[sender.id] = true;
|
||||||
}
|
}
|
||||||
let response;
|
let response;
|
||||||
@@ -270,7 +273,7 @@ const messageHandler = {
|
|||||||
}).catch((e) => {
|
}).catch((e) => {
|
||||||
throw e;
|
throw e;
|
||||||
});
|
});
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
// Lets do this last as theme manager did a check before connecting before
|
// Lets do this last as theme manager did a check before connecting before
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ const MozillaVPN_Background = {
|
|||||||
// invalid proxy connection.
|
// invalid proxy connection.
|
||||||
this.port.onDisconnect.addListener(() => this.increaseIsolationKey());
|
this.port.onDisconnect.addListener(() => this.increaseIsolationKey());
|
||||||
|
|
||||||
} catch(e) {
|
} catch {
|
||||||
this._installed = false;
|
this._installed = false;
|
||||||
this._connected = false;
|
this._connected = false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -338,7 +338,7 @@ async function reconcileIdentities(){
|
|||||||
if (deletedCookieStoreId){
|
if (deletedCookieStoreId){
|
||||||
try{
|
try{
|
||||||
await browser.contextualIdentities.remove(deletedCookieStoreId);
|
await browser.contextualIdentities.remove(deletedCookieStoreId);
|
||||||
} catch (error) {
|
} catch {
|
||||||
// if the identity we are deleting is not there, that's fine.
|
// if the identity we are deleting is not there, that's fine.
|
||||||
console.error("Error deleting contextualIdentity", deletedCookieStoreId);
|
console.error("Error deleting contextualIdentity", deletedCookieStoreId);
|
||||||
continue;
|
continue;
|
||||||
@@ -514,7 +514,7 @@ async function reconcileSiteAssignments() {
|
|||||||
await setAssignmentWithUUID(assignedSite, urlKey);
|
await setAssignmentWithUUID(assignedSite, urlKey);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch {
|
||||||
// this is probably old or incorrect site info in Sync
|
// this is probably old or incorrect site info in Sync
|
||||||
// skip and move on.
|
// skip and move on.
|
||||||
}
|
}
|
||||||
@@ -565,7 +565,6 @@ async function setAssignmentWithUUID(assignedSite, urlKey) {
|
|||||||
const uuid = assignedSite.identityMacAddonUUID;
|
const uuid = assignedSite.identityMacAddonUUID;
|
||||||
const cookieStoreId = await identityState.lookupCookieStoreId(uuid);
|
const cookieStoreId = await identityState.lookupCookieStoreId(uuid);
|
||||||
if (cookieStoreId) {
|
if (cookieStoreId) {
|
||||||
// eslint-disable-next-line require-atomic-updates
|
|
||||||
assignedSite.userContextId = cookieStoreId
|
assignedSite.userContextId = cookieStoreId
|
||||||
.replace(/^firefox-container-/, "");
|
.replace(/^firefox-container-/, "");
|
||||||
await assignManager.storageArea.set(
|
await assignManager.storageArea.set(
|
||||||
|
|||||||
+14
-7
@@ -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);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -88,7 +88,7 @@ const MozillaVPN = {
|
|||||||
el.classList.remove("display-none");
|
el.classList.remove("display-none");
|
||||||
});
|
});
|
||||||
this.setStatusIndicatorIcons(mozillaVpnInstalled);
|
this.setStatusIndicatorIcons(mozillaVpnInstalled);
|
||||||
} catch (e) {
|
} catch {
|
||||||
mozVpnLogotypes.forEach(el => {
|
mozVpnLogotypes.forEach(el => {
|
||||||
el.style.display = "none";
|
el.style.display = "none";
|
||||||
});
|
});
|
||||||
@@ -139,7 +139,7 @@ const MozillaVPN = {
|
|||||||
try {
|
try {
|
||||||
const proxy = await proxifiedContainers.retrieve(identity.cookieStoreId);
|
const proxy = await proxifiedContainers.retrieve(identity.cookieStoreId);
|
||||||
proxies[identity.cookieStoreId] = proxy;
|
proxies[identity.cookieStoreId] = proxy;
|
||||||
} catch (e) {
|
} catch {
|
||||||
proxies[identity.cookieStoreId] = {};
|
proxies[identity.cookieStoreId] = {};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+31
-48
@@ -137,19 +137,26 @@ const Logic = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
async showAchievementOrContainersListPanel() {
|
async showAchievementOrContainersListPanel() {
|
||||||
// Do we need to show an achievement panel?
|
|
||||||
let showAchievements = false;
|
|
||||||
const achievementsStorage = await browser.storage.local.get({ achievements: [] });
|
const achievementsStorage = await browser.storage.local.get({ achievements: [] });
|
||||||
for (const achievement of achievementsStorage.achievements) {
|
const achievements = achievementsStorage.achievements;
|
||||||
if (!achievement.done) {
|
|
||||||
showAchievements = true;
|
let saveAchievements = false;
|
||||||
|
for (const achievement of achievements.filter(a => !a.done)) {
|
||||||
|
if (achievement.name === "manyContainersOpened") {
|
||||||
|
this.showPanel(P_CONTAINERS_ACHIEVEMENT);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We have found an unknown achievement. Let's mark it as done.
|
||||||
|
achievement.done = true;
|
||||||
|
saveAchievements = true;
|
||||||
}
|
}
|
||||||
if (showAchievements) {
|
|
||||||
this.showPanel(P_CONTAINERS_ACHIEVEMENT);
|
if (saveAchievements) {
|
||||||
} else {
|
browser.storage.local.set({ achievements });
|
||||||
this.showPanel(P_CONTAINERS_LIST);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.showPanel(P_CONTAINERS_LIST);
|
||||||
},
|
},
|
||||||
|
|
||||||
// In case the user wants to click multiple actions,
|
// In case the user wants to click multiple actions,
|
||||||
@@ -198,7 +205,7 @@ const Logic = {
|
|||||||
// Handle old style rejection with null and also Promise.reject new style
|
// Handle old style rejection with null and also Promise.reject new style
|
||||||
try {
|
try {
|
||||||
return await browser.contextualIdentities.get(cookieStoreId) || defaultContainer;
|
return await browser.contextualIdentities.get(cookieStoreId) || defaultContainer;
|
||||||
} catch (e) {
|
} catch {
|
||||||
return defaultContainer;
|
return defaultContainer;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -425,7 +432,7 @@ const Logic = {
|
|||||||
cookieStoreId: identity.cookieStoreId
|
cookieStoreId: identity.cookieStoreId
|
||||||
});
|
});
|
||||||
window.close();
|
window.close();
|
||||||
} catch (e) {
|
} catch {
|
||||||
window.close();
|
window.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -762,12 +769,11 @@ Logic.registerPanel(P_CONTAINERS_LIST, {
|
|||||||
method: "sortTabs"
|
method: "sortTabs"
|
||||||
});
|
});
|
||||||
window.close();
|
window.close();
|
||||||
} catch (e) {
|
} catch {
|
||||||
window.close();
|
window.close();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
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 +782,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);
|
||||||
@@ -877,7 +858,7 @@ Logic.registerPanel(P_CONTAINERS_LIST, {
|
|||||||
cookieStoreId: identity.cookieStoreId
|
cookieStoreId: identity.cookieStoreId
|
||||||
});
|
});
|
||||||
window.close();
|
window.close();
|
||||||
} catch (e) {
|
} catch {
|
||||||
window.close();
|
window.close();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -888,7 +869,7 @@ Logic.registerPanel(P_CONTAINERS_LIST, {
|
|||||||
cookieStoreId: identity.cookieStoreId
|
cookieStoreId: identity.cookieStoreId
|
||||||
});
|
});
|
||||||
window.close();
|
window.close();
|
||||||
} catch (e) {
|
} catch {
|
||||||
window.close();
|
window.close();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -938,7 +919,7 @@ Logic.registerPanel(P_CONTAINER_INFO, {
|
|||||||
incompatible = await browser.runtime.sendMessage({
|
incompatible = await browser.runtime.sendMessage({
|
||||||
method: "checkIncompatibleAddons"
|
method: "checkIncompatibleAddons"
|
||||||
});
|
});
|
||||||
} catch (e) {
|
} catch {
|
||||||
throw new Error("Could not check for incompatible add-ons.");
|
throw new Error("Could not check for incompatible add-ons.");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -973,7 +954,7 @@ Logic.registerPanel(P_CONTAINER_INFO, {
|
|||||||
cookieStoreId: identity.cookieStoreId
|
cookieStoreId: identity.cookieStoreId
|
||||||
});
|
});
|
||||||
window.close();
|
window.close();
|
||||||
} catch (e) {
|
} catch {
|
||||||
window.close();
|
window.close();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -1035,7 +1016,7 @@ Logic.registerPanel(P_CONTAINER_INFO, {
|
|||||||
cookieStoreId: Logic.currentCookieStoreId()
|
cookieStoreId: Logic.currentCookieStoreId()
|
||||||
});
|
});
|
||||||
window.close();
|
window.close();
|
||||||
} catch (e) {
|
} catch {
|
||||||
window.close();
|
window.close();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -1112,7 +1093,7 @@ Logic.registerPanel(OPEN_NEW_CONTAINER_PICKER, {
|
|||||||
cookieStoreId: identity.cookieStoreId
|
cookieStoreId: identity.cookieStoreId
|
||||||
});
|
});
|
||||||
window.close();
|
window.close();
|
||||||
} catch (e) {
|
} catch {
|
||||||
window.close();
|
window.close();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -1306,7 +1287,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 +1318,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();
|
||||||
});
|
});
|
||||||
@@ -1872,7 +1855,7 @@ Logic.registerPanel(P_CONTAINER_EDIT, {
|
|||||||
});
|
});
|
||||||
await Logic.refreshIdentities();
|
await Logic.refreshIdentities();
|
||||||
Logic.showPreviousPanel();
|
Logic.showPreviousPanel();
|
||||||
} catch (e) {
|
} catch {
|
||||||
Logic.showPreviousPanel();
|
Logic.showPreviousPanel();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -2363,7 +2346,7 @@ Logic.registerPanel(P_CONTAINER_DELETE, {
|
|||||||
await Logic.removeIdentity(Utils.userContextId(Logic.currentIdentity().cookieStoreId));
|
await Logic.removeIdentity(Utils.userContextId(Logic.currentIdentity().cookieStoreId));
|
||||||
await Logic.refreshIdentities();
|
await Logic.refreshIdentities();
|
||||||
Logic.showPreviousPanel();
|
Logic.showPreviousPanel();
|
||||||
} catch (e) {
|
} catch {
|
||||||
Logic.showPreviousPanel();
|
Logic.showPreviousPanel();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -65,6 +65,10 @@ proxifiedContainers = {
|
|||||||
async delete(cookieStoreId) {
|
async delete(cookieStoreId) {
|
||||||
// Assumes proxy is a properly formatted object
|
// Assumes proxy is a properly formatted object
|
||||||
const proxifiedContainersStore = await proxifiedContainers.retrieveAll();
|
const proxifiedContainersStore = await proxifiedContainers.retrieveAll();
|
||||||
|
if(!proxifiedContainersStore) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
const index = proxifiedContainersStore.findIndex(i => i.cookieStoreId === cookieStoreId);
|
const index = proxifiedContainersStore.findIndex(i => i.cookieStoreId === cookieStoreId);
|
||||||
if (index !== -1) {
|
if (index !== -1) {
|
||||||
proxifiedContainersStore.splice(index, 1);
|
proxifiedContainersStore.splice(index, 1);
|
||||||
|
|||||||
+17
-3
@@ -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(
|
||||||
|
|||||||
+35
-6
@@ -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.7",
|
||||||
"incognito": "not_allowed",
|
"incognito": "not_allowed",
|
||||||
"description": "__MSG_extensionDescription__",
|
"description": "__MSG_extensionDescription__",
|
||||||
"icons": {
|
"icons": {
|
||||||
@@ -33,7 +33,10 @@
|
|||||||
"browser_specific_settings": {
|
"browser_specific_settings": {
|
||||||
"gecko": {
|
"gecko": {
|
||||||
"id": "@testpilot-containers",
|
"id": "@testpilot-containers",
|
||||||
"strict_min_version": "91.1.0"
|
"strict_min_version": "91.1.0",
|
||||||
|
"data_collection_permissions": {
|
||||||
|
"required": ["none"]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"commands": {
|
"commands": {
|
||||||
@@ -45,10 +48,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": {
|
||||||
@@ -110,6 +109,36 @@
|
|||||||
"default": "Ctrl+Shift+0"
|
"default": "Ctrl+Shift+0"
|
||||||
},
|
},
|
||||||
"description": "__MSG_containerShortcut__"
|
"description": "__MSG_containerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_0": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_1": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_2": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_3": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_4": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_5": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_6": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_7": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_8": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
|
},
|
||||||
|
"reopen_in_container_9": {
|
||||||
|
"description": "__MSG_reopenInContainerShortcut__"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"browser_action": {
|
"browser_action": {
|
||||||
|
|||||||
+2
-12
@@ -3,6 +3,7 @@
|
|||||||
<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>Firefox Multi-Account Containers</title>
|
||||||
<script type="text/javascript" src="./js/i18n.js"></script>
|
<script type="text/javascript" src="./js/i18n.js"></script>
|
||||||
|
<meta name="color-scheme" content="light dark">
|
||||||
<link rel="stylesheet" href="./css/popup.css">
|
<link rel="stylesheet" href="./css/popup.css">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
@@ -192,18 +193,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>
|
||||||
|
|||||||
@@ -1,12 +0,0 @@
|
|||||||
module.exports = {
|
|
||||||
env: {
|
|
||||||
"node": true,
|
|
||||||
"mocha": true
|
|
||||||
},
|
|
||||||
"parserOptions": {
|
|
||||||
"ecmaVersion": 2018
|
|
||||||
},
|
|
||||||
"rules": {
|
|
||||||
"no-restricted-globals": ["error", "browser"]
|
|
||||||
}
|
|
||||||
};
|
|
||||||
+4
-1
@@ -1,5 +1,4 @@
|
|||||||
if (!process.listenerCount("unhandledRejection")) {
|
if (!process.listenerCount("unhandledRejection")) {
|
||||||
// eslint-disable-next-line no-console
|
|
||||||
process.on("unhandledRejection", r => console.log(r));
|
process.on("unhandledRejection", r => console.log(r));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -32,6 +31,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();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -0,0 +1,44 @@
|
|||||||
|
const {initializeWithTab} = require("../common");
|
||||||
|
|
||||||
|
describe("Reopen Shortcuts Feature", function () {
|
||||||
|
beforeEach(async function () {
|
||||||
|
// Initialize with a tab in the default container
|
||||||
|
this.webExt = await initializeWithTab({
|
||||||
|
cookieStoreId: "firefox-default",
|
||||||
|
url: "https://example.com"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(function () {
|
||||||
|
this.webExt.destroy();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when using keyboard shortcut to reopen in container", function () {
|
||||||
|
beforeEach(async function () {
|
||||||
|
// Simulate the keyboard shortcut command
|
||||||
|
await this.webExt.background.browser.commands.onCommand.addListener.firstCall.args[0]("reopen_in_container_0");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should open the page in the assigned container and close the original tab", async function () {
|
||||||
|
this.webExt.background.browser.tabs.create.should.have.been.calledWithMatch({
|
||||||
|
url: "https://example.com",
|
||||||
|
cookieStoreId: "firefox-container-1",
|
||||||
|
index: 1,
|
||||||
|
active: true
|
||||||
|
});
|
||||||
|
|
||||||
|
this.webExt.background.browser.tabs.remove.should.have.been.called;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when container is set to 'none'", function () {
|
||||||
|
beforeEach(async function () {
|
||||||
|
await this.webExt.background.browser.commands.onCommand.addListener.firstCall.args[0]("reopen_in_container_9");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should not reopen the tab", function () {
|
||||||
|
this.webExt.background.browser.tabs.create.should.not.have.been.called;
|
||||||
|
this.webExt.background.browser.tabs.remove.should.not.have.been.called;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
Reference in New Issue
Block a user