// ==UserScript== // @name (改)百度网盘会员青春版 // @author hmjz100、Gwen // @namespace github.com/hmjz100 // @version 1.2.7 // @description 《也许同类型中最好用?》系列、LinkSwift 官方扩展 - 虚拟百度网盘身份信息为会员,支持隐藏云一朵、使用视频倍速、生成/查看/下载 AI字幕、修改头像、用户名、等级、SVIP 10 ID 等功能,自定义程度超高! // @icon  // @license AGPL-3.0-or-later // @homepage https://github.com/hmjz100/BaiduNetDiskYouthMember/ // @homepageURL https://github.com/hmjz100/BaiduNetDiskYouthMember/ // @support https://github.com/hmjz100/BaiduNetDiskYouthMember/issues // @supportURL https://github.com/hmjz100/BaiduNetDiskYouthMember/issues // @require https://unpkg.com/jquery@3.6.0/dist/jquery.min.js // @require https://unpkg.com/sweetalert2@11.4.8/dist/sweetalert2.min.js // @run-at document-start // @early-start // @match *://pan.baidu.com/* // @match *://yun.baidu.com/* // @grant GM_getValue // @grant GM_setValue // @grant GM_addStyle // @grant GM_registerMenuCommand // @grant unsafeWindow // @compatible Chrome // @compatible Edge // @compatible Firefox // @compatible Safari // @compatible Opera // ==/UserScript== (function $baidu() { // 严格模式,确保代码安全执行,不越界 "use strict"; // unsafeWindow 检测,适用于 Via 这类无 unsafeWindow 的浏览器 if (typeof (unsafeWindow) === "undefined") window.unsafeWindow = window; let temp = { // 脚本已包含设置 UI,所以请不要修改此处的代码。 defaults: { vip: true, svip: true, previous: false, name: "百湿不得其解", photo: "https://bkimg.cdn.bcebos.com/pic/cdbf6c81800a19d8bc3e5cfd95ab958ba61ea9d3e8b3", // desc: "What's your problem? | 业绩增长凌驾于用户体验,简单经营替代了简单可依赖,我们与用户渐行渐远,我们与创业初期坚守的使命和价值观渐行渐远。", level: 10, point: 100000, vip_id: "000001", endtime: 253402185600, unAnalytics: true, unAI_Yiduo: true, unPViewer: false, unVerSwitch: false, unTab_aiNote: true, unTab_aiStudy: false, unTab_aiCourseware: true, unTab_aiDraft: false, iLoveAD: false, debug: false, }, swalDefault: { position: "center", heightAuto: false, scrollbarPadding: false, confirmButtonText: `确认`, denyButtonText: `拒绝`, cancelButtonText: `取消` }, hooker: new WeakMap(), url: (url = location) => url.host + url.pathname + url.search + url.hash } /* 防止代码因其他原因被执行多次 代码出自 “Via 轻插件”,作者谷花泰 */ let key = encodeURIComponent("(改)百度网盘会员青春版:主代码"); if (window[key]) return; window[key] = true; let mount = idontknow("BaiduNetDiskYouthMember"); let info = { author: GM_info.script?.author, name: GM_info.script?.name, version: (GM_info.script?.version || "1.2.7"), icon: (GM_info.script?.icon || ""), maction: GM_info.scriptaction, mversion: GM_info.version, }; let base = { hooker: { toString: function () { if (temp.hooker.has(this)) return `function ${this.name}() { [native code] }`; return originalToString.call(this); }, toLocaleString: function () { if (temp.hooker.has(this)) return `function ${this.name}() { [native code] }`; return originalToLocaleString.call(this); }, attachShadow(options) { if (options?.mode === "closed") { options.mode = "open"; console.log("【(改)百度网盘会员青春版】Hook Shadow\n新增:", this); }; return originalAttachShadow.call(this, options); }, async fetch(input, init = {}) { let url = typeof input === "string" ? input : input?.url || ""; url = new URL(url, location.origin); if (rules.some(r => r.match(url) && r.condition() && r.runat === "url")) url = base.hooker.universal("fetch", url, url, "url"); if (rules.some(r => r.match(url) && r.condition() && r.runat === "header")) { if (!init.headers) init.headers = {}; let tempHeaders = {}; if (init.headers instanceof Headers) { for (let [key, value] of init.headers.entries()) { tempHeaders[key] = value; } } else { tempHeaders = { ...init.headers }; } init.headers = new Headers(base.hooker.universal("fetch", url, tempHeaders, "header")); } if (rules.some(r => r.match(url) && r.condition() && r.runat === "end")) { try { let response = await originalFetch.call(this, url, init); let responseText = await response.text(); let res = base.hooker.universal("fetch", url, responseText, "end"); return new Response(res, { status: response.status, statusText: response.statusText, headers: response.headers }); } catch (e) { return originalFetch.call(this, url, init); } } if (rules.some(r => r.match(url) && r.condition() && r.runat === "start")) { try { let res = base.hooker.universal("fetch", url, null, "start"); return new Response(res, { status: 200, statusText: "OK", headers: { "Content-Type": "plain/text" } }); } catch (e) { return originalFetch.call(this, url, init); } } return originalFetch.call(this, url, init); }, send(data) { let url = this[mount]; if (this.headers) { for (let [name, value] of Object.entries(this.headers)) { originalSetRequestHeader.call(this, name, value); } } if (rules.find(rule => rule.match(url) && rule.condition() && rule.runat === "start")) { try { let res = base.hooker.universal("XHR", url, null, "start"); Object.defineProperty(this, "responseText", { writable: true, }); Object.defineProperty(this, "response", { writable: true, }); this.response = res; this.responseText = res; ["readystatechange", "load", "loadend"].forEach(prop => { this.dispatchEvent(new Event(prop)); if (typeof this[prop] === "function") this[prop](); if (typeof this[`on${prop}`] === "function") this[`on${prop}`](); }); return true; // 显式 return } catch (e) { return originalSend.apply(this, arguments); } } return originalSend.apply(this, arguments); }, open(method, input) { let url = new URL(input, location.origin); if (rules.some(r => r.match(url) && r.condition() && r.runat === "url")) url = base.hooker.universal("XHR", url, url, "url"); this[mount] = url; this.addEventListener("readystatechange", function () { if (this.readyState === 4) { if (rules.some(r => r.match(url) && r.condition() && r.runat === "end")) { let response = this.response; if (this.responseType === "" || this.responseType === "text") response = this.responseText; let res = base.hooker.universal("XHR", url, response, "end"); Object.defineProperty(this, "responseText", { writable: true, }); Object.defineProperty(this, "response", { writable: true, }); this.response = res; this.responseText = res; } } }); return originalOpen.call(this, method, url); }, setRequestHeader(name, value) { let url = this[mount]; if (url && rules.some(r => r.match(url) && r.condition() && r.runat === "header")) { if (!this.headers) this.headers = {}; this.headers[name] = value; this.headers = base.hooker.universal("XHR", url, { ...this.headers }, "header"); return true; // 显式 return } return originalSetRequestHeader.call(this, name, value); }, sendBeacon(input, data) { let url = new URL(input, location.origin); if (rules.some(rule => rule.match(url) && rule.condition() && rule.runat === "start")) { data = base.hooker.universal("sendBeacon", url, data, "start"); return true; } return originalSendBeacon.call(this, url, data); }, Image: function Image(...args) { return new Proxy(new OriginalImage(...args), { set(target, prop, value) { if (prop === "src") { let url = new URL(value, location.origin); if (rules.some(r => r.match(url) && r.condition() && r.runat === "image")) { // if (user.debug) console.log(`${log_head} Image`, "\n请求地址:", url.href, "\n模拟回复: 成功"); return true; // 显式 return } } return (target[prop] = value); // 未匹配,设置属性 }, get(target, prop) { return target[prop]; // 获取属性值 } }); }, createElement(tagName) { let element = originalCreateElement.call(unsafeWindow.document, tagName) if (tagName.toLowerCase() === "img") { Object.defineProperty(element, "src", { set(value) { let url = new URL(value, location.origin); if (rules.some(r => r.match(url) && r.condition() && r.runat === "image")) { // if (user.debug) console.log(`${log_head} createElement img`, "\n请求地址:", url.href, "\n模拟回复: 成功"); return true; // 显式 return } return element.setAttribute("src", value); // 未匹配,设置属性 }, get() { return element.getAttribute("src"); } }); } return element; }, universal(from, url, data, type) { let res, oriRes; try { res = typeof data === "string" ? JSON.parse(data) : data; oriRes = typeof data === "string" ? JSON.parse(data) : base.deepClone(data); } catch (e) { res = data; oriRes = base.deepClone(data); } let rule = rules.find(r => r.match(url) && r.condition() && r.runat === type); if (rule && typeof rule.action === "function") { if (type === "variable") { var originalSet = res.set; if (originalSet) res.set = (...args) => { set("set", ...args) }; var originalMSet = res.mset; if (originalMSet) res.mset = (...args) => { set("mset", ...args) }; var originalSetData = res.setData; if (originalSetData) res.setData = (...args) => { set("setData", ...args) }; function set(type, ...args) { let object; if (typeof args[0] === "object" && !Array.isArray(args[0]) && args[0] !== null) { object = args[0]; } else if (args[1] && args[1] !== null) { object = { [args[0]]: args[1] }; } for (let key in object) { if (Object.prototype.hasOwnProperty.call(object, key)) { if (["function", "symbol", "undefined"].includes((typeof object[key]).toLowerCase())) { delete object[key]; } } } object = base.hooker.universal(from, url, object, "variable"); if (type === "set") for (key in object) { originalSet.call(res, key, object[key]); } if (type === "mset") { if (originalSet) for (key in object) { originalSet.call(res, key, object[key]); } else { originalMSet.call(res, object); } } if (type === "setData") originalSetData.call(res, object); } if (!originalMSet && !originalMSet && !originalSetData) res = rule.action(res, url); } if (!["variable"].includes(type)) res = rule.action(res, url); if (type === "header" && res !== null && typeof res === "object") try { let tempHeaders = {}; for (let key in res) tempHeaders[key.toLowerCase().split("-").map(word => word.charAt(0).toUpperCase() + word.slice(1)).join("-")] = res[key]; res = tempHeaders; } catch { } if (user.debug) { let log_head = `【(改)百度网盘会员青春版】Hook` if (!base.deepEqual(res, oriRes)) { let format = (data) => typeof data === "string" && data.startsWith("#EXTM3U") ? "(M3U8预览) " + data.split("\n").slice(0, 5).join("\\n") : data; type === "start" && console.log(`${log_head} ${from} Virt`, "\n请求地址:", url.href, "\n模拟回复:", format(res) || "空"); type === "end" && console.log(`${log_head} ${from} Repl`, "\n请求地址:", url.href, "\n原始回复:", format(oriRes) || "空", "\n修改回复:", format(res) || "空"); type === "header" && console.log(`${log_head} ${from} Head`, "\n请求地址:", url.href, "\n原始头部:", oriRes || "空", "\n修改头部:", res || "空"); type === "variable" && console.log(`${log_head} ${from}`, "\n原始内容:", oriRes || "空", "\n修改内容:", res || "空"); } type === "url" && console.log(`${log_head} ${from} URL`, "\n原始地址:", oriRes?.href || "空", "\n修改地址:", res?.href || "空"); } if (typeof res === "object" && !["header", "variable", "url"].includes(type)) try { res = JSON.stringify(res); } catch { } } return res; }, }, deepClone(obj) { if (obj instanceof URL) return new URL(obj.href); if (obj === null || typeof obj !== "object") return obj; if (obj instanceof Date) return new Date(obj); if (obj instanceof RegExp) return new RegExp(obj); if (typeof obj === "function") return obj; // 函数直接引用(无法真正克隆) if (Array.isArray(obj)) { return obj.map(item => this.deepClone(item)); } let cloned = {}; for (let key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { cloned[key] = this.deepClone(obj[key]); } } return cloned; }, deepEqual(a, b) { // 严格相等 if (Object.is(a, b)) return true; // 类型不同直接返回 false if (typeof a !== typeof b) return false; // null 或 undefined(typeof null === "object",需提前处理) if (a === null || b === null || a === undefined || b === undefined) { return a === b; } // Date if (a instanceof Date && b instanceof Date) { return a.getTime() === b.getTime(); } // RegExp if (a instanceof RegExp && b instanceof RegExp) { return a.toString() === b.toString(); } // 函数:只有同一个引用才算相等 if (typeof a === "function" && typeof b === "function") { return a === b; } // 非对象类型(string, number, boolean 等)已在 Object.is 处理,此处可跳过 if (typeof a !== "object") { return false; } // 数组 let aIsArray = Array.isArray(a); let bIsArray = Array.isArray(b); if (aIsArray !== bIsArray) return false; if (aIsArray) { if (a.length !== b.length) return false; for (let i = 0; i < a.length; i++) { if (!this.deepEqual(a[i], b[i])) return false; } return true; } // 普通对象 let aKeys = Object.keys(a); let bKeys = Object.keys(b); if (aKeys.length !== bKeys.length) return false; for (let key of aKeys) { if (!Object.prototype.hasOwnProperty.call(b, key)) return false; if (!this.deepEqual(a[key], b[key])) return false; } return true; }, getSettings() { var now = GM_getValue("user_settings", {}); return { ...temp.defaults, ...now }; }, async setSettings(settings) { await GM_setValue("user_settings", settings); }, /** * 显示脚本设置 * @author hmjz100 */ async showSettings() { let settings = base.getSettings(); function generateHTML(title, key, type, other = {}) { function escapeHtml(str) { if (typeof str !== "string") return str; return str.replace(/[&<>""]/g, tag => ({ "&": "&", "<": "<", ">": ">", "\"": `"`, "'": `'` }[tag] || tag) ); } // 将 other 对象转换为属性字符串 function buildAttrs(obj = {}) { let attrs = ""; for (let [k, v] of Object.entries(obj)) { // 跳过空值或 undefined if (v === undefined || v === null) continue; if (typeof v === "boolean") { if (v) attrs += ` ${k}`; continue; } // 其他值作为 k="value" attrs += ` ${k}="${escapeHtml(String(v))}"`; } return attrs; }; let control = ""; let attrs = buildAttrs(other); if (type === "boolean") { let checked = settings[key] == true ? " checked" : ""; control = `` } else if (type === "string") { let value = settings[key] || ""; let inputType = other?.type ? `type="${escapeHtml(String(other.type))}"` : `type="text"`; if (key === "vip_id") value = parseInt(value) || ""; control = `` } if (title.includes("|")) { let parts = title.split("|").map(part => `${part}`); title = parts.join("
"); } else { title = `${title}`; } return ``; } let dialog = await Swal.fire({ ...temp.swalDefault, icon: "info", iconHtml: "⚙︎", title: "设置", html: `
Tips: 设置自动保存,刷新页面后生效。
Tips: 您也可通过脚本管理器菜单再次打开本窗口。
风雨送春归,飞雪迎春到。已是悬崖百丈冰,犹有花枝俏。
俏也不争春,只把春来报。待到山花烂漫时,她在丛中笑。
${generateHTML("会员|所有功能的总开关,显示为 VIP", "vip", "boolean")} ${generateHTML("超级会员|显示为 SVIP", "svip", "boolean")} ${generateHTML("过期|显示为会员过期", "previous", "boolean")} ${generateHTML("昵称|对外展示信息", "name", "string", { "type": "string", "placeholder": "留空将使用已登录账户信息" })} ${generateHTML("头像|对外展示信息|可以是 http(s):data: 链接", "photo", "string", { "type": "string", "placeholder": "留空将使用已登录账户信息" })} ${generateHTML("等级|会员等级|1 ~ 10", "level", "string", { "type": "number", "min": "1", "max": "10", "placeholder": "留空将使用已登录账户信息" })} ${generateHTML("成长值|会员成长值|1 ~ 100000", "point", "string", { "type": "number", "min": "1", "max": "100000", "placeholder": "留空将使用已登录账户信息" })} ${generateHTML("SVIP 10 专属 ID|显示时会自动补零|1 ~ 999999", "vip_id", "string", { "type": "number", "min": "1", "max": "999999", "placeholder": "留空将使用已登录账户信息" })} ${generateHTML("会员过期时间|对外展示信息|Unix 时间戳", "endtime", "string", { "type": "number", "min": "-2147483648", "max": "253402185600", "placeholder": temp.defaults.endtime })} ${generateHTML("屏蔽网页数据分析", "unAnalytics", "boolean")} ${generateHTML("禁用 “云一朵” 相关功能入口", "unAI_Yiduo", "boolean")} ${generateHTML(`禁用新版在线文档查看器|勾选后新版的 PDF 在线预览会变为 WPS`, "unPViewer", "boolean")} ${generateHTML("隐藏新旧版切换入口", "unVerSwitch", "boolean")} ${generateHTML("隐藏视频页 “笔记” ", "unTab_aiNote", "boolean")} ${generateHTML("隐藏视频页 “AI看” ", "unTab_aiStudy", "boolean")} ${generateHTML("隐藏视频页 “课件” ", "unTab_aiCourseware", "boolean")} ${generateHTML("隐藏视频页 “文稿” ", "unTab_aiDraft", "boolean")} ${generateHTML("我爱看广告|修改视频广告播放时间为 200 秒,SVIP 下无效", "iLoveAD", "boolean")} ${generateHTML("显示调试信息到控制台", "debug", "boolean")}
`, allowOutsideClick: false, showCloseButton: true, showDenyButton: true, confirmButtonText: "刷新", denyButtonText: "关闭", didOpen: (toast) => { let element = $(toast); element.find(".ym-settings input").each((index, tag) => { $(tag).on("change", async (event) => { let element = $(event.target); let key = element.data("key"); if (element.attr("type") === "checkbox") { settings[key] = element.prop("checked"); } else { let value = element.val() || ""; if (element.attr("type") === "number") value = parseInt(value) || ""; if (key === "vip_id") value = String(value).padStart(6, "0"); settings[key] = value; } await base.setSettings(settings); }) }) }, footer: `

