Files
Tooloop-Kiosk-Browser/main.js
T

223 lines
6.4 KiB
JavaScript
Raw Normal View History

const { app, BrowserWindow, nativeTheme, Menu, MenuItem } = require('electron/main');
2024-05-28 16:43:25 +02:00
const path = require('node:path');
const fs = require('fs');
2024-05-28 18:27:50 +02:00
const log = require('electron-log/main');
2024-06-03 12:16:11 +02:00
const isReachable = require('is-reachable');
const { setInterval } = require('node:timers/promises');
const config = require('./js/config.js');
const { Config } = require('./js/config.js');
2024-05-28 18:27:50 +02:00
//------------------------------------------------------------------------------
// Properties
//------------------------------------------------------------------------------
2024-05-28 14:25:32 +02:00
2024-05-28 16:43:25 +02:00
let win;
2024-05-31 10:57:15 +02:00
let menu;
2024-05-28 14:25:32 +02:00
const isMac = process.platform === 'darwin';
2024-05-28 18:27:50 +02:00
//------------------------------------------------------------------------------
// Function
//------------------------------------------------------------------------------
/**
* Creates the browser window
*/
2024-06-03 12:16:11 +02:00
function createMainWindow() {
2024-05-28 16:43:25 +02:00
win = new BrowserWindow({
2024-05-28 14:25:32 +02:00
width: 1920,
height: 1080,
2024-05-28 16:43:25 +02:00
backgroundColor: '#000000',
2024-05-29 10:09:11 +02:00
icon: 'images/icon-512.png',
2024-05-29 22:29:17 +02:00
autoHideMenuBar: true,
2024-06-04 08:09:59 +02:00
kiosk: true,
2024-05-28 14:25:32 +02:00
webPreferences: {
2024-06-03 16:32:05 +02:00
preload: path.join(__dirname, 'js/preload.js'),
2024-05-28 14:25:32 +02:00
webSecurity: false,
2024-05-29 22:29:17 +02:00
disableDialogs: true
2024-05-28 14:25:32 +02:00
},
});
2024-05-29 22:29:17 +02:00
nativeTheme.themeSource = 'dark';
2024-05-29 21:38:55 +02:00
// register event callbacks
win.on("closed", function () { win = null; });
win.webContents.on("will-frame-navigate", (event) => validateDomain(event));
win.webContents.setWindowOpenHandler(({ url }) => {
// we need to manually validate the url as `loadURL`
// doesnt trigger the `will-navigate` event
let event = new Event("DummyNavigation");
event.url = url;
if (validateDomain(event)) {
win.loadURL(config.data.url);
2024-05-29 21:38:55 +02:00
}
return { action: 'deny' };
2024-05-28 14:25:32 +02:00
});
win.webContents.on("did-fail-load", (event) => log.error(event));
2024-06-03 12:16:11 +02:00
win.on("page-title-updated", (event) => event.preventDefault());
2024-05-28 14:25:32 +02:00
// Load page from config file
if (config.data && 'url' in config.data && config.data.url) {
loadUrlAsync(config.data.url);
} else {
try {
// Load file from data folder if available
fs.accessSync('/assets/data/index.html');
win.loadFile('/assets/data/index.html');
} catch {
// Load fallback page
2024-06-03 12:16:11 +02:00
win.loadFile('./html/onboarding.html');
}
2024-05-28 16:43:25 +02:00
}
2024-05-31 10:57:15 +02:00
}
2024-05-29 21:38:55 +02:00
2024-05-31 10:57:15 +02:00
/**
* Creates all available keyboard shortcuts
*/
function createMenu() {
menu = new Menu();
menu.append(new MenuItem({
label: app.name,
submenu: [
{
label: 'Toggle Fullscreen',
accelerator: isMac ? 'Cmd+F' : 'Ctrl+F',
click: () => { win.setKiosk(!win.kiosk); }
},
2024-06-03 12:16:11 +02:00
{ role: 'reload' },
{ role: 'quit' },
{
label: 'Config',
accelerator: isMac ? 'Cmd+,' : 'Ctrl+,',
click: () => {
config.showWindow(
win,
(newConfig) => { loadUrlAsync(newConfig.url); }
);
}
2024-06-03 12:16:11 +02:00
}
2024-05-31 10:57:15 +02:00
]
}));
Menu.setApplicationMenu(menu);
2024-05-28 14:25:32 +02:00
}
2024-05-31 10:57:15 +02:00
2024-05-28 18:27:50 +02:00
/**
* Validates the url of a navigatio event against the list of allowed domains in
* the config file. See https://www.electronjs.org/docs/latest/api/web-contents#event-will-frame-navigate
* @param {Event} event
* @returns `true` if the url is allowed, `false` otherwise
2024-05-28 18:27:50 +02:00
*/
function validateDomain(event) {
if (config == undefined) return true;
2024-05-29 21:38:55 +02:00
2024-05-28 18:27:50 +02:00
let url = new URL(event.url);
2024-05-29 21:38:55 +02:00
// allow local urls
if (['file:', 'file'].includes(url.protocol)) return true;
2024-06-03 16:32:05 +02:00
if ('whitelist' in config && !config.whitelist.includes(url.hostname)) {
2024-05-28 18:27:50 +02:00
event.preventDefault();
log.info("Navigation to " + event.url + " prevented");
2024-05-29 21:38:55 +02:00
return false;
2024-05-28 18:27:50 +02:00
}
2024-05-29 21:38:55 +02:00
return true;
2024-05-28 18:27:50 +02:00
}
/**
* Tests if the host of the url is reachable.
* Access is tried in a 1 second interval and the url is loaded if successfull.
* @param {string} testurl
*/
2024-05-31 17:28:43 +02:00
async function loadUrlAsync(testurl) {
try {
let url = new URL(testurl);
// If its an online url, test whether its reachable
2024-06-03 12:16:11 +02:00
if (['https:', 'http:', 'https', 'http'].includes(url.protocol)) {
reachable = await isReachable(url.href);
2024-05-31 17:28:43 +02:00
if (reachable) {
log.info("Successfull access to " + url.hostname);
log.info("Loading " + config.data.url);
win.loadURL(config.data.url);
2024-05-31 17:28:43 +02:00
} else {
log.warn("Could not reach " + url.hostname);
for await (_ of setInterval(1000)) {
2024-06-03 12:16:11 +02:00
reachable = await isReachable(url.href);
if (reachable) {
2024-05-31 17:28:43 +02:00
log.info("Successfull access to " + url.hostname);
2024-06-04 15:32:55 +02:00
log.info("Loading " + config.data.url);
win.loadURL(config.data.url);
// break the interval
break;
} else {
log.warn("Could not reach " + url.hostname);
}
}
2024-06-03 12:16:11 +02:00
2024-05-31 17:28:43 +02:00
}
}
// Load other protocols (i. e. offline) immedately
else {
win.loadURL(config.data.url);
}
} catch (err) {
log.error(err.message);
// Load fallback page
win.loadFile('./html/onboarding.html');
}
}
2024-05-28 18:27:50 +02:00
//------------------------------------------------------------------------------
2024-05-29 22:50:18 +02:00
// Init electron app
2024-05-28 18:27:50 +02:00
//------------------------------------------------------------------------------
2024-06-03 12:16:11 +02:00
// https://github.com/electron/electron/issues/17972
app.commandLine.appendSwitch('--no-sandbox');
2024-06-03 12:24:44 +02:00
app.commandLine.appendSwitch('no-sandbox');
2024-06-03 12:16:11 +02:00
2024-05-28 14:25:32 +02:00
app.whenReady().then(() => {
config.load();
2024-05-28 18:27:50 +02:00
log.initialize();
if (config.data != undefined && 'logPath' in config.data) {
log.transports.file.resolvePathFn = () => config.data.logPath;
2024-05-28 18:27:50 +02:00
}
2024-05-31 10:57:15 +02:00
log.info('----------------------------------------------');
log.info('Starting Tooloop Kiosk Browser...');
createMenu();
2024-06-03 12:16:11 +02:00
createMainWindow();
2024-05-28 14:25:32 +02:00
});
2024-05-31 10:57:15 +02:00
app.on('activate', () => {
if (BrowserWindow.getAllWindows().length === 0) {
2024-06-03 12:16:11 +02:00
createMainWindow();
2024-05-31 10:57:15 +02:00
}
})
2024-05-29 22:50:18 +02:00
2024-05-28 14:25:32 +02:00
app.on('window-all-closed', () => {
2024-05-28 16:43:25 +02:00
app.quit()
2024-05-28 18:27:50 +02:00
});
2024-05-29 22:50:18 +02:00
2024-05-31 10:57:15 +02:00
app.on('quit', () => {
log.info('Quit');
2024-05-31 17:28:43 +02:00
});
// prevent error dialogs in case of exceptions
process.on('uncaughtException', function (error) {
log.error(error);
2024-05-31 10:57:15 +02:00
});