electron/lib/renderer/web-view/web-view-attributes.ts

272 lines
9 KiB
TypeScript
Raw Normal View History

import { WEB_VIEW_ATTRIBUTES, WEB_VIEW_ERROR_MESSAGES } from '@electron/internal/renderer/web-view/web-view-constants';
import type { WebViewImpl } from '@electron/internal/renderer/web-view/web-view-impl';
2016-01-12 02:40:23 +00:00
const resolveURL = function (url?: string | null) {
2021-04-09 21:22:18 +00:00
return url ? new URL(url, location.href).href : '';
2020-03-20 20:28:31 +00:00
};
2016-01-12 02:40:23 +00:00
interface MutationHandler {
handleMutation (_oldValue: any, _newValue: any): any;
}
2016-01-14 19:10:12 +00:00
// Attribute objects.
// Default implementation of a WebView attribute.
export class WebViewAttribute implements MutationHandler {
public value: any;
public ignoreMutation = false;
constructor (public name: string, public webViewImpl: WebViewImpl) {
2020-03-20 20:28:31 +00:00
this.name = name;
this.value = (webViewImpl.webviewNode as Record<string, any>)[name] || '';
this.webViewImpl = webViewImpl;
this.defineProperty();
2016-01-15 22:28:12 +00:00
}
2016-01-12 02:40:23 +00:00
2016-01-15 22:28:12 +00:00
// Retrieves and returns the attribute's value.
public getValue () {
2020-03-20 20:28:31 +00:00
return this.webViewImpl.webviewNode.getAttribute(this.name) || this.value;
2016-01-15 22:28:12 +00:00
}
2016-01-12 02:40:23 +00:00
2016-01-15 22:28:12 +00:00
// Sets the attribute's value.
public setValue (value: any) {
2020-03-20 20:28:31 +00:00
this.webViewImpl.webviewNode.setAttribute(this.name, value || '');
2016-01-15 22:28:12 +00:00
}
2016-01-12 02:40:23 +00:00
2016-01-15 22:28:12 +00:00
// Changes the attribute's value without triggering its mutation handler.
public setValueIgnoreMutation (value: any) {
2020-03-20 20:28:31 +00:00
this.ignoreMutation = true;
this.setValue(value);
this.ignoreMutation = false;
2016-01-15 22:28:12 +00:00
}
2016-01-12 02:40:23 +00:00
2016-01-15 22:28:12 +00:00
// Defines this attribute as a property on the webview node.
public defineProperty () {
2016-01-15 22:28:12 +00:00
return Object.defineProperty(this.webViewImpl.webviewNode, this.name, {
2016-03-11 22:08:14 +00:00
get: () => {
2020-03-20 20:28:31 +00:00
return this.getValue();
2016-03-11 22:08:14 +00:00
},
set: (value) => {
2020-03-20 20:28:31 +00:00
return this.setValue(value);
2016-03-11 22:08:14 +00:00
},
2016-01-15 22:28:12 +00:00
enumerable: true
2020-03-20 20:28:31 +00:00
});
2016-01-15 23:10:26 +00:00
}
2016-01-12 02:40:23 +00:00
2016-01-15 22:28:12 +00:00
// Called when the attribute's value changes.
public handleMutation: MutationHandler['handleMutation'] = () => undefined;
}
2016-01-12 02:40:23 +00:00
2016-01-15 22:28:12 +00:00
// An attribute that is treated as a Boolean.
class BooleanAttribute extends WebViewAttribute {
2016-03-25 19:57:17 +00:00
getValue () {
2020-03-20 20:28:31 +00:00
return this.webViewImpl.webviewNode.hasAttribute(this.name);
2016-01-12 02:40:23 +00:00
}
setValue (value: boolean) {
2016-11-03 18:37:11 +00:00
if (value) {
2020-03-20 20:28:31 +00:00
this.webViewImpl.webviewNode.setAttribute(this.name, '');
2016-11-03 18:37:11 +00:00
} else {
2020-03-20 20:28:31 +00:00
this.webViewImpl.webviewNode.removeAttribute(this.name);
2016-01-15 22:28:12 +00:00
}
}
}
2016-01-12 02:40:23 +00:00
2016-01-14 18:35:29 +00:00
// Attribute representing the state of the storage partition.
export class PartitionAttribute extends WebViewAttribute {
public validPartitionId = true;
constructor (public webViewImpl: WebViewImpl) {
super(WEB_VIEW_ATTRIBUTES.PARTITION, webViewImpl);
2016-01-15 22:28:12 +00:00
}
2016-01-12 02:40:23 +00:00
public handleMutation = (oldValue: any, newValue: any) => {
2020-03-20 20:28:31 +00:00
newValue = newValue || '';
2016-01-12 02:40:23 +00:00
2016-01-15 22:28:12 +00:00
// The partition cannot change if the webview has already navigated.
if (!this.webViewImpl.beforeFirstNavigation) {
console.error(WEB_VIEW_ERROR_MESSAGES.ALREADY_NAVIGATED);
2020-03-20 20:28:31 +00:00
this.setValueIgnoreMutation(oldValue);
return;
2016-01-15 22:28:12 +00:00
}
if (newValue === 'persist:') {
2020-03-20 20:28:31 +00:00
this.validPartitionId = false;
console.error(WEB_VIEW_ERROR_MESSAGES.INVALID_PARTITION_ATTRIBUTE);
2016-01-15 22:28:12 +00:00
}
};
}
2016-01-15 22:28:12 +00:00
// Attribute that handles the location and navigation of the webview.
export class SrcAttribute extends WebViewAttribute {
public observer!: MutationObserver;
constructor (public webViewImpl: WebViewImpl) {
super(WEB_VIEW_ATTRIBUTES.SRC, webViewImpl);
2020-03-20 20:28:31 +00:00
this.setupMutationObserver();
2016-01-15 22:28:12 +00:00
}
public getValue () {
2016-01-15 22:28:12 +00:00
if (this.webViewImpl.webviewNode.hasAttribute(this.name)) {
2020-03-20 20:28:31 +00:00
return resolveURL(this.webViewImpl.webviewNode.getAttribute(this.name));
2016-01-15 22:28:12 +00:00
} else {
2020-03-20 20:28:31 +00:00
return this.value;
2016-01-15 22:28:12 +00:00
}
2016-01-12 02:40:23 +00:00
}
public setValueIgnoreMutation (value: any) {
2020-03-20 20:28:31 +00:00
super.setValueIgnoreMutation(value);
2016-01-12 02:40:23 +00:00
2016-01-15 22:28:12 +00:00
// takeRecords() is needed to clear queued up src mutations. Without it, it
// is possible for this change to get picked up asynchronously by src's
2016-01-15 22:28:12 +00:00
// mutation observer |observer|, and then get handled even though we do not
// want to handle this mutation.
2020-03-20 20:28:31 +00:00
this.observer.takeRecords();
2016-01-15 22:28:12 +00:00
}
2016-01-12 02:40:23 +00:00
public handleMutation = (oldValue: any, newValue: any) => {
2016-01-15 22:28:12 +00:00
// Once we have navigated, we don't allow clearing the src attribute.
// Once <webview> enters a navigated state, it cannot return to a
// placeholder state.
if (!newValue && oldValue) {
// src attribute changes normally initiate a navigation. We suppress
// the next src attribute handler call to avoid reloading the page
// on every guest-initiated navigation.
2020-03-20 20:28:31 +00:00
this.setValueIgnoreMutation(oldValue);
return;
2016-01-15 22:28:12 +00:00
}
2020-03-20 20:28:31 +00:00
this.parse();
};
2016-01-12 02:40:23 +00:00
2016-01-15 22:28:12 +00:00
// The purpose of this mutation observer is to catch assignment to the src
// attribute without any changes to its value. This is useful in the case
// where the webview guest has crashed and navigating to the same address
// spawns off a new process.
public setupMutationObserver () {
2016-03-11 22:08:14 +00:00
this.observer = new MutationObserver((mutations) => {
2016-11-03 18:09:53 +00:00
for (const mutation of mutations) {
2020-03-20 20:28:31 +00:00
const { oldValue } = mutation;
const newValue = this.getValue();
2016-03-11 22:08:14 +00:00
if (oldValue !== newValue) {
2020-03-20 20:28:31 +00:00
return;
2016-01-12 02:40:23 +00:00
}
2020-03-20 20:28:31 +00:00
this.handleMutation(oldValue, newValue);
2016-03-11 22:08:14 +00:00
}
2020-03-20 20:28:31 +00:00
});
2016-11-03 17:39:40 +00:00
const params = {
2016-01-15 22:28:12 +00:00
attributes: true,
attributeOldValue: true,
attributeFilter: [this.name]
2020-03-20 20:28:31 +00:00
};
2020-03-20 20:28:31 +00:00
this.observer.observe(this.webViewImpl.webviewNode, params);
}
2016-01-15 22:28:12 +00:00
public parse () {
if (!this.webViewImpl.elementAttached || !(this.webViewImpl.attributes.get(WEB_VIEW_ATTRIBUTES.PARTITION) as PartitionAttribute).validPartitionId || !this.getValue()) {
2020-03-20 20:28:31 +00:00
return;
2016-01-15 22:28:12 +00:00
}
if (this.webViewImpl.guestInstanceId == null) {
if (this.webViewImpl.beforeFirstNavigation) {
2020-03-20 20:28:31 +00:00
this.webViewImpl.beforeFirstNavigation = false;
this.webViewImpl.createGuest();
2016-01-15 22:28:12 +00:00
}
2020-03-20 20:28:31 +00:00
return;
2016-01-12 02:40:23 +00:00
}
2016-01-15 22:28:12 +00:00
// Navigate to |this.src|.
2020-03-20 20:28:31 +00:00
const opts: Record<string, string> = {};
const httpreferrer = this.webViewImpl.attributes.get(WEB_VIEW_ATTRIBUTES.HTTPREFERRER)!.getValue();
2016-01-15 22:28:12 +00:00
if (httpreferrer) {
2020-03-20 20:28:31 +00:00
opts.httpReferrer = httpreferrer;
2016-01-15 22:28:12 +00:00
}
const useragent = this.webViewImpl.attributes.get(WEB_VIEW_ATTRIBUTES.USERAGENT)!.getValue();
2016-01-15 22:28:12 +00:00
if (useragent) {
2020-03-20 20:28:31 +00:00
opts.userAgent = useragent;
2016-01-15 22:28:12 +00:00
}
(this.webViewImpl.webviewNode as Electron.WebviewTag).loadURL(this.getValue(), opts)
.catch(err => {
console.error('Unexpected error while loading URL', err);
});
2016-01-12 02:40:23 +00:00
}
2016-01-15 22:28:12 +00:00
}
2016-01-12 02:40:23 +00:00
// Attribute specifies HTTP referrer.
2016-01-15 22:28:12 +00:00
class HttpReferrerAttribute extends WebViewAttribute {
constructor (webViewImpl: WebViewImpl) {
super(WEB_VIEW_ATTRIBUTES.HTTPREFERRER, webViewImpl);
2016-01-15 22:28:12 +00:00
}
}
2016-01-12 02:40:23 +00:00
2016-01-14 18:35:29 +00:00
// Attribute specifies user agent
2016-01-15 22:28:12 +00:00
class UserAgentAttribute extends WebViewAttribute {
constructor (webViewImpl: WebViewImpl) {
super(WEB_VIEW_ATTRIBUTES.USERAGENT, webViewImpl);
2016-01-15 22:28:12 +00:00
}
}
2016-01-12 02:40:23 +00:00
2016-01-14 18:35:29 +00:00
// Attribute that set preload script.
2016-01-15 23:09:25 +00:00
class PreloadAttribute extends WebViewAttribute {
constructor (webViewImpl: WebViewImpl) {
super(WEB_VIEW_ATTRIBUTES.PRELOAD, webViewImpl);
}
2016-01-15 22:28:12 +00:00
public getValue () {
2016-01-15 22:28:12 +00:00
if (!this.webViewImpl.webviewNode.hasAttribute(this.name)) {
2020-03-20 20:28:31 +00:00
return this.value;
2016-01-15 22:28:12 +00:00
}
2020-03-20 20:28:31 +00:00
let preload = resolveURL(this.webViewImpl.webviewNode.getAttribute(this.name));
const protocol = preload.substr(0, 5);
2016-01-15 22:28:12 +00:00
if (protocol !== 'file:') {
console.error(WEB_VIEW_ERROR_MESSAGES.INVALID_PRELOAD_ATTRIBUTE);
2020-03-20 20:28:31 +00:00
preload = '';
2016-01-15 22:28:12 +00:00
}
2020-03-20 20:28:31 +00:00
return preload;
}
2016-01-15 22:28:12 +00:00
}
2016-01-12 02:40:23 +00:00
2016-01-21 10:13:41 +00:00
// Attribute that specifies the blink features to be enabled.
class BlinkFeaturesAttribute extends WebViewAttribute {
constructor (webViewImpl: WebViewImpl) {
super(WEB_VIEW_ATTRIBUTES.BLINKFEATURES, webViewImpl);
2016-01-21 10:13:41 +00:00
}
}
// Attribute that specifies the blink features to be disabled.
class DisableBlinkFeaturesAttribute extends WebViewAttribute {
constructor (webViewImpl: WebViewImpl) {
super(WEB_VIEW_ATTRIBUTES.DISABLEBLINKFEATURES, webViewImpl);
}
}
// Attribute that specifies the web preferences to be enabled.
class WebPreferencesAttribute extends WebViewAttribute {
constructor (webViewImpl: WebViewImpl) {
super(WEB_VIEW_ATTRIBUTES.WEBPREFERENCES, webViewImpl);
}
}
2016-01-14 18:35:29 +00:00
// Sets up all of the webview attributes.
export function setupWebViewAttributes (self: WebViewImpl) {
return new Map<string, WebViewAttribute>([
[WEB_VIEW_ATTRIBUTES.PARTITION, new PartitionAttribute(self)],
[WEB_VIEW_ATTRIBUTES.SRC, new SrcAttribute(self)],
[WEB_VIEW_ATTRIBUTES.HTTPREFERRER, new HttpReferrerAttribute(self)],
[WEB_VIEW_ATTRIBUTES.USERAGENT, new UserAgentAttribute(self)],
[WEB_VIEW_ATTRIBUTES.NODEINTEGRATION, new BooleanAttribute(WEB_VIEW_ATTRIBUTES.NODEINTEGRATION, self)],
[WEB_VIEW_ATTRIBUTES.NODEINTEGRATIONINSUBFRAMES, new BooleanAttribute(WEB_VIEW_ATTRIBUTES.NODEINTEGRATIONINSUBFRAMES, self)],
[WEB_VIEW_ATTRIBUTES.PLUGINS, new BooleanAttribute(WEB_VIEW_ATTRIBUTES.PLUGINS, self)],
[WEB_VIEW_ATTRIBUTES.DISABLEWEBSECURITY, new BooleanAttribute(WEB_VIEW_ATTRIBUTES.DISABLEWEBSECURITY, self)],
[WEB_VIEW_ATTRIBUTES.ALLOWPOPUPS, new BooleanAttribute(WEB_VIEW_ATTRIBUTES.ALLOWPOPUPS, self)],
[WEB_VIEW_ATTRIBUTES.PRELOAD, new PreloadAttribute(self)],
[WEB_VIEW_ATTRIBUTES.BLINKFEATURES, new BlinkFeaturesAttribute(self)],
[WEB_VIEW_ATTRIBUTES.DISABLEBLINKFEATURES, new DisableBlinkFeaturesAttribute(self)],
[WEB_VIEW_ATTRIBUTES.WEBPREFERENCES, new WebPreferencesAttribute(self)]
]);
}