感谢您选择会员青春版,喜欢的话就来给个 Star 吧~

`, }); if (dialog.isConfirmed) { location.reload(); // 刷新页面使设置生效 } }, /** * 等待指定元素加载完成并执行回调 * @author hmjz100 * @description 监听 DOM 元素是否出现,若未出现则每隔一段时间重试,直到找到为止。 * 支持在 iframe 内部查找元素,适用于异步加载场景。 * @param {string} selectorElem - 要等待的目标元素选择器 * @param {Function} actionFunction - 找到元素后执行的回调函数,接收 jQuery 元素作为参数,返回 true 可以不再继续寻找 * @param {boolean} [bWaitOnce=false] - 是否只执行一次回调,默认为 false,如果不设置为 true 的话需要自行判断是否对元素进行操作 * @param {string} [iframeSelector] - 若目标元素位于 iframe 中,传入 iframe 的选择器 * @param {string} [controlKey] - 控制唯一性的键名,用于避免重复处理 */ waitForKeyElements(selectorElem, actionFunction, bWaitOnce, iframeSelector, controlKey) { // 初始化管理器 let manager = this.waitForKeyElements.manager; if (!manager) { manager = this.waitForKeyElements.manager = { observers: new WeakMap(), tasks: new Map(), instanceCounter: 0 }; } // 获取目标文档 let targetDoc = iframeSelector ? $(iframeSelector).get(0)?.contentDocument : document; if (!targetDoc) return; // 生成唯一键 controlKey = controlKey || `wkfe_${manager.instanceCounter++}`; // 清理重复任务 let existingTask = manager.tasks.get(controlKey); if (existingTask) { existingTask.observer.disconnect(); manager.tasks.delete(controlKey); } // 递归扫描shadowRoot function scanShadowDOM(root, selector) { let results = []; try { // 扫描当前根节点 let currentMatches = root.querySelectorAll(selector); results.push(...Array.from(currentMatches)); // 递归扫描shadowRoot let walker = document.createTreeWalker(root, NodeFilter.SHOW_ELEMENT); let node; while (node = walker.nextNode()) { if (node.shadowRoot && node.shadowRoot.mode === 'open') { results.push(...scanShadowDOM(node.shadowRoot, selector)); } } } catch (e) { // 跨域shadowRoot静默跳过 } return results; } // 元素扫描核心逻辑 function scanElements() { // 全局扫描(含shadowRoot) let rawElements = scanShadowDOM(targetDoc.documentElement, selectorElem); let jElements = $(rawElements); let foundActive = false; jElements.each(function () { let jEl = $(this); if (jEl.data(controlKey)) return true; // 跳过已处理元素 let cancel = actionFunction(jEl); if (cancel) { foundActive = true; } else if (bWaitOnce) { jEl.data(controlKey, true); // 标记已处理 } }); // 一次性任务清理 if (bWaitOnce && foundActive) { observer.disconnect(); manager.tasks.delete(controlKey); } } // 创建观察器 let observer = new MutationObserver(scanElements); // 配置观察项(增强shadowRoot监听) observer.observe(targetDoc.documentElement, { childList: true, subtree: true, attributes: true, characterData: true }); // 注册任务 manager.tasks.set(controlKey, { observer: observer, targetDoc: targetDoc }); // 首次扫描 scanElements(); }, } function has(value) { return !["undefined", "null"].includes(typeof value) } let user = base.getSettings(); let rules = [ { // 综合变量 runat: "variable", match: (url) => !url.pathname.includes("buy/checkoutcounter"), condition: () => user.vip == true, action: (res, url) => { if (/\/s\/.*?\?fid=.*?/.test(url.href)) res.self = 1 if (res.userInfo) { if (has(res.userInfo?.username) && has(user.name)) res.userInfo.username = user.name; if (has(res.userInfo?.["photo_url"]) && has(user.photo)) res.userInfo["photo_url"] = user.photo; if (has(res.userInfo?.vip_level) && has(user.level)) res.userInfo.vip_level = user.level; if (has(res.userInfo?.svip10_id) && has(user.vip_id)) res.userInfo.svip10_id = user.vip_id; if (has(res.userInfo?.vip_identity) && user.previous == false) res.userInfo.vip_identity = Number(`${user.svip == true ? 2 : 1}1`); } if (has(user.name)) res.username = user.name; if (has(res.MYNAME) && has(user.name)) res.MYNAME = user.name; if (has(user.photo)) res.photo = user.photo; if (has(res.MYAVATAR) && has(user.photo)) res.MYAVATAR = user.photo; if (user.previous == false) res.is_vip = 1; if (has(res.ISVIP) && user.previous == false) res.ISVIP = 1; if (user.svip == true && user.previous == false) res.is_svip = 1; if (has(res.ISSVIP) && user.svip == true && user.previous == false) res.ISSVIP = 1; if (has(res.ISYEARVIP)) res.ISYEARVIP = 1; if (has(res.is_plus_buy)) res.is_plus_buy = 1; if (has(res.is_vip_v2) && has(user.level) && user.level >= 2) res.is_vip_v2 = 1; if (has(res.is_vip_v2_new) && has(user.level) && user.level >= 2) res.is_vip_v2_new = 1; if (has(res.is_evip) && user.previous == false) res.is_evip = 1; if (has(res.is_mvip) && user.previous == false) res.is_mvip = 1; if (has(res.vipsid) && has(user.vip_id)) { let idStr = String(user.vip_id); let six = idStr.match(/\d{6}/)?.[0]; let all = idStr.replace(/\D/g, ""); let finalId = six ? +six : (all ? +all : 1); res.vipsid = finalId; } if (has(user.vip_id)) res.v10_id = user.vip_id; if (has(user.level)) res.vip_level = user.level; if (has(res.show_vip_ad)) res.show_vip_ad = 0; return res; } }, { // 用户信息 基本 runat: "end", match: (url) => url.pathname.includes("api/user/getinfo"), condition: () => user.vip == true, action: (res, url) => { let key = localStorage.getItem("realNameVerify"); if (key && res.records?.[0] && res.records.some(item => item.uk === Number(key))) { let info = res.records.find(item => item.uk === Number(key)); if (has(info.uname) && has(user.name)) info.uname = user.name; if (has(info.priority_name) && has(user.name)) info.priority_name = user.name; if (has(info.avatar_url) && has(user.photo)) info.avatar_url = user.photo; } return res; } }, { // 登录信息 基本 runat: "end", match: (url) => url.pathname.includes("api/loginStatus"), condition: () => user.vip == true, action: (res, url) => { if (has(res.login_info?.username) && has(user.name)) res.login_info.username = user.name; if (has(res.login_info?.["photo_url"]) && has(user.photo)) res.login_info["photo_url"] = user.photo; if (has(res.login_info?.vip_level) && has(user.level)) res.login_info.vip_level = user.level; if (has(res.login_info?.vip_point) && has(user.point)) res.login_info.vip_point = user.point; if (has(res.login_info?.vip_type)) res.login_info.vip_type = user.svip == true ? "21" : "11"; if (has(res.login_info?.vip_identity)) res.login_info.vip_identity = user.svip == true ? 21 : 11; return res; } }, { // 会员信息 用户 runat: "end", match: (url) => url.pathname.includes("rest/2.0/membership/user") && (!url.searchParams.get("method") || url.searchParams.get("method") == "query"), condition: () => user.vip == true, action: (res, url) => { if (has(res.current_product) && user.previous == false) res.current_product = res.current_product_v2 = { "cluster": "vip", "detail_cluster": user.svip == true ? "svip" : "vip", // 决定会员标签变化 "product_type": user.svip == true ? `vip2_1m_auto` : `vip_1m_auto`, "product_id": "12187135090581539740" } // 决定会员等级变化 if (has(res.level_info)) res.level_info = { ...res.level_info, "current_value": has(user.point) && user.point != "" ? user.point : 0, // 决定所有会员标签 "current_level": has(user.level) && user.level != "" ? user.level : 1, // 决定所有会员标签 "history_value": has(user.point) && user.point != "" ? user.point : 0, "history_level": has(user.level) && user.level != "" ? user.level : 1, "v10_id": user.vip_id, "last_manual_collection_time": 0 } if (has(res.privilege_list)) [...res.privilege_list].forEach(item => { item.free_count = 100 }) // 决定会员剩余时间变化 if (has(res.reminder) && user.previous == false) res.reminder = { ...res.reminder, [user.svip == true ? "svip" : "vip"]: { "leftseconds": ((has(user.endtime) && user.endtime != "" ? user.endtime : 2147483648) * 1000 - Date.now()) / 1000, "nextState": "normal" } } // 其它(已有套餐、之前套餐) if (has(res.product_infos) && user.previous == false) res.product_infos.unshift({ "product_id": "12187135090581539740", "buy_time": "946656000", "start_time": 946656000, "end_time": has(user.endtime) && user.endtime != "" ? user.endtime : 2147483648, "cluster": "vip", "detail_cluster": user.svip == true ? "svip" : "vip", "product_name": user.svip == true ? "svip2_nd" : "vip1_nd", "cur_svip_type": "year", "function_num": 8, "status": 0, "buy_description": "超级SVIP套餐(永久)", "product_description": "自打百度一诞生我就买了!" }) if (has(res.previous_product)) res.previous_product = { "cluster": "vip", "detail_cluster": user.svip == true ? "svip" : "vip", // 决定之前会员标签 "product_type": user.svip == true ? `vip2_1m_auto` : `vip_1m_auto`, "expired_time": 946656000 } if (has(res.previous_product_v2)) res.previous_product_v2 = { "cluster": "vip", "detail_cluster": user.svip == true ? "svip" : "vip", // 决定之前会员标签 "product_type": user.svip == true ? `vip2_1m_auto` : `vip_1m_auto`, "expired_time": 946656000 } return res; } }, { // 会员信息 用户导览 runat: "end", match: (url) => url.pathname.includes("rest/2.0/membership/user") && url.searchParams.get("method") == "guide", condition: () => user.vip == true, action: (res, url) => { if (has(res.status_data) && has(user.endtime)) res.status_data = `${user.svip == true ? "SVIP" : "VIP"}至:${new Date(user.endtime * 1000).toLocaleDateString()}`; if (has(res.status_data) && has(user.endtime)) res.status_data_arr = [`${user.svip == true ? "SVIP" : "VIP"}至:${new Date(user.endtime * 1000).toLocaleDateString()}`]; return res; } }, { // 会员信息 用户信息 runat: "end", match: (url) => url.pathname.includes("rest/2.0/membership/user/info") && url.searchParams.get("method") == "query", condition: () => user.vip == true, action: (res, url) => { if (has(res.user_info?.username) && has(user.name)) res.user_info.username = user.name; if (has(res.user_info?.["photo_url"]) && has(user.photo)) res.user_info["photo_url"] = user.photo; if (has(res.user_info?.phone) && has(user.phone)) res.user_info.phone = Number(user.phone); if (has(res.user_info?.vipsid) && has(user.vip_id)) { let idStr = String(user.vip_id); let six = idStr.match(/\d{6}/)?.[0]; let all = idStr.replace(/\D/g, ""); let finalId = six ? +six : (all ? +all : 1); res.user_info.vipsid = finalId; } if (has(res.user_info?.is_vip) && user.previous == false) res.user_info.is_vip = 1; if (has(res.user_info?.is_svip) && user.svip == true && user.previous == false) res.user_info.is_svip = 1; if (has(res.user_info?.is_mvip) && user.previous == false) res.user_info.is_mvip = 1; if (has(res.user_info?.is_vip_v2) && has(user.level) && user.level >= 2) res.user_info.is_vip_v2 = 1; if (has(res.user_info?.is_vip_v2_new) && has(user.level) && user.level >= 2) res.user_info.is_vip_v2_new = 1; if (has(res.user_info?.is_first_charge)) res.user_info.is_first_charge = 1; if (has(res.user_info?.is_first_charge) && user.svip == true) res.user_info.is_first_charge_svip = 1; if (has(res.user_info?.is_plus_buy)) res.user_info.is_plus_buy = 1; if (has(res.user_info?.plus_buy_hit)) res.user_info.plus_buy_hit = 1; return res; } }, { // 会员信息 手机 runat: "end", match: (url) => url.pathname.includes("rest/2.0/membership/phone") && url.searchParams.get("method") == "query", condition: () => user.vip == true, action: (res, url) => { if (has(res.phone) && has(user.phone)) res.phone = user.phone; if (has(res.phone_enc) && has(user.phone)) res.phone_enc = user.phone; return res; } }, { // 会员信息 等级 runat: "end", match: (url) => url.pathname.includes("rest/2.0/membership/level") && url.searchParams.get("method", "") == "getv10info", condition: () => user.vip == true, action: (res, url) => { if (has(res.data.points) && has(user.point)) res.data.points = user.point; if (has(res.data.svip_expire_time) && has(user.endtime)) res.data.svip_expire_time = user.endtime; if (has(res.data.v10_expire_time) && has(user.endtime)) res.data.v10_expire_time = user.endtime; if (has(res.data.v10_id) && has(user.vip_id)) res.data.v10_id = user.vip_id; console.log(user) if (has(res.data.v10_rank) && has(user.vip_id)) { let idStr = String(user.vip_id); let six = idStr.match(/\d{6}/)?.[0]; let all = idStr.replace(/\D/g, ""); let finalId = six ? +six : (all ? +all : 1); res.data.v10_rank = finalId; } return res; } }, { // 视频信息 runat: "end", match: (url) => url.pathname.includes("api/filemetas"), condition: () => user.vip == true, action: (res, url) => { // 百度网盘不允许非会员用户在线观看 1080P 及以上分辨率,所以在此处将其移除。 if (Array.isArray(res?.info) && res.info.length > 0) { for (let item of res.info) { if (!item || typeof item !== "object") continue; // 从 resolution 解析宽高 let width = item.width || 0, height = item.height || 0; if (item.resolution) { let match = item.resolution.match(/width:(\d+).*?height:(\d+)/i); if (match) { width = parseInt(match[1], 10) || 0; height = parseInt(match[2], 10) || 0; } } // 判断是否高于 720P if (Math.min(width, height) >= 720 || Math.max(width, height) >= 1280) { if (width >= height) { width = 1280, height = 720; // 横屏 } else { width = 720, height = 1280; // 竖屏 } item.width = width; item.height = height; item.resolution = `width:${width},height:${height}`; } } } return res; } }, { // 视频信息 视频流 M3U8 runat: "url", match: (url) => url.pathname.includes("api/streaming") && has(url.searchParams.get("vip")) && url.searchParams.get("type"), condition: () => user.vip == true, action: (url) => { url.searchParams.set("vip", "0"); if (temp.adToken) { url.searchParams.set("adToken", temp.adToken); delete temp.adToken; } return url; } }, { // 视频信息 视频流 M3U8 runat: "end", match: (url) => url.pathname.includes("api/streaming") && has(url.searchParams.get("vip")) && /^M3U8_(AUTO|FLV)/.test(url.searchParams.get("type")), condition: () => user.vip == true, action: (res, url) => { if (typeof res === "object") { temp.adToken = res.adToken; res.adTime = 2; res.ltime = 1; if (user.iLoveAD) { res.adTime = 200; res.ltime = 100; } } return res; } }, { // 视频信息 字幕流 M3U8 runat: "end", match: (url) => url.pathname.includes("api/streaming") && has(url.searchParams.get("vip")) && url.searchParams.get("type") == "M3U8_SUBTITLE_SRT", condition: () => user.vip == true, action: (res, url) => { if (typeof res === "string" && res.startsWith("#EXTM3U") && res.includes("#MEDIA:SUBTITLES")) { function getSubtitles(text) { let regex = /#EXT-X-MEDIA:TYPE=SUBTITLES,[^#]*\n\s*(https?:\/\/[^\s]+)/g; let subtitles = []; let match; while ((match = regex.exec(text)) !== null) { let line = text.substring(match.index, regex.lastIndex).split('\n')[0]; let attrs = Object.fromEntries( line.replace('#EXT-X-MEDIA:', '').split(',').map(pair => { let [k, v = ''] = pair.split('='); return [k.trim(), v.replace(/"/g, '').trim()]; }) ); subtitles.push({ name: attrs.NAME || '', language: attrs.LANGUAGE || '', default: attrs.DEFAULT === 'YES', ai: attrs['AI-SUB'] === 'YES', url: match[1] }); } return subtitles; }; let subtitles = getSubtitles(res); console.log(subtitles); base.waitForKeyElements("div.vjs-subtitle-item", (element) => { if (!element.parent("style.ym").length) element.parent().append(``); let subName = element.find(".vjs-subtitle-item-text").text()?.trim(); let maybeAI = element.find(".vjs-subtitle-item-text").prev(".vjs-subtitle-item-icon").length; if (subName && subtitles.some(i => i.name === subName && i.ai == maybeAI)) { let subtitle = subtitles.find(i => i.name === subName && i.ai == maybeAI); let button = $(``); button.on('click', function (e) { e.preventDefault(); e.stopPropagation(); window.open(subtitle.url, '_blank', 'noopener,noreferrer'); }); element.append(button); } }, true) base.waitForKeyElements(".vp-video__control-bar--video-subtitles-select-item", (element) => { let subName = element.find(".vp-video__control-bar--video-subtitles-select-desc").text()?.trim(); let maybeAI = element.find(".vp-video__control-bar--video-subtitles-select-desc").prev(".vp-img-container").length; if (subName && subtitles.some(i => i.name === subName && i.ai == maybeAI)) { let subtitle = subtitles.find(i => i.name === subName && i.ai == maybeAI); let button = $(``); button.on('click', function (e) { e.preventDefault(); e.stopPropagation(); window.open(subtitle.url, '_blank', 'noopener,noreferrer'); }); if (element.find(".downloadSubtitle").length) element.find(".downloadSubtitle").remove(); element.append(button); } }, true) } return res; } }, { // 界面信息 runat: "end", match: (url) => url.pathname.includes("api/getsyscfg"), condition: () => user.vip == true, action: (res, url) => { // huge_upgrade if (res.huge_upgrade?.cfg_list && has(user.unVerSwitch)) { let hue_config = res.huge_upgrade.cfg_list.find(item => item.node_key === "huge_upgrade_entrance_config"); if (hue_config) { if (has(hue_config.new_nav_show)) hue_config.new_nav_show = (user.unVerSwitch == true ? "0" : "1") if (has(hue_config.new_nav_href)) hue_config.new_nav_href = "https://pan.baidu.com/disk/main" if (has(hue_config.old_nav_show)) hue_config.old_nav_show = (user.unVerSwitch == true ? "0" : "1") if (has(hue_config.old_nav_href)) hue_config.old_nav_href = "https://pan.baidu.com/disk/home?stayAtHome=true" } } // netdisk_docchat_config if (res.netdisk_docchat_config?.cfg_list && user.unAI_Yiduo == true) { let ce_config = res.netdisk_docchat_config.cfg_list.find(item => item.node_key === "chat_enter_config"); ce_config.is_chat_show = "0"; ce_config.is_docview_chat_auto_show = "0"; ce_config.is_docview_chat_show = "0"; ce_config.is_home_need_out_sug = "0"; ce_config.is_main_chat_show = "0"; ce_config.is_videoview_chat_auto_show = "0"; ce_config.is_videoview_chat_show = "0"; } // ai_collect if (res.ai_collect?.cfg_list && user.unAI_Yiduo == true) { let ac_config = res.ai_collect.cfg_list.find(item => item.node_key === "ai_collect_config"); ac_config.is_ai_collect_show = "0"; } // netdisk_docview_config if (res.netdisk_docview_config?.cfg_list) { let dv_type = res.netdisk_docview_config.cfg_list.find(item => item.node_key === "docview_type"); dv_type.auto_search_to_ai = (user.unAI_Yiduo == true ? "0" : "1"); dv_type.use_pfile_for_doc = (user.unPViewer == true ? "0" : "1"); dv_type.use_pfile_for_ppt = (user.unPViewer == true ? "0" : "1"); dv_type.use_pfile_reader = (user.unPViewer == true ? "0" : "1"); dv_type.use_wps_reader = (user.unPViewer == true ? "1" : "0"); } return res; } }, { // 界面信息 视频页 笔记 Tab 展示状态 runat: "end", match: (url) => url.pathname.includes("learn/query/strategy"), condition: () => true, action: (res, url) => { if (Array.isArray(res.data?.module_info) && res.data.module_info.length > 0 && res.data.module_info.some(i => i.module_name === "ainote")) { res.data.module_info.find(i => i.module_name === "ainote").display_strategy.enable = (user.unTab_aiNote !== true); } return res; } }, { // 界面信息 视频页 AI看 Tab 展示状态 runat: "end", match: (url) => url.pathname.includes("learn/query/roiinfo"), condition: () => true, action: (res, url) => { if (Array.isArray(res.data?.file_infos) && res.data?.file_infos.length > 0 && res.data.file_infos.some(i => has(i.roi_status))) { res.data.file_infos.forEach(i => { i.roi_status = user.unTab_aiStudy ? 0 : 1; }); } return res; } }, { // AI看 功能 runat: "end", match: (url) => url.pathname.includes("api/study/kpointinfo"), condition: () => user.vip == true, action: (res, url) => { if (res.data?.left_times != null && Array.isArray(res.data?.display_data) && res.data.display_data.length > 0) { for (const item of res.data.display_data) { if (!["ready", "fail"].includes(item.ai_status)) continue; // 仅当免费次数用完时才修改状态 if (res.data.left_times === 0) { if (item.id === 1) { item.ai_status = "done"; item.content = "您的 AI 看免费使用次数已用尽,此视频无法生成 AI 总结(百度云盘会员青春版)"; } else { item.ai_status = "done_empty"; } } } } return res; } }, { // 界面信息 视频页 课件/文稿 Tab 展示状态 runat: "end", match: (url) => url.pathname.includes("/common") && url.searchParams.get("method") === "status", condition: () => true, action: (res, url) => { // 课件 - aitrans/ppt if (url.pathname.includes("aitrans/ppt") && has(res.data?.roi_status)) { res.data.roi_status = (user.unTab_aiCourseware == true ? 0 : 1); } // 文稿 - richsearch/autosubtitle if (url.pathname.includes("richsearch/autosubtitle") && has(res.res?.roi_status)) { res.res.roi_status = (user.unTab_aiDraft == true ? 0 : 1) } return res; } }, { runat: "start", match: (url) => url.pathname.includes("cloudapp/api/v1/isLocateRight"), condition: () => user.vip == true, action: (res, url) => { res = { "errno": 0, "msg": "success", "result": false, "requestId": "" } return res; } }, { // 广告信息 弹窗 runat: "start", match: (url) => url.pathname.includes("rest/2.0/pcs/adx"), condition: () => user.vip == true, action: (res, url) => { res = { "list": { "tab": { "web-popup": [] }, "version": 0, "home_banner_version": 0 }, "request_id": "" } return res; } }, { // 分析信息 报告 runat: "start", match: (url) => { if (url.pathname.includes("api/report/user")) return true; if (url.pathname.includes("mwb2.gif")) return true; return false }, condition: () => user.unAnalytics == true, action: () => { } }, { // 分析信息 报告 runat: "image", match: (url) => { if (url.host == "mbd.baidu.com" && url.pathname.includes("ztbox")) return true; if (url.host == "hm.baidu.com" && url.pathname.includes("hm.gif")) return true; if (url.host == "update.pan.baidu.com" && url.pathname.includes("statistics")) return true; if (url.pathname.includes("api/analytics")) return true; return false }, condition: () => user.unAnalytics == true, action: () => { } }, ]; /** * 控制台输出 * @author 油小猴 * @author hmjz100 * @description 来自【网盘智能识别助手】,有改动 */ console.log(`%c %c (改)百度网盘会员青春版\n虚拟百度网盘身份信息为会员\n仓库:https://github.com/hmjz100/BaiduNetDiskYouthMember\n版本:${info.version}\n领域:${(window.self !== window.top ? "[iframe] " : "") + (document.title ? (document.title + " (" + location.origin + location.pathname + ")") : temp.url())}`, `background:url(${info.icon}) center center no-repeat;background-size:12px;padding:3px`, `padding:2px`); // 没有什么是 Hook 解决不了的。 var originalFetch = fetch; unsafeWindow.fetch = base.hooker.fetch; temp.hooker.set(fetch); var originalSend = XMLHttpRequest.prototype.send; unsafeWindow.XMLHttpRequest.prototype.send = base.hooker.send; temp.hooker.set(XMLHttpRequest.prototype.send); var originalOpen = XMLHttpRequest.prototype.open; unsafeWindow.XMLHttpRequest.prototype.open = base.hooker.open; temp.hooker.set(XMLHttpRequest.prototype.open); var originalSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader; unsafeWindow.XMLHttpRequest.prototype.setRequestHeader = base.hooker.setRequestHeader; temp.hooker.set(XMLHttpRequest.prototype.setRequestHeader); var originalSendBeacon = navigator.sendBeacon; unsafeWindow.navigator.sendBeacon = base.hooker.sendBeacon; temp.hooker.set(navigator.sendBeacon); var OriginalImage = Image; unsafeWindow.Image = base.hooker.Image; temp.hooker.set(Image); var originalCreateElement = document.createElement; unsafeWindow.document.createElement = base.hooker.createElement; temp.hooker.set(document.createElement); var originalToString = Function.prototype.toString; unsafeWindow.Function.prototype.toString = base.hooker.toString; temp.hooker.set(Function.prototype.toString); var originalToLocaleString = Function.prototype.toLocaleString; unsafeWindow.Function.prototype.toLocaleString = base.hooker.toLocaleString; temp.hooker.set(Function.prototype.toLocaleString); var originalAttachShadow = Element.prototype.attachShadow; unsafeWindow.Element.prototype.attachShadow = base.hooker.attachShadow; temp.hooker.set(Element.prototype.attachShadow); let localsTimer = setInterval(() => { if (!unsafeWindow.locals) return; clearInterval(localsTimer); let url = new URL(location.href); if (rules.some(r => r.match(url) && r.condition() && r.runat === "variable")) { unsafeWindow.locals = base.hooker.universal("locals", url, unsafeWindow.locals, "variable"); } }, 1) let yunDataTimer = setInterval(() => { if (!unsafeWindow.yunData) return; clearInterval(yunDataTimer); let url = new URL(location.href); if (rules.some(r => r.match(url) && r.condition() && r.runat === "variable")) { unsafeWindow.yunData = base.hooker.universal("yunData", url, unsafeWindow.yunData, "variable"); } }, 1) let yunDataRequireTimer = setInterval(() => { if (!unsafeWindow.require) return; clearInterval(yunDataRequireTimer); var originalRequire = unsafeWindow.require; originalRequire.async("pan-center:widget/data/yunData", (yunData) => { let url = new URL(location.href); return base.hooker.universal("require yunData", url, yunData, "variable"); }) /*unsafeWindow.require = function (url) { Object.assign(unsafeWindow.require, originalRequire); if (url.includes("widget/data/yunData")) { let url = new URL(location.href); return base.hooker.universal("require yunData", url, originalRequire.apply(this, arguments), "variable"); } return originalRequire.apply(this, arguments); }*/ }, 1) GM_registerMenuCommand("⚙️ 设置", () => { base.showSettings(); }); base.waitForKeyElements("head:not(:has(.style.listener-ym))", (element) => { element.append(``); }, true) // 这是啥?我不到啊 function idontknow(input) { let charArray = input.split(""); // 这是 Fisher-Yates 洗牌算法的实现 for (let i = charArray.length - 1; i > 0; i--) { let j = Math.floor(Math.random() * (i + 1)); [charArray[i], charArray[j]] = [charArray[j], charArray[i]]; } return charArray.join(""); } })();