SymphonyElectron/js/windowMgr.js

459 lines
14 KiB
JavaScript
Raw Normal View History

2017-03-01 18:32:21 -06:00
'use strict';
const electron = require('electron');
const app = electron.app;
const BrowserWindow = electron.BrowserWindow;
2017-03-01 18:32:21 -06:00
const path = require('path');
2017-04-18 11:02:25 -05:00
const nodeURL = require('url');
const querystring = require('querystring');
const filesize = require('filesize');
2017-03-01 18:32:21 -06:00
const { getTemplate, getMinimizeOnClose } = require('./menus/menuTemplate.js');
2017-03-01 18:32:21 -06:00
const loadErrors = require('./dialogs/showLoadError.js');
const isInDisplayBounds = require('./utils/isInDisplayBounds.js');
2017-03-09 12:12:28 -06:00
const getGuid = require('./utils/getGuid.js');
const log = require('./log.js');
2017-03-01 18:32:21 -06:00
const logLevels = require('./enums/logLevels.js');
2017-04-06 12:07:58 -05:00
const notify = require('./notify/electron-notify.js');
const eventEmitter = require('./eventEmitter');
const throttle = require('./utils/throttle.js');
const { getConfigField, updateConfigField } = require('./config.js');
2017-03-01 18:32:21 -06:00
//context menu
const contextMenu = require('./menus/contextMenu.js');
2017-03-01 18:32:21 -06:00
// show dialog when certificate errors occur
require('./dialogs/showCertError.js');
// Keep a global reference of the window object, if you don't, the window will
// be closed automatically when the JavaScript object is garbage collected.
let mainWindow;
let windows = {};
let willQuitApp = false;
let isOnline = true;
let boundsChangeWindow;
let alwaysOnTop = false;
let position = 'lower-right';
let display;
2017-03-07 16:44:31 -06:00
2017-04-18 11:02:25 -05:00
// note: this file is built using browserify in prebuild step.
2017-03-07 16:44:31 -06:00
const preloadMainScript = path.join(__dirname, 'preload/_preloadMain.js');
2017-03-01 18:32:21 -06:00
const MIN_WIDTH = 300;
const MIN_HEIGHT = 600;
2017-03-01 18:32:21 -06:00
function addWindowKey(key, browserWin) {
windows[key] = browserWin;
2017-03-01 18:32:21 -06:00
}
function removeWindowKey(key) {
delete windows[key];
2017-03-01 18:32:21 -06:00
}
function getParsedUrl(url) {
return nodeURL.parse(url);
2017-04-18 11:02:25 -05:00
}
function createMainWindow(initialUrl) {
getConfigField('mainWinPos').then(
function (bounds) {
doCreateMainWindow(initialUrl, bounds);
},
function () {
// failed, use default bounds
doCreateMainWindow(initialUrl, null);
}
)
}
function doCreateMainWindow(initialUrl, initialBounds) {
2017-04-18 11:02:25 -05:00
let url = initialUrl;
2017-03-01 18:32:21 -06:00
let key = getGuid();
log.send(logLevels.INFO, 'creating main window url: ' + url);
2017-04-18 11:02:25 -05:00
let newWinOpts = {
2017-03-01 18:32:21 -06:00
title: 'Symphony',
show: true,
minWidth: MIN_WIDTH,
minHeight: MIN_HEIGHT,
alwaysOnTop: false,
2017-03-01 18:32:21 -06:00
webPreferences: {
sandbox: true,
nodeIntegration: false,
2017-03-07 16:44:31 -06:00
preload: preloadMainScript,
2017-03-01 18:32:21 -06:00
}
2017-04-18 11:02:25 -05:00
};
// set size and position
let bounds = initialBounds;
// if bounds if not fully contained in some display then use default size
// and position.
if (!isInDisplayBounds(bounds)) {
bounds = null;
}
if (bounds && bounds.width && bounds.height) {
newWinOpts.width = bounds.width;
newWinOpts.height = bounds.height;
} else {
newWinOpts.width = 1024;
newWinOpts.height = 768;
}
// will center on screen if values not provided
if (bounds && bounds.x && bounds.y) {
newWinOpts.x = bounds.x;
newWinOpts.y = bounds.y;
}
// will set the main window on top as per the user prefs
if (alwaysOnTop){
newWinOpts.alwaysOnTop = alwaysOnTop;
}
2017-04-18 11:02:25 -05:00
// note: augmenting with some custom values
newWinOpts.winKey = key;
mainWindow = new BrowserWindow(newWinOpts);
2017-04-18 11:02:25 -05:00
mainWindow.winName = 'main';
2017-03-01 18:32:21 -06:00
let throttledMainWinBoundsChange = throttle(5000, saveMainWinBounds);
mainWindow.on('move', throttledMainWinBoundsChange);
mainWindow.on('resize', throttledMainWinBoundsChange);
2017-03-01 18:32:21 -06:00
function retry() {
2017-03-03 18:07:48 -06:00
if (!isOnline) {
2017-03-01 18:32:21 -06:00
loadErrors.showNetworkConnectivityError(mainWindow, url, retry);
2017-03-03 18:07:48 -06:00
return;
}
if (mainWindow.webContents) {
mainWindow.webContents.reload();
2017-03-01 18:32:21 -06:00
}
}
// content can be cached and will still finish load but
// we might not have network connectivity, so warn the user.
mainWindow.webContents.on('did-finish-load', function () {
2017-04-18 11:02:25 -05:00
url = mainWindow.webContents.getURL();
2017-03-01 18:32:21 -06:00
if (!isOnline) {
loadErrors.showNetworkConnectivityError(mainWindow, url, retry);
} else {
// updates the notify config with user preference
notify.updateConfig({position: position, display: display});
2017-04-06 12:07:58 -05:00
// removes all existing notifications when main window reloads
notify.reset();
log.send(logLevels.INFO, 'loaded main window url: ' + url);
2017-03-01 18:32:21 -06:00
}
});
mainWindow.webContents.on('did-fail-load', function (event, errorCode,
errorDesc, validatedURL) {
2017-03-09 12:12:28 -06:00
loadErrors.showLoadFailure(mainWindow, validatedURL, errorDesc, errorCode, retry);
2017-03-01 18:32:21 -06:00
});
addWindowKey(key, mainWindow);
mainWindow.loadURL(url);
const menu = electron.Menu.buildFromTemplate(getTemplate(app));
2017-03-01 18:32:21 -06:00
electron.Menu.setApplicationMenu(menu);
mainWindow.on('close', function(e) {
2017-03-01 18:32:21 -06:00
if (willQuitApp) {
2017-04-18 11:02:25 -05:00
destroyAllWindows();
2017-03-01 18:32:21 -06:00
return;
}
if (getMinimizeOnClose()) {
2017-03-01 18:32:21 -06:00
e.preventDefault();
mainWindow.minimize();
} else {
app.quit();
2017-03-01 18:32:21 -06:00
}
});
2017-04-18 11:02:25 -05:00
function destroyAllWindows() {
let keys = Object.keys(windows);
for (var i = 0, len = keys.length; i < len; i++) {
2017-04-18 11:02:25 -05:00
let winKey = keys[i];
removeWindowKey(winKey);
2017-03-03 18:07:48 -06:00
}
2017-04-18 11:02:25 -05:00
mainWindow = null;
}
2017-03-01 18:32:21 -06:00
2017-04-18 11:02:25 -05:00
mainWindow.on('closed', destroyAllWindows);
// Manage File Downloads
mainWindow.webContents.session.on('will-download', (event, item, webContents) => {
// When download is in progress, send necessary data to indicate the same
webContents.send('downloadProgress');
// Send file path when download is complete
item.once('done', (event, state) => {
if (state === 'completed') {
let data = {
_id: getGuid(),
savedPath: item.getSavePath() ? item.getSavePath() : '',
total: filesize(item.getTotalBytes() ? item.getTotalBytes() : 0),
fileName: item.getFilename() ? item.getFilename() : 'No name'
};
webContents.send('downloadCompleted', data);
}
});
});
// bug in electron is preventing this from working in sandboxed evt...
// https://github.com/electron/electron/issues/8841
mainWindow.webContents.on('will-navigate', function(event, willNavUrl) {
event.preventDefault();
openUrlInDefaultBrower(willNavUrl);
});
// open external links in default browser - a tag with href='_blank' or window.open
mainWindow.webContents.on('new-window', function (event, newWinUrl,
frameName, disposition, newWinOptions) {
let newWinParsedUrl = getParsedUrl(newWinUrl);
let mainWinParsedUrl = getParsedUrl(url);
let newWinHost = newWinParsedUrl && newWinParsedUrl.host;
let mainWinHost = mainWinParsedUrl && mainWinParsedUrl.host;
2017-04-18 11:02:25 -05:00
// only allow window.open to succeed is if coming from same hsot,
// otherwise open in default browser.
if (disposition === 'new-window' && newWinHost === mainWinHost) {
2017-04-18 11:02:25 -05:00
// handle: window.open
if (!frameName) {
// abort - no frame name provided.
return;
}
log.send(logLevels.INFO, 'creating pop-out window url: ' + newWinParsedUrl);
let x = 0;
let y = 0;
let width = newWinOptions.width || MIN_WIDTH;
let height = newWinOptions.height || MIN_HEIGHT;
// try getting x and y position from query parameters
var query = newWinParsedUrl && querystring.parse(newWinParsedUrl.query);
if (query && query.x && query.y) {
let newX = Number.parseInt(query.x, 10);
let newY = Number.parseInt(query.y, 10);
let newWinRect = {x: newX, y: newY, width, height};
// only accept if both are successfully parsed.
if (Number.isInteger(newX) && Number.isInteger(newY) &&
isInDisplayBounds(newWinRect)) {
x = newX;
y = newY;
} else {
x = 0;
y = 0;
}
} else {
// create new window at slight offset from main window.
({x, y} = getWindowSizeAndPosition(mainWindow));
x += 50;
y += 50;
}
2017-04-18 11:02:25 -05:00
/* eslint-disable no-param-reassign */
newWinOptions.x = x;
newWinOptions.y = y;
newWinOptions.width = Math.max(width, MIN_WIDTH);
newWinOptions.height = Math.max(height, MIN_HEIGHT);
newWinOptions.minWidth = MIN_WIDTH;
newWinOptions.minHeight = MIN_HEIGHT;
newWinOptions.alwaysOnTop = alwaysOnTop;
2017-04-18 11:02:25 -05:00
let newWinKey = getGuid();
2017-04-18 11:02:25 -05:00
newWinOptions.winKey = newWinKey;
/* eslint-enable no-param-reassign */
let webContents = newWinOptions.webContents;
2017-04-18 11:02:25 -05:00
webContents.once('did-finish-load', function () {
let browserWin = BrowserWindow.fromWebContents(webContents);
2017-04-18 11:02:25 -05:00
if (browserWin) {
log.send(logLevels.INFO, 'loaded pop-out window url: ' + newWinParsedUrl);
browserWin.winName = frameName;
browserWin.setAlwaysOnTop(alwaysOnTop);
2017-04-18 11:02:25 -05:00
browserWin.once('closed', function () {
removeWindowKey(newWinKey);
browserWin.removeListener('move', throttledBoundsChange);
browserWin.removeListener('resize', throttledBoundsChange);
});
2017-04-18 11:02:25 -05:00
addWindowKey(newWinKey, browserWin);
// throttle changes so we don't flood client.
let throttledBoundsChange = throttle(1000,
sendChildWinBoundsChange.bind(null, browserWin));
browserWin.on('move', throttledBoundsChange);
browserWin.on('resize', throttledBoundsChange);
}
});
} else {
event.preventDefault();
openUrlInDefaultBrower(newWinUrl)
2017-04-18 11:02:25 -05:00
}
2017-03-01 18:32:21 -06:00
});
2017-04-18 11:02:25 -05:00
contextMenu(mainWindow);
2017-03-01 18:32:21 -06:00
}
app.on('before-quit', function () {
2017-03-01 18:32:21 -06:00
willQuitApp = true;
});
function saveMainWinBounds() {
let newBounds = getWindowSizeAndPosition(mainWindow);
if (newBounds) {
updateConfigField('mainWinPos', newBounds);
}
}
2017-03-06 23:09:10 -06:00
function getMainWindow() {
return mainWindow;
}
function getWindowSizeAndPosition(window) {
if (window) {
let newPos = window.getPosition();
let newSize = window.getSize();
if (newPos && newPos.length === 2 &&
newSize && newSize.length === 2) {
return {
x: newPos[0],
y: newPos[1],
width: newSize[0],
height: newSize[1],
};
}
}
return null;
}
2017-03-01 18:32:21 -06:00
function showMainWindow() {
mainWindow.show();
}
function isMainWindow(win) {
return mainWindow === win;
}
function hasWindow(win, winKey) {
if (win instanceof BrowserWindow) {
let browserWin = windows[winKey];
2017-03-01 18:32:21 -06:00
return browserWin && win === browserWin;
}
return false;
}
function setIsOnline(status) {
isOnline = status;
}
/**
* Tries finding a window we have created with given name. If founds then
* brings to front and gives focus.
* @param {String} windowName Name of target window. Note: main window has
* name 'main'.
*/
2017-04-18 11:02:25 -05:00
function activate(windowName) {
let keys = Object.keys(windows);
for (let i = 0, len = keys.length; i < len; i++) {
let window = windows[keys[i]];
if (window && !window.isDestroyed() && window.winName === windowName) {
if (window.isMinimized()) {
window.restore();
} else {
window.show();
}
2017-04-18 11:02:25 -05:00
return;
}
}
}
/**
* name of renderer window to notify when bounds of child window changes.
* @param {object} window Renderer window to use IPC with to inform about size/
* position change.
*/
function setBoundsChangeWindow(window) {
boundsChangeWindow = window;
}
/**
* Called when bounds of child window changes size/position
* @param {object} window Child window which has changed size/position.
*/
function sendChildWinBoundsChange(window) {
let newBounds = getWindowSizeAndPosition(window);
if (newBounds && boundsChangeWindow) {
newBounds.windowName = window.winName;
// ipc msg back to renderer to inform bounds has changed.
boundsChangeWindow.send('boundsChange', newBounds);
}
}
function openUrlInDefaultBrower(urlToOpen) {
if (urlToOpen) {
electron.shell.openExternal(urlToOpen);
}
}
/**
* Called when an event is received from menu
* @param boolean weather to enable or disable alwaysOnTop.
*/
function isAlwaysOnTop(boolean) {
alwaysOnTop = boolean;
let browserWins = BrowserWindow.getAllWindows();
if (browserWins.length > 0) {
browserWins.forEach(function (browser) {
browser.setAlwaysOnTop(boolean);
});
}
}
// node event emitter to update always on top
eventEmitter.on('isAlwaysOnTop', (boolean) => {
isAlwaysOnTop(boolean);
});
// node event emitter for notification settings
eventEmitter.on('notificationSettings', (notificationSettings) => {
position = notificationSettings.position;
display = notificationSettings.display;
});
2017-03-01 18:32:21 -06:00
module.exports = {
createMainWindow: createMainWindow,
2017-03-06 23:09:10 -06:00
getMainWindow: getMainWindow,
2017-03-01 18:32:21 -06:00
showMainWindow: showMainWindow,
isMainWindow: isMainWindow,
hasWindow: hasWindow,
2017-04-18 11:02:25 -05:00
setIsOnline: setIsOnline,
activate: activate,
setBoundsChangeWindow: setBoundsChangeWindow
2017-03-01 18:32:21 -06:00
};