/* -*- indent-tabs-mode: nil; js-indent-level: 2 -*- */ /* 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/. */ /* This content script contains code that requires a tab browser. */ /* eslint-env mozilla/frame-script */ ChromeUtils.import("resource://gre/modules/XPCOMUtils.jsm"); ChromeUtils.import("resource://gre/modules/Services.jsm"); ChromeUtils.defineModuleGetter(this, "E10SUtils", "resource://gre/modules/E10SUtils.jsm"); ChromeUtils.defineModuleGetter(this, "BrowserUtils", "resource://gre/modules/BrowserUtils.jsm"); ChromeUtils.defineModuleGetter(this, "PageStyleHandler", "resource:///modules/PageStyleHandler.jsm"); ChromeUtils.import("resource://gre/modules/ActorManagerChild.jsm"); ActorManagerChild.attach(this, "browsers"); // TabChildGlobal var global = this; addMessageListener("Browser:HideSessionRestoreButton", function(message) { // Hide session restore button on about:home let doc = content.document; let container; if (doc.documentURI.toLowerCase() == "about:home" && (container = doc.getElementById("sessionRestoreContainer"))) { container.hidden = true; } }); XPCOMUtils.defineLazyProxy(this, "LightweightThemeChildHelper", "resource:///modules/LightweightThemeChildHelper.jsm"); XPCOMUtils.defineLazyProxy(this, "ManifestMessages", () => { let tmp = {}; ChromeUtils.import("resource://gre/modules/ManifestMessages.jsm", tmp); return new tmp.ManifestMessages(global); }); let themeablePagesWhitelist = new Set([ "about:home", "about:newtab", "about:welcome", ]); addEventListener("pageshow", function({ originalTarget }) { if (originalTarget.defaultView == content && themeablePagesWhitelist.has(content.document.documentURI)) { LightweightThemeChildHelper.listen(themeablePagesWhitelist); LightweightThemeChildHelper.update(chromeOuterWindowID, content); } }, false, true); var ContentSearchMediator = { whitelist: new Set([ "about:home", "about:newtab", "about:welcome", ]), init(chromeGlobal) { chromeGlobal.addEventListener("ContentSearchClient", this, true, true); addMessageListener("ContentSearch", this); this.init = null; }, handleEvent(event) { if (this._contentWhitelisted) { this._sendMsg(event.detail.type, event.detail.data); } }, receiveMessage(msg) { if (msg.data.type == "AddToWhitelist") { for (let uri of msg.data.data) { this.whitelist.add(uri); } this._sendMsg("AddToWhitelistAck"); return; } if (this._contentWhitelisted) { this._fireEvent(msg.data.type, msg.data.data); } }, get _contentWhitelisted() { return this.whitelist.has(content.document.documentURI); }, _sendMsg(type, data = null) { sendAsyncMessage("ContentSearch", { type, data, }); }, _fireEvent(type, data = null) { let event = Cu.cloneInto({ detail: { type, data, }, }, content); content.dispatchEvent(new content.CustomEvent("ContentSearchService", event)); }, }; ContentSearchMediator.init(this); addMessageListener("PageStyle:Switch", PageStyleHandler); addMessageListener("PageStyle:Disable", PageStyleHandler); addEventListener("pageshow", PageStyleHandler); // Keep a reference to the translation content handler to avoid it it being GC'ed. var trHandler = null; if (Services.prefs.getBoolPref("browser.translation.detectLanguage")) { ChromeUtils.import("resource:///modules/translation/TranslationContentHandler.jsm"); trHandler = new TranslationContentHandler(global, docShell); } function gKeywordURIFixup(fixupInfo) { fixupInfo.QueryInterface(Ci.nsIURIFixupInfo); if (!fixupInfo.consumer) { return; } // Ignore info from other docshells let parent = fixupInfo.consumer.QueryInterface(Ci.nsIDocShellTreeItem).sameTypeRootTreeItem; if (parent != docShell) return; let data = {}; for (let f of Object.keys(fixupInfo)) { if (f == "consumer" || typeof fixupInfo[f] == "function") continue; if (fixupInfo[f] && fixupInfo[f] instanceof Ci.nsIURI) { data[f] = fixupInfo[f].spec; } else { data[f] = fixupInfo[f]; } } sendAsyncMessage("Browser:URIFixup", data); } Services.obs.addObserver(gKeywordURIFixup, "keyword-uri-fixup"); addEventListener("unload", () => { Services.obs.removeObserver(gKeywordURIFixup, "keyword-uri-fixup"); }, false); var WebBrowserChrome = { onBeforeLinkTraversal(originalTarget, linkURI, linkNode, isAppTab) { return BrowserUtils.onBeforeLinkTraversal(originalTarget, linkURI, linkNode, isAppTab); }, // Check whether this URI should load in the current process shouldLoadURI(aDocShell, aURI, aReferrer, aHasPostData, aTriggeringPrincipal) { if (!E10SUtils.shouldLoadURI(aDocShell, aURI, aReferrer, aHasPostData)) { E10SUtils.redirectLoad(aDocShell, aURI, aReferrer, aTriggeringPrincipal, false); return false; } return true; }, shouldLoadURIInThisProcess(aURI) { return E10SUtils.shouldLoadURIInThisProcess(aURI); }, // Try to reload the currently active or currently loading page in a new process. reloadInFreshProcess(aDocShell, aURI, aReferrer, aTriggeringPrincipal, aLoadFlags) { E10SUtils.redirectLoad(aDocShell, aURI, aReferrer, aTriggeringPrincipal, true, aLoadFlags); return true; } }; if (Services.appinfo.processType == Services.appinfo.PROCESS_TYPE_CONTENT) { let tabchild = docShell.QueryInterface(Ci.nsIInterfaceRequestor) .getInterface(Ci.nsITabChild); tabchild.webBrowserChrome = WebBrowserChrome; } var DOMFullscreenHandler = { init() { addMessageListener("DOMFullscreen:Entered", this); addMessageListener("DOMFullscreen:CleanUp", this); addEventListener("MozDOMFullscreen:Request", this); addEventListener("MozDOMFullscreen:Entered", this); addEventListener("MozDOMFullscreen:NewOrigin", this); addEventListener("MozDOMFullscreen:Exit", this); addEventListener("MozDOMFullscreen:Exited", this); this.init = null; }, receiveMessage(aMessage) { let windowUtils = content && content.windowUtils; switch (aMessage.name) { case "DOMFullscreen:Entered": { this._lastTransactionId = windowUtils.lastTransactionId; if (!windowUtils.handleFullscreenRequests() && !content.document.fullscreenElement) { // If we don't actually have any pending fullscreen request // to handle, neither we have been in fullscreen, tell the // parent to just exit. sendAsyncMessage("DOMFullscreen:Exit"); } break; } case "DOMFullscreen:CleanUp": { // If we've exited fullscreen at this point, no need to record // transaction id or call exit fullscreen. This is especially // important for non-e10s, since in that case, it is possible // that no more paint would be triggered after this point. if (content.document.fullscreenElement && windowUtils) { this._lastTransactionId = windowUtils.lastTransactionId; windowUtils.exitFullscreen(); } break; } } }, handleEvent(aEvent) { switch (aEvent.type) { case "MozDOMFullscreen:Request": { sendAsyncMessage("DOMFullscreen:Request"); break; } case "MozDOMFullscreen:NewOrigin": { sendAsyncMessage("DOMFullscreen:NewOrigin", { originNoSuffix: aEvent.target.nodePrincipal.originNoSuffix, }); break; } case "MozDOMFullscreen:Exit": { sendAsyncMessage("DOMFullscreen:Exit"); break; } case "MozDOMFullscreen:Entered": case "MozDOMFullscreen:Exited": { addEventListener("MozAfterPaint", this); if (!content || !content.document.fullscreenElement) { // If we receive any fullscreen change event, and find we are // actually not in fullscreen, also ask the parent to exit to // ensure that the parent always exits fullscreen when we do. sendAsyncMessage("DOMFullscreen:Exit"); } break; } case "MozAfterPaint": { // Only send Painted signal after we actually finish painting // the transition for the fullscreen change. // Note that this._lastTransactionId is not set when in non-e10s // mode, so we need to check that explicitly. if (!this._lastTransactionId || aEvent.transactionId > this._lastTransactionId) { removeEventListener("MozAfterPaint", this); sendAsyncMessage("DOMFullscreen:Painted"); } break; } } } }; DOMFullscreenHandler.init(); Services.obs.notifyObservers(this, "tab-content-frameloader-created"); // Remove this once bug 1397365 is fixed. addEventListener("MozAfterPaint", function onFirstNonBlankPaint() { if (content.document.documentURI == "about:blank" && !content.opener) return; removeEventListener("MozAfterPaint", onFirstNonBlankPaint); sendAsyncMessage("Browser:FirstNonBlankPaint"); }); addMessageListener("DOM:WebManifest:hasManifestLink", ManifestMessages); addMessageListener("DOM:ManifestObtainer:Obtain", ManifestMessages); addMessageListener("DOM:Manifest:FireAppInstalledEvent", ManifestMessages); addMessageListener("DOM:WebManifest:fetchIcon", ManifestMessages);