This commit is contained in:
Araxeus
2022-04-10 19:19:20 +03:00
parent c25a6f9d2a
commit 768ec7bda7
5 changed files with 106 additions and 28 deletions

View File

@ -175,6 +175,8 @@ function createMainWindow() {
win.webContents.loadURL(urlToLoad); win.webContents.loadURL(urlToLoad);
win.on("closed", onClosed); win.on("closed", onClosed);
const setPiPOptions = (key, value) => config.plugins.setOptions("picture-in-picture", { [key]: value });
win.on("move", () => { win.on("move", () => {
if (win.isMaximized()) return; if (win.isMaximized()) return;
let position = win.getPosition(); let position = win.getPosition();
@ -183,6 +185,8 @@ function createMainWindow() {
config.plugins.getOptions("picture-in-picture")["isInPiP"]; config.plugins.getOptions("picture-in-picture")["isInPiP"];
if (!isPiPEnabled) { if (!isPiPEnabled) {
lateSave("window-position", { x: position[0], y: position[1] }); lateSave("window-position", { x: position[0], y: position[1] });
} else if(config.plugins.getOptions("picture-in-picture")["savePosition"]) {
lateSave("pip-position", position, setPiPOptions);
} }
}); });
@ -196,25 +200,29 @@ function createMainWindow() {
winWasMaximized = isMaximized; winWasMaximized = isMaximized;
config.set("window-maximized", isMaximized); config.set("window-maximized", isMaximized);
} }
if (isMaximized) return;
const isPiPEnabled = const isPiPEnabled =
config.plugins.isEnabled("picture-in-picture") && config.plugins.isEnabled("picture-in-picture") &&
config.plugins.getOptions("picture-in-picture")["isInPiP"]; config.plugins.getOptions("picture-in-picture")["isInPiP"];
if (!isMaximized && !isPiPEnabled) { if (!isPiPEnabled) {
lateSave("window-size", { lateSave("window-size", {
width: windowSize[0], width: windowSize[0],
height: windowSize[1], height: windowSize[1],
}); });
} else if(config.plugins.getOptions("picture-in-picture")["saveSize"]) {
lateSave("pip-size", windowSize, setPiPOptions);
} }
}); });
let savedTimeouts = {}; let savedTimeouts = {};
function lateSave(key, value) { function lateSave(key, value, fn = config.set) {
if (savedTimeouts[key]) clearTimeout(savedTimeouts[key]); if (savedTimeouts[key]) clearTimeout(savedTimeouts[key]);
savedTimeouts[key] = setTimeout(() => { savedTimeouts[key] = setTimeout(() => {
config.set(key, value); fn(key, value);
savedTimeouts[key] = undefined; savedTimeouts[key] = undefined;
}, 1000) }, 600);
} }
win.webContents.on("render-process-gone", (event, webContents, details) => { win.webContents.on("render-process-gone", (event, webContents, details) => {

View File

@ -5,18 +5,33 @@ const { app, ipcMain } = require("electron");
const { setOptions } = require("../../config/plugins"); const { setOptions } = require("../../config/plugins");
const { injectCSS } = require("../utils"); const { injectCSS } = require("../utils");
let isInPiPMode = false; let isInPiP = false;
let originalPosition; let originalPosition;
let originalSize; let originalSize;
let originalFullScreen;
let originalMaximized;
const pipPosition = [10, 10]; let win;
const pipSize = [450, 275]; let options;
const togglePiP = async (win) => { const pipPosition = () => (options.savePosition && options["pip-position"]) || [10, 10];
isInPiPMode = !isInPiPMode; const pipSize = () => (options.saveSize && options["pip-size"]) || [450, 275];
setOptions("picture-in-picture", { isInPiP: isInPiPMode });
if (isInPiPMode) { const setLocalOptions = (_options) => {
options = { ...options, ..._options };
setOptions("picture-in-picture", _options);
}
const togglePiP = async () => {
isInPiP = !isInPiP;
setLocalOptions({ isInPiP });
if (isInPiP) {
originalFullScreen = win.isFullScreen();
if (originalFullScreen) win.setFullScreen(false);
originalMaximized = win.isMaximized();
if (originalMaximized) win.unmaximize();
originalPosition = win.getPosition(); originalPosition = win.getPosition();
originalSize = win.getSize(); originalSize = win.getSize();
@ -26,6 +41,13 @@ const togglePiP = async (win) => {
await win.webContents.executeJavaScript( await win.webContents.executeJavaScript(
// Go fullscreen // Go fullscreen
` `
var exitButton = document.querySelector(".exit-fullscreen-button");
exitButton.replaceWith(exitButton.cloneNode(true));
document.querySelector(".exit-fullscreen-button").onclick = () => togglePictureInPicture();
var onPlayerDblClick = document.querySelector('#player').onDoubleClick_
document.querySelector('#player').onDoubleClick_ = () => {};
document.querySelector('#expanding-menu').onmouseleave = () => document.querySelector('.middle-controls').click();
if (!document.querySelector("ytmusic-player-page").playerPageOpen_) { if (!document.querySelector("ytmusic-player-page").playerPageOpen_) {
document.querySelector(".toggle-player-page-button").click(); document.querySelector(".toggle-player-page-button").click();
} }
@ -47,6 +69,9 @@ const togglePiP = async (win) => {
await win.webContents.executeJavaScript( await win.webContents.executeJavaScript(
// Exit fullscreen // Exit fullscreen
` `
document.querySelector('#player').onDoubleClick_ = onPlayerDblClick;
document.querySelector('#expanding-menu').onmouseleave = undefined;
document.querySelector(".exit-fullscreen-button").replaceWith(exitButton);
document.querySelector(".exit-fullscreen-button").click(); document.querySelector(".exit-fullscreen-button").click();
document.querySelector("ytmusic-player-bar").classList.remove("pip"); document.querySelector("ytmusic-player-bar").classList.remove("pip");
` `
@ -54,26 +79,39 @@ const togglePiP = async (win) => {
win.setVisibleOnAllWorkspaces(false); win.setVisibleOnAllWorkspaces(false);
win.setAlwaysOnTop(false); win.setAlwaysOnTop(false);
if (originalFullScreen) win.setFullScreen(true);
if (originalMaximized) win.maximize();
} }
const [x, y] = isInPiPMode ? pipPosition : originalPosition; const [x, y] = isInPiP ? pipPosition() : originalPosition;
const [w, h] = isInPiPMode ? pipSize : originalSize; const [w, h] = isInPiP ? pipSize() : originalSize;
win.setPosition(x, y); win.setPosition(x, y);
win.setSize(w, h); win.setSize(w, h);
win.setWindowButtonVisibility?.(!isInPiPMode); win.setWindowButtonVisibility?.(!isInPiP);
};
module.exports = (win) => {
injectCSS(win.webContents, path.join(__dirname, "style.css"));
ipcMain.on("picture-in-picture", async () => {
await togglePiP(win);
});
}; };
const blockShortcutsInPiP = (event, input) => { const blockShortcutsInPiP = (event, input) => {
const blockedShortcuts = ["f", "escape"]; const key = input.key.toLowerCase();
if (blockedShortcuts.includes(input.key.toLowerCase())) {
if (key === "f") {
event.preventDefault(); event.preventDefault();
} } else if (key === 'escape') {
togglePiP();
event.preventDefault();
};
}; };
module.exports = (_win, _options) => {
options ??= _options;
win ??= _win;
setLocalOptions({ isInPiP });
injectCSS(win.webContents, path.join(__dirname, "style.css"));
ipcMain.on("picture-in-picture", async () => {
await togglePiP();
});
};
module.exports.setOptions = setLocalOptions;

View File

@ -3,6 +3,8 @@ const { ipcRenderer } = require("electron");
const { getSongMenu } = require("../../providers/dom-elements"); const { getSongMenu } = require("../../providers/dom-elements");
const { ElementFromFile, templatePath } = require("../utils"); const { ElementFromFile, templatePath } = require("../utils");
function $(selector) { return document.querySelector(selector); }
let menu = null; let menu = null;
const pipButton = ElementFromFile( const pipButton = ElementFromFile(
templatePath(__dirname, "picture-in-picture.html") templatePath(__dirname, "picture-in-picture.html")
@ -14,7 +16,7 @@ const observer = new MutationObserver(() => {
if (!menu) return; if (!menu) return;
} }
if (menu.contains(pipButton)) return; if (menu.contains(pipButton)) return;
const menuUrl = document.querySelector( const menuUrl = $(
'tp-yt-paper-listbox [tabindex="0"] #navigation-endpoint' 'tp-yt-paper-listbox [tabindex="0"] #navigation-endpoint'
)?.href; )?.href;
if (menuUrl && !menuUrl.includes("watch?")) return; if (menuUrl && !menuUrl.includes("watch?")) return;
@ -30,7 +32,18 @@ function observeMenu(options) {
document.addEventListener( document.addEventListener(
"apiLoaded", "apiLoaded",
() => { () => {
observer.observe(document.querySelector("ytmusic-popup-container"), { const minButton = $(".player-minimize-button");
// remove native listeners
minButton.replaceWith(minButton.cloneNode(true));
$(".player-minimize-button").onclick = () => {
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
observer.observe($("ytmusic-popup-container"), {
childList: true, childList: true,
subtree: true, subtree: true,
}); });

View File

@ -0,0 +1,20 @@
const { setOptions } = require("./back.js");
module.exports = (_win, options) => [
{
label: "Save window position",
type: "checkbox",
checked: options.savePosition,
click: (item) => {
setOptions({ savePosition: item.checked });
},
},
{
label: "Save window size",
type: "checkbox",
checked: options.saveSize,
click: (item) => {
setOptions({ saveSize: item.checked });
},
}
];

View File

@ -18,8 +18,7 @@ function setup(event) {
$('.top-row-buttons.ytmusic-player').prepend(qualitySettingsButton); $('.top-row-buttons.ytmusic-player').prepend(qualitySettingsButton);
qualitySettingsButton.onclick = function chooseQuality() { qualitySettingsButton.onclick = function chooseQuality() {
if (api.getPlayerState() === 2) api.playVideo(); setTimeout(() => $('#player').click());
else if (api.getPlayerState() === 1) api.pauseVideo();
const currentIndex = api.getAvailableQualityLevels().indexOf(api.getPlaybackQuality()) const currentIndex = api.getAvailableQualityLevels().indexOf(api.getPlaybackQuality())