/* 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/. */ "use strict"; var EXPORTED_SYMBOLS = ["PermissionUI"]; /** * PermissionUI is responsible for exposing both a prototype * PermissionPrompt that can be used by arbitrary browser * components and add-ons, but also hosts the implementations of * built-in permission prompts. * * If you're developing a feature that requires web content to ask * for special permissions from the user, this module is for you. * * Suppose a system add-on wants to add a new prompt for a new request * for getting more low-level access to the user's sound card, and the * permission request is coming up from content by way of the * nsContentPermissionHelper. The system add-on could then do the following: * * Cu.import("resource://gre/modules/Integration.jsm"); * Cu.import("resource:///modules/PermissionUI.jsm"); * * const SoundCardIntegration = (base) => ({ * __proto__: base, * createPermissionPrompt(type, request) { * if (type != "sound-api") { * return super.createPermissionPrompt(...arguments); * } * * return { * __proto__: PermissionUI.PermissionPromptForRequestPrototype, * get permissionKey() { * return "sound-permission"; * } * // etc - see the documentation for PermissionPrompt for * // a better idea of what things one can and should override. * } * }, * }); * * // Add-on startup: * Integration.contentPermission.register(SoundCardIntegration); * // ... * // Add-on shutdown: * Integration.contentPermission.unregister(SoundCardIntegration); * * Note that PermissionPromptForRequestPrototype must be used as the * prototype, since the prompt is wrapping an nsIContentPermissionRequest, * and going through nsIContentPermissionPrompt. * * It is, however, possible to take advantage of PermissionPrompt without * having to go through nsIContentPermissionPrompt or with a * nsIContentPermissionRequest. The PermissionPromptPrototype can be * imported, subclassed, and have prompt() called directly, without * the caller having called into createPermissionPrompt. */ const { XPCOMUtils } = ChromeUtils.import( "resource://gre/modules/XPCOMUtils.jsm" ); ChromeUtils.defineModuleGetter( this, "Services", "resource://gre/modules/Services.jsm" ); ChromeUtils.defineModuleGetter( this, "SitePermissions", "resource:///modules/SitePermissions.jsm" ); ChromeUtils.defineModuleGetter( this, "PrivateBrowsingUtils", "resource://gre/modules/PrivateBrowsingUtils.jsm" ); ChromeUtils.defineModuleGetter( this, "PermissionUITelemetry", "resource:///modules/PermissionUITelemetry.jsm" ); XPCOMUtils.defineLazyServiceGetter( this, "IDNService", "@mozilla.org/network/idn-service;1", "nsIIDNService" ); XPCOMUtils.defineLazyServiceGetter( this, "ContentPrefService2", "@mozilla.org/content-pref/service;1", "nsIContentPrefService2" ); XPCOMUtils.defineLazyGetter(this, "gBrowserBundle", function() { return Services.strings.createBundle( "chrome://browser/locale/browser.properties" ); }); XPCOMUtils.defineLazyPreferenceGetter( this, "animationsEnabled", "toolkit.cosmeticAnimations.enabled" ); XPCOMUtils.defineLazyPreferenceGetter( this, "postPromptAnimationEnabled", "permissions.postPrompt.animate" ); var PermissionUI = {}; /** * PermissionPromptPrototype should be subclassed by callers that * want to display prompts to the user. See each method and property * below for guidance on what to override. * * Note that if you're creating a prompt for an * nsIContentPermissionRequest, you'll want to subclass * PermissionPromptForRequestPrototype instead. */ var PermissionPromptPrototype = { /** * Returns the associated for the request. This should * work for the e10s and non-e10s case. * * Subclasses must override this. * * @return {} */ get browser() { throw new Error("Not implemented."); }, /** * Returns the nsIPrincipal associated with the request. * * Subclasses must override this. * * @return {nsIPrincipal} */ get principal() { throw new Error("Not implemented."); }, /** * Provides the preferred name to use in the permission popups, * based on the principal URI (the URI.hostPort for any URI scheme * besides the moz-extension one which should default to the * extension name). */ get principalName() { if (this.principal.addonPolicy) { return this.principal.addonPolicy.name; } return this.principal.URI.hostPort; }, /** * If the nsIPermissionManager is being queried and written * to for this permission request, set this to the key to be * used. If this is undefined, no integration with temporary * permissions infrastructure will be provided. * * Note that if a permission is set, in any follow-up * prompting within the expiry window of that permission, * the prompt will be skipped and the allow or deny choice * will be selected automatically. */ get permissionKey() { return undefined; }, /** * A string that needs to be set to include this prompt in * experimental event telemetry collection. * * This needs to conform to event telemetry string rules, * i.e. it needs to be an alphabetic string under 20 characters. */ get permissionTelemetryKey() { return undefined; }, /** * If true, user permissions will be read from and written to. * When this is false, we still provide integration with * infrastructure such as temporary permissions. permissionKey should * still return a valid name in those cases for that integration to work. */ get usePermissionManager() { return true; }, /** * These are the options that will be passed to the * PopupNotification when it is shown. See the documentation * for PopupNotification for more details. * * Note that prompt() will automatically set displayURI to * be the URI of the requesting pricipal, unless the displayURI is exactly * set to false. */ get popupOptions() { return {}; }, /** * If true, automatically denied permission requests will * spawn a "post-prompt" that allows the user to correct the * automatic denial by giving permanent permission access to * the site. * * Note that if this function returns true, the permissionKey * and postPromptActions attributes must be implemented. */ get postPromptEnabled() { return false; }, /** * If true, the prompt will be cancelled automatically unless * request.isHandlingUserInput is true. */ get requiresUserInput() { return false; }, /** * PopupNotification requires a unique ID to open the notification. * You must return a unique ID string here, for which PopupNotification * will then create a node with the ID * "-notification". * * If there's a custom you're hoping to show, * then you need to make sure its ID has the "-notification" suffix, * and then return the prefix here. * * See PopupNotification.jsm for more details. * * @return {string} * The unique ID that will be used to as the * "-notification" ID for the * to use or create. */ get notificationID() { throw new Error("Not implemented."); }, /** * The ID of the element to anchor the PopupNotification to. * * @return {string} */ get anchorID() { return "default-notification-icon"; }, /** * The message to show to the user in the PopupNotification. A string * with "<>" as a placeholder that is usually replaced by an addon name * or a host name, formatted in bold, to describe the permission that is being requested. * * Subclasses must override this. * * @return {string} */ get message() { throw new Error("Not implemented."); }, /** * This will be called if the request is to be cancelled. * * Subclasses only need to override this if they provide a * permissionKey. */ cancel() { throw new Error("Not implemented."); }, /** * This will be called if the request is to be allowed. * * Subclasses only need to override this if they provide a * permissionKey. */ allow() { throw new Error("Not implemented."); }, /** * The actions that will be displayed in the PopupNotification * via a dropdown menu. The first item in this array will be * the default selection. Each action is an Object with the * following properties: * * label (string): * The label that will be displayed for this choice. * accessKey (string): * The access key character that will be used for this choice. * action (SitePermissions state) * The action that will be associated with this choice. * This should be either SitePermissions.ALLOW or SitePermissions.BLOCK. * scope (SitePermissions scope) * The scope of the associated action (e.g. SitePermissions.SCOPE_PERSISTENT) * * callback (function, optional) * A callback function that will fire if the user makes this choice, with * a single parameter, state. State is an Object that contains the property * checkboxChecked, which identifies whether the checkbox to remember this * decision was checked. */ get promptActions() { return []; }, /** * The actions that will be displayed in the PopupNotification * for post-prompt notifications via a dropdown menu. * The first item in this array will be the default selection. * Each action is an Object with the following properties: * * label (string): * The label that will be displayed for this choice. * accessKey (string): * The access key character that will be used for this choice. * action (SitePermissions state) * The action that will be associated with this choice. * This should be either SitePermissions.ALLOW or SitePermissions.BLOCK. * Note that the scope of this action will always be persistent. * * callback (function, optional) * A callback function that will fire if the user makes this choice. */ get postPromptActions() { return null; }, /** * If the prompt will be shown to the user, this callback will * be called just before. Subclasses may want to override this * in order to, for example, bump a counter Telemetry probe for * how often a particular permission request is seen. * * If this returns false, it cancels the process of showing the prompt. In * that case, it is the responsibility of the onBeforeShow() implementation * to ensure that allow() or cancel() are called on the object appropriately. */ onBeforeShow() { return true; }, /** * If the prompt was shown to the user, this callback will be called just * after it's been shown. */ onShown() {}, /** * If the prompt was shown to the user, this callback will be called just * after it's been hidden. */ onAfterShow() {}, /** * Will determine if a prompt should be shown to the user, and if so, * will show it. * * If a permissionKey is defined prompt() might automatically * allow or cancel itself based on the user's current * permission settings without displaying the prompt. * * If the permission is not already set and the that the request * is associated with does not belong to a browser window with the * PopupNotifications global set, the prompt request is ignored. */ prompt() { // We ignore requests from non-nsIStandardURLs let requestingURI = this.principal.URI; if (!(requestingURI instanceof Ci.nsIStandardURL)) { return; } if (this.usePermissionManager && this.permissionKey) { // If we're reading and setting permissions, then we need // to check to see if we already have a permission setting // for this particular principal. let { state } = SitePermissions.getForPrincipal( this.principal, this.permissionKey, this.browser ); if (state == SitePermissions.BLOCK) { // If this block was done based on a global user setting, we want to show // a post prompt to give the user some more granular control without // annoying them too much. if ( this.postPromptEnabled && SitePermissions.getDefault(this.permissionKey) == SitePermissions.BLOCK ) { this.postPrompt(); } this.cancel(); return; } if (state == SitePermissions.ALLOW) { this.allow(); return; } // Tell the browser to refresh the identity block display in case there // are expired permission states. this.browser.dispatchEvent( new this.browser.ownerGlobal.CustomEvent("PermissionStateChange") ); } else if (this.permissionKey) { // If we're reading a permission which already has a temporary value, // see if we can use the temporary value. let { state } = SitePermissions.getForPrincipal( null, this.permissionKey, this.browser ); if (state == SitePermissions.BLOCK) { this.cancel(); return; } } if (this.requiresUserInput && !this.request.isHandlingUserInput) { if (this.postPromptEnabled) { this.postPrompt(); } this.cancel(); return; } let chromeWin = this.browser.ownerGlobal; if (!chromeWin.PopupNotifications) { this.cancel(); return; } this._buttonAction = null; // Transform the PermissionPrompt actions into PopupNotification actions. let popupNotificationActions = []; for (let promptAction of this.promptActions) { let action = { label: promptAction.label, accessKey: promptAction.accessKey, callback: state => { if (promptAction.callback) { promptAction.callback(); } if (this.usePermissionManager && this.permissionKey) { if ( (state && state.checkboxChecked && state.source != "esc-press") || promptAction.scope == SitePermissions.SCOPE_PERSISTENT ) { // Permanently store permission. let scope = SitePermissions.SCOPE_PERSISTENT; // Only remember permission for session if in PB mode. if (PrivateBrowsingUtils.isBrowserPrivate(this.browser)) { scope = SitePermissions.SCOPE_SESSION; } SitePermissions.setForPrincipal( this.principal, this.permissionKey, promptAction.action, scope ); } else if (promptAction.action == SitePermissions.BLOCK) { // Temporarily store BLOCK permissions only // SitePermissions does not consider subframes when storing temporary // permissions on a tab, thus storing ALLOW could be exploited. SitePermissions.setForPrincipal( this.principal, this.permissionKey, promptAction.action, SitePermissions.SCOPE_TEMPORARY, this.browser ); } // Grant permission if action is ALLOW. // Record buttonAction for telemetry. if (promptAction.action == SitePermissions.ALLOW) { this._buttonAction = "accept"; this.allow(); } else { if (promptAction.scope == SitePermissions.SCOPE_PERSISTENT) { this._buttonAction = "never"; } else { this._buttonAction = "deny"; } this.cancel(); } } else if (this.permissionKey) { // TODO: Add support for permitTemporaryAllow if (promptAction.action == SitePermissions.BLOCK) { // Temporarily store BLOCK permissions. // We don't consider subframes when storing temporary // permissions on a tab, thus storing ALLOW could be exploited. SitePermissions.setForPrincipal( null, this.permissionKey, promptAction.action, SitePermissions.SCOPE_TEMPORARY, this.browser ); } } }, }; if (promptAction.dismiss) { action.dismiss = promptAction.dismiss; } popupNotificationActions.push(action); } this._showNotification(popupNotificationActions); }, postPrompt() { let browser = this.browser; let principal = this.principal; let chromeWin = browser.ownerGlobal; if (!chromeWin.PopupNotifications) { return; } if (!this.permissionKey) { throw new Error("permissionKey is required to show a post-prompt"); } if (!this.postPromptActions) { throw new Error("postPromptActions are required to show a post-prompt"); } // Transform the PermissionPrompt actions into PopupNotification actions. let popupNotificationActions = []; for (let promptAction of this.postPromptActions) { let action = { label: promptAction.label, accessKey: promptAction.accessKey, callback: state => { if (promptAction.callback) { promptAction.callback(); } // Post-prompt permissions are stored permanently by default. // Since we can not reply to the original permission request anymore, // the page will need to listen for permission changes which are triggered // by permanent entries in the permission manager. let scope = SitePermissions.SCOPE_PERSISTENT; // Only remember permission for session if in PB mode. if (PrivateBrowsingUtils.isBrowserPrivate(browser)) { scope = SitePermissions.SCOPE_SESSION; } SitePermissions.setForPrincipal( principal, this.permissionKey, promptAction.action, scope ); }, }; popupNotificationActions.push(action); } if (animationsEnabled && postPromptAnimationEnabled) { let anchor = chromeWin.document.getElementById(this.anchorID); // Only show the animation on the first request, not after e.g. tab switching. anchor.addEventListener( "animationend", () => anchor.removeAttribute("animate"), { once: true } ); anchor.setAttribute("animate", "true"); } this._showNotification(popupNotificationActions, true); }, _showNotification(actions, postPrompt = false) { let chromeWin = this.browser.ownerGlobal; let mainAction = actions.length ? actions[0] : null; let secondaryActions = actions.splice(1); let options = this.popupOptions; let telemetryData = null; if (this.request && this.permissionTelemetryKey) { telemetryData = { permissionTelemetryKey: this.permissionTelemetryKey, permissionKey: this.permissionKey, principal: this.principal, documentDOMContentLoadedTimestamp: this.request .documentDOMContentLoadedTimestamp, isHandlingUserInput: this.request.isHandlingUserInput, userHadInteractedWithDocument: this.request .userHadInteractedWithDocument, }; } if (!options.hasOwnProperty("displayURI") || options.displayURI) { options.displayURI = this.principal.URI; } if (!postPrompt) { // Permission prompts are always persistent; the close button is controlled by a pref. options.persistent = true; options.hideClose = true; } options.eventCallback = (topic, nextRemovalReason, isCancel) => { // When the docshell of the browser is aboout to be swapped to another one, // the "swapping" event is called. Returning true causes the notification // to be moved to the new browser. if (topic == "swapping") { return true; } // The prompt has been shown, notify the PermissionUI. // onShown() is currently not called for post-prompts, // because there is no prompt that would make use of this. // You can remove this restriction if you need it, but be // mindful of other consumers. if (topic == "shown" && !postPrompt) { this.onShown(); } // The prompt has been removed, notify the PermissionUI. // onAfterShow() is currently not called for post-prompts, // because there is no prompt that would make use of this. // You can remove this restriction if you need it, but be // mindful of other consumers. if (topic == "removed" && !postPrompt) { if (telemetryData) { PermissionUITelemetry.onRemoved( telemetryData, this._buttonAction, nextRemovalReason ); } if (isCancel) { this.cancel(); } this.onAfterShow(); } return false; }; // Post-prompts show up as dismissed. options.dismissed = postPrompt; // onBeforeShow() is currently not called for post-prompts, // because there is no prompt that would make use of this. // You can remove this restriction if you need it, but be // mindful of other consumers. if (postPrompt || this.onBeforeShow() !== false) { chromeWin.PopupNotifications.show( this.browser, this.notificationID, this.message, this.anchorID, mainAction, secondaryActions, options ); if (telemetryData) { PermissionUITelemetry.onShow(telemetryData); } } }, }; PermissionUI.PermissionPromptPrototype = PermissionPromptPrototype; /** * A subclass of PermissionPromptPrototype that assumes * that this.request is an nsIContentPermissionRequest * and fills in some of the required properties on the * PermissionPrompt. For callers that are wrapping an * nsIContentPermissionRequest, this should be subclassed * rather than PermissionPromptPrototype. */ var PermissionPromptForRequestPrototype = { __proto__: PermissionPromptPrototype, get browser() { // In the e10s-case, the will be at request.element. // In the single-process case, we have to use some XPCOM incantations // to resolve to the . if (this.request.element) { return this.request.element; } return this.request.window.docShell.chromeEventHandler; }, get principal() { return this.request.principal; }, cancel() { this.request.cancel(); }, allow(choices) { this.request.allow(choices); }, }; PermissionUI.PermissionPromptForRequestPrototype = PermissionPromptForRequestPrototype; /** * Creates a PermissionPrompt for a nsIContentPermissionRequest for * the GeoLocation API. * * @param request (nsIContentPermissionRequest) * The request for a permission from content. */ function GeolocationPermissionPrompt(request) { this.request = request; } GeolocationPermissionPrompt.prototype = { __proto__: PermissionPromptForRequestPrototype, get permissionKey() { return "geo"; }, get permissionTelemetryKey() { return "geo"; }, get popupOptions() { let pref = "browser.geolocation.warning.infoURL"; let options = { learnMoreURL: Services.urlFormatter.formatURLPref(pref), displayURI: false, name: this.principalName, }; if (this.principal.URI.schemeIs("file")) { options.checkbox = { show: false }; } else { // Don't offer "always remember" action in PB mode options.checkbox = { show: !PrivateBrowsingUtils.isWindowPrivate(this.browser.ownerGlobal), }; } if (options.checkbox.show) { options.checkbox.label = gBrowserBundle.GetStringFromName( "geolocation.remember" ); } return options; }, get notificationID() { return "geolocation"; }, get anchorID() { return "geo-notification-icon"; }, get message() { if (this.principal.URI.schemeIs("file")) { return gBrowserBundle.GetStringFromName("geolocation.shareWithFile3"); } return gBrowserBundle.formatStringFromName("geolocation.shareWithSite3", [ "<>", ]); }, get promptActions() { return [ { label: gBrowserBundle.GetStringFromName("geolocation.allowLocation"), accessKey: gBrowserBundle.GetStringFromName( "geolocation.allowLocation.accesskey" ), action: SitePermissions.ALLOW, }, { label: gBrowserBundle.GetStringFromName( "geolocation.dontAllowLocation" ), accessKey: gBrowserBundle.GetStringFromName( "geolocation.dontAllowLocation.accesskey" ), action: SitePermissions.BLOCK, }, ]; }, _updateGeoSharing(state) { let gBrowser = this.browser.ownerGlobal.gBrowser; if (gBrowser == null) { return; } gBrowser.updateBrowserSharing(this.browser, { geo: state }); if (!state) { return; } let host; try { host = this.browser.currentURI.host; } catch (e) { return; } if (host == null || host == "") { return; } ContentPrefService2.set( this.browser.currentURI.host, "permissions.geoLocation.lastAccess", new Date().toString(), this.browser.loadContext ); }, allow(...args) { this._updateGeoSharing(true); PermissionPromptForRequestPrototype.allow.apply(this, args); }, cancel(...args) { this._updateGeoSharing(false); PermissionPromptForRequestPrototype.cancel.apply(this, args); }, }; PermissionUI.GeolocationPermissionPrompt = GeolocationPermissionPrompt; /** * Creates a PermissionPrompt for a nsIContentPermissionRequest for * the Desktop Notification API. * * @param request (nsIContentPermissionRequest) * The request for a permission from content. * @return {PermissionPrompt} (see documentation in header) */ function DesktopNotificationPermissionPrompt(request) { this.request = request; XPCOMUtils.defineLazyPreferenceGetter( this, "requiresUserInput", "dom.webnotifications.requireuserinteraction" ); XPCOMUtils.defineLazyPreferenceGetter( this, "postPromptEnabled", "permissions.desktop-notification.postPrompt.enabled" ); } DesktopNotificationPermissionPrompt.prototype = { __proto__: PermissionPromptForRequestPrototype, get permissionKey() { return "desktop-notification"; }, get permissionTelemetryKey() { return "notifications"; }, get popupOptions() { let learnMoreURL = Services.urlFormatter.formatURLPref("app.support.baseURL") + "push"; return { learnMoreURL, displayURI: false, name: this.principalName, }; }, get notificationID() { return "web-notifications"; }, get anchorID() { return "web-notifications-notification-icon"; }, get message() { return gBrowserBundle.formatStringFromName( "webNotifications.receiveFromSite2", ["<>"] ); }, get promptActions() { let actions = [ { label: gBrowserBundle.GetStringFromName("webNotifications.allow"), accessKey: gBrowserBundle.GetStringFromName( "webNotifications.allow.accesskey" ), action: SitePermissions.ALLOW, scope: SitePermissions.SCOPE_PERSISTENT, }, { label: gBrowserBundle.GetStringFromName("webNotifications.notNow"), accessKey: gBrowserBundle.GetStringFromName( "webNotifications.notNow.accesskey" ), action: SitePermissions.BLOCK, }, ]; if (!PrivateBrowsingUtils.isBrowserPrivate(this.browser)) { actions.push({ label: gBrowserBundle.GetStringFromName("webNotifications.never"), accessKey: gBrowserBundle.GetStringFromName( "webNotifications.never.accesskey" ), action: SitePermissions.BLOCK, scope: SitePermissions.SCOPE_PERSISTENT, }); } return actions; }, get postPromptActions() { return [ { label: gBrowserBundle.GetStringFromName("webNotifications.allow"), accessKey: gBrowserBundle.GetStringFromName( "webNotifications.allow.accesskey" ), action: SitePermissions.ALLOW, }, { label: gBrowserBundle.GetStringFromName("webNotifications.never"), accessKey: gBrowserBundle.GetStringFromName( "webNotifications.never.accesskey" ), action: SitePermissions.BLOCK, }, ]; }, }; PermissionUI.DesktopNotificationPermissionPrompt = DesktopNotificationPermissionPrompt; /** * Creates a PermissionPrompt for a nsIContentPermissionRequest for * the persistent-storage API. * * @param request (nsIContentPermissionRequest) * The request for a permission from content. */ function PersistentStoragePermissionPrompt(request) { this.request = request; } PersistentStoragePermissionPrompt.prototype = { __proto__: PermissionPromptForRequestPrototype, get permissionKey() { return "persistent-storage"; }, get popupOptions() { let learnMoreURL = Services.urlFormatter.formatURLPref("app.support.baseURL") + "storage-permissions"; return { learnMoreURL, displayURI: false, name: this.principalName, }; }, get notificationID() { return "persistent-storage"; }, get anchorID() { return "persistent-storage-notification-icon"; }, get message() { return gBrowserBundle.formatStringFromName( "persistentStorage.allowWithSite", ["<>"] ); }, get promptActions() { return [ { label: gBrowserBundle.GetStringFromName("persistentStorage.allow"), accessKey: gBrowserBundle.GetStringFromName( "persistentStorage.allow.accesskey" ), action: Ci.nsIPermissionManager.ALLOW_ACTION, scope: SitePermissions.SCOPE_PERSISTENT, }, { label: gBrowserBundle.GetStringFromName( "persistentStorage.notNow.label" ), accessKey: gBrowserBundle.GetStringFromName( "persistentStorage.notNow.accesskey" ), action: Ci.nsIPermissionManager.DENY_ACTION, }, { label: gBrowserBundle.GetStringFromName( "persistentStorage.neverAllow.label" ), accessKey: gBrowserBundle.GetStringFromName( "persistentStorage.neverAllow.accesskey" ), action: SitePermissions.BLOCK, scope: SitePermissions.SCOPE_PERSISTENT, }, ]; }, }; PermissionUI.PersistentStoragePermissionPrompt = PersistentStoragePermissionPrompt; /** * Creates a PermissionPrompt for a nsIContentPermissionRequest for * the WebMIDI API. * * @param request (nsIContentPermissionRequest) * The request for a permission from content. */ function MIDIPermissionPrompt(request) { this.request = request; let types = request.types.QueryInterface(Ci.nsIArray); let perm = types.queryElementAt(0, Ci.nsIContentPermissionType); this.isSysexPerm = perm.options.length > 0 && perm.options.queryElementAt(0, Ci.nsISupportsString) == "sysex"; this.permName = "midi"; if (this.isSysexPerm) { this.permName = "midi-sysex"; } } MIDIPermissionPrompt.prototype = { __proto__: PermissionPromptForRequestPrototype, get permissionKey() { return this.permName; }, get popupOptions() { // TODO (bug 1433235) We need a security/permissions explanation URL for this let options = { displayURI: false, name: this.principalName, }; if (this.principal.URI.schemeIs("file")) { options.checkbox = { show: false }; } else { // Don't offer "always remember" action in PB mode options.checkbox = { show: !PrivateBrowsingUtils.isWindowPrivate(this.browser.ownerGlobal), }; } if (options.checkbox.show) { options.checkbox.label = gBrowserBundle.GetStringFromName( "midi.remember" ); } return options; }, get notificationID() { return "midi"; }, get anchorID() { return "midi-notification-icon"; }, get message() { let message; if (this.principal.URI.schemeIs("file")) { if (this.isSysexPerm) { message = gBrowserBundle.formatStringFromName( "midi.shareSysexWithFile.message" ); } else { message = gBrowserBundle.formatStringFromName( "midi.shareWithFile.message" ); } } else if (this.isSysexPerm) { message = gBrowserBundle.formatStringFromName( "midi.shareSysexWithSite.message", ["<>"] ); } else { message = gBrowserBundle.formatStringFromName( "midi.shareWithSite.message", ["<>"] ); } return message; }, get promptActions() { return [ { label: gBrowserBundle.GetStringFromName("midi.Allow.label"), accessKey: gBrowserBundle.GetStringFromName("midi.Allow.accesskey"), action: Ci.nsIPermissionManager.ALLOW_ACTION, }, { label: gBrowserBundle.GetStringFromName("midi.DontAllow.label"), accessKey: gBrowserBundle.GetStringFromName("midi.DontAllow.accesskey"), action: Ci.nsIPermissionManager.DENY_ACTION, }, ]; }, }; PermissionUI.MIDIPermissionPrompt = MIDIPermissionPrompt; function StorageAccessPermissionPrompt(request) { this.request = request; } StorageAccessPermissionPrompt.prototype = { __proto__: PermissionPromptForRequestPrototype, get usePermissionManager() { return false; }, get permissionKey() { // Make sure this name is unique per each third-party tracker return "storage-access-" + this.principal.origin; }, prettifyHostPort(uri) { try { uri = Services.uriFixup.createExposableURI(uri); } catch (e) { // ignore, since we can't do anything better } let host = IDNService.convertToDisplayIDN(uri.host, {}); if (uri.port != -1) { host += `:${uri.port}`; } return host; }, get popupOptions() { return { displayURI: false, name: this.prettifyHostPort(this.principal.URI), secondName: this.prettifyHostPort(this.topLevelPrincipal.URI), escAction: "buttoncommand", }; }, onShown() { let document = this.browser.ownerDocument; let label = gBrowserBundle.formatStringFromName( "storageAccess.description.label", [this.prettifyHostPort(this.request.principal.URI), "<>"] ); let parts = label.split("<>"); if (parts.length == 1) { parts.push(""); } let map = { "storage-access-perm-label": parts[0], "storage-access-perm-learnmore": gBrowserBundle.GetStringFromName( "storageAccess.description.learnmore" ), "storage-access-perm-endlabel": parts[1], }; for (let id in map) { let str = map[id]; document.getElementById(id).textContent = str; } let learnMoreURL = Services.urlFormatter.formatURLPref("app.support.baseURL") + "third-party-cookies"; document.getElementById( "storage-access-perm-learnmore" ).href = learnMoreURL; }, get notificationID() { return "storage-access"; }, get anchorID() { return "storage-access-notification-icon"; }, get message() { return gBrowserBundle.formatStringFromName("storageAccess.message", [ "<>", "{}", ]); }, get promptActions() { let self = this; return [ { label: gBrowserBundle.GetStringFromName( "storageAccess.DontAllow.label" ), accessKey: gBrowserBundle.GetStringFromName( "storageAccess.DontAllow.accesskey" ), action: Ci.nsIPermissionManager.DENY_ACTION, callback(state) { self.cancel(); }, }, { label: gBrowserBundle.GetStringFromName("storageAccess.Allow.label"), accessKey: gBrowserBundle.GetStringFromName( "storageAccess.Allow.accesskey" ), action: Ci.nsIPermissionManager.ALLOW_ACTION, callback(state) { self.allow({ "storage-access": "allow" }); }, }, { label: gBrowserBundle.GetStringFromName( "storageAccess.AllowOnAnySite.label" ), accessKey: gBrowserBundle.GetStringFromName( "storageAccess.AllowOnAnySite.accesskey" ), action: Ci.nsIPermissionManager.ALLOW_ACTION, callback(state) { self.allow({ "storage-access": "allow-on-any-site" }); }, }, ]; }, get topLevelPrincipal() { return this.request.topLevelPrincipal; }, }; PermissionUI.StorageAccessPermissionPrompt = StorageAccessPermissionPrompt;