// ==UserScript== // @name Netflix Plus // @namespace http://tampermonkey.net/ // @version 1.4 // @description Enable dolby plus audio and more video on Netflix // @author TGSAN // @match https://www.netflix.com/* // @icon https://www.google.com/s2/favicons?sz=64&domain=netflix.com // @run-at document-start // @grant unsafeWindow // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @downloadURL none // ==/UserScript== (() => { "use strict"; let windowCtx = self.window; if (self.unsafeWindow) { console.log("[Netflix Plus] use unsafeWindow mode"); windowCtx = self.unsafeWindow; } else { console.log("[Netflix Plus] use window mode (your userscript extensions not support unsafeWindow)"); } const Event = class { constructor(script, target) { this.script = script; this.target = target; this._cancel = false; this._replace = null; this._stop = false; } preventDefault() { this._cancel = true; } stopPropagation() { this._stop = true; } replacePayload(payload) { this._replace = payload; } }; let callbacks = []; windowCtx.addBeforeScriptExecuteListener = (f) => { if (typeof f !== "function") { throw new Error("Event handler must be a function."); } callbacks.push(f); }; windowCtx.removeBeforeScriptExecuteListener = (f) => { let i = callbacks.length; while (i--) { if (callbacks[i] === f) { callbacks.splice(i, 1); } } }; const dispatch = (script, target) => { if (script.tagName !== "SCRIPT") { return; } const e = new Event(script, target); if (typeof windowCtx.onbeforescriptexecute === "function") { try { windowCtx.onbeforescriptexecute(e); } catch (err) { console.error(err); } } for (const func of callbacks) { if (e._stop) { break; } try { func(e); } catch (err) { console.error(err); } } if (e._cancel) { script.textContent = ""; script.remove(); } else if (typeof e._replace === "string") { script.textContent = e._replace; } }; const observer = new MutationObserver((mutations) => { for (const m of mutations) { for (const n of m.addedNodes) { dispatch(n, m.target); } } }); observer.observe(document, { childList: true, subtree: true, }); const menuItems = [ ["setMaxBitrate", "Automatically select best bitrate available"], ["useFHDAndAVCH", "Focus 1080P and High-AVC"], ]; let hasLoaded = false; let menuCommandList = []; windowCtx.globalOptions = { useallSub: true, useddplus: true, useAVC: false, useFHDAndAVCH: ((!windowCtx.MSMediaKeys) || (!windowCtx.WebKitMediaKeys)), get ["useFHD"]() { return windowCtx.globalOptions.useFHDAndAVCH; }, usedef: false, useHA: true, get ["useAVCH"]() { return windowCtx.globalOptions.useFHDAndAVCH; }, usevp9: false, useav1: false, useprk: true, usehevc: false, usef4k: true, usef12k: true, closeimsc: true, setMaxBitrate: true }; windowCtx.onbeforescriptexecute = function (e) { let scripts = document.getElementsByTagName("script"); if (scripts.length === 0) return; for (let i = 0; scripts.length > i; i++) { let dom = scripts[i]; if (dom.src.includes("cadmium-playercore")) { let playercore = document.createElement('script'); playercore.src = "https://static-os.kumo.moe/js/netflix/cadmium-playercore.js"; playercore.crossOrigin = dom.crossOrigin; playercore.async = dom.async; playercore.id = dom.id; document.head.appendChild(playercore); let maxbitrate = document.createElement('script'); maxbitrate.src = 'https://static-os.kumo.moe/js/netflix/max-bitrate.js'; maxbitrate.crossOrigin = dom.crossOrigin; maxbitrate.async = dom.async; maxbitrate.id = dom.id; document.head.appendChild(maxbitrate); dom.remove(); windowCtx.onbeforescriptexecute = null; break; } } }; function checkSelected(type) { let selected = GM_getValue("NETFLIX_PLUS_" + type); if (typeof selected == "boolean") { return selected; } else { return windowCtx.globalOptions[type]; } } function registerSelectableVideoProcessingMenuCommand(name, type) { let selected = checkSelected(type); windowCtx.globalOptions[type] = selected; return GM_registerMenuCommand((checkSelected(type) ? "✅" : "🔲") + " " + name, function() { GM_setValue("NETFLIX_PLUS_" + type, !selected); windowCtx.globalOptions[type] = !selected; updateMenuCommand(); }); } async function updateMenuCommand() { for (let command of menuCommandList) { await GM_unregisterMenuCommand(command); } menuCommandList = []; for (let menuItem of menuItems) { menuCommandList.push(await registerSelectableVideoProcessingMenuCommand(menuItem[1], menuItem[0])); } } windowCtx.document.addEventListener("readystatechange", (event) => { // 防止双重载入(第二次一般不会有interactive,直接complete) if (event.target.readyState === "interactive") { // 防止在框架内再次载入 if (!windowCtx.frameElement) { if (!hasLoaded) { hasLoaded = true; updateMenuCommand(); } } } }); })();