electron/default_app/main.js

340 lines
8.6 KiB
JavaScript
Raw Normal View History

const {app, dialog, shell, Menu} = require('electron')
const fs = require('fs')
2016-05-24 16:33:55 +00:00
const Module = require('module')
const path = require('path')
const repl = require('repl')
const url = require('url')
// Parse command line options.
const argv = process.argv.slice(1)
2016-06-07 12:45:38 +00:00
const option = { file: null, help: null, version: null, abi: null, webdriver: null, modules: [] }
for (let i = 0; i < argv.length; i++) {
2016-03-27 17:38:32 +00:00
if (argv[i] === '--version' || argv[i] === '-v') {
option.version = true
break
2016-06-07 12:45:38 +00:00
} else if (argv[i] === '--abi') {
option.abi = true
break
} else if (argv[i].match(/^--app=/)) {
option.file = argv[i].split('=')[1]
break
2016-03-27 17:38:32 +00:00
} else if (argv[i] === '--help' || argv[i] === '-h') {
option.help = true
break
2016-03-27 17:38:32 +00:00
} else if (argv[i] === '--interactive' || argv[i] === '-i') {
option.interactive = true
2016-03-27 17:38:32 +00:00
} else if (argv[i] === '--test-type=webdriver') {
option.webdriver = true
2016-03-27 17:38:32 +00:00
} else if (argv[i] === '--require' || argv[i] === '-r') {
option.modules.push(argv[++i])
continue
2016-03-27 17:38:32 +00:00
} else if (argv[i][0] === '-') {
continue
} else {
option.file = argv[i]
break
}
}
// Quit when all windows are closed and no other one is listening to this.
app.on('window-all-closed', () => {
2016-03-27 17:38:32 +00:00
if (app.listeners('window-all-closed').length === 1 && !option.interactive) {
app.quit()
2016-03-27 17:38:32 +00:00
}
})
// Create default menu.
app.once('ready', () => {
2016-03-27 17:38:32 +00:00
if (Menu.getApplicationMenu()) return
const template = [
2015-09-01 15:34:56 +00:00
{
label: 'Edit',
submenu: [
{
label: 'Undo',
accelerator: 'CmdOrCtrl+Z',
role: 'undo'
},
{
label: 'Redo',
accelerator: 'Shift+CmdOrCtrl+Z',
role: 'redo'
},
{
type: 'separator'
},
{
label: 'Cut',
accelerator: 'CmdOrCtrl+X',
role: 'cut'
},
{
label: 'Copy',
accelerator: 'CmdOrCtrl+C',
role: 'copy'
},
{
label: 'Paste',
accelerator: 'CmdOrCtrl+V',
role: 'paste'
},
{
label: 'Select All',
accelerator: 'CmdOrCtrl+A',
role: 'selectall'
2016-03-27 17:38:32 +00:00
}
2015-09-01 15:34:56 +00:00
]
},
{
label: 'View',
submenu: [
{
label: 'Reload',
accelerator: 'CmdOrCtrl+R',
click (item, focusedWindow) {
2016-03-27 17:38:32 +00:00
if (focusedWindow) focusedWindow.reload()
}
2015-09-01 15:34:56 +00:00
},
{
label: 'Toggle Full Screen',
accelerator: (() => {
2016-03-28 20:19:41 +00:00
return (process.platform === 'darwin') ? 'Ctrl+Command+F' : 'F11'
2015-09-01 15:34:56 +00:00
})(),
click (item, focusedWindow) {
2016-03-28 20:19:41 +00:00
if (focusedWindow) focusedWindow.setFullScreen(!focusedWindow.isFullScreen())
2015-09-01 15:34:56 +00:00
}
},
{
label: 'Toggle Developer Tools',
accelerator: (() => {
2016-03-28 20:19:41 +00:00
return (process.platform === 'darwin') ? 'Alt+Command+I' : 'Ctrl+Shift+I'
2015-09-01 15:34:56 +00:00
})(),
click (item, focusedWindow) {
2016-03-28 20:19:41 +00:00
if (focusedWindow) focusedWindow.toggleDevTools()
2015-09-01 15:34:56 +00:00
}
2016-03-28 20:19:41 +00:00
}
2015-09-01 15:34:56 +00:00
]
},
{
label: 'Window',
role: 'window',
submenu: [
{
label: 'Minimize',
accelerator: 'CmdOrCtrl+M',
role: 'minimize'
},
{
label: 'Close',
accelerator: 'CmdOrCtrl+W',
role: 'close'
2016-03-28 20:19:41 +00:00
}
2015-09-01 15:34:56 +00:00
]
},
{
label: 'Help',
role: 'help',
submenu: [
{
label: 'Learn More',
click () {
shell.openExternal('http://electron.atom.io')
2016-01-19 22:53:59 +00:00
}
2015-09-01 15:34:56 +00:00
},
{
label: 'Documentation',
click () {
shell.openExternal(
2016-03-31 23:50:30 +00:00
`https://github.com/electron/electron/tree/v${process.versions.electron}/docs#readme`
)
}
2015-09-01 15:34:56 +00:00
},
{
label: 'Community Discussions',
click () {
shell.openExternal('https://discuss.atom.io/c/electron')
2016-01-19 22:53:59 +00:00
}
2015-09-01 15:34:56 +00:00
},
{
label: 'Search Issues',
click () {
2016-03-31 23:50:30 +00:00
shell.openExternal('https://github.com/electron/electron/issues')
2016-01-19 22:53:59 +00:00
}
2015-09-01 15:34:56 +00:00
}
]
2016-03-28 20:19:41 +00:00
}
]
2015-09-01 15:34:56 +00:00
2016-03-27 17:38:32 +00:00
if (process.platform === 'darwin') {
2015-09-01 15:34:56 +00:00
template.unshift({
label: 'Electron',
submenu: [
{
label: 'About Electron',
role: 'about'
},
{
type: 'separator'
},
{
label: 'Services',
role: 'services',
submenu: []
},
{
type: 'separator'
},
{
label: 'Hide Electron',
accelerator: 'Command+H',
role: 'hide'
},
{
label: 'Hide Others',
2016-01-21 22:48:06 +00:00
accelerator: 'Command+Alt+H',
role: 'hideothers'
2015-09-01 15:34:56 +00:00
},
{
label: 'Show All',
role: 'unhide'
2015-09-01 15:34:56 +00:00
},
{
type: 'separator'
},
{
label: 'Quit ' + app.getName(),
2015-09-01 15:34:56 +00:00
accelerator: 'Command+Q',
click () { app.quit() }
2016-03-28 20:19:41 +00:00
}
2015-09-01 15:34:56 +00:00
]
})
2015-09-01 15:34:56 +00:00
template[3].submenu.push(
{
2015-09-01 15:34:56 +00:00
type: 'separator'
},
{
2015-09-01 15:34:56 +00:00
label: 'Bring All to Front',
role: 'front'
}
)
}
const menu = Menu.buildFromTemplate(template)
Menu.setApplicationMenu(menu)
})
if (option.modules.length > 0) {
2016-05-24 16:33:55 +00:00
Module._preloadModules(option.modules)
}
function loadApplicationPackage (packagePath) {
// Add a flag indicating app is started from default app.
process.defaultApp = true
try {
// Override app name and version.
packagePath = path.resolve(packagePath)
const packageJsonPath = path.join(packagePath, 'package.json')
if (fs.existsSync(packageJsonPath)) {
let packageJson
try {
packageJson = JSON.parse(fs.readFileSync(packageJsonPath))
} catch (e) {
2016-05-24 16:38:04 +00:00
showErrorMessage(`Unable to parse ${packageJsonPath}\n\n${e.message}`)
return
}
2016-03-28 20:19:41 +00:00
2016-05-24 16:42:31 +00:00
if (packageJson.version) {
app.setVersion(packageJson.version)
}
2016-03-28 20:19:41 +00:00
if (packageJson.productName) {
app.setName(packageJson.productName)
2016-03-28 20:19:41 +00:00
} else if (packageJson.name) {
app.setName(packageJson.name)
2016-03-28 20:19:41 +00:00
}
app.setPath('userData', path.join(app.getPath('appData'), app.getName()))
app.setPath('userCache', path.join(app.getPath('cache'), app.getName()))
app.setAppPath(packagePath)
}
2016-05-24 16:33:55 +00:00
try {
Module._resolveFilename(packagePath, module, true)
} catch (e) {
showErrorMessage(`Unable to find Electron app at ${packagePath}\n\n${e.message}`)
2016-05-24 16:41:31 +00:00
return
}
2016-05-24 16:41:31 +00:00
// Run the app.
Module._load(packagePath, module, true)
2016-03-29 01:00:30 +00:00
} catch (e) {
console.error('App threw an error during load')
2016-05-24 16:51:09 +00:00
console.error(e.stack || e)
throw e
}
}
function showErrorMessage (message) {
app.focus()
2016-05-24 16:37:52 +00:00
dialog.showErrorBox('Error launching app', message)
process.exit(1)
}
function loadApplicationByUrl (appUrl) {
require('./default_app').load(appUrl)
}
function startRepl () {
repl.start('> ').on('exit', () => {
process.exit(0)
})
2016-03-22 20:34:46 +00:00
}
// Start the specified app if there is one specified in command line, otherwise
// start the default app.
if (option.file && !option.webdriver) {
const file = option.file
const protocol = url.parse(file).protocol
const extension = path.extname(file)
2016-02-04 18:34:36 +00:00
if (protocol === 'http:' || protocol === 'https:' || protocol === 'file:') {
loadApplicationByUrl(file)
} else if (extension === '.html' || extension === '.htm') {
loadApplicationByUrl('file://' + path.resolve(file))
} else {
loadApplicationPackage(file)
}
} else if (option.version) {
console.log('v' + process.versions.electron)
process.exit(0)
2016-06-07 12:45:38 +00:00
} else if (option.abi) {
console.log(process.versions.modules)
process.exit(0)
2015-04-24 21:02:49 +00:00
} else if (option.help) {
2016-05-24 16:43:25 +00:00
const helpMessage = `Electron ${process.versions.electron} - Build cross platform desktop apps with JavaScript, HTML, and CSS
Usage: electron [options] [path]
2016-05-24 16:53:23 +00:00
A path to an Electron app may be specified. The path must be one of the following:
- index.js file.
- Folder containing a package.json file.
- Folder containing an index.js file.
- .html/.htm file.
- http://, https://, or file:// URL.
Options:
-h, --help Print this usage message.
-i, --interactive Open a REPL to the main process.
-r, --require Module to preload (option can be repeated)
2016-06-07 12:45:38 +00:00
-v, --version Print the version.
--abi Print the application binary interface.`
console.log(helpMessage)
process.exit(0)
2016-03-22 20:22:39 +00:00
} else if (option.interactive) {
startRepl()
} else {
const indexPath = path.join(__dirname, '/index.html')
2016-03-29 01:00:30 +00:00
loadApplicationByUrl(`file://${indexPath}`)
2013-04-15 07:39:54 +00:00
}