| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  | /* 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"; | 
					
						
							|  |  |  | const Ci = Components.interfaces; | 
					
						
							|  |  |  | const Cr = Components.results; | 
					
						
							|  |  |  | const Cu = Components.utils; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const {Helper} = ChromeUtils.import('chrome://juggler/content/Helper.js'); | 
					
						
							|  |  |  | const {SimpleChannel} = ChromeUtils.import('chrome://juggler/content/SimpleChannel.js'); | 
					
						
							|  |  |  | const {Runtime} = ChromeUtils.import('chrome://juggler/content/content/Runtime.js'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const helper = new Helper(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FrameTree { | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |   constructor(rootBrowsingContext) { | 
					
						
							| 
									
										
										
										
											2023-01-23 11:29:48 -08:00
										 |  |  |     helper.decorateAsEventEmitter(this); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     this._rootBrowsingContext = rootBrowsingContext; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     this._browsingContextGroup = rootBrowsingContext.group; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     if (!this._browsingContextGroup.__jugglerFrameTrees) | 
					
						
							|  |  |  |       this._browsingContextGroup.__jugglerFrameTrees = new Set(); | 
					
						
							|  |  |  |     this._browsingContextGroup.__jugglerFrameTrees.add(this); | 
					
						
							| 
									
										
										
										
											2021-04-29 22:09:10 +00:00
										 |  |  |     this._isolatedWorlds = new Map(); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 16:33:50 -07:00
										 |  |  |     this._webSocketEventService = Cc[ | 
					
						
							|  |  |  |       "@mozilla.org/websocketevent/service;1" | 
					
						
							|  |  |  |     ].getService(Ci.nsIWebSocketEventService); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this._runtime = new Runtime(false /* isWorker */); | 
					
						
							|  |  |  |     this._workers = new Map(); | 
					
						
							|  |  |  |     this._frameIdToFrame = new Map(); | 
					
						
							|  |  |  |     this._pageReady = false; | 
					
						
							| 
									
										
										
										
											2023-01-23 11:29:48 -08:00
										 |  |  |     this._javaScriptDisabled = false; | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     for (const browsingContext of helper.collectAllBrowsingContexts(rootBrowsingContext)) | 
					
						
							|  |  |  |       this._createFrame(browsingContext); | 
					
						
							|  |  |  |     this._mainFrame = this.frameForBrowsingContext(rootBrowsingContext); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const webProgress = rootBrowsingContext.docShell.QueryInterface(Ci.nsIInterfaceRequestor) | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |                                 .getInterface(Ci.nsIWebProgress); | 
					
						
							|  |  |  |     this.QueryInterface = ChromeUtils.generateQI([ | 
					
						
							|  |  |  |       Ci.nsIWebProgressListener, | 
					
						
							|  |  |  |       Ci.nsIWebProgressListener2, | 
					
						
							|  |  |  |       Ci.nsISupportsWeakReference, | 
					
						
							|  |  |  |     ]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 08:06:34 -07:00
										 |  |  |     this._addedScrollbarsStylesheetSymbol = Symbol('_addedScrollbarsStylesheetSymbol'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this._wdm = Cc["@mozilla.org/dom/workers/workerdebuggermanager;1"].createInstance(Ci.nsIWorkerDebuggerManager); | 
					
						
							|  |  |  |     this._wdmListener = { | 
					
						
							|  |  |  |       QueryInterface: ChromeUtils.generateQI([Ci.nsIWorkerDebuggerManagerListener]), | 
					
						
							|  |  |  |       onRegister: this._onWorkerCreated.bind(this), | 
					
						
							|  |  |  |       onUnregister: this._onWorkerDestroyed.bind(this), | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     this._wdm.addListener(this._wdmListener); | 
					
						
							|  |  |  |     for (const workerDebugger of this._wdm.getWorkerDebuggerEnumerator()) | 
					
						
							|  |  |  |       this._onWorkerCreated(workerDebugger); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const flags = Ci.nsIWebProgress.NOTIFY_STATE_DOCUMENT | | 
					
						
							| 
									
										
										
										
											2021-04-01 18:49:01 -05:00
										 |  |  |                   Ci.nsIWebProgress.NOTIFY_LOCATION; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this._eventListeners = [ | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |       helper.addObserver((docShell, topic, loadIdentifier) => { | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |         const frame = this.frameForDocShell(docShell); | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |         if (!frame) | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         frame._pendingNavigationId = helper.toProtocolNavigationId(loadIdentifier); | 
					
						
							|  |  |  |         this.emit(FrameTree.Events.NavigationStarted, frame); | 
					
						
							|  |  |  |       }, 'juggler-navigation-started-renderer'), | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |       helper.addObserver(this._onDOMWindowCreated.bind(this), 'content-document-global-created'), | 
					
						
							| 
									
										
										
										
											2020-07-01 13:28:13 -07:00
										 |  |  |       helper.addObserver(this._onDOMWindowCreated.bind(this), 'juggler-dom-window-reused'), | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |       helper.addObserver((browsingContext, topic, why) => { | 
					
						
							|  |  |  |         this._onBrowsingContextAttached(browsingContext); | 
					
						
							|  |  |  |       }, 'browsing-context-attached'), | 
					
						
							|  |  |  |       helper.addObserver((browsingContext, topic, why) => { | 
					
						
							|  |  |  |         this._onBrowsingContextDetached(browsingContext); | 
					
						
							|  |  |  |       }, 'browsing-context-discarded'), | 
					
						
							| 
									
										
										
										
											2023-06-06 15:08:23 -07:00
										 |  |  |       helper.addObserver((subject, topic, eventInfo) => { | 
					
						
							|  |  |  |         const [type, jugglerEventId] = eventInfo.split(' '); | 
					
						
							|  |  |  |         this.emit(FrameTree.Events.InputEvent, { type, jugglerEventId: +(jugglerEventId ?? '0') }); | 
					
						
							|  |  |  |       }, 'juggler-mouse-event-hit-renderer'), | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |       helper.addProgressListener(webProgress, this, flags), | 
					
						
							|  |  |  |     ]; | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-06 15:08:23 -07:00
										 |  |  |     this._dragEventListeners = []; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   workers() { | 
					
						
							|  |  |  |     return [...this._workers.values()]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   runtime() { | 
					
						
							|  |  |  |     return this._runtime; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-14 20:32:12 -08:00
										 |  |  |   setInitScripts(scripts) { | 
					
						
							|  |  |  |     for (const world of this._isolatedWorlds.values()) | 
					
						
							|  |  |  |       world._scriptsToEvaluateOnNewDocument = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (let { worldName, script } of scripts) { | 
					
						
							|  |  |  |       worldName = worldName || ''; | 
					
						
							|  |  |  |       const existing = this._isolatedWorlds.has(worldName); | 
					
						
							|  |  |  |       const world = this._ensureWorld(worldName); | 
					
						
							|  |  |  |       world._scriptsToEvaluateOnNewDocument.push(script); | 
					
						
							|  |  |  |       // FIXME: 'should inherit http credentials from browser context' fails without this
 | 
					
						
							|  |  |  |       if (worldName && !existing) { | 
					
						
							|  |  |  |         for (const frame of this.frames()) | 
					
						
							|  |  |  |           frame._createIsolatedContext(worldName); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2021-04-29 22:09:10 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |   _ensureWorld(worldName) { | 
					
						
							|  |  |  |     worldName = worldName || ''; | 
					
						
							|  |  |  |     let world = this._isolatedWorlds.get(worldName); | 
					
						
							|  |  |  |     if (!world) { | 
					
						
							|  |  |  |       world = new IsolatedWorld(worldName); | 
					
						
							|  |  |  |       this._isolatedWorlds.set(worldName, world); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return world; | 
					
						
							| 
									
										
										
										
											2021-04-29 22:09:10 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   _frameForWorker(workerDebugger) { | 
					
						
							|  |  |  |     if (workerDebugger.type !== Ci.nsIWorkerDebugger.TYPE_DEDICATED) | 
					
						
							|  |  |  |       return null; | 
					
						
							| 
									
										
										
										
											2020-06-08 15:54:47 -07:00
										 |  |  |     if (!workerDebugger.window) | 
					
						
							|  |  |  |       return null; | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     return this.frameForDocShell(workerDebugger.window.docShell); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   _onDOMWindowCreated(window) { | 
					
						
							| 
									
										
										
										
											2021-05-14 08:06:34 -07:00
										 |  |  |     if (!window[this._addedScrollbarsStylesheetSymbol] && this.scrollbarsHidden) { | 
					
						
							|  |  |  |       const styleSheetService = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Components.interfaces.nsIStyleSheetService); | 
					
						
							|  |  |  |       const ioService = Cc["@mozilla.org/network/io-service;1"].getService(Components.interfaces.nsIIOService); | 
					
						
							|  |  |  |       const uri = ioService.newURI('chrome://juggler/content/content/hidden-scrollbars.css', null, null); | 
					
						
							|  |  |  |       const sheet = styleSheetService.preloadSheet(uri, styleSheetService.AGENT_SHEET); | 
					
						
							|  |  |  |       window.windowUtils.addSheet(sheet, styleSheetService.AGENT_SHEET); | 
					
						
							|  |  |  |       window[this._addedScrollbarsStylesheetSymbol] = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     const frame = this.frameForDocShell(window.docShell); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     if (!frame) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     frame._onGlobalObjectCleared(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 08:06:34 -07:00
										 |  |  |   setScrollbarsHidden(hidden) { | 
					
						
							|  |  |  |     this.scrollbarsHidden = hidden; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-23 11:29:48 -08:00
										 |  |  |   setJavaScriptDisabled(javaScriptDisabled) { | 
					
						
							|  |  |  |     this._javaScriptDisabled = javaScriptDisabled; | 
					
						
							|  |  |  |     for (const frame of this.frames()) | 
					
						
							|  |  |  |       frame._updateJavaScriptDisabled(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   _onWorkerCreated(workerDebugger) { | 
					
						
							|  |  |  |     // Note: we do not interoperate with firefox devtools.
 | 
					
						
							|  |  |  |     if (workerDebugger.isInitialized) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     const frame = this._frameForWorker(workerDebugger); | 
					
						
							|  |  |  |     if (!frame) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     const worker = new Worker(frame, workerDebugger); | 
					
						
							|  |  |  |     this._workers.set(workerDebugger, worker); | 
					
						
							|  |  |  |     this.emit(FrameTree.Events.WorkerCreated, worker); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   _onWorkerDestroyed(workerDebugger) { | 
					
						
							|  |  |  |     const worker = this._workers.get(workerDebugger); | 
					
						
							|  |  |  |     if (!worker) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     worker.dispose(); | 
					
						
							|  |  |  |     this._workers.delete(workerDebugger); | 
					
						
							|  |  |  |     this.emit(FrameTree.Events.WorkerDestroyed, worker); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   allFramesInBrowsingContextGroup(group) { | 
					
						
							|  |  |  |     const frames = []; | 
					
						
							| 
									
										
										
										
											2023-02-21 11:46:10 -08:00
										 |  |  |     for (const frameTree of (group.__jugglerFrameTrees || [])) { | 
					
						
							|  |  |  |       for (const frame of frameTree.frames()) { | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |           // Try accessing docShell and domWindow to filter out dead frames.
 | 
					
						
							|  |  |  |           // This might happen for print-preview frames, but maybe for something else as well.
 | 
					
						
							|  |  |  |           frame.docShell(); | 
					
						
							|  |  |  |           frame.domWindow(); | 
					
						
							|  |  |  |           frames.push(frame); | 
					
						
							|  |  |  |         } catch (e) { | 
					
						
							|  |  |  |           dump(`WARNING: unable to access docShell and domWindow of the frame[id=${frame.id()}]\n`); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     return frames; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   isPageReady() { | 
					
						
							|  |  |  |     return this._pageReady; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   forcePageReady() { | 
					
						
							|  |  |  |     if (this._pageReady) | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     this._pageReady = true; | 
					
						
							|  |  |  |     this.emit(FrameTree.Events.PageReady); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-11 16:27:39 +00:00
										 |  |  |   addBinding(worldName, name, script) { | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |     worldName = worldName || ''; | 
					
						
							|  |  |  |     const world = this._ensureWorld(worldName); | 
					
						
							|  |  |  |     world._bindings.set(name, script); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     for (const frame of this.frames()) | 
					
						
							| 
									
										
										
										
											2021-05-11 16:27:39 +00:00
										 |  |  |       frame._addBinding(worldName, name, script); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |   frameForBrowsingContext(browsingContext) { | 
					
						
							|  |  |  |     if (!browsingContext) | 
					
						
							|  |  |  |       return null; | 
					
						
							|  |  |  |     const frameId = helper.browsingContextToFrameId(browsingContext); | 
					
						
							|  |  |  |     return this._frameIdToFrame.get(frameId) ?? null; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   frameForDocShell(docShell) { | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     if (!docShell) | 
					
						
							|  |  |  |       return null; | 
					
						
							|  |  |  |     const frameId = helper.browsingContextToFrameId(docShell.browsingContext); | 
					
						
							|  |  |  |     return this._frameIdToFrame.get(frameId) ?? null; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   frame(frameId) { | 
					
						
							|  |  |  |     return this._frameIdToFrame.get(frameId) || null; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   frames() { | 
					
						
							|  |  |  |     let result = []; | 
					
						
							|  |  |  |     collect(this._mainFrame); | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function collect(frame) { | 
					
						
							|  |  |  |       result.push(frame); | 
					
						
							|  |  |  |       for (const subframe of frame._children) | 
					
						
							|  |  |  |         collect(subframe); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   mainFrame() { | 
					
						
							|  |  |  |     return this._mainFrame; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dispose() { | 
					
						
							|  |  |  |     this._browsingContextGroup.__jugglerFrameTrees.delete(this); | 
					
						
							|  |  |  |     this._wdm.removeListener(this._wdmListener); | 
					
						
							|  |  |  |     this._runtime.dispose(); | 
					
						
							|  |  |  |     helper.removeListeners(this._eventListeners); | 
					
						
							| 
									
										
										
										
											2023-06-06 15:08:23 -07:00
										 |  |  |     helper.removeListeners(this._dragEventListeners); | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   onWindowEvent(event) { | 
					
						
							|  |  |  |     if (event.type !== 'DOMDocElementInserted' || !event.target.ownerGlobal) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const docShell = event.target.ownerGlobal.docShell; | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     const frame = this.frameForDocShell(docShell); | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |     if (!frame) { | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |       dump(`WARNING: ${event.type} for unknown frame ${helper.browsingContextToFrameId(docShell.browsingContext)}\n`); | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (frame._pendingNavigationId) { | 
					
						
							| 
									
										
										
										
											2023-07-06 12:22:45 -07:00
										 |  |  |       docShell.QueryInterface(Ci.nsIWebNavigation); | 
					
						
							|  |  |  |       this._frameNavigationCommitted(frame, docShell.currentURI.spec); | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (frame === this._mainFrame) { | 
					
						
							| 
									
										
										
										
											2023-06-06 15:08:23 -07:00
										 |  |  |       helper.removeListeners(this._dragEventListeners); | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |       const chromeEventHandler = docShell.chromeEventHandler; | 
					
						
							|  |  |  |       const options = { | 
					
						
							|  |  |  |         mozSystemGroup: true, | 
					
						
							|  |  |  |         capture: true, | 
					
						
							|  |  |  |       }; | 
					
						
							| 
									
										
										
										
											2023-06-06 15:08:23 -07:00
										 |  |  |       const emitInputEvent = (event) => this.emit(FrameTree.Events.InputEvent, { type: event.type, jugglerEventId: 0 }); | 
					
						
							|  |  |  |       // Drag events are dispatched from content process, so these we don't see in the
 | 
					
						
							|  |  |  |       // `juggler-mouse-event-hit-renderer` instrumentation.
 | 
					
						
							|  |  |  |       this._dragEventListeners = [ | 
					
						
							|  |  |  |         helper.addEventListener(chromeEventHandler, 'dragstart', emitInputEvent, options), | 
					
						
							|  |  |  |         helper.addEventListener(chromeEventHandler, 'dragover', emitInputEvent, options), | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |       ]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   _frameNavigationCommitted(frame, url) { | 
					
						
							|  |  |  |     for (const subframe of frame._children) | 
					
						
							|  |  |  |       this._detachFrame(subframe); | 
					
						
							|  |  |  |     const navigationId = frame._pendingNavigationId; | 
					
						
							|  |  |  |     frame._pendingNavigationId = null; | 
					
						
							|  |  |  |     frame._lastCommittedNavigationId = navigationId; | 
					
						
							|  |  |  |     frame._url = url; | 
					
						
							|  |  |  |     this.emit(FrameTree.Events.NavigationCommitted, frame); | 
					
						
							|  |  |  |     if (frame === this._mainFrame) | 
					
						
							|  |  |  |       this.forcePageReady(); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   onStateChange(progress, request, flag, status) { | 
					
						
							|  |  |  |     if (!(request instanceof Ci.nsIChannel)) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     const channel = request.QueryInterface(Ci.nsIChannel); | 
					
						
							|  |  |  |     const docShell = progress.DOMWindow.docShell; | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     const frame = this.frameForDocShell(docShell); | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |     if (!frame) | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |       return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-09 09:29:34 -08:00
										 |  |  |     if (!channel.isDocument) { | 
					
						
							|  |  |  |       // Somehow, we can get worker requests here,
 | 
					
						
							|  |  |  |       // while we are only interested in frame documents.
 | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     const isStop = flag & Ci.nsIWebProgressListener.STATE_STOP; | 
					
						
							| 
									
										
										
										
											2023-03-21 01:23:12 +00:00
										 |  |  |     if (isStop && frame._pendingNavigationId && status) { | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |       // Navigation is aborted.
 | 
					
						
							|  |  |  |       const navigationId = frame._pendingNavigationId; | 
					
						
							|  |  |  |       frame._pendingNavigationId = null; | 
					
						
							|  |  |  |       // Always report download navigation as failure to match other browsers.
 | 
					
						
							| 
									
										
										
										
											2020-10-06 00:15:24 -07:00
										 |  |  |       const errorText = helper.getNetworkErrorStatusText(status); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |       this.emit(FrameTree.Events.NavigationAborted, frame, navigationId, errorText); | 
					
						
							| 
									
										
										
										
											2020-10-06 00:15:24 -07:00
										 |  |  |       if (frame === this._mainFrame && status !== Cr.NS_BINDING_ABORTED) | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |         this.forcePageReady(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 18:49:01 -05:00
										 |  |  |   onLocationChange(progress, request, location, flags) { | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     const docShell = progress.DOMWindow.docShell; | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     const frame = this.frameForDocShell(docShell); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     const sameDocumentNavigation = !!(flags & Ci.nsIWebProgressListener.LOCATION_CHANGE_SAME_DOCUMENT); | 
					
						
							|  |  |  |     if (frame && sameDocumentNavigation) { | 
					
						
							|  |  |  |       frame._url = location.spec; | 
					
						
							|  |  |  |       this.emit(FrameTree.Events.SameDocumentNavigation, frame); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |   _onBrowsingContextAttached(browsingContext) { | 
					
						
							|  |  |  |     // If this browsing context doesn't belong to our frame tree - do nothing.
 | 
					
						
							|  |  |  |     if (browsingContext.top !== this._rootBrowsingContext) | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |       return; | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     this._createFrame(browsingContext); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   _onBrowsingContextDetached(browsingContext) { | 
					
						
							|  |  |  |     const frame = this.frameForBrowsingContext(browsingContext); | 
					
						
							|  |  |  |     if (frame) | 
					
						
							|  |  |  |       this._detachFrame(frame); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |   _createFrame(browsingContext) { | 
					
						
							|  |  |  |     const parentFrame = this.frameForBrowsingContext(browsingContext.parent); | 
					
						
							| 
									
										
										
										
											2023-01-23 11:29:48 -08:00
										 |  |  |     if (!parentFrame && this._mainFrame) { | 
					
						
							|  |  |  |       dump(`WARNING: found docShell with the same root, but no parent!\n`); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     const frame = new Frame(this, this._runtime, browsingContext, parentFrame); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this._frameIdToFrame.set(frame.id(), frame); | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     if (browsingContext.docShell?.domWindow && browsingContext.docShell?.domWindow.location) | 
					
						
							|  |  |  |       frame._url = browsingContext.docShell.domWindow.location.href; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this.emit(FrameTree.Events.FrameAttached, frame); | 
					
						
							|  |  |  |     // Create execution context **after** reporting frame.
 | 
					
						
							|  |  |  |     // This is our protocol contract.
 | 
					
						
							|  |  |  |     if (frame.domWindow()) | 
					
						
							|  |  |  |       frame._onGlobalObjectCleared(); | 
					
						
							|  |  |  |     return frame; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   _detachFrame(frame) { | 
					
						
							|  |  |  |     // Detach all children first
 | 
					
						
							|  |  |  |     for (const subframe of frame._children) | 
					
						
							|  |  |  |       this._detachFrame(subframe); | 
					
						
							| 
									
										
										
										
											2023-01-23 11:29:48 -08:00
										 |  |  |     if (frame === this._mainFrame) { | 
					
						
							|  |  |  |       // Do not detach main frame (happens during cross-process navigation),
 | 
					
						
							|  |  |  |       // as it confuses the client.
 | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this._frameIdToFrame.delete(frame.id()); | 
					
						
							|  |  |  |     if (frame._parentFrame) | 
					
						
							|  |  |  |       frame._parentFrame._children.delete(frame); | 
					
						
							|  |  |  |     frame._parentFrame = null; | 
					
						
							|  |  |  |     frame.dispose(); | 
					
						
							|  |  |  |     this.emit(FrameTree.Events.FrameDetached, frame); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | FrameTree.Events = { | 
					
						
							|  |  |  |   FrameAttached: 'frameattached', | 
					
						
							|  |  |  |   FrameDetached: 'framedetached', | 
					
						
							|  |  |  |   WorkerCreated: 'workercreated', | 
					
						
							|  |  |  |   WorkerDestroyed: 'workerdestroyed', | 
					
						
							| 
									
										
										
										
											2020-10-29 16:33:50 -07:00
										 |  |  |   WebSocketCreated: 'websocketcreated', | 
					
						
							|  |  |  |   WebSocketOpened: 'websocketopened', | 
					
						
							|  |  |  |   WebSocketClosed: 'websocketclosed', | 
					
						
							|  |  |  |   WebSocketFrameReceived: 'websocketframereceived', | 
					
						
							|  |  |  |   WebSocketFrameSent: 'websocketframesent', | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   NavigationStarted: 'navigationstarted', | 
					
						
							|  |  |  |   NavigationCommitted: 'navigationcommitted', | 
					
						
							|  |  |  |   NavigationAborted: 'navigationaborted', | 
					
						
							|  |  |  |   SameDocumentNavigation: 'samedocumentnavigation', | 
					
						
							|  |  |  |   PageReady: 'pageready', | 
					
						
							| 
									
										
										
										
											2023-06-06 15:08:23 -07:00
										 |  |  |   InputEvent: 'inputevent', | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  | class IsolatedWorld { | 
					
						
							|  |  |  |   constructor(name) { | 
					
						
							|  |  |  |     this._name = name; | 
					
						
							|  |  |  |     this._scriptsToEvaluateOnNewDocument = []; | 
					
						
							|  |  |  |     this._bindings = new Map(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  | class Frame { | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |   constructor(frameTree, runtime, browsingContext, parentFrame) { | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this._frameTree = frameTree; | 
					
						
							|  |  |  |     this._runtime = runtime; | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     this._browsingContext = browsingContext; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this._children = new Set(); | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     this._frameId = helper.browsingContextToFrameId(browsingContext); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this._parentFrame = null; | 
					
						
							|  |  |  |     this._url = ''; | 
					
						
							|  |  |  |     if (parentFrame) { | 
					
						
							|  |  |  |       this._parentFrame = parentFrame; | 
					
						
							|  |  |  |       parentFrame._children.add(this); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     this._lastCommittedNavigationId = null; | 
					
						
							|  |  |  |     this._pendingNavigationId = null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     this._textInputProcessor = null; | 
					
						
							| 
									
										
										
										
											2020-10-29 16:33:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |     this._worldNameToContext = new Map(); | 
					
						
							| 
									
										
										
										
											2021-04-29 22:09:10 +00:00
										 |  |  |     this._initialNavigationDone = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 16:33:50 -07:00
										 |  |  |     this._webSocketListenerInnerWindowId = 0; | 
					
						
							|  |  |  |     // WebSocketListener calls frameReceived event before webSocketOpened.
 | 
					
						
							|  |  |  |     // To avoid this, serialize event reporting.
 | 
					
						
							|  |  |  |     this._webSocketInfos = new Map(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const dispatchWebSocketFrameReceived = (webSocketSerialID, frame) => this._frameTree.emit(FrameTree.Events.WebSocketFrameReceived, { | 
					
						
							|  |  |  |       frameId: this._frameId, | 
					
						
							|  |  |  |       wsid: webSocketSerialID + '', | 
					
						
							|  |  |  |       opcode: frame.opCode, | 
					
						
							|  |  |  |       data: frame.opCode !== 1 ? btoa(frame.payload) : frame.payload, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     this._webSocketListener = { | 
					
						
							|  |  |  |       QueryInterface: ChromeUtils.generateQI([Ci.nsIWebSocketEventListener, ]), | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       webSocketCreated: (webSocketSerialID, uri, protocols) => { | 
					
						
							|  |  |  |         this._frameTree.emit(FrameTree.Events.WebSocketCreated, { | 
					
						
							|  |  |  |           frameId: this._frameId, | 
					
						
							|  |  |  |           wsid: webSocketSerialID + '', | 
					
						
							|  |  |  |           requestURL: uri, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         this._webSocketInfos.set(webSocketSerialID, { | 
					
						
							|  |  |  |           opened: false, | 
					
						
							|  |  |  |           pendingIncomingFrames: [], | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       webSocketOpened: (webSocketSerialID, effectiveURI, protocols, extensions, httpChannelId) => { | 
					
						
							|  |  |  |         this._frameTree.emit(FrameTree.Events.WebSocketOpened, { | 
					
						
							|  |  |  |           frameId: this._frameId, | 
					
						
							|  |  |  |           requestId: httpChannelId + '', | 
					
						
							|  |  |  |           wsid: webSocketSerialID + '', | 
					
						
							|  |  |  |           effectiveURL: effectiveURI, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         const info = this._webSocketInfos.get(webSocketSerialID); | 
					
						
							|  |  |  |         info.opened = true; | 
					
						
							|  |  |  |         for (const frame of info.pendingIncomingFrames) | 
					
						
							|  |  |  |           dispatchWebSocketFrameReceived(webSocketSerialID, frame); | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       webSocketMessageAvailable: (webSocketSerialID, data, messageType) => { | 
					
						
							|  |  |  |         // We don't use this event.
 | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       webSocketClosed: (webSocketSerialID, wasClean, code, reason) => { | 
					
						
							|  |  |  |         this._webSocketInfos.delete(webSocketSerialID); | 
					
						
							|  |  |  |         let error = ''; | 
					
						
							|  |  |  |         if (!wasClean) { | 
					
						
							|  |  |  |           const keys = Object.keys(Ci.nsIWebSocketChannel); | 
					
						
							|  |  |  |           for (const key of keys) { | 
					
						
							|  |  |  |             if (Ci.nsIWebSocketChannel[key] === code) | 
					
						
							|  |  |  |               error = key; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         this._frameTree.emit(FrameTree.Events.WebSocketClosed, { | 
					
						
							|  |  |  |           frameId: this._frameId, | 
					
						
							|  |  |  |           wsid: webSocketSerialID + '', | 
					
						
							|  |  |  |           error, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       frameReceived: (webSocketSerialID, frame) => { | 
					
						
							|  |  |  |         // Report only text and binary frames.
 | 
					
						
							|  |  |  |         if (frame.opCode !== 1 && frame.opCode !== 2) | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         const info = this._webSocketInfos.get(webSocketSerialID); | 
					
						
							|  |  |  |         if (info.opened) | 
					
						
							|  |  |  |           dispatchWebSocketFrameReceived(webSocketSerialID, frame); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           info.pendingIncomingFrames.push(frame); | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       frameSent: (webSocketSerialID, frame) => { | 
					
						
							|  |  |  |         // Report only text and binary frames.
 | 
					
						
							|  |  |  |         if (frame.opCode !== 1 && frame.opCode !== 2) | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         this._frameTree.emit(FrameTree.Events.WebSocketFrameSent, { | 
					
						
							|  |  |  |           frameId: this._frameId, | 
					
						
							|  |  |  |           wsid: webSocketSerialID + '', | 
					
						
							|  |  |  |           opcode: frame.opCode, | 
					
						
							|  |  |  |           data: frame.opCode !== 1 ? btoa(frame.payload) : frame.payload, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |   _createIsolatedContext(name) { | 
					
						
							| 
									
										
										
										
											2021-04-29 22:09:10 +00:00
										 |  |  |     const principal = [this.domWindow()]; // extended principal
 | 
					
						
							|  |  |  |     const sandbox = Cu.Sandbox(principal, { | 
					
						
							|  |  |  |       sandboxPrototype: this.domWindow(), | 
					
						
							|  |  |  |       wantComponents: false, | 
					
						
							|  |  |  |       wantExportHelpers: false, | 
					
						
							|  |  |  |       wantXrays: true, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     const world = this._runtime.createExecutionContext(this.domWindow(), sandbox, { | 
					
						
							|  |  |  |       frameId: this.id(), | 
					
						
							|  |  |  |       name, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |     this._worldNameToContext.set(name, world); | 
					
						
							| 
									
										
										
										
											2021-04-29 22:09:10 +00:00
										 |  |  |     return world; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   unsafeObject(objectId) { | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |     for (const context of this._worldNameToContext.values()) { | 
					
						
							| 
									
										
										
										
											2021-04-29 22:09:10 +00:00
										 |  |  |       const result = context.unsafeObject(objectId); | 
					
						
							|  |  |  |       if (result) | 
					
						
							|  |  |  |         return result.object; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw new Error('Cannot find object with id = ' + objectId); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   dispose() { | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |     for (const context of this._worldNameToContext.values()) | 
					
						
							|  |  |  |       this._runtime.destroyExecutionContext(context); | 
					
						
							|  |  |  |     this._worldNameToContext.clear(); | 
					
						
							| 
									
										
										
										
											2021-05-11 16:27:39 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   _addBinding(worldName, name, script) { | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |     let executionContext = this._worldNameToContext.get(worldName); | 
					
						
							|  |  |  |     if (worldName && !executionContext) | 
					
						
							|  |  |  |       executionContext = this._createIsolatedContext(worldName); | 
					
						
							| 
									
										
										
										
											2021-05-11 16:27:39 +00:00
										 |  |  |     if (executionContext) | 
					
						
							|  |  |  |       executionContext.addBinding(name, script); | 
					
						
							| 
									
										
										
										
											2021-05-08 18:06:56 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   _onGlobalObjectCleared() { | 
					
						
							| 
									
										
										
										
											2020-10-29 16:33:50 -07:00
										 |  |  |     const webSocketService = this._frameTree._webSocketEventService; | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     if (this._webSocketListenerInnerWindowId && webSocketService.hasListenerFor(this._webSocketListenerInnerWindowId)) | 
					
						
							| 
									
										
										
										
											2020-10-29 16:33:50 -07:00
										 |  |  |       webSocketService.removeListener(this._webSocketListenerInnerWindowId, this._webSocketListener); | 
					
						
							|  |  |  |     this._webSocketListenerInnerWindowId = this.domWindow().windowGlobalChild.innerWindowId; | 
					
						
							|  |  |  |     webSocketService.addListener(this._webSocketListenerInnerWindowId, this._webSocketListener); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |     for (const context of this._worldNameToContext.values()) | 
					
						
							|  |  |  |       this._runtime.destroyExecutionContext(context); | 
					
						
							|  |  |  |     this._worldNameToContext.clear(); | 
					
						
							| 
									
										
										
										
											2021-05-11 16:27:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |     this._worldNameToContext.set('', this._runtime.createExecutionContext(this.domWindow(), this.domWindow(), { | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |       frameId: this._frameId, | 
					
						
							|  |  |  |       name: '', | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |     })); | 
					
						
							|  |  |  |     for (const [name, world] of this._frameTree._isolatedWorlds) { | 
					
						
							|  |  |  |       if (name) | 
					
						
							|  |  |  |         this._createIsolatedContext(name); | 
					
						
							|  |  |  |       const executionContext = this._worldNameToContext.get(name); | 
					
						
							|  |  |  |       // Add bindings before evaluating scripts.
 | 
					
						
							|  |  |  |       for (const [name, script] of world._bindings) | 
					
						
							|  |  |  |         executionContext.addBinding(name, script); | 
					
						
							|  |  |  |       for (const script of world._scriptsToEvaluateOnNewDocument) | 
					
						
							|  |  |  |         executionContext.evaluateScriptSafely(script); | 
					
						
							| 
									
										
										
										
											2021-04-29 22:09:10 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-01-23 11:29:48 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const url = this.domWindow().location?.href; | 
					
						
							|  |  |  |     if (url === 'about:blank' && !this._url) { | 
					
						
							|  |  |  |       // Sometimes FrameTree is created too early, before the location has been set.
 | 
					
						
							|  |  |  |       this._url = url; | 
					
						
							|  |  |  |       this._frameTree.emit(FrameTree.Events.NavigationCommitted, this); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     this._updateJavaScriptDisabled(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   _updateJavaScriptDisabled() { | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     if (this._browsingContext.currentWindowContext) | 
					
						
							|  |  |  |       this._browsingContext.currentWindowContext.allowJavascript = !this._frameTree._javaScriptDisabled; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:52:08 +00:00
										 |  |  |   mainExecutionContext() { | 
					
						
							|  |  |  |     return this._worldNameToContext.get(''); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   textInputProcessor() { | 
					
						
							|  |  |  |     if (!this._textInputProcessor) { | 
					
						
							|  |  |  |       this._textInputProcessor = Cc["@mozilla.org/text-input-processor;1"].createInstance(Ci.nsITextInputProcessor); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     this._textInputProcessor.beginInputTransactionForTests(this.docShell().DOMWindow); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     return this._textInputProcessor; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   pendingNavigationId() { | 
					
						
							|  |  |  |     return this._pendingNavigationId; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   lastCommittedNavigationId() { | 
					
						
							|  |  |  |     return this._lastCommittedNavigationId; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   docShell() { | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     return this._browsingContext.docShell; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   domWindow() { | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     return this.docShell()?.domWindow; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   name() { | 
					
						
							| 
									
										
										
										
											2023-04-24 21:28:08 +00:00
										 |  |  |     const frameElement = this.domWindow()?.frameElement; | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     let name = ''; | 
					
						
							|  |  |  |     if (frameElement) | 
					
						
							|  |  |  |       name = frameElement.getAttribute('name') || frameElement.getAttribute('id') || ''; | 
					
						
							|  |  |  |     return name; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   parentFrame() { | 
					
						
							|  |  |  |     return this._parentFrame; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   id() { | 
					
						
							|  |  |  |     return this._frameId; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   url() { | 
					
						
							|  |  |  |     return this._url; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Worker { | 
					
						
							|  |  |  |   constructor(frame, workerDebugger) { | 
					
						
							|  |  |  |     this._frame = frame; | 
					
						
							|  |  |  |     this._workerId = helper.generateId(); | 
					
						
							|  |  |  |     this._workerDebugger = workerDebugger; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     workerDebugger.initialize('chrome://juggler/content/content/WorkerMain.js'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-11 18:16:33 -07:00
										 |  |  |     this._channel = new SimpleChannel(`content::worker[${this._workerId}]`, 'worker-' + this._workerId); | 
					
						
							| 
									
										
										
										
											2020-11-02 16:21:34 -08:00
										 |  |  |     this._channel.setTransport({ | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |       sendMessage: obj => workerDebugger.postMessage(JSON.stringify(obj)), | 
					
						
							|  |  |  |       dispose: () => {}, | 
					
						
							| 
									
										
										
										
											2020-11-02 16:21:34 -08:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |     this._workerDebuggerListener = { | 
					
						
							|  |  |  |       QueryInterface: ChromeUtils.generateQI([Ci.nsIWorkerDebuggerListener]), | 
					
						
							|  |  |  |       onMessage: msg => void this._channel._onMessage(JSON.parse(msg)), | 
					
						
							|  |  |  |       onClose: () => void this._channel.dispose(), | 
					
						
							|  |  |  |       onError: (filename, lineno, message) => { | 
					
						
							| 
									
										
										
										
											2023-01-23 11:29:48 -08:00
										 |  |  |         dump(`WARNING: Error in worker: ${message} @${filename}:${lineno}\n`); | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  |       }, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     workerDebugger.addListener(this._workerDebuggerListener); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   channel() { | 
					
						
							|  |  |  |     return this._channel; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   frame() { | 
					
						
							|  |  |  |     return this._frame; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   id() { | 
					
						
							|  |  |  |     return this._workerId; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   url() { | 
					
						
							|  |  |  |     return this._workerDebugger.url; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dispose() { | 
					
						
							|  |  |  |     this._channel.dispose(); | 
					
						
							|  |  |  |     this._workerDebugger.removeListener(this._workerDebuggerListener); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 16:33:50 -07:00
										 |  |  | function channelId(channel) { | 
					
						
							|  |  |  |   if (channel instanceof Ci.nsIIdentChannel) { | 
					
						
							|  |  |  |     const identChannel = channel.QueryInterface(Ci.nsIIdentChannel); | 
					
						
							|  |  |  |     return String(identChannel.channelId); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return helper.generateId(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 16:51:13 -07:00
										 |  |  | var EXPORTED_SYMBOLS = ['FrameTree']; | 
					
						
							|  |  |  | this.FrameTree = FrameTree; | 
					
						
							|  |  |  | 
 |