You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			1144 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			TypeScript
		
	
			
		
		
	
	
			1144 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			TypeScript
		
	
/* eslint-disable @typescript-eslint/no-misused-promises */
 | 
						|
/* eslint-disable no-void */
 | 
						|
/* eslint-disable import/first */
 | 
						|
/* eslint-disable import/order */
 | 
						|
/* eslint-disable no-console */
 | 
						|
 | 
						|
import {
 | 
						|
  app,
 | 
						|
  BrowserWindow,
 | 
						|
  dialog,
 | 
						|
  ipcMain as ipc,
 | 
						|
  Menu,
 | 
						|
  protocol as electronProtocol,
 | 
						|
  screen,
 | 
						|
  shell,
 | 
						|
  systemPreferences,
 | 
						|
} from 'electron';
 | 
						|
 | 
						|
import path, { join } from 'path';
 | 
						|
import { platform as osPlatform } from 'process';
 | 
						|
import url from 'url';
 | 
						|
import os from 'os';
 | 
						|
import fs from 'fs';
 | 
						|
import crypto from 'crypto';
 | 
						|
 | 
						|
import _, { isEmpty } from 'lodash';
 | 
						|
import pify from 'pify';
 | 
						|
import Logger from 'bunyan';
 | 
						|
 | 
						|
import { setup as setupSpellChecker } from '../node/spell_check'; // checked - only node
 | 
						|
import { setupGlobalErrorHandler } from '../node/global_errors'; // checked - only node
 | 
						|
 | 
						|
import packageJson from '../../package.json'; // checked - only node
 | 
						|
 | 
						|
setupGlobalErrorHandler();
 | 
						|
import electronLocalshortcut from 'electron-localshortcut';
 | 
						|
 | 
						|
const getRealPath = pify(fs.realpath);
 | 
						|
 | 
						|
// Hardcoding appId to prevent build failures on release.
 | 
						|
// const appUserModelId = packageJson.build.appId;
 | 
						|
const appUserModelId = 'com.loki-project.messenger-desktop';
 | 
						|
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: BrowserWindow | null = null;
 | 
						|
 | 
						|
function getMainWindow() {
 | 
						|
  return mainWindow;
 | 
						|
}
 | 
						|
 | 
						|
let readyForShutdown: boolean = false;
 | 
						|
 | 
						|
// Tray icon and related objects
 | 
						|
let tray: any = null;
 | 
						|
 | 
						|
import { config } from '../node/config'; // checked - only node
 | 
						|
 | 
						|
// Very important to put before the single instance check, since it is based on the
 | 
						|
//   userData directory.
 | 
						|
import { userConfig } from '../node/config/user_config'; // checked - only node
 | 
						|
import * as PasswordUtil from '../util/passwordUtils'; // checked - only node
 | 
						|
 | 
						|
const development = (config as any).environment === 'development';
 | 
						|
const appInstance = config.util.getEnv('NODE_APP_INSTANCE') || 0;
 | 
						|
 | 
						|
// We generally want to pull in our own modules after this point, after the user
 | 
						|
//   data directory has been set.
 | 
						|
import { initAttachmentsChannel } from '../node/attachment_channel';
 | 
						|
 | 
						|
import * as updater from '../updater/index'; // checked - only node
 | 
						|
 | 
						|
import { createTrayIcon } from '../node/tray_icon'; // checked - only node
 | 
						|
import { ephemeralConfig } from '../node/config/ephemeral_config'; // checked - only node
 | 
						|
import { getLogger, initializeLogger } from '../node/logging'; // checked - only node
 | 
						|
import { sqlNode } from '../node/sql'; // checked - only node
 | 
						|
import * as sqlChannels from '../node/sql_channel'; // checked - only node
 | 
						|
import { windowMarkShouldQuit, windowShouldQuit } from '../node/window_state'; // checked - only node
 | 
						|
import { createTemplate } from '../node/menu'; // checked - only node
 | 
						|
import { installFileHandler, installWebHandler } from '../node/protocol_filter'; // checked - only node
 | 
						|
import { installPermissionsHandler } from '../node/permissions'; // checked - only node
 | 
						|
 | 
						|
let appStartInitialSpellcheckSetting = true;
 | 
						|
 | 
						|
const isTestIntegration = Boolean(
 | 
						|
  process.env.NODE_APP_INSTANCE && process.env.NODE_APP_INSTANCE.includes('test-integration')
 | 
						|
);
 | 
						|
const openDevToolsTestIntegration = isTestIntegration && !isEmpty(process.env.TEST_OPEN_DEV_TOOLS);
 | 
						|
 | 
						|
