mirror of
https://github.com/th-ch/youtube-music.git
synced 2026-01-10 10:11:46 +00:00
473 lines
15 KiB
TypeScript
473 lines
15 KiB
TypeScript
import is from 'electron-is';
|
|
import { app, BrowserWindow, clipboard, dialog, Menu } from 'electron';
|
|
import prompt from 'custom-electron-prompt';
|
|
|
|
import { restart } from './providers/app-controls';
|
|
import config from './config';
|
|
import { startingPages } from './providers/extracted-data';
|
|
import promptOptions from './providers/prompt-options';
|
|
|
|
import adblockerMenu from './plugins/adblocker/menu';
|
|
import captionsSelectorMenu from './plugins/captions-selector/menu';
|
|
import crossfadeMenu from './plugins/crossfade/menu';
|
|
import disableAutoplayMenu from './plugins/disable-autoplay/menu';
|
|
import discordMenu from './plugins/discord/menu';
|
|
import downloaderMenu from './plugins/downloader/menu';
|
|
import lyricsGeniusMenu from './plugins/lyrics-genius/menu';
|
|
import notificationsMenu from './plugins/notifications/menu';
|
|
import pictureInPictureMenu from './plugins/picture-in-picture/menu';
|
|
import preciseVolumeMenu from './plugins/precise-volume/menu';
|
|
import shortcutsMenu from './plugins/shortcuts/menu';
|
|
import videoToggleMenu from './plugins/video-toggle/menu';
|
|
import visualizerMenu from './plugins/visualizer/menu';
|
|
import { getAvailablePluginNames } from './plugins/utils';
|
|
|
|
export type MenuTemplate = Electron.MenuItemConstructorOptions[];
|
|
|
|
// True only if in-app-menu was loaded on launch
|
|
const inAppMenuActive = config.plugins.isEnabled('in-app-menu');
|
|
|
|
const betaPlugins = ['crossfade', 'lumiastream'];
|
|
|
|
const pluginMenus = {
|
|
'adblocker': adblockerMenu,
|
|
'disable-autoplay': disableAutoplayMenu,
|
|
'captions-selector': captionsSelectorMenu,
|
|
'crossfade': crossfadeMenu,
|
|
'discord': discordMenu,
|
|
'downloader': downloaderMenu,
|
|
'lyrics-genius': lyricsGeniusMenu,
|
|
'notifications': notificationsMenu,
|
|
'picture-in-picture': pictureInPictureMenu,
|
|
'precise-volume': preciseVolumeMenu,
|
|
'shortcuts': shortcutsMenu,
|
|
'video-toggle': videoToggleMenu,
|
|
'visualizer': visualizerMenu,
|
|
};
|
|
|
|
const pluginEnabledMenu = (plugin: string, label = '', hasSubmenu = false, refreshMenu: (() => void ) | undefined = undefined): Electron.MenuItemConstructorOptions => ({
|
|
label: label || plugin,
|
|
type: 'checkbox',
|
|
checked: config.plugins.isEnabled(plugin),
|
|
click(item: Electron.MenuItem) {
|
|
if (item.checked) {
|
|
config.plugins.enable(plugin);
|
|
} else {
|
|
config.plugins.disable(plugin);
|
|
}
|
|
|
|
if (hasSubmenu) {
|
|
refreshMenu?.();
|
|
}
|
|
},
|
|
});
|
|
|
|
export const mainMenuTemplate = (win: BrowserWindow): MenuTemplate => {
|
|
const refreshMenu = () => {
|
|
setApplicationMenu(win);
|
|
if (inAppMenuActive) {
|
|
win.webContents.send('refreshMenu');
|
|
}
|
|
};
|
|
|
|
return [
|
|
{
|
|
label: 'Plugins',
|
|
submenu:
|
|
getAvailablePluginNames().map((pluginName) => {
|
|
let pluginLabel = pluginName;
|
|
if (betaPlugins.includes(pluginLabel)) {
|
|
pluginLabel += ' [beta]';
|
|
}
|
|
|
|
if (Object.hasOwn(pluginMenus, pluginName)) {
|
|
const getPluginMenu = pluginMenus[pluginName as keyof typeof pluginMenus];
|
|
|
|
if (!config.plugins.isEnabled(pluginName)) {
|
|
return pluginEnabledMenu(pluginName, pluginLabel, true, refreshMenu);
|
|
}
|
|
|
|
return {
|
|
label: pluginLabel,
|
|
submenu: [
|
|
pluginEnabledMenu(pluginName, 'Enabled', true, refreshMenu),
|
|
{ type: 'separator' },
|
|
...getPluginMenu(win, config.plugins.getOptions(pluginName), refreshMenu),
|
|
],
|
|
} satisfies Electron.MenuItemConstructorOptions;
|
|
}
|
|
|
|
return pluginEnabledMenu(pluginName, pluginLabel);
|
|
}),
|
|
},
|
|
{
|
|
label: 'Options',
|
|
submenu: [
|
|
{
|
|
label: 'Auto-update',
|
|
type: 'checkbox',
|
|
checked: config.get('options.autoUpdates'),
|
|
click(item) {
|
|
config.setMenuOption('options.autoUpdates', item.checked);
|
|
},
|
|
},
|
|
{
|
|
label: 'Resume last song when app starts',
|
|
type: 'checkbox',
|
|
checked: config.get('options.resumeOnStart'),
|
|
click(item) {
|
|
config.setMenuOption('options.resumeOnStart', item.checked);
|
|
},
|
|
},
|
|
{
|
|
label: 'Starting page',
|
|
submenu: (() => {
|
|
const subMenuArray: Electron.MenuItemConstructorOptions[] = Object.keys(startingPages).map((name) => ({
|
|
label: name,
|
|
type: 'radio',
|
|
checked: config.get('options.startingPage') === name,
|
|
click() {
|
|
config.set('options.startingPage', name);
|
|
},
|
|
}));
|
|
subMenuArray.unshift({
|
|
label: 'Unset',
|
|
type: 'radio',
|
|
checked: config.get('options.startingPage') === '',
|
|
click() {
|
|
config.set('options.startingPage', '');
|
|
},
|
|
});
|
|
return subMenuArray;
|
|
})(),
|
|
},
|
|
{
|
|
label: 'Visual Tweaks',
|
|
submenu: [
|
|
{
|
|
label: 'Remove upgrade button',
|
|
type: 'checkbox',
|
|
checked: config.get('options.removeUpgradeButton'),
|
|
click(item) {
|
|
config.setMenuOption('options.removeUpgradeButton', item.checked);
|
|
},
|
|
},
|
|
{
|
|
label: 'Like buttons',
|
|
submenu: [
|
|
{
|
|
label: 'Default',
|
|
type: 'radio',
|
|
checked: !config.get('options.likeButtons'),
|
|
click() {
|
|
config.set('options.likeButtons', '');
|
|
},
|
|
},
|
|
{
|
|
label: 'Force show',
|
|
type: 'radio',
|
|
checked: config.get('options.likeButtons') === 'force',
|
|
click() {
|
|
config.set('options.likeButtons', 'force');
|
|
},
|
|
},
|
|
{
|
|
label: 'Hide',
|
|
type: 'radio',
|
|
checked: config.get('options.likeButtons') === 'hide',
|
|
click() {
|
|
config.set('options.likeButtons', 'hide');
|
|
},
|
|
},
|
|
],
|
|
},
|
|
{
|
|
label: 'Theme',
|
|
submenu: [
|
|
{
|
|
label: 'No theme',
|
|
type: 'radio',
|
|
checked: config.get('options.themes')?.length === 0, // Todo rename "themes"
|
|
click() {
|
|
config.set('options.themes', []);
|
|
},
|
|
},
|
|
{ type: 'separator' },
|
|
{
|
|
label: 'Import custom CSS file',
|
|
type: 'normal',
|
|
async click() {
|
|
const { filePaths } = await dialog.showOpenDialog({
|
|
filters: [{ name: 'CSS Files', extensions: ['css'] }],
|
|
properties: ['openFile', 'multiSelections'],
|
|
});
|
|
if (filePaths) {
|
|
config.set('options.themes', filePaths);
|
|
}
|
|
},
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
{
|
|
label: 'Single instance lock',
|
|
type: 'checkbox',
|
|
checked: true,
|
|
click(item) {
|
|
if (!item.checked && app.hasSingleInstanceLock()) {
|
|
app.releaseSingleInstanceLock();
|
|
} else if (item.checked && !app.hasSingleInstanceLock()) {
|
|
app.requestSingleInstanceLock();
|
|
}
|
|
},
|
|
},
|
|
{
|
|
label: 'Always on top',
|
|
type: 'checkbox',
|
|
checked: config.get('options.alwaysOnTop'),
|
|
click(item) {
|
|
config.setMenuOption('options.alwaysOnTop', item.checked);
|
|
win.setAlwaysOnTop(item.checked);
|
|
},
|
|
},
|
|
...(is.windows() || is.linux()
|
|
? [
|
|
{
|
|
label: 'Hide menu',
|
|
type: 'checkbox',
|
|
checked: config.get('options.hideMenu'),
|
|
click(item) {
|
|
config.setMenuOption('options.hideMenu', item.checked);
|
|
if (item.checked && !config.get('options.hideMenuWarned')) {
|
|
dialog.showMessageBox(win, {
|
|
type: 'info', title: 'Hide Menu Enabled',
|
|
message: 'Menu will be hidden on next launch, use [Alt] to show it (or backtick [`] if using in-app-menu)',
|
|
});
|
|
}
|
|
},
|
|
},
|
|
]
|
|
: []) satisfies Electron.MenuItemConstructorOptions[],
|
|
...(is.windows() || is.macOS()
|
|
? // Only works on Win/Mac
|
|
// https://www.electronjs.org/docs/api/app#appsetloginitemsettingssettings-macos-windows
|
|
[
|
|
{
|
|
label: 'Start at login',
|
|
type: 'checkbox',
|
|
checked: config.get('options.startAtLogin'),
|
|
click(item) {
|
|
config.setMenuOption('options.startAtLogin', item.checked);
|
|
},
|
|
},
|
|
]
|
|
: []) satisfies Electron.MenuItemConstructorOptions[],
|
|
{
|
|
label: 'Tray',
|
|
submenu: [
|
|
{
|
|
label: 'Disabled',
|
|
type: 'radio',
|
|
checked: !config.get('options.tray'),
|
|
click() {
|
|
config.setMenuOption('options.tray', false);
|
|
config.setMenuOption('options.appVisible', true);
|
|
},
|
|
},
|
|
{
|
|
label: 'Enabled + app visible',
|
|
type: 'radio',
|
|
checked: config.get('options.tray') && config.get('options.appVisible'),
|
|
click() {
|
|
config.setMenuOption('options.tray', true);
|
|
config.setMenuOption('options.appVisible', true);
|
|
},
|
|
},
|
|
{
|
|
label: 'Enabled + app hidden',
|
|
type: 'radio',
|
|
checked: config.get('options.tray') && !config.get('options.appVisible'),
|
|
click() {
|
|
config.setMenuOption('options.tray', true);
|
|
config.setMenuOption('options.appVisible', false);
|
|
},
|
|
},
|
|
{ type: 'separator' },
|
|
{
|
|
label: 'Play/Pause on click',
|
|
type: 'checkbox',
|
|
checked: config.get('options.trayClickPlayPause'),
|
|
click(item) {
|
|
config.setMenuOption('options.trayClickPlayPause', item.checked);
|
|
},
|
|
},
|
|
],
|
|
},
|
|
{ type: 'separator' },
|
|
{
|
|
label: 'Advanced options',
|
|
submenu: [
|
|
{
|
|
label: 'Set Proxy',
|
|
type: 'normal',
|
|
async click(item) {
|
|
await setProxy(item, win);
|
|
},
|
|
},
|
|
{
|
|
label: 'Override useragent',
|
|
type: 'checkbox',
|
|
checked: config.get('options.overrideUserAgent'),
|
|
click(item) {
|
|
config.setMenuOption('options.overrideUserAgent', item.checked);
|
|
},
|
|
},
|
|
{
|
|
label: 'Disable hardware acceleration',
|
|
type: 'checkbox',
|
|
checked: config.get('options.disableHardwareAcceleration'),
|
|
click(item) {
|
|
config.setMenuOption('options.disableHardwareAcceleration', item.checked);
|
|
},
|
|
},
|
|
{
|
|
label: 'Restart on config changes',
|
|
type: 'checkbox',
|
|
checked: config.get('options.restartOnConfigChanges'),
|
|
click(item) {
|
|
config.setMenuOption('options.restartOnConfigChanges', item.checked);
|
|
},
|
|
},
|
|
{
|
|
label: 'Reset App cache when app starts',
|
|
type: 'checkbox',
|
|
checked: config.get('options.autoResetAppCache'),
|
|
click(item) {
|
|
config.setMenuOption('options.autoResetAppCache', item.checked);
|
|
},
|
|
},
|
|
{ type: 'separator' },
|
|
is.macOS()
|
|
? {
|
|
label: 'Toggle DevTools',
|
|
// Cannot use "toggleDevTools" role in macOS
|
|
click() {
|
|
const { webContents } = win;
|
|
if (webContents.isDevToolsOpened()) {
|
|
webContents.closeDevTools();
|
|
} else {
|
|
webContents.openDevTools();
|
|
}
|
|
},
|
|
}
|
|
: { role: 'toggleDevTools' },
|
|
{
|
|
label: 'Edit config.json',
|
|
click() {
|
|
config.edit();
|
|
},
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
{
|
|
label: 'View',
|
|
submenu: [
|
|
{ role: 'reload' },
|
|
{ role: 'forceReload' },
|
|
{ type: 'separator' },
|
|
{ role: 'zoomIn', accelerator: process.platform === 'darwin' ? 'Cmd+I' : 'Ctrl+I' },
|
|
{ role: 'zoomOut', accelerator: process.platform === 'darwin' ? 'Cmd+O' : 'Ctrl+O' },
|
|
{ role: 'resetZoom' },
|
|
{ type: 'separator' },
|
|
{ role: 'togglefullscreen' },
|
|
],
|
|
},
|
|
{
|
|
label: 'Navigation',
|
|
submenu: [
|
|
{
|
|
label: 'Go back',
|
|
click() {
|
|
if (win.webContents.canGoBack()) {
|
|
win.webContents.goBack();
|
|
}
|
|
},
|
|
},
|
|
{
|
|
label: 'Go forward',
|
|
click() {
|
|
if (win.webContents.canGoForward()) {
|
|
win.webContents.goForward();
|
|
}
|
|
},
|
|
},
|
|
{
|
|
label: 'Copy current URL',
|
|
click() {
|
|
const currentURL = win.webContents.getURL();
|
|
clipboard.writeText(currentURL);
|
|
},
|
|
},
|
|
{
|
|
label: 'Restart App',
|
|
click: restart,
|
|
},
|
|
{ role: 'quit' },
|
|
],
|
|
},
|
|
];
|
|
};
|
|
export const setApplicationMenu = (win: Electron.BrowserWindow) => {
|
|
const menuTemplate: MenuTemplate = [...mainMenuTemplate(win)];
|
|
if (process.platform === 'darwin') {
|
|
const { name } = app;
|
|
menuTemplate.unshift({
|
|
label: name,
|
|
submenu: [
|
|
{ role: 'about' },
|
|
{ type: 'separator' },
|
|
{ role: 'hide' },
|
|
{ role: 'hideOthers' },
|
|
{ role: 'unhide' },
|
|
{ type: 'separator' },
|
|
{ role: 'selectAll' },
|
|
{ role: 'cut' },
|
|
{ role: 'copy' },
|
|
{ role: 'paste' },
|
|
{ type: 'separator' },
|
|
{ role: 'minimize' },
|
|
{ role: 'close' },
|
|
{ role: 'quit' },
|
|
],
|
|
});
|
|
}
|
|
|
|
const menu = Menu.buildFromTemplate(menuTemplate);
|
|
Menu.setApplicationMenu(menu);
|
|
};
|
|
|
|
async function setProxy(item: Electron.MenuItem, win: BrowserWindow) {
|
|
const output = await prompt({
|
|
title: 'Set Proxy',
|
|
label: 'Enter Proxy Address: (leave empty to disable)',
|
|
value: config.get('options.proxy'),
|
|
type: 'input',
|
|
inputAttrs: {
|
|
type: 'url',
|
|
placeholder: "Example: 'socks5://127.0.0.1:9999",
|
|
},
|
|
width: 450,
|
|
...promptOptions(),
|
|
}, win);
|
|
|
|
if (typeof output === 'string') {
|
|
config.setMenuOption('options.proxy', output);
|
|
item.checked = output !== '';
|
|
} else { // User pressed cancel
|
|
item.checked = !item.checked; // Reset checkbox
|
|
}
|
|
}
|