Skip to content

Commit 768ec7b

Browse files
committed
v2
1 parent c25a6f9 commit 768ec7b

File tree

5 files changed

+104
-26
lines changed

5 files changed

+104
-26
lines changed

index.js

+12-4
Original file line numberDiff line numberDiff line change
@@ -175,6 +175,8 @@ function createMainWindow() {
175175
win.webContents.loadURL(urlToLoad);
176176
win.on("closed", onClosed);
177177

178+
const setPiPOptions = (key, value) => config.plugins.setOptions("picture-in-picture", { [key]: value });
179+
178180
win.on("move", () => {
179181
if (win.isMaximized()) return;
180182
let position = win.getPosition();
@@ -183,6 +185,8 @@ function createMainWindow() {
183185
config.plugins.getOptions("picture-in-picture")["isInPiP"];
184186
if (!isPiPEnabled) {
185187
lateSave("window-position", { x: position[0], y: position[1] });
188+
} else if(config.plugins.getOptions("picture-in-picture")["savePosition"]) {
189+
lateSave("pip-position", position, setPiPOptions);
186190
}
187191
});
188192

@@ -196,25 +200,29 @@ function createMainWindow() {
196200
winWasMaximized = isMaximized;
197201
config.set("window-maximized", isMaximized);
198202
}
203+
if (isMaximized) return;
204+
199205
const isPiPEnabled =
200206
config.plugins.isEnabled("picture-in-picture") &&
201207
config.plugins.getOptions("picture-in-picture")["isInPiP"];
202-
if (!isMaximized && !isPiPEnabled) {
208+
if (!isPiPEnabled) {
203209
lateSave("window-size", {
204210
width: windowSize[0],
205211
height: windowSize[1],
206212
});
213+
} else if(config.plugins.getOptions("picture-in-picture")["saveSize"]) {
214+
lateSave("pip-size", windowSize, setPiPOptions);
207215
}
208216
});
209217

210218
let savedTimeouts = {};
211-
function lateSave(key, value) {
219+
function lateSave(key, value, fn = config.set) {
212220
if (savedTimeouts[key]) clearTimeout(savedTimeouts[key]);
213221

214222
savedTimeouts[key] = setTimeout(() => {
215-
config.set(key, value);
223+
fn(key, value);
216224
savedTimeouts[key] = undefined;
217-
}, 1000)
225+
}, 600);
218226
}
219227

