import { webFrame, IpcMessageEvent } from 'electron' import { ipcRendererInternal } from '@electron/internal/renderer/ipc-renderer-internal' import { invoke, invokeSync } from '@electron/internal/renderer/ipc-renderer-internal-utils' import { WebViewImpl } from '@electron/internal/renderer/web-view/web-view-impl' const WEB_VIEW_EVENTS: Record> = { 'load-commit': ['url', 'isMainFrame'], 'did-attach': [], 'did-finish-load': [], 'did-fail-load': ['errorCode', 'errorDescription', 'validatedURL', 'isMainFrame', 'frameProcessId', 'frameRoutingId'], 'did-frame-finish-load': ['isMainFrame', 'frameProcessId', 'frameRoutingId'], 'did-start-loading': [], 'did-stop-loading': [], 'dom-ready': [], 'console-message': ['level', 'message', 'line', 'sourceId'], 'context-menu': ['params'], 'devtools-opened': [], 'devtools-closed': [], 'devtools-focused': [], 'new-window': ['url', 'frameName', 'disposition', 'options'], 'will-navigate': ['url'], 'did-start-navigation': ['url', 'isInPlace', 'isMainFrame', 'frameProcessId', 'frameRoutingId'], 'did-navigate': ['url', 'httpResponseCode', 'httpStatusText'], 'did-frame-navigate': ['url', 'httpResponseCode', 'httpStatusText', 'isMainFrame', 'frameProcessId', 'frameRoutingId'], 'did-navigate-in-page': ['url', 'isMainFrame', 'frameProcessId', 'frameRoutingId'], 'focus-change': ['focus', 'guestInstanceId'], 'close': [], 'crashed': [], 'plugin-crashed': ['name', 'version'], 'destroyed': [], 'page-title-updated': ['title', 'explicitSet'], 'page-favicon-updated': ['favicons'], 'enter-html-full-screen': [], 'leave-html-full-screen': [], 'media-started-playing': [], 'media-paused': [], 'found-in-page': ['result'], 'did-change-theme-color': ['themeColor'], 'update-target-url': ['url'] } const DEPRECATED_EVENTS: Record = { 'page-title-updated': 'page-title-set' } const dispatchEvent = function ( webView: WebViewImpl, eventName: string, eventKey: string, ...args: Array ) { if (DEPRECATED_EVENTS[eventName] != null) { dispatchEvent(webView, DEPRECATED_EVENTS[eventName], eventKey, ...args) } const domEvent = new Event(eventName) as ElectronInternal.WebViewEvent WEB_VIEW_EVENTS[eventKey].forEach((prop, index) => { (domEvent as any)[prop] = args[index] }) webView.dispatchEvent(domEvent) if (eventName === 'load-commit') { webView.onLoadCommit(domEvent) } else if (eventName === 'focus-change') { webView.onFocusChange() } } export function registerEvents (webView: WebViewImpl, viewInstanceId: number) { ipcRendererInternal.on(`ELECTRON_GUEST_VIEW_INTERNAL_DESTROY_GUEST-${viewInstanceId}`, function () { webView.guestInstanceId = undefined webView.reset() const domEvent = new Event('destroyed') webView.dispatchEvent(domEvent) }) ipcRendererInternal.on(`ELECTRON_GUEST_VIEW_INTERNAL_DISPATCH_EVENT-${viewInstanceId}`, function (event, eventName, ...args) { dispatchEvent(webView, eventName, eventName, ...args) }) ipcRendererInternal.on(`ELECTRON_GUEST_VIEW_INTERNAL_IPC_MESSAGE-${viewInstanceId}`, function (event, channel, ...args) { const domEvent = new Event('ipc-message') as IpcMessageEvent domEvent.channel = channel domEvent.args = args webView.dispatchEvent(domEvent) }) } export function deregisterEvents (viewInstanceId: number) { ipcRendererInternal.removeAllListeners(`ELECTRON_GUEST_VIEW_INTERNAL_DESTROY_GUEST-${viewInstanceId}`) ipcRendererInternal.removeAllListeners(`ELECTRON_GUEST_VIEW_INTERNAL_DISPATCH_EVENT-${viewInstanceId}`) ipcRendererInternal.removeAllListeners(`ELECTRON_GUEST_VIEW_INTERNAL_IPC_MESSAGE-${viewInstanceId}`) } export function createGuest (params: Record): Promise { return invoke('ELECTRON_GUEST_VIEW_MANAGER_CREATE_GUEST', params) } export function createGuestSync (params: Record): number { return invokeSync('ELECTRON_GUEST_VIEW_MANAGER_CREATE_GUEST', params) } export function attachGuest ( elementInstanceId: number, guestInstanceId: number, params: Record, contentWindow: Window ) { const embedderFrameId = (webFrame as ElectronInternal.WebFrameInternal).getWebFrameId(contentWindow) if (embedderFrameId < 0) { // this error should not happen. throw new Error('Invalid embedder frame') } invoke('ELECTRON_GUEST_VIEW_MANAGER_ATTACH_GUEST', embedderFrameId, elementInstanceId, guestInstanceId, params) } export const guestViewInternalModule = { deregisterEvents, createGuest, createGuestSync, attachGuest }