diff --git a/config/defaults.js b/config/defaults.js index 0035f0b2..d01d059a 100644 --- a/config/defaults.js +++ b/config/defaults.js @@ -46,6 +46,7 @@ const defaultConfig = { }, discord: { enabled: false, + autoReconnect: true, // if enabled, will try to reconnect to discord every 5 seconds after disconnecting or failing to connect activityTimoutEnabled: true, // if enabled, the discord rich presence gets cleared when music paused after the time specified below activityTimoutTime: 10 * 60 * 1000, // 10 minutes listenAlong: true, // add a "listen along" button to rich presence diff --git a/index.js b/index.js index da6947cc..6de4c18b 100644 --- a/index.js +++ b/index.js @@ -492,13 +492,12 @@ function removeContentSecurityPolicy( // Custom listener to tweak the content security policy session.webRequest.onHeadersReceived(function (details, callback) { - if ( - !details.responseHeaders["content-security-policy-report-only"] && - !details.responseHeaders["content-security-policy"] - ) - return callback({ cancel: false }); + details.responseHeaders ??= {} + + // Remove the content security policy delete details.responseHeaders["content-security-policy-report-only"]; delete details.responseHeaders["content-security-policy"]; + callback({ cancel: false, responseHeaders: details.responseHeaders }); }); diff --git a/package.json b/package.json index 9dcdc15d..8f52197d 100644 --- a/package.json +++ b/package.json @@ -35,8 +35,20 @@ "!plugins/touchbar${/*}" ], "target": [ - "nsis", - "portable" + { + "target": "nsis", + "arch": [ + "x64", + "arm64" + ] + }, + { + "target": "portable", + "arch": [ + "x64", + "arm64" + ] + } ] }, "nsis": { @@ -91,13 +103,14 @@ }, "engines": { "node": ">=16.0.0", - "npm": "Please use yarn and not npm" + "npm": "Please use yarn instead" }, "dependencies": { "@cliqz/adblocker-electron": "^1.25.2", "@ffmpeg/core": "^0.11.0", "@ffmpeg/ffmpeg": "^0.11.6", "@foobar404/wave": "^2.0.4", + "@xhayper/discord-rpc": "^1.0.15", "async-mutex": "^0.4.0", "browser-id3-writer": "^4.4.0", "butterchurn": "^2.6.7", @@ -105,7 +118,6 @@ "chokidar": "^3.5.3", "custom-electron-prompt": "^1.5.1", "custom-electron-titlebar": "^4.1.5", - "discord-rpc": "^4.0.1", "electron-better-web-request": "^1.0.1", "electron-debug": "^3.2.0", "electron-is": "^3.0.0", @@ -116,6 +128,8 @@ "filenamify": "^4.3.0", "howler": "^2.2.3", "html-to-text": "^9.0.3", + "keyboardevent-from-electron-accelerator": "^2.0.0", + "keyboardevents-areequal": "^0.2.2", "md5": "^2.3.0", "mpris-service": "^2.1.2", "node-fetch": "^2.6.8", @@ -137,11 +151,6 @@ "playwright": "^1.29.2", "xo": "^0.53.1" }, - "resolutions": { - "glob-parent": "5.1.2", - "minimist": "1.2.7", - "yargs-parser": "21.1.1" - }, "auto-changelog": { "hideCredit": true, "package": true, diff --git a/plugins/adblocker/preload.js b/plugins/adblocker/preload.js index 621c9fae..2b5a12df 100644 --- a/plugins/adblocker/preload.js +++ b/plugins/adblocker/preload.js @@ -1,4 +1,4 @@ module.exports = () => { // Preload adblocker to inject scripts/styles - require("@cliqz/adblocker-electron-preload/dist/preload.cjs"); + require("@cliqz/adblocker-electron-preload"); }; diff --git a/plugins/crossfade/front.js b/plugins/crossfade/front.js index 433f5c19..f08d7d87 100644 --- a/plugins/crossfade/front.js +++ b/plugins/crossfade/front.js @@ -6,7 +6,7 @@ require("./fader"); let transitionAudio; // Howler audio used to fade out the current music let firstVideo = true; -let transitioning = false; +let waitForTransition; // Crossfade options that can be overridden in plugin options let crossfadeOptions = { @@ -109,10 +109,9 @@ const syncVideoWithTransitionAudio = async () => { const onApiLoaded = () => { watchVideoIDChanges(async (videoID) => { - if (!transitioning) { - const url = await getStreamURL(videoID); - await createAudioForCrossfade(url); - } + await waitForTransition; + const url = await getStreamURL(videoID); + await createAudioForCrossfade(url); }); }; @@ -121,7 +120,11 @@ const crossfade = (cb) => { cb(); return; } - transitioning = true; + + let resolveTransition; + waitForTransition = new Promise(function (resolve, reject) { + resolveTransition = resolve; + }); const video = document.querySelector("video"); @@ -134,7 +137,7 @@ const crossfade = (cb) => { // Fade out the music video.volume = 0; fader.fadeOut(() => { - transitioning = false; + resolveTransition(); cb(); }); }; diff --git a/plugins/discord/back.js b/plugins/discord/back.js index d1ff3a01..a8a75986 100644 --- a/plugins/discord/back.js +++ b/plugins/discord/back.js @@ -1,4 +1,5 @@ -const Discord = require("discord-rpc"); +"use strict"; +const Discord = require("@xhayper/discord-rpc"); const { dev } = require("electron-is"); const { dialog, app } = require("electron"); @@ -9,58 +10,81 @@ const clientId = "1043858434585526382"; /** * @typedef {Object} Info - * @property {import('discord-rpc').Client} rpc + * @property {import('@xhayper/discord-rpc').Client} rpc * @property {boolean} ready + * @property {boolean} autoReconnect * @property {import('../../providers/song-info').SongInfo} lastSongInfo */ /** * @type {Info} */ const info = { - rpc: null, + rpc: new Discord.Client({ + clientId + }), ready: false, + autoReconnect: true, lastSongInfo: null, }; + /** * @type {(() => void)[]} */ const refreshCallbacks = []; + const resetInfo = () => { - info.rpc = null; info.ready = false; clearTimeout(clearActivity); if (dev()) console.log("discord disconnected"); refreshCallbacks.forEach(cb => cb()); }; +info.rpc.on("connected", () => { + if (dev()) console.log("discord connected"); + refreshCallbacks.forEach(cb => cb()); +}); + +info.rpc.on("ready", () => { + info.ready = true; + if (info.lastSongInfo) updateActivity(info.lastSongInfo) +}); + +info.rpc.on("disconnected", () => { + resetInfo(); + + if (info.autoReconnect) { + connectTimeout(); + } +}); + +const connectTimeout = () => new Promise((resolve, reject) => setTimeout(() => { + if (!info.autoReconnect || info.rpc.isConnected) return; + info.rpc.login().then(resolve).catch(reject); +}, 5000)); + +const connectRecursive = () => { + if (!info.autoReconnect || info.rpc.isConnected) return; + connectTimeout().catch(connectRecursive); +} + let window; const connect = (showErr = false) => { - if (info.rpc) { + if (info.rpc.isConnected) { if (dev()) - console.log('Attempted to connect with active RPC object'); + console.log('Attempted to connect with active connection'); return; } - info.rpc = new Discord.Client({ - transport: "ipc", - }); info.ready = false; - info.rpc.once("connected", () => { - if (dev()) console.log("discord connected"); - refreshCallbacks.forEach(cb => cb()); - }); - info.rpc.once("ready", () => { - info.ready = true; - if (info.lastSongInfo) updateActivity(info.lastSongInfo) - }); - info.rpc.once("disconnected", resetInfo); - // Startup the rpc client info.rpc.login({ clientId }).catch(err => { resetInfo(); if (dev()) console.error(err); - if (showErr) dialog.showMessageBox(window, { title: 'Connection failed', message: err.message || String(err), type: 'error' }); + if (info.autoReconnect) { + connectRecursive(); + } + else if (showErr) dialog.showMessageBox(window, { title: 'Connection failed', message: err.message || String(err), type: 'error' }); }); }; @@ -70,7 +94,9 @@ let clearActivity; */ let updateActivity; -module.exports = (win, { activityTimoutEnabled, activityTimoutTime, listenAlong, hideDurationLeft }) => { +module.exports = (win, { autoReconnect, activityTimoutEnabled, activityTimoutTime, listenAlong, hideDurationLeft }) => { + info.autoReconnect = autoReconnect; + window = win; // We get multiple events // Next song: PAUSE(n), PAUSE(n+1), PLAY(n+1) @@ -92,7 +118,7 @@ module.exports = (win, { activityTimoutEnabled, activityTimoutTime, listenAlong, // clear directly if timeout is 0 if (songInfo.isPaused && activityTimoutEnabled && activityTimoutTime === 0) { - info.rpc.clearActivity().catch(console.error); + info.rpc.user?.clearActivity().catch(console.error); return; } @@ -100,7 +126,6 @@ module.exports = (win, { activityTimoutEnabled, activityTimoutTime, listenAlong, // @see https://discord.com/developers/docs/topics/gateway#activity-object // not all options are transfered through https://github.com/discordjs/RPC/blob/6f83d8d812c87cb7ae22064acd132600407d7d05/src/client.js#L518-530 const activityInfo = { - type: 2, // Listening, addressed in https://github.com/discordjs/RPC/pull/149 details: songInfo.title, state: songInfo.artist, largeImageKey: songInfo.imageSrc, @@ -116,7 +141,7 @@ module.exports = (win, { activityTimoutEnabled, activityTimoutTime, listenAlong, activityInfo.smallImageText = "Paused"; // Set start the timer so the activity gets cleared after a while if enabled if (activityTimoutEnabled) - clearActivity = setTimeout(() => info.rpc.clearActivity().catch(console.error), activityTimoutTime ?? 10000); + clearActivity = setTimeout(() => info.rpc.user?.clearActivity().catch(console.error), activityTimoutTime ?? 10000); } else if (!hideDurationLeft) { // Add the start and end time of the song const songStartTime = Date.now() - songInfo.elapsedSeconds * 1000; @@ -125,7 +150,7 @@ module.exports = (win, { activityTimoutEnabled, activityTimoutTime, listenAlong, songStartTime + songInfo.songDuration * 1000; } - info.rpc.setActivity(activityInfo).catch(console.error); + info.rpc.user?.setActivity(activityInfo).catch(console.error); }; // If the page is ready, register the callback @@ -137,9 +162,10 @@ module.exports = (win, { activityTimoutEnabled, activityTimoutTime, listenAlong, }; module.exports.clear = () => { - if (info.rpc) info.rpc.clearActivity(); + if (info.rpc) info.rpc.user?.clearActivity(); clearTimeout(clearActivity); }; + module.exports.connect = connect; module.exports.registerRefresh = (cb) => refreshCallbacks.push(cb); module.exports.isConnected = () => info.rpc !== null; diff --git a/plugins/discord/menu.js b/plugins/discord/menu.js index 5ce72704..2853a502 100644 --- a/plugins/discord/menu.js +++ b/plugins/discord/menu.js @@ -18,6 +18,15 @@ module.exports = (win, options, refreshMenu) => { enabled: !isConnected(), click: connect, }, + { + label: "Auto reconnect", + type: "checkbox", + checked: options.autoReconnect, + click: (item) => { + options.autoReconnect = item.checked; + setMenuOptions('discord', options); + }, + }, { label: "Clear activity", click: clear, diff --git a/plugins/downloader/youtube-dl.js b/plugins/downloader/youtube-dl.js index 443059fc..8a7b0bf1 100644 --- a/plugins/downloader/youtube-dl.js +++ b/plugins/downloader/youtube-dl.js @@ -82,7 +82,7 @@ const downloadVideoToMP3 = async ( sendFeedback("Download: " + progress + "%", ratio); }) .on("info", (info, format) => { - videoName = info.videoDetails.title.replace("|", "").toString("ascii"); + videoName = info.videoDetails.title.replaceAll("|", "").toString("ascii"); if (is.dev()) { console.log( "Downloading video - name:", diff --git a/plugins/in-app-menu/front.js b/plugins/in-app-menu/front.js index da5841f9..17c89d4e 100644 --- a/plugins/in-app-menu/front.js +++ b/plugins/in-app-menu/front.js @@ -43,9 +43,18 @@ module.exports = (options) => { setNavbarMargin(); const playPageObserver = new MutationObserver(setNavbarMargin); playPageObserver.observe($('ytmusic-app-layout'), { attributeFilter: ['player-page-open_', 'playerPageOpen_'] }) + setupSearchOpenObserver(); }, { once: true, passive: true }) }; +function setupSearchOpenObserver() { + const searchOpenObserver = new MutationObserver(mutations => { + $('#nav-bar-background').style.webkitAppRegion = + mutations[0].target.opened ? 'no-drag' : 'drag'; + }); + searchOpenObserver.observe($('ytmusic-search-box'), { attributeFilter: ["opened"] }) +} + function setNavbarMargin() { $('#nav-bar-background').style.right = $('ytmusic-app-layout').playerPageOpen_ ? diff --git a/plugins/in-app-menu/style.css b/plugins/in-app-menu/style.css index 946b3e4a..c27b2d57 100644 --- a/plugins/in-app-menu/style.css +++ b/plugins/in-app-menu/style.css @@ -79,3 +79,15 @@ yt-page-navigation-progress, .cet-menubar-menu-container .cet-action-item { background-color: inherit } + +#nav-bar-background { + -webkit-app-region: drag; +} + +ytmusic-nav-bar input, +ytmusic-nav-bar span, +ytmusic-nav-bar [role="button"], +ytmusic-nav-bar yt-icon, +tp-yt-iron-dropdown { + -webkit-app-region: no-drag; +} diff --git a/plugins/picture-in-picture/adaptors/in-app-menu.js b/plugins/picture-in-picture/adaptors/in-app-menu.js deleted file mode 100644 index a96ae967..00000000 --- a/plugins/picture-in-picture/adaptors/in-app-menu.js +++ /dev/null @@ -1,37 +0,0 @@ -const { Menu, app } = require("electron"); -const { setApplicationMenu } = require("../../../menu"); - -module.exports = (win, options, setOptions, togglePip, isInPip) => { - if (isInPip) { - Menu.setApplicationMenu(Menu.buildFromTemplate([ - { - label: "App", - submenu: [ - { - label: "Exit Picture in Picture", - click: togglePip, - }, - { - label: "Always on top", - type: "checkbox", - checked: options.alwaysOnTop, - click: (item) => { - setOptions({ alwaysOnTop: item.checked }); - win.setAlwaysOnTop(item.checked); - }, - }, - { - label: "Restart", - click: () => { - app.relaunch(); - app.quit(); - }, - }, - { role: "quit" }, - ], - }, - ])); - } else { - setApplicationMenu(win); - } -}; diff --git a/plugins/picture-in-picture/back.js b/plugins/picture-in-picture/back.js index d71acd49..0bfdfa36 100644 --- a/plugins/picture-in-picture/back.js +++ b/plugins/picture-in-picture/back.js @@ -3,7 +3,7 @@ const path = require("path"); const { app, ipcMain } = require("electron"); const electronLocalshortcut = require("electron-localshortcut"); -const { setOptions, isEnabled } = require("../../config/plugins"); +const { setOptions } = require("../../config/plugins"); const { injectCSS } = require("../utils"); let isInPiP = false; @@ -23,15 +23,6 @@ const setLocalOptions = (_options) => { setOptions("picture-in-picture", _options); } - -const adaptors = []; -const runAdaptors = () => adaptors.forEach(a => a()); - -if (isEnabled("in-app-menu")) { - let adaptor = require("./adaptors/in-app-menu"); - adaptors.push(() => adaptor(win, options, setLocalOptions, togglePiP, isInPiP)); -} - const togglePiP = async () => { isInPiP = !isInPiP; setLocalOptions({ isInPiP }); @@ -50,7 +41,6 @@ const togglePiP = async () => { win.setMaximizable(false); win.setFullScreenable(false); - runAdaptors(); win.webContents.send("pip-toggle", true); app.dock?.hide(); @@ -66,7 +56,6 @@ const togglePiP = async () => { win.setMaximizable(true); win.setFullScreenable(true); - runAdaptors(); win.webContents.send("pip-toggle", false); win.setVisibleOnAllWorkspaces(false); @@ -103,9 +92,6 @@ module.exports = (_win, _options) => { ipcMain.on("picture-in-picture", async () => { await togglePiP(); }); - if (options.hotkey) { - electronLocalshortcut.register(win, options.hotkey, togglePiP); - } }; module.exports.setOptions = setLocalOptions; diff --git a/plugins/picture-in-picture/front.js b/plugins/picture-in-picture/front.js index 788a908e..941a4333 100644 --- a/plugins/picture-in-picture/front.js +++ b/plugins/picture-in-picture/front.js @@ -1,10 +1,14 @@ const { ipcRenderer } = require("electron"); +const { toKeyEvent } = require("keyboardevent-from-electron-accelerator"); +const keyEventAreEqual = require("keyboardevents-areequal"); + const { getSongMenu } = require("../../providers/dom-elements"); const { ElementFromFile, templatePath } = require("../utils"); function $(selector) { return document.querySelector(selector); } +let useNativePiP = false; let menu = null; const pipButton = ElementFromFile( templatePath(__dirname, "picture-in-picture.html") @@ -39,8 +43,24 @@ const observer = new MutationObserver(() => { menu.prepend(pipButton); }); -global.togglePictureInPicture = () => { +global.togglePictureInPicture = async () => { + if (useNativePiP) { + const isInPiP = document.pictureInPictureElement !== null; + const video = $("video"); + const togglePiP = () => + isInPiP + ? document.exitPictureInPicture.call(document) + : video.requestPictureInPicture.call(video); + + try { + await togglePiP(); + $("#icon").click(); // Close the menu + return true; + } catch {} + } + ipcRenderer.send("picture-in-picture"); + return false; }; const listenForToggle = () => { @@ -54,9 +74,12 @@ const listenForToggle = () => { const player = $('#player'); const onPlayerDblClick = player.onDoubleClick_; - ipcRenderer.on('pip-toggle', (_, isPip) => { + const titlebar = $(".cet-titlebar"); + + ipcRenderer.on("pip-toggle", (_, isPip) => { if (isPip) { - replaceButton(".exit-fullscreen-button", originalExitButton).onclick = () => togglePictureInPicture(); + replaceButton(".exit-fullscreen-button", originalExitButton).onclick = + () => togglePictureInPicture(); player.onDoubleClick_ = () => {}; expandMenu.onmouseleave = () => middleControls.click(); if (!playerPage.playerPageOpen_) { @@ -64,30 +87,33 @@ const listenForToggle = () => { } fullScreenButton.click(); appLayout.classList.add("pip"); + if (titlebar) titlebar.style.display = "none"; } else { $(".exit-fullscreen-button").replaceWith(originalExitButton); player.onDoubleClick_ = onPlayerDblClick; expandMenu.onmouseleave = undefined; originalExitButton.click(); appLayout.classList.remove("pip"); + if (titlebar) titlebar.style.display = "flex"; } }); } function observeMenu(options) { + useNativePiP = options.useNativePiP; document.addEventListener( "apiLoaded", () => { listenForToggle(); - // remove native listeners - cloneButton(".player-minimize-button").onclick = () => { - global.togglePictureInPicture(); - setTimeout(() => $('#player').click()); + + cloneButton(".player-minimize-button").onclick = async () => { + await global.togglePictureInPicture(); + setTimeout(() => $("#player").click()); }; // allows easily closing the menu by programmatically clicking outside of it $("#expanding-menu").removeAttribute("no-cancel-on-outside-click"); - // TODO: think about wether an additional button in songMenu is needed + // TODO: think about wether an additional button in songMenu is needed observer.observe($("ytmusic-popup-container"), { childList: true, subtree: true, @@ -97,4 +123,18 @@ function observeMenu(options) { ); } -module.exports = observeMenu; +module.exports = (options) => { + observeMenu(options); + + if (options.hotkey) { + const hotkeyEvent = toKeyEvent(options.hotkey); + window.addEventListener("keydown", (event) => { + if ( + keyEventAreEqual(event, hotkeyEvent) && + !$("ytmusic-search-box").opened + ) { + togglePictureInPicture(); + } + }); + } +}; diff --git a/plugins/picture-in-picture/menu.js b/plugins/picture-in-picture/menu.js index bdbc1cee..61014c22 100644 --- a/plugins/picture-in-picture/menu.js +++ b/plugins/picture-in-picture/menu.js @@ -45,16 +45,24 @@ module.exports = (win, options) => [ }], ...promptOptions() }, win) - + if (output) { const { value, accelerator } = output[0]; setOptions({ [value]: accelerator }); - + item.checked = !!accelerator; } else { // Reset checkbox if prompt was canceled item.checked = !item.checked; } }, + }, + { + label: "Use native PiP", + type: "checkbox", + checked: options.useNativePiP, + click: (item) => { + setOptions({ useNativePiP: item.checked }); + }, } ]; diff --git a/plugins/picture-in-picture/style.css b/plugins/picture-in-picture/style.css index 1856e01b..a7430739 100644 --- a/plugins/picture-in-picture/style.css +++ b/plugins/picture-in-picture/style.css @@ -3,9 +3,9 @@ ytmusic-app-layout.pip ytmusic-player-bar svg, ytmusic-app-layout.pip ytmusic-player-bar .time-info, ytmusic-app-layout.pip ytmusic-player-bar yt-formatted-string, ytmusic-app-layout.pip ytmusic-player-bar .yt-formatted-string { - filter: drop-shadow(2px 4px 6px black); - color: white !important; - fill: white !important; + filter: drop-shadow(2px 4px 6px black); + color: white !important; + fill: white !important; } /* improve the style of the player bar expanding menu */ @@ -20,6 +20,23 @@ ytmusic-app-layout.pip ytmusic-player-expanding-menu { top: 22px !important; } +/* make player-bar not draggable if in-app-menu is enabled */ +.cet-container ytmusic-app-layout.pip ytmusic-player-bar { + -webkit-app-region: no-drag !important; +} + +/* make player draggable if in-app-menu is enabled */ +.cet-container ytmusic-app-layout.pip #player { + -webkit-app-region: drag !important; +} + +/* remove info, thumbnail and menu from player-bar */ +ytmusic-app-layout.pip ytmusic-player-bar .content-info-wrapper, +ytmusic-app-layout.pip ytmusic-player-bar .thumbnail-image-wrapper, +ytmusic-app-layout.pip ytmusic-player-bar ytmusic-menu-renderer { + display: none !important; +} + /* disable the video-toggle button when in PiP mode */ ytmusic-app-layout.pip .video-switch-button { display: none !important; diff --git a/yarn.lock b/yarn.lock index de7c4175..5107917c 100644 --- a/yarn.lock +++ b/yarn.lock @@ -1190,6 +1190,17 @@ __metadata: languageName: node linkType: hard +"@xhayper/discord-rpc@npm:^1.0.15": + version: 1.0.15 + resolution: "@xhayper/discord-rpc@npm:1.0.15" + dependencies: + axios: ^1.2.1 + discord-api-types: ^0.37.24 + ws: ^8.11.0 + checksum: a592f20e524ab6ef9b58f5125cd5b30cfd040e5f74f78f935afd88c3bd251523dcf5f599586f5bb24cf7e157fe370711d6109dbfb47fc312bd8890cef6b6d4ea + languageName: node + linkType: hard + "abbrev@npm:^1.0.0": version: 1.1.1 resolution: "abbrev@npm:1.1.1" @@ -1634,6 +1645,17 @@ __metadata: languageName: node linkType: hard +"axios@npm:^1.2.1": + version: 1.3.2 + resolution: "axios@npm:1.3.2" + dependencies: + follow-redirects: ^1.15.0 + form-data: ^4.0.0 + proxy-from-env: ^1.1.0 + checksum: 9791af75a6df137b15ef45d13ad11eb357b3860d2496347ee18778db9d0abc2320362a4452f1e070e3160f1dbcc518fcefdc9e005be097e7db39acb22cf608e5 + languageName: node + linkType: hard + "babel-runtime@npm:^6.26.0": version: 6.26.0 resolution: "babel-runtime@npm:6.26.0" @@ -1674,7 +1696,7 @@ __metadata: languageName: node linkType: hard -"bindings@npm:^1.2.1, bindings@npm:^1.3.0": +"bindings@npm:^1.2.1": version: 1.5.0 resolution: "bindings@npm:1.5.0" dependencies: @@ -2683,17 +2705,10 @@ __metadata: languageName: node linkType: hard -"discord-rpc@npm:^4.0.1": - version: 4.0.1 - resolution: "discord-rpc@npm:4.0.1" - dependencies: - node-fetch: ^2.6.1 - register-scheme: "github:devsnek/node-register-scheme" - ws: ^7.3.1 - dependenciesMeta: - register-scheme: - optional: true - checksum: f96e4e32751402c9a689b24eccfa319315995f8906bb86e167c6a94bc6466f3641b41f8a082ac5b4e92baa7cefdb3456e7c4d86ccd2d9373281c9a93b7335c56 +"discord-api-types@npm:^0.37.24": + version: 0.37.33 + resolution: "discord-api-types@npm:0.37.33" + checksum: d1e31106081ef79d3ea721b8991c5de155879e6d01c47a025b2c41e0496b551608e72e383caf23680938219099b6c3bd43cbacf48edeb7fc9a3c1730c1811ba8 languageName: node linkType: hard @@ -3978,6 +3993,16 @@ __metadata: languageName: node linkType: hard +"follow-redirects@npm:^1.15.0": + version: 1.15.2 + resolution: "follow-redirects@npm:1.15.2" + peerDependenciesMeta: + debug: + optional: true + checksum: faa66059b66358ba65c234c2f2a37fcec029dc22775f35d9ad6abac56003268baf41e55f9ee645957b32c7d9f62baf1f0b906e68267276f54ec4b4c597c2b190 + languageName: node + linkType: hard + "for-each@npm:^0.3.3": version: 0.3.3 resolution: "for-each@npm:0.3.3" @@ -4229,7 +4254,7 @@ __metadata: languageName: node linkType: hard -"glob-parent@npm:5.1.2": +"glob-parent@npm:^5.1.2, glob-parent@npm:~5.1.2": version: 5.1.2 resolution: "glob-parent@npm:5.1.2" dependencies: @@ -4238,6 +4263,15 @@ __metadata: languageName: node linkType: hard +"glob-parent@npm:^6.0.2": + version: 6.0.2 + resolution: "glob-parent@npm:6.0.2" + dependencies: + is-glob: ^4.0.3 + checksum: c13ee97978bef4f55106b71e66428eb1512e71a7466ba49025fc2aec59a5bfb0954d5abd58fc5ee6c9b076eef4e1f6d3375c2e964b88466ca390da4419a786a8 + languageName: node + linkType: hard + "glob@npm:^7.1.3, glob@npm:^7.1.4, glob@npm:^7.1.6": version: 7.2.3 resolution: "glob@npm:7.2.3" @@ -5511,7 +5545,7 @@ __metadata: languageName: node linkType: hard -"keyboardevents-areequal@npm:^0.2.1": +"keyboardevents-areequal@npm:^0.2.1, keyboardevents-areequal@npm:^0.2.2": version: 0.2.2 resolution: "keyboardevents-areequal@npm:0.2.2" checksum: 05d846f75170238bbb9ed45d13ca9c6cd3e68ea8ba6b7727971790fa55b44c3386ec8b9c321ae72dae0d0944678d0dc2b922148fe67d3f9720bf30a23999dd65 @@ -6055,10 +6089,10 @@ __metadata: languageName: node linkType: hard -"minimist@npm:1.2.7": - version: 1.2.7 - resolution: "minimist@npm:1.2.7" - checksum: 7346574a1038ca23c32e02252f603801f09384dd1d78b69a943a4e8c2c28730b80e96193882d3d3b22a063445f460e48316b29b8a25addca2d7e5e8f75478bec +"minimist@npm:^1.2.0, minimist@npm:^1.2.5, minimist@npm:^1.2.6": + version: 1.2.8 + resolution: "minimist@npm:1.2.8" + checksum: 75a6d645fb122dad29c06a7597bddea977258957ed88d7a6df59b5cd3fe4a527e253e9bbf2e783e4b73657f9098b96a5fe96ab8a113655d4109108577ecf85b0 languageName: node linkType: hard @@ -6235,7 +6269,7 @@ __metadata: languageName: node linkType: hard -"node-addon-api@npm:^1.3.0, node-addon-api@npm:^1.6.3": +"node-addon-api@npm:^1.6.3": version: 1.7.2 resolution: "node-addon-api@npm:1.7.2" dependencies: @@ -7034,6 +7068,13 @@ __metadata: languageName: node linkType: hard +"proxy-from-env@npm:^1.1.0": + version: 1.1.0 + resolution: "proxy-from-env@npm:1.1.0" + checksum: ed7fcc2ba0a33404958e34d95d18638249a68c430e30fcb6c478497d72739ba64ce9810a24f53a7d921d0c065e5b78e3822759800698167256b04659366ca4d4 + languageName: node + linkType: hard + "psl@npm:^1.1.28": version: 1.9.0 resolution: "psl@npm:1.9.0" @@ -7284,16 +7325,6 @@ __metadata: languageName: node linkType: hard -"register-scheme@github:devsnek/node-register-scheme": - version: 0.0.2 - resolution: "register-scheme@https://github.com/devsnek/node-register-scheme.git#commit=e7cc9a63a1f512565da44cb57316d9fb10750e17" - dependencies: - bindings: ^1.3.0 - node-addon-api: ^1.3.0 - checksum: e3281b9b81f5718ebc05c27988f6ef33bcd452048a414ad5862f29348d14513b988ec117cc73ef46955c40f78185aaec1430361dc1d1fe0a662eec864a2bf058 - languageName: node - linkType: hard - "request-progress@npm:^2.0.1": version: 2.0.1 resolution: "request-progress@npm:2.0.1" @@ -8705,18 +8736,18 @@ __metadata: languageName: node linkType: hard -"ws@npm:^7.3.1": - version: 7.5.9 - resolution: "ws@npm:7.5.9" +"ws@npm:^8.11.0": + version: 8.12.0 + resolution: "ws@npm:8.12.0" peerDependencies: bufferutil: ^4.0.1 - utf-8-validate: ^5.0.2 + utf-8-validate: ">=5.0.2" peerDependenciesMeta: bufferutil: optional: true utf-8-validate: optional: true - checksum: c3c100a181b731f40b7f2fddf004aa023f79d64f489706a28bc23ff88e87f6a64b3c6651fbec3a84a53960b75159574d7a7385709847a62ddb7ad6af76f49138 + checksum: 818ff3f8749c172a95a114cceb8b89cedd27e43a82d65c7ad0f7882b1e96a2ee6709e3746a903c3fa88beec0c8bae9a9fcd75f20858b32a166dfb7519316a5d7 languageName: node linkType: hard @@ -8851,13 +8882,29 @@ __metadata: languageName: node linkType: hard -"yargs-parser@npm:21.1.1": +"yargs-parser@npm:^20.2.9": + version: 20.2.9 + resolution: "yargs-parser@npm:20.2.9" + checksum: 8bb69015f2b0ff9e17b2c8e6bfe224ab463dd00ca211eece72a4cd8a906224d2703fb8a326d36fdd0e68701e201b2a60ed7cf81ce0fd9b3799f9fe7745977ae3 + languageName: node + linkType: hard + +"yargs-parser@npm:^21.1.1": version: 21.1.1 resolution: "yargs-parser@npm:21.1.1" checksum: ed2d96a616a9e3e1cc7d204c62ecc61f7aaab633dcbfab2c6df50f7f87b393993fe6640d017759fe112d0cb1e0119f2b4150a87305cc873fd90831c6a58ccf1c languageName: node linkType: hard +"yargs-parser@npm:^4.2.0": + version: 4.2.1 + resolution: "yargs-parser@npm:4.2.1" + dependencies: + camelcase: ^3.0.0 + checksum: 955f7cd7fa25ee559ea4f3a4bfc5961ae38b340b32ff58834d043b4145c03f8d6a8891b31d258d55f63fb3c81415943e13f1dd45cef9bba05d2303e25e811030 + languageName: node + linkType: hard + "yargs@npm:^17.5.1": version: 17.6.2 resolution: "yargs@npm:17.6.2" @@ -8927,6 +8974,7 @@ __metadata: "@ffmpeg/ffmpeg": ^0.11.6 "@foobar404/wave": ^2.0.4 "@playwright/test": ^1.29.2 + "@xhayper/discord-rpc": ^1.0.15 async-mutex: ^0.4.0 auto-changelog: ^2.4.0 browser-id3-writer: ^4.4.0 @@ -8936,7 +8984,6 @@ __metadata: custom-electron-prompt: ^1.5.1 custom-electron-titlebar: ^4.1.5 del-cli: ^5.0.0 - discord-rpc: ^4.0.1 electron: ^22.0.2 electron-better-web-request: ^1.0.1 electron-builder: ^23.6.0 @@ -8951,6 +8998,8 @@ __metadata: filenamify: ^4.3.0 howler: ^2.2.3 html-to-text: ^9.0.3 + keyboardevent-from-electron-accelerator: ^2.0.0 + keyboardevents-areequal: ^0.2.2 md5: ^2.3.0 mpris-service: ^2.1.2 node-fetch: ^2.6.8 diff --git a/youtube-music.css b/youtube-music.css index 9a36c00c..ee48dc3a 100644 --- a/youtube-music.css +++ b/youtube-music.css @@ -39,3 +39,8 @@ img { ytmusic-cast-button { display: none !important; } + +/* Remove useless inaccessible button on top-right corner of the video player */ +.ytp-chrome-top-buttons { + display: none !important; +}