electron/lib/browser/api/web-contents.js

285 lines
8.2 KiB
JavaScript
Raw Normal View History

'use strict'
2016-01-13 03:55:49 +00:00
const {EventEmitter} = require('events')
2016-06-13 15:59:03 +00:00
const {app, ipcMain, session, Menu, NavigationController} = require('electron')
// session is not used here, the purpose is to make sure session is initalized
// before the webContents module.
session
2016-01-12 02:40:23 +00:00
const binding = process.atomBinding('web_contents')
const debuggerBinding = process.atomBinding('debugger')
2016-01-12 02:40:23 +00:00
let nextId = 0
const getNextId = function () {
return ++nextId
}
2016-01-12 02:40:23 +00:00
2016-06-01 06:24:53 +00:00
// Stock page sizes
const PDFPageSizes = {
2016-01-12 02:40:23 +00:00
A5: {
custom_display_name: 'A5',
2016-01-12 02:40:23 +00:00
height_microns: 210000,
name: 'ISO_A5',
2016-01-12 02:40:23 +00:00
width_microns: 148000
},
A4: {
custom_display_name: 'A4',
2016-01-12 02:40:23 +00:00
height_microns: 297000,
name: 'ISO_A4',
is_default: 'true',
2016-01-12 02:40:23 +00:00
width_microns: 210000
},
A3: {
custom_display_name: 'A3',
2016-01-12 02:40:23 +00:00
height_microns: 420000,
name: 'ISO_A3',
2016-01-12 02:40:23 +00:00
width_microns: 297000
},
Legal: {
custom_display_name: 'Legal',
2016-01-12 02:40:23 +00:00
height_microns: 355600,
name: 'NA_LEGAL',
2016-01-12 02:40:23 +00:00
width_microns: 215900
},
Letter: {
custom_display_name: 'Letter',
2016-01-12 02:40:23 +00:00
height_microns: 279400,
name: 'NA_LETTER',
2016-01-12 02:40:23 +00:00
width_microns: 215900
},
Tabloid: {
height_microns: 431800,
name: 'NA_LEDGER',
2016-01-12 02:40:23 +00:00
width_microns: 279400,
custom_display_name: 'Tabloid'
2016-01-12 02:40:23 +00:00
}
}
2016-01-12 02:40:23 +00:00
2016-06-01 06:24:53 +00:00
// Default printing setting
const defaultPrintingSetting = {
pageRage: [],
mediaSize: {},
landscape: false,
color: 2,
headerFooterEnabled: false,
marginsType: 0,
isFirstRequest: false,
requestID: getNextId(),
previewModifiable: true,
printToPDF: true,
printWithCloudPrint: false,
printWithPrivet: false,
printWithExtension: false,
deviceName: 'Save as PDF',
generateDraftData: true,
fitToPageEnabled: false,
duplex: 0,
copies: 1,
collate: true,
shouldPrintBackgrounds: false,
shouldPrintSelectionOnly: false
}
2016-01-13 03:55:49 +00:00
// Following methods are mapped to webFrame.
const webFrameMethods = [
'insertText',
'setZoomFactor',
'setZoomLevel',
2016-01-19 22:49:40 +00:00
'setZoomLevelLimits'
]
2016-01-13 03:55:49 +00:00
const webFrameMethodsWithResult = [
'getZoomFactor',
'getZoomLevel'
]
// Add JavaScript wrappers for WebContents class.
const wrapWebContents = function (webContents) {
2016-01-14 18:35:29 +00:00
// webContents is an EventEmitter.
Object.setPrototypeOf(webContents, EventEmitter.prototype)
2016-01-12 02:40:23 +00:00
// Every remote callback from renderer process would add a listenter to the
// render-view-deleted event, so ignore the listenters warning.
webContents.setMaxListeners(0)
2016-01-14 18:35:29 +00:00
// WebContents::send(channel, args..)
// WebContents::sendToAll(channel, args..)
const sendWrapper = (allFrames, channel, ...args) => {
if (channel == null) {
throw new Error('Missing required channel argument')
}
return webContents._send(allFrames, channel, args)
}
webContents.send = sendWrapper.bind(null, false)
webContents.sendToAll = sendWrapper.bind(null, true)
2016-01-14 18:35:29 +00:00
// The navigation controller.
const controller = new NavigationController(webContents)
for (const name in NavigationController.prototype) {
const method = NavigationController.prototype[name]
2016-01-12 02:40:23 +00:00
if (method instanceof Function) {
webContents[name] = function () {
return method.apply(controller, arguments)
}
2016-01-12 02:40:23 +00:00
}
}
const asyncWebFrameMethods = function (requestId, method, callback, ...args) {
this.send('ELECTRON_INTERNAL_RENDERER_ASYNC_WEB_FRAME_METHOD', requestId, method, args)
ipcMain.once(`ELECTRON_INTERNAL_BROWSER_ASYNC_WEB_FRAME_RESPONSE_${requestId}`, function (event, result) {
if (callback) callback(result)
})
}
const syncWebFrameMethods = function (requestId, method, callback, ...args) {
this.send('ELECTRON_INTERNAL_RENDERER_SYNC_WEB_FRAME_METHOD', requestId, method, args)
ipcMain.once(`ELECTRON_INTERNAL_BROWSER_SYNC_WEB_FRAME_RESPONSE_${requestId}`, function (event, result) {
if (callback) callback(result)
})
}
2016-01-13 03:55:49 +00:00
// Mapping webFrame methods.
for (const method of webFrameMethods) {
webContents[method] = function (...args) {
this.send('ELECTRON_INTERNAL_RENDERER_WEB_FRAME_METHOD', method, args)
}
2016-01-13 03:55:49 +00:00
}
for (const method of webFrameMethodsWithResult) {
webContents[method] = function (...args) {
const callback = args[args.length - 1]
const actualArgs = args.slice(0, args.length - 2)
syncWebFrameMethods.call(this, getNextId(), method, callback, ...actualArgs)
}
}
2016-02-22 14:00:21 +00:00
2016-01-13 04:11:46 +00:00
// Make sure webContents.executeJavaScript would run the code only when the
// webContents has been loaded.
webContents.executeJavaScript = function (code, hasUserGesture, callback) {
const requestId = getNextId()
if (typeof hasUserGesture === 'function') {
callback = hasUserGesture
hasUserGesture = false
}
if (this.getURL() && !this.isLoadingMainFrame()) {
asyncWebFrameMethods.call(this, requestId, 'executeJavaScript', callback, code, hasUserGesture)
2016-03-29 00:35:49 +00:00
} else {
this.once('did-finish-load', () => {
asyncWebFrameMethods.call(this, requestId, 'executeJavaScript', callback, code, hasUserGesture)
})
2016-03-29 00:35:49 +00:00
}
}
2016-01-13 04:11:46 +00:00
2016-01-14 18:35:29 +00:00
// Dispatch IPC messages to the ipc module.
webContents.on('ipc-message', function (event, [channel, ...args]) {
ipcMain.emit(channel, event, ...args)
})
webContents.on('ipc-message-sync', function (event, [channel, ...args]) {
2016-01-12 02:40:23 +00:00
Object.defineProperty(event, 'returnValue', {
set: function (value) {
return event.sendReply(JSON.stringify(value))
2016-03-30 21:06:50 +00:00
},
get: function () {}
})
ipcMain.emit(channel, event, ...args)
})
2016-01-12 02:40:23 +00:00
2016-01-14 18:35:29 +00:00
// Handle context menu action request from pepper plugin.
webContents.on('pepper-context-menu', function (event, params) {
2016-05-19 22:28:08 +00:00
const menu = Menu.buildFromTemplate(params.menu)
menu.popup(params.x, params.y)
})
2016-01-12 02:40:23 +00:00
// The devtools requests the webContents to reload.
webContents.on('devtools-reload-page', function () {
webContents.reload()
})
2016-01-14 18:35:29 +00:00
// Delays the page-title-updated event to next tick.
webContents.on('-page-title-updated', function (...args) {
setImmediate(() => {
this.emit.apply(this, ['page-title-updated'].concat(args))
})
})
2016-01-12 02:40:23 +00:00
2016-03-29 00:40:40 +00:00
webContents.printToPDF = function (options, callback) {
2016-06-01 06:24:53 +00:00
const printingSetting = Object.assign({}, defaultPrintingSetting)
2016-01-12 02:40:23 +00:00
if (options.landscape) {
printingSetting.landscape = options.landscape
2016-01-12 02:40:23 +00:00
}
if (options.marginsType) {
printingSetting.marginsType = options.marginsType
2016-01-12 02:40:23 +00:00
}
if (options.printSelectionOnly) {
printingSetting.shouldPrintSelectionOnly = options.printSelectionOnly
2016-01-12 02:40:23 +00:00
}
if (options.printBackground) {
printingSetting.shouldPrintBackgrounds = options.printBackground
2016-01-12 02:40:23 +00:00
}
2016-05-31 16:59:07 +00:00
if (options.pageSize) {
2016-06-01 06:24:53 +00:00
const pageSize = options.pageSize
if (typeof pageSize === 'object') {
if (!pageSize.height || !pageSize.width) {
return callback(new Error('Must define height and width for pageSize'))
}
2016-05-31 16:59:07 +00:00
// Dimensions in Microns
// 1 meter = 10^6 microns
printingSetting.mediaSize = {
name: 'CUSTOM',
2016-06-01 06:24:53 +00:00
custom_display_name: 'Custom',
height_microns: pageSize.height,
width_microns: pageSize.width
2016-05-31 16:59:07 +00:00
}
2016-06-01 06:24:53 +00:00
} else if (PDFPageSizes[pageSize]) {
printingSetting.mediaSize = PDFPageSizes[pageSize]
2016-05-31 16:59:07 +00:00
} else {
2016-06-01 06:24:53 +00:00
return callback(new Error(`Does not support pageSize with ${pageSize}`))
2016-05-31 16:59:07 +00:00
}
2016-06-01 06:24:53 +00:00
} else {
printingSetting.mediaSize = PDFPageSizes['A4']
2016-01-12 02:40:23 +00:00
}
2016-05-31 16:59:07 +00:00
2016-06-01 06:24:53 +00:00
this._printToPDF(printingSetting, callback)
}
2016-06-13 15:59:03 +00:00
app.emit('web-contents-created', {}, webContents)
}
2016-01-12 02:40:23 +00:00
binding._setWrapWebContents(wrapWebContents)
// Add JavaScript wrappers for Debugger class.
const wrapDebugger = function (webContentsDebugger) {
2016-01-23 04:02:21 +00:00
// debugger is an EventEmitter.
Object.setPrototypeOf(webContentsDebugger, EventEmitter.prototype)
}
2016-01-23 04:02:21 +00:00
debuggerBinding._setWrapDebugger(wrapDebugger)
2016-01-12 02:40:23 +00:00
2016-06-13 15:59:03 +00:00
module.exports = {
2016-06-13 16:06:42 +00:00
create (options = {}) {
2016-06-13 15:59:03 +00:00
return binding.create(options)
},
fromId (id) {
return binding.fromId(id)
2016-07-13 15:54:40 +00:00
},
getFocusedWebContents () {
let focused = null
2016-07-14 15:59:49 +00:00
for (let contents of binding.getAllWebContents()) {
if (!contents.isFocused()) continue
2016-07-13 21:49:25 +00:00
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
2016-07-14 15:59:49 +00:00
},
getAllWebContents () {
return binding.getAllWebContents()
2016-06-13 15:59:03 +00:00
}
}