c94d4efd18
* Script for beta config; unique data dir, in-app env/type display To release a beta build, increment the version and add -beta-N to the end, then go through all the standard release activities. The prepare-build npm script then updates key bits of the package.json to ensure that the beta build can be installed alongside a production build. This includes a new name ('Signal Beta') and a different location for application data. Note: Beta builds can be installed alongside production builds. As part of this, a couple new bits of data are shown across the app: - Environment (development or test, not shown if production) - App Instance (disabled in production; used for multiple accounts) These are shown in: - The window title - both environment and app instance. You can tell beta builds because the app name, preceding these data bits, is different. - The about window - both environment and app instance. You can tell beta builds from the version number. - The header added to the debug log - just environment. The version number will tell us if it's a beta build, and app instance isn't helpful. * Turn on single-window mode in non-production modes Because it's really frightening when you see 'unable to read from db' errors in the console. * aply.sh: More instructions for initial setup and testing * Gruntfile: Get consistent with use of package.json datas * Linux: manually update desktop keys, since macros not available
351 lines
9.2 KiB
JavaScript
351 lines
9.2 KiB
JavaScript
const path = require('path');
|
|
const url = require('url');
|
|
const os = require('os');
|
|
|
|
const _ = require('lodash');
|
|
const electron = require('electron');
|
|
|
|
const BrowserWindow = electron.BrowserWindow;
|
|
const app = electron.app;
|
|
const ipc = electron.ipcMain;
|
|
const Menu = electron.Menu;
|
|
const shell = electron.shell;
|
|
|
|
const packageJson = require('./package.json');
|
|
const autoUpdate = require('./app/auto_update');
|
|
const windowState = require('./app/window_state');
|
|
|
|
|
|
const aumid = 'org.whispersystems.' + packageJson.name;
|
|
console.log('setting AUMID to ' + aumid);
|
|
app.setAppUserModelId(aumid);
|
|
|
|
// 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;
|
|
|
|
const config = require("./app/config");
|
|
|
|
// Very important to put before the single instance check, since it is based on the
|
|
// userData directory.
|
|
const userConfig = require('./app/user_config');
|
|
|
|
if (!process.mas) {
|
|
console.log('making app single instance');
|
|
var shouldQuit = app.makeSingleInstance(function(commandLine, workingDirectory) {
|
|
// Someone tried to run a second instance, we should focus our window
|
|
if (mainWindow) {
|
|
if (mainWindow.isMinimized()) mainWindow.restore();
|
|
mainWindow.focus();
|
|
}
|
|
return true;
|
|
});
|
|
|
|
if (shouldQuit) {
|
|
console.log('quitting; we are the second instance');
|
|
app.quit();
|
|
return;
|
|
}
|
|
}
|
|
|
|
const logging = require('./app/logging');
|
|
|
|
// This must be after we set up appPath in user_config.js, so we know where logs go
|
|
logging.initialize();
|
|
const logger = logging.getLogger();
|
|
|
|
let windowConfig = userConfig.get('window');
|
|
const loadLocale = require('./app/locale').load;
|
|
|
|
let locale;
|
|
|
|
function prepareURL(pathSegments) {
|
|
return url.format({
|
|
pathname: path.join.apply(null, pathSegments),
|
|
protocol: 'file:',
|
|
slashes: true,
|
|
query: {
|
|
name: packageJson.productName,
|
|
locale: locale.name,
|
|
version: app.getVersion(),
|
|
buildExpiration: config.get('buildExpiration'),
|
|
serverUrl: config.get('serverUrl'),
|
|
cdnUrl: config.get('cdnUrl'),
|
|
certificateAuthorities: config.get('certificateAuthorities'),
|
|
environment: config.environment,
|
|
node_version: process.versions.node,
|
|
hostname: os.hostname(),
|
|
appInstance: process.env.NODE_APP_INSTANCE,
|
|
}
|
|
})
|
|
}
|
|
|
|
function handleUrl(event, target) {
|
|
event.preventDefault();
|
|
const protocol = url.parse(target).protocol;
|
|
if (protocol === 'http:' || protocol === 'https:') {
|
|
shell.openExternal(target);
|
|
}
|
|
}
|
|
|
|
function captureClicks(window) {
|
|
window.webContents.on('will-navigate', handleUrl)
|
|
window.webContents.on('new-window', handleUrl);
|
|
}
|
|
|
|
function createWindow () {
|
|
const windowOptions = Object.assign({
|
|
width: 800,
|
|
height: 610,
|
|
minWidth: 700,
|
|
minHeight: 360,
|
|
autoHideMenuBar: false,
|
|
webPreferences: {
|
|
nodeIntegration: false,
|
|
//sandbox: true,
|
|
preload: path.join(__dirname, 'preload.js')
|
|
}
|
|
}, windowConfig);
|
|
|
|
if (windowOptions.fullscreen === false) {
|
|
delete windowOptions.fullscreen;
|
|
}
|
|
|
|
logger.info('Initializing BrowserWindow config: %s', JSON.stringify(windowOptions));
|
|
|
|
// Create the browser window.
|
|
mainWindow = new BrowserWindow(windowOptions);
|
|
|
|
function captureAndSaveWindowStats() {
|
|
const size = mainWindow.getSize();
|
|
const position = mainWindow.getPosition();
|
|
|
|
// so if we need to recreate the window, we have the most recent settings
|
|
windowConfig = {
|
|
maximized: mainWindow.isMaximized(),
|
|
autoHideMenuBar: mainWindow.isMenuBarAutoHide(),
|
|
width: size[0],
|
|
height: size[1],
|
|
x: position[0],
|
|
y: position[1]
|
|
};
|
|
|
|
if (mainWindow.isFullScreen()) {
|
|
// Only include this property if true, because when explicitly set to
|
|
// false the fullscreen button will be disabled on osx
|
|
windowConfig.fullscreen = true;
|
|
}
|
|
|
|
logger.info('Updating BrowserWindow config: %s', JSON.stringify(windowConfig));
|
|
userConfig.set('window', windowConfig);
|
|
}
|
|
|
|
const debouncedCaptureStats = _.debounce(captureAndSaveWindowStats, 500);
|
|
mainWindow.on('resize', debouncedCaptureStats);
|
|
mainWindow.on('move', debouncedCaptureStats);
|
|
mainWindow.on('close', captureAndSaveWindowStats);
|
|
|
|
mainWindow.on('focus', function() {
|
|
mainWindow.flashFrame(false);
|
|
});
|
|
|
|
// Ingested in preload.js via a sendSync call
|
|
ipc.on('locale-data', function(event, arg) {
|
|
event.returnValue = locale.messages;
|
|
});
|
|
|
|
if (config.environment === 'test') {
|
|
mainWindow.loadURL(prepareURL([__dirname, 'test', 'index.html']));
|
|
} else {
|
|
mainWindow.loadURL(prepareURL([__dirname, 'background.html']));
|
|
}
|
|
|
|
if (config.get('openDevTools')) {
|
|
// Open the DevTools.
|
|
mainWindow.webContents.openDevTools()
|
|
}
|
|
|
|
captureClicks(mainWindow);
|
|
|
|
mainWindow.webContents.on('will-navigate', function(e) {
|
|
logger.info('will-navigate');
|
|
e.preventDefault();
|
|
});
|
|
|
|
// Emitted when the window is about to be closed.
|
|
mainWindow.on('close', function (e) {
|
|
if (process.platform === 'darwin' && !windowState.shouldQuit() && config.environment !== 'test') {
|
|
e.preventDefault();
|
|
mainWindow.hide();
|
|
}
|
|
});
|
|
|
|
// 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
|
|
});
|
|
|
|
ipc.on('show-window', function() {
|
|
// Using focus() instead of show() seems to be important on Windows when our window
|
|
// has been docked using Aero Snap/Snap Assist. A full .show() call here will cause
|
|
// the window to reposition:
|
|
// https://github.com/WhisperSystems/Signal-Desktop/issues/1429
|
|
if (mainWindow.isVisible()) {
|
|
mainWindow.focus();
|
|
} else {
|
|
mainWindow.show();
|
|
}
|
|
});
|
|
}
|
|
|
|
function showDebugLog() {
|
|
if (mainWindow) {
|
|
mainWindow.webContents.send('debug-log')
|
|
}
|
|
}
|
|
|
|
function showWindow() {
|
|
if (mainWindow) {
|
|
mainWindow.show();
|
|
}
|
|
};
|
|
|
|
function openReleaseNotes() {
|
|
shell.openExternal('https://github.com/WhisperSystems/Signal-Desktop/releases/tag/v' + app.getVersion());
|
|
}
|
|
|
|
function openNewBugForm() {
|
|
shell.openExternal('https://github.com/WhisperSystems/Signal-Desktop/issues/new');
|
|
}
|
|
|
|
function openSupportPage() {
|
|
shell.openExternal('https://support.signal.org/hc/en-us/categories/202319038-Desktop');
|
|
}
|
|
|
|
function openForums() {
|
|
shell.openExternal('https://whispersystems.discoursehosting.net/');
|
|
}
|
|
|
|
|
|
let aboutWindow;
|
|
function showAbout() {
|
|
if (aboutWindow) {
|
|
aboutWindow.show();
|
|
return;
|
|
}
|
|
|
|
const options = {
|
|
width: 500,
|
|
height: 400,
|
|
resizable: false,
|
|
title: locale.messages.aboutSignalDesktop.message,
|
|
autoHideMenuBar: true,
|
|
backgroundColor: '#2090EA',
|
|
show: false,
|
|
webPreferences: {
|
|
nodeIntegration: false,
|
|
preload: path.join(__dirname, 'preload.js')
|
|
}
|
|
};
|
|
|
|
aboutWindow = new BrowserWindow(options);
|
|
|
|
captureClicks(aboutWindow);
|
|
|
|
aboutWindow.loadURL(prepareURL([__dirname, 'about.html']));
|
|
|
|
aboutWindow.on('closed', function () {
|
|
aboutWindow = null;
|
|
});
|
|
|
|
aboutWindow.once('ready-to-show', function() {
|
|
aboutWindow.show();
|
|
});
|
|
}
|
|
|
|
// 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.
|
|
app.on('ready', function() {
|
|
logger.info('app ready');
|
|
|
|
if (!locale) {
|
|
locale = loadLocale();
|
|
}
|
|
|
|
autoUpdate.initialize(locale.messages);
|
|
|
|
createWindow();
|
|
|
|
const options = {
|
|
showDebugLog,
|
|
showWindow,
|
|
showAbout,
|
|
openReleaseNotes,
|
|
openNewBugForm,
|
|
openSupportPage,
|
|
openForums,
|
|
};
|
|
const createTemplate = require('./app/menu.js');
|
|
const template = createTemplate(options, locale.messages);
|
|
|
|
const menu = Menu.buildFromTemplate(template);
|
|
Menu.setApplicationMenu(menu);
|
|
})
|
|
|
|
app.on('before-quit', function() {
|
|
windowState.markShouldQuit();
|
|
});
|
|
|
|
// 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' || config.environment === 'test') {
|
|
app.quit()
|
|
}
|
|
})
|
|
|
|
app.on('activate', function () {
|
|
// On OS X it's common to re-create a window in the app when the
|
|
// dock icon is clicked and there are no other windows open.
|
|
if (mainWindow) {
|
|
mainWindow.show();
|
|
} else {
|
|
createWindow();
|
|
}
|
|
})
|
|
|
|
// In this file you can include the rest of your app's specific main process
|
|
// code. You can also put them in separate files and require them here.
|
|
|
|
ipc.on('set-badge-count', function(event, count) {
|
|
app.setBadgeCount(count);
|
|
});
|
|
|
|
ipc.on('draw-attention', function(event, count) {
|
|
if (process.platform === 'darwin') {
|
|
app.dock.bounce();
|
|
} else if (process.platform == 'win32') {
|
|
mainWindow.flashFrame(true);
|
|
setTimeout(function() {
|
|
mainWindow.flashFrame(false);
|
|
}, 1000);
|
|
}
|
|
});
|
|
|
|
ipc.on('restart', function(event) {
|
|
app.relaunch();
|
|
app.quit();
|
|
});
|
|
|
|
ipc.on("set-auto-hide-menu-bar", function(event, autoHide) {
|
|
mainWindow.setAutoHideMenuBar(autoHide);
|
|
});
|
|
|
|
ipc.on("set-menu-bar-visibility", function(event, visibility) {
|
|
mainWindow.setMenuBarVisibility(visibility);
|
|
});
|