File indexing completed on 2025-10-19 05:18:40

0001 /*
0002     Copyright (C) 2017 Kai Uwe Broulik <kde@privat.broulik.de>
0003     Copyright (C) 2018 David Edmundson <davidedmundson@kde.org>
0004 
0005     This program is free software; you can redistribute it and/or
0006     modify it under the terms of the GNU General Public License as
0007     published by the Free Software Foundation; either version 3 of
0008     the License, or (at your option) any later version.
0009 
0010     This program is distributed in the hope that it will be useful,
0011     but WITHOUT ANY WARRANTY; without even the implied warranty of
0012     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
0013     GNU General Public License for more details.
0014 
0015     You should have received a copy of the GNU General Public License
0016     along with this program.  If not, see <http://www.gnu.org/licenses/>.
0017  */
0018 
0019 var callbacks = {};
0020 
0021 function addCallback(subsystem, action, callback)
0022 {
0023     if (!callbacks[subsystem]) {
0024         callbacks[subsystem] = {};
0025     }
0026     callbacks[subsystem][action] = callback;
0027 }
0028 
0029 function initPageScript(cb) {
0030     // On reloads, unload the previous page-script.
0031     executePageAction({"action": "unload"});
0032 
0033     // The script is only run later, wait for that before sending events.
0034     window.addEventListener("pbiInited", cb, {"once": true});
0035 
0036     var element = document.createElement('script');
0037     element.src = chrome.runtime.getURL("page-script.js");
0038     (document.body || document.head || document.documentElement).prepend(element);
0039     // We need to remove the script tag after inserting or else websites relying on the order of items in
0040     // document.getElementsByTagName("script") will break (looking at you, Google Hangouts)
0041     element.parentNode.removeChild(element);
0042 }
0043 
0044 function executePageAction(args) {
0045     // The page script injection and communication mechanism
0046     // was inspired by https://github.com/x0a/uBO-YouTube
0047     if (IS_FIREFOX) {
0048         args = cloneInto(args, window);
0049     }
0050 
0051     window.dispatchEvent(new CustomEvent('pbiEvent', {detail: args}));
0052 }
0053 
0054 chrome.runtime.onMessage.addListener(function (message, sender) {
0055     // TODO do something with sender (check privilige or whatever)
0056 
0057     var subsystem = message.subsystem;
0058     var action = message.action;
0059 
0060     if (!subsystem || !action) {
0061         return;
0062     }
0063 
0064     if (callbacks[subsystem] && callbacks[subsystem][action]) {
0065         callbacks[subsystem][action](message.payload);
0066     }
0067 });
0068 
0069 initPageScript(() => {
0070     SettingsUtils.get().then((items) => {
0071         if (items.breezeScrollBars.enabled) {
0072             loadBreezeScrollBars();
0073         }
0074 
0075         const mpris = items.mpris;
0076         if (mpris.enabled) {
0077             const origin = window.location.origin;
0078 
0079             const websiteSettings = mpris.websiteSettings || {};
0080 
0081             let mprisAllowed = true;
0082             if (typeof MPRIS_WEBSITE_SETTINGS[origin] === "boolean") {
0083                 mprisAllowed = MPRIS_WEBSITE_SETTINGS[origin];
0084             }
0085             if (typeof websiteSettings[origin] === "boolean") {
0086                 mprisAllowed = websiteSettings[origin];
0087             }
0088 
0089             if (mprisAllowed) {
0090                 loadMpris();
0091                 if (items.mprisMediaSessions.enabled) {
0092                     loadMediaSessionsShim();
0093                 }
0094             }
0095         }
0096 
0097         if (items.purpose.enabled) {
0098             sendMessage("settings", "getSubsystemStatus").then((status) => {
0099                 if (status && status.purpose) {
0100                     loadPurpose();
0101                 }
0102             }, (err) => {
0103                 // No warning, can also happen when port isn't connected for unsupported OS
0104                 console.log("Failed to get subsystem status for purpose", err);
0105             });
0106         }
0107     });
0108 });
0109 
0110 // BREEZE SCROLL BARS
0111 // ------------------------------------------------------------------------
0112 //
0113 function loadBreezeScrollBars() {
0114     if (IS_FIREFOX) {
0115         return;
0116     }
0117 
0118     if (!document.head) {
0119         return;
0120     }
0121 
0122     // You cannot access cssRules for <link rel="stylesheet" ..> on a different domain.
0123     // Since our chrome-extension:// URL for a stylesheet would be, this can
0124     // lead to problems in e.g modernizr, so include the <style> inline instead.
0125     // "Failed to read the 'cssRules' property from 'CSSStyleSheet': Cannot access rules"
0126     var styleTag = document.createElement("style");
0127     styleTag.appendChild(document.createTextNode(`
0128 html::-webkit-scrollbar {
0129     /* we'll add padding as "border" in the thumb*/
0130     height: 20px;
0131     width: 20px;
0132     background: white;
0133 }
0134 
0135 html::-webkit-scrollbar-track {
0136     border-radius: 20px;
0137     border: 7px solid white; /* FIXME why doesn't "transparent" work here?! */
0138     background-color: white;
0139     width: 6px !important; /* 20px scrollbar - 2 * 7px border */
0140     box-sizing: content-box;
0141 }
0142 html::-webkit-scrollbar-track:hover {
0143     background-color: #BFC0C2;
0144 }
0145 
0146 html::-webkit-scrollbar-thumb {
0147     background-color: #3DAEE9; /* default blue breeze color */
0148     border: 7px solid transparent;
0149     border-radius: 20px;
0150     background-clip: content-box;
0151     width: 6px !important; /* 20px scrollbar - 2 * 7px border */
0152     box-sizing: content-box;
0153     min-height: 30px;
0154 }
0155 html::-webkit-scrollbar-thumb:window-inactive {
0156    background-color: #949699; /* when window is inactive it's gray */
0157 }
0158 html::-webkit-scrollbar-thumb:hover {
0159     background-color: #93CEE9; /* hovered is a lighter blue */
0160 }
0161 
0162 html::-webkit-scrollbar-corner {
0163     background-color: white; /* FIXME why doesn't "transparent" work here?! */
0164 }
0165     `));
0166 
0167     document.head.appendChild(styleTag);
0168 }
0169 
0170 
0171 // MPRIS
0172 // ------------------------------------------------------------------------
0173 //
0174 
0175 var activePlayer;
0176 // When a player has no duration yet, we'll wait for it becoming known
0177 // to determine whether to ignore it (short sound) or make it active
0178 var pendingActivePlayer;
0179 var playerMetadata = {};
0180 var playerCallbacks = [];
0181 
0182 // Playback state communicated via media sessions api
0183 var playerPlaybackState = "";
0184 
0185 var players = new WeakSet();
0186 
0187 var pendingSeekingUpdate = 0;
0188 
0189 var titleTagObserver = null;
0190 var oldPageTitle = "";
0191 
0192 addCallback("mpris", "play", function () {
0193     playerPlay();
0194 });
0195 
0196 addCallback("mpris", "pause", function () {
0197     playerPause();
0198 });
0199 
0200 addCallback("mpris", "playPause", function () {
0201     if (activePlayer) {
0202         if (activePlayer.paused) { // TODO take into account media sessions playback state
0203             playerPlay();
0204         } else {
0205             playerPause();
0206         }
0207     }
0208 });
0209 
0210 addCallback("mpris", "stop", function () {
0211     // When available, use the "stop" media sessions action
0212     if (playerCallbacks.indexOf("stop") > -1) {
0213         executePageAction({"action": "mpris", "mprisCallbackName": "stop"});
0214         return;
0215     }
0216 
0217     // otherwise since there's no "stop" on the player, simulate it be rewinding and reloading
0218     if (activePlayer) {
0219         activePlayer.pause();
0220         activePlayer.currentTime = 0;
0221         // calling load() now as is suggested in some "how to fake video Stop" code snippets
0222         // utterly breaks stremaing sites
0223         //activePlayer.load();
0224 
0225         // needs to be delayed slightly otherwise we pause(), then send "stopped", and only after that
0226         // the "paused" signal is handled and we end up in Paused instead of Stopped state
0227         setTimeout(function() {
0228             sendMessage("mpris", "stopped");
0229         }, 1);
0230         return;
0231     }
0232 });
0233 
0234 addCallback("mpris", "next", function () {
0235     if (playerCallbacks.indexOf("nexttrack") > -1) {
0236         executePageAction({"action": "mpris", "mprisCallbackName": "nexttrack"});
0237     }
0238 });
0239 
0240 addCallback("mpris", "previous", function () {
0241     if (playerCallbacks.indexOf("previoustrack") > -1) {
0242         executePageAction({"action": "mpris", "mprisCallbackName": "previoustrack"});
0243     }
0244 });
0245 
0246 addCallback("mpris", "setFullscreen", (message) => {
0247     if (activePlayer) {
0248         if (message.fullscreen) {
0249             activePlayer.requestFullscreen();
0250         } else {
0251             document.exitFullscreen();
0252         }
0253     }
0254 });
0255 
0256 addCallback("mpris", "setPosition", function (message) {
0257     if (activePlayer) {
0258         activePlayer.currentTime = message.position;
0259     }
0260 });
0261 
0262 addCallback("mpris", "setPlaybackRate", function (message) {
0263     if (activePlayer) {
0264         activePlayer.playbackRate = message.playbackRate;
0265     }
0266 });
0267 
0268 addCallback("mpris", "setVolume", function (message) {
0269     if (activePlayer) {
0270         activePlayer.volume = message.volume;
0271         activePlayer.muted = (message.volume == 0.0);
0272     }
0273 });
0274 
0275 addCallback("mpris", "setLoop", function (message) {
0276     if (activePlayer) {
0277         activePlayer.loop = message.loop;
0278     }
0279 });
0280 
0281 addCallback("mpris", "identify", function (message) {
0282     if (activePlayer) {
0283         // We don't have a dedicated "send player info" callback, so we instead send a "playing"
0284         // and if we're paused, we'll send a "paused" event right after
0285         // TODO figure out a way how to add this to the host without breaking compat
0286 
0287         var paused = activePlayer.paused;
0288         playerPlaying(activePlayer);
0289         if (paused) {
0290             playerPaused(activePlayer);
0291         }
0292     }
0293 });
0294 
0295 function playerPlaying(player) {
0296     setPlayerActive(player);
0297 }
0298 
0299 function playerPaused(player) {
0300     sendPlayerInfo(player, "paused");
0301 }
0302 
0303 function setPlayerActive(player) {
0304     pendingActivePlayer = player;
0305 
0306     if (isNaN(player.duration)) {
0307         // Ignore this player for now until we know a duration
0308         // In durationchange event handler we'll check for this and end up here again
0309         return;
0310     }
0311 
0312     // Ignore short sounds, they are most likely a chat notification sound
0313     // A stream has a duration of Infinity
0314     // Note that "NaN" is also not finite but we already returned earlier for that
0315     if (isFinite(player.duration) && player.duration > 0 && player.duration < 8) {
0316         return;
0317     }
0318 
0319     pendingActivePlayer = undefined;
0320     activePlayer = player;
0321 
0322     // when playback starts, send along metadata
0323     // a website might have set Media Sessions metadata prior to playing
0324     // and then we would have ignored the metadata signal because there was no player
0325     sendMessage("mpris", "playing", {
0326         mediaSrc: player.currentSrc || player.src,
0327         pageTitle: document.title,
0328         poster: player.poster,
0329         duration: player.duration,
0330         currentTime: player.currentTime,
0331         playbackRate: player.playbackRate,
0332         volume: player.volume,
0333         muted: player.muted,
0334         loop: player.loop,
0335         metadata: playerMetadata,
0336         callbacks: playerCallbacks,
0337         fullscreen: document.fullscreenElement !== null,
0338         canSetFullscreen: player.tagName.toLowerCase() === "video"
0339     });
0340 
0341     if (!titleTagObserver) {
0342 
0343         // Observe changes to the <title> tag in case it is updated after the player has started playing
0344         let titleTag = document.querySelector("head > title");
0345         if (titleTag) {
0346             oldPageTitle = titleTag.innerText;
0347 
0348             titleTagObserver = new MutationObserver((mutations) => {
0349                 mutations.forEach((mutation) => {
0350                     const pageTitle = mutation.target.textContent;
0351                     if (pageTitle && oldPageTitle !== pageTitle) {
0352                         sendMessage("mpris", "titlechange", {
0353                             pageTitle: pageTitle
0354                         });
0355                     }
0356                     oldPageTitle = pageTitle;
0357                 });
0358             });
0359 
0360             titleTagObserver.observe(titleTag, {
0361                 childList: true, // text content is technically a child node
0362                 subtree: true,
0363                 characterData: true
0364             });
0365         }
0366     }
0367 }
0368 
0369 function sendPlayerGone() {
0370     activePlayer = undefined;
0371     pendingActivePlayer = undefined;
0372     playerMetadata = {};
0373     playerCallbacks = [];
0374     sendMessage("mpris", "gone");
0375 
0376     if (titleTagObserver) {
0377         titleTagObserver.disconnect();
0378         titleTagObserver = null;
0379     }
0380 }
0381 
0382 function sendPlayerInfo(player, event, payload) {
0383     if (player != activePlayer) {
0384         return;
0385     }
0386 
0387     sendMessage("mpris", event, payload);
0388 }
0389 
0390 function registerPlayer(player) {
0391     if (players.has(player)) {
0392         //console.log("Already know", player);
0393         return;
0394     }
0395 
0396     // auto-playing player, become active right away
0397     if (!player.paused) {
0398         playerPlaying(player);
0399     }
0400     player.addEventListener("play", function () {
0401         playerPlaying(player);
0402     });
0403 
0404     player.addEventListener("pause", function () {
0405         playerPaused(player);
0406     });
0407 
0408     // what about "stalled" event?
0409     player.addEventListener("waiting", function () {
0410         sendPlayerInfo(player, "waiting");
0411     });
0412 
0413     // playlist is now empty or being reloaded, stop player
0414     // e.g. when using Ajax page navigation and the user nagivated away
0415     player.addEventListener("emptied", function () {
0416         // When the player is emptied but the website tells us it's just "paused"
0417         // keep it around (Bug 402324: Soundcloud does this)
0418         if (player === activePlayer && playerPlaybackState === "paused") {
0419             return;
0420         }
0421 
0422         // could have its own signal but for compat it's easier just to pretend to have stopped
0423         sendPlayerInfo(player, "stopped");
0424     });
0425 
0426     // opposite of "waiting", we finished buffering enough
0427     // only if we are playing, though, should we set playback state back to playing
0428     player.addEventListener("canplay", function () {
0429         if (!player.paused) {
0430             sendPlayerInfo(player, "canplay");
0431         }
0432     });
0433 
0434     player.addEventListener("timeupdate", function () {
0435         sendPlayerInfo(player, "timeupdate", {
0436             currentTime: player.currentTime
0437         });
0438     });
0439 
0440     player.addEventListener("ratechange", function () {
0441         sendPlayerInfo(player, "ratechange", {
0442             playbackRate: player.playbackRate
0443         });
0444     });
0445 
0446     // TODO use player.seekable for determining whether we can seek?
0447     player.addEventListener("durationchange", function () {
0448         // Deferred active due to unknown duration
0449         if (pendingActivePlayer == player) {
0450             setPlayerActive(pendingActivePlayer);
0451             return;
0452         }
0453 
0454         sendPlayerInfo(player, "duration", {
0455             duration: player.duration
0456         });
0457     });
0458 
0459     player.addEventListener("seeking", function () {
0460         if (pendingSeekingUpdate) {
0461             return;
0462         }
0463 
0464         // Compress "seeking" signals, this is invoked continuously as the user drags the slider
0465         pendingSeekingUpdate = setTimeout(function() {
0466             pendingSeekingUpdate = 0;
0467         }, 250);
0468 
0469         sendPlayerInfo(player, "seeking", {
0470             currentTime: player.currentTime
0471         });
0472     });
0473 
0474     player.addEventListener("seeked", function () {
0475         sendPlayerInfo(player, "seeked", {
0476             currentTime: player.currentTime
0477         });
0478     });
0479 
0480     player.addEventListener("volumechange", function () {
0481         sendPlayerInfo(player, "volumechange", {
0482             volume: player.volume,
0483             muted: player.muted
0484         });
0485     });
0486 
0487     players.add(player);
0488 }
0489 
0490 function findAllPlayersFromNode(node) {
0491     if (typeof node.getElementsByTagName !== "function") {
0492         return [];
0493     }
0494 
0495     return [...node.getElementsByTagName("video"), ...node.getElementsByTagName("audio")];
0496 }
0497 
0498 
0499 function registerAllPlayers() {
0500     var players = findAllPlayersFromNode(document);
0501     players.forEach(registerPlayer);
0502 }
0503 
0504 function playerPlay() {
0505     // if a media sessions callback is registered, it takes precedence over us manually messing with the player
0506     if (playerCallbacks.indexOf("play") > -1) {
0507         executePageAction({"action": "mpris", "mprisCallbackName": "play"});
0508     } else if (activePlayer) {
0509         activePlayer.play();
0510     }
0511 }
0512 
0513 function playerPause() {
0514     if (playerCallbacks.indexOf("pause") > -1) {
0515         executePageAction({"action": "mpris", "mprisCallbackName": "pause"});
0516     } else if (activePlayer) {
0517         activePlayer.pause();
0518     }
0519 }
0520 
0521 function loadMpris() {
0522     // TODO figure out somehow when a <video> tag is added dynamically and autoplays
0523     // as can happen on Ajax-heavy pages like YouTube
0524     // could also be done if we just look for the "audio playing in this tab" and only then check for player?
0525     // cf. "checkPlayer" event above
0526 
0527     var observer = new MutationObserver(function (mutations) {
0528         let nodesRemoved = false;
0529         mutations.forEach(function (mutation) {
0530             mutation.addedNodes.forEach(function (node) {
0531                 if (typeof node.matches !== "function") {
0532                     return;
0533                 }
0534 
0535                 // Check whether the node itself or any of its children is a player
0536                 var players = findAllPlayersFromNode(node);
0537                 if (node.matches("video,audio")) {
0538                     players.unshift(node);
0539                 }
0540 
0541                 players.forEach(function (player) {
0542                     registerPlayer(player);
0543                 });
0544             });
0545 
0546             mutation.removedNodes.forEach((node) => {
0547                 if (typeof node.matches !== "function") {
0548                     return;
0549                 }
0550 
0551                 // Check whether the node itself or any of its children is the current player
0552                 const players = findAllPlayersFromNode(node);
0553                 if (node.matches("video,audio")) {
0554                     players.unshift(node);
0555                 }
0556 
0557                 for (let player of players) {
0558                     if (player !== activePlayer) {
0559                         continue;
0560                     }
0561 
0562                     // If the player is still in the visible DOM, don't consider it gone
0563                     if (document.body.contains(player)) {
0564                         continue;
0565                     }
0566 
0567                     // If the player got temporarily added by us, don't consider it gone
0568                     if (player.dataset.pbiPausedForDomRemoval === "true") {
0569                         continue;
0570                     }
0571 
0572                     sendPlayerGone();
0573                     break;
0574                 }
0575             });
0576         });
0577     });
0578 
0579     window.addEventListener("pagehide", function () {
0580         // about to navigate to a different page, tell our extension that the player will be gone shortly
0581         // we listen for tab closed in the extension but we don't for navigating away as URL change doesn't
0582         // neccesarily mean a navigation.
0583         // NOTE beforeunload is not emitted for iframes!
0584         sendPlayerGone();
0585     });
0586 
0587     function documentReady() {
0588         registerAllPlayers();
0589 
0590         observer.observe(document, {
0591             childList: true,
0592             subtree: true
0593         });
0594     }
0595 
0596     // In some cases DOMContentLoaded won't fire, e.g. when watching a video file directly in the browser
0597     // it generates a "video player" page for you but won't fire the event.
0598     // Also, make sure to install the mutation observer if this codepath is executed after the page is already ready.
0599     if (["interactive", "complete"].includes(document.readyState)) {
0600         documentReady();
0601     } else {
0602         registerAllPlayers(); // in case the document isn't ready but the event also doesn't fire...
0603         document.addEventListener("DOMContentLoaded", documentReady);
0604     }
0605 
0606     document.addEventListener("fullscreenchange", () => {
0607         if (activePlayer) {
0608             sendPlayerInfo(activePlayer, "fullscreenchange", {
0609                 fullscreen: document.fullscreenElement !== null
0610             });
0611         }
0612     });
0613 }
0614 
0615 // This adds a shim for the Chrome media sessions API which is currently only supported on Android
0616 // Documentation: https://developers.google.com/web/updates/2017/02/media-session
0617 // Try it here: https://googlechrome.github.io/samples/media-session/video.html
0618 
0619 // Bug 379087: Only inject this stuff if we're a proper HTML page
0620 // otherwise we might end up messing up XML stuff
0621 // only if our documentElement is a "html" tag we'll do it
0622 // the rest is only set up in DOMContentLoaded which is only executed for proper pages anyway
0623 
0624 // tagName always returned "HTML" for me but I wouldn't trust it always being uppercase
0625 function loadMediaSessionsShim() {
0626     if (document.documentElement.tagName.toLowerCase() === "html") {
0627 
0628         window.addEventListener("pbiMprisMessage", (e) => {
0629             let data = e.detail || {};
0630 
0631             let action = data.action;
0632             let payload = data.payload;
0633 
0634             switch (action) {
0635             case "metadata":
0636                 playerMetadata = {};
0637 
0638                 if (typeof payload !== "object") {
0639                     return;
0640                 }
0641 
0642                 playerMetadata = payload;
0643                 sendMessage("mpris", "metadata", payload);
0644 
0645                 return;
0646 
0647             case "playbackState":
0648                 if (!["none", "paused", "playing"].includes(payload)) {
0649                     return;
0650                 }
0651 
0652                 playerPlaybackState = payload;
0653 
0654                 if (!activePlayer) {
0655                     return;
0656                 }
0657 
0658                 if (playerPlaybackState === "playing") {
0659                     playerPlaying(activePlayer);
0660                 } else if (playerPlaybackState === "paused") {
0661                     playerPaused(activePlayer);
0662                 }
0663 
0664                 return;
0665 
0666             case "callbacks":
0667                 if (Array.isArray(payload)) {
0668                     playerCallbacks = payload;
0669                 } else {
0670                     playerCallbacks = [];
0671                 }
0672                 sendMessage("mpris", "callbacks", playerCallbacks);
0673 
0674                 return;
0675             }
0676         });
0677 
0678         executePageAction({"action": "mediaSessionsRegister"});
0679     }
0680 }
0681 
0682 // PURPOSE / WEB SHARE API
0683 // ------------------------------------------------------------------------
0684 //
0685 var purposeLoaded = false;
0686 function loadPurpose() {
0687     if (purposeLoaded) {
0688         return;
0689     }
0690 
0691     purposeLoaded = true;
0692 
0693     // navigator.share must only be defined in secure (https) context
0694     if (!window.isSecureContext) {
0695         return;
0696     }
0697 
0698      window.addEventListener("pbiPurposeMessage", (e) => {
0699         const data = e.detail || {};
0700 
0701         const action = data.action;
0702         const payload = data.payload;
0703 
0704         if (action !== "share") {
0705             return;
0706         }
0707 
0708         sendMessage("purpose", "share", payload).then((response) => {
0709             executePageAction({"action": "purposeShare"});
0710         }, (err) => {
0711             // Deliberately not giving any more details about why it got rejected
0712             executePageAction({"action": "purposeReject"});
0713         }).finally(() => {
0714             executePageAction({"action": "purposeReset"});
0715         });;
0716     });
0717 
0718     executePageAction({"action": "purposeRegister"});
0719 }