mattermost-desktop/src/main.js

322 lines
10 KiB
JavaScript
Raw Normal View History

'use strict';
2015-10-23 09:44:10 -07:00
2015-11-20 01:54:44 -08:00
const electron = require('electron');
const app = electron.app; // Module to control application life.
if (require('electron-squirrel-startup')) app.quit();
2015-11-20 01:54:44 -08:00
const BrowserWindow = electron.BrowserWindow; // Module to create native browser window.
const Menu = electron.Menu;
const Tray = electron.Tray;
const ipc = electron.ipcMain;
2016-04-07 07:15:12 -07:00
const nativeImage = electron.nativeImage;
const fs = require('fs');
2016-01-30 07:50:43 -08:00
const path = require('path');
2015-12-23 02:25:14 -08:00
var settings = require('./common/settings');
var certificateStore = require('./main/certificateStore').load(path.resolve(app.getPath('userData'), 'certificate.json'));
2016-01-30 07:50:43 -08:00
var appMenu = require('./main/menus/app');
const allowProtocolDialog = require('./main/allowProtocolDialog');
2015-10-23 09:44:10 -07:00
var argv = require('yargs').argv;
var client = null;
if (argv.livereload) {
client = require('electron-connect').client.create();
2015-12-23 05:04:21 -08:00
client.on('reload', function() {
mainWindow.reload();
});
}
if (argv['config-file']) {
global['config-file'] = argv['config-file'];
}
else {
global['config-file'] = app.getPath('userData') + '/config.json'
}
var config = {};
2015-12-23 02:25:14 -08:00
try {
var configFile = global['config-file'];
config = settings.readFileSync(configFile);
2015-12-23 02:25:14 -08:00
if (config.version != settings.version) {
config = settings.upgrade(config);
settings.writeFileSync(configFile, config);
}
}
catch (e) {
2016-01-30 07:50:43 -08:00
config = settings.loadDefault();
2015-12-23 02:25:14 -08:00
console.log('Failed to read or upgrade config.json');
}
2015-10-23 09:44:10 -07:00
// 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.
var mainWindow = null;
var trayIcon = null;
2016-04-07 07:15:12 -07:00
const trayImages = function() {
switch (process.platform) {
case 'win32':
return {
normal: nativeImage.createFromPath(path.resolve(__dirname, 'resources/tray.png')),
unread: nativeImage.createFromPath(path.resolve(__dirname, 'resources/tray_unread.png')),
mention: nativeImage.createFromPath(path.resolve(__dirname, 'resources/tray_mention.png'))
};
case 'darwin':
return {
normal: nativeImage.createFromPath(path.resolve(__dirname, 'resources/osx/MenuIconTemplate.png')),
unread: nativeImage.createFromPath(path.resolve(__dirname, 'resources/osx/MenuIconUnreadTemplate.png')),
mention: nativeImage.createFromPath(path.resolve(__dirname, 'resources/osx/MenuIconMentionTemplate.png'))
};
case 'linux':
var resourcesDir = 'resources/linux/' + (config.trayIconTheme || 'light') + '/';
return {
normal: nativeImage.createFromPath(path.resolve(__dirname, resourcesDir + 'MenuIconTemplate.png')),
unread: nativeImage.createFromPath(path.resolve(__dirname, resourcesDir + 'MenuIconUnreadTemplate.png')),
mention: nativeImage.createFromPath(path.resolve(__dirname, resourcesDir + 'MenuIconMentionTemplate.png'))
};
2016-04-07 07:15:12 -07:00
default:
return {};
}
}();
2015-10-23 09:44:10 -07:00
var willAppQuit = false;
2016-04-11 05:51:24 -07:00
function shouldShowTrayIcon() {
if (process.platform === 'win32') {
return true;
}
if (['darwin', 'linux'].includes(process.platform) && config.showTrayIcon === true) {
2016-04-11 05:51:24 -07:00
return true;
}
return false;
}
2015-10-23 09:44:10 -07:00
// Quit when all windows are closed.
app.on('window-all-closed', function() {
// On OS X it is common for applications and their menu bar
// to stay active until the user quits explicitly with Cmd + Q
if (process.platform !== 'darwin') {
app.quit();
}
});
// For win32, auto-hide menu bar.
app.on('browser-window-created', function(event, window) {
if (process.platform === 'win32' || process.platform === 'linux') {
if (config.hideMenuBar) {
window.setAutoHideMenuBar(true);
window.setMenuBarVisibility(false);
}
2015-10-23 09:44:10 -07:00
}
});
// For OSX, show hidden mainWindow when clicking dock icon.
app.on('activate', function(event) {
mainWindow.show();
});
app.on('before-quit', function() {
willAppQuit = true;
});
app.on('certificate-error', function(event, webContents, url, error, certificate, callback) {
if (certificateStore.isTrusted(url, certificate)) {
event.preventDefault();
callback(true);
}
else {
var detail = `URL: ${url}\nError: ${error}`;
if (certificateStore.isExisting(url)) {
detail = `Certificate is different from previous one.\n\n` + detail;
}
electron.dialog.showMessageBox(mainWindow, {
title: 'Certificate error',
message: `Do you trust certificate from "${certificate.issuerName}"?`,
detail: detail,
type: 'warning',
buttons: [
'Yes',
'No'
],
cancelId: 1
}, function(response) {
if (response === 0) {
certificateStore.add(url, certificate);
certificateStore.save();
webContents.loadURL(url);
}
});
callback(false);
}
});
2016-04-21 06:43:18 -07:00
const loginCallbackMap = new Map();
ipc.on('login-credentials', function(event, request, user, password) {
const callback = loginCallbackMap.get(JSON.stringify(request));
if (callback != null) {
callback(user, password);
}
})
app.on('login', function(event, webContents, request, authInfo, callback) {
event.preventDefault();
loginCallbackMap.set(JSON.stringify(request), callback);
mainWindow.webContents.send('login-request', request, authInfo);
});
allowProtocolDialog.init(mainWindow);
2016-05-21 21:39:52 -07:00
2015-10-23 09:44:10 -07:00
// This method will be called when Electron has finished
// initialization and is ready to create browser windows.
app.on('ready', function() {
2016-04-11 05:51:24 -07:00
if (shouldShowTrayIcon()) {
2016-04-06 08:49:20 -07:00
// set up tray icon
2016-04-07 07:15:12 -07:00
trayIcon = new Tray(trayImages.normal);
trayIcon.setToolTip(app.getName());
2015-11-20 01:04:20 -08:00
trayIcon.on('click', function() {
mainWindow.focus();
});
trayIcon.on('right-click', () => {
trayIcon.popUpContextMenu();
});
2015-11-20 01:04:20 -08:00
trayIcon.on('balloon-click', function() {
mainWindow.focus();
});
ipc.on('notified', function(event, arg) {
trayIcon.displayBalloon({
2016-01-30 07:50:43 -08:00
icon: path.resolve(__dirname, 'resources/appicon.png'),
title: arg.title,
content: arg.options.body
});
});
// Set overlay icon from dataURL
// Set trayicon to show "dot"
2016-04-06 08:49:20 -07:00
ipc.on('update-unread', function(event, arg) {
if (process.platform === 'win32') {
const overlay = arg.overlayDataURL ? electron.nativeImage.createFromDataURL(arg.overlayDataURL) : null;
mainWindow.setOverlayIcon(overlay, arg.description);
}
2016-04-07 07:15:12 -07:00
if (arg.mentionCount > 0) {
trayIcon.setImage(trayImages.mention);
2016-06-03 05:45:02 -07:00
trayIcon.setToolTip(arg.mentionCount + ' unread mentions');
2016-04-07 07:15:12 -07:00
}
else if (arg.unreadCount > 0) {
trayIcon.setImage(trayImages.unread);
2016-06-03 05:45:02 -07:00
trayIcon.setToolTip(arg.unreadCount + ' unread channels');
2016-04-07 07:15:12 -07:00
}
else {
trayIcon.setImage(trayImages.normal);
2016-06-03 05:45:02 -07:00
trayIcon.setToolTip(app.getName());
}
});
}
2015-10-23 09:44:10 -07:00
// Create the browser window.
var bounds_info_path = path.resolve(app.getPath("userData"), "bounds-info.json");
var window_options;
try {
window_options = JSON.parse(fs.readFileSync(bounds_info_path, 'utf-8'));
}
catch (e) {
// follow Electron's defaults
window_options = {};
}
if (process.platform === 'win32' || process.platform === 'linux') {
// On HiDPI Windows environment, the taskbar icon is pixelated. So this line is necessary.
window_options.icon = path.resolve(__dirname, 'resources/appicon.png');
}
window_options.title = app.getName();
mainWindow = new BrowserWindow(window_options);
mainWindow.setFullScreenable(true); // fullscreenable option has no effect.
if (window_options.maximized) {
mainWindow.maximize();
}
if (window_options.fullscreen) {
mainWindow.setFullScreen(true);
}
2015-10-23 09:44:10 -07:00
// and load the index.html of the app.
2015-12-22 06:35:22 -08:00
mainWindow.loadURL('file://' + __dirname + '/browser/index.html');
2015-10-23 09:44:10 -07:00
2016-05-21 00:13:22 -07:00
// Set application menu
ipc.on('update-menu', (event, config) => {
var app_menu = appMenu.createMenu(mainWindow, config);
Menu.setApplicationMenu(app_menu);
});
ipc.emit('update-menu', true, config);
// set up context menu for tray icon
if (shouldShowTrayIcon()) {
const tray_menu = require('./main/menus/tray').createDefault(mainWindow);
trayIcon.setContextMenu(tray_menu);
}
2015-10-23 09:44:10 -07:00
// Open the DevTools.
// mainWindow.openDevTools();
var saveWindowState = function(file, window) {
var window_state = window.getBounds();
window_state.maximized = window.isMaximized();
window_state.fullscreen = window.isFullScreen();
2016-05-03 08:36:54 -07:00
try {
fs.writeFileSync(bounds_info_path, JSON.stringify(window_state));
}
catch (e) {
// [Linux] error happens only when the window state is changed before the config dir is creatied.
}
};
2015-10-23 09:44:10 -07:00
mainWindow.on('close', function(event) {
if (willAppQuit) { // when [Ctrl|Cmd]+Q
saveWindowState(bounds_info_path, mainWindow);
}
else { // Minimize or hide the window for close button.
2015-10-23 09:44:10 -07:00
event.preventDefault();
switch (process.platform) {
case 'win32':
case 'linux':
2015-10-23 09:44:10 -07:00
mainWindow.minimize();
break;
case 'darwin':
mainWindow.hide();
break;
default:
}
}
});
// App should save bounds when a window is closed.
// However, 'close' is not fired in some situations(shutdown, ctrl+c)
// because main process is killed in such situations.
// 'blur' event was effective in order to avoid this.
// Ideally, app should detect that OS is shutting down.
mainWindow.on('blur', function() {
saveWindowState(bounds_info_path, mainWindow);
});
2015-10-23 09:44:10 -07:00
// Emitted when the window is closed.
mainWindow.on('closed', function() {
// Dereference the window object, usually you would store windows
// in an array if your app supports multi windows, this is the time
// when you should delete the corresponding element.
mainWindow = null;
});
// Deny drag&drop navigation in mainWindow.
// Drag&drop is allowed in webview of index.html.
mainWindow.webContents.on('will-navigate', function(event, url) {
var dirname = __dirname;
if (process.platform === 'win32') {
dirname = '/' + dirname.replace(/\\/g, '/');
}
var index = url.indexOf('file://' + dirname);
if (index !== 0) {
event.preventDefault();
}
});
});