diff --git a/src/packager/electron-superpowers.js b/src/packager/electron-superpowers.js new file mode 100644 index 00000000..6578ade2 --- /dev/null +++ b/src/packager/electron-superpowers.js @@ -0,0 +1,80 @@ +const exposeBrowserWindowMethods = [ + // https://www.electronjs.org/docs/latest/api/browser-window + 'focus', + 'blur', + 'isFocused', + 'show', + 'showInactive', + 'hide', + 'isVisible', + 'maximize', + 'unmaximize', + 'isMaximized', + 'minimize', + 'restore', + 'isMinimized', + 'isNormal', + 'setAspectRatio', + 'setBounds', + 'getBounds', + 'setContentBounds', + 'getContentBounds', + 'getNormalBounds', + 'setEnabled', + 'isEnabled', + 'setSize', + 'getSize', + 'setContentSize', + 'getContentSize', + 'setMinimumSize', + 'getMinimumSize', + 'setMaximumSize', + 'getMaximumSize', + 'setResizable', + 'isResizable', + 'setMovable', + 'isMovable', + 'setMinimizable', + 'isMinimizable', + 'setMaximizable', + 'isMaximizable', + 'setClosable', + 'isClosable', + 'setAlwaysOnTop', + 'isAlwaysOnTop', + 'moveTop', + 'center', + 'setPosition', + 'getPosition', + 'setProgressBar', + 'setOpacity', + 'getOpacity', + 'setFocusable', + 'isFocusable' +]; + +const main = ` +const {ipcMain, BrowserWindow} = require('electron'); + +const ALLOWED_METHODS = ${JSON.stringify(exposeBrowserWindowMethods)}; + +ipcMain.handle('superpowers/BrowserWindow', (event, name, ...args) => { + if (!ALLOWED_METHODS.includes(name)) { + throw new Error('Unsupported BrowserWindow method: ' + name); + } + return BrowserWindow.fromWebContents(event.sender)[name](...args); +}); +`.trimStart(); + +const preload = ` +const {contextBridge, ipcRenderer} = require('electron'); + +contextBridge.exposeInMainWorld('PackagerSuperpowers', { + BrowserWindow: (name, ...args) => ipcRenderer.invoke('superpowers/BrowserWindow', name, ...args) +}); +`.trimStart(); + +export default { + main, + preload +}; diff --git a/src/packager/packager.js b/src/packager/packager.js index 85a88195..c9ebc3f0 100644 --- a/src/packager/packager.js +++ b/src/packager/packager.js @@ -11,6 +11,7 @@ import {APP_NAME, WEBSITE, COPYRIGHT_NOTICE, ACCENT_COLOR} from './brand'; import {OutdatedPackagerError} from '../common/errors'; import {darken} from './colors'; import {Adapter} from './adapter'; +import electronSuperpowerScripts from './electron-superpowers'; const PROGRESS_LOADED_SCRIPTS = 0.1; @@ -512,6 +513,8 @@ cd "$(dirname "$0")" const contentsPrefix = isMac ? `${rootPrefix}${packageName}.app/Contents/` : rootPrefix; const resourcesPrefix = isMac ? `${contentsPrefix}Resources/app/` : `${contentsPrefix}resources/app/`; const electronMainName = 'electron-main.js'; + const electronSuperpowersMainName = 'electron-superpowers.js'; + const electronSuperpowersPreloadName = 'electron-preload.js'; const iconName = 'icon.png'; const icon = await Adapter.getAppIcon(this.options.app.icon); @@ -525,7 +528,7 @@ cd "$(dirname "$0")" zip.file(`${resourcesPrefix}package.json`, JSON.stringify(manifest, null, 4)); const mainJS = `'use strict'; -const {app, BrowserWindow, Menu, shell, screen, dialog} = require('electron'); +const {app, BrowserWindow, Menu, shell, screen, dialog, ipcMain} = require('electron'); const path = require('path'); const isWindows = process.platform === 'win32'; @@ -558,6 +561,7 @@ const createWindow = (windowOptions) => { sandbox: true, contextIsolation: true, nodeIntegration: false, + preload: path.resolve(__dirname, ${JSON.stringify(electronSuperpowersPreloadName)}), }, show: true, width: 480, @@ -695,8 +699,12 @@ app.on('window-all-closed', () => { app.whenReady().then(() => { createProjectWindow(defaultProjectURL); }); + +require('./' + ${JSON.stringify(electronSuperpowersMainName)}); `; zip.file(`${resourcesPrefix}${electronMainName}`, mainJS); + zip.file(`${resourcesPrefix}${electronSuperpowersMainName}`, electronSuperpowerScripts.main) + zip.file(`${resourcesPrefix}${electronSuperpowersPreloadName}`, electronSuperpowerScripts.preload); for (const [path, data] of Object.entries(projectZip.files)) { setFileFast(zip, `${resourcesPrefix}${path}`, data);