2016-03-25 20:03:49 +00:00
const assert = require ( 'assert' )
const path = require ( 'path' )
const http = require ( 'http' )
const url = require ( 'url' )
2017-02-03 20:21:46 +00:00
const { ipcRenderer , remote } = require ( 'electron' )
2017-01-16 20:56:39 +00:00
const { app , session , getGuestWebContents , ipcMain , BrowserWindow , webContents } = remote
2016-11-03 23:43:16 +00:00
const { closeWindow } = require ( './window-helpers' )
2016-03-30 17:33:15 +00:00
2017-07-11 11:12:57 +00:00
const isCI = remote . getGlobal ( 'isCi' )
2017-05-25 23:40:15 +00:00
const nativeModulesEnabled = remote . getGlobal ( 'nativeModulesEnabled' )
2017-01-16 20:56:39 +00:00
2017-11-23 22:22:43 +00:00
/* Most of the APIs here don't use standard callbacks */
/* eslint-disable standard/no-callback-literal */
2017-05-07 05:14:52 +00:00
describe ( '<webview> tag' , function ( ) {
2016-12-22 21:31:30 +00:00
this . timeout ( 3 * 60 * 1000 )
2016-02-05 18:27:05 +00:00
2017-10-27 20:45:58 +00:00
const fixtures = path . join ( _ _dirname , 'fixtures' )
let webview = null
2016-05-17 08:04:13 +00:00
let w = null
2016-02-05 18:27:05 +00:00
2017-10-27 20:45:58 +00:00
beforeEach ( ( ) => {
2016-03-28 20:47:31 +00:00
webview = new WebView ( )
2016-03-25 20:03:49 +00:00
} )
2016-02-05 18:27:05 +00:00
2017-10-27 20:45:58 +00:00
afterEach ( ( ) => {
2016-09-30 16:55:24 +00:00
if ( ! document . body . contains ( webview ) ) {
document . body . appendChild ( webview )
2016-01-12 02:40:23 +00:00
}
2016-09-30 16:55:24 +00:00
webview . remove ( )
2017-10-27 20:45:58 +00:00
return closeWindow ( w ) . then ( ( ) => { w = null } )
2016-03-25 20:03:49 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'works without script tag in page' , ( done ) => {
2016-05-17 08:04:13 +00:00
w = new BrowserWindow ( { show : false } )
2017-10-27 20:45:58 +00:00
ipcMain . once ( 'pong' , ( ) => done ( ) )
2016-04-12 06:10:26 +00:00
w . loadURL ( 'file://' + fixtures + '/pages/webview-no-script.html' )
2017-05-07 05:10:42 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'is disabled when nodeIntegration is disabled' , ( done ) => {
2017-05-07 05:10:42 +00:00
w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : false ,
preload : path . join ( fixtures , 'module' , 'preload-webview.js' )
}
} )
2017-10-27 20:45:58 +00:00
ipcMain . once ( 'webview' , ( event , type ) => {
2017-05-07 05:10:42 +00:00
if ( type === 'undefined' ) {
done ( )
} else {
done ( 'WebView still exists' )
}
} )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-no-script.html ` )
2017-05-07 05:10:42 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'is enabled when the webviewTag option is enabled and the nodeIntegration option is disabled' , ( done ) => {
2017-05-07 05:10:42 +00:00
w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : false ,
preload : path . join ( fixtures , 'module' , 'preload-webview.js' ) ,
2017-05-17 20:09:24 +00:00
webviewTag : true
2017-05-07 05:10:42 +00:00
}
} )
2017-10-27 20:45:58 +00:00
ipcMain . once ( 'webview' , ( event , type ) => {
2017-05-07 05:10:42 +00:00
if ( type !== 'undefined' ) {
done ( )
} else {
done ( 'WebView is not created' )
}
} )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-no-script.html ` )
2016-04-12 06:10:26 +00:00
} )
2017-10-27 20:45:58 +00:00
describe ( 'src attribute' , ( ) => {
it ( 'specifies the page to load' , ( done ) => {
webview . addEventListener ( 'console-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'a' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'navigates to new page when changed' , ( done ) => {
const listener = ( ) => {
webview . src = ` file:// ${ fixtures } /pages/b.html `
webview . addEventListener ( 'console-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'b' )
done ( )
} )
webview . removeEventListener ( 'did-finish-load' , listener )
}
webview . addEventListener ( 'did-finish-load' , listener )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2016-09-08 23:40:31 +00:00
2017-10-27 20:45:58 +00:00
it ( 'resolves relative URLs' , ( done ) => {
const listener = ( e ) => {
2016-09-09 00:38:40 +00:00
assert . equal ( e . message , 'Window script is loaded before preload script' )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
webview . src = '../fixtures/pages/e.html'
document . body . appendChild ( webview )
2016-09-08 23:56:29 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'ignores empty values' , ( ) => {
2016-09-08 23:40:31 +00:00
assert . equal ( webview . src , '' )
webview . src = ''
assert . equal ( webview . src , '' )
2016-09-08 23:48:30 +00:00
webview . src = null
assert . equal ( webview . src , '' )
webview . src = undefined
assert . equal ( webview . src , '' )
2016-09-08 23:40:31 +00:00
} )
2016-03-25 20:03:49 +00:00
} )
2017-10-27 20:45:58 +00:00
describe ( 'nodeintegration attribute' , ( ) => {
it ( 'inserts no node symbols when not set' , ( done ) => {
webview . addEventListener ( 'console-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'undefined undefined undefined undefined' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/c.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'inserts node symbols when set' , ( done ) => {
webview . addEventListener ( 'console-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'function object object' )
done ( )
} )
webview . setAttribute ( 'nodeintegration' , 'on' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/d.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-11-15 21:05:46 +00:00
it ( 'loads node symbols after POST navigation when set' , function ( done ) {
2017-03-27 16:15:40 +00:00
// FIXME Figure out why this is timing out on AppVeyor
2017-11-15 21:05:46 +00:00
if ( process . env . APPVEYOR === 'True' ) {
// FIXME(alexeykuzmin): Skip the test.
// this.skip()
return done ( )
}
2017-03-27 16:15:40 +00:00
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'console-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'function object object' )
done ( )
} )
webview . setAttribute ( 'nodeintegration' , 'on' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/post.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
2016-03-30 17:33:15 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'disables node integration on child windows when it is disabled on the webview' , ( done ) => {
app . once ( 'browser-window-created' , ( event , window ) => {
2016-03-30 18:17:24 +00:00
assert . equal ( window . webContents . getWebPreferences ( ) . nodeIntegration , false )
2016-03-30 17:33:15 +00:00
done ( )
} )
webview . setAttribute ( 'allowpopups' , 'on' )
2016-04-01 23:53:44 +00:00
webview . src = url . format ( {
2016-03-30 17:33:15 +00:00
pathname : ` ${ fixtures } /pages/webview-opener-no-node-integration.html ` ,
protocol : 'file' ,
query : {
p : ` ${ fixtures } /pages/window-opener-node.html `
} ,
slashes : true
} )
document . body . appendChild ( webview )
2016-03-25 20:03:49 +00:00
} )
2016-02-05 18:27:05 +00:00
2017-10-27 20:45:58 +00:00
it ( 'loads native modules when navigation happens' , ( done ) => {
2017-05-25 23:40:15 +00:00
if ( ! nativeModulesEnabled ) return done ( )
2017-10-27 20:45:58 +00:00
const listener = ( ) => {
2017-05-25 23:40:15 +00:00
webview . removeEventListener ( 'did-finish-load' , listener )
2017-10-27 20:45:58 +00:00
const listener2 = ( e ) => {
2017-05-25 23:40:15 +00:00
assert . equal ( e . message , 'function' )
done ( )
2016-03-25 20:03:49 +00:00
}
2017-05-25 23:40:15 +00:00
webview . addEventListener ( 'console-message' , listener2 )
webview . reload ( )
}
webview . addEventListener ( 'did-finish-load' , listener )
webview . setAttribute ( 'nodeintegration' , 'on' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/native-module.html `
2017-05-25 23:40:15 +00:00
document . body . appendChild ( webview )
} )
2016-03-25 20:03:49 +00:00
} )
2017-10-27 20:45:58 +00:00
describe ( 'preload attribute' , ( ) => {
it ( 'loads the script before other scripts in window' , ( done ) => {
const listener = ( e ) => {
2017-02-06 21:54:41 +00:00
assert . equal ( e . message , 'function object object function' )
2016-03-25 20:03:49 +00:00
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
2017-10-27 20:45:58 +00:00
webview . setAttribute ( 'preload' , ` ${ fixtures } /module/preload.js ` )
webview . src = ` file:// ${ fixtures } /pages/e.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'preload script can still use "process" and "Buffer" when nodeintegration is off' , ( done ) => {
webview . addEventListener ( 'console-message' , ( e ) => {
2017-02-06 21:54:41 +00:00
assert . equal ( e . message , 'object undefined object function' )
2016-03-25 20:03:49 +00:00
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . setAttribute ( 'preload' , ` ${ fixtures } /module/preload-node-off.js ` )
webview . src = ` file:// ${ fixtures } /api/blank.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'preload script can require modules that still use "process" and "Buffer" when nodeintegration is off' , ( done ) => {
webview . addEventListener ( 'console-message' , ( e ) => {
2017-02-23 19:35:28 +00:00
assert . equal ( e . message , 'object undefined object function undefined' )
2017-02-23 19:26:37 +00:00
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . setAttribute ( 'preload' , ` ${ fixtures } /module/preload-node-off-wrapper.js ` )
webview . src = ` file:// ${ fixtures } /api/blank.html `
2017-02-23 19:26:37 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'receives ipc message in preload script' , ( done ) => {
const message = 'boom!'
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . channel , 'pong' )
assert . deepEqual ( e . args , [ message ] )
webview . removeEventListener ( 'ipc-message' , listener )
done ( )
}
2017-10-27 20:45:58 +00:00
const listener2 = ( ) => {
2016-03-25 20:03:49 +00:00
webview . send ( 'ping' , message )
webview . removeEventListener ( 'did-finish-load' , listener2 )
}
webview . addEventListener ( 'ipc-message' , listener )
webview . addEventListener ( 'did-finish-load' , listener2 )
2017-10-27 20:45:58 +00:00
webview . setAttribute ( 'preload' , ` ${ fixtures } /module/preload-ipc.js ` )
webview . src = ` file:// ${ fixtures } /pages/e.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2016-04-12 05:57:40 +00:00
2017-10-27 20:45:58 +00:00
it ( 'works without script tag in page' , ( done ) => {
const listener = ( e ) => {
2017-02-06 21:54:41 +00:00
assert . equal ( e . message , 'function object object function' )
2016-04-12 05:57:40 +00:00
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
2017-10-27 20:45:58 +00:00
webview . setAttribute ( 'preload' , ` ${ fixtures } /module/preload.js ` )
webview . src = ` file:// ${ fixtures } pages/base-page.html `
2016-04-12 05:57:40 +00:00
document . body . appendChild ( webview )
} )
2016-09-08 23:40:31 +00:00
2017-10-27 20:45:58 +00:00
it ( 'resolves relative URLs' , ( done ) => {
const listener = ( e ) => {
2017-02-06 21:54:41 +00:00
assert . equal ( e . message , 'function object object function' )
2016-09-09 00:38:40 +00:00
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/e.html `
2016-09-09 00:38:40 +00:00
webview . preload = '../fixtures/module/preload.js'
document . body . appendChild ( webview )
2016-09-08 23:56:29 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'ignores empty values' , ( ) => {
2016-09-08 23:40:31 +00:00
assert . equal ( webview . preload , '' )
webview . preload = ''
assert . equal ( webview . preload , '' )
2016-09-08 23:48:30 +00:00
webview . preload = null
assert . equal ( webview . preload , '' )
webview . preload = undefined
assert . equal ( webview . preload , '' )
2016-09-08 23:40:31 +00:00
} )
2016-03-25 20:03:49 +00:00
} )
2017-10-27 20:45:58 +00:00
describe ( 'httpreferrer attribute' , ( ) => {
it ( 'sets the referrer url' , ( done ) => {
const referrer = 'http://github.com/'
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , referrer )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
webview . setAttribute ( 'httpreferrer' , referrer )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/referrer.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'useragent attribute' , ( ) => {
it ( 'sets the user agent' , ( done ) => {
const referrer = 'Mozilla/5.0 (Windows NT 6.1; WOW64; Trident/7.0; AS; rv:11.0) like Gecko'
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , referrer )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
webview . setAttribute ( 'useragent' , referrer )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/useragent.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'disablewebsecurity attribute' , ( ) => {
it ( 'does not disable web security when not set' , ( done ) => {
const jqueryPath = path . join ( _ _dirname , '/static/jquery-2.0.3.min.js' )
const src = ` <script src='file:// ${ jqueryPath } '></script> <script>console.log('ok');</script> `
const encoded = btoa ( unescape ( encodeURIComponent ( src ) ) )
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert ( /Not allowed to load local resource/ . test ( e . message ) )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
2017-10-27 20:45:58 +00:00
webview . src = ` data:text/html;base64, ${ encoded } `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'disables web security when set' , ( done ) => {
const jqueryPath = path . join ( _ _dirname , '/static/jquery-2.0.3.min.js' )
const src = ` <script src='file:// ${ jqueryPath } '></script> <script>console.log('ok');</script> `
const encoded = btoa ( unescape ( encodeURIComponent ( src ) ) )
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'ok' )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
webview . setAttribute ( 'disablewebsecurity' , '' )
2017-10-27 20:45:58 +00:00
webview . src = ` data:text/html;base64, ${ encoded } `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2016-05-30 06:20:53 +00:00
2017-10-27 20:45:58 +00:00
it ( 'does not break node integration' , ( done ) => {
webview . addEventListener ( 'console-message' , ( e ) => {
2016-05-30 06:20:53 +00:00
assert . equal ( e . message , 'function object object' )
done ( )
} )
webview . setAttribute ( 'nodeintegration' , 'on' )
webview . setAttribute ( 'disablewebsecurity' , '' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/d.html `
2016-05-30 06:20:53 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'does not break preload script' , ( done ) => {
const listener = ( e ) => {
2017-02-13 17:33:11 +00:00
assert . equal ( e . message , 'function object object function' )
2016-05-30 06:20:53 +00:00
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
webview . setAttribute ( 'disablewebsecurity' , '' )
2017-10-27 20:45:58 +00:00
webview . setAttribute ( 'preload' , ` ${ fixtures } /module/preload.js ` )
webview . src = ` file:// ${ fixtures } /pages/e.html `
2016-05-30 06:20:53 +00:00
document . body . appendChild ( webview )
} )
2016-03-25 20:03:49 +00:00
} )
2017-10-27 20:45:58 +00:00
describe ( 'partition attribute' , ( ) => {
it ( 'inserts no node symbols when not set' , ( done ) => {
webview . addEventListener ( 'console-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'undefined undefined undefined undefined' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/c.html `
2016-03-25 20:03:49 +00:00
webview . partition = 'test1'
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'inserts node symbols when set' , ( done ) => {
webview . addEventListener ( 'console-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'function object object' )
done ( )
} )
webview . setAttribute ( 'nodeintegration' , 'on' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/d.html `
2016-03-25 20:03:49 +00:00
webview . partition = 'test2'
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'isolates storage for different id' , ( done ) => {
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , ' 0' )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
window . localStorage . setItem ( 'test' , 'one' )
webview . addEventListener ( 'console-message' , listener )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/partition/one.html `
2016-03-25 20:03:49 +00:00
webview . partition = 'test3'
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'uses current session storage when no id is provided' , ( done ) => {
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'one 1' )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
window . localStorage . setItem ( 'test' , 'one' )
webview . addEventListener ( 'console-message' , listener )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/partition/one.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'allowpopups attribute' , ( ) => {
it ( 'can not open new window when not set' , ( done ) => {
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'null' )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/window-open-hide.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'can open new window when set' , ( done ) => {
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . message , 'window' )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
webview . setAttribute ( 'allowpopups' , 'on' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/window-open-hide.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'webpreferences attribute' , ( ) => {
it ( 'can enable nodeintegration' , ( done ) => {
webview . addEventListener ( 'console-message' , ( e ) => {
2016-10-14 23:04:33 +00:00
assert . equal ( e . message , 'function object object' )
done ( )
} )
webview . setAttribute ( 'webpreferences' , 'nodeIntegration' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/d.html `
2016-10-14 23:04:33 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'can disables web security and enable nodeintegration' , ( done ) => {
const jqueryPath = path . join ( _ _dirname , '/static/jquery-2.0.3.min.js' )
const src = ` <script src='file:// ${ jqueryPath } '></script> <script>console.log('ok '+(typeof require));</script> `
const encoded = btoa ( unescape ( encodeURIComponent ( src ) ) )
const listener = ( e ) => {
2016-10-14 23:04:33 +00:00
assert . equal ( e . message , 'ok function' )
webview . removeEventListener ( 'console-message' , listener )
done ( )
}
webview . addEventListener ( 'console-message' , listener )
webview . setAttribute ( 'webpreferences' , 'webSecurity=no, nodeIntegration=yes' )
2017-10-27 20:45:58 +00:00
webview . src = ` data:text/html;base64, ${ encoded } `
2016-10-14 23:04:33 +00:00
document . body . appendChild ( webview )
} )
2017-01-05 18:17:06 +00:00
it ( 'can enable context isolation' , ( done ) => {
ipcMain . once ( 'isolated-world' , ( event , data ) => {
assert . deepEqual ( data , {
preloadContext : {
preloadProperty : 'number' ,
pageProperty : 'undefined' ,
typeofRequire : 'function' ,
typeofProcess : 'object' ,
typeofArrayPush : 'function' ,
typeofFunctionApply : 'function'
} ,
pageContext : {
preloadProperty : 'undefined' ,
pageProperty : 'string' ,
typeofRequire : 'undefined' ,
typeofProcess : 'undefined' ,
typeofArrayPush : 'number' ,
typeofFunctionApply : 'boolean' ,
2017-01-12 00:36:59 +00:00
typeofPreloadExecuteJavaScriptProperty : 'number' ,
2017-05-22 18:10:10 +00:00
typeofOpenedWindow : 'object'
2017-01-05 18:17:06 +00:00
}
} )
done ( )
} )
webview . setAttribute ( 'preload' , path . join ( fixtures , 'api' , 'isolated-preload.js' ) )
2017-01-12 00:36:59 +00:00
webview . setAttribute ( 'allowpopups' , 'yes' )
2017-01-05 18:17:06 +00:00
webview . setAttribute ( 'webpreferences' , 'contextIsolation=yes' )
webview . src = 'file://' + fixtures + '/api/isolated.html'
document . body . appendChild ( webview )
} )
2016-10-14 23:04:33 +00:00
} )
2017-10-27 20:45:58 +00:00
describe ( 'new-window event' , ( ) => {
it ( 'emits when window.open is called' , ( done ) => {
webview . addEventListener ( 'new-window' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . url , 'http://host/' )
assert . equal ( e . frameName , 'host' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/window-open.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'emits when link with target is called' , ( done ) => {
webview . addEventListener ( 'new-window' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . url , 'http://host/' )
assert . equal ( e . frameName , 'target' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/target-name.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'ipc-message event' , ( ) => {
it ( 'emits when guest sends a ipc message to browser' , ( done ) => {
webview . addEventListener ( 'ipc-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . channel , 'channel' )
assert . deepEqual ( e . args , [ 'arg1' , 'arg2' ] )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/ipc-message.html `
2016-03-25 20:03:49 +00:00
webview . setAttribute ( 'nodeintegration' , 'on' )
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'page-title-set event' , ( ) => {
it ( 'emits when title is set' , ( done ) => {
webview . addEventListener ( 'page-title-set' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . title , 'test' )
assert ( e . explicitSet )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'page-favicon-updated event' , ( ) => {
it ( 'emits when favicon urls are received' , ( done ) => {
webview . addEventListener ( 'page-favicon-updated' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . favicons . length , 2 )
2017-02-16 03:02:11 +00:00
if ( process . platform === 'win32' ) {
2017-02-16 17:26:20 +00:00
assert ( /^file:\/\/\/[A-Z]:\/favicon.png$/i . test ( e . favicons [ 0 ] ) )
2017-02-16 03:02:11 +00:00
} else {
assert . equal ( e . favicons [ 0 ] , 'file:///favicon.png' )
}
2016-03-25 20:03:49 +00:00
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'will-navigate event' , ( ) => {
it ( 'emits when a url that leads to oustide of the page is clicked' , ( done ) => {
webview . addEventListener ( 'will-navigate' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . url , 'http://host/' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/webview-will-navigate.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'did-navigate event' , ( ) => {
let p = path . join ( fixtures , 'pages' , 'webview-will-navigate.html' )
2016-03-25 20:03:49 +00:00
p = p . replace ( /\\/g , '/' )
2017-10-27 20:45:58 +00:00
const pageUrl = url . format ( {
2016-01-12 02:40:23 +00:00
protocol : 'file' ,
slashes : true ,
pathname : p
2016-03-25 20:03:49 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'emits when a url that leads to outside of the page is clicked' , ( done ) => {
webview . addEventListener ( 'did-navigate' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . url , pageUrl )
done ( )
} )
webview . src = pageUrl
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'did-navigate-in-page event' , ( ) => {
it ( 'emits when an anchor link is clicked' , ( done ) => {
let p = path . join ( fixtures , 'pages' , 'webview-did-navigate-in-page.html' )
2016-03-25 20:03:49 +00:00
p = p . replace ( /\\/g , '/' )
2017-10-27 20:45:58 +00:00
const pageUrl = url . format ( {
2016-01-12 02:40:23 +00:00
protocol : 'file' ,
slashes : true ,
pathname : p
2016-03-25 20:03:49 +00:00
} )
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'did-navigate-in-page' , ( e ) => {
assert . equal ( e . url , ` ${ pageUrl } #test_content ` )
2016-03-25 20:03:49 +00:00
done ( )
} )
webview . src = pageUrl
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'emits when window.history.replaceState is called' , ( done ) => {
webview . addEventListener ( 'did-navigate-in-page' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . url , 'http://host/' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/webview-did-navigate-in-page-with-history.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'emits when window.location.hash is changed' , ( done ) => {
let p = path . join ( fixtures , 'pages' , 'webview-did-navigate-in-page-with-hash.html' )
2016-03-25 20:03:49 +00:00
p = p . replace ( /\\/g , '/' )
2017-10-27 20:45:58 +00:00
const pageUrl = url . format ( {
2016-01-12 02:40:23 +00:00
protocol : 'file' ,
slashes : true ,
pathname : p
2016-03-25 20:03:49 +00:00
} )
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'did-navigate-in-page' , ( e ) => {
assert . equal ( e . url , ` ${ pageUrl } #test ` )
2016-03-25 20:03:49 +00:00
done ( )
} )
webview . src = pageUrl
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'close event' , ( ) => {
it ( 'should fire when interior page calls window.close' , ( done ) => {
webview . addEventListener ( 'close' , ( ) => done ( ) )
webview . src = ` file:// ${ fixtures } /pages/close.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-11-30 12:04:50 +00:00
describe ( 'setDevToolsWebCotnents() API' , ( ) => {
it ( 'sets webContents of webview as devtools' , ( done ) => {
const webview2 = new WebView ( )
webview2 . addEventListener ( 'did-attach' , ( ) => {
webview2 . addEventListener ( 'dom-ready' , ( ) => {
const devtools = webview2 . getWebContents ( )
assert . ok ( devtools . getURL ( ) . startsWith ( 'chrome-devtools://devtools' ) )
devtools . executeJavaScript ( 'InspectorFrontendHost.constructor.name' , ( name ) => {
assert . ok ( name , 'InspectorFrontendHostImpl' )
document . body . removeChild ( webview2 )
done ( )
} )
} )
webview . addEventListener ( 'dom-ready' , ( ) => {
webview . getWebContents ( ) . setDevToolsWebContents ( webview2 . getWebContents ( ) )
webview . getWebContents ( ) . openDevTools ( )
} )
webview . src = 'about:blank'
document . body . appendChild ( webview )
} )
document . body . appendChild ( webview2 )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'devtools-opened event' , ( ) => {
it ( 'should fire when webview.openDevTools() is called' , ( done ) => {
const listener = ( ) => {
2016-03-25 20:03:49 +00:00
webview . removeEventListener ( 'devtools-opened' , listener )
webview . closeDevTools ( )
done ( )
}
webview . addEventListener ( 'devtools-opened' , listener )
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'dom-ready' , ( ) => {
2016-03-25 20:03:49 +00:00
webview . openDevTools ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/base-page.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'devtools-closed event' , ( ) => {
it ( 'should fire when webview.closeDevTools() is called' , ( done ) => {
const listener2 = ( ) => {
2016-03-25 20:03:49 +00:00
webview . removeEventListener ( 'devtools-closed' , listener2 )
done ( )
}
2017-10-27 20:45:58 +00:00
const listener = ( ) => {
2016-03-25 20:03:49 +00:00
webview . removeEventListener ( 'devtools-opened' , listener )
webview . closeDevTools ( )
}
webview . addEventListener ( 'devtools-opened' , listener )
webview . addEventListener ( 'devtools-closed' , listener2 )
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'dom-ready' , ( ) => {
2016-03-25 20:03:49 +00:00
webview . openDevTools ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/base-page.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'devtools-focused event' , ( ) => {
it ( 'should fire when webview.openDevTools() is called' , ( done ) => {
const listener = ( ) => {
2016-03-25 20:03:49 +00:00
webview . removeEventListener ( 'devtools-focused' , listener )
webview . closeDevTools ( )
done ( )
}
webview . addEventListener ( 'devtools-focused' , listener )
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'dom-ready' , ( ) => {
2016-03-25 20:03:49 +00:00
webview . openDevTools ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/base-page.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( '<webview>.reload()' , ( ) => {
it ( 'should emit beforeunload handler' , ( done ) => {
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . channel , 'onbeforeunload' )
webview . removeEventListener ( 'ipc-message' , listener )
done ( )
}
2017-10-27 20:45:58 +00:00
const listener2 = ( ) => {
2016-03-25 20:03:49 +00:00
webview . reload ( )
webview . removeEventListener ( 'did-finish-load' , listener2 )
}
webview . addEventListener ( 'ipc-message' , listener )
webview . addEventListener ( 'did-finish-load' , listener2 )
webview . setAttribute ( 'nodeintegration' , 'on' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/beforeunload-false.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( '<webview>.goForward()' , ( ) => {
it ( 'should work after a replaced history entry' , ( done ) => {
let loadCount = 1
const listener = ( e ) => {
2016-09-17 00:06:16 +00:00
if ( loadCount === 1 ) {
assert . equal ( e . channel , 'history' )
assert . equal ( e . args [ 0 ] , 1 )
assert ( ! webview . canGoBack ( ) )
assert ( ! webview . canGoForward ( ) )
} else if ( loadCount === 2 ) {
assert . equal ( e . channel , 'history' )
assert . equal ( e . args [ 0 ] , 2 )
assert ( ! webview . canGoBack ( ) )
assert ( webview . canGoForward ( ) )
webview . removeEventListener ( 'ipc-message' , listener )
}
}
2017-10-27 20:45:58 +00:00
const loadListener = ( e ) => {
2016-09-17 00:06:16 +00:00
if ( loadCount === 1 ) {
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/base-page.html `
2016-09-17 00:06:16 +00:00
} else if ( loadCount === 2 ) {
assert ( webview . canGoBack ( ) )
assert ( ! webview . canGoForward ( ) )
webview . goBack ( )
} else if ( loadCount === 3 ) {
webview . goForward ( )
} else if ( loadCount === 4 ) {
assert ( webview . canGoBack ( ) )
assert ( ! webview . canGoForward ( ) )
webview . removeEventListener ( 'did-finish-load' , loadListener )
done ( )
}
2017-10-27 20:45:58 +00:00
loadCount += 1
2016-09-17 00:06:16 +00:00
}
webview . addEventListener ( 'ipc-message' , listener )
webview . addEventListener ( 'did-finish-load' , loadListener )
webview . setAttribute ( 'nodeintegration' , 'on' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/history-replace.html `
2016-09-17 00:06:16 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( '<webview>.clearHistory()' , ( ) => {
it ( 'should clear the navigation history' , ( done ) => {
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . channel , 'history' )
assert . equal ( e . args [ 0 ] , 2 )
assert ( webview . canGoBack ( ) )
webview . clearHistory ( )
assert ( ! webview . canGoBack ( ) )
webview . removeEventListener ( 'ipc-message' , listener )
done ( )
}
webview . addEventListener ( 'ipc-message' , listener )
webview . setAttribute ( 'nodeintegration' , 'on' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/history.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'basic auth' , ( ) => {
const auth = require ( 'basic-auth' )
2016-03-25 20:03:49 +00:00
2017-10-27 20:45:58 +00:00
it ( 'should authenticate with correct credentials' , ( done ) => {
const message = 'Authenticated'
const server = http . createServer ( ( req , res ) => {
const credentials = auth ( req )
2016-01-12 02:40:23 +00:00
if ( credentials . name === 'test' && credentials . pass === 'test' ) {
2016-03-25 20:03:49 +00:00
res . end ( message )
2016-01-12 02:40:23 +00:00
} else {
2016-03-25 20:03:49 +00:00
res . end ( 'failed' )
2016-01-12 02:40:23 +00:00
}
2016-03-25 20:03:49 +00:00
server . close ( )
} )
2017-10-27 20:45:58 +00:00
server . listen ( 0 , '127.0.0.1' , ( ) => {
const port = server . address ( ) . port
webview . addEventListener ( 'ipc-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . channel , message )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/basic-auth.html?port= ${ port } `
2016-03-25 20:03:49 +00:00
webview . setAttribute ( 'nodeintegration' , 'on' )
document . body . appendChild ( webview )
} )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'dom-ready event' , ( ) => {
it ( 'emits when document is loaded' , ( done ) => {
const server = http . createServer ( ( ) => { } )
server . listen ( 0 , '127.0.0.1' , ( ) => {
const port = server . address ( ) . port
webview . addEventListener ( 'dom-ready' , ( ) => {
2016-03-25 20:03:49 +00:00
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/dom-ready.html?port= ${ port } `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
it ( 'throws a custom error when an API method is called before the event is emitted' , ( ) => {
assert . throws ( ( ) => {
2016-03-25 20:03:49 +00:00
webview . stop ( )
} , 'Cannot call stop because the webContents is unavailable. The WebView must be attached to the DOM and the dom-ready event emitted before this method can be called.' )
} )
} )
2016-02-05 18:27:05 +00:00
2017-10-27 20:45:58 +00:00
describe ( 'executeJavaScript' , ( ) => {
2017-11-15 21:05:46 +00:00
it ( 'should support user gesture' , function ( done ) {
2017-10-27 20:45:58 +00:00
const listener = ( ) => {
2016-03-25 20:03:49 +00:00
webview . removeEventListener ( 'enter-html-full-screen' , listener )
done ( )
}
2017-10-27 20:45:58 +00:00
const listener2 = ( ) => {
const jsScript = "document.querySelector('video').webkitRequestFullscreen()"
2016-03-25 20:03:49 +00:00
webview . executeJavaScript ( jsScript , true )
webview . removeEventListener ( 'did-finish-load' , listener2 )
}
webview . addEventListener ( 'enter-html-full-screen' , listener )
webview . addEventListener ( 'did-finish-load' , listener2 )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/fullscreen.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
2017-11-15 21:05:46 +00:00
it ( 'can return the result of the executed script' , function ( done ) {
2017-10-27 20:45:58 +00:00
const listener = ( ) => {
const jsScript = "'4'+2"
webview . executeJavaScript ( jsScript , false , ( result ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( result , '42' )
done ( )
} )
webview . removeEventListener ( 'did-finish-load' , listener )
}
webview . addEventListener ( 'did-finish-load' , listener )
webview . src = 'about:blank'
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'sendInputEvent' , ( ) => {
it ( 'can send keyboard event' , ( done ) => {
webview . addEventListener ( 'ipc-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . channel , 'keyup' )
2016-10-07 21:57:31 +00:00
assert . deepEqual ( e . args , [ 'C' , 'KeyC' , 67 , true , false ] )
2016-03-25 20:03:49 +00:00
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'dom-ready' , ( ) => {
2016-02-05 18:27:05 +00:00
webview . sendInputEvent ( {
2016-01-12 02:40:23 +00:00
type : 'keyup' ,
keyCode : 'c' ,
modifiers : [ 'shift' ]
2016-03-25 20:03:49 +00:00
} )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/onkeyup.html `
2016-03-25 20:03:49 +00:00
webview . setAttribute ( 'nodeintegration' , 'on' )
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'can send mouse event' , ( done ) => {
webview . addEventListener ( 'ipc-message' , ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . channel , 'mouseup' )
assert . deepEqual ( e . args , [ 10 , 20 , false , true ] )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'dom-ready' , ( ) => {
2016-02-05 18:27:05 +00:00
webview . sendInputEvent ( {
2016-01-12 02:40:23 +00:00
type : 'mouseup' ,
modifiers : [ 'ctrl' ] ,
x : 10 ,
y : 20
2016-03-25 20:03:49 +00:00
} )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/onmouseup.html `
2016-03-25 20:03:49 +00:00
webview . setAttribute ( 'nodeintegration' , 'on' )
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'media-started-playing media-paused events' , ( ) => {
it ( 'emits when audio starts and stops playing' , ( done ) => {
let audioPlayed = false
webview . addEventListener ( 'media-started-playing' , ( ) => {
2016-03-25 20:03:49 +00:00
audioPlayed = true
} )
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'media-paused' , ( ) => {
2016-03-25 20:03:49 +00:00
assert ( audioPlayed )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/audio.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-11-25 12:21:26 +00:00
describe ( 'found-in-page event' , ( ) => {
2017-10-27 20:45:58 +00:00
it ( 'emits when a request is made' , ( done ) => {
2016-09-08 05:27:10 +00:00
let requestId = null
let activeMatchOrdinal = [ ]
2017-10-27 20:45:58 +00:00
const listener = ( e ) => {
2016-03-25 20:03:49 +00:00
assert . equal ( e . result . requestId , requestId )
2016-09-08 05:27:10 +00:00
assert . equal ( e . result . matches , 3 )
activeMatchOrdinal . push ( e . result . activeMatchOrdinal )
if ( e . result . activeMatchOrdinal === e . result . matches ) {
assert . deepEqual ( activeMatchOrdinal , [ 1 , 2 , 3 ] )
webview . stopFindInPage ( 'clearSelection' )
done ( )
2016-03-14 01:19:45 +00:00
} else {
2016-09-08 05:27:10 +00:00
listener2 ( )
2016-01-12 02:40:23 +00:00
}
2016-03-25 20:03:49 +00:00
}
2017-10-27 20:45:58 +00:00
const listener2 = ( ) => {
2016-03-25 20:03:49 +00:00
requestId = webview . findInPage ( 'virtual' )
}
webview . addEventListener ( 'found-in-page' , listener )
webview . addEventListener ( 'did-finish-load' , listener2 )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/content.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
2017-11-25 14:48:01 +00:00
// TODO(deepak1556): With https://codereview.chromium.org/2836973002
// focus of the webContents is required when triggering the api.
// Remove this workaround after determining the cause for
// incorrect focus.
webview . focus ( )
2016-03-25 20:03:49 +00:00
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'did-change-theme-color event' , ( ) => {
it ( 'emits when theme color changes' , ( done ) => {
webview . addEventListener ( 'did-change-theme-color' , ( ) => done ( ) )
webview . src = ` file:// ${ fixtures } /pages/theme-color.html `
2016-03-25 20:03:49 +00:00
document . body . appendChild ( webview )
} )
} )
2017-10-27 20:45:58 +00:00
describe ( 'permission-request event' , ( ) => {
2016-06-29 16:37:10 +00:00
function setUpRequestHandler ( webview , requestedPermission , completed ) {
2017-10-27 20:45:58 +00:00
const listener = function ( webContents , permission , callback ) {
2016-03-25 20:03:49 +00:00
if ( webContents . getId ( ) === webview . getId ( ) ) {
2016-12-19 23:11:17 +00:00
// requestMIDIAccess with sysex requests both midi and midiSysex so
// grant the first midi one and then reject the midiSysex one
2016-12-20 00:27:38 +00:00
if ( requestedPermission === 'midiSysex' && permission === 'midi' ) {
2016-12-19 23:11:17 +00:00
return callback ( true )
}
2016-06-29 16:37:10 +00:00
assert . equal ( permission , requestedPermission )
2016-03-25 20:03:49 +00:00
callback ( false )
2016-06-29 16:37:10 +00:00
if ( completed ) completed ( )
2016-01-31 21:35:34 +00:00
}
2016-03-25 20:03:49 +00:00
}
session . fromPartition ( webview . partition ) . setPermissionRequestHandler ( listener )
2016-01-31 21:35:34 +00:00
}
2017-10-27 20:45:58 +00:00
it ( 'emits when using navigator.getUserMedia api' , ( done ) => {
if ( isCI ) return done ( )
2017-07-11 11:12:57 +00:00
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'ipc-message' , ( e ) => {
2016-10-28 18:14:14 +00:00
assert . equal ( e . channel , 'message' )
assert . deepEqual ( e . args , [ 'PermissionDeniedError' ] )
2016-03-25 20:03:49 +00:00
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/permissions/media.html `
2016-03-25 20:03:49 +00:00
webview . partition = 'permissionTest'
webview . setAttribute ( 'nodeintegration' , 'on' )
setUpRequestHandler ( webview , 'media' )
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'emits when using navigator.geolocation api' , ( done ) => {
webview . addEventListener ( 'ipc-message' , ( e ) => {
2016-10-28 18:14:14 +00:00
assert . equal ( e . channel , 'message' )
assert . deepEqual ( e . args , [ 'User denied Geolocation' ] )
2016-03-25 20:03:49 +00:00
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/permissions/geolocation.html `
2016-03-25 20:03:49 +00:00
webview . partition = 'permissionTest'
webview . setAttribute ( 'nodeintegration' , 'on' )
setUpRequestHandler ( webview , 'geolocation' )
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'emits when using navigator.requestMIDIAccess without sysex api' , ( done ) => {
webview . addEventListener ( 'ipc-message' , ( e ) => {
2016-10-28 18:14:14 +00:00
assert . equal ( e . channel , 'message' )
assert . deepEqual ( e . args , [ 'SecurityError' ] )
2016-03-25 20:03:49 +00:00
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/permissions/midi.html `
2016-03-25 20:03:49 +00:00
webview . partition = 'permissionTest'
webview . setAttribute ( 'nodeintegration' , 'on' )
2016-12-19 23:11:17 +00:00
setUpRequestHandler ( webview , 'midi' )
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'emits when using navigator.requestMIDIAccess with sysex api' , ( done ) => {
webview . addEventListener ( 'ipc-message' , ( e ) => {
2016-12-19 23:11:17 +00:00
assert . equal ( e . channel , 'message' )
assert . deepEqual ( e . args , [ 'SecurityError' ] )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/permissions/midi-sysex.html `
2016-12-19 23:11:17 +00:00
webview . partition = 'permissionTest'
webview . setAttribute ( 'nodeintegration' , 'on' )
2016-03-25 20:03:49 +00:00
setUpRequestHandler ( webview , 'midiSysex' )
document . body . appendChild ( webview )
} )
2016-04-20 16:55:15 +00:00
2017-10-27 20:45:58 +00:00
it ( 'emits when accessing external protocol' , ( done ) => {
2016-04-20 16:55:15 +00:00
webview . src = 'magnet:test'
webview . partition = 'permissionTest'
setUpRequestHandler ( webview , 'openExternal' , done )
document . body . appendChild ( webview )
} )
2016-10-28 17:34:11 +00:00
2017-10-27 20:45:58 +00:00
it ( 'emits when using Notification.requestPermission' , ( done ) => {
webview . addEventListener ( 'ipc-message' , ( e ) => {
2016-10-28 17:34:11 +00:00
assert . equal ( e . channel , 'message' )
assert . deepEqual ( e . args , [ 'granted' ] )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/permissions/notification.html `
2016-10-28 17:34:11 +00:00
webview . partition = 'permissionTest'
webview . setAttribute ( 'nodeintegration' , 'on' )
2017-10-27 20:45:58 +00:00
session . fromPartition ( webview . partition ) . setPermissionRequestHandler ( ( webContents , permission , callback ) => {
2016-10-28 17:34:11 +00:00
if ( webContents . getId ( ) === webview . getId ( ) ) {
assert . equal ( permission , 'notifications' )
2017-10-27 20:45:58 +00:00
setTimeout ( ( ) => { callback ( true ) } , 10 )
2016-10-28 17:34:11 +00:00
}
} )
document . body . appendChild ( webview )
} )
2016-03-25 20:03:49 +00:00
} )
2017-10-27 20:45:58 +00:00
describe ( '<webview>.getWebContents' , ( ) => {
it ( 'can return the webcontents associated' , ( done ) => {
webview . addEventListener ( 'did-finish-load' , ( ) => {
2016-03-25 20:03:49 +00:00
const webviewContents = webview . getWebContents ( )
assert ( webviewContents )
assert . equal ( webviewContents . getURL ( ) , 'about:blank' )
done ( )
} )
webview . src = 'about:blank'
document . body . appendChild ( webview )
} )
} )
2016-04-08 19:55:20 +00:00
2017-10-27 20:45:58 +00:00
describe ( 'did-get-response-details event' , ( ) => {
it ( 'emits for the page and its resources' , ( done ) => {
2016-04-08 19:55:20 +00:00
// expected {fileName: resourceType} pairs
2017-10-27 20:45:58 +00:00
const expectedResources = {
2016-04-08 19:55:20 +00:00
'did-get-response-details.html' : 'mainFrame' ,
'logo.png' : 'image'
}
2017-10-27 20:45:58 +00:00
let responses = 0
webview . addEventListener ( 'did-get-response-details' , ( event ) => {
responses += 1
const fileName = event . newURL . slice ( event . newURL . lastIndexOf ( '/' ) + 1 )
const expectedType = expectedResources [ fileName ]
2016-04-08 19:55:20 +00:00
assert ( ! ! expectedType , ` Unexpected response details for ${ event . newURL } ` )
assert ( typeof event . status === 'boolean' , 'status should be boolean' )
assert . equal ( event . httpResponseCode , 200 )
assert . equal ( event . requestMethod , 'GET' )
assert ( typeof event . referrer === 'string' , 'referrer should be string' )
assert ( ! ! event . headers , 'headers should be present' )
assert ( typeof event . headers === 'object' , 'headers should be object' )
assert . equal ( event . resourceType , expectedType , 'Incorrect resourceType' )
2017-10-27 20:45:58 +00:00
if ( responses === Object . keys ( expectedResources ) . length ) done ( )
2016-04-08 19:55:20 +00:00
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ path . join ( fixtures , 'pages' , 'did-get-response-details.html' ) } `
2016-04-08 19:55:20 +00:00
document . body . appendChild ( webview )
} )
} )
2016-05-19 21:46:11 +00:00
2017-10-27 20:45:58 +00:00
describe ( 'document.visibilityState/hidden' , ( ) => {
afterEach ( ( ) => {
2017-06-12 19:51:27 +00:00
ipcMain . removeAllListeners ( 'pong' )
2016-05-24 19:37:26 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'updates when the window is shown after the ready-to-show event' , ( done ) => {
w = new BrowserWindow ( { show : false } )
w . once ( 'ready-to-show' , ( ) => {
2017-06-12 19:51:27 +00:00
w . show ( )
2016-05-24 19:37:26 +00:00
} )
2017-02-24 19:56:27 +00:00
2017-10-27 20:45:58 +00:00
ipcMain . on ( 'pong' , ( event , visibilityState , hidden ) => {
2017-06-12 19:51:27 +00:00
if ( ! hidden ) {
assert . equal ( visibilityState , 'visible' )
done ( )
}
} )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-visibilitychange.html ` )
2016-05-24 19:37:26 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'inherits the parent window visibility state and receives visibilitychange events' , ( done ) => {
w = new BrowserWindow ( { show : false } )
2017-06-12 19:51:27 +00:00
2017-10-27 20:45:58 +00:00
ipcMain . once ( 'pong' , ( event , visibilityState , hidden ) => {
2017-06-12 19:51:27 +00:00
assert . equal ( visibilityState , 'hidden' )
assert . equal ( hidden , true )
ipcMain . once ( 'pong' , function ( event , visibilityState , hidden ) {
assert . equal ( visibilityState , 'visible' )
assert . equal ( hidden , false )
done ( )
} )
w . webContents . emit ( '-window-visibility-change' , 'visible' )
} )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-visibilitychange.html ` )
2017-06-12 19:51:27 +00:00
} )
2016-05-24 19:37:26 +00:00
} )
2016-06-07 16:50:06 +00:00
2017-02-09 19:49:14 +00:00
describe ( 'will-attach-webview event' , ( ) => {
2017-02-03 20:21:46 +00:00
it ( 'supports changing the web preferences' , ( done ) => {
2017-02-03 20:55:37 +00:00
ipcRenderer . send ( 'disable-node-on-next-will-attach-webview' )
2017-02-03 20:21:46 +00:00
webview . addEventListener ( 'console-message' , ( event ) => {
assert . equal ( event . message , 'undefined undefined undefined undefined' )
done ( )
} )
webview . setAttribute ( 'nodeintegration' , 'yes' )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2017-02-03 20:21:46 +00:00
document . body . appendChild ( webview )
} )
it ( 'supports preventing a webview from being created' , ( done ) => {
2017-02-03 20:55:37 +00:00
ipcRenderer . send ( 'prevent-next-will-attach-webview' )
2017-02-03 20:21:46 +00:00
webview . addEventListener ( 'destroyed' , ( ) => {
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/c.html `
2017-02-03 20:21:46 +00:00
document . body . appendChild ( webview )
} )
2017-05-17 20:47:00 +00:00
it ( 'supports removing the preload script' , ( done ) => {
ipcRenderer . send ( 'disable-preload-on-next-will-attach-webview' )
webview . addEventListener ( 'console-message' , ( event ) => {
assert . equal ( event . message , 'undefined' )
done ( )
} )
webview . setAttribute ( 'nodeintegration' , 'yes' )
webview . setAttribute ( 'preload' , path . join ( fixtures , 'module' , 'preload-set-global.js' ) )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2017-05-17 20:47:00 +00:00
document . body . appendChild ( webview )
} )
2017-02-03 20:21:46 +00:00
} )
2017-10-06 16:31:41 +00:00
describe ( 'did-attach-webview event' , ( ) => {
it ( 'is emitted when a webview has been attached' , ( done ) => {
2017-10-27 20:45:58 +00:00
w = new BrowserWindow ( { show : false } )
2017-10-06 16:31:41 +00:00
w . webContents . on ( 'did-attach-webview' , ( event , webContents ) => {
ipcMain . once ( 'webview-dom-ready' , ( event , id ) => {
assert . equal ( webContents . id , id )
done ( )
} )
} )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-did-attach-event.html ` )
2017-10-06 16:31:41 +00:00
} )
} )
2017-10-27 20:45:58 +00:00
it ( 'loads devtools extensions registered on the parent window' , ( done ) => {
w = new BrowserWindow ( { show : false } )
2016-06-07 16:50:06 +00:00
BrowserWindow . removeDevToolsExtension ( 'foo' )
2017-10-27 20:45:58 +00:00
const extensionPath = path . join ( _ _dirname , 'fixtures' , 'devtools-extensions' , 'foo' )
2016-06-07 16:50:06 +00:00
BrowserWindow . addDevToolsExtension ( extensionPath )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-devtools.html ` )
2016-06-07 16:50:06 +00:00
2017-10-27 20:45:58 +00:00
ipcMain . once ( 'answer' , ( event , message ) => {
2016-06-07 16:50:06 +00:00
assert . equal ( message . runtimeId , 'foo' )
2016-06-07 17:38:01 +00:00
assert . notEqual ( message . tabId , w . webContents . id )
2016-06-07 16:50:06 +00:00
done ( )
} )
} )
2016-09-08 17:01:01 +00:00
2017-10-27 20:45:58 +00:00
describe ( 'guestinstance attribute' , ( ) => {
it ( 'before loading there is no attribute' , ( ) => {
2016-09-08 17:01:01 +00:00
document . body . appendChild ( webview )
assert ( ! webview . hasAttribute ( 'guestinstance' ) )
} )
2017-10-27 20:45:58 +00:00
it ( 'loading a page sets the guest view' , ( done ) => {
const loadListener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
const instance = webview . getAttribute ( 'guestinstance' )
2016-09-08 17:01:01 +00:00
assert . equal ( instance , parseInt ( instance ) )
2017-10-27 20:45:58 +00:00
const guest = getGuestWebContents ( parseInt ( instance ) )
2016-09-08 17:01:01 +00:00
assert . equal ( guest , webview . getWebContents ( ) )
done ( )
}
webview . addEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /api/blank.html `
2016-09-08 17:01:01 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'deleting the attribute destroys the webview' , ( done ) => {
const loadListener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
const destroyListener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'destroyed' , destroyListener , false )
assert . equal ( getGuestWebContents ( instance ) , null )
done ( )
}
webview . addEventListener ( 'destroyed' , destroyListener , false )
2017-10-27 20:45:58 +00:00
const instance = parseInt ( webview . getAttribute ( 'guestinstance' ) )
2016-09-08 17:01:01 +00:00
webview . removeAttribute ( 'guestinstance' )
}
webview . addEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /api/blank.html `
2016-09-08 17:01:01 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'setting the attribute on a new webview moves the contents' , ( done ) => {
const loadListener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
const webContents = webview . getWebContents ( )
const instance = webview . getAttribute ( 'guestinstance' )
2016-09-08 17:01:01 +00:00
2017-10-27 20:45:58 +00:00
const destroyListener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'destroyed' , destroyListener , false )
assert . equal ( webContents , webview2 . getWebContents ( ) )
// Make sure that events are hooked up to the right webview now
2017-10-27 20:45:58 +00:00
webview2 . addEventListener ( 'console-message' , ( e ) => {
2016-09-08 17:01:01 +00:00
assert . equal ( e . message , 'a' )
document . body . removeChild ( webview2 )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview2 . src = ` file:// ${ fixtures } /pages/a.html `
2016-09-08 17:01:01 +00:00
}
webview . addEventListener ( 'destroyed' , destroyListener , false )
2017-10-27 20:45:58 +00:00
const webview2 = new WebView ( )
2016-09-08 17:01:01 +00:00
webview2 . setAttribute ( 'guestinstance' , instance )
document . body . appendChild ( webview2 )
}
webview . addEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /api/blank.html `
2016-09-08 17:01:01 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'setting the attribute to an invalid guestinstance does nothing' , ( done ) => {
const loadListener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'did-finish-load' , loadListener , false )
webview . setAttribute ( 'guestinstance' , 55 )
// Make sure that events are still hooked up to the webview
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'console-message' , ( e ) => {
2016-09-08 17:01:01 +00:00
assert . equal ( e . message , 'a' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-09-08 17:01:01 +00:00
}
webview . addEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /api/blank.html `
2016-09-08 17:01:01 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'setting the attribute on an existing webview moves the contents' , ( done ) => {
const load1Listener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'did-finish-load' , load1Listener , false )
2017-10-27 20:45:58 +00:00
const webContents = webview . getWebContents ( )
const instance = webview . getAttribute ( 'guestinstance' )
let destroyedInstance
2016-09-08 17:01:01 +00:00
2017-10-27 20:45:58 +00:00
const destroyListener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'destroyed' , destroyListener , false )
assert . equal ( webContents , webview2 . getWebContents ( ) )
assert . equal ( null , getGuestWebContents ( parseInt ( destroyedInstance ) ) )
// Make sure that events are hooked up to the right webview now
2017-10-27 20:45:58 +00:00
webview2 . addEventListener ( 'console-message' , ( e ) => {
2016-09-08 17:01:01 +00:00
assert . equal ( e . message , 'a' )
document . body . removeChild ( webview2 )
done ( )
} )
webview2 . src = 'file://' + fixtures + '/pages/a.html'
}
webview . addEventListener ( 'destroyed' , destroyListener , false )
2017-10-27 20:45:58 +00:00
const webview2 = new WebView ( )
const load2Listener = ( ) => {
2016-09-08 17:01:01 +00:00
webview2 . removeEventListener ( 'did-finish-load' , load2Listener , false )
destroyedInstance = webview2 . getAttribute ( 'guestinstance' )
assert . notEqual ( instance , destroyedInstance )
webview2 . setAttribute ( 'guestinstance' , instance )
}
webview2 . addEventListener ( 'did-finish-load' , load2Listener , false )
webview2 . src = 'file://' + fixtures + '/api/blank.html'
document . body . appendChild ( webview2 )
}
webview . addEventListener ( 'did-finish-load' , load1Listener , false )
webview . src = 'file://' + fixtures + '/api/blank.html'
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'moving a guest back to its original webview should work' , ( done ) => {
const loadListener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
const webContents = webview . getWebContents ( )
const instance = webview . getAttribute ( 'guestinstance' )
2016-09-08 17:01:01 +00:00
2017-10-27 20:45:58 +00:00
const destroy1Listener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'destroyed' , destroy1Listener , false )
assert . equal ( webContents , webview2 . getWebContents ( ) )
assert . equal ( null , webview . getWebContents ( ) )
2017-10-27 20:45:58 +00:00
const destroy2Listener = ( ) => {
2016-09-08 17:01:01 +00:00
webview2 . removeEventListener ( 'destroyed' , destroy2Listener , false )
assert . equal ( webContents , webview . getWebContents ( ) )
assert . equal ( null , webview2 . getWebContents ( ) )
// Make sure that events are hooked up to the right webview now
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'console-message' , ( e ) => {
2016-09-08 17:01:01 +00:00
assert . equal ( e . message , 'a' )
document . body . removeChild ( webview2 )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-09-08 17:01:01 +00:00
}
webview2 . addEventListener ( 'destroyed' , destroy2Listener , false )
webview . setAttribute ( 'guestinstance' , instance )
}
webview . addEventListener ( 'destroyed' , destroy1Listener , false )
2017-10-27 20:45:58 +00:00
const webview2 = new WebView ( )
2016-09-08 17:01:01 +00:00
webview2 . setAttribute ( 'guestinstance' , instance )
document . body . appendChild ( webview2 )
}
webview . addEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /api/blank.html `
2016-09-08 17:01:01 +00:00
document . body . appendChild ( webview )
} )
2017-10-27 20:45:58 +00:00
it ( 'setting the attribute on a webview in a different window moves the contents' , ( done ) => {
const loadListener = ( ) => {
2016-09-08 17:01:01 +00:00
webview . removeEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
const instance = webview . getAttribute ( 'guestinstance' )
2016-09-08 17:01:01 +00:00
w = new BrowserWindow ( { show : false } )
2017-10-27 20:45:58 +00:00
w . webContents . once ( 'did-finish-load' , ( ) => {
ipcMain . once ( 'pong' , ( ) => {
2016-09-08 17:01:01 +00:00
assert ( ! webview . hasAttribute ( 'guestinstance' ) )
done ( )
} )
w . webContents . send ( 'guestinstance' , instance )
} )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-move-to-window.html ` )
2016-09-08 17:01:01 +00:00
}
webview . addEventListener ( 'did-finish-load' , loadListener , false )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /api/blank.html `
2016-09-08 17:01:01 +00:00
document . body . appendChild ( webview )
2016-11-02 19:00:28 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'does not delete the guestinstance attribute when moving the webview to another parent node' , ( done ) => {
2016-11-03 00:25:52 +00:00
webview . addEventListener ( 'dom-ready' , function domReadyListener ( ) {
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'did-attach' , ( ) => {
2016-11-03 00:25:52 +00:00
assert ( webview . guestinstance != null )
assert ( webview . getWebContents ( ) != null )
done ( )
} )
document . body . replaceChild ( webview , div )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-11-03 00:25:52 +00:00
const div = document . createElement ( 'div' )
div . appendChild ( webview )
document . body . appendChild ( div )
} )
2017-10-27 20:45:58 +00:00
it ( 'does not destroy the webContents when hiding/showing the webview (regression)' , ( done ) => {
2016-11-02 19:00:28 +00:00
webview . addEventListener ( 'dom-ready' , function domReadyListener ( ) {
const instance = webview . getAttribute ( 'guestinstance' )
2016-11-03 00:25:52 +00:00
assert ( instance != null )
2016-11-02 19:00:28 +00:00
// Wait for event directly since attach happens asynchronously over IPC
2017-10-27 20:45:58 +00:00
ipcMain . once ( 'ELECTRON_GUEST_VIEW_MANAGER_ATTACH_GUEST' , ( ) => {
2016-11-02 19:00:28 +00:00
assert ( webview . getWebContents ( ) != null )
assert . equal ( instance , webview . getAttribute ( 'guestinstance' ) )
done ( )
} )
webview . style . display = 'none'
2017-11-23 22:22:43 +00:00
webview . offsetHeight // eslint-disable-line
2016-11-02 19:00:28 +00:00
webview . style . display = 'block'
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-11-02 19:00:28 +00:00
document . body . appendChild ( webview )
2016-11-02 22:35:00 +00:00
} )
2017-10-27 20:45:58 +00:00
it ( 'does not reload the webContents when hiding/showing the webview (regression)' , ( done ) => {
2016-11-02 22:35:00 +00:00
webview . addEventListener ( 'dom-ready' , function domReadyListener ( ) {
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'did-start-loading' , ( ) => {
2016-11-02 22:35:00 +00:00
done ( new Error ( 'webview started loading unexpectedly' ) )
} )
// Wait for event directly since attach happens asynchronously over IPC
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'did-attach' , ( ) => {
2016-11-02 22:35:00 +00:00
done ( )
} )
webview . style . display = 'none'
2017-11-23 22:22:43 +00:00
webview . offsetHeight // eslint-disable-line
2016-11-02 22:35:00 +00:00
webview . style . display = 'block'
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-11-02 22:35:00 +00:00
document . body . appendChild ( webview )
2016-09-08 17:01:01 +00:00
} )
} )
2016-11-03 22:12:54 +00:00
2017-10-27 20:45:58 +00:00
describe ( 'DOM events' , ( ) => {
2016-11-03 22:12:54 +00:00
let div
2017-10-27 20:45:58 +00:00
beforeEach ( ( ) => {
2016-11-03 22:12:54 +00:00
div = document . createElement ( 'div' )
div . style . width = '100px'
div . style . height = '10px'
div . style . overflow = 'hidden'
webview . style . height = '100%'
webview . style . width = '100%'
} )
2017-10-27 20:45:58 +00:00
afterEach ( ( ) => {
2016-11-03 22:12:54 +00:00
if ( div != null ) div . remove ( )
} )
2017-10-27 20:45:58 +00:00
it ( 'emits resize events' , ( done ) => {
webview . addEventListener ( 'dom-ready' , ( ) => {
2016-11-03 22:12:54 +00:00
div . style . width = '1234px'
div . style . height = '789px'
} )
webview . addEventListener ( 'resize' , function onResize ( event ) {
webview . removeEventListener ( 'resize' , onResize )
assert . equal ( event . newWidth , 1234 )
assert . equal ( event . newHeight , 789 )
assert . equal ( event . target , webview )
done ( )
} )
2016-11-07 16:28:02 +00:00
webview . src = ` file:// ${ fixtures } /pages/a.html `
2016-11-03 22:12:54 +00:00
div . appendChild ( webview )
document . body . appendChild ( div )
} )
} )
2016-10-25 05:17:38 +00:00
describe ( 'disableguestresize attribute' , ( ) => {
it ( 'does not have attribute by default' , ( ) => {
document . body . appendChild ( webview )
assert ( ! webview . hasAttribute ( 'disableguestresize' ) )
} )
2017-10-27 20:45:58 +00:00
it ( 'resizes guest when attribute is not present' , ( done ) => {
2016-11-15 21:47:14 +00:00
w = new BrowserWindow ( { show : false , width : 200 , height : 200 } )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-guest-resize.html ` )
2016-10-25 05:17:38 +00:00
w . webContents . once ( 'did-finish-load' , ( ) => {
const CONTENT _SIZE = 300
const elementResizePromise = new Promise ( resolve => {
ipcMain . once ( 'webview-element-resize' , ( event , width , height ) => {
assert . equal ( width , CONTENT _SIZE )
2016-11-15 21:47:14 +00:00
assert . equal ( height , CONTENT _SIZE )
2016-10-25 05:17:38 +00:00
resolve ( )
} )
} )
const guestResizePromise = new Promise ( resolve => {
ipcMain . once ( 'webview-guest-resize' , ( event , width , height ) => {
assert . equal ( width , CONTENT _SIZE )
2016-11-15 21:47:14 +00:00
assert . equal ( height , CONTENT _SIZE )
2016-10-25 05:17:38 +00:00
resolve ( )
} )
} )
Promise . all ( [ elementResizePromise , guestResizePromise ] ) . then ( ( ) => done ( ) )
w . setContentSize ( CONTENT _SIZE , CONTENT _SIZE )
} )
} )
it ( 'does not resize guest when attribute is present' , done => {
2016-11-15 21:47:14 +00:00
w = new BrowserWindow ( { show : false , width : 200 , height : 200 } )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-no-guest-resize.html ` )
2016-10-25 05:17:38 +00:00
w . webContents . once ( 'did-finish-load' , ( ) => {
const CONTENT _SIZE = 300
const elementResizePromise = new Promise ( resolve => {
ipcMain . once ( 'webview-element-resize' , ( event , width , height ) => {
assert . equal ( width , CONTENT _SIZE )
2016-11-15 21:47:14 +00:00
assert . equal ( height , CONTENT _SIZE )
2016-10-25 05:17:38 +00:00
resolve ( )
} )
} )
const noGuestResizePromise = new Promise ( resolve => {
const onGuestResize = ( event , width , height ) => {
2016-11-15 21:47:14 +00:00
done ( new Error ( 'Unexpected guest resize message' ) )
2016-10-25 05:17:38 +00:00
}
ipcMain . once ( 'webview-guest-resize' , onGuestResize )
setTimeout ( ( ) => {
ipcMain . removeListener ( 'webview-guest-resize' , onGuestResize )
resolve ( )
} , 500 )
} )
Promise . all ( [ elementResizePromise , noGuestResizePromise ] ) . then ( ( ) => done ( ) )
w . setContentSize ( CONTENT _SIZE , CONTENT _SIZE )
} )
} )
it ( 'dispatches element resize event even when attribute is present' , done => {
2016-11-15 21:47:14 +00:00
w = new BrowserWindow ( { show : false , width : 200 , height : 200 } )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-no-guest-resize.html ` )
2016-10-25 05:17:38 +00:00
w . webContents . once ( 'did-finish-load' , ( ) => {
const CONTENT _SIZE = 300
ipcMain . once ( 'webview-element-resize' , ( event , width , height ) => {
assert . equal ( width , CONTENT _SIZE )
done ( )
} )
w . setContentSize ( CONTENT _SIZE , CONTENT _SIZE )
} )
} )
2017-11-15 21:05:46 +00:00
it ( 'can be manually resized with setSize even when attribute is present' , function ( done ) {
2016-11-15 21:47:14 +00:00
w = new BrowserWindow ( { show : false , width : 200 , height : 200 } )
2017-10-27 20:45:58 +00:00
w . loadURL ( ` file:// ${ fixtures } /pages/webview-no-guest-resize.html ` )
2016-10-25 05:17:38 +00:00
w . webContents . once ( 'did-finish-load' , ( ) => {
const GUEST _WIDTH = 10
const GUEST _HEIGHT = 20
ipcMain . once ( 'webview-guest-resize' , ( event , width , height ) => {
assert . equal ( width , GUEST _WIDTH )
assert . equal ( height , GUEST _HEIGHT )
done ( )
} )
2016-11-15 21:47:14 +00:00
for ( const wc of webContents . getAllWebContents ( ) ) {
2016-10-25 05:17:38 +00:00
if ( wc . hostWebContents &&
wc . hostWebContents . id === w . webContents . id ) {
wc . setSize ( {
normal : {
width : GUEST _WIDTH ,
height : GUEST _HEIGHT
}
} )
}
}
} )
} )
} )
2017-02-17 15:48:30 +00:00
describe ( 'zoom behavior' , ( ) => {
2017-02-17 17:58:29 +00:00
const zoomScheme = remote . getGlobal ( 'zoomScheme' )
const webviewSession = session . fromPartition ( 'webview-temp' )
before ( ( done ) => {
const protocol = webviewSession . protocol
protocol . registerStringProtocol ( zoomScheme , ( request , callback ) => {
callback ( 'hello' )
} , ( error ) => done ( error ) )
} )
after ( ( done ) => {
const protocol = webviewSession . protocol
protocol . unregisterProtocol ( zoomScheme , ( error ) => done ( error ) )
} )
2017-02-17 15:48:30 +00:00
it ( 'inherits the zoomFactor of the parent window' , ( done ) => {
w = new BrowserWindow ( {
show : false ,
webPreferences : {
zoomFactor : 1.2
}
} )
2017-02-17 17:58:29 +00:00
ipcMain . once ( 'webview-parent-zoom-level' , ( event , zoomFactor , zoomLevel ) => {
2017-02-17 15:48:30 +00:00
assert . equal ( zoomFactor , 1.2 )
assert . equal ( zoomLevel , 1 )
done ( )
} )
w . loadURL ( ` file:// ${ fixtures } /pages/webview-zoom-factor.html ` )
} )
2017-02-17 17:58:29 +00:00
it ( 'maintains zoom level on navigation' , ( done ) => {
2017-02-17 15:48:30 +00:00
w = new BrowserWindow ( {
show : false ,
webPreferences : {
zoomFactor : 1.2
}
} )
2017-02-17 17:58:29 +00:00
ipcMain . on ( 'webview-zoom-level' , ( event , zoomLevel , zoomFactor , newHost , final ) => {
if ( ! newHost ) {
2017-02-17 15:48:30 +00:00
assert . equal ( zoomFactor , 1.44 )
assert . equal ( zoomLevel , 2.0 )
} else {
assert . equal ( zoomFactor , 1.2 )
assert . equal ( zoomLevel , 1 )
}
2017-02-17 17:58:29 +00:00
if ( final ) done ( )
2017-02-17 15:48:30 +00:00
} )
w . loadURL ( ` file:// ${ fixtures } /pages/webview-custom-zoom-level.html ` )
} )
2017-02-17 17:58:29 +00:00
it ( 'maintains zoom level when navigating within same page' , ( done ) => {
w = new BrowserWindow ( {
show : false ,
webPreferences : {
zoomFactor : 1.2
}
} )
ipcMain . on ( 'webview-zoom-in-page' , ( event , zoomLevel , zoomFactor , final ) => {
assert . equal ( zoomFactor , 1.44 )
assert . equal ( zoomLevel , 2.0 )
if ( final ) done ( )
} )
w . loadURL ( ` file:// ${ fixtures } /pages/webview-in-page-navigate.html ` )
} )
2017-03-08 13:35:24 +00:00
it ( 'inherits zoom level for the origin when available' , ( done ) => {
w = new BrowserWindow ( {
show : false ,
webPreferences : {
zoomFactor : 1.2
}
} )
ipcMain . once ( 'webview-origin-zoom-level' , ( event , zoomLevel ) => {
assert . equal ( zoomLevel , 2.0 )
done ( )
} )
w . loadURL ( ` file:// ${ fixtures } /pages/webview-origin-zoom-level.html ` )
} )
2017-02-17 15:48:30 +00:00
} )
2017-05-23 20:24:51 +00:00
describe ( 'nativeWindowOpen option' , ( ) => {
2017-10-27 20:45:58 +00:00
beforeEach ( ( ) => {
2017-05-23 21:22:09 +00:00
webview . setAttribute ( 'allowpopups' , 'on' )
2017-05-23 20:59:13 +00:00
webview . setAttribute ( 'nodeintegration' , 'on' )
webview . setAttribute ( 'webpreferences' , 'nativeWindowOpen=1' )
} )
2017-05-23 20:28:24 +00:00
it ( 'opens window of about:blank with cross-scripting enabled' , ( done ) => {
2017-05-23 20:24:51 +00:00
ipcMain . once ( 'answer' , ( event , content ) => {
assert . equal ( content , 'Hello' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ path . join ( fixtures , 'api' , 'native-window-open-blank.html' ) } `
2017-05-23 20:24:51 +00:00
document . body . appendChild ( webview )
} )
2017-05-23 20:28:24 +00:00
it ( 'opens window of same domain with cross-scripting enabled' , ( done ) => {
ipcMain . once ( 'answer' , ( event , content ) => {
assert . equal ( content , 'Hello' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ path . join ( fixtures , 'api' , 'native-window-open-file.html' ) } `
2017-05-23 20:28:24 +00:00
document . body . appendChild ( webview )
2017-05-23 22:56:06 +00:00
} )
it ( 'returns null from window.open when allowpopups is not set' , ( done ) => {
webview . removeAttribute ( 'allowpopups' )
ipcMain . once ( 'answer' , ( event , { windowOpenReturnedNull } ) => {
assert . equal ( windowOpenReturnedNull , true )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ path . join ( fixtures , 'api' , 'native-window-open-no-allowpopups.html' ) } `
2017-05-23 22:56:06 +00:00
document . body . appendChild ( webview )
2017-05-25 16:39:40 +00:00
} )
it ( 'blocks accessing cross-origin frames' , ( done ) => {
ipcMain . once ( 'answer' , ( event , content ) => {
assert . equal ( content , 'Blocked a frame with origin "file://" from accessing a cross-origin frame.' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ path . join ( fixtures , 'api' , 'native-window-open-cross-origin.html' ) } `
2017-05-25 16:39:40 +00:00
document . body . appendChild ( webview )
2017-05-23 20:28:24 +00:00
} )
2017-05-23 20:59:13 +00:00
it ( 'emits a new-window event' , ( done ) => {
2017-10-27 20:45:58 +00:00
webview . addEventListener ( 'new-window' , ( e ) => {
2017-05-23 20:59:13 +00:00
assert . equal ( e . url , 'http://host/' )
assert . equal ( e . frameName , 'host' )
done ( )
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/window-open.html `
2017-05-23 20:59:13 +00:00
document . body . appendChild ( webview )
} )
2017-05-23 21:31:20 +00:00
it ( 'emits a browser-window-created event' , ( done ) => {
2017-10-27 20:45:58 +00:00
app . once ( 'browser-window-created' , ( ) => done ( ) )
webview . src = ` file:// ${ fixtures } /pages/window-open.html `
2017-05-23 21:31:20 +00:00
document . body . appendChild ( webview )
} )
it ( 'emits a web-contents-created event' , ( done ) => {
app . on ( 'web-contents-created' , function listener ( event , contents ) {
if ( contents . getType ( ) === 'window' ) {
app . removeListener ( 'web-contents-created' , listener )
done ( )
}
} )
2017-10-27 20:45:58 +00:00
webview . src = ` file:// ${ fixtures } /pages/window-open.html `
2017-05-23 21:31:20 +00:00
document . body . appendChild ( webview )
} )
2017-05-23 20:24:51 +00:00
} )
2016-03-25 20:03:49 +00:00
} )