2018-03-09 20:59:12 +00:00
|
|
|
const { _ } = require('lib/locale.js');
|
|
|
|
const { BrowserWindow, Menu, Tray } = require('electron');
|
|
|
|
const { shim } = require('lib/shim');
|
|
|
|
const url = require('url')
|
|
|
|
const path = require('path')
|
|
|
|
const urlUtils = require('lib/urlUtils.js');
|
|
|
|
const { dirname, basename } = require('lib/path-utils');
|
|
|
|
const fs = require('fs-extra');
|
2017-11-04 12:38:53 +00:00
|
|
|
|
|
|
|
class ElectronAppWrapper {
|
2018-03-09 20:59:12 +00:00
|
|
|
|
2017-11-13 18:47:35 +00:00
|
|
|
constructor(electronApp, env) {
|
2017-11-04 12:38:53 +00:00
|
|
|
this.electronApp_ = electronApp;
|
2017-11-13 18:47:35 +00:00
|
|
|
this.env_ = env;
|
2017-11-04 12:38:53 +00:00
|
|
|
this.win_ = null;
|
2017-11-17 18:05:25 +00:00
|
|
|
this.willQuitApp_ = false;
|
2018-01-31 20:10:32 +00:00
|
|
|
this.tray_ = null;
|
2018-02-06 13:11:59 +00:00
|
|
|
this.buildDir_ = null;
|
2017-11-04 13:23:15 +00:00
|
|
|
}
|
2017-11-04 12:38:53 +00:00
|
|
|
|
2017-11-04 13:23:15 +00:00
|
|
|
electronApp() {
|
|
|
|
return this.electronApp_;
|
|
|
|
}
|
|
|
|
|
|
|
|
setLogger(v) {
|
|
|
|
this.logger_ = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
logger() {
|
|
|
|
return this.logger_;
|
2017-11-04 12:38:53 +00:00
|
|
|
}
|
|
|
|
|
2017-11-05 00:17:48 +00:00
|
|
|
window() {
|
|
|
|
return this.win_;
|
2017-11-04 19:46:37 +00:00
|
|
|
}
|
|
|
|
|
2017-11-04 12:38:53 +00:00
|
|
|
createWindow() {
|
2018-03-09 20:59:12 +00:00
|
|
|
const windowStateKeeper = require('electron-window-state');
|
2017-11-14 18:02:58 +00:00
|
|
|
|
|
|
|
// Load the previous state with fallback to defaults
|
|
|
|
const windowState = windowStateKeeper({
|
|
|
|
defaultWidth: 800,
|
|
|
|
defaultHeight: 600,
|
2018-03-09 20:59:12 +00:00
|
|
|
file: 'window-state-' + this.env_ + '.json',
|
2017-11-14 18:02:58 +00:00
|
|
|
});
|
|
|
|
|
2017-12-15 07:31:57 +00:00
|
|
|
const windowOptions = {
|
2018-02-27 20:04:38 +00:00
|
|
|
x: windowState.x,
|
|
|
|
y: windowState.y,
|
|
|
|
width: windowState.width,
|
|
|
|
height: windowState.height,
|
2017-12-15 07:31:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Linux icon workaround for bug https://github.com/electron-userland/electron-builder/issues/2098
|
|
|
|
// Fix: https://github.com/electron-userland/electron-builder/issues/2269
|
2018-03-09 20:59:12 +00:00
|
|
|
if (shim.isLinux()) windowOptions.icon = __dirname + '/build/icons/128x128.png';
|
2017-12-15 07:31:57 +00:00
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
this.win_ = new BrowserWindow(windowOptions)
|
2017-11-04 12:38:53 +00:00
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
this.win_.loadURL(url.format({
|
|
|
|
pathname: path.join(__dirname, 'index.html'),
|
|
|
|
protocol: 'file:',
|
|
|
|
slashes: true
|
|
|
|
}))
|
2017-11-04 12:38:53 +00:00
|
|
|
|
2017-11-27 22:50:46 +00:00
|
|
|
// Uncomment this to view errors if the application does not start
|
2018-02-16 18:08:02 +00:00
|
|
|
// if (this.env_ === 'dev') this.win_.webContents.openDevTools();
|
2017-11-04 12:38:53 +00:00
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
this.win_.on('close', (event) => {
|
2018-01-31 20:10:32 +00:00
|
|
|
// If it's on macOS, the app is completely closed only if the user chooses to close the app (willQuitApp_ will be true)
|
|
|
|
// otherwise the window is simply hidden, and will be re-open once the app is "activated" (which happens when the
|
|
|
|
// user clicks on the icon in the task bar).
|
|
|
|
|
|
|
|
// On Windows and Linux, the app is closed when the window is closed *except* if the tray icon is used. In which
|
|
|
|
// case the app must be explicitely closed with Ctrl+Q or by right-clicking on the tray icon and selecting "Exit".
|
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
if (process.platform === 'darwin') {
|
2018-01-31 20:10:32 +00:00
|
|
|
if (this.willQuitApp_) {
|
|
|
|
this.win_ = null;
|
|
|
|
} else {
|
|
|
|
event.preventDefault();
|
2018-02-27 21:54:40 +00:00
|
|
|
this.hide();
|
2018-01-31 20:10:32 +00:00
|
|
|
}
|
2017-11-17 18:05:25 +00:00
|
|
|
} else {
|
2018-01-31 20:10:32 +00:00
|
|
|
if (this.trayShown() && !this.willQuitApp_) {
|
|
|
|
event.preventDefault();
|
|
|
|
this.win_.hide();
|
|
|
|
} else {
|
|
|
|
this.win_ = null;
|
|
|
|
}
|
2017-11-17 18:05:25 +00:00
|
|
|
}
|
2018-03-09 20:59:12 +00:00
|
|
|
})
|
2017-11-14 18:02:58 +00:00
|
|
|
|
|
|
|
// Let us register listeners on the window, so we can update the state
|
|
|
|
// automatically (the listeners will be removed when the window is closed)
|
|
|
|
// and restore the maximized or full screen state
|
|
|
|
windowState.manage(this.win_);
|
2017-11-04 12:38:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async waitForElectronAppReady() {
|
2017-11-04 13:23:15 +00:00
|
|
|
if (this.electronApp().isReady()) return Promise.resolve();
|
2017-11-04 12:38:53 +00:00
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const iid = setInterval(() => {
|
2017-11-04 13:23:15 +00:00
|
|
|
if (this.electronApp().isReady()) {
|
2017-11-04 12:38:53 +00:00
|
|
|
clearInterval(iid);
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
}, 10);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-03-07 19:11:55 +00:00
|
|
|
async quit() {
|
2017-11-04 13:23:15 +00:00
|
|
|
this.electronApp_.quit();
|
|
|
|
}
|
|
|
|
|
2018-03-07 19:11:55 +00:00
|
|
|
exit(errorCode = 0) {
|
|
|
|
this.electronApp_.exit(errorCode);
|
|
|
|
}
|
|
|
|
|
2018-01-31 20:10:32 +00:00
|
|
|
trayShown() {
|
|
|
|
return !!this.tray_;
|
|
|
|
}
|
|
|
|
|
2018-02-27 21:54:40 +00:00
|
|
|
// This method is used in macOS only to hide the whole app (and not just the main window)
|
|
|
|
// including the menu bar. This follows the macOS way of hidding an app.
|
|
|
|
hide() {
|
|
|
|
this.electronApp_.hide();
|
|
|
|
}
|
|
|
|
|
2018-02-05 22:19:21 +00:00
|
|
|
buildDir() {
|
2018-02-06 13:11:59 +00:00
|
|
|
if (this.buildDir_) return this.buildDir_;
|
2018-03-09 20:59:12 +00:00
|
|
|
let dir = __dirname + '/build';
|
2018-02-06 13:11:59 +00:00
|
|
|
if (!fs.pathExistsSync(dir)) {
|
2018-03-09 20:59:12 +00:00
|
|
|
dir = dirname(__dirname) + '/build';
|
|
|
|
if (!fs.pathExistsSync(dir)) throw new Error('Cannot find build dir');
|
2018-02-06 13:11:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
this.buildDir_ = dir;
|
|
|
|
return dir;
|
2018-02-05 22:19:21 +00:00
|
|
|
}
|
|
|
|
|
2018-02-20 00:41:52 +00:00
|
|
|
trayIconFilename_() {
|
2018-03-09 20:59:12 +00:00
|
|
|
let output = '';
|
2018-02-20 00:41:52 +00:00
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
if (process.platform === 'darwin') {
|
|
|
|
output = 'macos-16x16Template.png'; // Electron Template Image format
|
2018-02-20 00:41:52 +00:00
|
|
|
} else {
|
2018-03-09 20:59:12 +00:00
|
|
|
output = '16x16.png';
|
2018-02-20 00:41:52 +00:00
|
|
|
}
|
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
if (this.env_ === 'dev') output = '16x16-dev.png'
|
2018-02-20 00:41:52 +00:00
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2018-01-31 20:10:32 +00:00
|
|
|
// Note: this must be called only after the "ready" event of the app has been dispatched
|
|
|
|
createTray(contextMenu) {
|
2018-02-06 13:11:59 +00:00
|
|
|
try {
|
2018-03-09 20:59:12 +00:00
|
|
|
this.tray_ = new Tray(this.buildDir() + '/icons/' + this.trayIconFilename_())
|
|
|
|
this.tray_.setToolTip(this.electronApp_.getName())
|
|
|
|
this.tray_.setContextMenu(contextMenu)
|
2018-02-06 13:11:59 +00:00
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
this.tray_.on('click', () => {
|
2018-02-06 13:11:59 +00:00
|
|
|
this.window().show();
|
|
|
|
});
|
|
|
|
} catch (error) {
|
|
|
|
console.error("Cannot create tray", error);
|
|
|
|
}
|
2018-01-31 20:10:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
destroyTray() {
|
|
|
|
if (!this.tray_) return;
|
|
|
|
this.tray_.destroy();
|
|
|
|
this.tray_ = null;
|
|
|
|
}
|
|
|
|
|
2018-02-23 17:51:23 +00:00
|
|
|
ensureSingleInstance() {
|
2018-03-09 20:59:12 +00:00
|
|
|
if (this.env_ === 'dev') return false;
|
2018-03-01 20:14:06 +00:00
|
|
|
|
2018-02-23 17:51:23 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const alreadyRunning = this.electronApp_.makeSingleInstance((commandLine, workingDirectory) => {
|
|
|
|
const win = this.window();
|
|
|
|
if (!win) return;
|
|
|
|
if (win.isMinimized()) win.restore();
|
|
|
|
win.show();
|
|
|
|
win.focus();
|
|
|
|
});
|
|
|
|
|
|
|
|
if (alreadyRunning) this.electronApp_.quit();
|
|
|
|
|
|
|
|
resolve(alreadyRunning);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-11-04 12:38:53 +00:00
|
|
|
async start() {
|
|
|
|
// Since we are doing other async things before creating the window, we might miss
|
|
|
|
// the "ready" event. So we use the function below to make sure that the app is ready.
|
|
|
|
await this.waitForElectronAppReady();
|
|
|
|
|
2018-02-23 17:51:23 +00:00
|
|
|
const alreadyRunning = await this.ensureSingleInstance();
|
|
|
|
if (alreadyRunning) return;
|
|
|
|
|
2017-11-04 12:38:53 +00:00
|
|
|
this.createWindow();
|
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
this.electronApp_.on('before-quit', () => {
|
2017-11-17 18:05:25 +00:00
|
|
|
this.willQuitApp_ = true;
|
2018-03-09 20:59:12 +00:00
|
|
|
})
|
2017-11-17 18:05:25 +00:00
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
this.electronApp_.on('window-all-closed', () => {
|
2017-11-17 18:05:25 +00:00
|
|
|
this.electronApp_.quit();
|
2018-03-09 20:59:12 +00:00
|
|
|
})
|
2017-11-04 12:38:53 +00:00
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
this.electronApp_.on('activate', () => {
|
2017-11-17 18:05:25 +00:00
|
|
|
this.win_.show();
|
2018-03-09 20:59:12 +00:00
|
|
|
})
|
2017-11-04 12:38:53 +00:00
|
|
|
}
|
2018-03-09 20:59:12 +00:00
|
|
|
|
2017-11-04 12:38:53 +00:00
|
|
|
}
|
|
|
|
|
2018-03-09 20:59:12 +00:00
|
|
|
module.exports = { ElectronAppWrapper };
|