// http://infocatcher.ucoz.net/js/cb/cookiesPermissions.js // https://forum.mozilla-russia.org/viewtopic.php?id=56039 // https://github.com/Infocatcher/Custom_Buttons/tree/master/Cookies_Permissions // Cookies Permissions button for Custom Buttons // (code for "initialization" section) // (c) Infocatcher 2010-2021 // version 0.2.1pre8 - 2021-02-14 var {Components} = window; // Prevent garbage collection in Firefox 3.6 and older var cp = Components.interfaces.nsICookiePermission; var options = { removeUnprotectedCookiesEnabled: false, // true - periodically remove unprotected cookies by default // false - don't remove by default removeUnprotectedCookiesInterval: 10*60*1000, // Periodically remove unprotected cookies (leave only cookies with "Allow" permission) // Time in milliseconds like 30*60*1000 (30 minutes) or -1 to disable removeAllUnprotectedCookies: false, // true - periodically ("removeUnprotectedCookiesInterval" option) remove all unprotected cookies // false - or exclude cookies from opened sites showTempPermissions: true, // Show items about temporary permissions (only Gecko 2.0+) tempExpire: -1, // Type of temporary permissions // -1 - session, otherwise - expire after given time (in milliseconds) useBaseDomain: { // 0 - use full domain name: addons.mozilla.org, www.google.com // 1 - strip "www." prefix from full domain name: addons.mozilla.org, google.com // 2 - use top-level domains (TLDs): mozilla.org, google.com addPermission: 1, // Add (and toggle) permission action openPermissions: 0, // Filter in "Show Exceptions" window showCookies: 2, // Filter in "Show Cookies" window removeCurrentSiteCookies: 2, // For "Remove All Current Site Cookies" action preserveCurrentSitesCookies: 2 // For "removeAllUnprotectedCookies: false" }, showDefaultPolicy: true, // Show default cookies policy toggleMode: [cp.ACCESS_ALLOW, cp.ACCESS_DENY, cp.ACCESS_DEFAULT], // ACCESS_DENY, ACCESS_SESSION, ACCESS_ALLOW or ACCESS_DEFAULT useCookiesManagerPlus: true, // https://addons.mozilla.org/firefox/addon/cookies-manager-plus/ reusePermissionsWindow: false, // Use any already opened permissions window // E.g. "Show Exceptions" may convert "Allowed Sites - Add-ons Installation" to "Exceptions - Cookies" prefillMode: 1, // 0 - move caret to start, 1 - select all, 2 - move caret to end confirmRemoval: true, moveToStatusBar: { // Move button to Status Bar, only for SeaMonkey or Firefox < 4.0 // Be careful, has some side-effects and button can't be edited w/o restart enabled: false, insertAfter: "download-monitor,popupIcon,statusbar-progresspanel" // Like https://developer.mozilla.org/en-US/docs/XUL/Attribute/insertafter // Also looks for nodes with "cb_id" attribute } }; function _localize(sid) { var strings = { en: { defaultTooltiptext: "Cookies: Default", denyTooltiptext: "Cookies: Block", allowSessionTooltiptext: "Cookies: Allow for Session", allowTooltiptext: "Cookies: Allow", notAvailableTooltiptext: "Cookies: n/a", unknownTooltiptext: "Cookies: ???", errorTooltiptext: "Cookies: Error!", defaultDenyTooltiptext: "Cookies: Block (Default)", defaultAllowSessionTooltiptext: "Cookies: Allow for Session (Default)", defaultAllowTooltiptext: "Cookies: Allow (Default)", defaultLabel: "Default", defaultAccesskey: "D", denyLabel: "Block", denyAccesskey: "B", denyTempLabel: "Temporarily Block", denyTempAccesskey: "k", allowSessionLabel: "Allow for Session", allowSessionAccesskey: "S", allowSessionTempLabel: "Temporarily Allow for Session", allowSessionTempAccesskey: "e", allowLabel: "Allow", allowAccesskey: "A", allowTempLabel: "Temporarily Allow", allowTempAccesskey: "w", showPermissionsLabel: "Show Exceptions…", showPermissionsAccesskey: "x", showCookiesLabel: "Show Cookies…", showCookiesAccesskey: "h", removeTempPermissionsLabel: "Remove Temporary Permissions", removeTempPermissionsAccesskey: "T", autoRemoveUnprotectedCookiesLabel: "Automatically Remove Unprotected Cookies", autoRemoveUnprotectedCookiesTip: "Periodically remove unprotected cookies (if checked)", autoRemoveUnprotectedCookiesAccesskey: "n", removeUnprotectedCookiesLabel: "Remove Unprotected Cookies", removeUnprotectedCookiesTip: "Except cookies marked as “Allow” and except cookies from opened sites", removeUnprotectedCookiesAccesskey: "U", removeUnprotectedCookiesConfirm: "Remove unprotected cookies?", removeAllUnprotectedCookiesLabel: "Remove All Unprotected Cookies", removeAllUnprotectedCookiesTip: "Except cookies marked as “Allow”; unprotected cookies from opened sites will be removed", removeAllUnprotectedCookiesAccesskey: "R", removeAllUnprotectedCookiesConfirm: "Remove ALL unprotected cookies?", removeCurrentSiteCookiesLabel: "Remove All Current Site Cookies", removeCurrentSiteCookiesAccesskey: "C", removeCurrentSiteCookiesConfirm: "Remove ALL current site cookies?", removeAllCookiesLabel: "Remove ALL Cookies", removeAllCookiesAccesskey: "L", removeAllCookiesConfirm: "Remove ALL cookies?", buttonMenu: "Button Menu", buttonMenuAccesskey: "M" }, ru: { defaultTooltiptext: "Cookies: По умолчанию", denyTooltiptext: "Cookies: Блокировать", allowSessionTooltiptext: "Cookies: Разрешить на сессию", allowTooltiptext: "Cookies: Разрешить", notAvailableTooltiptext: "Cookies: н/д", unknownTooltiptext: "Cookies: ???", errorTooltiptext: "Cookies: Ошибка!", defaultDenyTooltiptext: "Cookies: Блокировать (по умолчанию)", defaultAllowSessionTooltiptext: "Cookies: Разрешить на сессию (по умолчанию)", defaultAllowTooltiptext: "Cookies: Разрешить (по умолчанию)", defaultLabel: "По умолчанию", defaultAccesskey: "у", denyLabel: "Блокировать", denyAccesskey: "Б", denyTempLabel: "Временно блокировать", denyTempAccesskey: "л", allowSessionLabel: "Разрешить на сессию", allowSessionAccesskey: "с", allowSessionTempLabel: "Временно разрешить на сессию", allowSessionTempAccesskey: "е", allowLabel: "Разрешить", allowAccesskey: "Р", allowTempLabel: "Временно разрешить", allowTempAccesskey: "ш", showPermissionsLabel: "Показать исключения…", showPermissionsAccesskey: "и", showCookiesLabel: "Показать cookies…", showCookiesAccesskey: "П", removeTempPermissionsLabel: "Удалить временные исключения", removeTempPermissionsAccesskey: "ы", autoRemoveUnprotectedCookiesLabel: "Автоматически удалять незащищённые cookies", autoRemoveUnprotectedCookiesTip: "Периодически удалять незащищённые cookies (если установлена галочка)", autoRemoveUnprotectedCookiesAccesskey: "А", removeUnprotectedCookiesLabel: "Удалить незащищённые cookies", removeUnprotectedCookiesTip: "Исключая cookies, помеченные как «Разрешить», и исключая cookies из открытых сайтов", removeUnprotectedCookiesAccesskey: "н", removeUnprotectedCookiesConfirm: "Удалить незащищённые cookies?", removeAllUnprotectedCookiesLabel: "Удалить все незащищённые cookies", removeAllUnprotectedCookiesTip: "Исключая cookies, помеченные как «Разрешить»; незащищённые cookies из открытых сайтов будут удалены", removeAllUnprotectedCookiesAccesskey: "д", removeAllUnprotectedCookiesConfirm: "Удалить ВСЕ незащищённые cookies?", removeCurrentSiteCookiesLabel: "Удалить все cookies текущего сайта", removeCurrentSiteCookiesAccesskey: "в", removeCurrentSiteCookiesConfirm: "Удалить все cookies текущего сайта?", removeAllCookiesLabel: "Удалить ВСЕ cookies", removeAllCookiesAccesskey: "Е", removeAllCookiesConfirm: "Удалить ВСЕ cookies?", buttonMenu: "Меню кнопки", buttonMenuAccesskey: "М" } }; var locale = (function() { if("Services" in window && "locale" in Services) { var locales = Services.locale.requestedLocales // Firefox 64+ || Services.locale.getRequestedLocales && Services.locale.getRequestedLocales(); if(locales) return locales[0]; } var prefs = "Services" in window && Services.prefs || Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefBranch); function pref(name, type) { return prefs.getPrefType(name) != prefs.PREF_INVALID ? prefs["get" + type + "Pref"](name) : undefined; } if(!pref("intl.locale.matchOS", "Bool")) { // Also see https://bugzilla.mozilla.org/show_bug.cgi?id=1414390 var locale = pref("general.useragent.locale", "Char"); if(locale && locale.substr(0, 9) != "chrome://") return locale; } return Components.classes["@mozilla.org/chrome/chrome-registry;1"] .getService(Components.interfaces.nsIXULChromeRegistry) .getSelectedLocale("global"); })().match(/^[a-z]*/)[0]; _localize = function(sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize.apply(this, arguments); } this.onclick = function(e) { if(e.target != this) return; var btn = e.button; if(btn == 1 || btn == 0 && this.permissions.hasModifier(e)) this.permissions.openPermissions(); else if(btn == 0) { this.permissions.togglePermission(this.permissions.options.toggleMode); // Prevent "command" event to use "command" section only from hotkey e.preventDefault(); e.stopPropagation(); } }; if(!this.hasOwnProperty("defaultContextId")) this.defaultContextId = this.getAttribute("context") || "custombuttons-contextpopup"; this.oncontextmenu = function(e) { if(e.target != this) return; this.permissions.initContextOnce(); this.setAttribute( "context", this.permissions.hasModifier(e) ? this.defaultContextId : this.permissions.mpId ); }; this.permissions = { permissionType: "cookie", timerId: "customButtonsCookiesCleanupTimer", timer: null, popupClass: "cbCookiesPermissionsPopup", button: this, options: options, cp: Components.interfaces.nsICookiePermission, PERMISSIONS_NOT_SUPPORTED: -1, PERMISSIONS_ERROR: -2, errPrefix: "[Custom Buttons :: Cookies Permissions] ", get pm() { delete this.pm; return this.pm = Components.classes["@mozilla.org/permissionmanager;1"] .getService(Components.interfaces.nsIPermissionManager); }, get cm() { delete this.cm; return this.cm = Components.classes["@mozilla.org/cookiemanager;1"] .getService(Components.interfaces.nsICookieManager); }, get io() { delete this.io; return this.io = Components.classes["@mozilla.org/network/io-service;1"] .getService(Components.interfaces.nsIIOService); }, get oSvc() { return Components.classes["@mozilla.org/observer-service;1"] .getService(Components.interfaces.nsIObserverService); }, get wm() { delete this.wm; return this.wm = Components.classes["@mozilla.org/appshell/window-mediator;1"] .getService(Components.interfaces.nsIWindowMediator); }, get tld() { delete this.tld; return this.tld = Components.classes["@mozilla.org/network/effective-tld-service;1"] .getService(Components.interfaces.nsIEffectiveTLDService); }, get app() { delete this.app; return this.app = Components.classes["@mozilla.org/xre/app-info;1"] .getService(Components.interfaces.nsIXULAppInfo); }, get platformVersion() { var pv = parseFloat(this.app.platformVersion); if(this.app.name == "Pale Moon" || this.app.name == "Basilisk") pv = pv >= 4.1 ? 56 : 28; delete this.platformVersion; return this.platformVersion = pv; }, initialized: false, mp: null, init: function() { if(this.initialized) return; this.initialized = true; if(this.options.moveToStatusBar.enabled) this.moveToStatusBar(); var dummy = function() {}; this.progressListener = { context: this, onStateChange: dummy, onProgressChange: dummy, onLocationChange: function(aWebProgress, aRequest, aLocation) { setTimeout(function(_this) { _this.context.updButtonState(); }, 0, this); }, onStatusChange: dummy, onSecurityChange: dummy }; gBrowser.addProgressListener(this.progressListener/*, Components.interfaces.nsIWebProgress.NOTIFY_LOCATION*/); this.permissionsObserver = { context: this, observe: function(subject, topic, data) { if(topic != "perm-changed") return; var permission = subject.QueryInterface(Components.interfaces.nsIPermission); var type = this.context.permissionType; if(permission.type != type) return; this.context.updButtonState(); if(data == "deleted") { // See chrome://browser/content/preferences/permissions.js // observe: function (aSubject, aTopic, aData) let win = this.context.wm.getMostRecentWindow("Browser:Permissions"); if(win && "gPermissionManager" in win && win.gPermissionManager._type == type) { let pm = win.gPermissionManager; let perms = pm._permissions; for(let i = 0, l = perms.length; i < l; ++i) { if(this.context.getPermissionHost(perms[i]) == this.context.getPermissionHost(permission)) { perms.splice(i, 1); --pm._view._rowCount; pm._tree.treeBoxObject.rowCountChanged(i, -1); pm._tree.treeBoxObject.invalidate(); break; } } } } } }; this.oSvc.addObserver(this.permissionsObserver, "perm-changed", false); if(this.options.showDefaultPolicy) { let po = this.prefsObserver = { context: this, get prefs() { delete this.prefs; return this.prefs = Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefService) .getBranch("network.cookie.") .QueryInterface(Components.interfaces.nsIPrefBranch2 || Components.interfaces.nsIPrefBranch); }, getIntPref: function(name) { var dv = 0; try { return this.prefs.getIntPref(name, dv); } catch(e) { Components.utils.reportError(e); } return dv; }, observe: function(subject, topic, data) { if(topic != "nsPref:changed") return; if(data != "cookieBehavior" && data != "lifetimePolicy") return; this.context.prefs[data] = this.getIntPref(data); this.context.updButtonState(); } }; this.prefs = { "cookieBehavior": po.getIntPref("cookieBehavior"), "lifetimePolicy": po.getIntPref("lifetimePolicy"), __proto__: null }; po.prefs.addObserver("", po, false); } this.initCleanupTimer(); this.updButtonState(); }, destroy: function() { if(!this.initialized) return; this.initialized = false; gBrowser.removeProgressListener(this.progressListener); this.oSvc.removeObserver(this.permissionsObserver, "perm-changed"); if(this.options.showDefaultPolicy) this.prefsObserver.prefs.removeObserver("", this.prefsObserver); this.progressListener = this.permissionsObserver = this.prefsObserver = null; }, initContextOnce: function() { this.initContextOnce = function() {}; this.mpId = this.button.id + "-context"; var cp = this.cp; var noTempPermissions = !this.options.showTempPermissions || !this.hasTempPermissions; var mp = this.mp = this.button.appendChild(this.parseXULFromString('\ \ \ \ \ \ \ \ \ \ \ \ \ ' )); var cbPopup = document.getElementById(this.button.defaultContextId); if(!cbPopup) Components.utils.reportError(this.errPrefix + "cb menu not found"); else { cbPopup = cbPopup.cloneNode(true); let id = "-" + this.button.id.match(/\d*$/)[0] + "-cloned"; cbPopup.id += id; Array.prototype.slice.call(cbPopup.getElementsByAttribute("id", "*")).forEach(function(node) { node.id += id; }); cbPopup.setAttribute( "onpopupshowing", '\ var btn = document.popupNode = this.parentNode.parentNode.parentNode;\n\ custombutton.setContextMenuVisibility(btn);' ); let menu = mp.lastChild; menu.appendChild(cbPopup); } }, setAutoRemove: function(enable) { if(enable) { this._forceEnableCleanup = true; this.initCleanupTimer(); // Ensure initialized } var timer = this.timer || null; if(timer) timer.enabled = enable; }, _forceEnableCleanup: false, initCleanupTimer: function() { if(!this.options.removeUnprotectedCookiesEnabled && !this._forceEnableCleanup) return; var interval = this.options.removeUnprotectedCookiesInterval; if(interval <= 0) return; var timerId = this.timerId; var timer = this.timer = this.storage.get(timerId, null); if(timer) return; timer = this.timer = { timerId: timerId, interval: interval, enabled: true, permissions: this, get timer() { delete this.timer; return this.timer = Components.classes["@mozilla.org/timer;1"] .createInstance(Components.interfaces.nsITimer); }, init: function() { window.addEventListener("unload", this, false); this.permissions.oSvc.addObserver(this, "quit-application-granted", false); this.timer.init(this, this.interval, this.timer.TYPE_REPEATING_SLACK); }, destroy: function() { this.permissions.oSvc.removeObserver(this, "quit-application-granted"); this.timer.cancel(); this.permissions.storage.set(this.timerId, null); this.permissions = null; }, handleEvent: function(e) { if(e.type != "unload") return; window.removeEventListener("unload", this, false); var p = this.permissions; p.button = p.mp = null; }, observe: function(subject, topic, data) { if(topic == "quit-application-granted") this.destroy(); else if(topic == "timer-callback") this.enabled && this.permissions.removeUnprotectedCookies(); } }; this.storage.set(timerId, timer); timer.init(); }, moveToStatusBar: function() { var insPoint; this.options.moveToStatusBar.insertAfter .split(/,\s*/) .some(function(id) { insPoint = document.getElementsByAttribute("cb_id", id)[0] || document.getElementById(id); return insPoint; }); if(!insPoint) return; var btn = this.button; // Make looks like , see CSS btn.className += " custombuttons-insideStatusbarpanel"; // And insert it into var spId = btn.id + "-statusbarpanel"; var sp = document.getElementById(spId); sp && sp.parentNode.removeChild(sp); sp = document.createElement("statusbarpanel"); sp.id = spId; sp.setAttribute("cb_id", "custombuttons-cookiesPermissionsSBPanel"); sp.appendChild(btn); insPoint.parentNode.insertBefore(sp, insPoint.nextSibling); }, get currentHost() { return this.getHostFromBrowser(gBrowser); }, get currentHosts() { // returns hosts from all visible tabs in all windows var tmp = { __proto__: null }; var ws = this.wm.getEnumerator("navigator:browser"); while(ws.hasMoreElements()) { let gBrowser = ws.getNext().gBrowser; let tabs = gBrowser.visibleTabs || gBrowser.tabs || gBrowser.tabContainer.childNodes; for(let i = 0, l = tabs.length; i < l; ++i) { let browser = tabs[i].linkedBrowser; let host = browser && this.getHostFromBrowser(browser); if(!host) continue; host = this.getHost(this.options.useBaseDomain.preserveCurrentSitesCookies, host); tmp[host] = true; } } var hosts = []; for(var host in tmp) hosts.push(host); return hosts; }, getHostFromBrowser: function(browser) { try { var uri = browser.currentURI; if(["chrome", "resource"].indexOf(uri.scheme) != -1) return ""; return uri.host; } catch(e) { } return ""; }, get currentProtocol() { var scheme = gBrowser.currentURI.scheme; if(scheme == "https") return scheme; return "http"; }, getHost: function(useBaseDomain, host) { if(host === undefined) host = this.currentHost; switch(useBaseDomain) { case 1: return this.stripWww(host); case 2: return this.getBaseDomain(host); } return host; }, getURI: function(host, protocol) { if(host.indexOf(":") != -1 && /^[:\da-f.]+$/.test(host)) // IPv6 host = "[" + host + "]"; host = host.replace(/^\./, ""); try { return this.io.newURI((protocol || this.currentProtocol) + "://" + host, null, null); } catch(e) { Components.utils.reportError(this.errPrefix + "Invalid host: \"" + host + "\""); throw e; } }, stripWww: function(host) { return host && host.replace(/^www\./i, ""); }, getBaseDomain: function(host) { if(host) try { return this.tld.getBaseDomainFromHost(host); } catch(e) { } return host; }, showMenu: function(e, isContext, mp) { document.popupNode = this.button.ownerDocument.popupNode = this.button; if(!mp) { this.initContextOnce(); mp = this.mp; } if("openPopupAtScreen" in mp) mp.openPopupAtScreen(e.screenX, e.screenY, isContext); else mp.showPopup(this, e.screenX, e.screenY, isContext ? "context" : "popup", null, null); }, updMenu: function() { var permission = this.options.showTempPermissions ? this.getPermissionEx() : this.getPermission(); var noPermissions = permission == this.PERMISSIONS_NOT_SUPPORTED; Array.prototype.forEach.call( this.mp.getElementsByAttribute("cb_permission", "*"), function(mi) { mi.hidden = noPermissions; var ns = mi.nextSibling; if(ns && ns.localName == "menuseparator") ns.hidden = noPermissions; } ); if(!noPermissions) { let cbPermission = permission.capability || permission; if( this.options.showTempPermissions && permission instanceof Components.interfaces.nsIPermission && "expireType" in permission && permission.expireType != this.pm.EXPIRE_NEVER ) cbPermission += "-temp"; let mi = this.mp.getElementsByAttribute("cb_permission", cbPermission)[0] || null; mi && mi.setAttribute("checked", "true"); } if(this.hasTempPermissions) { let maxItems = 10; let removeItem = this.mp.getElementsByAttribute("cb_id", "removeTempPermissions")[0]; let tempPermissions = this.tempPermissions; removeItem.disabled = !tempPermissions.length; if(tempPermissions.length > maxItems) tempPermissions.splice(maxItems - 2, tempPermissions.length - maxItems + 1, "…"); let cp = this.cp; removeItem.tooltipText = tempPermissions.map(function(permission) { if(typeof permission == "string") return permission; var action = "???"; switch(permission.capability) { case cp.ACCESS_ALLOW: action = "allowLabel"; break; case cp.ACCESS_DENY: action = "denyLabel"; break; case cp.ACCESS_SESSION: action = "allowSessionLabel"; } return (permission.host || permission.principal.URI.spec.replace(/\/$/, "")) + ": " + _localize(action).toLowerCase(); }, this).join(", \n"); } var removeCurrent = this.mp.getElementsByAttribute("cb_id", "removeCurrentSiteCookies")[0]; removeCurrent.hidden = noPermissions; if(!noPermissions) removeCurrent.tooltipText = this.removeCurrentSiteCookiesHost; var timer = this.timer || null; var autoRemove = this.mp.getElementsByAttribute("cb_id", "autoRemoveUnprotectedCookies")[0]; autoRemove.setAttribute("checked", timer ? timer.enabled : false); autoRemove.hidden = this.options.removeUnprotectedCookiesInterval <= 0; return true; }, openPermissions: function() { var host = this.getHost(this.options.useBaseDomain.openPermissions); if(host && this.platformVersion >= 42) host = this.currentProtocol + "://" + host; if(this.app.name == "SeaMonkey") { this.openPermissionsSM(host); return; } // chrome://browser/content/preferences/privacy.js // gPrivacyPane.showCookieExceptions() var bundle = Components.classes["@mozilla.org/intl/stringbundle;1"] .getService(Components.interfaces.nsIStringBundleService) .createBundle("chrome://browser/locale/preferences/preferences.properties"); try { var cpTitle = bundle.GetStringFromName("cookiepermissionstitle"); var cpText = bundle.GetStringFromName("cookiepermissionstext"); } catch(e) { // Not used in Firefox 62+ } var params = { blockVisible : true, sessionVisible : true, allowVisible : true, prefilledHost : host, permissionType : this.permissionType, windowTitle : cpTitle, introText : cpText }; var win; var ws = this.wm.getEnumerator("Browser:Permissions"); while(ws.hasMoreElements()) { win = ws.getNext(); if( this.options.reusePermissionsWindow || "gPermissionManager" in win && win.gPermissionManager._type == this.permissionType ) break; win = null; } var _this = this; var setFilter = function setFilter(e) { e && win.removeEventListener("load", setFilter, false); setTimeout(function() { _this.setTextboxValue(win.document.getElementById("url"), host, !!e); }, 0); }; if(win) { // See in chrome://global/content/bindings/preferences.xml#prefwindow if("initWithParams" in win) win.initWithParams(params); win.focus(); host && setFilter(); } else { var ext = this.platformVersion >= 72 ? ".xhtml" : ".xul"; var sub = this.platformVersion >= 77 ? "dialogs/" : ""; win = window.openDialog("chrome://browser/content/preferences/" + sub + "permissions" + ext, "_blank", "", params); host && win.addEventListener("load", setFilter, false); } this.tweakWindow(win); }, openPermissionsSM: function(host) { var win = this.wm.getMostRecentWindow("mozilla:cookieviewer"); var _this = this; if(!host) host = ""; var setFilter = function setFilter(e) { e && win.removeEventListener("load", setFilter, false); var doc = win.document; doc.getElementById("tabbox").selectedTab = doc.getElementById("permissionsTab"); _this.setTextboxValue(doc.getElementById("cookie-site"), host); }; if(win) { win.focus(); setFilter(); } else { win = window.openDialog("chrome://communicator/content/permissions/cookieViewer.xul", "_blank", ""); win.addEventListener("load", setFilter, false); } }, tweakWindow: function(win) { if("__cbPermissionsTweaked" in win) return; win.__cbPermissionsTweaked = true; function keypressHandler(e) { if(e.keyCode == e.DOM_VK_ESCAPE) win.close(); } win.addEventListener("keypress", keypressHandler, false); win.addEventListener("unload", function destroy(e) { var win = e.target.defaultView; if(win != e.currentTarget) return; win.removeEventListener(e.type, destroy, false); win.removeEventListener("keypress", keypressHandler, false); }, false); }, setTextboxValue: function(tb, val, onlySelect) { if(!tb) return; if(!onlySelect) tb.value = val; tb.focus(); if(val && "inputField" in tb) { let ifi = tb.inputField; switch(this.options.prefillMode) { case 0: ifi.selectionStart = ifi.selectionEnd = 0; break; case 2: ifi.selectionStart = ifi.selectionEnd = val.length; break; default: tb.select(); } } if(onlySelect) return; setTimeout(function() { // For Browser:Cookies in Firefox 14 tb.doCommand(); // Should be faster than "input" emulation }, 0); var evt = document.createEvent("UIEvents"); evt.initUIEvent("input", true, true, tb.ownerDocument.defaultView, 0); tb.dispatchEvent(evt); }, get hasTempPermissions() { delete this.hasTempPermissions; return this.hasTempPermissions = "EXPIRE_SESSION" in this.pm && (!("add" in this.pm) || this.pm.add.length > 3); }, get pmw() { delete this.pmw; var pm = this.pm; if("testPermission" in pm) { return this.pmw = { testPermission: pm.testPermission.bind(pm), add: pm.add .bind(pm), remove: pm.remove .bind(pm), __proto__: pm }; } // Firefox 71+ var make = function(fn) { return function(uri, permission) { var principal = Services.scriptSecurityManager.createContentPrincipal(uri, {}); var args = Array.from(arguments); args[0] = principal; return fn.apply(pm, args); }; }; return this.pmw = { _context: this, testPermission: make(pm.testPermissionFromPrincipal), add: make(pm.addFromPrincipal), remove: make(pm.removeFromPrincipal), get enumerator() { // Firefox 72+ return pm.enumerator || this._context.arrayToEnumerator(pm.all); } }; }, arrayToEnumerator: function(arr) { var i = 0, l = arr.length; return { hasMoreElements: function() { return i < l; }, getNext: function() { return arr[i++]; } } }, addPermission: function(capability, temporary) { // capability: // this.cp.ACCESS_ALLOW (this.pm.ALLOW_ACTION) // this.cp.ACCESS_SESSION // this.cp.ACCESS_DENY (this.pm.DENY_ACTION) var host = this.getHost(this.options.useBaseDomain.addPermission); if(!host) return; if(temporary && !this.hasTempPermissions) temporary = false; this.updButtonState(capability); // Faster than ProgressListener (70-80 ms for me) if(this.hasTempPermissions) this.removePermissionForHost(host); var pm = this.pm; var args = [this.getURI(host), this.permissionType, capability]; if(temporary) { let expire = this.options.tempExpire; if(expire < 0) args.push(pm.EXPIRE_SESSION); else args.push(pm.EXPIRE_TIME, expire + Date.now()); } this.pmw.add.apply(this.pmw, args); }, removePermission: function() { var host = this.currentHost; if(!host) return; this.updButtonState(this.cp.ACCESS_DEFAULT); // Faster than ProgressListener (70-80 ms for me) var uri = this.getURI(host); var permission = this.pmw.testPermission(uri, this.permissionType); this.removePermissionForHost(host); while(this.pmw.testPermission(uri, this.permissionType) == permission) { let parentHost = host.replace(/^[^.]*\./, ""); if(parentHost == host) break; host = parentHost; this.removePermissionForHost(host); } }, togglePermission: function(capabilities) { var permission = this.getPermissionEx(this.getHost(this.options.useBaseDomain.addPermission), true); if(permission instanceof Components.interfaces.nsIPermission) permission = permission.capability; if(permission == this.PERMISSIONS_NOT_SUPPORTED) return; var capability = capabilities[(capabilities.indexOf(permission) + 1) % capabilities.length]; if(capability == this.cp.ACCESS_DEFAULT) this.removePermission(); else this.addPermission(capability); }, get tempPermissions() { var out = []; if(!this.hasTempPermissions) return out; var pm = this.pm; var enumerator = this.pmw.enumerator; while(enumerator.hasMoreElements()) { let permission = enumerator.getNext() .QueryInterface(Components.interfaces.nsIPermission); if( permission.type == this.permissionType && permission.expireType != pm.EXPIRE_NEVER ) out.push(permission); } return out; }, removeTempPermissions: function() { this.tempPermissions.forEach(this.removeRawPermission, this); }, getPermission: function() { var host = this.currentHost; return host ? this.pmw.testPermission(this.getURI(host), this.permissionType) : this.PERMISSIONS_NOT_SUPPORTED; }, getPermissionEx: function(host, dontGetInherited) { // Unfortunately no API like nsIPermissionManager.testPermission() // for temporary permissions if(!host) host = this.currentHost; if(!host) return this.PERMISSIONS_NOT_SUPPORTED; var pm = this.pm; var matchedPermission = pm.UNKNOWN_ACTION; var protocol = this.currentProtocol; var maxHostLen = -1; var enumerator = this.pmw.enumerator; while(enumerator.hasMoreElements()) { let permission = enumerator.getNext() .QueryInterface(Components.interfaces.nsIPermission); if(permission.type != this.permissionType) continue; if("principal" in permission && permission.principal.URI.scheme != protocol) // Firefox 42+ continue; let permissionHost = this.getPermissionHost(permission); if(permissionHost == host) return permission; if(dontGetInherited) continue; let hostLen = permissionHost.length; if( hostLen > maxHostLen && host.substr(-hostLen - 1) == "." + permissionHost // ~= checkCookieHost() ) { matchedPermission = permission; maxHostLen = hostLen; } } return matchedPermission; }, removePermissionForHost: function(host) { try { this.pmw.remove(host, this.permissionType); } catch(e) { // See https://bugzilla.mozilla.org/show_bug.cgi?id=1170200 if("Services" in window) try { // Firefox 42+ let uri = Services.io.newURI(this.currentProtocol + "://" + host, null, null); this.pmw.remove(uri, this.permissionType); return; } catch(e2) { Components.utils.reportError(e2); } Components.utils.reportError(e); } }, removeRawPermission: function(permission) { if("principal" in permission) // Firefox 42+ this.pmw.remove(permission.principal.URI, this.permissionType); else this.removePermissionForHost(permission.host); }, getPermissionHost: function(permission) { if("host" in permission) return permission.host; // See https://bugzilla.mozilla.org/show_bug.cgi?id=1173523 return permission.principal.URI.host; // Firefox 42+ }, get defaultPermission() { // http://kb.mozillazine.org/Network.cookie.cookieBehavior // http://kb.mozillazine.org/Network.cookie.lifetimePolicy if(this.prefs.cookieBehavior == 2) return this.cp.ACCESS_DENY; if(this.prefs.lifetimePolicy == 2) return this.cp.ACCESS_SESSION; return this.cp.ACCESS_ALLOW; }, showCookiesEvent: function(e) { if(e.type == "command") this.showCookies(this.hasModifier(e)); else if(e.type == "click" && e.button == 1) { this.mp.hidePopup(); this.showCookies(true); } }, showCookies: function(showAll) { var host = showAll ? "" : this.getHost(this.options.useBaseDomain.showCookies); if("coomanPlus" in window && "coomanPlusCore" in window && this.options.useCookiesManagerPlus) { // https://addons.mozilla.org/firefox/addon/cookies-manager-plus/ this.showCookiesCMP(host); return; } if(this.app.name == "SeaMonkey") { this.showCookiesSM(host); return; } if( (this.app.name == "Firefox" || this.app.name == "Waterfox") && parseFloat(this.app.version) >= 61 ) { this.showSiteData(host); return; } var win = this.wm.getMostRecentWindow("Browser:Cookies"); var _this = this; var setFilter = function setFilter(e) { e && win.removeEventListener("load", setFilter, false); _this.setTextboxValue(win.document.getElementById("filter"), host); }; if(win) { win.focus(); (host || showAll) && setFilter(); } else { win = window.openDialog("chrome://browser/content/preferences/cookies.xul", "_blank", ""); host && win.addEventListener("load", setFilter, false); } this.tweakWindow(win); }, showCookiesCMP: function(host) { // See openCMP() function in resource://cookiesmanagerplus/coomanPlusCore.jsm var win = coomanPlusCore.aWindow; var _this = this; var setFilter = function setFilter(e) { e && win.removeEventListener("load", setFilter, false); var doc = win.document; setTimeout(function() { // Just loaded window aren't ready _this.setTextboxValue(doc.getElementById("lookupcriterium"), host); }, 0); }; if(win) { win.focus(); host && setFilter(); } else { win = window.openDialog( "chrome://cookiesmanagerplus/content/cookiesmanagerplus.xul", "coomanPlusWindow", "chrome,resizable=yes,toolbar=no,statusbar=no,scrollbar=no,centerscreen" ); host && win.addEventListener("load", setFilter, false); } }, showCookiesSM: function(host) { var win = this.wm.getMostRecentWindow("mozilla:cookieviewer"); var _this = this; if(!host) host = ""; var setFilter = function setFilter(e) { e && win.removeEventListener("load", setFilter, false); var doc = win.document; doc.getElementById("tabbox").selectedTab = doc.getElementById("cookiesTab"); _this.setTextboxValue(doc.getElementById("filter"), host); }; if(win) { win.focus(); setFilter(); } else { win = window.openDialog("chrome://communicator/content/permissions/cookieViewer.xul", "_blank", ""); win.addEventListener("load", setFilter, false); } }, showSiteData: function(host) { var win = this.wm.getMostRecentWindow("Browser:SiteDataSettings"); var _this = this; var setFilter = function setFilter(e) { e && win.removeEventListener("load", setFilter, false); _this.setTextboxValue(win.document.getElementById("searchBox"), host); }; if(win) { win.focus(); host && setFilter(); } else { var {SiteDataManager} = Components.utils.import("resource:///modules/SiteDataManager.jsm", {}); SiteDataManager.updateSites().then(function() { var ext = _this.platformVersion >= 72 ? ".xhtml" : ".xul"; var sub = _this.platformVersion >= 77 ? "dialogs/" : ""; win = window.openDialog("chrome://browser/content/preferences/" + sub + "siteDataSettings" + ext, "_blank", ""); host && win.addEventListener("load", setFilter, false); }, Components.utils.reportError); } }, confirm: function(msg, method/*, arg1, arg2*/) { if( !this.options.confirmRemoval || Components.classes["@mozilla.org/embedcomp/prompt-service;1"] .getService(Components.interfaces.nsIPromptService) .confirm(window, _localize("Cookies Permissions"), _localize(msg)) ) this[method].apply(this, Array.prototype.slice.call(arguments, 2)); }, removeUnprotectedCookies: function(removeAll) { if(removeAll == undefined) removeAll = this.options.removeAllUnprotectedCookies; var cp = this.cp; var checkCookieHosts; if(!removeAll) { let hosts = this.currentHosts; let checkCookieHost = this.checkCookieHost; checkCookieHosts = function(cookieHost) { return !hosts.some(function(host) { return checkCookieHost(cookieHost, host); }); }; } this.removeCookies([ cp.ACCESS_DEFAULT, /*cp.ACCESS_ALLOW,*/ cp.ACCESS_DENY, cp.ACCESS_SESSION ], checkCookieHosts); }, get removeCurrentSiteCookiesHost() { return this.getHost(this.options.useBaseDomain.removeCurrentSiteCookies); }, removeCurrentSiteCookies: function() { var host = this.removeCurrentSiteCookiesHost; var checkCookieHost = this.checkCookieHost; this.removeCookies(null, function(cookieHost) { return checkCookieHost(cookieHost, host); }); }, get checkCookieHost() { delete this.checkCookieHost; return this.checkCookieHost = "endsWith" in String.prototype // Firefox 17+ ? function(cookieHost, host) { return host == cookieHost || cookieHost.endsWith("." + host); } : function(cookieHost, host) { return host == cookieHost || cookieHost.substr(-host.length - 1) == "." + host; }; }, removeCookies: function(types, checkHost) { var cm = this.cm; var pm = this.pm; var cookies = cm.enumerator || this.arrayToEnumerator(cm.cookies); // Firefox 73+ while(cookies.hasMoreElements()) { let cookie = cookies.getNext() .QueryInterface(Components.interfaces.nsICookie); let cookieHost = cookie.host; if(checkHost && !checkHost(cookieHost)) continue; if(types) { // Trick for Firefox 42+, assumed pm.UNKNOWN_ACTION == 0 let permission = this.pmw.testPermission(this.getURI(cookieHost, "http"), this.permissionType) || this.pmw.testPermission(this.getURI(cookieHost, "https"), this.permissionType); if(types.indexOf(permission) == -1) continue; } if("testPermission" in pm) cm.remove(cookieHost, cookie.name, cookie.path, false, cookie.originAttributes || undefined); else // Firefox 71+ cm.remove(cookieHost, cookie.name, cookie.path, cookie.originAttributes || undefined); } }, updButtonState: function(permission) { var ttAdd = ""; if(permission === undefined) try { permission = this.getPermission(); } catch(e) { // See this.getURI() Components.utils.reportError(e); ttAdd = " \n" + e; permission = this.PERMISSIONS_ERROR; } var cp = this.cp; var key; switch(permission) { case cp.ACCESS_DEFAULT: key = "default"; if(this.options.showDefaultPolicy) switch(this.defaultPermission) { case cp.ACCESS_ALLOW: key = "defaultAllow"; break; case cp.ACCESS_DENY: key = "defaultDeny"; break; case cp.ACCESS_SESSION: key = "defaultAllowSession"; } break; case cp.ACCESS_ALLOW: key = "allow"; break; case cp.ACCESS_DENY: key = "deny"; break; case cp.ACCESS_SESSION: key = "allowSession"; break; case this.PERMISSIONS_NOT_SUPPORTED: key = "notAvailable"; break; case this.PERMISSIONS_ERROR: key = "error"; break; default: key = "unknown"; } var btn = this.button; if(btn.getAttribute("cb_cookies") == key) return; btn.disabled = permission == this.PERMISSIONS_NOT_SUPPORTED; btn.setAttribute("cb_cookies", key); btn.tooltipText = _localize(key + "Tooltiptext") + ttAdd; }, hasModifier: function(e) { return e.ctrlKey || e.shiftKey || e.altKey || e.metaKey; }, parseXULFromString: function(xul) { xul = xul.replace(/>\s+<"); try { return new DOMParser().parseFromString(xul, "application/xml").documentElement; } catch(e) { // See http://custombuttons.sourceforge.net/forum/viewtopic.php?f=5&t=3720 // + https://forum.mozilla-russia.org/viewtopic.php?pid=732243#p732243 var dummy = document.createElement("dummy"); dummy.innerHTML = xul.trimLeft(); return dummy.firstChild; } }, get storage() { delete this.storage; if(!("Services" in window)) // Firefox 3.6 and older return this.storage = Application.storage; // Simple replacement for Application.storage // See https://bugzilla.mozilla.org/show_bug.cgi?id=1090880 //var global = Components.utils.getGlobalForObject(Services); // Ensure, that we have global object (because window.Services may be overwritten) var global = Components.utils.import("resource://gre/modules/Services.jsm", {}); var ns = "_cbCookiesPermissionsStorage"; var storage = global[ns] || (global[ns] = Components.utils.getGlobalForObject(global).Object.create(null)); return this.storage = { get: function(key, defaultVal) { if(key in storage) return storage[key]; return defaultVal; }, set: function(key, val) { if(key === null) delete storage[key]; else storage[key] = val; } }; } }; //=================== // Styles // Used Fugue and Diagona icons (http://p.yusukekamiyamane.com/) // Styles can't override hardcoded icon var icon = this.icon || this.ownerDocument.getAnonymousElementByAttribute(this, "class", "toolbarbutton-icon"); if(icon) icon.src = ""; else this.image = ""; var cssStr = ('\ @namespace url("http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul");\n\ @-moz-document url("' + window.location.href + '") {\n\ %button% {\n\ list-style-image: url("") !important;\n\ -moz-image-region: rect(0, 16px, 16px, 0) !important;\n\ }\n\ %button%[cb_cookies="default"] { -moz-image-region: rect(0, 16px, 16px, 0) !important; }\n\ %button%[cb_cookies="allow"] { -moz-image-region: rect(0, 32px, 16px, 16px) !important; }\n\ %button%[cb_cookies="allowSession"] { -moz-image-region: rect(0, 48px, 16px, 32px) !important; }\n\ %button%[cb_cookies="deny"] { -moz-image-region: rect(0, 64px, 16px, 48px) !important; }\n\ %button%[cb_cookies="unknown"],\n\ %button%[cb_cookies="error"] { -moz-image-region: rect(0, 80px, 16px, 64px) !important; }\n\ %button%[cb_cookies="notAvailable"] { -moz-image-region: rect(0, 96px, 16px, 80px) !important; }\n\ %button%[cb_cookies="defaultAllow"] { -moz-image-region: rect(16px, 32px, 32px, 16px) !important; }\n\ %button%[cb_cookies="defaultAllowSession"] { -moz-image-region: rect(16px, 48px, 32px, 32px) !important; }\n\ %button%[cb_cookies="defaultDeny"] { -moz-image-region: rect(16px, 64px, 32px, 48px) !important; }\n\ /* "moveToStatusBar" option */\n\ %button%.custombuttons-insideStatusbarpanel {\n\ -moz-appearance: none !important;\n\ border: none !important;\n\ margin: 0 !important;\n\ padding: 0 !important;\n\ min-width: 0 !important;\n\ max-width: none !important;\n\ }\n\ %button%.custombuttons-insideStatusbarpanel > .toolbarbutton-icon {\n\ margin: 0 !important;\n\ padding: 0 !important;\n\ }\n\ %button%.custombuttons-insideStatusbarpanel > .toolbarbutton-text {\n\ display: none !important;\n\ }\n\ %button% .cbTempPermission {\n\ font-style: italic !important;\n\ /*-moz-padding-start: 0.7em !important;*/\n\ }\n\ }') .replace(/%button%/g, "#" + this.id); var cssURI = this.cssURI = Components.classes["@mozilla.org/network/io-service;1"] .getService(Components.interfaces.nsIIOService) .newURI("data:text/css," + encodeURIComponent(cssStr), null, null); var sss = this.sss = Components.classes["@mozilla.org/content/style-sheet-service;1"] .getService(Components.interfaces.nsIStyleSheetService); if(!sss.sheetRegistered(cssURI, sss.USER_SHEET)) sss.loadAndRegisterSheet(cssURI, sss.USER_SHEET); this.onDestroy = function(reason) { if(reason == "update" || reason == "delete") { let sss = this.sss; let cssURI = this.cssURI; if(sss.sheetRegistered(cssURI, sss.USER_SHEET)) sss.unregisterSheet(cssURI, sss.USER_SHEET); if(this.permissions.timer) this.permissions.timer.destroy(); } this.permissions.destroy(); }; this.permissions.init();