123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507 |
- 'use strict'
- const features = process.atomBinding('features')
- const { EventEmitter } = require('events')
- const electron = require('electron')
- const path = require('path')
- const url = require('url')
- const v8Util = process.atomBinding('v8_util')
- const { app, ipcMain, session, NavigationController, deprecate } = electron
- const ipcMainInternal = require('@electron/internal/browser/ipc-main-internal')
- const errorUtils = require('@electron/internal/common/error-utils')
- // session is not used here, the purpose is to make sure session is initalized
- // before the webContents module.
- // eslint-disable-next-line
- session
- let nextId = 0
- const getNextId = function () {
- return ++nextId
- }
- // Stock page sizes
- const PDFPageSizes = {
- A5: {
- custom_display_name: 'A5',
- height_microns: 210000,
- name: 'ISO_A5',
- width_microns: 148000
- },
- A4: {
- custom_display_name: 'A4',
- height_microns: 297000,
- name: 'ISO_A4',
- is_default: 'true',
- width_microns: 210000
- },
- A3: {
- custom_display_name: 'A3',
- height_microns: 420000,
- name: 'ISO_A3',
- width_microns: 297000
- },
- Legal: {
- custom_display_name: 'Legal',
- height_microns: 355600,
- name: 'NA_LEGAL',
- width_microns: 215900
- },
- Letter: {
- custom_display_name: 'Letter',
- height_microns: 279400,
- name: 'NA_LETTER',
- width_microns: 215900
- },
- Tabloid: {
- height_microns: 431800,
- name: 'NA_LEDGER',
- width_microns: 279400,
- custom_display_name: 'Tabloid'
- }
- }
- // Default printing setting
- const defaultPrintingSetting = {
- pageRage: [],
- mediaSize: {},
- landscape: false,
- color: 2,
- headerFooterEnabled: false,
- marginsType: 0,
- isFirstRequest: false,
- requestID: getNextId(),
- previewUIID: 0,
- previewModifiable: true,
- printToPDF: true,
- printWithCloudPrint: false,
- printWithPrivet: false,
- printWithExtension: false,
- deviceName: 'Save as PDF',
- generateDraftData: true,
- fitToPageEnabled: false,
- scaleFactor: 1,
- dpiHorizontal: 72,
- dpiVertical: 72,
- rasterizePDF: false,
- duplex: 0,
- copies: 1,
- collate: true,
- shouldPrintBackgrounds: false,
- shouldPrintSelectionOnly: false
- }
- // JavaScript implementations of WebContents.
- const binding = process.atomBinding('web_contents')
- const { WebContents } = binding
- Object.setPrototypeOf(NavigationController.prototype, EventEmitter.prototype)
- Object.setPrototypeOf(WebContents.prototype, NavigationController.prototype)
- // WebContents::send(channel, args..)
- // WebContents::sendToAll(channel, args..)
- WebContents.prototype.send = function (channel, ...args) {
- if (typeof channel !== 'string') {
- throw new Error('Missing required channel argument')
- }
- const internal = false
- const sendToAll = false
- return this._send(internal, sendToAll, channel, args)
- }
- WebContents.prototype.sendToAll = function (channel, ...args) {
- if (typeof channel !== 'string') {
- throw new Error('Missing required channel argument')
- }
- const internal = false
- const sendToAll = true
- return this._send(internal, sendToAll, channel, args)
- }
- WebContents.prototype._sendInternal = function (channel, ...args) {
- if (typeof channel !== 'string') {
- throw new Error('Missing required channel argument')
- }
- const internal = true
- const sendToAll = false
- return this._send(internal, sendToAll, channel, args)
- }
- WebContents.prototype._sendInternalToAll = function (channel, ...args) {
- if (typeof channel !== 'string') {
- throw new Error('Missing required channel argument')
- }
- const internal = true
- const sendToAll = true
- return this._send(internal, sendToAll, channel, args)
- }
- // Following methods are mapped to webFrame.
- const webFrameMethods = [
- 'insertCSS',
- 'insertText',
- 'setLayoutZoomLevelLimits',
- 'setVisualZoomLevelLimits'
- ]
- const asyncWebFrameMethods = function (requestId, method, callback, ...args) {
- return new Promise((resolve, reject) => {
- ipcMainInternal.once(`ELECTRON_INTERNAL_BROWSER_ASYNC_WEB_FRAME_RESPONSE_${requestId}`, function (event, error, result) {
- if (error == null) {
- if (typeof callback === 'function') callback(result)
- resolve(result)
- } else {
- reject(errorUtils.deserialize(error))
- }
- })
- this._sendInternal('ELECTRON_INTERNAL_RENDERER_ASYNC_WEB_FRAME_METHOD', requestId, method, args)
- })
- }
- for (const method of webFrameMethods) {
- WebContents.prototype[method] = function (...args) {
- this._sendInternal('ELECTRON_INTERNAL_RENDERER_WEB_FRAME_METHOD', method, args)
- }
- }
- // Make sure WebContents::executeJavaScript would run the code only when the
- // WebContents has been loaded.
- WebContents.prototype.executeJavaScript = function (code, hasUserGesture, callback) {
- const requestId = getNextId()
- if (typeof hasUserGesture === 'function') {
- // Shift.
- callback = hasUserGesture
- hasUserGesture = null
- }
- if (hasUserGesture == null) {
- hasUserGesture = false
- }
- if (this.getURL() && !this.isLoadingMainFrame()) {
- return asyncWebFrameMethods.call(this, requestId, 'executeJavaScript', callback, code, hasUserGesture)
- } else {
- return new Promise((resolve, reject) => {
- this.once('did-stop-loading', () => {
- asyncWebFrameMethods.call(this, requestId, 'executeJavaScript', callback, code, hasUserGesture).then(resolve).catch(reject)
- })
- })
- }
- }
- WebContents.prototype.takeHeapSnapshot = function (filePath) {
- return new Promise((resolve, reject) => {
- const channel = `ELECTRON_TAKE_HEAP_SNAPSHOT_RESULT_${getNextId()}`
- ipcMainInternal.once(channel, (event, success) => {
- if (success) {
- resolve()
- } else {
- reject(new Error('takeHeapSnapshot failed'))
- }
- })
- if (!this._takeHeapSnapshot(filePath, channel)) {
- ipcMainInternal.emit(channel, false)
- }
- })
- }
- // Translate the options of printToPDF.
- WebContents.prototype.printToPDF = function (options, callback) {
- const printingSetting = Object.assign({}, defaultPrintingSetting)
- if (options.landscape) {
- printingSetting.landscape = options.landscape
- }
- if (options.marginsType) {
- printingSetting.marginsType = options.marginsType
- }
- if (options.printSelectionOnly) {
- printingSetting.shouldPrintSelectionOnly = options.printSelectionOnly
- }
- if (options.printBackground) {
- printingSetting.shouldPrintBackgrounds = options.printBackground
- }
- if (options.pageSize) {
- const pageSize = options.pageSize
- if (typeof pageSize === 'object') {
- if (!pageSize.height || !pageSize.width) {
- return callback(new Error('Must define height and width for pageSize'))
- }
- // Dimensions in Microns
- // 1 meter = 10^6 microns
- printingSetting.mediaSize = {
- name: 'CUSTOM',
- custom_display_name: 'Custom',
- height_microns: Math.ceil(pageSize.height),
- width_microns: Math.ceil(pageSize.width)
- }
- } else if (PDFPageSizes[pageSize]) {
- printingSetting.mediaSize = PDFPageSizes[pageSize]
- } else {
- return callback(new Error(`Does not support pageSize with ${pageSize}`))
- }
- } else {
- printingSetting.mediaSize = PDFPageSizes['A4']
- }
- // Chromium expects this in a 0-100 range number, not as float
- printingSetting.scaleFactor *= 100
- if (features.isPrintingEnabled()) {
- this._printToPDF(printingSetting, callback)
- } else {
- console.error('Error: Printing feature is disabled.')
- }
- }
- WebContents.prototype.print = function (...args) {
- if (features.isPrintingEnabled()) {
- this._print(...args)
- } else {
- console.error('Error: Printing feature is disabled.')
- }
- }
- WebContents.prototype.getPrinters = function () {
- if (features.isPrintingEnabled()) {
- return this._getPrinters()
- } else {
- console.error('Error: Printing feature is disabled.')
- }
- }
- WebContents.prototype.getZoomLevel = function (callback) {
- if (typeof callback !== 'function') {
- throw new Error('Must pass function as an argument')
- }
- process.nextTick(() => {
- const zoomLevel = this._getZoomLevel()
- callback(zoomLevel)
- })
- }
- WebContents.prototype.loadFile = function (filePath, options = {}) {
- if (typeof filePath !== 'string') {
- throw new Error('Must pass filePath as a string')
- }
- const { query, search, hash } = options
- return this.loadURL(url.format({
- protocol: 'file',
- slashes: true,
- pathname: path.resolve(app.getAppPath(), filePath),
- query,
- search,
- hash
- }))
- }
- WebContents.prototype.getZoomFactor = function (callback) {
- if (typeof callback !== 'function') {
- throw new Error('Must pass function as an argument')
- }
- process.nextTick(() => {
- const zoomFactor = this._getZoomFactor()
- callback(zoomFactor)
- })
- }
- WebContents.prototype.findInPage = function (text, options = {}) {
- // TODO (nitsakh): Remove in 5.0
- if (options.wordStart != null || options.medialCapitalAtWordStart != null) {
- deprecate.log('wordStart and medialCapitalAtWordStart options are deprecated')
- }
- return this._findInPage(text, options)
- }
- const safeProtocols = new Set([
- 'chrome-devtools:',
- 'chrome-extension:'
- ])
- const isWebContentsTrusted = function (contents) {
- const pageURL = contents._getURL()
- const { protocol } = url.parse(pageURL)
- return safeProtocols.has(protocol)
- }
- // Add JavaScript wrappers for WebContents class.
- WebContents.prototype._init = function () {
- // The navigation controller.
- NavigationController.call(this, this)
- // Every remote callback from renderer process would add a listenter to the
- // render-view-deleted event, so ignore the listenters warning.
- this.setMaxListeners(0)
- // Dispatch IPC messages to the ipc module.
- this.on('ipc-message', function (event, [channel, ...args]) {
- ipcMain.emit(channel, event, ...args)
- })
- this.on('ipc-message-sync', function (event, [channel, ...args]) {
- Object.defineProperty(event, 'returnValue', {
- set: function (value) {
- return event.sendReply([value])
- },
- get: function () {}
- })
- ipcMain.emit(channel, event, ...args)
- })
- this.on('ipc-internal-message', function (event, [channel, ...args]) {
- ipcMainInternal.emit(channel, event, ...args)
- })
- this.on('ipc-internal-message-sync', function (event, [channel, ...args]) {
- Object.defineProperty(event, 'returnValue', {
- set: function (value) {
- return event.sendReply([value])
- },
- get: function () {}
- })
- ipcMainInternal.emit(channel, event, ...args)
- })
- // Handle context menu action request from pepper plugin.
- this.on('pepper-context-menu', function (event, params, callback) {
- // Access Menu via electron.Menu to prevent circular require.
- const menu = electron.Menu.buildFromTemplate(params.menu)
- menu.popup({
- window: event.sender.getOwnerBrowserWindow(),
- x: params.x,
- y: params.y,
- callback
- })
- })
- const forwardedEvents = [
- 'remote-require',
- 'remote-get-global',
- 'remote-get-builtin',
- 'remote-get-current-window',
- 'remote-get-current-web-contents',
- 'remote-get-guest-web-contents'
- ]
- for (const eventName of forwardedEvents) {
- this.on(eventName, (event, ...args) => {
- if (!isWebContentsTrusted(event.sender)) {
- app.emit(eventName, event, this, ...args)
- }
- })
- }
- deprecate.event(this, 'did-get-response-details', '-did-get-response-details')
- deprecate.event(this, 'did-get-redirect-request', '-did-get-redirect-request')
- // The devtools requests the webContents to reload.
- this.on('devtools-reload-page', function () {
- this.reload()
- })
- // Handle window.open for BrowserWindow and BrowserView.
- if (['browserView', 'window'].includes(this.getType())) {
- // Make new windows requested by links behave like "window.open"
- this.webContents.on('-new-window', (event, url, frameName, disposition,
- additionalFeatures, postData,
- referrer) => {
- const options = {
- show: true,
- width: 800,
- height: 600
- }
- ipcMainInternal.emit('ELECTRON_GUEST_WINDOW_MANAGER_INTERNAL_WINDOW_OPEN',
- event, url, referrer, frameName, disposition,
- options, additionalFeatures, postData)
- })
- this.webContents.on('-web-contents-created', (event, webContents, url,
- frameName) => {
- v8Util.setHiddenValue(webContents, 'url-framename', { url, frameName })
- })
- // Create a new browser window for the native implementation of
- // "window.open", used in sandbox and nativeWindowOpen mode
- this.webContents.on('-add-new-contents', (event, webContents, disposition,
- userGesture, left, top, width,
- height) => {
- const urlFrameName = v8Util.getHiddenValue(webContents, 'url-framename')
- if ((disposition !== 'foreground-tab' && disposition !== 'new-window' &&
- disposition !== 'background-tab') || !urlFrameName) {
- event.preventDefault()
- return
- }
- if (webContents.getLastWebPreferences().nodeIntegration === true) {
- const message =
- 'Enabling Node.js integration in child windows opened with the ' +
- '"nativeWindowOpen" option will cause memory leaks, please turn off ' +
- 'the "nodeIntegration" option.\\n' +
- 'From 5.x child windows opened with the "nativeWindowOpen" option ' +
- 'will always have Node.js integration disabled.\\n' +
- 'See https://github.com/electron/electron/pull/15076 for more.'
- // console is only available after DOM is created.
- const printWarning = () => this.webContents.executeJavaScript(`console.warn('${message}')`)
- if (this.webContents.isDomReady()) {
- printWarning()
- } else {
- this.webContents.once('dom-ready', printWarning)
- }
- }
- const { url, frameName } = urlFrameName
- v8Util.deleteHiddenValue(webContents, 'url-framename')
- const options = {
- show: true,
- x: left,
- y: top,
- width: width || 800,
- height: height || 600,
- webContents
- }
- const referrer = { url: '', policy: 'default' }
- ipcMainInternal.emit('ELECTRON_GUEST_WINDOW_MANAGER_INTERNAL_WINDOW_OPEN',
- event, url, referrer, frameName, disposition, options)
- })
- }
- app.emit('web-contents-created', {}, this)
- }
- // JavaScript wrapper of Debugger.
- const { Debugger } = process.atomBinding('debugger')
- Object.setPrototypeOf(Debugger.prototype, EventEmitter.prototype)
- // Public APIs.
- module.exports = {
- create (options = {}) {
- return binding.create(options)
- },
- fromId (id) {
- return binding.fromId(id)
- },
- getFocusedWebContents () {
- let focused = null
- for (const contents of binding.getAllWebContents()) {
- if (!contents.isFocused()) continue
- if (focused == null) focused = contents
- // Return webview web contents which may be embedded inside another
- // web contents that is also reporting as focused
- if (contents.getType() === 'webview') return contents
- }
- return focused
- },
- getAllWebContents () {
- return binding.getAllWebContents()
- }
- }
|