mirror of
https://github.com/microsoft/playwright.git
synced 2025-06-26 21:40:17 +00:00

This patch: 1. Changes `SimpleChannel` to buffer messages to the namespace that hasn't been registered yet. This allows us to create `SimpleChannel` per target on the browser side right away. 2. Removes multisession support. Now there's only one `PageAgent` in the content process, which talks to a single `PageHandler` on the browser side. Both ends can be created as-soon-as-needed; thanks to `SimpleChannel` bufferring, no messages will be lost and all messages will be delivered in proper order. (This is currently the reason why build 1178 flakes on windows). 3. Straightens up the target reporting. Targets are reported as soon as they appear on the browser side. **NOTE:** this doesn't yet remove sessions from protocol. References #3995
158 lines
4.4 KiB
JavaScript
158 lines
4.4 KiB
JavaScript
/* 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/. */
|
|
|
|
const {Services} = ChromeUtils.import("resource://gre/modules/Services.jsm");
|
|
const {Helper} = ChromeUtils.import('chrome://juggler/content/Helper.js');
|
|
const {FrameTree} = ChromeUtils.import('chrome://juggler/content/content/FrameTree.js');
|
|
const {SimpleChannel} = ChromeUtils.import('chrome://juggler/content/SimpleChannel.js');
|
|
const {PageAgent} = ChromeUtils.import('chrome://juggler/content/content/PageAgent.js');
|
|
|
|
let frameTree;
|
|
const helper = new Helper();
|
|
const messageManager = this;
|
|
|
|
let pageAgent;
|
|
|
|
let failedToOverrideTimezone = false;
|
|
|
|
const applySetting = {
|
|
geolocation: (geolocation) => {
|
|
if (geolocation) {
|
|
docShell.setGeolocationOverride({
|
|
coords: {
|
|
latitude: geolocation.latitude,
|
|
longitude: geolocation.longitude,
|
|
accuracy: geolocation.accuracy,
|
|
altitude: NaN,
|
|
altitudeAccuracy: NaN,
|
|
heading: NaN,
|
|
speed: NaN,
|
|
},
|
|
address: null,
|
|
timestamp: Date.now()
|
|
});
|
|
} else {
|
|
docShell.setGeolocationOverride(null);
|
|
}
|
|
},
|
|
|
|
onlineOverride: (onlineOverride) => {
|
|
if (!onlineOverride) {
|
|
docShell.onlineOverride = Ci.nsIDocShell.ONLINE_OVERRIDE_NONE;
|
|
return;
|
|
}
|
|
docShell.onlineOverride = onlineOverride === 'online' ?
|
|
Ci.nsIDocShell.ONLINE_OVERRIDE_ONLINE : Ci.nsIDocShell.ONLINE_OVERRIDE_OFFLINE;
|
|
},
|
|
|
|
bypassCSP: (bypassCSP) => {
|
|
docShell.bypassCSPEnabled = bypassCSP;
|
|
},
|
|
|
|
timezoneId: (timezoneId) => {
|
|
failedToOverrideTimezone = !docShell.overrideTimezone(timezoneId);
|
|
},
|
|
|
|
locale: (locale) => {
|
|
docShell.languageOverride = locale;
|
|
},
|
|
|
|
javaScriptDisabled: (javaScriptDisabled) => {
|
|
docShell.allowJavascript = !javaScriptDisabled;
|
|
},
|
|
|
|
hasTouch: (hasTouch) => {
|
|
docShell.touchEventsOverride = hasTouch ? Ci.nsIDocShell.TOUCHEVENTS_OVERRIDE_ENABLED : Ci.nsIDocShell.TOUCHEVENTS_OVERRIDE_NONE;
|
|
},
|
|
|
|
colorScheme: (colorScheme) => {
|
|
frameTree.setColorScheme(colorScheme);
|
|
},
|
|
|
|
deviceScaleFactor: (deviceScaleFactor) => {
|
|
docShell.contentViewer.overrideDPPX = deviceScaleFactor || this._initialDPPX;
|
|
docShell.deviceSizeIsPageSize = !!deviceScaleFactor;
|
|
},
|
|
};
|
|
|
|
const channel = SimpleChannel.createForMessageManager('content::page', messageManager);
|
|
|
|
function initialize() {
|
|
const response = sendSyncMessage('juggler:content-ready')[0];
|
|
// If we didn't get a response, then we don't want to do anything
|
|
// as a part of this frame script.
|
|
if (!response)
|
|
return;
|
|
const {
|
|
scriptsToEvaluateOnNewDocument = [],
|
|
bindings = [],
|
|
settings = {}
|
|
} = response || {};
|
|
|
|
// Enforce focused state for all top level documents.
|
|
docShell.overrideHasFocus = true;
|
|
frameTree = new FrameTree(docShell);
|
|
for (const [name, value] of Object.entries(settings)) {
|
|
if (value !== undefined)
|
|
applySetting[name](value);
|
|
}
|
|
for (const script of scriptsToEvaluateOnNewDocument)
|
|
frameTree.addScriptToEvaluateOnNewDocument(script);
|
|
for (const { name, script } of bindings)
|
|
frameTree.addBinding(name, script);
|
|
|
|
pageAgent = new PageAgent(messageManager, channel, frameTree);
|
|
pageAgent.enable();
|
|
|
|
channel.register('', {
|
|
addScriptToEvaluateOnNewDocument(script) {
|
|
frameTree.addScriptToEvaluateOnNewDocument(script);
|
|
},
|
|
|
|
addBinding({name, script}) {
|
|
frameTree.addBinding(name, script);
|
|
},
|
|
|
|
applyContextSetting({name, value}) {
|
|
applySetting[name](value);
|
|
},
|
|
|
|
ensurePermissions() {
|
|
// noop, just a rountrip.
|
|
},
|
|
|
|
hasFailedToOverrideTimezone() {
|
|
return failedToOverrideTimezone;
|
|
},
|
|
|
|
async awaitViewportDimensions({width, height}) {
|
|
const win = docShell.domWindow;
|
|
if (win.innerWidth === width && win.innerHeight === height)
|
|
return;
|
|
await new Promise(resolve => {
|
|
const listener = helper.addEventListener(win, 'resize', () => {
|
|
if (win.innerWidth === width && win.innerHeight === height) {
|
|
helper.removeListeners([listener]);
|
|
resolve();
|
|
}
|
|
});
|
|
});
|
|
},
|
|
|
|
dispose() {
|
|
},
|
|
});
|
|
|
|
const gListeners = [
|
|
helper.addEventListener(messageManager, 'unload', msg => {
|
|
helper.removeListeners(gListeners);
|
|
pageAgent.dispose();
|
|
frameTree.dispose();
|
|
channel.dispose();
|
|
}),
|
|
];
|
|
}
|
|
|
|
initialize();
|