2017-02-14 19:44:15 -06:00
|
|
|
'use strict';
|
|
|
|
|
2016-11-07 14:31:39 -06:00
|
|
|
const electron = require('electron');
|
2017-02-20 12:24:46 -06:00
|
|
|
const app = electron.app;
|
|
|
|
const nodeURL = require('url');
|
2017-03-03 18:07:48 -06:00
|
|
|
const squirrelStartup = require('electron-squirrel-startup');
|
2017-05-09 16:44:14 -05:00
|
|
|
const AutoLaunch = require('auto-launch');
|
2017-05-13 13:23:44 -05:00
|
|
|
const urlParser = require('url');
|
2017-05-23 22:22:07 -05:00
|
|
|
const { getConfigField } = require('./config.js');
|
2017-05-31 12:38:57 -05:00
|
|
|
const { isMac, isDevEnv } = require('./utils/misc.js');
|
2017-05-13 13:23:44 -05:00
|
|
|
const protocolHandler = require('./protocolHandler');
|
2017-07-12 14:57:21 -05:00
|
|
|
const getCmdLineArg = require('./utils/getCmdLineArg.js');
|
|
|
|
const childProcess = require('child_process');
|
|
|
|
const path = require('path');
|
|
|
|
const AppDirectory = require('appdirectory');
|
|
|
|
const dirs = new AppDirectory('Symphony');
|
2017-05-13 13:23:44 -05:00
|
|
|
|
2017-06-27 10:08:58 -05:00
|
|
|
require('electron-dl')();
|
|
|
|
|
2017-05-13 13:23:44 -05:00
|
|
|
// used to check if a url was opened when the app was already open
|
|
|
|
let isAppAlreadyOpen = false;
|
2016-09-26 21:39:43 -05:00
|
|
|
|
2017-03-01 18:32:21 -06:00
|
|
|
// exit early for squirrel installer
|
2017-03-03 18:07:48 -06:00
|
|
|
if (squirrelStartup) {
|
2017-02-12 18:57:56 -06:00
|
|
|
return;
|
|
|
|
}
|
2016-11-07 14:31:39 -06:00
|
|
|
|
2017-03-01 18:32:21 -06:00
|
|
|
require('./mainApiMgr.js');
|
2017-02-14 19:44:15 -06:00
|
|
|
|
2017-03-01 18:32:21 -06:00
|
|
|
// monitor memory of main process
|
|
|
|
require('./memoryMonitor.js');
|
2017-02-10 20:20:09 -06:00
|
|
|
|
2017-03-01 18:32:21 -06:00
|
|
|
const windowMgr = require('./windowMgr.js');
|
2017-02-10 20:20:09 -06:00
|
|
|
|
2017-04-28 11:00:16 -05:00
|
|
|
// only allow a single instance of app.
|
2017-05-13 13:23:44 -05:00
|
|
|
const shouldQuit = app.makeSingleInstance((argv) => {
|
2017-04-28 11:00:16 -05:00
|
|
|
// Someone tried to run a second instance, we should focus our window.
|
|
|
|
let mainWin = windowMgr.getMainWindow();
|
|
|
|
if (mainWin) {
|
2017-05-13 13:23:44 -05:00
|
|
|
isAppAlreadyOpen = true;
|
2017-04-28 11:00:16 -05:00
|
|
|
if (mainWin.isMinimized()) {
|
|
|
|
mainWin.restore();
|
|
|
|
}
|
|
|
|
mainWin.focus();
|
|
|
|
}
|
2017-05-13 13:23:44 -05:00
|
|
|
processProtocolAction(argv);
|
2017-04-28 11:00:16 -05:00
|
|
|
});
|
|
|
|
|
2017-06-01 11:04:17 -05:00
|
|
|
// quit if another instance is already running, ignore for dev env
|
|
|
|
if (!isDevEnv && shouldQuit) {
|
2017-04-28 11:00:16 -05:00
|
|
|
app.quit();
|
|
|
|
}
|
|
|
|
|
2017-05-09 16:44:14 -05:00
|
|
|
var symphonyAutoLauncher = new AutoLaunch({
|
|
|
|
name: 'Symphony',
|
|
|
|
path: process.execPath,
|
|
|
|
});
|
|
|
|
|
2017-02-13 18:31:42 -06:00
|
|
|
/**
|
|
|
|
* This method will be called when Electron has finished
|
|
|
|
* initialization and is ready to create browser windows.
|
|
|
|
* Some APIs can only be used after this event occurs.
|
|
|
|
*/
|
2017-05-23 22:22:07 -05:00
|
|
|
app.on('ready', setupThenOpenMainWindow);
|
2017-02-20 12:24:46 -06:00
|
|
|
|
2017-05-23 22:22:07 -05:00
|
|
|
app.on('window-all-closed', function () {
|
|
|
|
app.quit();
|
|
|
|
});
|
|
|
|
|
|
|
|
app.on('activate', function () {
|
|
|
|
if (windowMgr.isMainWindow(null)) {
|
|
|
|
setupThenOpenMainWindow();
|
|
|
|
} else {
|
|
|
|
windowMgr.showMainWindow();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// adds 'symphony' as a protocol
|
|
|
|
// in the system. plist file in macOS
|
|
|
|
// and registry keys in windows
|
|
|
|
app.setAsDefaultProtocolClient('symphony');
|
|
|
|
|
|
|
|
// This event is emitted only on macOS
|
|
|
|
// at this moment, support for windows
|
|
|
|
// is in pipeline (https://github.com/electron/electron/pull/8052)
|
|
|
|
app.on('open-url', function (event, url) {
|
|
|
|
handleProtocolAction(url);
|
|
|
|
});
|
|
|
|
|
|
|
|
function setupThenOpenMainWindow() {
|
2017-05-13 13:23:44 -05:00
|
|
|
|
|
|
|
processProtocolAction(process.argv);
|
|
|
|
|
|
|
|
isAppAlreadyOpen = true;
|
|
|
|
|
2017-05-23 22:22:07 -05:00
|
|
|
// allows installer to launch app and set auto startup mode then
|
|
|
|
// immediately quit.
|
2017-05-31 23:39:08 -05:00
|
|
|
let hasInstallFlag = getCmdLineArg(process.argv, '--install', true);
|
|
|
|
if (!isMac && hasInstallFlag) {
|
|
|
|
getConfigField('launchOnStartup')
|
2017-07-12 14:57:21 -05:00
|
|
|
.then(setStartup)
|
|
|
|
.then(updateUserConfigWin)
|
|
|
|
.then(app.quit)
|
|
|
|
.catch(app.quit);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// allows mac installer to overwrite user config
|
|
|
|
if (isMac && hasInstallFlag) {
|
|
|
|
updateUserConfigMac()
|
|
|
|
.then(app.quit)
|
|
|
|
.catch(app.quit);
|
2017-05-31 23:39:08 -05:00
|
|
|
return;
|
2017-05-09 16:44:14 -05:00
|
|
|
}
|
2017-05-31 23:39:08 -05:00
|
|
|
|
|
|
|
getUrlAndCreateMainWindow();
|
2017-05-09 16:44:14 -05:00
|
|
|
}
|
|
|
|
|
2017-05-18 11:18:25 -05:00
|
|
|
function setStartup(lStartup){
|
|
|
|
return symphonyAutoLauncher.isEnabled()
|
|
|
|
.then(function(isEnabled){
|
|
|
|
if (!isEnabled && lStartup) {
|
|
|
|
return symphonyAutoLauncher.enable();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isEnabled && !lStartup) {
|
|
|
|
return symphonyAutoLauncher.disable();
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
});
|
2017-05-09 16:44:14 -05:00
|
|
|
}
|
|
|
|
|
2017-07-12 14:57:21 -05:00
|
|
|
// Method to overwrite user config on mac installer
|
|
|
|
function updateUserConfigMac() {
|
|
|
|
return new Promise((resolve, reject) => {
|
2017-07-13 00:13:56 -05:00
|
|
|
let userConfigPath = dirs.userConfig() + '/';
|
2017-07-12 14:57:21 -05:00
|
|
|
let globalConfigPath = process.argv[2];
|
|
|
|
let userName = process.env.USER;
|
|
|
|
|
2017-07-13 00:42:20 -05:00
|
|
|
childProcess.exec(`rsync -r "${globalConfigPath}" "${userConfigPath}" && chown -R "${userName}" "${userConfigPath}"`, {timeout: 60000}, (err) => {
|
2017-07-13 00:13:56 -05:00
|
|
|
if (err) {
|
|
|
|
reject(err);
|
2017-07-12 14:57:21 -05:00
|
|
|
}
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Method to overwrite user config on windows installer
|
|
|
|
function updateUserConfigWin() {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let userConfigPath = app.getPath('userData');
|
|
|
|
let globalConfigPath = path.join(__dirname, '..', '..', '..', 'config/Symphony.config');
|
|
|
|
|
2017-07-13 00:42:20 -05:00
|
|
|
childProcess.exec(`echo D|xcopy /y /e /s /c "${globalConfigPath}" "${userConfigPath}"`, {timeout: 60000}, (err) => {
|
2017-07-12 14:57:21 -05:00
|
|
|
if (err) {
|
|
|
|
reject(err);
|
|
|
|
}
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-05-23 22:22:07 -05:00
|
|
|
function getUrlAndCreateMainWindow() {
|
|
|
|
// for dev env allow passing url argument
|
2017-03-21 11:15:18 -05:00
|
|
|
if (isDevEnv) {
|
2017-05-31 23:39:08 -05:00
|
|
|
let url = getCmdLineArg(process.argv, '--url=')
|
2017-03-21 11:15:18 -05:00
|
|
|
if (url) {
|
2017-05-31 23:39:08 -05:00
|
|
|
windowMgr.createMainWindow(url.substr(6));
|
2017-03-21 11:15:18 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-20 13:54:11 -05:00
|
|
|
getConfigField('url')
|
2017-05-09 16:44:14 -05:00
|
|
|
.then(createWin).catch(function (err) {
|
|
|
|
let title = 'Error loading configuration';
|
|
|
|
electron.dialog.showErrorBox(title, title + ': ' + err);
|
|
|
|
});
|
2017-03-22 10:10:40 -05:00
|
|
|
}
|
|
|
|
|
2017-05-09 16:44:14 -05:00
|
|
|
function createWin(urlFromConfig) {
|
2017-03-22 10:10:40 -05:00
|
|
|
let protocol = '';
|
|
|
|
// add https protocol if none found.
|
2017-04-20 13:54:11 -05:00
|
|
|
let parsedUrl = nodeURL.parse(urlFromConfig);
|
2017-03-22 10:10:40 -05:00
|
|
|
if (!parsedUrl.protocol) {
|
|
|
|
protocol = 'https';
|
|
|
|
}
|
|
|
|
var url = nodeURL.format({
|
|
|
|
protocol: protocol,
|
|
|
|
slahes: true,
|
|
|
|
pathname: parsedUrl.href
|
2017-02-20 12:24:46 -06:00
|
|
|
});
|
2017-05-31 23:39:08 -05:00
|
|
|
|
2017-03-22 10:10:40 -05:00
|
|
|
windowMgr.createMainWindow(url);
|
2017-02-20 12:24:46 -06:00
|
|
|
}
|
2016-11-08 18:32:04 -06:00
|
|
|
|
2017-05-13 13:23:44 -05:00
|
|
|
/**
|
|
|
|
* processes protocol action for windows clients
|
|
|
|
* @param argv {Array} an array of command line arguments
|
|
|
|
*/
|
|
|
|
function processProtocolAction(argv) {
|
|
|
|
|
|
|
|
// In case of windows, we need to handle protocol handler
|
|
|
|
// manually because electron doesn't emit
|
|
|
|
// 'open-url' event on windows
|
|
|
|
if (!(process.platform === 'win32')) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-05-31 23:39:08 -05:00
|
|
|
let protocolUri = getCmdLineArg(argv, 'symphony://');
|
2017-05-13 13:23:44 -05:00
|
|
|
|
|
|
|
if (protocolUri) {
|
|
|
|
|
|
|
|
const parsedURL = urlParser.parse(protocolUri);
|
|
|
|
|
|
|
|
if (!parsedURL.protocol || !parsedURL.slashes) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
handleProtocolAction(protocolUri);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function handleProtocolAction(uri) {
|
|
|
|
if (!isAppAlreadyOpen) {
|
|
|
|
// app is opened by the protocol url, cache the protocol url to be used later
|
|
|
|
protocolHandler.setProtocolUrl(uri);
|
|
|
|
} else {
|
|
|
|
// app is already open, so, just trigger the protocol action method
|
|
|
|
protocolHandler.processProtocolAction(uri);
|
|
|
|
}
|
|
|
|
}
|