async function getSpellCheckSetting() {
 | 
						|
  const json = sqlNode.getItemById('spell-check');
 | 
						|
  // Default to `true` if setting doesn't exist yet
 | 
						|
  if (!json) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  return json.value;
 | 
						|
}
 | 
						|
 | 
						|
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 gotLock = app.requestSingleInstanceLock();
 | 
						|
  if (!gotLock) {
 | 
						|
    // Don't allow second instance if we are in prod
 | 
						|
    if (appInstance === 0) {
 | 
						|
      console.log('quitting; we are the second instance');
 | 
						|
      app.exit();
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    app.on('second-instance', () => {
 | 
						|
      // Someone tried to run a second instance, we should focus our window
 | 
						|
      if (mainWindow) {
 | 
						|
        if (mainWindow.isMinimized()) {
 | 
						|
          mainWindow.restore();
 | 
						|
        }
 | 
						|
 | 
						|
        showWindow();
 | 
						|
      }
 | 
						|
      return true;
 | 
						|
    });
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
const windowFromUserConfig = userConfig.get('window');
 | 
						|
const windowFromEphemeral = ephemeralConfig.get('window');
 | 
						|
let windowConfig = windowFromEphemeral || windowFromUserConfig;
 | 
						|
if (windowFromUserConfig) {
 | 
						|
  userConfig.set('window', null);
 | 
						|
  ephemeralConfig.set('window', windowConfig);
 | 
						|
}
 | 
						|
 | 
						|
// import {load as loadLocale} from '../..'
 | 
						|
import { load as loadLocale, LocaleMessagesWithNameType } from '../node/locale';
 | 
						|
import { setLastestRelease } from '../node/latest_desktop_release';
 | 
						|
import { getAppRootPath } from '../node/getRootPath';
 | 
						|
import { classicDark } from '../themes';
 | 
						|
 | 
						|
// Both of these will be set after app fires the 'ready' event
 | 
						|
let logger: Logger | null = null;
 | 
						|
let locale: LocaleMessagesWithNameType;
 | 
						|
 | 
						|
function assertLogger(): Logger {
 | 
						|
  if (!logger) {
 | 
						|
    throw new Error('assertLogger: logger is not set');
 | 
						|
  }
 | 
						|
  return logger;
 | 
						|
}
 | 
						|
 | 
						|
function prepareURL(pathSegments: Array<string>, moreKeys?: { theme: any }) {
 | 
						|
  const urlObject: url.UrlObject = {
 | 
						|
    pathname: join(...pathSegments),
 | 
						|
    protocol: 'file:',
 | 
						|
    slashes: true,
 | 
						|
    query: {
 | 
						|
      name: packageJson.productName,
 | 
						|
      locale: locale.name,
 | 
						|
      version: app.getVersion(),
 | 
						|
      commitHash: config.get('commitHash'),
 | 
						|
      environment: (config as any).environment,
 | 
						|
      node_version: process.versions.node,
 | 
						|
      hostname: os.hostname(),
 | 
						|
      appInstance: process.env.NODE_APP_INSTANCE,
 | 
						|
      proxyUrl: process.env.HTTPS_PROXY || process.env.https_proxy,
 | 
						|
      appStartInitialSpellcheckSetting,
 | 
						|
      ...moreKeys,
 | 
						|
    },
 | 
						|
  };
 | 
						|
  return url.format(urlObject);
 | 
						|
}
 | 
						|
 | 
						|
function handleUrl(event: any, target: string) {
 | 
						|
  event?.preventDefault();
 | 
						|
  const { protocol } = url.parse(target);
 | 
						|
 | 
						|
  if (protocol === 'http:' || protocol === 'https:') {
 | 
						|
    void shell.openExternal(target);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
function captureClicks(window: BrowserWindow) {
 | 
						|
  window.webContents.on('will-navigate', handleUrl);
 | 
						|
 | 
						|
  window.webContents.setWindowOpenHandler(({ url: urlToOpen }) => {
 | 
						|
    handleUrl(undefined, urlToOpen);
 | 
						|
    return { action: 'deny' };
 | 
						|
  });
 | 
						|
}
 | 
						|
 | 
						|
function getDefaultWindowSize() {
 | 
						|
  return {
 | 
						|
    defaultWidth: 880,
 | 
						|
    defaultHeight: openDevToolsTestIntegration ? 1000 : 820, // the dev tools open at the bottom hide some stuff which should be visible
 | 
						|
    minWidth: 880,
 | 
						|
    minHeight: 600,
 | 
						|
  };
 | 
						|
}
 | 
						|
 | 
						|
function getWindowSize() {
 | 
						|
  const screenSize = screen.getPrimaryDisplay().workAreaSize;
 | 
						|
  const { minWidth, minHeight, defaultWidth, defaultHeight } = getDefaultWindowSize();
 | 
						|
  // Ensure that the screen can fit within the default size
 | 
						|
  const width = Math.min(defaultWidth, Math.max(minWidth, screenSize.width));
 | 
						|
  const height = Math.min(defaultHeight, Math.max(minHeight, screenSize.height));
 | 
						|
 | 
						|
  return { width, height, minWidth, minHeight };
 | 
						|
}
 | 
						|
 | 
						|
function isVisible(window: { x: number; y: number; width: number }, bounds: any) {
 | 
						|
  const boundsX = _.get(bounds, 'x') || 0;
 | 
						|
  const boundsY = _.get(bounds, 'y') || 0;
 | 
						|
  const boundsWidth = _.get(bounds, 'width') || getDefaultWindowSize().defaultWidth;
 | 
						|
  const boundsHeight = _.get(bounds, 'height') || getDefaultWindowSize().defaultHeight;
 | 
						|
  const BOUNDS_BUFFER = 100;
 | 
						|
 | 
						|
  // 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 getStartInTray() {
 | 
						|
  const startInTray =
 | 
						|
    process.argv.some(arg => arg === '--start-in-tray') || userConfig.get('startInTray');
 | 
						|
  const usingTrayIcon = startInTray || process.argv.some(arg => arg === '--use-tray-icon');
 | 
						|
  return { usingTrayIcon, startInTray };
 | 
						|
}
 | 
						|
 | 
						|
async function createWindow() {
 | 
						|
  const { minWidth, minHeight, width, height } = getWindowSize();
 | 
						|
  windowConfig = windowConfig || {};
 | 
						|
  const picked = {
 | 
						|
    maximized: (windowConfig as any).maximized || false,
 | 
						|
    autoHideMenuBar: (windowConfig as any).autoHideMenuBar || false,
 | 
						|
    width: (windowConfig as any).width || width,
 | 
						|
    height: (windowConfig as any).height || height,
 | 
						|
    x: (windowConfig as any).x,
 | 
						|
    y: (windowConfig as any).y,
 | 
						|
  };
 | 
						|
 | 
						|
  if (isTestIntegration) {
 | 
						|
    const screenWidth =
 | 
						|
      screen.getPrimaryDisplay().workAreaSize.width - getDefaultWindowSize().defaultWidth;
 | 
						|
    const screenHeight =
 | 
						|
      screen.getPrimaryDisplay().workAreaSize.height - getDefaultWindowSize().defaultHeight;
 | 
						|
 | 
						|
    picked.x = Math.floor(Math.random() * screenWidth);
 | 
						|
    picked.y = Math.floor(Math.random() * screenHeight);
 | 
						|
  }
 | 
						|
 | 
						|
  const windowOptions = {
 | 
						|
    show: true,
 | 
						|
    minWidth,
 | 
						|
    minHeight,
 | 
						|
    fullscreen: false as boolean | undefined,
 | 
						|
    // Default theme is Classic Dark
 | 
						|
    backgroundColor: classicDark['--background-primary-color'],
 | 
						|
    webPreferences: {
 | 
						|
      nodeIntegration: true,
 | 
						|
      enableRemoteModule: true,
 | 
						|
      nodeIntegrationInWorker: true,
 | 
						|
      contextIsolation: false,
 | 
						|
      preload: path.join(getAppRootPath(), 'preload.js'),
 | 
						|
      nativeWindowOpen: true,
 | 
						|
      spellcheck: await getSpellCheckSetting(),
 | 
						|
    },
 | 
						|
    // only set icon for Linux, the executable one will be used by default for other platforms
 | 
						|
    icon:
 | 
						|
      (osPlatform === 'linux' && path.join(getAppRootPath(), 'images/session/session_icon.png')) ||
 | 
						|
      undefined,
 | 
						|
    ...picked,
 | 
						|
  };
 | 
						|
 | 
						|
  if (!_.isNumber(windowOptions.width) || windowOptions.width < minWidth) {
 | 
						|
    windowOptions.width = Math.max(minWidth, width);
 | 
						|
  }
 | 
						|
  if (!_.isNumber(windowOptions.height) || windowOptions.height < minHeight) {
 | 
						|
    windowOptions.height = Math.max(minHeight, 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;
 | 
						|
  }
 | 
						|
 | 
						|
  assertLogger().info('Initializing BrowserWindow config: %s', JSON.stringify(windowOptions));
 | 
						|
 | 
						|
  // Create the browser window.
 | 
						|
  mainWindow = new BrowserWindow(windowOptions);
 | 
						|
  setupSpellChecker(mainWindow, locale.messages);
 | 
						|
 | 
						|
  const setWindowFocus = () => {
 | 
						|
    if (!mainWindow) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    mainWindow.webContents.send('set-window-focus', mainWindow.isFocused());
 | 
						|
  };
 | 
						|
  mainWindow.on('focus', setWindowFocus);
 | 
						|
  mainWindow.on('blur', setWindowFocus);
 | 
						|
  mainWindow.once('ready-to-show', setWindowFocus);
 | 
						|
  // This is a fallback in case we drop an event for some reason.
 | 
						|
  global.setInterval(setWindowFocus, 5000);
 | 
						|
 | 
						|
  electronLocalshortcut.register(mainWindow, 'F5', () => {
 | 
						|
    if (!mainWindow) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    mainWindow.reload();
 | 
						|
  });
 | 
						|
  electronLocalshortcut.register(mainWindow, 'CommandOrControl+R', () => {
 | 
						|
    if (!mainWindow) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    mainWindow.reload();
 | 
						|
  });
 | 
						|
 | 
						|
  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],
 | 
						|
      fullscreen: false as boolean | undefined,
 | 
						|
    };
 | 
						|
 | 
						|
    if (mainWindow.isFullScreen()) {
 | 
						|
      // Only include this property if true, because when explicitly set to
 | 
						|
      // false the fullscreen button will be disabled on osx
 | 
						|
      (windowConfig as any).fullscreen = true;
 | 
						|
    }
 | 
						|
 | 
						|
    assertLogger().info('Updating BrowserWindow config: %s', JSON.stringify(windowConfig));
 | 
						|
    ephemeralConfig.set('window', windowConfig);
 | 
						|
  }
 | 
						|
 | 
						|
  const debouncedCaptureStats = _.debounce(captureAndSaveWindowStats, 500);
 | 
						|
  mainWindow.on('resize', debouncedCaptureStats);
 | 
						|
  mainWindow.on('move', debouncedCaptureStats);
 | 
						|
 | 
						|
  mainWindow.on('focus', () => {
 | 
						|
    if (!mainWindow) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    mainWindow.flashFrame(false);
 | 
						|
    if (passwordWindow) {
 | 
						|
      passwordWindow.close();
 | 
						|
      passwordWindow = null;
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  const urlToLoad = prepareURL([getAppRootPath(), 'background.html']);
 | 
						|
 | 
						|
  await mainWindow.loadURL(urlToLoad);
 | 
						|
  if (openDevToolsTestIntegration) {
 | 
						|
    setTimeout(() => {
 | 
						|
      if (mainWindow && mainWindow.webContents) {
 | 
						|
        mainWindow.webContents.openDevTools({
 | 
						|
          mode: 'bottom',
 | 
						|
          activate: false,
 | 
						|
        });
 | 
						|
      }
 | 
						|
    }, 5000);
 | 
						|
  }
 | 
						|
 | 
						|
  if ((process.env.NODE_APP_INSTANCE || '').startsWith('devprod')) {
 | 
						|
    // Open the DevTools.
 | 
						|
    mainWindow.webContents.openDevTools({
 | 
						|
      mode: 'bottom',
 | 
						|
      activate: false,
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  captureClicks(mainWindow);
 | 
						|
 | 
						|
  // Emitted when the window is about to be closed.
 | 
						|
  // Note: We do most of our shutdown logic here because all windows are closed by
 | 
						|
  //   Electron before the app quits.
 | 
						|
  mainWindow.on('close', async e => {
 | 
						|
    console.log('close event', {
 | 
						|
      readyForShutdown: mainWindow ? readyForShutdown : null,
 | 
						|
      shouldQuit: windowShouldQuit(),
 | 
						|
    });
 | 
						|
    // If the application is terminating, just do the default
 | 
						|
    if (mainWindow && readyForShutdown && windowShouldQuit()) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // Prevent the shutdown
 | 
						|
    e.preventDefault();
 | 
						|
    mainWindow?.hide();
 | 
						|
 | 
						|
    // 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 (!windowShouldQuit() && (getStartInTray().usingTrayIcon || process.platform === 'darwin')) {
 | 
						|
      // toggle the visibility of the show/hide tray icon menu entries
 | 
						|
      if (tray) {
 | 
						|
        tray.updateContextMenu();
 | 
						|
      }
 | 
						|
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    await requestShutdown();
 | 
						|
    if (mainWindow) {
 | 
						|
      readyForShutdown = true;
 | 
						|
    }
 | 
						|
    app.quit();
 | 
						|
  });
 | 
						|
 | 
						|
  // 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();
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('set-release-from-file-server', (_event, releaseGotFromFileServer) => {
 | 
						|
  setLastestRelease(releaseGotFromFileServer);
 | 
						|
});
 | 
						|
 | 
						|
let isReadyForUpdates = false;
 | 
						|
async function readyForUpdates() {
 | 
						|
  console.log('[updater] isReadyForUpdates', isReadyForUpdates);
 | 
						|
  if (isReadyForUpdates) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  isReadyForUpdates = true;
 | 
						|
 | 
						|
  // Second, start checking for app updates
 | 
						|
  try {
 | 
						|
    // if the user disabled auto updates, this will actually not start the updater
 | 
						|
    await updater.start(getMainWindow, userConfig, locale.messages, logger);
 | 
						|
  } catch (error) {
 | 
						|
    const log = logger || console;
 | 
						|
    log.error('Error starting update checks:', error && error.stack ? error.stack : error);
 | 
						|
  }
 | 
						|
}
 | 
						|
ipc.once('ready-for-updates', readyForUpdates);
 | 
						|
 | 
						|
// Forcefully call readyForUpdates after 10 minutes.
 | 
						|
// This ensures we start the updater.
 | 
						|
const TEN_MINUTES = 10 * 60 * 1000;
 | 
						|
setTimeout(readyForUpdates, TEN_MINUTES);
 | 
						|
 | 
						|
function openReleaseNotes() {
 | 
						|
  void shell.openExternal(
 | 
						|
    `https://github.com/oxen-io/session-desktop/releases/tag/v${app.getVersion()}`
 | 
						|
  );
 | 
						|
}
 | 
						|
 | 
						|
function openSupportPage() {
 | 
						|
  void shell.openExternal('https://docs.oxen.io/products-built-on-oxen/session');
 | 
						|
}
 | 
						|
 | 
						|
let passwordWindow: BrowserWindow | null = null;
 | 
						|
async function showPasswordWindow() {
 | 
						|
  if (passwordWindow) {
 | 
						|
    passwordWindow.show();
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  const { minWidth, minHeight, width, height } = getWindowSize();
 | 
						|
  const windowOptions = {
 | 
						|
    show: true, // allow to start minimised in tray
 | 
						|
    width,
 | 
						|
    height,
 | 
						|
    minWidth,
 | 
						|
    minHeight,
 | 
						|
    autoHideMenuBar: false,
 | 
						|
    // Default theme is Classic Dark
 | 
						|
    backgroundColor: classicDark['--background-primary-color'],
 | 
						|
    webPreferences: {
 | 
						|
      nodeIntegration: true,
 | 
						|
      enableRemoteModule: true,
 | 
						|
      nodeIntegrationInWorker: false,
 | 
						|
      contextIsolation: false,
 | 
						|
 | 
						|
      // sandbox: true,
 | 
						|
      preload: path.join(getAppRootPath(), 'password_preload.js'),
 | 
						|
      nativeWindowOpen: true,
 | 
						|
    },
 | 
						|
    // don't setup icon, the executable one will be used by default
 | 
						|
  };
 | 
						|
 | 
						|
  passwordWindow = new BrowserWindow(windowOptions);
 | 
						|
 | 
						|
  await passwordWindow.loadURL(prepareURL([getAppRootPath(), 'password.html']));
 | 
						|
 | 
						|
  captureClicks(passwordWindow);
 | 
						|
 | 
						|
  passwordWindow.on('close', e => {
 | 
						|
    // If the application is terminating, just do the default
 | 
						|
    if (windowShouldQuit()) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // Prevent the shutdown
 | 
						|
    e.preventDefault();
 | 
						|
    passwordWindow?.hide();
 | 
						|
 | 
						|
    // 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 (!windowShouldQuit() && (getStartInTray().usingTrayIcon || process.platform === 'darwin')) {
 | 
						|
      // toggle the visibility of the show/hide tray icon menu entries
 | 
						|
      if (tray) {
 | 
						|
        tray.updateContextMenu();
 | 
						|
      }
 | 
						|
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (passwordWindow) {
 | 
						|
      (passwordWindow as any).readyForShutdown = true;
 | 
						|
    }
 | 
						|
    // Quit the app if we don't have a main window
 | 
						|
    if (!mainWindow) {
 | 
						|
      app.quit();
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  passwordWindow.on('closed', () => {
 | 
						|
    passwordWindow = null;
 | 
						|
  });
 | 
						|
}
 | 
						|
 | 
						|
let aboutWindow: BrowserWindow | null;
 | 
						|
 | 
						|
async function showAbout() {
 | 
						|
  if (aboutWindow) {
 | 
						|
    aboutWindow.show();
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mainWindow) {
 | 
						|
    console.info('about window needs mainwindow as parent');
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  const theme = await getThemeFromMainWindow();
 | 
						|
  const options = {
 | 
						|
    width: 500,
 | 
						|
    height: 500,
 | 
						|
    resizable: true,
 | 
						|
    title: locale.messages.about,
 | 
						|
    autoHideMenuBar: true,
 | 
						|
    backgroundColor: classicDark['--background-primary-color'],
 | 
						|
    show: false,
 | 
						|
    webPreferences: {
 | 
						|
      nodeIntegration: true,
 | 
						|
      nodeIntegrationInWorker: false,
 | 
						|
      contextIsolation: false,
 | 
						|
      preload: path.join(getAppRootPath(), 'about_preload.js'),
 | 
						|
      nativeWindowOpen: true,
 | 
						|
    },
 | 
						|
    parent: mainWindow,
 | 
						|
  };
 | 
						|
 | 
						|
  aboutWindow = new BrowserWindow(options);
 | 
						|
 | 
						|
  captureClicks(aboutWindow);
 | 
						|
 | 
						|
  await aboutWindow.loadURL(prepareURL([getAppRootPath(), 'about.html'], { theme }));
 | 
						|
 | 
						|
  aboutWindow.on('closed', () => {
 | 
						|
    aboutWindow = null;
 | 
						|
  });
 | 
						|
 | 
						|
  aboutWindow.once('ready-to-show', () => {
 | 
						|
    aboutWindow?.setBackgroundColor(classicDark['--background-primary-color']);
 | 
						|
  });
 | 
						|
 | 
						|
  // looks like sometimes ready-to-show is not fired by electron.
 | 
						|
  // the fix mentioned here does not work neither: https://github.com/electron/electron/issues/7779.
 | 
						|
  // But, just showing the aboutWindow right away works correctly, so just force it to be shown when just created.
 | 
						|
  // It might take half a second to render it's content though.
 | 
						|
  aboutWindow?.show();
 | 
						|
}
 | 
						|
 | 
						|
let debugLogWindow: BrowserWindow | null = null;
 | 
						|
async function showDebugLogWindow() {
 | 
						|
  if (debugLogWindow) {
 | 
						|
    debugLogWindow.show();
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mainWindow) {
 | 
						|
    console.info('debug log needs mainwindow size to open');
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  const theme = await getThemeFromMainWindow();
 | 
						|
  const size = mainWindow.getSize();
 | 
						|
  const options = {
 | 
						|
    width: Math.max(size[0] - 100, getDefaultWindowSize().minWidth),
 | 
						|
    height: Math.max(size[1] - 100, getDefaultWindowSize().minHeight),
 | 
						|
    resizable: true,
 | 
						|
    title: locale.messages.debugLog,
 | 
						|
    autoHideMenuBar: true,
 | 
						|
    backgroundColor: classicDark['--background-primary-color'],
 | 
						|
    shadow: true,
 | 
						|
    show: false,
 | 
						|
    modal: true,
 | 
						|
    webPreferences: {
 | 
						|
      nodeIntegration: true,
 | 
						|
      nodeIntegrationInWorker: false,
 | 
						|
      contextIsolation: false,
 | 
						|
      preload: path.join(getAppRootPath(), 'debug_log_preload.js'),
 | 
						|
      nativeWindowOpen: true,
 | 
						|
    },
 | 
						|
    parent: mainWindow,
 | 
						|
  };
 | 
						|
 | 
						|
  debugLogWindow = new BrowserWindow(options);
 | 
						|
 | 
						|
  captureClicks(debugLogWindow);
 | 
						|
 | 
						|
  await debugLogWindow.loadURL(prepareURL([getAppRootPath(), 'debug_log.html'], { theme }));
 | 
						|
 | 
						|
  debugLogWindow.on('closed', () => {
 | 
						|
    debugLogWindow = null;
 | 
						|
  });
 | 
						|
 | 
						|
  debugLogWindow.once('ready-to-show', () => {
 | 
						|
    debugLogWindow?.setBackgroundColor(classicDark['--background-primary-color']);
 | 
						|
  });
 | 
						|
 | 
						|
  // see above: looks like sometimes ready-to-show is not fired by electron
 | 
						|
  debugLogWindow?.show();
 | 
						|
}
 | 
						|
 | 
						|
async function saveDebugLog(_event: any, logText: any) {
 | 
						|
  const options: Electron.SaveDialogOptions = {
 | 
						|
    title: 'Save debug log',
 | 
						|
    defaultPath: path.join(app.getPath('desktop'), `session_debug_${Date.now()}.txt`),
 | 
						|
    properties: ['createDirectory'],
 | 
						|
  };
 | 
						|
 | 
						|
  try {
 | 
						|
    const result = await dialog.showSaveDialog(options);
 | 
						|
    const outputPath = result.filePath;
 | 
						|
    console.info(`Trying to save logs to ${outputPath}`);
 | 
						|
    if (result === undefined || outputPath === undefined || outputPath === '') {
 | 
						|
      throw Error("User clicked Save button but didn't create a file");
 | 
						|
    }
 | 
						|
 | 
						|
    fs.writeFile(outputPath, logText, err => {
 | 
						|
      if (err) {
 | 
						|
        throw Error(`${err}`);
 | 
						|
      }
 | 
						|
      console.info(`Saved log - ${outputPath}`);
 | 
						|
    });
 | 
						|
  } catch (err) {
 | 
						|
    console.error('Error saving debug log', err);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// 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(join(app.getAppPath(), '..', '..'));
 | 
						|
 | 
						|
  installFileHandler({
 | 
						|
    protocol: electronProtocol,
 | 
						|
    userDataPath,
 | 
						|
    installPath,
 | 
						|
    isWindows: process.platform === 'win32',
 | 
						|
  });
 | 
						|
 | 
						|
  installWebHandler({
 | 
						|
    protocol: electronProtocol,
 | 
						|
  });
 | 
						|
 | 
						|
  installPermissionsHandler({ userConfig });
 | 
						|
 | 
						|
  await initializeLogger();
 | 
						|
  logger = getLogger();
 | 
						|
  assertLogger().info('app ready');
 | 
						|
  assertLogger().info(`starting version ${packageJson.version}`);
 | 
						|
  if (!locale) {
 | 
						|
    const appLocale = process.env.LANGUAGE || app.getLocale() || 'en';
 | 
						|
    locale = loadLocale({ appLocale, logger });
 | 
						|
    assertLogger().info(`locale is ${appLocale}`);
 | 
						|
  }
 | 
						|
 | 
						|
  const key = getDefaultSQLKey();
 | 
						|
 | 
						|
  // Try to show the main window with the default key
 | 
						|
  // If that fails then show the password window
 | 
						|
  const dbHasPassword = userConfig.get('dbHasPassword');
 | 
						|
  if (dbHasPassword) {
 | 
						|
    await showPasswordWindow();
 | 
						|
  } else {
 | 
						|
    await showMainWindow(key);
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
function getDefaultSQLKey() {
 | 
						|
  let key = userConfig.get('key');
 | 
						|
  if (!key) {
 | 
						|
    console.log('key/initialize: Generating new encryption key, since we did not find it on disk');
 | 
						|
    // https://www.zetetic.net/sqlcipher/sqlcipher-api/#key
 | 
						|
    key = crypto.randomBytes(32).toString('hex');
 | 
						|
    userConfig.set('key', key);
 | 
						|
  }
 | 
						|
 | 
						|
  return key as string;
 | 
						|
}
 | 
						|
 | 
						|
async function removeDB() {
 | 
						|
  // this don't remove attachments and stuff like that...
 | 
						|
  const userDir = await getRealPath(app.getPath('userData'));
 | 
						|
  sqlNode.removeDB(userDir);
 | 
						|
 | 
						|
  try {
 | 
						|
    console.error('Remove DB: removing.', userDir);
 | 
						|
 | 
						|
    userConfig.remove();
 | 
						|
    ephemeralConfig.remove();
 | 
						|
  } catch (e) {
 | 
						|
    console.error('Remove DB: Failed to remove configs.', e);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
async function showMainWindow(sqlKey: string, passwordAttempt = false) {
 | 
						|
  const userDataPath = await getRealPath(app.getPath('userData'));
 | 
						|
 | 
						|
  await sqlNode.initializeSql({
 | 
						|
    configDir: userDataPath,
 | 
						|
    key: sqlKey,
 | 
						|
    messages: locale.messages,
 | 
						|
    passwordAttempt,
 | 
						|
  });
 | 
						|
  appStartInitialSpellcheckSetting = await getSpellCheckSetting();
 | 
						|
  sqlChannels.initializeSqlChannel();
 | 
						|
 | 
						|
  await initAttachmentsChannel({
 | 
						|
    userDataPath,
 | 
						|
  });
 | 
						|
 | 
						|
  ready = true;
 | 
						|
 | 
						|
  await createWindow();
 | 
						|
 | 
						|
  if (getStartInTray().usingTrayIcon) {
 | 
						|
    tray = createTrayIcon(getMainWindow, locale.messages);
 | 
						|
  }
 | 
						|
 | 
						|
  setupMenu();
 | 
						|
}
 | 
						|
 | 
						|
function setupMenu() {
 | 
						|
  const { platform } = process;
 | 
						|
  const menuOptions = {
 | 
						|
    development,
 | 
						|
    showDebugLog: showDebugLogWindow,
 | 
						|
    showWindow,
 | 
						|
    showAbout,
 | 
						|
    openReleaseNotes,
 | 
						|
    openSupportPage,
 | 
						|
    platform,
 | 
						|
  };
 | 
						|
  const template = createTemplate(menuOptions, locale.messages);
 | 
						|
  const menu = Menu.buildFromTemplate(template);
 | 
						|
  Menu.setApplicationMenu(menu);
 | 
						|
}
 | 
						|
 | 
						|
async function requestShutdown() {
 | 
						|
  if (!mainWindow || !mainWindow.webContents) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  console.log('requestShutdown: Requesting close of mainWindow...');
 | 
						|
  const request = new Promise((resolve, reject) => {
 | 
						|
    ipc.once('now-ready-for-shutdown', (_event, error) => {
 | 
						|
      console.log('requestShutdown: Response received');
 | 
						|
 | 
						|
      if (error) {
 | 
						|
        reject(error);
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
      resolve(undefined);
 | 
						|
    });
 | 
						|
    mainWindow?.webContents.send('get-ready-for-shutdown');
 | 
						|
 | 
						|
    // We'll wait two minutes, then force the app to go down. This can happen if someone
 | 
						|
    //   exits the app before we've set everything up in preload() (so the browser isn't
 | 
						|
    //   yet listening for these events), or if there are a whole lot of stacked-up tasks.
 | 
						|
    // Note: two minutes is also our timeout for SQL tasks in data.ts in the browser.
 | 
						|
    setTimeout(() => {
 | 
						|
      console.log('requestShutdown: Response never received; forcing shutdown.');
 | 
						|
      resolve(undefined);
 | 
						|
    }, 2 * 60 * 1000);
 | 
						|
  });
 | 
						|
 | 
						|
  try {
 | 
						|
    await request;
 | 
						|
  } catch (error) {
 | 
						|
    console.log('requestShutdown error:', error && error.stack ? error.stack : error);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
app.on('before-quit', () => {
 | 
						|
  console.log('before-quit event', {
 | 
						|
    readyForShutdown: mainWindow ? readyForShutdown : null,
 | 
						|
    shouldQuit: windowShouldQuit(),
 | 
						|
  });
 | 
						|
 | 
						|
  if (tray) {
 | 
						|
    tray.destroy();
 | 
						|
  }
 | 
						|
 | 
						|
  windowMarkShouldQuit();
 | 
						|
});
 | 
						|
 | 
						|
// 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') {
 | 
						|
    app.quit();
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
app.on('activate', async () => {
 | 
						|
  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 {
 | 
						|
    await 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.setWindowOpenHandler(() => ({ action: 'deny' }));
 | 
						|
});
 | 
						|
 | 
						|
// Ingested in preload.js via a sendSync call
 | 
						|
ipc.on('locale-data', event => {
 | 
						|
  // eslint-disable-next-line no-param-reassign
 | 
						|
  event.returnValue = locale.messages;
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('draw-attention', () => {
 | 
						|
  if (!mainWindow) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  if (process.platform === 'win32') {
 | 
						|
    mainWindow.flashFrame(true);
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('restart', () => {
 | 
						|
  app.relaunch();
 | 
						|
  app.quit();
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('resetDatabase', async () => {
 | 
						|
  await removeDB();
 | 
						|
  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();
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
// Password screen related IPC calls
 | 
						|
ipc.on('password-window-login', async (event, passPhrase) => {
 | 
						|
  const sendResponse = (e: string | undefined) => {
 | 
						|
    event.sender.send('password-window-login-response', e);
 | 
						|
  };
 | 
						|
 | 
						|
  try {
 | 
						|
    const passwordAttempt = true;
 | 
						|
    await showMainWindow(passPhrase, passwordAttempt);
 | 
						|
    sendResponse(undefined);
 | 
						|
  } catch (e) {
 | 
						|
    const localisedError = locale.messages.removePasswordInvalid;
 | 
						|
    sendResponse(localisedError);
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('start-in-tray-on-start', (event, newValue) => {
 | 
						|
  try {
 | 
						|
    userConfig.set('startInTray', newValue);
 | 
						|
    if (newValue) {
 | 
						|
      if (!tray) {
 | 
						|
        tray = createTrayIcon(getMainWindow, locale.messages);
 | 
						|
      }
 | 
						|
    } else {
 | 
						|
      // destroy is not working for a lot of desktop env. So for simplicity, we don't destroy it here but just
 | 
						|
      // show a toast to explain to the user that he needs to restart
 | 
						|
      // tray.destroy();
 | 
						|
      // tray = null;
 | 
						|
    }
 | 
						|
    event.sender.send('start-in-tray-on-start-response', null);
 | 
						|
  } catch (e) {
 | 
						|
    event.sender.send('start-in-tray-on-start-response', e);
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('get-start-in-tray', event => {
 | 
						|
  try {
 | 
						|
    const val = userConfig.get('startInTray');
 | 
						|
    event.sender.send('get-start-in-tray-response', val);
 | 
						|
  } catch (e) {
 | 
						|
    event.sender.send('get-start-in-tray-response', false);
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('get-opengroup-pruning', event => {
 | 
						|
  try {
 | 
						|
    const val = userConfig.get('opengroupPruning');
 | 
						|
    event.sender.send('get-opengroup-pruning-response', val);
 | 
						|
  } catch (e) {
 | 
						|
    event.sender.send('get-opengroup-pruning-response', false);
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('set-opengroup-pruning', (event, newValue) => {
 | 
						|
  try {
 | 
						|
    userConfig.set('opengroupPruning', newValue);
 | 
						|
    event.sender.send('set-opengroup-pruning-response', null);
 | 
						|
  } catch (e) {
 | 
						|
    event.sender.send('set-opengroup-pruning-response', e);
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('set-password', async (event, passPhrase, oldPhrase) => {
 | 
						|
  const sendResponse = (response: string | undefined) => {
 | 
						|
    event.sender.send('set-password-response', response);
 | 
						|
  };
 | 
						|
 | 
						|
  try {
 | 
						|
    // Check if the hash we have stored matches the hash of the old passphrase.
 | 
						|
    const hash = sqlNode.getPasswordHash();
 | 
						|
 | 
						|
    const hashMatches = oldPhrase && PasswordUtil.matchesHash(oldPhrase, hash);
 | 
						|
    if (hash && !hashMatches) {
 | 
						|
      const incorrectOldPassword = locale.messages.invalidOldPassword;
 | 
						|
      sendResponse(
 | 
						|
        incorrectOldPassword || 'Failed to set password: Old password provided is invalid'
 | 
						|
      );
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (_.isEmpty(passPhrase)) {
 | 
						|
      const defaultKey = getDefaultSQLKey();
 | 
						|
      sqlNode.setSQLPassword(defaultKey);
 | 
						|
      sqlNode.removePasswordHash();
 | 
						|
      userConfig.set('dbHasPassword', false);
 | 
						|
    } else {
 | 
						|
      sqlNode.setSQLPassword(passPhrase);
 | 
						|
      const newHash = PasswordUtil.generateHash(passPhrase);
 | 
						|
      sqlNode.savePasswordHash(newHash);
 | 
						|
      userConfig.set('dbHasPassword', true);
 | 
						|
    }
 | 
						|
 | 
						|
    sendResponse(undefined);
 | 
						|
  } catch (e) {
 | 
						|
    const localisedError = locale.messages.setPasswordFail;
 | 
						|
    sendResponse(localisedError || 'Failed to set password');
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
// Debug Log-related IPC calls
 | 
						|
 | 
						|
ipc.on('show-debug-log', showDebugLogWindow);
 | 
						|
ipc.on('close-debug-log', () => {
 | 
						|
  if (debugLogWindow) {
 | 
						|
    debugLogWindow.close();
 | 
						|
  }
 | 
						|
});
 | 
						|
ipc.on('save-debug-log', saveDebugLog);
 | 
						|
 | 
						|
// This should be called with an ipc sendSync
 | 
						|
ipc.on('get-media-permissions', event => {
 | 
						|
  // eslint-disable-next-line no-param-reassign
 | 
						|
  event.returnValue = 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({ userConfig });
 | 
						|
 | 
						|
  event.sender.send('set-success-media-permissions', null);
 | 
						|
});
 | 
						|
 | 
						|
// This should be called with an ipc sendSync
 | 
						|
ipc.on('get-call-media-permissions', event => {
 | 
						|
  // eslint-disable-next-line no-param-reassign
 | 
						|
  event.returnValue = userConfig.get('callMediaPermissions') || false;
 | 
						|
});
 | 
						|
ipc.on('set-call-media-permissions', (event, value) => {
 | 
						|
  userConfig.set('callMediaPermissions', value);
 | 
						|
 | 
						|
  // We reinstall permissions handler to ensure that a revoked permission takes effect
 | 
						|
  installPermissionsHandler({ userConfig });
 | 
						|
 | 
						|
  event.sender.send('set-success-call-media-permissions', null);
 | 
						|
});
 | 
						|
 | 
						|
// Session - Auto updating
 | 
						|
ipc.on('get-auto-update-setting', event => {
 | 
						|
  const configValue = userConfig.get('autoUpdate');
 | 
						|
  // eslint-disable-next-line no-param-reassign
 | 
						|
  event.returnValue = typeof configValue !== 'boolean' ? true : configValue;
 | 
						|
});
 | 
						|
 | 
						|
ipc.on('set-auto-update-setting', async (_event, enabled) => {
 | 
						|
  userConfig.set('autoUpdate', !!enabled);
 | 
						|
 | 
						|
  if (enabled) {
 | 
						|
    await readyForUpdates();
 | 
						|
  } else {
 | 
						|
    updater.stop();
 | 
						|
    isReadyForUpdates = false;
 | 
						|
  }
 | 
						|
});
 | 
						|
 | 
						|
async function getThemeFromMainWindow() {
 | 
						|
  return new Promise(resolve => {
 | 
						|
    ipc.once('get-success-theme-setting', (_event, value) => {
 | 
						|
      resolve(value);
 | 
						|
    });
 | 
						|
    mainWindow?.webContents.send('get-theme-setting');
 | 
						|
  });
 | 
						|
}
 | 
						|
 | 
						|
async function askForMediaAccess() {
 | 
						|
  // Microphone part
 | 
						|
  let status = systemPreferences.getMediaAccessStatus('microphone');
 | 
						|
  if (status !== 'granted') {
 | 
						|
    await systemPreferences.askForMediaAccess('microphone');
 | 
						|
  }
 | 
						|
  // Camera part
 | 
						|
  status = systemPreferences.getMediaAccessStatus('camera');
 | 
						|
  if (status !== 'granted') {
 | 
						|
    await systemPreferences.askForMediaAccess('camera');
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
ipc.on('media-access', async () => {
 | 
						|
  await askForMediaAccess();
 | 
						|
});
 |