diff --git a/Makefile b/Makefile
index 9e3ca29..1e193cb 100644
--- a/Makefile
+++ b/Makefile
@@ -2,6 +2,8 @@
PREFIX := /usr
DATADIR := $(PREFIX)/share
+# the profile dir is copied and left unmodified by web-ext
+PROFILE := /home/xavier/.thunderbird/6ujxy9ck.default-release-2/
# UUID below is for Thunderbird
TB_EXT_DIR := $(DATADIR)/mozilla/extensions/{3550f703-e582-4d05-9a08-453d09bdfdc6}
@@ -17,3 +19,8 @@ clean:
install: all
install -d $(DESTDIR)$(TB_EXT_DIR)
install --mode=644 *.xpi $(DESTDIR)$(TB_EXT_DIR)
+
+run:
+ web-ext run --verbose --firefox=/usr/bin/thunderbird --firefox-profile=$(PROFILE) -s xul-ext/
+lint:
+ web-ext lint -s xul-ext/
\ No newline at end of file
diff --git a/xul-ext/api/windowListener/implementation.js b/xul-ext/api/windowListener/implementation.js
new file mode 100644
index 0000000..2d65b1a
--- /dev/null
+++ b/xul-ext/api/windowListener/implementation.js
@@ -0,0 +1,1196 @@
+/*
+ * This file is provided by the addon-developer-support repository at
+ * https://github.com/thundernest/addon-developer-support
+ *
+ * Version: 1.56
+ *
+ * Author: John Bieling (john@thunderbird.net)
+ *
+ * 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/.
+ */
+
+// Import some things we need.
+var { ExtensionCommon } = ChromeUtils.import(
+ "resource://gre/modules/ExtensionCommon.jsm"
+);
+var { ExtensionSupport } = ChromeUtils.import(
+ "resource:///modules/ExtensionSupport.jsm"
+);
+var { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm");
+
+var WindowListener = class extends ExtensionCommon.ExtensionAPI {
+ log(msg) {
+ if (this.debug) console.log("WindowListener API: " + msg);
+ }
+
+ getThunderbirdVersion() {
+ let parts = Services.appinfo.version.split(".");
+ return {
+ major: parseInt(parts[0]),
+ minor: parseInt(parts[1]),
+ revision: parts.length > 2 ? parseInt(parts[2]) : 0,
+ }
+ }
+
+ getCards(e) {
+ // This gets triggered by real events but also manually by providing the outer window.
+ // The event is attached to the outer browser, get the inner one.
+ let doc;
+
+ // 78,86, and 87+ need special handholding. *Yeah*.
+ if (this.getThunderbirdVersion().major < 86) {
+ let ownerDoc = e.document || e.target.ownerDocument;
+ doc = ownerDoc.getElementById("html-view-browser").contentDocument;
+ } else if (this.getThunderbirdVersion().major < 87) {
+ let ownerDoc = e.document || e.target;
+ doc = ownerDoc.getElementById("html-view-browser").contentDocument;
+ } else {
+ doc = e.document || e.target;
+ }
+ return doc.querySelectorAll("addon-card");
+ }
+
+ // Add pref entry to 68
+ add68PrefsEntry(event) {
+ let id = this.menu_addonPrefs_id + "_" + this.uniqueRandomID;
+
+ // Get the best size of the icon (16px or bigger)
+ let iconSizes = this.extension.manifest.icons
+ ? Object.keys(this.extension.manifest.icons)
+ : [];
+ iconSizes.sort((a, b) => a - b);
+ let bestSize = iconSizes.filter((e) => parseInt(e) >= 16).shift();
+ let icon = bestSize ? this.extension.manifest.icons[bestSize] : "";
+
+ let name = this.extension.manifest.name;
+ let entry = icon
+ ? event.target.ownerGlobal.MozXULElement.parseXULToFragment(
+ ``
+ )
+ : event.target.ownerGlobal.MozXULElement.parseXULToFragment(
+ ``
+ );
+
+ event.target.appendChild(entry);
+ let noPrefsElem = event.target.querySelector('[disabled="true"]');
+ // using collapse could be undone by core, so we use display none
+ // noPrefsElem.setAttribute("collapsed", "true");
+ noPrefsElem.style.display = "none";
+ event.target.ownerGlobal.document
+ .getElementById(id)
+ .addEventListener("command", this);
+ }
+
+ // Event handler for the addon manager, to update the state of the options button.
+ handleEvent(e) {
+ switch (e.type) {
+ // 68 add-on options menu showing
+ case "popupshowing":
+ {
+ this.add68PrefsEntry(e);
+ }
+ break;
+
+ // 78/88 add-on options menu/button click
+ case "click":
+ {
+ e.preventDefault();
+ e.stopPropagation();
+ let WL = {};
+ WL.extension = this.extension;
+ WL.messenger = this.getMessenger(this.context);
+ let w = Services.wm.getMostRecentWindow("mail:3pane");
+ w.openDialog(
+ this.pathToOptionsPage,
+ "AddonOptions",
+ "chrome,resizable,centerscreen",
+ WL
+ );
+ }
+ break;
+
+ // 68 add-on options menu command
+ case "command":
+ {
+ let WL = {};
+ WL.extension = this.extension;
+ WL.messenger = this.getMessenger(this.context);
+ e.target.ownerGlobal.openDialog(
+ this.pathToOptionsPage,
+ "AddonOptions",
+ "chrome,resizable,centerscreen",
+ WL
+ );
+ }
+ break;
+
+ // update, ViewChanged and manual call for add-on manager options overlay
+ default: {
+ let cards = this.getCards(e);
+ for (let card of cards) {
+ // Setup either the options entry in the menu or the button
+ //window.document.getElementById(id).addEventListener("command", function() {window.openDialog(self.pathToOptionsPage, "AddonOptions", "chrome,resizable,centerscreen", WL)});
+ if (card.addon.id == this.extension.id) {
+ let optionsMenu =
+ (this.getThunderbirdVersion().major > 78 && this.getThunderbirdVersion().major < 88) ||
+ (this.getThunderbirdVersion().major == 78 && this.getThunderbirdVersion().minor < 10) ||
+ (this.getThunderbirdVersion().major == 78 && this.getThunderbirdVersion().minor == 10 && this.getThunderbirdVersion().revision < 2);
+ if (optionsMenu) {
+ // Options menu in 78.0-78.10 and 79-87
+ let addonOptionsLegacyEntry = card.querySelector(
+ ".extension-options-legacy"
+ );
+ if (card.addon.isActive && !addonOptionsLegacyEntry) {
+ let addonOptionsEntry = card.querySelector(
+ "addon-options panel-list panel-item[action='preferences']"
+ );
+ addonOptionsLegacyEntry = card.ownerDocument.createElement(
+ "panel-item"
+ );
+ addonOptionsLegacyEntry.setAttribute(
+ "data-l10n-id",
+ "preferences-addon-button"
+ );
+ addonOptionsLegacyEntry.classList.add(
+ "extension-options-legacy"
+ );
+ addonOptionsEntry.parentNode.insertBefore(
+ addonOptionsLegacyEntry,
+ addonOptionsEntry
+ );
+ card
+ .querySelector(".extension-options-legacy")
+ .addEventListener("click", this);
+ } else if (!card.addon.isActive && addonOptionsLegacyEntry) {
+ addonOptionsLegacyEntry.remove();
+ }
+ } else {
+ // Add-on button in 88
+ let addonOptionsButton = card.querySelector(
+ ".windowlistener-options-button"
+ );
+ if (card.addon.isActive && !addonOptionsButton) {
+ addonOptionsButton = card.ownerDocument.createElement("button");
+ addonOptionsButton.classList.add("windowlistener-options-button");
+ addonOptionsButton.classList.add("extension-options-button");
+ card.optionsButton.parentNode.insertBefore(
+ addonOptionsButton,
+ card.optionsButton
+ );
+ card
+ .querySelector(".windowlistener-options-button")
+ .addEventListener("click", this);
+ } else if (!card.addon.isActive && addonOptionsButton) {
+ addonOptionsButton.remove();
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+
+ // Some tab/add-on-manager related functions
+ getTabMail(window) {
+ return window.document.getElementById("tabmail");
+ }
+
+ // returns the outer browser, not the nested browser of the add-on manager
+ // events must be attached to the outer browser
+ getAddonManagerFromTab(tab) {
+ if (tab.browser) {
+ let win = tab.browser.contentWindow;
+ if (win && win.location.href == "about:addons") {
+ return win;
+ }
+ }
+ }
+
+ getAddonManagerFromWindow(window) {
+ let tabMail = this.getTabMail(window);
+ for (let tab of tabMail.tabInfo) {
+ let win = this.getAddonManagerFromTab(tab);
+ if (win) {
+ return win;
+ }
+ }
+ }
+
+ setupAddonManager(managerWindow, forceLoad = false) {
+ if (!managerWindow) {
+ return;
+ }
+ if (!(
+ managerWindow &&
+ managerWindow[this.uniqueRandomID] &&
+ managerWindow[this.uniqueRandomID].hasAddonManagerEventListeners
+ )) {
+ managerWindow.document.addEventListener("ViewChanged", this);
+ managerWindow.document.addEventListener("update", this);
+ managerWindow.document.addEventListener("view-loaded", this);
+ managerWindow[this.uniqueRandomID] = {};
+ managerWindow[this.uniqueRandomID].hasAddonManagerEventListeners = true;
+ }
+ if (forceLoad) this.handleEvent(managerWindow);
+ }
+
+ getMessenger(context) {
+ let apis = ["storage", "runtime", "extension", "i18n"];
+
+ function getStorage() {
+ let localstorage = null;
+ try {
+ localstorage = context.apiCan.findAPIPath("storage");
+ localstorage.local.get = (...args) =>
+ localstorage.local.callMethodInParentProcess("get", args);
+ localstorage.local.set = (...args) =>
+ localstorage.local.callMethodInParentProcess("set", args);
+ localstorage.local.remove = (...args) =>
+ localstorage.local.callMethodInParentProcess("remove", args);
+ localstorage.local.clear = (...args) =>
+ localstorage.local.callMethodInParentProcess("clear", args);
+ } catch (e) {
+ console.info("Storage permission is missing");
+ }
+ return localstorage;
+ }
+
+ let messenger = {};
+ for (let api of apis) {
+ switch (api) {
+ case "storage":
+ XPCOMUtils.defineLazyGetter(messenger, "storage", () => getStorage());
+ break;
+
+ default:
+ XPCOMUtils.defineLazyGetter(messenger, api, () =>
+ context.apiCan.findAPIPath(api)
+ );
+ }
+ }
+ return messenger;
+ }
+
+ error(msg) {
+ if (this.debug) console.error("WindowListener API: " + msg);
+ }
+
+ // async sleep function using Promise
+ async sleep(delay) {
+ let timer = Components.classes["@mozilla.org/timer;1"].createInstance(
+ Components.interfaces.nsITimer
+ );
+ return new Promise(function (resolve, reject) {
+ let event = {
+ notify: function (timer) {
+ resolve();
+ },
+ };
+ timer.initWithCallback(
+ event,
+ delay,
+ Components.interfaces.nsITimer.TYPE_ONE_SHOT
+ );
+ });
+ }
+
+ getAPI(context) {
+ // Track if this is the background/main context
+ if (context.viewType != "background")
+ throw new Error(
+ "The WindowListener API may only be called from the background page."
+ );
+
+ this.context = context;
+
+ this.uniqueRandomID = "AddOnNS" + context.extension.instanceId;
+ this.menu_addonPrefs_id = "addonPrefs";
+
+ this.registeredWindows = {};
+ this.pathToStartupScript = null;
+ this.pathToShutdownScript = null;
+ this.pathToOptionsPage = null;
+ this.chromeHandle = null;
+ this.chromeData = null;
+ this.resourceData = null;
+ this.openWindows = [];
+ this.debug = context.extension.addonData.temporarilyInstalled;
+
+ const aomStartup = Cc[
+ "@mozilla.org/addons/addon-manager-startup;1"
+ ].getService(Ci.amIAddonManagerStartup);
+ const resProto = Cc[
+ "@mozilla.org/network/protocol;1?name=resource"
+ ].getService(Ci.nsISubstitutingProtocolHandler);
+
+ let self = this;
+
+ // TabMonitor to detect opening of tabs, to setup the options button in the add-on manager.
+ this.tabMonitor = {
+ onTabTitleChanged(aTab) { },
+ onTabClosing(aTab) { },
+ onTabPersist(aTab) { },
+ onTabRestored(aTab) { },
+ onTabSwitched(aNewTab, aOldTab) {
+ //self.setupAddonManager(self.getAddonManagerFromTab(aNewTab));
+ },
+ async onTabOpened(aTab) {
+ if (aTab.browser) {
+ if (!aTab.pageLoaded) {
+ // await a location change if browser is not loaded yet
+ await new Promise((resolve) => {
+ let reporterListener = {
+ QueryInterface: ChromeUtils.generateQI([
+ "nsIWebProgressListener",
+ "nsISupportsWeakReference",
+ ]),
+ onStateChange() { },
+ onProgressChange() { },
+ onLocationChange(
+ /* in nsIWebProgress*/ aWebProgress,
+ /* in nsIRequest*/ aRequest,
+ /* in nsIURI*/ aLocation
+ ) {
+ aTab.browser.removeProgressListener(reporterListener);
+ resolve();
+ },
+ onStatusChange() { },
+ onSecurityChange() { },
+ onContentBlockingEvent() { },
+ };
+ aTab.browser.addProgressListener(reporterListener);
+ });
+ }
+ self.setupAddonManager(self.getAddonManagerFromTab(aTab));
+ }
+ },
+ };
+
+ return {
+ WindowListener: {
+ async waitForMasterPassword() {
+ // Wait until master password has been entered (if needed)
+ while (!Services.logins.isLoggedIn) {
+ self.log("Waiting for master password.");
+ await self.sleep(1000);
+ }
+ self.log("Master password has been entered.");
+ },
+
+ aDocumentExistsAt(uriString) {
+ self.log(
+ "Checking if document at <" +
+ uriString +
+ "> used in registration actually exists."
+ );
+ try {
+ let uriObject = Services.io.newURI(uriString);
+ } catch (e) {
+ Components.utils.reportError(e);
+ return false;
+ }
+ return true;
+ },
+
+ registerOptionsPage(optionsUrl) {
+ self.pathToOptionsPage = optionsUrl.startsWith("chrome://")
+ ? optionsUrl
+ : context.extension.rootURI.resolve(optionsUrl);
+ },
+
+ registerDefaultPrefs(defaultUrl) {
+ let url = context.extension.rootURI.resolve(defaultUrl);
+
+ let prefsObj = {};
+ prefsObj.Services = ChromeUtils.import(
+ "resource://gre/modules/Services.jsm"
+ ).Services;
+ prefsObj.pref = function (aName, aDefault) {
+ let defaults = Services.prefs.getDefaultBranch("");
+ switch (typeof aDefault) {
+ case "string":
+ return defaults.setStringPref(aName, aDefault);
+
+ case "number":
+ return defaults.setIntPref(aName, aDefault);
+
+ case "boolean":
+ return defaults.setBoolPref(aName, aDefault);
+
+ default:
+ throw new Error(
+ "Preference <" +
+ aName +
+ "> has an unsupported type <" +
+ typeof aDefault +
+ ">. Allowed are string, number and boolean."
+ );
+ }
+ };
+ Services.scriptloader.loadSubScript(url, prefsObj, "UTF-8");
+ },
+
+ registerChromeUrl(data) {
+ let chromeData = [];
+ let resourceData = [];
+ for (let entry of data) {
+ if (entry[0] == "resource") resourceData.push(entry);
+ else chromeData.push(entry);
+ }
+
+ if (chromeData.length > 0) {
+ const manifestURI = Services.io.newURI(
+ "manifest.json",
+ null,
+ context.extension.rootURI
+ );
+ self.chromeHandle = aomStartup.registerChrome(
+ manifestURI,
+ chromeData
+ );
+ }
+
+ for (let res of resourceData) {
+ // [ "resource", "shortname" , "path" ]
+ let uri = Services.io.newURI(
+ res[2],
+ null,
+ context.extension.rootURI
+ );
+ resProto.setSubstitutionWithFlags(
+ res[1],
+ uri,
+ resProto.ALLOW_CONTENT_ACCESS
+ );
+ }
+
+ self.chromeData = chromeData;
+ self.resourceData = resourceData;
+ },
+
+ registerWindow(windowHref, jsFile) {
+ if (self.debug && !this.aDocumentExistsAt(windowHref)) {
+ self.error(
+ "Attempt to register an injector script for non-existent window: " +
+ windowHref
+ );
+ return;
+ }
+
+ if (!self.registeredWindows.hasOwnProperty(windowHref)) {
+ // path to JS file can either be chrome:// URL or a relative URL
+ let path = jsFile.startsWith("chrome://")
+ ? jsFile
+ : context.extension.rootURI.resolve(jsFile);
+ if (windowHref == "chrome://messenger/content/commonDialog.xhtml") {
+ // @XN-BUGFIX
+ // there is a bug somewhere with aDocumentExistsAt and this very specific window
+ // the existence test fails for chrome://global/content/commonDialog.xhtml
+ // and succeed for chrome://messenger/content/commonDialog.xhtml
+ // BUT this commonDialog from messenger fails to _loadIntoWindow(window, isAddonActivation) and does not activate
+ // WORKAROUD : set xhtml namespace to global for this url when it's supposed to be in messenger
+ self.log(
+ "reverting chrome commonDialog from messenger to global"
+ );
+ windowHref = "chrome://global/content/commonDialog.xhtml";
+ }
+ self.registeredWindows[windowHref] = path;
+ } else {
+ self.error(
+ "Window <" + windowHref + "> has already been registered"
+ );
+ }
+ },
+
+ registerStartupScript(aPath) {
+ self.pathToStartupScript = aPath.startsWith("chrome://")
+ ? aPath
+ : context.extension.rootURI.resolve(aPath);
+ },
+
+ registerShutdownScript(aPath) {
+ self.pathToShutdownScript = aPath.startsWith("chrome://")
+ ? aPath
+ : context.extension.rootURI.resolve(aPath);
+ },
+
+ openOptionsDialog(windowId) {
+ let window = context.extension.windowManager.get(windowId, context)
+ .window;
+ let WL = {};
+ WL.extension = self.extension;
+ WL.messenger = self.getMessenger(self.context);
+ window.openDialog(
+ self.pathToOptionsPage,
+ "AddonOptions",
+ "chrome,resizable,centerscreen",
+ WL
+ );
+ },
+
+ async startListening() {
+ // load the registered startup script, if one has been registered
+ // (mail3:pane may not have been fully loaded yet)
+ if (self.pathToStartupScript) {
+ let startupJS = {};
+ startupJS.WL = {};
+ startupJS.WL.extension = self.extension;
+ startupJS.WL.messenger = self.getMessenger(self.context);
+ try {
+ if (self.pathToStartupScript) {
+ Services.scriptloader.loadSubScript(
+ self.pathToStartupScript,
+ startupJS,
+ "UTF-8"
+ );
+ // delay startup until startup has been finished
+ self.log(
+ "Waiting for async startup() in <" +
+ self.pathToStartupScript +
+ "> to finish."
+ );
+ if (startupJS.startup) {
+ await startupJS.startup();
+ self.log(
+ "startup() in <" + self.pathToStartupScript + "> finished"
+ );
+ } else {
+ self.log(
+ "No startup() in <" + self.pathToStartupScript + "> found."
+ );
+ }
+ }
+ } catch (e) {
+ Components.utils.reportError(e);
+ }
+ }
+
+ let urls = Object.keys(self.registeredWindows);
+ if (urls.length > 0) {
+ // Before registering the window listener, check which windows are already open
+ self.openWindows = [];
+ for (let window of Services.wm.getEnumerator(null)) {
+ self.openWindows.push(window);
+ }
+
+ // Register window listener for all pre-registered windows
+ ExtensionSupport.registerWindowListener(
+ "injectListener_" + self.uniqueRandomID,
+ {
+ // React on all windows and manually reduce to the registered
+ // windows, so we can do special actions when the main
+ // messenger window is opened.
+ //chromeURLs: Object.keys(self.registeredWindows),
+ async onLoadWindow(window) {
+ // Create add-on scope
+ window[self.uniqueRandomID] = {};
+
+ // Special action #1: If this is the main messenger window
+ if (
+ window.location.href ==
+ "chrome://messenger/content/messenger.xul" ||
+ window.location.href ==
+ "chrome://messenger/content/messenger.xhtml"
+ ) {
+ if (self.pathToOptionsPage) {
+ if (self.getThunderbirdVersion().major < 78) {
+ let element_addonPrefs = window.document.getElementById(
+ self.menu_addonPrefs_id
+ );
+ element_addonPrefs.addEventListener(
+ "popupshowing",
+ self
+ );
+ } else {
+ // Setup the options button/menu in the add-on manager, if it is already open.
+ self.setupAddonManager(
+ self.getAddonManagerFromWindow(window),
+ true
+ );
+ // Add a tabmonitor, to be able to setup the options button/menu in the add-on manager.
+ self
+ .getTabMail(window)
+ .registerTabMonitor(self.tabMonitor);
+ window[self.uniqueRandomID].hasTabMonitor = true;
+ }
+ }
+ }
+
+ // Special action #2: If this page contains browser elements
+ let browserElements = window.document.getElementsByTagName(
+ "browser"
+ );
+ if (browserElements.length > 0) {
+ //register a MutationObserver
+ window[
+ self.uniqueRandomID
+ ]._mObserver = new window.MutationObserver(function (
+ mutations
+ ) {
+ mutations.forEach(async function (mutation) {
+ if (
+ mutation.attributeName == "src" &&
+ self.registeredWindows.hasOwnProperty(
+ mutation.target.getAttribute("src")
+ )
+ ) {
+ // When the MutationObserver callsback, the window is still showing "about:black" and it is going
+ // to unload and then load the new page. Any eventListener attached to the window will be removed
+ // so we cannot listen for the load event. We have to poll manually to learn when loading has finished.
+ // On my system it takes 70ms.
+ let loaded = false;
+ for (let i = 0; i < 100 && !loaded; i++) {
+ await self.sleep(100);
+ let targetWindow =
+ mutation.target.contentWindow.wrappedJSObject;
+ if (
+ targetWindow &&
+ targetWindow.location.href ==
+ mutation.target.getAttribute("src") &&
+ targetWindow.document.readyState == "complete"
+ ) {
+ loaded = true;
+ break;
+ }
+ }
+ if (loaded) {
+ let targetWindow =
+ mutation.target.contentWindow.wrappedJSObject;
+ // Create add-on scope
+ targetWindow[self.uniqueRandomID] = {};
+ // Inject with isAddonActivation = false
+ self._loadIntoWindow(targetWindow, false);
+ }
+ }
+ });
+ });
+
+ for (let element of browserElements) {
+ if (
+ self.registeredWindows.hasOwnProperty(
+ element.getAttribute("src")
+ )
+ ) {
+ let targetWindow =
+ element.contentWindow.wrappedJSObject;
+ // Create add-on scope
+ targetWindow[self.uniqueRandomID] = {};
+ // Inject with isAddonActivation = true
+ self._loadIntoWindow(targetWindow, true);
+ } else {
+ // Window/Browser is not yet fully loaded, postpone injection via MutationObserver
+ window[self.uniqueRandomID]._mObserver.observe(
+ element,
+ {
+ attributes: true,
+ childList: false,
+ characterData: false,
+ }
+ );
+ }
+ }
+ }
+
+ // Load JS into window
+ self._loadIntoWindow(
+ window,
+ self.openWindows.includes(window)
+ );
+ },
+
+ onUnloadWindow(window) {
+ // Remove JS from window, window is being closed, addon is not shut down
+ self._unloadFromWindow(window, false);
+ },
+ }
+ );
+ } else {
+ self.error("Failed to start listening, no windows registered");
+ }
+ },
+ },
+ };
+ }
+
+ _loadIntoWindow(window, isAddonActivation) {
+ if (
+ window.hasOwnProperty(this.uniqueRandomID) &&
+ this.registeredWindows.hasOwnProperty(window.location.href)
+ ) {
+ try {
+ let uniqueRandomID = this.uniqueRandomID;
+ let extension = this.extension;
+
+ // Add reference to window to add-on scope
+ window[this.uniqueRandomID].window = window;
+ window[this.uniqueRandomID].document = window.document;
+
+ // Keep track of toolbarpalettes we are injecting into
+ window[this.uniqueRandomID]._toolbarpalettes = {};
+
+ //Create WLDATA object
+ window[this.uniqueRandomID].WL = {};
+ window[this.uniqueRandomID].WL.scopeName = this.uniqueRandomID;
+
+ // Add helper function to inject CSS to WLDATA object
+ window[this.uniqueRandomID].WL.injectCSS = function (cssFile) {
+ let element;
+ let v = parseInt(Services.appinfo.version.split(".").shift());
+
+ // using createElementNS in TB78 delays the insert process and hides any security violation errors
+ if (v > 68) {
+ element = window.document.createElement("link");
+ } else {
+ let ns = window.document.documentElement.lookupNamespaceURI("html");
+ element = window.document.createElementNS(ns, "link");
+ }
+
+ element.setAttribute("wlapi_autoinjected", uniqueRandomID);
+ element.setAttribute("rel", "stylesheet");
+ element.setAttribute("href", cssFile);
+ return window.document.documentElement.appendChild(element);
+ };
+
+ // Add helper function to inject XUL to WLDATA object
+ window[this.uniqueRandomID].WL.injectElements = function (
+ xulString,
+ dtdFiles = [],
+ debug = false
+ ) {
+ let toolbarsToResolve = [];
+
+ function checkElements(stringOfIDs) {
+ let arrayOfIDs = stringOfIDs.split(",").map((e) => e.trim());
+ for (let id of arrayOfIDs) {
+ let element = window.document.getElementById(id);
+ if (element) {
+ return element;
+ }
+ }
+ return null;
+ }
+
+ function localize(entity) {
+ let msg = entity.slice("__MSG_".length, -2);
+ return extension.localeData.localizeMessage(msg);
+ }
+
+ function injectChildren(elements, container) {
+ if (debug) console.log(elements);
+
+ for (let i = 0; i < elements.length; i++) {
+ // take care of persists
+ const uri = window.document.documentURI;
+ for (const persistentNode of elements[i].querySelectorAll(
+ "[persist]"
+ )) {
+ for (const persistentAttribute of persistentNode
+ .getAttribute("persist")
+ .trim()
+ .split(" ")) {
+ if (
+ Services.xulStore.hasValue(
+ uri,
+ persistentNode.id,
+ persistentAttribute
+ )
+ ) {
+ persistentNode.setAttribute(
+ persistentAttribute,
+ Services.xulStore.getValue(
+ uri,
+ persistentNode.id,
+ persistentAttribute
+ )
+ );
+ }
+ }
+ }
+
+ if (
+ elements[i].hasAttribute("insertafter") &&
+ checkElements(elements[i].getAttribute("insertafter"))
+ ) {
+ let insertAfterElement = checkElements(
+ elements[i].getAttribute("insertafter")
+ );
+
+ if (debug)
+ console.log(
+ elements[i].tagName +
+ "#" +
+ elements[i].id +
+ ": insertafter " +
+ insertAfterElement.id
+ );
+ if (
+ debug &&
+ elements[i].id &&
+ window.document.getElementById(elements[i].id)
+ ) {
+ console.error(
+ "The id <" +
+ elements[i].id +
+ "> of the injected element already exists in the document!"
+ );
+ }
+ elements[i].setAttribute("wlapi_autoinjected", uniqueRandomID);
+ insertAfterElement.parentNode.insertBefore(
+ elements[i],
+ insertAfterElement.nextSibling
+ );
+ } else if (
+ elements[i].hasAttribute("insertbefore") &&
+ checkElements(elements[i].getAttribute("insertbefore"))
+ ) {
+ let insertBeforeElement = checkElements(
+ elements[i].getAttribute("insertbefore")
+ );
+
+ if (debug)
+ console.log(
+ elements[i].tagName +
+ "#" +
+ elements[i].id +
+ ": insertbefore " +
+ insertBeforeElement.id
+ );
+ if (
+ debug &&
+ elements[i].id &&
+ window.document.getElementById(elements[i].id)
+ ) {
+ console.error(
+ "The id <" +
+ elements[i].id +
+ "> of the injected element already exists in the document!"
+ );
+ }
+ elements[i].setAttribute("wlapi_autoinjected", uniqueRandomID);
+ insertBeforeElement.parentNode.insertBefore(
+ elements[i],
+ insertBeforeElement
+ );
+ } else if (
+ elements[i].id &&
+ window.document.getElementById(elements[i].id)
+ ) {
+ // existing container match, dive into recursivly
+ if (debug)
+ console.log(
+ elements[i].tagName +
+ "#" +
+ elements[i].id +
+ " is an existing container, injecting into " +
+ elements[i].id
+ );
+ injectChildren(
+ Array.from(elements[i].children),
+ window.document.getElementById(elements[i].id)
+ );
+ } else if (elements[i].localName === "toolbarpalette") {
+ // These vanish from the document but still exist via the palette property
+ if (debug) console.log(elements[i].id + " is a toolbarpalette");
+ let boxes = [
+ ...window.document.getElementsByTagName("toolbox"),
+ ];
+ let box = boxes.find(
+ (box) => box.palette && box.palette.id === elements[i].id
+ );
+ let palette = box ? box.palette : null;
+
+ if (!palette) {
+ if (debug)
+ console.log(
+ `The palette for ${elements[i].id} could not be found, deferring to later`
+ );
+ continue;
+ }
+
+ if (debug)
+ console.log(`The toolbox for ${elements[i].id} is ${box.id}`);
+
+ toolbarsToResolve.push(...box.querySelectorAll("toolbar"));
+ toolbarsToResolve.push(
+ ...window.document.querySelectorAll(
+ `toolbar[toolboxid="${box.id}"]`
+ )
+ );
+ for (let child of elements[i].children) {
+ child.setAttribute("wlapi_autoinjected", uniqueRandomID);
+ }
+ window[uniqueRandomID]._toolbarpalettes[palette.id] = palette;
+ injectChildren(Array.from(elements[i].children), palette);
+ } else {
+ // append element to the current container
+ if (debug)
+ console.log(
+ elements[i].tagName +
+ "#" +
+ elements[i].id +
+ ": append to " +
+ container.id
+ );
+ elements[i].setAttribute("wlapi_autoinjected", uniqueRandomID);
+ container.appendChild(elements[i]);
+ }
+ }
+ }
+
+ if (debug) console.log("Injecting into root document:");
+ let localizedXulString = xulString.replace(
+ /__MSG_(.*?)__/g,
+ localize
+ );
+ injectChildren(
+ Array.from(
+ window.MozXULElement.parseXULToFragment(
+ localizedXulString,
+ dtdFiles
+ ).children
+ ),
+ window.document.documentElement
+ );
+
+ for (let bar of toolbarsToResolve) {
+ let currentset = Services.xulStore.getValue(
+ window.location,
+ bar.id,
+ "currentset"
+ );
+ if (currentset) {
+ bar.currentSet = currentset;
+ } else if (bar.getAttribute("defaultset")) {
+ bar.currentSet = bar.getAttribute("defaultset");
+ }
+ }
+ };
+
+ // Add extension object to WLDATA object
+ window[this.uniqueRandomID].WL.extension = this.extension;
+ // Add messenger object to WLDATA object
+ window[this.uniqueRandomID].WL.messenger = this.getMessenger(
+ this.context
+ );
+ // Load script into add-on scope
+ Services.scriptloader.loadSubScript(
+ this.registeredWindows[window.location.href],
+ window[this.uniqueRandomID],
+ "UTF-8"
+ );
+ window[this.uniqueRandomID].onLoad(isAddonActivation);
+ } catch (e) {
+ Components.utils.reportError(e);
+ }
+ }
+ }
+
+ _unloadFromWindow(window, isAddonDeactivation) {
+ // unload any contained browser elements
+ if (
+ window.hasOwnProperty(this.uniqueRandomID) &&
+ window[this.uniqueRandomID].hasOwnProperty("_mObserver")
+ ) {
+ window[this.uniqueRandomID]._mObserver.disconnect();
+ let browserElements = window.document.getElementsByTagName("browser");
+ for (let element of browserElements) {
+ if (element.contentWindow) {
+ this._unloadFromWindow(
+ element.contentWindow.wrappedJSObject,
+ isAddonDeactivation
+ );
+ }
+ }
+ }
+
+ if (
+ window.hasOwnProperty(this.uniqueRandomID) &&
+ this.registeredWindows.hasOwnProperty(window.location.href)
+ ) {
+ // Remove this window from the list of open windows
+ this.openWindows = this.openWindows.filter((e) => e != window);
+
+ if (window[this.uniqueRandomID].onUnload) {
+ try {
+ // Call onUnload()
+ window[this.uniqueRandomID].onUnload(isAddonDeactivation);
+ } catch (e) {
+ Components.utils.reportError(e);
+ }
+ }
+
+ // Remove all auto injected objects
+ let elements = Array.from(
+ window.document.querySelectorAll(
+ '[wlapi_autoinjected="' + this.uniqueRandomID + '"]'
+ )
+ );
+ for (let element of elements) {
+ element.remove();
+ }
+
+ // Remove all autoinjected toolbarpalette items
+ for (const palette of Object.values(
+ window[this.uniqueRandomID]._toolbarpalettes
+ )) {
+ let elements = Array.from(
+ palette.querySelectorAll(
+ '[wlapi_autoinjected="' + this.uniqueRandomID + '"]'
+ )
+ );
+ for (let element of elements) {
+ element.remove();
+ }
+ }
+ }
+
+ // Remove add-on scope, if it exists
+ if (window.hasOwnProperty(this.uniqueRandomID)) {
+ delete window[this.uniqueRandomID];
+ }
+ }
+
+ onShutdown(isAppShutdown) {
+ if (isAppShutdown) {
+ return; // the application gets unloaded anyway
+ }
+
+ // Unload from all still open windows
+ let urls = Object.keys(this.registeredWindows);
+ if (urls.length > 0) {
+ for (let window of Services.wm.getEnumerator(null)) {
+ //remove our entry in the add-on options menu
+ if (
+ this.pathToOptionsPage &&
+ (window.location.href == "chrome://messenger/content/messenger.xul" ||
+ window.location.href ==
+ "chrome://messenger/content/messenger.xhtml")
+ ) {
+ if (this.getThunderbirdVersion().major < 78) {
+ let element_addonPrefs = window.document.getElementById(
+ this.menu_addonPrefs_id
+ );
+ element_addonPrefs.removeEventListener("popupshowing", this);
+ // Remove our entry.
+ let entry = window.document.getElementById(
+ this.menu_addonPrefs_id + "_" + this.uniqueRandomID
+ );
+ if (entry) entry.remove();
+ // Do we have to unhide the noPrefsElement?
+ if (element_addonPrefs.children.length == 1) {
+ let noPrefsElem = element_addonPrefs.querySelector(
+ '[disabled="true"]'
+ );
+ noPrefsElem.style.display = "inline";
+ }
+ } else {
+ // Remove event listener for addon manager view changes
+ let managerWindow = this.getAddonManagerFromWindow(window);
+ if (
+ managerWindow &&
+ managerWindow[this.uniqueRandomID] &&
+ managerWindow[this.uniqueRandomID].hasAddonManagerEventListeners
+ ) {
+ managerWindow.document.removeEventListener("ViewChanged", this);
+ managerWindow.document.removeEventListener("view-loaded", this);
+ managerWindow.document.removeEventListener("update", this);
+
+ let cards = this.getCards(managerWindow);
+ if (this.getThunderbirdVersion().major < 88) {
+ // Remove options menu in 78-87
+ for (let card of cards) {
+ let addonOptionsLegacyEntry = card.querySelector(
+ ".extension-options-legacy"
+ );
+ if (addonOptionsLegacyEntry) addonOptionsLegacyEntry.remove();
+ }
+ } else {
+ // Remove options button in 88
+ for (let card of cards) {
+ if (card.addon.id == this.extension.id) {
+ let addonOptionsButton = card.querySelector(
+ ".windowlistener-options-button"
+ );
+ if (addonOptionsButton) addonOptionsButton.remove();
+ break;
+ }
+ }
+ }
+ }
+
+ // Remove tabmonitor
+ if (window[this.uniqueRandomID].hasTabMonitor) {
+ this.getTabMail(window).unregisterTabMonitor(this.tabMonitor);
+ window[this.uniqueRandomID].hasTabMonitor = false;
+ }
+ }
+ }
+
+ // if it is app shutdown, it is not just an add-on deactivation
+ this._unloadFromWindow(window, !isAppShutdown);
+ }
+ // Stop listening for new windows.
+ ExtensionSupport.unregisterWindowListener(
+ "injectListener_" + this.uniqueRandomID
+ );
+ }
+
+ // Load registered shutdown script
+ let shutdownJS = {};
+ shutdownJS.extension = this.extension;
+ try {
+ if (this.pathToShutdownScript)
+ Services.scriptloader.loadSubScript(
+ this.pathToShutdownScript,
+ shutdownJS,
+ "UTF-8"
+ );
+ } catch (e) {
+ Components.utils.reportError(e);
+ }
+
+ // Extract all registered chrome content urls
+ let chromeUrls = [];
+ if (this.chromeData) {
+ for (let chromeEntry of this.chromeData) {
+ if (chromeEntry[0].toLowerCase().trim() == "content") {
+ chromeUrls.push("chrome://" + chromeEntry[1] + "/");
+ }
+ }
+ }
+
+ // Unload JSMs of this add-on
+ const rootURI = this.extension.rootURI.spec;
+ for (let module of Cu.loadedModules) {
+ if (
+ module.startsWith(rootURI) ||
+ (module.startsWith("chrome://") &&
+ chromeUrls.find((s) => module.startsWith(s)))
+ ) {
+ this.log("Unloading: " + module);
+ Cu.unload(module);
+ }
+ }
+
+ // Flush all caches
+ Services.obs.notifyObservers(null, "startupcache-invalidate");
+ this.registeredWindows = {};
+
+ if (this.resourceData) {
+ const resProto = Cc[
+ "@mozilla.org/network/protocol;1?name=resource"
+ ].getService(Ci.nsISubstitutingProtocolHandler);
+ for (let res of this.resourceData) {
+ // [ "resource", "shortname" , "path" ]
+ resProto.setSubstitution(res[1], null);
+ }
+ }
+
+ if (this.chromeHandle) {
+ this.chromeHandle.destruct();
+ this.chromeHandle = null;
+ }
+ }
+};
diff --git a/xul-ext/api/windowListener/schema.json b/xul-ext/api/windowListener/schema.json
new file mode 100644
index 0000000..03963a5
--- /dev/null
+++ b/xul-ext/api/windowListener/schema.json
@@ -0,0 +1,108 @@
+[
+ {
+ "namespace": "WindowListener",
+ "functions": [
+ {
+ "name": "registerDefaultPrefs",
+ "type": "function",
+ "parameters": [
+ {
+ "name": "aPath",
+ "type": "string",
+ "description": "Relative path to the default file."
+ }
+ ]
+ },
+ {
+ "name": "registerOptionsPage",
+ "type": "function",
+ "parameters": [
+ {
+ "name": "aPath",
+ "type": "string",
+ "description": "Path to the options page, which should be made accessible in the (legacy) Add-On Options menu."
+ }
+ ]
+ },
+ {
+ "name": "registerChromeUrl",
+ "type": "function",
+ "description": "Register folders which should be available as chrome:// urls (as defined in the legacy chrome.manifest)",
+ "parameters": [
+ {
+ "name": "data",
+ "type": "array",
+ "items": {
+ "type": "array",
+ "items": {
+ "type": "string"
+ }
+ },
+ "description": "Array of manifest url definitions (content, locale, resource)"
+ }
+ ]
+ },
+ {
+ "name": "waitForMasterPassword",
+ "type": "function",
+ "async": true,
+ "parameters": []
+ },
+ {
+ "name": "openOptionsDialog",
+ "type": "function",
+ "parameters": [
+ {
+ "name": "windowId",
+ "type": "integer",
+ "description": "Id of the window the dialog should be opened from."
+ }
+ ]
+ },
+ {
+ "name": "startListening",
+ "type": "function",
+ "async": true,
+ "parameters": []
+ },
+ {
+ "name": "registerWindow",
+ "type": "function",
+ "parameters": [
+ {
+ "name": "windowHref",
+ "type": "string",
+ "description": "Url of the window, which should be listen for."
+ },
+ {
+ "name": "jsFile",
+ "type": "string",
+ "description": "Path to the JavaScript file, which should be loaded into the window."
+ }
+ ]
+ },
+ {
+ "name": "registerStartupScript",
+ "type": "function",
+ "parameters": [
+ {
+ "name": "aPath",
+ "type": "string",
+ "description": "Path to a JavaScript file, which should be executed on add-on startup. The script will be executed after the main application window has been sucessfully loaded."
+ }
+ ]
+ },
+ {
+ "name": "registerShutdownScript",
+ "type": "function",
+ "parameters": [
+ {
+ "name": "aPath",
+ "type": "string",
+ "description": "Path to a JavaScript file, which should be executed on add-on shutdown."
+ }
+ ]
+ }
+ ]
+ }
+]
diff --git a/xul-ext/background.js b/xul-ext/background.js
new file mode 100644
index 0000000..3cd5673
--- /dev/null
+++ b/xul-ext/background.js
@@ -0,0 +1,63 @@
+/**
+ * commented out locales are not complete enough on Transifex
+ */
+const locales = [
+ 'cs',
+ //'da',
+ 'de',
+ //'el',
+ 'en-US',
+ 'es-AR',
+ 'fi',
+ 'fr',
+ //'hi',
+ 'hu',
+ //'it',
+ //'ja',
+ //'ko-KR',
+ 'nl',
+ //'pl',
+ //'pt-BR',
+ //'pt-PT',
+ //'ro',
+ 'ru',
+ 'sl',
+ //'sv-SE',
+ 'tr',
+ 'ul',
+ 'zh-CN'
+];
+(async () => {
+ // https://github.com/thundernest/addon-developer-support/wiki/WindowListener-API:-Getting-Started#windowlistenerregisterdefaultprefs
+ messenger.WindowListener.registerDefaultPrefs("defaults/preferences/prefs.js")
+ /**
+ * https://github.com/thundernest/addon-developer-support/wiki/WindowListener-API:-Getting-Started#windowlistenerregisterchromeurl
+ * Register the content, resource and locale entries from your legacy chrome.manifest via a call to registerChromeUrl().
+ */
+ messenger.WindowListener.registerChromeUrl([
+ ["content", "keefox", "chrome/content/"],
+ ["resource", "keefox", /*"classic/1.0", */ "chrome/skin/"],
+ ["resource", "kfmod", "modules/"],
+ ...locales.map((localeId) =>
+ ["locale", "keefox", localeId, `chrome/locale/${localeId}/`])
+ ]);
+ //xul-ext/chrome/skin
+ // https://github.com/thundernest/addon-developer-support/wiki/WindowListener-API:-Getting-Started#windowlistenerregisteroptionspage
+ messenger.WindowListener.registerOptionsPage("chrome://keefox/content/options.xhtml");
+
+ //
+ messenger.WindowListener.registerWindow(
+ "chrome://messenger/content/messenger.xhtml",
+ "chrome://keefox/content/scripts/messengerPanel.js");
+
+ messenger.WindowListener.registerWindow(
+ "chrome://messenger/content/commonDialog.xhtml",
+ "chrome://keefox/content/scripts/KFcommonDialog.js");
+
+ /*
+ messenger.WindowListener.registerWindow(
+ "chrome://global/content/win.xul",
+ "chrome://keefox/content/scripts/install.js");*/
+
+ await messenger.WindowListener.startListening();
+})()
\ No newline at end of file
diff --git a/xul-ext/chrome.manifest b/xul-ext/chrome.manifest
index 9a3b2fd..c6bd3f3 100644
--- a/xul-ext/chrome.manifest
+++ b/xul-ext/chrome.manifest
@@ -11,7 +11,7 @@ overlay chrome://messenger/content/messenger.xul chrome://keefox/content/panel.x
overlay chrome://gdata-provider/content/browserRequest.xul chrome://keefox/content/gdata-provider.xul
overlay chrome://messenger/content/browserRequest.xul chrome://keefox/content/gdata-provider.xul
-skin keefox classic/1.0 chrome/skin/
+resource keefox classic/1.0 chrome/skin/
resource kfmod modules/
# locales
diff --git a/xul-ext/chrome/content/KFcommonDialog.js b/xul-ext/chrome/content/KFcommonDialog.js
index 36aab27..59b16eb 100644
--- a/xul-ext/chrome/content/KFcommonDialog.js
+++ b/xul-ext/chrome/content/KFcommonDialog.js
@@ -39,6 +39,7 @@ if (!Cu)
const { keefox_org } = ChromeUtils.import("resource://kfmod/KF.js");
const { keeFoxLoginInfo } = ChromeUtils.import("resource://kfmod/kfDataModel.js");
+// var { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm");
var keeFoxDialogManager = {
scriptLoader : Components.classes["@mozilla.org/moz/jssubscript-loader;1"]
@@ -47,9 +48,7 @@ var keeFoxDialogManager = {
__promptBundle : null, // String bundle for L10N
get _promptBundle() {
if (!this.__promptBundle) {
- var bunService = Components.classes["@mozilla.org/intl/stringbundle;1"].
- getService(Components.interfaces.nsIStringBundleService);
- this.__promptBundle = bunService.createBundle(
+ this.__promptBundle = Services.strings.createBundle(
"chrome://global/locale/prompts.properties");
if (!this.__promptBundle)
throw "Prompt string bundle not present!";
@@ -60,9 +59,7 @@ var keeFoxDialogManager = {
__cdBundle : null, // String bundle for L10N
get _cdBundle() {
if (!this.__cdBundle) {
- var bunService = Components.classes["@mozilla.org/intl/stringbundle;1"].
- getService(Components.interfaces.nsIStringBundleService);
- this.__cdBundle = bunService.createBundle(
+ this.__cdBundle = Services.strings.createBundle(
"chrome://global/locale/commonDialogs.properties");
if (!this.__cdBundle)
throw "Common Dialogs string bundle not present!";
@@ -73,9 +70,7 @@ var keeFoxDialogManager = {
__messengerBundle : null, // string bundle for thunderbird l10n
get _messengerBundle() {
if (!this.__messengerBundle) {
- var bunService = Components.classes["@mozilla.org/intl/stringbundle;1"].
- getService(Components.interfaces.nsIStringBundleService);
- this.__messengerBundle = bunService.createBundle(
+ this.__messengerBundle = Services.strings.createBundle(
"chrome://messenger/locale/messenger.properties");
if (!this.__messengerBundle)
throw "Messenger string bundle not present!";
@@ -86,9 +81,7 @@ var keeFoxDialogManager = {
__localMsgsBundle : null, // string bundle for thunderbird l10n
get _localMsgsBundle() {
if (!this.__localMsgsBundle) {
- var bunService = Components.classes["@mozilla.org/intl/stringbundle;1"].
- getService(Components.interfaces.nsIStringBundleService);
- this.__localMsgsBundle = bunService.createBundle(
+ this.__localMsgsBundle = Services.strings.createBundle(
"chrome://messenger/locale/localMsgs.properties");
if (!this.__localMsgsBundle)
throw "localMsgs string bundle not present!";
@@ -99,9 +92,7 @@ var keeFoxDialogManager = {
__imapMsgsBundle : null, // string bundle for thunderbird l10n
get _imapMsgsBundle() {
if (!this.__imapMsgsBundle) {
- var bunService = Components.classes["@mozilla.org/intl/stringbundle;1"].
- getService(Components.interfaces.nsIStringBundleService);
- this.__imapMsgsBundle = bunService.createBundle(
+ this.__imapMsgsBundle = Services.strings.createBundle(
"chrome://messenger/locale/imapMsgs.properties");
if (!this.__imapMsgsBundle)
throw "imapMsgs string bundle not present!";
@@ -128,9 +119,7 @@ var keeFoxDialogManager = {
__newsBundle : null, // string bundle for thunderbird l10n
get _newsBundle() {
if (!this.__newsBundle) {
- var bunService = Components.classes["@mozilla.org/intl/stringbundle;1"].
- getService(Components.interfaces.nsIStringBundleService);
- this.__newsBundle = bunService.createBundle(
+ this.__newsBundle = Services.strings.createBundle(
"chrome://messenger/locale/news.properties");
if (!this.__newsBundle)
throw "news string bundle not present!";
@@ -141,9 +130,7 @@ var keeFoxDialogManager = {
__composeBundle : null, // string bundle for thunderbird l10n
get _composeBundle() {
if (!this.__composeBundle) {
- var bunService = Components.classes["@mozilla.org/intl/stringbundle;1"].
- getService(Components.interfaces.nsIStringBundleService);
- this.__composeBundle = bunService.createBundle(
+ this.__composeBundle = Services.strings.createBundle(
"chrome://messenger/locale/messengercompose/composeMsgs.properties");
if (!this.__composeBundle)
throw "Compose Message string bundle not present!";
@@ -155,7 +142,7 @@ var keeFoxDialogManager = {
.getService(Components.interfaces.nsIXULAppInfo),
dialogInit : function(e) {
- window.removeEventListener("load", keeFoxDialogManager.dialogInit);
+ //window.removeEventListener("load", keeFoxDialogManager.dialogInit);
try
{
document.addEventListener("dialogaccept", event => {
@@ -283,112 +270,139 @@ var keeFoxDialogManager = {
{
let regexChars = /[\[\{\(\)\*\+\?\.\\\^\$\|]/g;
protocols[aDialogType] = aDialogType.split("-")[0];
- titles[aDialogType] = aStringBundle.GetStringFromName(aTitlePropertyName);
- prompts[aDialogType] = aStringBundle.GetStringFromName(aPromptPropertyName);
- prompts[aDialogType] = prompts[aDialogType].replace(regexChars, "\\$&");
- aHostPlaceholder = aHostPlaceholder.replace(regexChars, "\\$&");
- // use null as a flag to indicate that there was only one
- // placeholder and hostIsFirst and secondIsUserName are not applicable
- hostIsFirst[aDialogType] = null;
- if (aUserPlaceholder != null)
- {
- aUserPlaceholder = aUserPlaceholder.replace(regexChars, "\\$&");
- hostIsFirst[aDialogType] = prompts[aDialogType].indexOf(aHostPlaceholder) <
- prompts[aDialogType].indexOf(aUserPlaceholder);
- secondIsUserName[aDialogType] = true;
- }
- if (aRealmPlaceholder != null)
- {
- aRealmPlaceholder = aRealmPlaceholder.replace(regexChars, "\\$&");
- hostIsFirst[aDialogType] = prompts[aDialogType].indexOf(aHostPlaceholder) <
- prompts[aDialogType].indexOf(aRealmPlaceholder);
- secondIsUserName[aDialogType] = false;
- }
- prompts[aDialogType] = prompts[aDialogType].replace(aHostPlaceholder, "([^\\s]+)");
- if (aUserPlaceholder != null)
- {
- prompts[aDialogType] = prompts[aDialogType].replace(aUserPlaceholder, "([^\\s]+)");
+ // dirty fix
+ // use a big try/catch here to avoid errors on null string resources
+ try {
+ titles[aDialogType] = aStringBundle.GetStringFromName(aTitlePropertyName);
+ prompts[aDialogType] = aStringBundle.GetStringFromName(aPromptPropertyName);
+ titles[aDialogType] = titles[aDialogType].replace(regexChars, "\\$&");
+ prompts[aDialogType] = prompts[aDialogType].replace(regexChars, "\\$&");
+ aHostPlaceholder = aHostPlaceholder.replace(regexChars, "\\$&");
+ // use null as a flag to indicate that there was only one
+ // placeholder and hostIsFirst and secondIsUserName are not applicable
+ hostIsFirst[aDialogType] = null;
+ if (aUserPlaceholder != null)
+ {
+ aUserPlaceholder = aUserPlaceholder.replace(regexChars, "\\$&");
+ hostIsFirst[aDialogType] = prompts[aDialogType].indexOf(aHostPlaceholder) <
+ prompts[aDialogType].indexOf(aUserPlaceholder);
+ secondIsUserName[aDialogType] = true;
+ }
+ if (aRealmPlaceholder != null)
+ {
+ aRealmPlaceholder = aRealmPlaceholder.replace(regexChars, "\\$&");
+ hostIsFirst[aDialogType] = prompts[aDialogType].indexOf(aHostPlaceholder) <
+ prompts[aDialogType].indexOf(aRealmPlaceholder);
+ secondIsUserName[aDialogType] = false;
+ }
+ titles[aDialogType] = titles[aDialogType].replace(aHostPlaceholder, "([^\\s]+)");
+ prompts[aDialogType] = prompts[aDialogType].replace(aHostPlaceholder, "([^\\s]+)");
+ if (aUserPlaceholder != null)
+ {
+ titles[aDialogType] = titles[aDialogType].replace(aUserPlaceholder, "([^\\s]+)");
+ prompts[aDialogType] = prompts[aDialogType].replace(aUserPlaceholder, "([^\\s]+)");
+ }
+ if (aRealmPlaceholder != null)
+ {
+ prompts[aDialogType] = prompts[aDialogType].replace(aRealmPlaceholder, "([^\\s]+)");
+ }
+ extractUserFromHost[aDialogType] = aExtractUserFromHost;
}
- if (aRealmPlaceholder != null)
- {
- prompts[aDialogType] = prompts[aDialogType].replace(aRealmPlaceholder, "([^\\s]+)");
+ catch (resourceError) {
+ keefox_org._KFLog.error(`LoadDialogData failed for ${aDialogType},${aTitlePropertyName},${aPromptPropertyName}`, resourceError);
}
- extractUserFromHost[aDialogType] = aExtractUserFromHost;
}
- }
-
- LoadDialogData(this._composeBundle, "smtp", "smtpEnterPasswordPromptTitle",
+ };
+ LoadDialogData(this._composeBundle, "smtp", "smtpEnterPasswordPrompt",
"smtpEnterPasswordPromptWithUsername", "%1$S", null, "%2$S");
+
+ // SMTP alt
+ // title "Password Required for Outgoing (SMTP) Server %1$S"
+ // prompt "Enter your password for %2$S on %1$S:"
+ // see omni.ja/chrome/en-US/locale/en-US/messenger/messengercompose/composeMsgs.properties
+ LoadDialogData(this._composeBundle,
+ "smtp2",
+ "smtpEnterPasswordPromptTitleWithHostname",
+ "smtpEnterPasswordPromptWithUsername",
+ "%1$S",
+ null,
+ "%2$S");
+
var imapEnterPasswordPromptTitle = this._imapBundleUsesStrings
- ? "imapEnterPasswordPromptTitle" : "5051";
+ ? "imapEnterPasswordPromptTitleWithUsername" : "5051";
var imapEnterPasswordPrompt = this._imapBundleUsesStrings
- ? "imapEnterPasswordPrompt" : "5047";
- var isPreTB40ImapStrings = true;
- try {
+ ? "imapEnterServerPasswordPrompt" : "5047";
+ var isPreTB40ImapStrings = false;
+ /*try {
// Take a peek at imapEnterPasswordPrompt
this._imapMsgsBundle.GetStringFromName(imapEnterPasswordPrompt);
} catch (e) {
// The string identifier changed again in TB 40
imapEnterPasswordPrompt = "imapEnterServerPasswordPrompt";
isPreTB40ImapStrings = false;
- }
+ }*/
// The prompt changed from using one parameter to using two in TB40
LoadDialogData(this._imapMsgsBundle, "imap",
imapEnterPasswordPromptTitle, imapEnterPasswordPrompt,
isPreTB40ImapStrings ? "%S" : "%2$S", null,
isPreTB40ImapStrings ? null : "%1$S", isPreTB40ImapStrings);
- LoadDialogData(this._localMsgsBundle, "pop3", "pop3EnterPasswordPromptTitle",
- "pop3EnterPasswordPrompt", "%2$S", null, "%1$S");
+
+ LoadDialogData(this._localMsgsBundle, "pop3", "pop3EnterPasswordPromptTitleWithUsername",
+ "pop3EnterPasswordPromptTitleWithUsername", "%2$S", null, "%1$S");
+
LoadDialogData(this._newsBundle, "nntp-1", "enterUserPassTitle",
"enterUserPassServer", "%S");
+
LoadDialogData(this._newsBundle, "nntp-2", "enterUserPassTitle",
"enterUserPassGroup", "%2$S", "%1$S");
+
LoadDialogData(this._messengerBundle, "mail", "passwordTitle",
"passwordPrompt", "%2$S", null, "%1$S");
- for (let type in titles)
- {
- if (Dialog.args.title == titles[type])
- {
- // some types have the same title, so we have more checking to do
- let regEx = new RegExp(prompts[type]);
- let matches = Dialog.args.text.match(regEx);
- if (!matches)
- {
- continue;
- }
- if (hostIsFirst[type] === null) {
- // there is only one parameter, so nothing is first
- if (matches.length == 2) {
- if (extractUserFromHost[type])
- {
- // user and host are separated by @ character
- let lastAtSym = matches[1].lastIndexOf("@");
- username = matches[1].substring(0, lastAtSym);
- host = protocols[type] + "://" +
- matches[1].substring(lastAtSym + 1, matches[1].length);
- } else {
- host = protocols[type] + "://" + matches[1];
- }
- break;
+ const matchings = Object.keys(titles).filter((type) => {
+ // refactor the title identification with a regEx because some have placeholders like $S
+ let reTitle = new RegExp(titles[type]);
+ return reTitle.test(Dialog.args.title);
+ }).map((type) => {
+ let regEx = new RegExp(prompts[type]);
+ let matches = Dialog.args.text.match(regEx);
+ return {
+ type,
+ matches
+ };
+ });
+
+ matchings.forEach((m) => {
+ const type = m.type;
+ const matches = m.matches;
+ if (hostIsFirst[type] === null) {
+ // there is only one parameter, so nothing is first
+ if (matches.length == 2) {
+ if (extractUserFromHost[type])
+ {
+ // user and host are separated by @ character
+ let lastAtSym = matches[1].lastIndexOf("@");
+ username = matches[1].substring(0, lastAtSym);
+ host = protocols[type] + "://" +
+ matches[1].substring(lastAtSym + 1, matches[1].length);
+ } else {
+ host = protocols[type] + "://" + matches[1];
}
- } else
- {
- if (matches.length == 3) {
- if (hostIsFirst[type]) {
- host = protocols[type] + "://" + matches[1];
- username = matches[2];
- } else {
- host = protocols[type] + "://" + matches[2];
- username = matches[1];
- }
- break;
+ }
+ } else {
+ if (matches.length == 3) {
+ if (hostIsFirst[type]) {
+ host = protocols[type] + "://" + matches[1];
+ username = matches[2];
+ } else {
+ host = protocols[type] + "://" + matches[2];
+ username = matches[1];
}
}
}
- }
+ });
} // end if Thunderbird
-
+
if (host.length < 1)
{
// e.g. en-US:
@@ -560,24 +574,23 @@ var keeFoxDialogManager = {
this.mustAutoSubmit = mustAutoSubmit;
/* add ui elements to dialog */
-
- var row = document.createElement("row");
+
+ var row = document.createElement("div");
row.setAttribute("id","keefox-autoauth-row");
- row.setAttribute("flex", "1");
+ row.setAttribute("class","dialogRow");
// spacer to take up first column in layout
- var spacer = document.createElement("spacer");
- spacer.setAttribute("flex", "1");
+ var spacer = document.createElement("div");
row.appendChild(spacer);
// this box displays labels and also the list of entries when fetched
- var box = document.createElement("hbox");
+ var box = document.createXULElement("hbox");
box.setAttribute("id","keefox-autoauth-box");
box.setAttribute("align", "center");
box.setAttribute("flex", "1");
box.setAttribute("pack", "start");
- var loadingPasswords = document.createElement("description");
+ var loadingPasswords = document.createXULElement("description");
loadingPasswords.setAttribute("id","keefox-autoauth-description");
loadingPasswords.setAttribute("align", "start");
loadingPasswords.setAttribute("flex", "1");
@@ -585,7 +598,7 @@ var keeFoxDialogManager = {
row.appendChild(box);
// button to lauch KeePass
- var launchKeePassButton = document.createElement("button");
+ var launchKeePassButton = document.createXULElement("button");
launchKeePassButton.setAttribute("id", "keefox-launch-kp-button");
launchKeePassButton.setAttribute("label", keefox_org.locale.$STR("launchKeePass.label"));
launchKeePassButton.addEventListener("command", function (event) { keefox_org.launchKeePass(''); }, false);
@@ -597,6 +610,10 @@ var keeFoxDialogManager = {
}
},
+ /**
+ * This is where we send the identification request to keefox RPC
+ * Assuming that the validation token has been stored
+ */
updateDialog : function()
{
// check to make sure prepareFill was called
@@ -761,9 +778,9 @@ var keeFoxDialogManager = {
if (showList) {
var box = dialogFindLoginStorage.document.getElementById("keefox-autoauth-box");
- var list = dialogFindLoginStorage.document.createElement("menulist");
+ var list = dialogFindLoginStorage.document.createXULElement("menulist");
list.setAttribute("id","autoauth-list");
- var popup = dialogFindLoginStorage.document.createElement("menupopup");
+ var popup = dialogFindLoginStorage.document.createXULElement("menupopup");
var done = false;
for (var i = 0; i < matchedLogins.length; i++) {
@@ -779,7 +796,7 @@ var keeFoxDialogManager = {
});
for (var i = 0; i < matchedLogins.length; i++){
- var item = dialogFindLoginStorage.document.createElement("menuitem");
+ var item = dialogFindLoginStorage.document.createXULElement("menuitem");
item.setAttribute("label", keefox_org.locale.$STRF("matchedLogin.label",
[matchedLogins[i].username, matchedLogins[i].host]));
item.setAttribute("tooltiptext", keefox_org.locale.$STRF("matchedLogin.tip",
@@ -928,4 +945,5 @@ KPRPCConnectionObserver.prototype = {
keeFoxDialogManager.Logger = keefox_org._KFLog;
keeFoxDialogManager.scriptLoader.loadSubScript(
"chrome://keefox/content/shared/uriUtils.js", keeFoxDialogManager);
-window.addEventListener("load", keeFoxDialogManager.dialogInit, false);
+//window.addEventListener("load", keeFoxDialogManager.dialogInit, false);
+
diff --git a/xul-ext/chrome/content/UninstallHelper.js b/xul-ext/chrome/content/UninstallHelper.js
index c725cf0..9a0a496 100644
--- a/xul-ext/chrome/content/UninstallHelper.js
+++ b/xul-ext/chrome/content/UninstallHelper.js
@@ -214,14 +214,12 @@ keefox_win.UninstallHelper.prototype =
extraLabel.setAttribute("control", "keefox-uninstall-helper-feedback-extra");
container.appendChild(extraLabel);
- // FIXME: textbox multiline="true" is removed in TB68
- // https://developer.thunderbird.net/add-ons/updates/tb68#less-than-textbox-multiline-true-greater-than
- let extra = this.doc.createElement('textbox');
+ let extra = this.doc.createElementNS("http://www.w3.org/1999/xhtml",'textarea');
extra.setAttribute("id","keefox-uninstall-helper-feedback-extra");
- extra.setAttribute("multiline","true");
extra.addEventListener("keypress", pnCountNoteChars, false);
container.appendChild(extra);
+
if (!varients.smallScreen && varients.whyLocation == 2)
attachWhy.call(this);
diff --git a/xul-ext/chrome/content/context.js b/xul-ext/chrome/content/context.js
index 04d5b0a..72026b9 100644
--- a/xul-ext/chrome/content/context.js
+++ b/xul-ext/chrome/content/context.js
@@ -135,7 +135,7 @@ keefox_win.context = {
var tempButton = i == 0 ?
container :
- this._currentWindow.document.createElement("menuitem");
+ this._currentWindow.document.createXULElement("menuitem");
tempButton.setAttribute("label", keefox_org.locale.$STRF("matchedLogin.label"
, [usernameDisplayValue, login.title]));
tempButton.setAttribute("class", "menuitem-iconic");
diff --git a/xul-ext/chrome/content/context.xul b/xul-ext/chrome/content/context.xhtml
similarity index 97%
rename from xul-ext/chrome/content/context.xul
rename to xul-ext/chrome/content/context.xhtml
index f505848..a86e876 100644
--- a/xul-ext/chrome/content/context.xul
+++ b/xul-ext/chrome/content/context.xhtml
@@ -56,7 +56,7 @@
diff --git a/xul-ext/chrome/content/famsOptions.js b/xul-ext/chrome/content/famsOptions.js
index 6f84b13..9e9b00c 100644
--- a/xul-ext/chrome/content/famsOptions.js
+++ b/xul-ext/chrome/content/famsOptions.js
@@ -62,6 +62,10 @@ function onLoad()
event.preventDefault();
}
});
+ document.addEventListener("dialogcancel", event => {
+ event.preventDefault(); // Prevent the dialog closing.
+ return;
+ });
go();
}
@@ -91,7 +95,7 @@ function go() {
function renderAllMessageGroups()
{
var msgGroupContainer = window.document.getElementById("msgGroupContainer");
- var famsDescription = window.document.createElement("description");
+ var famsDescription = window.document.createXULElement("description");
famsDescription.textContent = FAMS.locale.internationaliseString(config.description);
msgGroupContainer.appendChild(famsDescription);
@@ -108,11 +112,11 @@ function renderAllMessageGroups()
function renderDownloadOptions()
{
var msgGroupContainer = window.document.getElementById("msgGroupContainer");
- var downloadSliderComplete = window.document.createElement("hbox");
- var downloadLabelExplanation = window.document.createElement("label");
- var downloadLabelNote = window.document.createElement("description");
- var downloadSlider = window.document.createElement("scale");
- var downloadLabel = window.document.createElement("label");
+ var downloadSliderComplete = window.document.createXULElement("hbox");
+ var downloadLabelExplanation = window.document.createXULElement("label");
+ var downloadLabelNote = window.document.createXULElement("description");
+ var downloadSlider = window.document.createXULElement("scale");
+ var downloadLabel = window.document.createXULElement("label");
downloadLabelExplanation.setAttribute("value", FAMS.getLocalisedString("Options-Download-Freq.label"));
downloadLabelNote.textContent = FAMS.getLocalisedString("Options-Download-Freq.desc", config.name);
@@ -142,14 +146,14 @@ function renderMessageGroup(msgGroupIndex)
var msgGroup = config.messageGroups[msgGroupIndex];
var msgGroupContainer = window.document.getElementById("msgGroupContainer");
- var singleMsgGroupContainer = window.document.createElement("vbox");
- var desc = window.document.createElement("description");
- var appearanceLabelNote = window.document.createElement("description");
- var enabledCheckbox = window.document.createElement("checkbox");
- var appearanceSliderComplete = window.document.createElement("hbox");
- var appearanceLabelExplanation = window.document.createElement("label");
- var appearanceSlider = window.document.createElement("scale");
- var appearanceLabel = window.document.createElement("label");
+ var singleMsgGroupContainer = window.document.createXULElement("vbox");
+ var desc = window.document.createXULElement("description");
+ var appearanceLabelNote = window.document.createXULElement("description");
+ var enabledCheckbox = window.document.createXULElement("checkbox");
+ var appearanceSliderComplete = window.document.createXULElement("hbox");
+ var appearanceLabelExplanation = window.document.createXULElement("label");
+ var appearanceSlider = window.document.createXULElement("scale");
+ var appearanceLabel = window.document.createXULElement("label");
//var seeAllButton = window.document.createElement("button");
desc.textContent = FAMS.locale.internationaliseString(msgGroup.description);
diff --git a/xul-ext/chrome/content/famsOptions.xul b/xul-ext/chrome/content/famsOptions.xhtml
similarity index 98%
rename from xul-ext/chrome/content/famsOptions.xul
rename to xul-ext/chrome/content/famsOptions.xhtml
index 7738c19..b4d8b1f 100644
--- a/xul-ext/chrome/content/famsOptions.xul
+++ b/xul-ext/chrome/content/famsOptions.xhtml
@@ -26,7 +26,6 @@