simpliplay-desktop/simpliplay/main.js
Anirudh Sevugan ef5e34331e
Some checks are pending
Build Linux / build_x64 (push) Waiting to run
Build Linux / build_arm64 (push) Waiting to run
Build macOS / build (push) Waiting to run
Build Windows / build-x64 (push) Waiting to run
Build Windows / build-arm64 (push) Waiting to run
Update main.js
2025-09-14 22:35:14 -05:00

472 lines
14 KiB
JavaScript

const { app, BrowserWindow, Menu, MenuItem, shell, dialog, globalShortcut } = require('electron');
const path = require('path');
const fs = require('fs');
const os = require('os');
const { pathToFileURL } = require("url");
const { checkForUpdate } = require('./updateChecker');
let gpuAccel = "";
let didRegisterShortcuts = false;
let version = "2.1.4.0"
if (process.platform === 'darwin') {
if (process.argv.includes('--use-gl')) {
app.commandLine.appendSwitch('disable-features', 'Metal');
app.commandLine.appendSwitch('use-gl', 'desktop');
}
}
// random change just to make sure that snapcraft releases fixed version on new channel to delete arm64 versions
let mainWindow;
if (process.argv.includes('--disable-gpu')) {
app.disableHardwareAcceleration();
gpuAccel = "disabled";
}
// Handle file opening from Finder or File Explorer
app.on('open-file', (event, filePath) => {
event.preventDefault();
openFile(filePath);
});
const openFile = (filePath) => {
app.whenReady().then(() => {
const fileURL = pathToFileURL(filePath).href;
if (mainWindow) {
if (mainWindow.webContents.isLoading()) {
mainWindow.webContents.once("did-finish-load", () => {
mainWindow.webContents.send("play-media", fileURL);
});
} else {
mainWindow.webContents.send("play-media", fileURL);
}
} else {
createWindow(() => {
mainWindow.webContents.send("play-media", fileURL);
});
}
});
};
const takeSnapshot = async () => {
if (!mainWindow) return;
try {
const image = await mainWindow.webContents.capturePage();
const png = image.toPNG();
const snapshotsDir = path.join(os.homedir(), 'simpliplay-snapshots');
fs.mkdirSync(snapshotsDir, { recursive: true });
const filePath = path.join(snapshotsDir, `snapshot-${Date.now()}.png`);
fs.writeFileSync(filePath, png);
const { response } = await dialog.showMessageBox(mainWindow, {
type: 'info',
title: 'Snapshot Saved',
message: `Snapshot saved to:\n${filePath}`,
buttons: ['OK', 'Open File'],
defaultId: 0,
});
if (response === 1) shell.openPath(filePath);
} catch (error) {
dialog.showErrorBox("Snapshot Error", `Failed to capture snapshot: ${error.message}`);
}
};
const createWindow = (onReadyCallback) => {
if (!app.isReady()) {
app.whenReady().then(() => createWindow(onReadyCallback));
return;
}
if (mainWindow) mainWindow.close();
mainWindow = new BrowserWindow({
width: 1920,
height: 1080,
webPreferences: {
preload: path.join(__dirname, "preload.js"),
contextIsolation: true,
enableRemoteModule: false,
nodeIntegration: false, // Keep this false for security
sandbox: true,
},
});
mainWindow.loadFile("index.html");
if (process.platform === 'darwin') {
if (process.argv.includes('--use-gl')) {
mainWindow.webContents.on('did-finish-load', () => {
mainWindow.webContents.executeJavaScript("navigator.userAgent").then(ua => {
console.log("User Agent:", ua);
});
mainWindow.webContents.executeJavaScript("chrome.loadTimes ? chrome.loadTimes() : {}").then(loadTimes => {
console.log("GPU Info (legacy):", loadTimes);
});
});
}
}
mainWindow.once("ready-to-show", () => {
if (gpuAccel === "disabled") {
dialog.showMessageBox(mainWindow, {
type: 'warning',
buttons: ['Ok'],
defaultId: 0,
title: 'Warning!',
message: "Disabling GPU acceleration greatly decreases performance and is not recommended, but if you're curious, I don't wanna stop you.",
});
}
if (onReadyCallback) onReadyCallback();
});
setupContextMenu();
};
// Set up context menu (prevents errors if `mainWindow` is undefined)
const setupContextMenu = () => {
if (!mainWindow) return;
const contextMenu = new Menu();
contextMenu.append(new MenuItem({ label: 'Take a Snapshot', click: takeSnapshot }));
contextMenu.append(new MenuItem({ type: 'separator' }));
contextMenu.append(new MenuItem({ label: 'Inspect', click: () => mainWindow.webContents.openDevTools() }));
mainWindow.webContents.on('context-menu', (event) => {
event.preventDefault();
contextMenu.popup({ window: mainWindow });
});
};
// Set up application menu
const setupMenu = () => {
const menu = Menu.getApplicationMenu();
if (!menu) return;
const fileMenu = menu.items.find(item => item.label === 'File');
if (fileMenu && !fileMenu.submenu.items.some(item => item.label === 'Take a Snapshot')) {
fileMenu.submenu.append(new MenuItem({ label: 'Take a Snapshot', accelerator: 'CommandOrControl+Shift+S', click: takeSnapshot }));
}
const appMenu = menu.items.find(item => item.label === 'SimpliPlay');
if (appMenu && !appMenu.submenu.items.some(item => item.label === 'Check for Updates')) {
const submenu = appMenu.submenu;
const separatorIndex = submenu.items.findIndex(item => item.type === 'separator');
const updateMenuItem = new MenuItem({
label: 'Check for Updates',
accelerator: 'CommandOrControl+Shift+U',
click: () => checkForUpdate(version)
});
if (separatorIndex === -1) {
// No separator found — just append
submenu.append(updateMenuItem);
} else {
// Insert right before the separator
submenu.insert(separatorIndex, updateMenuItem);
}
}
const helpMenu = menu.items.find(item => item.label === 'Help');
if (helpMenu) {
const addMenuItem = (label, url) => {
if (!helpMenu.submenu.items.some(item => item.label === label)) {
helpMenu.submenu.append(new MenuItem({ label, click: () => shell.openExternal(url) }));
}
};
addMenuItem('Source Code', 'https://github.com/A-Star100/simpliplay-desktop');
addMenuItem('Website', 'https://simpliplay.netlify.app');
addMenuItem('Help Center', 'https://simpliplay.netlify.app/help');
// Check for Updates
if (!helpMenu.submenu.items.some(item => item.label === 'Check for Updates')) {
helpMenu.submenu.append(
new MenuItem({
label: 'Check for Updates',
accelerator: 'CommandOrControl+Shift+U',
click: () => checkForUpdate(version)
})
);
}
if (!helpMenu.submenu.items.some(item => item.label === 'Quit')) {
helpMenu.submenu.append(new MenuItem({ type: 'separator' }));
helpMenu.submenu.append(new MenuItem({ label: 'Quit', click: () => app.quit() }));
}
}
const loadedAddons = new Map(); // key: addon filepath, value: MenuItem
const newMenuItems = menu ? [...menu.items] : [];
let addonsMenu;
// Check if Add-ons menu already exists
const existingAddonsMenuItem = newMenuItems.find(item => item.label === 'Add-ons');
if (existingAddonsMenuItem) {
addonsMenu = existingAddonsMenuItem.submenu;
} else {
addonsMenu = new Menu();
// "Load Add-on" menu item
addonsMenu.append(new MenuItem({
label: 'Load Add-on',
accelerator: 'CommandOrControl+Shift+A',
click: async () => {
const result = await dialog.showOpenDialog(mainWindow, {
title: 'Load Add-on',
filters: [{ name: 'JavaScript Files', extensions: ['simpliplay'] }],
properties: ['openFile'],
});
if (!result.canceled && result.filePaths.length > 0) {
const filePath = result.filePaths[0];
const fileName = path.basename(filePath);
const fileURL = pathToFileURL(filePath).href;
// Check if an addon with the same filename is already loaded
const alreadyLoaded = [...loadedAddons.keys()].some(
loadedPath => path.basename(loadedPath) === fileName
);
if (alreadyLoaded) {
await dialog.showMessageBox(mainWindow, {
type: 'error',
title: 'Could not load addon',
message: `An add-on named "${fileName}" has already been loaded before.`,
buttons: ['OK']
});
return;
}
if (!loadedAddons.has(filePath)) {
mainWindow.webContents.send('load-addon', fileURL);
const addonMenuItem = new MenuItem({
label: fileName,
type: 'checkbox',
checked: true,
click: (menuItem) => {
if (menuItem.checked) {
fs.access(filePath, (err) => {
if (!err) {
mainWindow.webContents.send('load-addon', fileURL);
} else {
dialog.showMessageBox(mainWindow, {
type: 'error',
title: 'Could not load addon',
message: `The add-on "${fileName}" could not be found or doesn't exist anymore.`,
buttons: ['OK']
}).then(() => {
// Delay unchecking to ensure dialog closes first
setTimeout(() => {
menuItem.checked = false;
}, 100);
});
}
});
} else {
mainWindow.webContents.send('unload-addon', fileURL);
}
}
});
if (!addonsMenu.items.some(item => item.type === 'separator')) {
addonsMenu.append(new MenuItem({ type: 'separator' }));
}
addonsMenu.append(addonMenuItem);
loadedAddons.set(filePath, addonMenuItem);
// Rebuild the menu after adding the new addon item
Menu.setApplicationMenu(Menu.buildFromTemplate(newMenuItems));
}
}
}
}));
// "About Addons" menu item (info dialog version)
addonsMenu.append(new MenuItem({
label: 'About Addons',
click: async () => {
const result = await dialog.showMessageBox(mainWindow, {
type: 'info',
buttons: ['OK'],
defaultId: 0,
title: 'About Addons',
message: 'Addons can do almost anything from adding features to the media player to adding an entire game within the app!',
detail: 'Addons are regular, client-side JavaScript files with the [.simpliplay] extension.'
});
if (result.response === 0) {
console.log('User clicked OK');
// no need for dialog.closeDialog()
}
}
}));
// "Store" menu item (info dialog version)
addonsMenu.append(new MenuItem({
label: 'Store',
click: () => {
shell.openExternal("https://simpliplay.netlify.app/addons/")
}
}));
// Add the Add-ons menu only once here:
newMenuItems.push(new MenuItem({ label: 'Add-ons', submenu: addonsMenu }));
// Set the application menu after adding Add-ons menu
Menu.setApplicationMenu(Menu.buildFromTemplate(newMenuItems));
}
// Re-apply the full menu if you add newMenuItems outside of the if above
//Menu.setApplicationMenu(Menu.buildFromTemplate(newMenuItems));
//Menu.setApplicationMenu(menu);
};
const setupShortcuts = () => {
if (didRegisterShortcuts === false) {
globalShortcut.register('CommandOrControl+Q', () => {
const focusedWindow = BrowserWindow.getFocusedWindow(); // Get the currently focused window
if (!focusedWindow) return; // Do nothing if no window is focused
dialog.showMessageBox(focusedWindow, {
type: 'question',
buttons: ['Cancel', 'Quit'],
defaultId: 1,
title: 'Quit?',
message: 'Are you sure you want to quit SimpliPlay?',
}).then(({ response }) => {
if (response === 1) app.quit();
});
});
globalShortcut.register('CommandOrControl+Shift+S', () => {
const focusedWindow = BrowserWindow.getFocusedWindow(); // Get the currently focused window
if (!focusedWindow) return; // Do nothing if no window is focused
takeSnapshot();
});
globalShortcut.register('CommandOrControl+S', () => {
const focusedWindow = BrowserWindow.getFocusedWindow(); // Get the currently focused window
if (!focusedWindow) return; // Do nothing if no window is focused
takeSnapshot();
});
// globalShortcut.register('CommandOrControl+Shift+S', takeSnapshot);
didRegisterShortcuts = true;
}
};
function unregisterShortcuts() {
didRegisterShortcuts = false;
globalShortcut.unregisterAll();
console.log("Shortcuts unregistered");
}
app.whenReady().then(() => {
createWindow();
setupMenu();
mainWindow?.on("focus", () => {
if (!didRegisterShortcuts) setupShortcuts();
});
mainWindow?.on("blur", unregisterShortcuts);
// Store but delay opening
const args = process.argv.slice(2);
const fileArg = args.find(isValidFileArg);
if (fileArg) {
app.whenReady().then(() => {
if (mainWindow) openFileSafely(fileArg);
});
}
app.on("open-file", (event, filePath) => {
event.preventDefault();
openFileSafely(filePath);
});
if (["win32", "linux"].includes(process.platform)) {
if (!app.requestSingleInstanceLock()) {
app.quit();
} else {
app.on("second-instance", (event, argv) => {
const fileArg = argv.find(isValidFileArg);
if (fileArg) openFileSafely(fileArg);
});
}
}
});
let hasOpenedFile = false;
function openFileSafely(filePath) {
if (!hasOpenedFile) {
hasOpenedFile = true;
const absPath = path.resolve(filePath); // ensure absolute path
if (mainWindow?.webContents) {
const winFileURL = pathToFileURL(filePath).href; // ✅ Convert and encode file path
mainWindow.webContents.send("play-media", winFileURL);
}
setTimeout(() => (hasOpenedFile = false), 1000);
}
}
function isValidFileArg(arg) {
if (!arg || arg.startsWith('-') || arg.includes('electron')) return false;
const resolvedPath = path.resolve(arg);
if (!fs.existsSync(resolvedPath)) return false;
// Reject known executable/script extensions
const badExtensions = ['.exe', '.bat', '.cmd', '.sh', '.msi', '.com', '.vbs', '.ps1', '.jar', '.scr'];
const ext = path.extname(resolvedPath).toLowerCase();
return !badExtensions.includes(ext);
}
app.on("window-all-closed", () => {
globalShortcut.unregisterAll();
app.quit();
/* once bug fixed replace above with:
if (process.platform !== 'darwin') app.quit() */
});
app.on("will-quit", () => {
globalShortcut.unregisterAll();
});