import electron, { BrowserWindow, MenuItemConstructorOptions } from 'electron'; import fs from 'fs'; import { Curl } from 'node-libcurl'; import * as os from 'os'; import path from 'path'; import { changelogUrl, getAppLongName, getAppName, getAppReleaseDate, getAppVersion, isDevelopment, isLinux, isMac, MNEMONIC_SYM, } from '../common/constants'; import { docsBase } from '../common/documentation'; import { clickLink, getDataDirectory, restartApp } from '../common/electron-helpers'; import * as log from '../common/log'; import LocalStorage from './local-storage'; const { app, Menu, shell, dialog, clipboard } = electron; const DEFAULT_WIDTH = 1280; const DEFAULT_HEIGHT = 700; const MINIMUM_WIDTH = 500; const MINIMUM_HEIGHT = 400; let mainWindow: BrowserWindow | null = null; let localStorage: LocalStorage | null = null; interface Bounds { height?: number; width?: number; x?: number; y?: number; } export function init() { initLocalStorage(); initContextMenus(); } export function createWindow() { const zoomFactor = getZoomFactor(); const { bounds, fullscreen, maximize } = getBounds(); const { x, y, width, height } = bounds; const appLogo = 'static/insomnia-core-logo_16x.png'; let isVisibleOnAnyDisplay = true; for (const d of electron.screen.getAllDisplays()) { const isVisibleOnDisplay = // @ts-expect-error -- TSCONVERSION genuine error x >= d.bounds.x && // @ts-expect-error -- TSCONVERSION genuine error y >= d.bounds.y && // @ts-expect-error -- TSCONVERSION genuine error x + width <= d.bounds.x + d.bounds.width && // @ts-expect-error -- TSCONVERSION genuine error y + height <= d.bounds.y + d.bounds.height; if (!isVisibleOnDisplay) { isVisibleOnAnyDisplay = false; } } mainWindow = new BrowserWindow({ // Make sure we don't initialize the window outside the bounds x: isVisibleOnAnyDisplay ? x : undefined, y: isVisibleOnAnyDisplay ? y : undefined, // Other options backgroundColor: '#2C2C2C', fullscreen: fullscreen, fullscreenable: true, title: getAppName(), width: width || DEFAULT_WIDTH, height: height || DEFAULT_HEIGHT, minHeight: MINIMUM_HEIGHT, minWidth: MINIMUM_WIDTH, acceptFirstMouse: true, icon: path.resolve(__dirname, appLogo), webPreferences: { zoomFactor: zoomFactor, nodeIntegration: true, webviewTag: true, enableRemoteModule: true, // TODO: enable context isolation contextIsolation: false, disableBlinkFeatures: 'Auxclick', }, }); // BrowserWindow doesn't have an option for this, so we have to do it manually :( if (maximize) { mainWindow?.maximize(); } mainWindow?.on('resize', () => saveBounds()); mainWindow?.on('maximize', () => saveBounds()); mainWindow?.on('unmaximize', () => saveBounds()); mainWindow?.on('move', () => saveBounds()); mainWindow?.on('unresponsive', () => { showUnresponsiveModal(); }); // Open generic links () in default browser mainWindow?.webContents.on('will-navigate', (e, url) => { if (url === appUrl) { return; } console.log('[app] Navigate to ' + url); e.preventDefault(); clickLink(url); }); mainWindow?.webContents.on('new-window', e => { e.preventDefault(); }); // Load the html of the app. const url = process.env.APP_RENDER_URL; const appUrl = url || `file://${app.getAppPath()}/renderer.html`; console.log(`[main] Loading ${appUrl}`); mainWindow?.loadURL(appUrl); // 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; }); const applicationMenu: MenuItemConstructorOptions = { label: `${MNEMONIC_SYM}Application`, submenu: [ { label: `${MNEMONIC_SYM}Preferences`, click: function(_menuItem, window) { if (!window || !window.webContents) { return; } window.webContents.send('toggle-preferences'); }, }, { label: `${MNEMONIC_SYM}Changelog`, click: function(_menuItem, window) { if (!window || !window.webContents) { return; } clickLink(changelogUrl()); }, }, { type: 'separator', }, { role: 'hide', }, { // @ts-expect-error -- TSCONVERSION appears to be a genuine error role: 'hideothers', }, { type: 'separator', }, { label: `${MNEMONIC_SYM}Quit`, accelerator: 'CmdOrCtrl+Q', click: () => app.quit(), }, ], }; const editMenu: MenuItemConstructorOptions = { label: `${MNEMONIC_SYM}Edit`, submenu: [ { label: `${MNEMONIC_SYM}Undo`, accelerator: 'CmdOrCtrl+Z', role: 'undo', }, { label: `${MNEMONIC_SYM}Redo`, accelerator: 'Shift+CmdOrCtrl+Z', role: 'redo', }, { type: 'separator', }, { label: `Cu${MNEMONIC_SYM}t`, accelerator: 'CmdOrCtrl+X', role: 'cut', }, { label: `${MNEMONIC_SYM}Copy`, accelerator: 'CmdOrCtrl+C', role: 'copy', }, { label: `${MNEMONIC_SYM}Paste`, accelerator: 'CmdOrCtrl+V', role: 'paste', }, { label: `Select ${MNEMONIC_SYM}All`, accelerator: 'CmdOrCtrl+A', role: 'selectAll', }, ], }; const viewMenu: MenuItemConstructorOptions = { label: `${MNEMONIC_SYM}View`, submenu: [ { label: `Toggle ${MNEMONIC_SYM}Full Screen`, role: 'togglefullscreen', }, { label: `${MNEMONIC_SYM}Actual Size`, accelerator: 'CmdOrCtrl+0', click: () => { const w = BrowserWindow.getFocusedWindow(); if (!w || !w.webContents) { return; } const zoomFactor = 1; w.webContents.setZoomFactor(zoomFactor); saveZoomFactor(zoomFactor); }, }, { label: `Zoom ${MNEMONIC_SYM}In`, accelerator: 'CmdOrCtrl+=', click: () => { const w = BrowserWindow.getFocusedWindow(); if (!w || !w.webContents) { return; } const zoomFactor = Math.min(1.8, getZoomFactor() + 0.05); w.webContents.setZoomFactor(zoomFactor); saveZoomFactor(zoomFactor); }, }, { label: `Zoom ${MNEMONIC_SYM}Out`, accelerator: 'CmdOrCtrl+-', click: () => { const w = BrowserWindow.getFocusedWindow(); if (!w || !w.webContents) { return; } const zoomFactor = Math.max(0.5, getZoomFactor() - 0.05); w.webContents.setZoomFactor(zoomFactor); saveZoomFactor(zoomFactor); }, }, { type: 'separator', }, { label: 'Toggle Sidebar', click: () => { const w = BrowserWindow.getFocusedWindow(); if (!w || !w.webContents) { return; } w.webContents.send('toggle-sidebar'); }, }, { label: `Toggle ${MNEMONIC_SYM}DevTools`, accelerator: 'Alt+CmdOrCtrl+I', // @ts-expect-error -- TSCONVERSION needs global module augmentation click: () => mainWindow?.toggleDevTools(), }, ], }; const windowMenu: MenuItemConstructorOptions = { label: `${MNEMONIC_SYM}Window`, role: 'window', submenu: [ { label: `${MNEMONIC_SYM}Minimize`, role: 'minimize', }, // @ts-expect-error -- TSCONVERSION missing in official electron types ...(isMac() ? [ { label: `${MNEMONIC_SYM}Close`, role: 'close', }, ] : []), ], }; const helpMenu: MenuItemConstructorOptions = { label: `${MNEMONIC_SYM}Help`, role: 'help', id: 'help', submenu: [ { label: `${MNEMONIC_SYM}Help and Support`, ...(isMac() ? {} : { accelerator: 'F1' }), click: () => { clickLink(docsBase); }, }, { label: `${MNEMONIC_SYM}Keyboard Shortcuts`, accelerator: 'CmdOrCtrl+Shift+?', click: (_menuItem, w) => { if (!w || !w.webContents) { return; } w.webContents.send('toggle-preferences-shortcuts'); }, }, { type: 'separator', }, { label: `Show App ${MNEMONIC_SYM}Data Folder`, click: () => { const directory = getDataDirectory(); shell.showItemInFolder(directory); }, }, { label: `Show App ${MNEMONIC_SYM}Logs Folder`, click: () => { const directory = log.getLogDirectory(); shell.showItemInFolder(directory); }, }, { type: 'separator', }, { label: 'Show Open Source Licenses', click: () => { const licensePath = path.resolve(app.getAppPath(), '../opensource-licenses.txt'); shell.openPath(licensePath); }, }, { label: 'Show Software License', click: () => { clickLink('https://insomnia.rest/license'); }, }, ], }; const aboutMenuClickHandler = async () => { const copy = 'Copy'; const ok = 'OK'; const buttons = isLinux() ? [copy, ok] : [ok, copy]; const detail = [ `Version: ${getAppLongName()} ${getAppVersion()}`, `Release date: ${getAppReleaseDate()}`, `OS: ${os.type()} ${os.arch()} ${os.release()}`, `Electron: ${process.versions.electron}`, `Node: ${process.versions.node}`, `V8: ${process.versions.v8}`, `Architecture: ${process.arch}`, `node-libcurl: ${Curl.getVersion()}`, ].join('\n'); const msgBox = await dialog.showMessageBox({ type: 'info', title: getAppName(), message: getAppLongName(), detail, buttons, defaultId: buttons.indexOf(ok), cancelId: buttons.indexOf(ok), noLink: true, }); if (msgBox.response === buttons.indexOf(copy)) { clipboard.writeText(detail); } }; if (isMac()) { // @ts-expect-error -- TSCONVERSION type splitting applicationMenu.submenu?.unshift( { label: `A${MNEMONIC_SYM}bout ${getAppName()}`, click: aboutMenuClickHandler, }, { type: 'separator', }, ); } else { // @ts-expect-error -- TSCONVERSION type splitting helpMenu.submenu?.push({ type: 'separator', }, { label: `${MNEMONIC_SYM}About`, click: aboutMenuClickHandler, }); } const developerMenu: MenuItemConstructorOptions = { label: `${MNEMONIC_SYM}Developer`, // @ts-expect-error -- TSCONVERSION missing in official electron types position: 'before=help', submenu: [ { label: `${MNEMONIC_SYM}Reload`, accelerator: 'Shift+F5', click: () => mainWindow?.reload(), }, { label: `Resize to Defaul${MNEMONIC_SYM}t`, click: () => mainWindow?.setBounds({ x: 100, y: 100, width: DEFAULT_WIDTH, height: DEFAULT_HEIGHT, }), }, { label: `Take ${MNEMONIC_SYM}Screenshot`, click: function() { // @ts-expect-error -- TSCONVERSION not accounted for in the electron types to provide a function mainWindow?.capturePage(image => { const buffer = image.toPNG(); const dir = app.getPath('desktop'); fs.writeFileSync(path.join(dir, `Screenshot-${new Date()}.png`), buffer); }); }, }, { label: `${MNEMONIC_SYM}Clear a model`, click: function(_menuItem, window) { window?.webContents?.send('clear-model'); }, }, { label: `Clear ${MNEMONIC_SYM}all models`, click: function(_menuItem, window) { window?.webContents?.send('clear-all-models'); }, }, { label: `R${MNEMONIC_SYM}estart`, click: restartApp, }, ], }; const toolsMenu: MenuItemConstructorOptions = { label: `${MNEMONIC_SYM}Tools`, submenu: [ { label: `${MNEMONIC_SYM}Reload Plugins`, click: () => { const w = BrowserWindow.getFocusedWindow(); if (!w || !w.webContents) { return; } w.webContents.send('reload-plugins'); }, }, ], }; const template: MenuItemConstructorOptions[] = []; template.push(applicationMenu); template.push(editMenu); template.push(viewMenu); template.push(windowMenu); template.push(toolsMenu); template.push(helpMenu); if (isDevelopment() || process.env.INSOMNIA_FORCE_DEBUG) { template.push(developerMenu); } Menu.setApplicationMenu(Menu.buildFromTemplate(template)); return mainWindow; } async function showUnresponsiveModal() { const id = await dialog.showMessageBox({ type: 'info', buttons: ['Cancel', 'Reload'], defaultId: 1, cancelId: 0, title: 'Unresponsive', message: 'Insomnia has become unresponsive. Do you want to reload?', }); // @ts-expect-error -- TSCONVERSION appears to be a genuine error if (id === 1) { mainWindow?.destroy(); createWindow(); } } function saveBounds() { if (!mainWindow) { return; } const fullscreen = mainWindow?.isFullScreen(); // Only save the size if we're not in fullscreen if (!fullscreen) { localStorage?.setItem('bounds', mainWindow?.getBounds()); localStorage?.setItem('maximize', mainWindow?.isMaximized()); localStorage?.setItem('fullscreen', false); } else { localStorage?.setItem('fullscreen', true); } } function getBounds() { let bounds: Bounds = {}; let fullscreen = false; let maximize = false; try { bounds = localStorage?.getItem('bounds', {}); fullscreen = localStorage?.getItem('fullscreen', false); maximize = localStorage?.getItem('maximize', false); } catch (e) { // This should never happen, but if it does...! console.error('Failed to parse window bounds', e); } return { bounds, fullscreen, maximize, }; } function saveZoomFactor(zoomFactor) { localStorage?.setItem('zoomFactor', zoomFactor); } function getZoomFactor() { let zoomFactor = 1; try { zoomFactor = localStorage?.getItem('zoomFactor', 1); } catch (e) { // This should never happen, but if it does...! console.error('Failed to parse zoomFactor', e); } return zoomFactor; } function initLocalStorage() { const localStoragePath = path.join(getDataDirectory(), 'localStorage'); localStorage = new LocalStorage(localStoragePath); } function initContextMenus() { // eslint-disable-next-line @typescript-eslint/no-var-requires require('electron-context-menu')({}); }