// ==UserScript== // @name 123云盘优化助手 // @namespace https://geoisam.github.io // @version 1.0.4 // @description 优化123云盘页面及解除下载限制(需登录),支持电脑端和移动端 // @author geoisam@qq.com // @icon  // @supportURL https://github.com/geoisam/FuckScripts/issues // @require https://cdnjs.cloudflare.com/ajax/libs/sweetalert2/11.22.2/sweetalert2.min.js // @resource swalStyle https://cdnjs.cloudflare.com/ajax/libs/sweetalert2/11.22.2/sweetalert2.min.css // @match http*://*.123pan.com/* // @match http*://*.123pan.cn/* // @match http*://*.123684.com/* // @match http*://*.123865.com/* // @match http*://*.123912.com/* // @run-at document-start // @grant unsafeWindow // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // @grant GM_openInTab // @grant GM_getResourceText // @grant GM_registerMenuCommand // ==/UserScript== (function () { "use strict"; const pjs = { option: [{ name: "download_limit_remove", value: true }, { name: "display_svip_block", value: true }, { name: "popup_payment_hidden", value: true }, { name: "video_speed_diy", value: [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2, 2.5, 3, 5] }], getValue(name, value) { return GM_getValue(name, value) }, setValue(name, value) { GM_setValue(name, value) } } const main = { initValue() { pjs.option.forEach((v) => { pjs.getValue(v.name) === undefined && pjs.setValue(v.name, v.value) }) }, addPluginStyle() { GM_addStyle(` .swal2-title { margin-bottom: 1.25em !important;} .pjs-popup { font-size: 14px !important;font-weight: bold !important;} .pjs-setting-label { display: flex;align-items: center;justify-content: space-between;padding: 12px 0;} .pjs-setting-label input[type="checkbox"] { position: absolute;opacity: 0;width: 0;height: 0;} .pjs-setting-label input[type="text"] { flex: 1; padding: 8px 10px; border: 1px solid; border-radius: 5px; font-size: 14px } .pjs-setting-checkbox { position: relative;display: inline-block;width: 48px;height: 26px;background-color: #e0e0e0;border-radius: 19px;transition: background-color 0.3s;} .pjs-setting-checkbox::before { content: "";position: absolute;left: 2px;top: 2px;width: 22px;height: 22px;background-color: #fff;border-radius: 50%;box-shadow: 0 2px 4px rgba(0,0,0,0.3);transition: transform 0.3s;} .pjs-setting-label input:checked + .pjs-setting-checkbox { background-color: #7066e0;} .pjs-setting-label input:checked + .pjs-setting-checkbox::before { transform: translateX(22px);} `) GM_addStyle(GM_getResourceText("swalStyle")) }, registerMenuCommand() { GM_registerMenuCommand("⚙️ 设置", () => { let dom = `
` Swal.fire({ title: "123pan Config", html: dom, showCloseButton: true, confirmButtonText: "保存", footer: `
✨ 助手免费开源 谨防上当受骗 ✨
`, customClass: { popup: "pjs-popup", }, }).then((res) => { if (res.isConfirmed) { pjs.option.forEach((v) => { pjs.setValue(v.name, v.value) }) history.go(0) } }) function isValidFormat(input) { if (!input) return "视频倍速列表输入内容为空" if (/(^,+|,+$|,,+)/.test(input)) return "视频倍速列表含有多余逗号" const inputs = input.split(",") const numbers = inputs.map(num => num.trim()) for (let num of numbers) { if (isNaN(num)) return "视频倍速列表含有非法字符" if (parseFloat(num) < 0.1 || parseFloat(num) > 32) return "视频倍速范围: 0.1 ≤ 倍速 ≤ 32" if (!(/^(0|[1-9]\d*)(\.\d{1,2})?$/.test(num))) return "视频倍速列表内容格式错误" } if (new Set(inputs.map(Number)).size !== numbers.length) return "视频倍速列表含有重复内容" return true } document.querySelector("#S-Speed").addEventListener("input", (e) => { const isOKtext = isValidFormat(e.currentTarget.value) const isOKobj = document.querySelector("#swal2-validation-message") if (isOKtext == true) { isOKobj.style.display = "none" } else { isOKobj.innerHTML = isOKtext isOKobj.style.display = "flex" } }) document.querySelector("#S-Download").addEventListener("change", (e) => { const targetItem = pjs.option.find(item => item.name == "download_limit_remove") targetItem.value = e.currentTarget.checked }) document.querySelector("#S-SVIP").addEventListener("change", (e) => { const targetItem = pjs.option.find(item => item.name == "display_svip_block") targetItem.value = e.currentTarget.checked }) document.querySelector("#S-Payment").addEventListener("change", (e) => { const targetItem = pjs.option.find(item => item.name == "popup_payment_hidden") targetItem.value = e.currentTarget.checked }) document.querySelector("#S-Speed").addEventListener("change", (e) => { const targetItem = pjs.option.find(item => item.name == "video_speed_diy") const speedArr = e.currentTarget.value if (isValidFormat(speedArr) == true) targetItem.value = speedArr.split(",").map(Number) }) }) }, autoGoGoGo() { const isMobile = /Android|webOS|iPhone|iPad|iPod|Opera Mini|Mobile/i.test(navigator.userAgent) const optionNames = pjs.option.map(item => item.name) optionNames.forEach(name => { const targetItem = pjs.option.find(item => item.name === name) if (targetItem) targetItem.value = pjs.getValue(name) }) GM_addStyle(` .advBanner, [class*="banner_guide"], [class*="adv-container"], [class*="banner-container"], [class*="header-waning-tips"], .btn_reward, .ant-carousel, .activity-box, #top_container, .banner_all_wrap, #banner_container, .verifyBox img.default-img-title, .uppy-Dashboard-slowSpeed-banner, .video-new-user-tips:has(.open-vip), .ant-modal-content .download-msgInfo-phone, .bg_svip_block_ads { display: none !important; } .contentBorder .conter { margin-bottom: 0 !important; } .appBottomBtn.banner-bottom { bottom: 0 !important; } .webbody.svip-body,.backgroundImage,.verifyBox { background-image: none !important; } `) if (!isMobile) { GM_addStyle(` .verifyBox .leftCard { width: 100% !important; } .contentBorder { height: calc(-64px + 100vh) !important;width: 100% !important;max-width: 100% !important;margin: auto !important;} .contentBorder > div:not(.cleanNet) { width: 75%; } .web-body .content { width: 100% !important;margin: auto !important;flex-direction: unset !important;justify-content: center; } .content-layout-page { width: 75%; } .content-layout:has(.single-file-sharing-container) { margin: 64px auto 0; } `) } const originalAtob = atob const originalFetch = fetch const originalOpen = XMLHttpRequest.prototype.open const originalSend = XMLHttpRequest.prototype.send const originalSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader const fuckRules = [ { runat: "end", match: (url) => url.pathname.includes("/api/user/info"), action: (res, url) => { if (pjs.getValue("display_svip_block")) { const timestamp = 253402185600 const standardTime = new Date(timestamp * 1000) res.data.Vip = true res.data.GrowSpaceAddCount = 128 res.data.IsShowAdvertisement = false res.data.VipExpire = standardTime.toLocaleString() res.data.VipLevel = res.data.UserVipDetail.VipCode = 2 res.data.UserVipDetailInfos = res.data.UserVipDetail.UserVipDetailInfos = [ { VipDesc: "SVIP会员", TimeDesc: ` ${standardTime.toLocaleDateString()} 到期`, IsUse: standardTime >= new Date(), endTime: timestamp, EndTime: timestamp, StartTime: 1638288000 } ] } return res } }, { runat: "start", match: (url) => url.pathname.includes("/user/report/info"), action: (res, url) => { if (pjs.getValue("display_svip_block")) { res.data.vipType = 2 } return res } }, { runat: "header", match: (url) => ["/file/download_info", "/file/batch_download_info", "/share/download/info", "/file/batch_download_share_info"].some(path => url.pathname.includes(path)), action: (res, url) => { if (pjs.getValue("download_limit_remove")) { res.platform = Math.random() < 0.5 ? "android" : "ios" } return res } }, { runat: "end", match: (url) => ["/file/download_info", "/file/batch_download_info", "/share/download/info", "/file/batch_download_share_info"].some(path => url.pathname.includes(path)), action: (res, url) => { if (pjs.getValue("popup_payment_hidden")) { if (res?.code === 5113 || res?.code === 5114 || res?.message?.includes("下载流量已超出")) { if (url.pathname.includes("batch_download")) { res = { code: 400, message: "请勿多选文件/文件夹下载!已为您拦截付费弹窗", data: null } } else { res = { code: 400, message: "您今日下载流量已超出限制!已为您拦截付费弹窗", data: null } } } } if (res.data && pjs.getValue("download_limit_remove") && (res.data.DownloadUrl || res.data.DownloadURL)) { const origKey = res.data.DownloadUrl ? "DownloadUrl" : "DownloadURL" const origURL = new URL(res.data[origKey]) let finalURL if (origURL.origin.includes("web-pro")) { const params = ((url) => { try { return decodeURIComponent(atob(url)) } catch { return atob(url) } })(origURL.searchParams.get("params")) const directURL = new URL(params, origURL.origin) directURL.searchParams.set("auto_redirect", 0) origURL.searchParams.set("params", btoa(directURL.href)) finalURL = decodeURIComponent(origURL.href) } else { origURL.searchParams.set("auto_redirect", 0) const newURL = new URL("https://web-pro2.123952.com/download-v2/", origURL.origin) newURL.searchParams.set("params", btoa(encodeURI(origURL.href))) newURL.searchParams.set("is_s3", 0) finalURL = decodeURIComponent(newURL.href) } res.data[origKey] = finalURL } return res } }, { runat: "end", match: (url) => url.pathname.includes("/api/video/play/info"), action: (res, url) => { if (res.data?.video_play_info) res.data.video_play_info = res.data.video_play_info.filter(item => item.url !== "") return res } }, { runat: "end", match: (url) => url.pathname.includes("/api/video/play/conf"), action: (res, url) => { res.data.speedList = pjs.getValue("video_speed_diy") res.data.userVideoResolution = "2160p" res.data.vipResolutionList = res.data.timeLimitSpeedList = res.data.vipSpeedList = null return res } }, { runat: "start", match: (url) => ["/web_logs", "/metrics", "/advert", "/r.png"].some(path => url.pathname.includes(path)), action: (res, url) => { res = null return res } }, ] unsafeWindow.fetch = async function (input, init = {}) { let url = typeof input === "string" ? input : input?.url || ""; url = new URL(url, location.origin) if (fuckRules.some(rule => rule.match(url) && rule.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(fuckUniversal("fetch", url, tempHeaders, "header")) } if (fuckRules.some(rule => rule.match(url) && rule.runat === "end")) { try { const response = await originalFetch.apply(this, arguments) const responseText = await response.text() const res = fuckUniversal("fetch", url, responseText, "end") return new Response(res, { status: response.status, statusText: response.statusText, headers: response.headers }) } catch (e) { return originalFetch.apply(this, arguments) } } if (fuckRules.some(rule => rule.match(url) && rule.runat === "start")) { try { const res = fuckUniversal("fetch", url, null, "start") return new Response(res, { status: 200, statusText: "OK", headers: { "Content-Type": "plain/text" } }) } catch (e) { return originalFetch.apply(this, arguments) } } return originalFetch.apply(this, arguments) } unsafeWindow.XMLHttpRequest.prototype.open = function (method, input) { let url = new URL(input, location.origin) this._currentUrl = url this.addEventListener("readystatechange", function () { if (this.readyState === 4) { if (fuckRules.some(rule => rule.match(url) && rule.runat === "end")) { const res = fuckUniversal("XHR", url, this.responseText, "end") Object.defineProperty(this, "responseText", { writable: true, }) Object.defineProperty(this, "response", { writable: true, }) this.response = res this.responseText = res } } }) return originalOpen.apply(this, arguments) } unsafeWindow.XMLHttpRequest.prototype.send = function (data) { const url = this._currentUrl if (this.headers) { for (let [name, value] of Object.entries(this.headers)) { originalSetRequestHeader.call(this, name, value) } } if (fuckRules.find(rule => rule.match(url) && rule.runat === "start")) { try { const res = fuckUniversal("XHR", url, null, "start") Object.defineProperty(this, "responseText", { writable: true, }) Object.defineProperty(this, "response", { writable: true, }) this.response = 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 } catch (e) { return originalSend.apply(this, arguments) } } return originalSend.apply(this, arguments) } unsafeWindow.XMLHttpRequest.prototype.setRequestHeader = function (name, value) { const url = this._currentUrl if (fuckRules.some(rule => rule.match(url) && rule.runat === "header")) { if (!this.headers) this.headers = {} this.headers[name] = value this.headers = fuckUniversal("XHR", url, { ...this.headers }, "header") return } return originalSetRequestHeader.call(this, name, value) } unsafeWindow.atob = function (input) { try { return originalAtob(decodeURIComponent(input)) } catch (e) { return originalAtob(input) } } function fuckUniversal(from, url, data, type) { let res try { res = typeof data === "string" ? JSON.parse(data) : data } catch (e) { res = data } const rule = fuckRules.find(r => r.match(url) && r.runat === type) if (rule) { res = rule.action(res, url) try { if (res !== null && typeof res === "object" && type === "header") { 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 (e) { return } try { if (res !== null && typeof res === "object" && type !== "header") res = JSON.stringify(res) } catch (e) { return } } return res } if (pjs.getValue("popup_payment_hidden")) { GM_addStyle(`.loginModal-footer.payment-footer,.download-pay-footer { display: none !important; }`) } }, init() { this.initValue() this.addPluginStyle() this.registerMenuCommand() this.autoGoGoGo() } } main.init() })()