220228
win.webContents.on("render-process-gone", (event, webContents, details) => {

plugins/picture-in-picture/back.js

+56-18
Original file line numberDiff line numberDiff line change
@@ -5,18 +5,33 @@ const { app, ipcMain } = require("electron");
55
const { setOptions } = require("../../config/plugins");
66
const { injectCSS } = require("../utils");
77

8-
let isInPiPMode = false;
8+
let isInPiP = false;
99
let originalPosition;
1010
let originalSize;
11+
let originalFullScreen;
12+
let originalMaximized;
1113

12-
const pipPosition = [10, 10];
13-
const pipSize = [450, 275];
14+
let win;
15+
let options;
1416

15-
const togglePiP = async (win) => {
16-
isInPiPMode = !isInPiPMode;
17-
setOptions("picture-in-picture", { isInPiP: isInPiPMode });
17+
const pipPosition = () => (options.savePosition && options["pip-position"]) || [10, 10];
18+
const pipSize = () => (options.saveSize && options["pip-size"]) || [450, 275];
1819

19-
if (isInPiPMode) {
20+
const setLocalOptions = (_options) => {
21+
options = { ...options, ..._options };
22+
setOptions("picture-in-picture", _options);
23+
}
24+
25+
const togglePiP = async () => {
26+
isInPiP = !isInPiP;
27+
setLocalOptions({ isInPiP });
28+
29+
if (isInPiP) {
30+
originalFullScreen = win.isFullScreen();
31+
if (originalFullScreen) win.setFullScreen(false);
32+
originalMaximized = win.isMaximized();
33+
if (originalMaximized) win.unmaximize();
34+
2035
originalPosition = win.getPosition();
2136
originalSize = win.getSize();
2237

@@ -26,6 +41,13 @@ const togglePiP = async (win) => {
2641
await win.webContents.executeJavaScript(
2742
// Go fullscreen
2843
`
44+
var exitButton = document.querySelector(".exit-fullscreen-button");
45+
exitButton.replaceWith(exitButton.cloneNode(true));
46+
document.querySelector(".exit-fullscreen-button").onclick = () => togglePictureInPicture();
47+
48+
var onPlayerDblClick = document.querySelector('#player').onDoubleClick_
49+
document.querySelector('#player').onDoubleClick_ = () => {};
50+
document.querySelector('#expanding-menu').onmouseleave = () => document.querySelector('.middle-controls').click();
2951
if (!document.querySelector("ytmusic-player-page").playerPageOpen_) {
3052
document.querySelector(".toggle-player-page-button").click();
3153
}
@@ -47,33 +69,49 @@ const togglePiP = async (win) => {
4769
await win.webContents.executeJavaScript(
4870
// Exit fullscreen
4971
`
72+
document.querySelector('#player').onDoubleClick_ = onPlayerDblClick;
73+
document.querySelector('#expanding-menu').onmouseleave = undefined;
74+
document.querySelector(".exit-fullscreen-button").replaceWith(exitButton);
5075
document.querySelector(".exit-fullscreen-button").click();
5176
document.querySelector("ytmusic-player-bar").classList.remove("pip");
5277
`
5378
);
5479

5580
win.setVisibleOnAllWorkspaces(false);
5681
win.setAlwaysOnTop(false);
82+
83+
if (originalFullScreen) win.setFullScreen(true);
84+
if (originalMaximized) win.maximize();
5785
}
5886

59-
const [x, y] = isInPiPMode ? pipPosition : originalPosition;
60-
const [w, h] = isInPiPMode ? pipSize : originalSize;
87+
const [x, y] = isInPiP ? pipPosition() : originalPosition;
88+
const [w, h] = isInPiP ? pipSize() : originalSize;
6189
win.setPosition(x, y);
6290
win.setSize(w, h);
6391

64-
win.setWindowButtonVisibility?.(!isInPiPMode);
92+
win.setWindowButtonVisibility?.(!isInPiP);
6593
};
6694

67-
module.exports = (win) => {
95+
const blockShortcutsInPiP = (event, input) => {
96+
const key = input.key.toLowerCase();
97+
98+
if (key === "f") {
99+
event.preventDefault();
100+
} else if (key === 'escape') {
101+
togglePiP();
102+
event.preventDefault();
103+
};
104+
};
105+
106+
module.exports = (_win, _options) => {
107+
options ??= _options;
108+
win ??= _win;
109+
setLocalOptions({ isInPiP });
68110
injectCSS(win.webContents, path.join(__dirname, "style.css"));
69111
ipcMain.on("picture-in-picture", async () => {
70-
await togglePiP(win);
112+
await togglePiP();
71113
});
72114
};
73115

74-
const blockShortcutsInPiP = (event, input) => {
75-
const blockedShortcuts = ["f", "escape"];
76-
if (blockedShortcuts.includes(input.key.toLowerCase())) {
77-
event.preventDefault();
78-
}
79-
};
116+
module.exports.setOptions = setLocalOptions;
117+

plugins/picture-in-picture/front.js

+15-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,8 @@ const { ipcRenderer } = require("electron");
33
const { getSongMenu } = require("../../providers/dom-elements");
44
const { ElementFromFile, templatePath } = require("../utils");
55

6+
function $(selector) { return document.querySelector(selector); }
7+
68
let menu = null;
79
const pipButton = ElementFromFile(
810
templatePath(__dirname, "picture-in-picture.html")
@@ -14,7 +16,7 @@ const observer = new MutationObserver(() => {
1416
if (!menu) return;
1517
}
1618
if (menu.contains(pipButton)) return;
17-
const menuUrl = document.querySelector(
19+
const menuUrl = $(
1820
'tp-yt-paper-listbox [tabindex="0"] #navigation-endpoint'
1921
)?.href;
2022
if (menuUrl && !menuUrl.includes("watch?")) return;
@@ -30,7 +32,18 @@ function observeMenu(options) {
3032
document.addEventListener(
3133
"apiLoaded",
3234
() => {
33-
observer.observe(document.querySelector("ytmusic-popup-container"), {
35+
const minButton = $(".player-minimize-button");
36+
// remove native listeners
37+
minButton.replaceWith(minButton.cloneNode(true));
38+
$(".player-minimize-button").onclick = () => {
39+
global.togglePictureInPicture();
40+
setTimeout(() => $('#player').click());
41+
};
42+
43+
// allows easily closing the menu by programmatically clicking outside of it
44+
$("#expanding-menu").removeAttribute("no-cancel-on-outside-click");
45+
// TODO: think about wether an additional button in songMenu is needed
46+
observer.observe($("ytmusic-popup-container"), {
3447
childList: true,
3548
subtree: true,
3649
});

plugins/picture-in-picture/menu.js

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
const { setOptions } = require("./back.js");
2+
3+
module.exports = (_win, options) => [
4+
{
5+
label: "Save window position",
6+
type: "checkbox",
7+
checked: options.savePosition,
8+
click: (item) => {
9+
setOptions({ savePosition: item.checked });
10+
},
11+
},
12+
{
13+
label: "Save window size",
14+
type: "checkbox",
15+
checked: options.saveSize,
16+
click: (item) => {
17+
setOptions({ saveSize: item.checked });
18+
},
19+
}
20+
];

plugins/quality-changer/front.js

+1-2
Original file line numberDiff line numberDiff line change
@@ -18,8 +18,7 @@ function setup(event) {
1818
$('.top-row-buttons.ytmusic-player').prepend(qualitySettingsButton);
1919

2020
qualitySettingsButton.onclick = function chooseQuality() {
21-
if (api.getPlayerState() === 2) api.playVideo();
22-
else if (api.getPlayerState() === 1) api.pauseVideo();
21+
setTimeout(() => $('#player').click());
2322

2423
const currentIndex = api.getAvailableQualityLevels().indexOf(api.getPlaybackQuality())
2524

0 commit comments

Comments
 (0)