insomnia/packages/insomnia-app/app/main/window-utils.ts
James Gatz 7cb1cbef0d
Prevent opening new windows in the app (#3874)
* Force auxclick (middle-click) events to be handled as click events

* Prevent opening new windows

* Formatting: Remove parentheses from arrow fn
2021-08-10 12:28:12 +03:00

602 lines
16 KiB
TypeScript

import electron, { BrowserWindow } 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;
// So we can use native modules in renderer
// NOTE: This will be deprecated in Electron 10 and impossible in 11
// https://github.com/electron/electron/issues/18397
app.allowRendererProcessReuse = false;
// Note: this hack is required because MenuItemConstructorOptions is not exported from the electron types as of 9.3.5
type MenuItemConstructorOptions = Parameters<typeof Menu.buildFromTemplate>[0][0];
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 (<a .../>) 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',
// @ts-expect-error -- TSCONVERSION missing in official electron types
selector: 'undo:',
},
{
label: `${MNEMONIC_SYM}Redo`,
accelerator: 'Shift+CmdOrCtrl+Z',
// @ts-expect-error -- TSCONVERSION missing in official electron types
selector: 'redo:',
},
{
type: 'separator',
},
{
label: `Cu${MNEMONIC_SYM}t`,
accelerator: 'CmdOrCtrl+X',
// @ts-expect-error -- TSCONVERSION missing in official electron types
selector: 'cut:',
},
{
label: `${MNEMONIC_SYM}Copy`,
accelerator: 'CmdOrCtrl+C',
// @ts-expect-error -- TSCONVERSION missing in official electron types
selector: 'copy:',
},
{
label: `${MNEMONIC_SYM}Paste`,
accelerator: 'CmdOrCtrl+V',
// @ts-expect-error -- TSCONVERSION missing in official electron types
selector: 'paste:',
},
{
label: `Select ${MNEMONIC_SYM}All`,
accelerator: 'CmdOrCtrl+A',
// @ts-expect-error -- TSCONVERSION missing in official electron types
selector: '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`,
// @ts-expect-error -- TSCONVERSION TSCONVERSION `Accelerator` type from electron is needed here as a cast but is not exported as of the 9.3.5 types
accelerator: !isMac() ? 'F1' : null,
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')({});
}