const path = require('path'); const url = require('url'); const os = require('os'); const fs = require('fs'); const _ = require('lodash'); const pify = require('pify'); const electron = require('electron'); const getRealPath = pify(fs.realpath); const { app, BrowserWindow, ipcMain: ipc, Menu, protocol: electronProtocol, session, shell, } = electron; const packageJson = require('./package.json'); const Attachments = require('./app/attachments'); const autoUpdate = require('./app/auto_update'); const createTrayIcon = require('./app/tray_icon'); const GlobalErrors = require('./js/modules/global_errors'); const logging = require('./app/logging'); const windowState = require('./app/window_state'); const { createTemplate } = require('./app/menu'); const { installFileHandler, installWebHandler, } = require('./app/protocol_filter'); const { installPermissionsHandler } = require('./app/permissions'); GlobalErrors.addHandler(); const appUserModelId = `org.whispersystems.${packageJson.name}`; console.log('Set Windows Application User Model ID (AUMID)', { appUserModelId, }); app.setAppUserModelId(appUserModelId); // 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; function getMainWindow() { return mainWindow; } // Tray icon and related objects let tray = null; const startInTray = process.argv.some(arg => arg === '--start-in-tray'); const usingTrayIcon = startInTray || process.argv.some(arg => arg === '--use-tray-icon'); const config = require('./app/config'); const importMode = process.argv.some(arg => arg === '--import') || config.get('import'); const development = config.environment === 'development'; // Very important to put before the single instance check, since it is based on the // userData directory. const userConfig = require('./app/user_config'); function showWindow() { if (!mainWindow) { return; } // 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/signalapp/Signal-Desktop/issues/1429 if (mainWindow.isVisible()) { mainWindow.focus(); } else { mainWindow.show(); } // toggle the visibility of the show/hide tray icon menu entries if (tray) { tray.updateContextMenu(); } } if (!process.mas) { console.log('making app single instance'); const shouldQuit = app.makeSingleInstance(() => { // Someone tried to run a second instance, we should focus our window if (mainWindow) { if (mainWindow.isMinimized()) { mainWindow.restore(); } showWindow(); } return true; }); if (shouldQuit) { console.log('quitting; we are the second instance'); app.exit(); } } let windowConfig = userConfig.get('window'); const loadLocale = require('./app/locale').load; // Both of these will be set after app fires the 'ready' event let logger; let locale; function prepareURL(pathSegments, moreKeys) { 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'), certificateAuthority: config.get('certificateAuthority'), environment: config.environment, node_version: process.versions.node, hostname: os.hostname(), appInstance: process.env.NODE_APP_INSTANCE, proxyUrl: process.env.HTTPS_PROXY || process.env.https_proxy, importMode: importMode ? true : undefined, // for stringify() ...moreKeys, }, }); } function handleUrl(event, target) { event.preventDefault(); const { protocol } = url.parse(target); if (protocol === 'http:' || protocol === 'https:') { shell.openExternal(target); } } function captureClicks(window) { window.webContents.on('will-navigate', handleUrl); window.webContents.on('new-window', handleUrl); } const DEFAULT_WIDTH = 800; const DEFAULT_HEIGHT = 610; const MIN_WIDTH = 640; const MIN_HEIGHT = 360; const BOUNDS_BUFFER = 100; function isVisible(window, bounds) { const boundsX = _.get(bounds, 'x') || 0; const boundsY = _.get(bounds, 'y') || 0; const boundsWidth = _.get(bounds, 'width') || DEFAULT_WIDTH; const boundsHeight = _.get(bounds, 'height') || DEFAULT_HEIGHT; // requiring BOUNDS_BUFFER pixels on the left or right side const rightSideClearOfLeftBound = window.x + window.width >= boundsX + BOUNDS_BUFFER; const leftSideClearOfRightBound = window.x <= boundsX + boundsWidth - BOUNDS_BUFFER; // top can't be offscreen, and must show at least BOUNDS_BUFFER pixels at bottom const topClearOfUpperBound = window.y >= boundsY; const topClearOfLowerBound = window.y <= boundsY + boundsHeight - BOUNDS_BUFFER; return ( rightSideClearOfLeftBound && leftSideClearOfRightBound && topClearOfUpperBound && topClearOfLowerBound ); } function createWindow() { const { screen } = electron; const windowOptions = Object.assign( { show: !startInTray, // allow to start minimised in tray width: DEFAULT_WIDTH, height: DEFAULT_HEIGHT, minWidth: MIN_WIDTH, minHeight: MIN_HEIGHT, autoHideMenuBar: false, webPreferences: { nodeIntegration: false, nodeIntegrationInWorker: false, // sandbox: true, preload: path.join(__dirname, 'preload.js'), nativeWindowOpen: true, }, icon: path.join(__dirname, 'images', 'icon_256.png'), }, _.pick(windowConfig, [ 'maximized', 'autoHideMenuBar', 'width', 'height', 'x', 'y', ]) ); if (!_.isNumber(windowOptions.width) || windowOptions.width < MIN_WIDTH) { windowOptions.width = DEFAULT_WIDTH; } if (!_.isNumber(windowOptions.height) || windowOptions.height < MIN_HEIGHT) { windowOptions.height = DEFAULT_HEIGHT; } if (!_.isBoolean(windowOptions.maximized)) { delete windowOptions.maximized; } if (!_.isBoolean(windowOptions.autoHideMenuBar)) { delete windowOptions.autoHideMenuBar; } const visibleOnAnyScreen = _.some(screen.getAllDisplays(), display => { if (!_.isNumber(windowOptions.x) || !_.isNumber(windowOptions.y)) { return false; } return isVisible(windowOptions, _.get(display, 'bounds')); }); if (!visibleOnAnyScreen) { console.log('Location reset needed'); delete windowOptions.x; delete windowOptions.y; } 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() { if (!mainWindow) { return; } 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', () => { mainWindow.flashFrame(false); }); // Ingested in preload.js via a sendSync call ipc.on('locale-data', event => { // eslint-disable-next-line no-param-reassign event.returnValue = locale.messages; }); if (config.environment === 'test') { mainWindow.loadURL(prepareURL([__dirname, 'test', 'index.html'])); } else if (config.environment === 'test-lib') { mainWindow.loadURL( prepareURL([__dirname, 'libtextsecure', 'test', 'index.html']) ); } else { mainWindow.loadURL(prepareURL([__dirname, 'background.html'])); } if (config.get('openDevTools')) { // Open the DevTools. mainWindow.webContents.openDevTools(); } captureClicks(mainWindow); // Emitted when the window is about to be closed. mainWindow.on('close', e => { // If the application is terminating, just do the default if ( windowState.shouldQuit() || config.environment === 'test' || config.environment === 'test-lib' ) { return; } // On Mac, or on other platforms when the tray icon is in use, the window // should be only hidden, not closed, when the user clicks the close button if (usingTrayIcon || process.platform === 'darwin') { e.preventDefault(); mainWindow.hide(); // toggle the visibility of the show/hide tray icon menu entries if (tray) { tray.updateContextMenu(); } } }); // Emitted when the window is closed. mainWindow.on('closed', () => { // 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', () => { showWindow(); }); } function openReleaseNotes() { shell.openExternal( `https://github.com/signalapp/Signal-Desktop/releases/tag/v${app.getVersion()}` ); } function openNewBugForm() { shell.openExternal('https://github.com/signalapp/Signal-Desktop/issues/new'); } function openSupportPage() { shell.openExternal( 'https://support.signal.org/hc/en-us/categories/202319038-Desktop' ); } function openForums() { shell.openExternal('https://community.signalusers.org/'); } function setupWithImport() { if (mainWindow) { mainWindow.webContents.send('set-up-with-import'); } } function setupAsNewDevice() { if (mainWindow) { mainWindow.webContents.send('set-up-as-new-device'); } } function setupAsStandalone() { if (mainWindow) { mainWindow.webContents.send('set-up-as-standalone'); } } 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, nodeIntegrationInWorker: false, preload: path.join(__dirname, 'about_preload.js'), // sandbox: true, nativeWindowOpen: true, }, parent: mainWindow, }; aboutWindow = new BrowserWindow(options); captureClicks(aboutWindow); aboutWindow.loadURL(prepareURL([__dirname, 'about.html'])); aboutWindow.on('closed', () => { aboutWindow = null; }); aboutWindow.once('ready-to-show', () => { aboutWindow.show(); }); } let settingsWindow; async function showSettingsWindow() { if (settingsWindow) { settingsWindow.show(); return; } if (!mainWindow) { return; } const theme = await pify(getDataFromMainWindow)('theme-setting'); const size = mainWindow.getSize(); const options = { width: Math.min(500, size[0]), height: Math.max(size[1] - 100, MIN_HEIGHT), resizable: false, title: locale.messages.signalDesktopPreferences.message, autoHideMenuBar: true, backgroundColor: '#FFFFFF', show: false, modal: true, webPreferences: { nodeIntegration: false, nodeIntegrationInWorker: false, preload: path.join(__dirname, 'settings_preload.js'), // sandbox: true, nativeWindowOpen: true, }, parent: mainWindow, }; settingsWindow = new BrowserWindow(options); captureClicks(settingsWindow); settingsWindow.loadURL(prepareURL([__dirname, 'settings.html'], { theme })); settingsWindow.on('closed', () => { removeDarkOverlay(); settingsWindow = null; }); settingsWindow.once('ready-to-show', () => { addDarkOverlay(); settingsWindow.show(); }); } let debugLogWindow; async function showDebugLogWindow() { if (debugLogWindow) { debugLogWindow.show(); return; } const theme = await pify(getDataFromMainWindow)('theme-setting'); const size = mainWindow.getSize(); const options = { width: Math.max(size[0] - 100, MIN_WIDTH), height: Math.max(size[1] - 100, MIN_HEIGHT), resizable: false, title: locale.messages.signalDesktopPreferences.message, autoHideMenuBar: true, backgroundColor: '#FFFFFF', show: false, modal: true, webPreferences: { nodeIntegration: false, nodeIntegrationInWorker: false, preload: path.join(__dirname, 'debug_log_preload.js'), // sandbox: true, nativeWindowOpen: true, }, parent: mainWindow, }; debugLogWindow = new BrowserWindow(options); captureClicks(debugLogWindow); debugLogWindow.loadURL(prepareURL([__dirname, 'debug_log.html'], { theme })); debugLogWindow.on('closed', () => { removeDarkOverlay(); debugLogWindow = null; }); debugLogWindow.once('ready-to-show', () => { addDarkOverlay(); debugLogWindow.show(); }); } let permissionsPopupWindow; async function showPermissionsPopupWindow() { if (permissionsPopupWindow) { permissionsPopupWindow.show(); return; } if (!mainWindow) { return; } const theme = await pify(getDataFromMainWindow)('theme-setting'); const size = mainWindow.getSize(); const options = { width: Math.min(400, size[0]), height: Math.min(150, size[1]), resizable: false, title: locale.messages.signalDesktopPreferences.message, autoHideMenuBar: true, backgroundColor: '#FFFFFF', show: false, modal: true, webPreferences: { nodeIntegration: false, nodeIntegrationInWorker: false, preload: path.join(__dirname, 'permissions_popup_preload.js'), // sandbox: true, nativeWindowOpen: true, }, parent: mainWindow, }; permissionsPopupWindow = new BrowserWindow(options); captureClicks(permissionsPopupWindow); permissionsPopupWindow.loadURL( prepareURL([__dirname, 'permissions_popup.html'], { theme }) ); permissionsPopupWindow.on('closed', () => { removeDarkOverlay(); permissionsPopupWindow = null; }); permissionsPopupWindow.once('ready-to-show', () => { addDarkOverlay(); permissionsPopupWindow.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. let ready = false; app.on('ready', async () => { const userDataPath = await getRealPath(app.getPath('userData')); const installPath = await getRealPath(app.getAppPath()); if (process.env.NODE_ENV !== 'test' && process.env.NODE_ENV !== 'test-lib') { installFileHandler({ protocol: electronProtocol, userDataPath, installPath, isWindows: process.platform === 'win32', }); } installWebHandler({ protocol: electronProtocol, }); installPermissionsHandler({ session, userConfig }); // NOTE: Temporarily allow `then` until we convert the entire file to `async` / `await`: /* eslint-disable more/no-then */ let loggingSetupError; logging .initialize() .catch(error => { loggingSetupError = error; }) .then(async () => { /* eslint-enable more/no-then */ logger = logging.getLogger(); logger.info('app ready'); if (loggingSetupError) { logger.error('Problem setting up logging', loggingSetupError.stack); } if (!locale) { const appLocale = process.env.NODE_ENV === 'test' ? 'en' : app.getLocale(); locale = loadLocale({ appLocale, logger }); } console.log('Ensure attachments directory exists'); await Attachments.ensureDirectory(userDataPath); ready = true; autoUpdate.initialize(getMainWindow, locale.messages); createWindow(); if (usingTrayIcon) { tray = createTrayIcon(getMainWindow, locale.messages); } setupMenu(); }); }); function setupMenu(options) { const { platform } = process; const menuOptions = Object.assign({}, options, { development, showDebugLog: showDebugLogWindow, showWindow, showAbout, showSettings: showSettingsWindow, openReleaseNotes, openNewBugForm, openSupportPage, openForums, platform, setupWithImport, setupAsNewDevice, setupAsStandalone, }); const template = createTemplate(menuOptions, locale.messages); const menu = Menu.buildFromTemplate(template); Menu.setApplicationMenu(menu); } app.on('before-quit', () => { windowState.markShouldQuit(); }); // Quit when all windows are closed. app.on('window-all-closed', () => { // 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' || config.environment === 'test-lib' ) { app.quit(); } }); app.on('activate', () => { if (!ready) { return; } // 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(); } }); // Defense in depth. We never intend to open webviews or windows. Prevent it completely. app.on('web-contents-created', (createEvent, contents) => { contents.on('will-attach-webview', attachEvent => { attachEvent.preventDefault(); }); contents.on('new-window', newEvent => { newEvent.preventDefault(); }); }); ipc.on('set-badge-count', (event, count) => { app.setBadgeCount(count); }); ipc.on('remove-setup-menu-items', () => { setupMenu(); }); ipc.on('add-setup-menu-items', () => { setupMenu({ includeSetup: true, }); }); ipc.on('draw-attention', () => { if (process.platform === 'darwin') { app.dock.bounce(); } else if (process.platform === 'win32') { mainWindow.flashFrame(true); } else if (process.platform === 'linux') { mainWindow.flashFrame(true); } }); ipc.on('set-media-permissions', (event, enabled) => { userConfig.set('mediaPermissions', enabled); }); ipc.on('get-media-permissions', event => { // eslint-disable-next-line no-param-reassign event.returnValue = userConfig.get('mediaPermissions') || false; }); ipc.on('restart', () => { app.relaunch(); app.quit(); }); ipc.on('set-auto-hide-menu-bar', (event, autoHide) => { if (mainWindow) { mainWindow.setAutoHideMenuBar(autoHide); } }); ipc.on('set-menu-bar-visibility', (event, visibility) => { if (mainWindow) { mainWindow.setMenuBarVisibility(visibility); } }); ipc.on('close-about', () => { if (aboutWindow) { aboutWindow.close(); } }); ipc.on('update-tray-icon', (event, unreadCount) => { if (tray) { tray.updateIcon(unreadCount); } }); // Debug Log-related IPC calls ipc.on('show-debug-log', showDebugLogWindow); ipc.on('close-debug-log', () => { if (debugLogWindow) { debugLogWindow.close(); } }); // Permissions Popup-related IPC calls ipc.on('show-permissions-popup', showPermissionsPopupWindow); ipc.on('close-permissions-popup', () => { if (permissionsPopupWindow) { permissionsPopupWindow.close(); } }); // Settings-related IPC calls function addDarkOverlay() { if (mainWindow && mainWindow.webContents) { mainWindow.webContents.send('add-dark-overlay'); } } function removeDarkOverlay() { if (mainWindow && mainWindow.webContents) { mainWindow.webContents.send('remove-dark-overlay'); } } ipc.on('show-settings', showSettingsWindow); ipc.on('close-settings', () => { if (settingsWindow) { settingsWindow.close(); } }); installSettingsGetter('device-name'); installSettingsGetter('theme-setting'); installSettingsSetter('theme-setting'); installSettingsGetter('hide-menu-bar'); installSettingsSetter('hide-menu-bar'); installSettingsGetter('notification-setting'); installSettingsSetter('notification-setting'); installSettingsGetter('audio-notification'); installSettingsSetter('audio-notification'); // This one is different because its single source of truth is userConfig, not IndexedDB ipc.on('get-media-permissions', event => { event.sender.send( 'get-success-media-permissions', null, userConfig.get('mediaPermissions') || false ); }); ipc.on('set-media-permissions', (event, value) => { userConfig.set('mediaPermissions', value); // We reinstall permissions handler to ensure that a revoked permission takes effect installPermissionsHandler({ session, userConfig }); event.sender.send('set-success-media-permissions', null); }); installSettingsGetter('is-primary'); installSettingsGetter('sync-request'); installSettingsGetter('sync-time'); installSettingsSetter('sync-time'); ipc.on('delete-all-data', () => { if (mainWindow && mainWindow.webContents) { mainWindow.webContents.send('delete-all-data'); } }); function getDataFromMainWindow(name, callback) { ipc.once(`get-success-${name}`, (_event, error, value) => callback(error, value) ); mainWindow.webContents.send(`get-${name}`); } function installSettingsGetter(name) { ipc.on(`get-${name}`, event => { if (mainWindow && mainWindow.webContents) { getDataFromMainWindow(name, (error, value) => event.sender.send(`get-success-${name}`, error, value) ); } }); } function installSettingsSetter(name) { ipc.on(`set-${name}`, (event, value) => { if (mainWindow && mainWindow.webContents) { ipc.once(`set-success-${name}`, (_event, error) => event.sender.send(`set-success-${name}`, error) ); mainWindow.webContents.send(`set-${name}`, value); } }); }