2016-04-05 14:35:11 -04:00
|
|
|
"use strict";
|
2016-03-27 20:40:07 +02:00
|
|
|
|
2016-04-05 14:35:11 -04:00
|
|
|
const electron = require("electron");
|
2023-03-22 23:53:10 +01:00
|
|
|
const core = require("./app");
|
|
|
|
const Log = require("./logger");
|
2016-03-30 12:20:46 +02:00
|
|
|
|
|
|
|
// Config
|
2021-01-05 18:48:55 +01:00
|
|
|
let config = process.env.config ? JSON.parse(process.env.config) : {};
|
2016-03-27 20:40:07 +02:00
|
|
|
// Module to control application life.
|
|
|
|
const app = electron.app;
|
2023-12-13 22:49:35 +01:00
|
|
|
// Per default electron is started with --disable-gpu flag, if you want the gpu enabled,
|
|
|
|
// you must set the env var ELECTRON_ENABLE_GPU=1 on startup.
|
2022-03-29 21:24:17 +02:00
|
|
|
// See https://www.electronjs.org/docs/latest/tutorial/offscreen-rendering for more info.
|
2023-12-13 22:49:35 +01:00
|
|
|
if (process.env.ELECTRON_ENABLE_GPU !== "1") {
|
2022-03-29 20:38:45 +02:00
|
|
|
app.disableHardwareAcceleration();
|
|
|
|
}
|
|
|
|
|
2016-03-27 20:40:07 +02:00
|
|
|
// Module to create native browser window.
|
|
|
|
const BrowserWindow = electron.BrowserWindow;
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
|
2020-07-27 14:24:30 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
2023-12-25 08:17:11 +01:00
|
|
|
function createWindow () {
|
2023-08-12 22:57:42 +02:00
|
|
|
// see https://www.electronjs.org/docs/latest/api/screen
|
|
|
|
// Create a window that fills the screen's available work area.
|
|
|
|
let electronSize = (800, 600);
|
|
|
|
try {
|
|
|
|
electronSize = electron.screen.getPrimaryDisplay().workAreaSize;
|
|
|
|
} catch {
|
|
|
|
Log.warn("Could not get display size, using defaults ...");
|
|
|
|
}
|
|
|
|
|
2021-08-30 11:32:24 +02:00
|
|
|
let electronSwitchesDefaults = ["autoplay-policy", "no-user-gesture-required"];
|
|
|
|
app.commandLine.appendSwitch(...new Set(electronSwitchesDefaults, config.electronSwitches));
|
2021-01-05 18:48:55 +01:00
|
|
|
let electronOptionsDefaults = {
|
2023-08-12 22:57:42 +02:00
|
|
|
width: electronSize.width,
|
|
|
|
height: electronSize.height,
|
2016-12-29 22:23:08 -03:00
|
|
|
x: 0,
|
2016-12-02 21:09:16 +01:00
|
|
|
y: 0,
|
2016-12-29 22:23:08 -03:00
|
|
|
darkTheme: true,
|
2016-12-02 21:09:16 +01:00
|
|
|
webPreferences: {
|
2021-02-27 13:19:16 +01:00
|
|
|
contextIsolation: true,
|
2016-12-29 22:23:08 -03:00
|
|
|
nodeIntegration: false,
|
2016-12-02 21:09:16 +01:00
|
|
|
zoomFactor: config.zoom
|
2017-01-04 12:52:21 -03:00
|
|
|
},
|
|
|
|
backgroundColor: "#000000"
|
2017-06-11 11:53:55 +02:00
|
|
|
};
|
2016-12-02 21:09:16 +01:00
|
|
|
|
2016-12-10 17:02:54 +01:00
|
|
|
// DEPRECATED: "kioskmode" backwards compatibility, to be removed
|
|
|
|
// settings these options directly instead provides cleaner interface
|
2016-07-27 12:50:59 +02:00
|
|
|
if (config.kioskmode) {
|
2016-12-10 17:02:54 +01:00
|
|
|
electronOptionsDefaults.kiosk = true;
|
2016-07-27 12:50:59 +02:00
|
|
|
} else {
|
2023-01-26 22:16:50 +01:00
|
|
|
electronOptionsDefaults.show = false;
|
|
|
|
electronOptionsDefaults.frame = false;
|
|
|
|
electronOptionsDefaults.transparent = true;
|
|
|
|
electronOptionsDefaults.hasShadow = false;
|
2023-09-14 08:01:50 +02:00
|
|
|
electronOptionsDefaults.fullscreen = true;
|
2016-07-27 12:50:59 +02:00
|
|
|
}
|
2016-03-30 12:20:46 +02:00
|
|
|
|
2021-01-05 18:48:55 +01:00
|
|
|
const electronOptions = Object.assign({}, electronOptionsDefaults, config.electronOptions);
|
2016-12-10 17:02:54 +01:00
|
|
|
|
2016-12-02 21:09:16 +01:00
|
|
|
// Create the browser window.
|
|
|
|
mainWindow = new BrowserWindow(electronOptions);
|
|
|
|
|
2016-03-30 12:20:46 +02:00
|
|
|
// and load the index.html of the app.
|
2017-07-17 14:23:24 +02:00
|
|
|
// If config.address is not defined or is an empty string (listening on all interfaces), connect to localhost
|
2020-03-28 12:37:50 +01:00
|
|
|
|
2021-01-05 18:48:55 +01:00
|
|
|
let prefix;
|
2021-12-26 17:43:27 +01:00
|
|
|
if ((config["tls"] !== null && config["tls"]) || config.useHttps) {
|
2020-04-03 12:37:33 +02:00
|
|
|
prefix = "https://";
|
2020-03-28 12:37:50 +01:00
|
|
|
} else {
|
2020-04-03 12:37:33 +02:00
|
|
|
prefix = "http://";
|
2020-03-28 12:37:50 +01:00
|
|
|
}
|
|
|
|
|
2023-11-15 12:44:08 -06:00
|
|
|
let address = (config.address === void 0) | (config.address === "") | (config.address === "0.0.0.0") ? (config.address = "localhost") : config.address;
|
2023-12-28 21:58:31 +01:00
|
|
|
const port = process.env.MM_PORT || config.port;
|
|
|
|
mainWindow.loadURL(`${prefix}${address}:${port}`);
|
2016-03-30 12:20:46 +02:00
|
|
|
|
2016-07-30 09:42:31 -04:00
|
|
|
// Open the DevTools if run with "npm start dev"
|
2017-04-18 23:34:14 -03:00
|
|
|
if (process.argv.includes("dev")) {
|
2021-06-18 21:39:55 +02:00
|
|
|
if (process.env.JEST_WORKER_ID !== undefined) {
|
|
|
|
// if we are running with jest
|
2021-07-02 17:24:29 +02:00
|
|
|
const devtools = new BrowserWindow(electronOptions);
|
2021-06-18 21:39:55 +02:00
|
|
|
mainWindow.webContents.setDevToolsWebContents(devtools.webContents);
|
|
|
|
}
|
2016-07-30 15:58:51 +02:00
|
|
|
mainWindow.webContents.openDevTools();
|
|
|
|
}
|
2016-03-30 12:20:46 +02:00
|
|
|
|
2021-08-29 22:12:14 +02:00
|
|
|
// simulate mouse move to hide black cursor on start
|
2021-08-30 19:55:26 +02:00
|
|
|
mainWindow.webContents.on("dom-ready", (event) => {
|
|
|
|
mainWindow.webContents.sendInputEvent({ type: "mouseMove", x: 0, y: 0 });
|
2021-08-29 22:12:14 +02:00
|
|
|
});
|
|
|
|
|
2016-08-09 10:22:29 +02:00
|
|
|
// Set responders for window events.
|
2020-05-11 22:22:32 +02:00
|
|
|
mainWindow.on("closed", function () {
|
2016-03-30 12:20:46 +02:00
|
|
|
mainWindow = null;
|
|
|
|
});
|
2016-08-09 10:22:29 +02:00
|
|
|
|
|
|
|
if (config.kioskmode) {
|
2020-05-11 22:22:32 +02:00
|
|
|
mainWindow.on("blur", function () {
|
2016-08-09 10:22:29 +02:00
|
|
|
mainWindow.focus();
|
|
|
|
});
|
|
|
|
|
2020-05-11 22:22:32 +02:00
|
|
|
mainWindow.on("leave-full-screen", function () {
|
2016-08-09 10:22:29 +02:00
|
|
|
mainWindow.setFullScreen(true);
|
|
|
|
});
|
|
|
|
|
2020-05-11 22:22:32 +02:00
|
|
|
mainWindow.on("resize", function () {
|
|
|
|
setTimeout(function () {
|
2016-08-09 10:22:29 +02:00
|
|
|
mainWindow.reload();
|
|
|
|
}, 1000);
|
|
|
|
});
|
|
|
|
}
|
2022-11-07 07:42:27 +01:00
|
|
|
|
|
|
|
//remove response headers that prevent sites of being embedded into iframes if configured
|
|
|
|
mainWindow.webContents.session.webRequest.onHeadersReceived((details, callback) => {
|
|
|
|
let curHeaders = details.responseHeaders;
|
|
|
|
if (config["ignoreXOriginHeader"] || false) {
|
2023-12-25 08:17:11 +01:00
|
|
|
curHeaders = Object.fromEntries(Object.entries(curHeaders).filter((header) => !(/x-frame-options/i).test(header[0])));
|
2022-11-07 07:42:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (config["ignoreContentSecurityPolicy"] || false) {
|
2023-12-25 08:17:11 +01:00
|
|
|
curHeaders = Object.fromEntries(Object.entries(curHeaders).filter((header) => !(/content-security-policy/i).test(header[0])));
|
2022-11-07 07:42:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
callback({ responseHeaders: curHeaders });
|
|
|
|
});
|
2023-01-26 22:16:50 +01:00
|
|
|
|
|
|
|
mainWindow.once("ready-to-show", () => {
|
|
|
|
mainWindow.show();
|
|
|
|
});
|
2016-03-27 20:40:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Quit when all windows are closed.
|
2020-05-11 22:22:32 +02:00
|
|
|
app.on("window-all-closed", function () {
|
2021-07-04 17:46:22 +02:00
|
|
|
if (process.env.JEST_WORKER_ID !== undefined) {
|
|
|
|
// if we are running with jest
|
|
|
|
app.quit();
|
|
|
|
} else {
|
|
|
|
createWindow();
|
2021-07-04 19:21:17 +02:00
|
|
|
}
|
2016-03-27 20:40:07 +02:00
|
|
|
});
|
|
|
|
|
2020-05-11 22:22:32 +02:00
|
|
|
app.on("activate", function () {
|
2016-03-30 12:20:46 +02:00
|
|
|
// 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 === null) {
|
|
|
|
createWindow();
|
|
|
|
}
|
2016-03-27 20:40:07 +02:00
|
|
|
});
|
2016-04-08 22:16:22 +02:00
|
|
|
|
2017-10-13 16:43:11 -05:00
|
|
|
/* This method will be called when SIGINT is received and will call
|
|
|
|
* each node_helper's stop function if it exists. Added to fix #1056
|
|
|
|
*
|
|
|
|
* Note: this is only used if running Electron. Otherwise
|
|
|
|
* core.stop() is called by process.on("SIGINT"... in `app.js`
|
|
|
|
*/
|
2023-02-22 18:58:00 +01:00
|
|
|
app.on("before-quit", async (event) => {
|
2020-05-11 07:25:42 +02:00
|
|
|
Log.log("Shutting down server...");
|
2017-10-13 16:43:11 -05:00
|
|
|
event.preventDefault();
|
2020-05-11 22:22:32 +02:00
|
|
|
setTimeout(() => {
|
|
|
|
process.exit(0);
|
|
|
|
}, 3000); // Force-quit after 3 seconds.
|
2023-02-22 18:58:00 +01:00
|
|
|
await core.stop();
|
2017-10-13 16:43:11 -05:00
|
|
|
process.exit(0);
|
|
|
|
});
|
|
|
|
|
2023-02-22 18:58:00 +01:00
|
|
|
/**
|
|
|
|
* Handle errors from self-signed certificates
|
|
|
|
*/
|
2021-12-26 17:43:27 +01:00
|
|
|
app.on("certificate-error", (event, webContents, url, error, certificate, callback) => {
|
|
|
|
event.preventDefault();
|
|
|
|
callback(true);
|
|
|
|
});
|
|
|
|
|
2023-07-16 16:04:58 -05:00
|
|
|
if (process.env.clientonly) {
|
|
|
|
app.whenReady().then(() => {
|
|
|
|
Log.log("Launching client viewer application.");
|
|
|
|
createWindow();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-06-25 12:18:59 +02:00
|
|
|
// Start the core application if server is run on localhost
|
2016-04-08 22:16:22 +02:00
|
|
|
// This starts all node helpers and starts the webserver.
|
2021-01-05 18:48:55 +01:00
|
|
|
if (["localhost", "127.0.0.1", "::1", "::ffff:127.0.0.1", undefined].includes(config.address)) {
|
2023-04-12 08:25:07 +02:00
|
|
|
core.start().then((c) => {
|
|
|
|
config = c;
|
|
|
|
app.whenReady().then(() => {
|
|
|
|
Log.log("Launching application.");
|
|
|
|
createWindow();
|
|
|
|
});
|
|
|
|
});
|
2017-07-22 15:40:35 +02:00
|
|
|
}
|