// ==UserScript== // @name 123云盘解锁 // @author QingJ // @namespace https://github.com/QingJ01/123pan_unlock // @version 1.2.0 // @description 专业的123云盘增强脚本 - 完美解锁会员功能、突破下载限制、去广告、支持自定义用户信息。整合秒传链接功能,支持生成和保存秒传文件,快速分享和保存文件。界面精美,功能强大,让你的123云盘体验更美好! // @contributor Baoqing、Chaofan、lipkiat - 123FastLink秒传功能核心贡献者 // @contributor hmjz100 - 借鉴了部分适配代码 // @license Apache Licence 2 // @icon  // @match *://*.123pan.com/* // @match *://*.123pan.cn/* // @match *://*.123684.com/* // @match *://*.123865.com/* // @match *://*.123952.com/* // @match *://*.123912.com/* // @grant GM_setValue // @grant GM_getValue // @grant unsafeWindow // @grant GM_info // @grant GM_registerMenuCommand // @grant GM_setClipboard // @grant GM_unregisterMenuCommand // @grant GM_xmlhttpRequest // @run-at document-start // ==/UserScript== (function () { 'use strict'; // 检测unsafeWindow if (typeof (unsafeWindow) === 'undefined') window.unsafeWindow = window; // 配置验证和加载函数 function loadConfig() { const defaultConfig = { vip: 1, svip: 1, pvip: 0, ad: 1, name: "QingJ", photo: "http://q.qlogo.cn/headimg_dl?dst_uin=2903609300&spec=640&img_type=jpg", mail: "", phone: "", id: "", level: 128, endtime: 253402185600, debug: 0 }; const config = {}; for (const [key, defaultValue] of Object.entries(defaultConfig)) { let value = GM_getValue(key, defaultValue); // 类型验证 if (typeof value !== typeof defaultValue) { console.warn(`[123云盘解锁] 配置项 ${key} 类型错误,使用默认值`); value = defaultValue; } // 数值范围验证 if (key === 'level' && (typeof value !== 'number' || value < 0 || value > 128)) { console.warn(`[123云盘解锁] 等级值超出范围,使用默认值`); value = 128; } // VIP逻辑验证 if (key === 'vip' && (value !== 0 && value !== 1)) { value = 1; } if (key === 'svip' && (value !== 0 && value !== 1)) { value = 1; } if (key === 'pvip' && (value !== 0 && value !== 1)) { value = 0; } // URL验证(宽松检查) if (key === 'photo' && value && typeof value === 'string') { if (value.length > 2000) { console.warn(`[123云盘解锁] 头像URL过长,使用默认值`); value = defaultValue; } } config[key] = value; } return config; } // 从存储中读取配置(带验证) var user = loadConfig(); // ============================================================================= // 秒传功能模块 - 整合自 123FastLink // 特别鸣谢:@Baoqing、@Chaofan、@lipkiat // ============================================================================= // 秒传配置 const FastLinkConfig = { enabled: GM_getValue('fastlink_enabled', 1), getFileListPageDelay: 500, getFileInfoBatchSize: 100, getFileInfoDelay: 200, getFolderInfoDelay: 300, saveLinkDelay: 100, mkdirDelay: 100, usesBase62EtagsInExport: true, COMMON_PATH_LINK_PREFIX: "123FLCPV2$" }; // 1. API通信类 class PanApiClient { constructor() { this.host = 'https://' + window.location.host; this.authToken = null; this.loginUuid = null; this.progress = 0; this.progressDesc = ""; } init() { this.authToken = localStorage['authorToken']; this.loginUuid = localStorage['LoginUuid']; } buildURL(path, queryParams) { const queryString = new URLSearchParams(queryParams || {}).toString(); return `${this.host}${path}?${queryString}`; } async sendRequest(method, path, queryParams, body) { const headers = { 'Content-Type': 'application/json;charset=UTF-8', 'Authorization': 'Bearer ' + this.authToken, 'platform': 'web', 'App-Version': '3', 'LoginUuid': this.loginUuid, 'Origin': this.host, 'Referer': document.location.href, }; try { const response = await fetch(this.buildURL(path, queryParams), { method, headers, body, credentials: 'include' }); const data = await response.json(); if (data.code !== 0) { throw new Error(data.message); } return data; } catch (e) { console.error('[123云盘解锁] API请求失败:', e); throw e; } } async getOnePageFileList(parentFileId, page) { const urlParams = { driveId: '0', limit: '100', next: '0', orderBy: 'file_name', orderDirection: 'asc', parentFileId: parentFileId.toString(), trashed: 'false', SearchData: '', Page: page.toString(), OnlyLookAbnormalFile: '0', event: 'homeListFile', operateType: '1', inDirectSpace: 'false' }; const data = await this.sendRequest("GET", "/b/api/file/list/new", urlParams); return { data: { InfoList: data.data.InfoList }, total: data.data.Total }; } async getFileList(parentFileId) { let InfoList = []; this.progress = 0; this.progressDesc = `获取文件列表 ID:${parentFileId}`; const info = await this.getOnePageFileList(parentFileId, 1); InfoList.push(...info.data.InfoList); const total = info.total; if (total > 100) { const times = Math.ceil(total / 100); for (let i = 2; i < times + 1; i++) { this.progress = Math.ceil((i / times) * 100); const pageInfo = await this.getOnePageFileList(parentFileId, i); InfoList.push(...pageInfo.data.InfoList); await new Promise(resolve => setTimeout(resolve, FastLinkConfig.getFileListPageDelay)); } } this.progress = 100; return { data: { InfoList }, total: total }; } async getFileInfo(idList) { const fileIdList = idList.map(fileId => ({ fileId })); const data = await this.sendRequest("POST", "/b/api/file/info", {}, JSON.stringify({ fileIdList })); return { data: { InfoList: data.data.infoList } }; } async uploadRequest(fileInfo) { try { const response = await this.sendRequest('POST', '/b/api/file/upload_request', {}, JSON.stringify({ ...fileInfo, RequestSource: null })); const reuse = response['data']['Reuse']; if (response['code'] !== 0) { return [false, response['message']]; } if (!reuse) { return [false, "未能实现秒传"]; } else { return [true, null]; } } catch (error) { console.error('[123云盘解锁] 上传请求失败:', error); return [false, '请求失败']; } } async getParentFileId() { const homeFilePath = JSON.parse(sessionStorage['filePath'])['homeFilePath']; const parentFileId = (homeFilePath[homeFilePath.length - 1] || 0); return parentFileId.toString(); } async getFile(fileInfo, parentFileId) { if (!parentFileId) { parentFileId = await this.getParentFileId(); } return await this.uploadRequest({ driveId: 0, etag: fileInfo.etag, fileName: fileInfo.fileName, parentFileId, size: fileInfo.size, type: 0, duplicate: 1 }); } async mkdir(parentFileId, folderName = "New Folder") { let folderFileId = null; try { const response = await this.sendRequest('POST', '/b/api/file/upload_request', {}, JSON.stringify({ driveId: 0, etag: "", fileName: folderName, parentFileId, size: 0, type: 1, duplicate: 1, NotReuse: true, event: "newCreateFolder", operateType: 1, RequestSource: null })); folderFileId = response['data']['Info']['FileId']; } catch (error) { console.error('[123云盘解锁] 创建文件夹失败:', error); return { 'folderFileId': null, 'folderName': folderName, 'success': false }; } return { 'folderFileId': folderFileId, 'folderName': folderName, 'success': true }; } } // 2. 选中文件管理类 class TableRowSelector { constructor() { this.selectedRowKeys = []; this.unselectedRowKeys = []; this.isSelectAll = false; this._inited = false; } init() { if (this._inited) return; this._inited = true; const originalCreateElement = document.createElement; const self = this; document.createElement = function (tagName, options) { const element = originalCreateElement.call(document, tagName, options); const observer = new MutationObserver(() => { if (element.classList.contains('ant-table-row') && element.classList.contains('ant-table-row-level-0')) { const input = element.querySelector('input'); if (input) { input.addEventListener('click', function () { const rowKey = element.getAttribute('data-row-key'); if (self.isSelectAll) { if (!this.checked) { if (!self.unselectedRowKeys.includes(rowKey)) { self.unselectedRowKeys.push(rowKey); } } else { const idx = self.unselectedRowKeys.indexOf(rowKey); if (idx > -1) { self.unselectedRowKeys.splice(idx, 1); } } } else { if (this.checked) { if (!self.selectedRowKeys.includes(rowKey)) { self.selectedRowKeys.push(rowKey); } } else { const idx = self.selectedRowKeys.indexOf(rowKey); if (idx > -1) { self.selectedRowKeys.splice(idx, 1); } } } }); } observer.disconnect(); } else if (element.classList.contains('ant-checkbox-input') && element.getAttribute('aria-label') === 'Select all') { if (!(element.parentElement.classList.contains('ant-checkbox-indeterminate') || element.parentElement.classList.contains('ant-checkbox-checked'))) { self.unselectedRowKeys = []; self.selectedRowKeys = []; self.isSelectAll = false; } self._bindSelectAllEvent(element); } else if (element.classList.contains('ant-btn') && element.classList.contains('ant-btn-link')) { element.addEventListener('click', function () { self.selectedRowKeys = []; self.unselectedRowKeys = []; self.isSelectAll = false; }); } }); observer.observe(element, { attributes: true, attributeFilter: ['class', 'aria-label'] }); return element; }; } _bindSelectAllEvent(checkbox) { if (checkbox.dataset.selectAllBound) return; checkbox.dataset.selectAllBound = 'true'; checkbox.addEventListener('click', () => { if (checkbox.checked) { this.isSelectAll = true; this.unselectedRowKeys = []; this.selectedRowKeys = []; } else { this.isSelectAll = false; this.selectedRowKeys = []; this.unselectedRowKeys = []; } }); } getSelection() { return { isSelectAll: this.isSelectAll, selectedRowKeys: [...this.selectedRowKeys], unselectedRowKeys: [...this.unselectedRowKeys] }; } } // 3. 秒传链接管理类 class ShareLinkManager { constructor(apiClient) { this.apiClient = apiClient; this.progress = 0; this.progressDesc = ""; this.taskCancel = false; this.fileInfoList = []; this.commonPath = ""; } async _getAllFileInfoByFolderId(parentFileId, folderName = '', total) { this.progressDesc = `正在扫描文件夹:${folderName}`; let progress = this.progress; const progressUpdater = setInterval(() => { this.progress = progress + this.apiClient.progress / total; this.progressDesc = this.apiClient.progressDesc; if (this.progress > 100) { clearInterval(progressUpdater); } }, 500); const allFileInfoList = (await this.apiClient.getFileList(parentFileId)).data.InfoList.map(file => ({ fileName: file.FileName, etag: file.Etag, size: file.Size, type: file.Type, fileId: file.FileId })); clearInterval(progressUpdater); const fileInfo = allFileInfoList.filter(file => file.type !== 1); fileInfo.forEach(file => { file.path = folderName + file.fileName; }); this.fileInfoList.push(...fileInfo); const directoryFileInfo = allFileInfoList.filter(file => file.type === 1); for (const folder of directoryFileInfo) { await new Promise(resolve => setTimeout(resolve, FastLinkConfig.getFolderInfoDelay)); if (this.taskCancel) { this.progressDesc = "任务已取消"; return; } await this._getAllFileInfoByFolderId(folder.fileId, folderName + folder.fileName + "/", total * directoryFileInfo.length); } this.progress = progress + 100 / total; } async _getFileInfoBatch(idList) { const total = idList.length; let completed = 0; let allFileInfo = []; for (let i = 0; i < total; i += FastLinkConfig.getFileInfoBatchSize) { const batch = idList.slice(i, i + FastLinkConfig.getFileInfoBatchSize); try { const response = await this.apiClient.getFileInfo(batch); allFileInfo = allFileInfo.concat(response.data.InfoList || []); } catch (e) { console.error('[123云盘解锁] 获取文件信息失败:', e); } completed += batch.length; this.progress = Math.round((completed / total) * 100 - 1); this.progressDesc = `正在获取文件信息... (${completed} / ${total})`; await new Promise(resolve => setTimeout(resolve, FastLinkConfig.getFileInfoDelay)); } return allFileInfo.map(file => ({ fileName: file.FileName, etag: file.Etag, size: file.Size, type: file.Type, fileId: file.FileId })); } async _getCommonPath() { if (!this.fileInfoList || this.fileInfoList.length === 0) return ''; const pathArrays = this.fileInfoList.map(file => { const path = file.path || ''; const lastSlashIndex = path.lastIndexOf('/'); return lastSlashIndex === -1 ? [] : path.substring(0, lastSlashIndex).split('/'); }); let commonPrefix = []; const firstPath = pathArrays[0]; for (let i = 0; i < firstPath.length; i++) { const currentComponent = firstPath[i]; const allMatch = pathArrays.every(pathArray => pathArray.length > i && pathArray[i] === currentComponent); if (allMatch) { commonPrefix.push(currentComponent); } else { break; } } const commonPath = commonPrefix.length > 0 ? commonPrefix.join('/') + '/' : ''; this.commonPath = commonPath; return commonPath; } async _getSelectedFilesInfo(fileSelectionDetails) { this.fileInfoList = []; if (!fileSelectionDetails.isSelectAll && fileSelectionDetails.selectedRowKeys.length === 0) { return false; } let fileSelectFolderInfoList = []; if (fileSelectionDetails.isSelectAll) { this.progress = 10; this.progressDesc = "正在递归获取选择的文件..." let allFileInfo = (await this.apiClient.getFileList(await this.apiClient.getParentFileId())).data.InfoList.map(file => ({ fileName: file.FileName, etag: file.Etag, size: file.Size, type: file.Type, fileId: file.FileId })); let fileInfo = allFileInfo.filter(file => file.type !== 1); fileInfo.filter(file => !fileSelectionDetails.unselectedRowKeys.includes(file.fileId.toString())).forEach(file => { file.path = file.fileName; }); this.fileInfoList.push(...fileInfo); fileSelectFolderInfoList = allFileInfo.filter(file => file.type === 1).filter(file => !fileSelectionDetails.unselectedRowKeys.includes(file.fileId.toString())); } else { let fileSelectIdList = fileSelectionDetails.selectedRowKeys; if (!fileSelectIdList.length) { this.progress = 100; this.progressDesc = "未选择文件"; return false; } const allFileInfo = await this._getFileInfoBatch(fileSelectIdList); const fileInfo = allFileInfo.filter(info => info.type !== 1); fileInfo.forEach(file => { file.path = file.fileName; }); this.fileInfoList.push(...fileInfo); fileSelectFolderInfoList = allFileInfo.filter(info => info.type === 1); } for (let i = 0; i < fileSelectFolderInfoList.length; i++) { const folderInfo = fileSelectFolderInfoList[i]; this.progress = Math.round((i / fileSelectFolderInfoList.length) * 100); await new Promise(resolve => setTimeout(resolve, FastLinkConfig.getFolderInfoDelay)); if (this.taskCancel) { this.progressDesc = "任务已取消"; return true; } await this._getAllFileInfoByFolderId(folderInfo.fileId, folderInfo.fileName + "/", fileSelectFolderInfoList.length); } const commonPath = await this._getCommonPath(); if (commonPath) { this.fileInfoList.forEach(info => { info.path = info.path.slice(commonPath.length); }); } return true; } async generateShareLink(fileSelectionDetails) { this.progress = 0; this.progressDesc = "准备获取文件信息..."; const result = await this._getSelectedFilesInfo(fileSelectionDetails); if (!result) return ''; this.progressDesc = "秒传链接生成完成"; return this.buildShareLink(this.fileInfoList, this.commonPath); } buildShareLink(fileInfoList, commonPath) { const shareLinkFileInfo = fileInfoList.map(info => { return [FastLinkConfig.usesBase62EtagsInExport ? this._hexToBase62(info.etag) : info.etag, info.size, info.path.replace(/[%#$]/g, '')].join('#'); }).filter(Boolean).join('$'); const shareLink = `${FastLinkConfig.COMMON_PATH_LINK_PREFIX}${commonPath}%${shareLinkFileInfo}`; return shareLink; } _parseShareLink(shareLink, InputUsesBase62 = true) { let commonPath = ''; let shareFileInfo = ''; if (shareLink.slice(0, 4) === "123F") { const commonPathLinkPrefix = shareLink.split('$')[0]; shareLink = shareLink.replace(`${commonPathLinkPrefix}$`, ''); if (commonPathLinkPrefix + "$" === FastLinkConfig.COMMON_PATH_LINK_PREFIX) { commonPath = shareLink.split('%')[0]; shareFileInfo = shareLink.replace(`${commonPath}%`, ''); } else { return null; } } else { shareFileInfo = shareLink; InputUsesBase62 = false; } const shareLinkList = Array.from(shareFileInfo.replace(/\r?\n/g, '$').split('$')); this.commonPath = commonPath; return shareLinkList.map(singleShareLink => { const singleFileInfoList = singleShareLink.split('#'); if (singleFileInfoList.length < 3) return null; return { etag: InputUsesBase62 ? this._base62ToHex(singleFileInfoList[0]) : singleFileInfoList[0], size: singleFileInfoList[1], path: singleFileInfoList[2], fileName: singleFileInfoList[2].split('/').pop() }; }).filter(Boolean); } async _makeDirForFiles(shareFileList) { const total = shareFileList.length; this.progressDesc = `正在创建文件夹...`; let folder = {}; const rootFolderId = await this.apiClient.getParentFileId(); if (this.commonPath) { const commonPathParts = this.commonPath.split('/').filter(part => part !== ''); let currentParentId = rootFolderId; for (let i = 0; i < commonPathParts.length; i++) { const currentPath = commonPathParts.slice(0, i + 1).join('/'); const folderName = commonPathParts[i]; if (!folder[currentPath]) { const newFolder = await this.apiClient.mkdir(currentParentId, folderName); await new Promise(resolve => setTimeout(resolve, FastLinkConfig.mkdirDelay)); folder[currentPath] = newFolder.folderFileId; } currentParentId = folder[currentPath]; } } else { folder[''] = rootFolderId; } for (let i = 0; i < shareFileList.length; i++) { const item = shareFileList[i]; const itemPath = item.path.split('/').slice(0, -1); let nowParentFolderId = folder[this.commonPath.slice(0, -1)] || rootFolderId; for (let i = 0; i < itemPath.length; i++) { const path = itemPath.slice(0, i + 1).join('/'); if (!folder[path]) { const newFolderID = await this.apiClient.mkdir(nowParentFolderId, itemPath[i]); await new Promise(resolve => setTimeout(resolve, FastLinkConfig.mkdirDelay)); folder[path] = newFolderID.folderFileId; nowParentFolderId = newFolderID.folderFileId; } else { nowParentFolderId = folder[path]; } if (this.taskCancel) { this.progressDesc = "任务已取消"; return shareFileList; } } shareFileList[i].parentFolderId = nowParentFolderId; this.progress = Math.round((i / total) * 100); this.progressDesc = `正在创建文件夹... (${i + 1} / ${total})`; } return shareFileList; } async _saveFileList(shareFileList) { let completed = 0; let successList = []; let failedList = []; const total = shareFileList.length; for (let i = 0; i < shareFileList.length; i++) { if (this.taskCancel) { this.progressDesc = "任务已取消"; break; } const fileInfo = shareFileList[i]; if (i > 0) { await new Promise(resolve => setTimeout(resolve, FastLinkConfig.saveLinkDelay)); } const reuse = await this.apiClient.getFile({ etag: fileInfo.etag, size: fileInfo.size, fileName: fileInfo.fileName }, fileInfo.parentFolderId); if (reuse[0]) { successList.push(fileInfo); } else { fileInfo.error = reuse[1]; failedList.push(fileInfo); } completed++; this.progress = Math.round((completed / total) * 100); this.progressDesc = `正在保存第 ${completed} / ${total} 个文件...`; } return { success: successList, failed: failedList, commonPath: this.commonPath }; } async saveShareLink(content) { let saveResult = { success: [], failed: [] }; try { const jsonData = this.safeParse(content); if (jsonData && this.validateJson(jsonData)) { saveResult = await this.saveJsonShareLink(jsonData); } else { const shareFileList = this._parseShareLink(content); if (!shareFileList) return { success: [], failed: [] }; saveResult = await this._saveFileList(await this._makeDirForFiles(shareFileList)); } } catch (error) { console.error('[123云盘解锁] 保存失败:', error); saveResult = { success: [], failed: [] }; } return saveResult; } _base62chars() { return '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; } _hexToBase62(hex) { if (!hex) return ''; let num = BigInt('0x' + hex); if (num === 0n) return '0'; let chars = []; const base62 = this._base62chars(); while (num > 0n) { chars.push(base62[Number(num % 62n)]); num = num / 62n; } return chars.reverse().join(''); } _base62ToHex(base62) { if (!base62) return ''; const chars = this._base62chars(); let num = 0n; for (let i = 0; i < base62.length; i++) { num = num * 62n + BigInt(chars.indexOf(base62[i])); } let hex = num.toString(16); if (hex.length % 2) hex = '0' + hex; while (hex.length < 32) hex = '0' + hex; return hex; } // JSON相关功能 safeParse(str) { try { return JSON.parse(str); } catch { return null; } } _formatSize(size) { if (size < 1024) return size + ' B'; if (size < 1024 * 1024) return (size / 1024).toFixed(2) + ' KB'; if (size < 1024 * 1024 * 1024) return (size / 1024 / 1024).toFixed(2) + ' MB'; return (size / 1024 / 1024 / 1024).toFixed(2) + ' GB'; } validateJson(json) { return (json && Array.isArray(json.files) && json.files.every(f => f.etag && f.size && f.path)); } shareLinkToJson(shareLink) { const fileInfo = this._parseShareLink(shareLink); if (fileInfo.length === 0) { console.error('[123云盘解锁] 解析秒传链接失败:', shareLink); return { error: '解析秒传链接失败' }; } if (FastLinkConfig.usesBase62EtagsInExport) { fileInfo.forEach(f => { f.etag = this._hexToBase62(f.etag); }); } const totalSize = fileInfo.reduce((sum, f) => sum + Number(f.size), 0); return { scriptVersion: "1.2.0", exportVersion: "1.0", usesBase62EtagsInExport: FastLinkConfig.usesBase62EtagsInExport, commonPath: this.commonPath, totalFilesCount: fileInfo.length, totalSize, formattedTotalSize: this._formatSize(totalSize), files: fileInfo.map(f => ({ etag: f.etag, size: f.size, path: f.path })) }; } _parseJsonShareLink(jsonData) { this.commonPath = jsonData['commonPath'] || ''; const shareFileList = jsonData['files']; if (jsonData['usesBase62EtagsInExport']) { shareFileList.forEach(file => { file.etag = this._base62ToHex(file.etag); }); } shareFileList.forEach(file => { file.fileName = file.path.split('/').pop(); }); return shareFileList; } async saveJsonShareLink(jsonContent) { const shareFileList = this._parseJsonShareLink(jsonContent); return this._saveFileList(await this._makeDirForFiles(shareFileList)); } async retrySaveFailed(FileList) { return this._saveFileList(FileList); } } // 秒传功能初始化 let panApiClient = null; let tableRowSelector = null; let shareLinkManager = null; // 任务队列系统 let taskList = []; let isTaskRunning = false; let taskIdCounter = 0; let currentTask = null; let isProgressMinimized = false; const minimizeWidgetId = 'fastlink-progress-minimize-widget'; function initFastLink() { if (!FastLinkConfig.enabled) return; panApiClient = new PanApiClient(); tableRowSelector = new TableRowSelector(); shareLinkManager = new ShareLinkManager(panApiClient); // 初始化 panApiClient.init(); tableRowSelector.init(); } // ============================================================================= // 秒传功能模块结束 // ============================================================================= // 保存原始方法 const originalXHR = unsafeWindow.XMLHttpRequest; const originalFetch = unsafeWindow.fetch; const originalOpen = XMLHttpRequest.prototype.open; const originalSend = XMLHttpRequest.prototype.send; const originalSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader; // 创建唯一标识符 const requestURLSymbol = Symbol('requestURL'); const modifiedHeadersSymbol = Symbol('modifiedHeaders'); const handlerSymbol = Symbol('readyStateHandler'); // 规则配置 const rules = [ { // 用户信息 runat: "end", match: (url) => url.pathname.includes('api/user/info'), condition: () => user.vip === 1, action: (res) => { if (!res.data) return res; res.data.Vip = true; res.data.VipLevel = user.pvip ? 3 : (user.svip ? 2 : 1); if (user.ad === 1) res.data.IsShowAdvertisement = false; // 确保UserVipDetail存在 if (!res.data.UserVipDetail) { res.data.UserVipDetail = {}; } res.data.UserVipDetail.VipCode = res.data.VipLevel; if (user.pvip === 1) { // 长期会员 res.data.VipExpire = "永久有效"; res.data.UserVipDetail.UserPermanentVIPDetailInfos = [{ VipDesc: "长期VIP会员", TimeDesc: " 永久有效", IsUse: true }]; res.data.UserVipDetailInfos = []; } else if (user.svip === 1) { // 超级会员 let time = new Date(user.endtime * 1000); res.data.VipExpire = time.toLocaleString(); res.data.UserVipDetailInfos = [{ VipDesc: "SVIP 会员", TimeDesc: time.toLocaleDateString() + " 到期", IsUse: time >= new Date() }]; } else { // 普通会员 let time = new Date(user.endtime * 1000); res.data.VipExpire = time.toLocaleString(); res.data.UserVipDetailInfos = [{ VipDesc: "VIP 会员", TimeDesc: time.toLocaleDateString() + " 到期", IsUse: time >= new Date() }]; } if (user.name) res.data.Nickname = user.name; if (user.photo) res.data.HeadImage = user.photo; if (user.mail) res.data.Mail = user.mail; if (user.phone) res.data.Passport = Number(user.phone); if (user.id) res.data.UID = Number(user.id); if (user.level) res.data.GrowSpaceAddCount = Number(user.level); return res; } }, { // 用户报告信息 runat: "end", match: (url) => url.pathname.includes('user/report/info'), condition: () => user.vip === 1, action: (res) => { if (res && res.data) { res.data.vipType = user.pvip ? 3 : (user.svip ? 2 : 1); res.data.vipSub = user.pvip ? 3 : (user.svip ? 2 : 1); res.data.developSub = user.pvip ? 3 : (user.svip ? 2 : 1); } 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)), condition: () => true, action: (headers) => { headers.platform = 'android'; return headers; } }, { // 下载信息处理 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)), condition: () => true, action: (res, url) => { // 处理下载限制错误 if (res?.code === 5113 || res?.code === 5114 || res?.message?.includes("下载流量已超出")) { if (url.pathname.includes("batch_download")) { showFastLinkToast("请勿多选文件!已为您拦截支付下载窗口", 'warning', 3000); return { code: 400, message: "已拦截", data: null }; } else { showFastLinkToast("您今日下载流量已超出限制,已为您拦截支付窗口", 'warning', 3000); return { code: 400, message: "已拦截", data: null }; } } if (res.data && (res.data.DownloadUrl || res.data.DownloadURL)) { // 统一处理下载链接 let origKey = res.data.DownloadUrl ? 'DownloadUrl' : 'DownloadURL'; let origURL = new URL(res.data[origKey]); let finalURL; if (origURL.origin.includes("web-pro")) { let params = (() => { try { return decodeURIComponent(atob(origURL.searchParams.get('params'))); } catch { return atob(origURL.searchParams.get('params')); } })(); let directURL = new URL(params, origURL.origin); directURL.searchParams.set('auto_redirect', 0); origURL.searchParams.set('params', btoa(encodeURI(directURL.href))); finalURL = decodeURIComponent(origURL.href); } else { origURL.searchParams.set('auto_redirect', 0); let 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: "start", match: (url) => url.pathname.includes('web_logs') || url.pathname.includes('metrics'), condition: () => true, action: () => { throw new Error('【123云盘解锁】已屏蔽此数据收集器'); } } ]; // 工具函数 function findMatchingRule(url, phase) { try { return rules.find(rule => rule.match(url) && rule.condition() && rule.runat === phase ); } catch (error) { console.error('[123云盘解锁] 规则匹配失败:', error); if (user.debug) { console.error('错误详情:', { url: url.href, phase }); } return null; } } function processData(data) { if (typeof data === 'string') { try { return JSON.parse(data); } catch { return data; } } return data; } function debugLog(method, phase, url, original, modified) { if (user.debug) { console.log(`[123云盘解锁] ${method} ${phase}`, { url: url.href, original: original, modified: modified }); } } function applyRule(rule, data, url, method, phase) { const originalData = processData(data); let result = rule.action(originalData, url); // 处理header格式化 if (phase === 'header' && result && typeof result === 'object') { const headers = {}; Object.entries(result).forEach(([key, value]) => { const formattedKey = key.toLowerCase() .split('-') .map(word => word.charAt(0).toUpperCase() + word.slice(1)) .join('-'); headers[formattedKey] = value; }); result = headers; } debugLog(method, phase, url, originalData, result); // 非header返回字符串 if (phase !== 'header' && result && typeof result === 'object') { return JSON.stringify(result); } return result; } // 统一错误处理包装函数 function safeApplyRule(rule, data, url, method, phase) { try { return applyRule(rule, data, url, method, phase); } catch (error) { console.error(`[123云盘解锁] 规则执行失败 [${phase}]:`, error); if (user.debug) { console.error('错误详情:', { url: url.href, method: method, phase: phase, stack: error.stack }); } // 返回原始数据以保证功能不中断 return data; } } // 修复后的Fetch拦截(带统一错误处理) unsafeWindow.fetch = async function (input, init = {}) { try { const url = new URL(typeof input === 'string' ? input : input.url, location.origin); // 检查start规则 const startRule = findMatchingRule(url, 'start'); if (startRule) { try { const result = safeApplyRule(startRule, null, url, 'fetch', 'start'); return new Response(result, { status: 200, statusText: 'OK', headers: { 'Content-Type': 'application/json' } }); } catch (error) { console.warn('[123云盘解锁] fetch start错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } // 检查header规则 const headerRule = findMatchingRule(url, 'header'); if (headerRule) { try { if (!init.headers) init.headers = {}; let headers = {}; if (init.headers instanceof Headers) { init.headers.forEach((value, key) => headers[key] = value); } else { headers = { ...init.headers }; } const modifiedHeaders = safeApplyRule(headerRule, headers, url, 'fetch', 'header'); init.headers = new Headers(modifiedHeaders); } catch (error) { console.warn('[123云盘解锁] fetch header错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } // 执行原始请求 const response = await originalFetch.call(this, input, init); // 检查end规则 const endRule = findMatchingRule(url, 'end'); if (endRule) { try { const responseText = await response.clone().text(); const modifiedResponse = safeApplyRule(endRule, responseText, url, 'fetch', 'end'); return new Response(modifiedResponse, { status: response.status, statusText: response.statusText, headers: response.headers }); } catch (error) { console.warn('[123云盘解锁] fetch end错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } return response; } catch (error) { console.error('[123云盘解锁] fetch拦截失败:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } // 失败时调用原始fetch return originalFetch.call(this, input, init); } }; // 修复后的XMLHttpRequest拦截(修复内存泄漏) XMLHttpRequest.prototype.open = function (method, url, ...args) { try { const fullUrl = new URL(url, location.origin); this[requestURLSymbol] = fullUrl; // 移除旧的事件监听器(如果存在) if (this[handlerSymbol]) { this.removeEventListener('readystatechange', this[handlerSymbol]); } // 使用箭头函数保持this上下文 const handleStateChange = () => { if (this.readyState === 4) { const endRule = findMatchingRule(fullUrl, 'end'); if (endRule) { try { const modifiedResponse = safeApplyRule( endRule, this.responseText, fullUrl, 'XHR', 'end' ); Object.defineProperty(this, 'responseText', { value: modifiedResponse, writable: false, configurable: true }); Object.defineProperty(this, 'response', { value: modifiedResponse, writable: false, configurable: true }); } catch (error) { console.warn('[123云盘解锁] XHR响应处理错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } } }; // 保存事件处理器引用,以便后续移除 this[handlerSymbol] = handleStateChange; this.addEventListener('readystatechange', handleStateChange); return originalOpen.call(this, method, url, ...args); } catch (error) { console.error('[123云盘解锁] XHR.open错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } // 失败时调用原始方法 return originalOpen.call(this, method, url, ...args); } }; XMLHttpRequest.prototype.setRequestHeader = function (name, value) { try { const url = this[requestURLSymbol]; if (!url) return originalSetRequestHeader.call(this, name, value); const headerRule = findMatchingRule(url, 'header'); if (headerRule) { try { if (!this[modifiedHeadersSymbol]) this[modifiedHeadersSymbol] = {}; this[modifiedHeadersSymbol][name] = value; const modifiedHeaders = safeApplyRule(headerRule, this[modifiedHeadersSymbol], url, 'XHR', 'header'); this[modifiedHeadersSymbol] = modifiedHeaders; return; } catch (error) { console.warn('[123云盘解锁] XHR header处理错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } return originalSetRequestHeader.call(this, name, value); } catch (error) { console.error('[123云盘解锁] XHR.setRequestHeader错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } // 失败时调用原始方法 return originalSetRequestHeader.call(this, name, value); } }; XMLHttpRequest.prototype.send = function (data) { try { const url = this[requestURLSymbol]; if (!url) return originalSend.call(this, data); // 应用修改的headers const modifiedHeaders = this[modifiedHeadersSymbol]; if (modifiedHeaders) { try { Object.entries(modifiedHeaders).forEach(([name, value]) => { originalSetRequestHeader.call(this, name, value); }); } catch (error) { console.warn('[123云盘解锁] 应用修改的headers失败:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } // 检查start规则 const startRule = findMatchingRule(url, 'start'); if (startRule) { try { const result = safeApplyRule(startRule, null, url, 'XHR', 'start'); // 设置响应属性 Object.defineProperty(this, 'readyState', { value: 4, configurable: true }); Object.defineProperty(this, 'status', { value: 200, configurable: true }); Object.defineProperty(this, 'responseText', { value: result, configurable: true }); Object.defineProperty(this, 'response', { value: result, configurable: true }); // 触发事件 setTimeout(() => { ['readystatechange', 'load', 'loadend'].forEach(eventType => { try { this.dispatchEvent(new Event(eventType)); const handler = this[`on${eventType}`]; if (typeof handler === 'function') handler.call(this); } catch (error) { console.warn(`[123云盘解锁] 事件错误 ${eventType}:`, error); if (user.debug) { console.error('错误堆栈:', error.stack); } } }); }, 0); return; } catch (error) { console.warn('[123云盘解锁] XHR start处理错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } return originalSend.call(this, data); } catch (error) { console.error('[123云盘解锁] XHR.send错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } // 失败时调用原始方法 return originalSend.call(this, data); } }; // 格式化设置项 const formatSetting = (key, value, comment) => { const item = document.createElement('div'); item.className = 'setting-item'; const content = document.createElement('div'); content.className = 'setting-content'; const keyElement = document.createElement('div'); keyElement.className = 'setting-key'; keyElement.textContent = key; content.appendChild(keyElement); // 判断设置类型 - 修复等级1被误判为开关的问题 const switchKeys = ['VIP状态', 'SVIP显示', '长期会员显示', '广告控制', '秒传功能', '调试模式']; const isSwitch = switchKeys.includes(key) && typeof value === 'number' && (value === 0 || value === 1); const isEditable = ['用户名', '头像', '等级', '过期时间'].includes(key); if (isSwitch) { // 创建开关按钮 const switchContainer = document.createElement('label'); switchContainer.className = 'switch'; const input = document.createElement('input'); input.type = 'checkbox'; input.checked = value === 1; const slider = document.createElement('span'); slider.className = 'slider round'; switchContainer.appendChild(input); switchContainer.appendChild(slider); // 添加点击事件 input.addEventListener('change', () => { let newValue = input.checked ? 1 : 0; // 更新用户配置 switch (key) { case 'VIP状态': user.vip = newValue; GM_setValue('vip', newValue); break; case 'SVIP显示': user.svip = newValue; GM_setValue('svip', newValue); // 如果SVIP关闭,长期会员也应该关闭 if (newValue === 0 && user.pvip === 1) { user.pvip = 0; GM_setValue('pvip', 0); } break; case '长期会员显示': user.pvip = newValue; GM_setValue('pvip', newValue); // 如果长期会员开启,SVIP必须开启 if (newValue === 1 && user.svip === 0) { user.svip = 1; GM_setValue('svip', 1); } break; case '广告控制': user.ad = newValue; GM_setValue('ad', newValue); break; case '秒传功能': FastLinkConfig.enabled = newValue; GM_setValue('fastlink_enabled', newValue); break; case '调试模式': user.debug = newValue; GM_setValue('debug', newValue); break; } // 刷新页面以应用更改 setTimeout(() => location.reload(), 300); }); content.appendChild(switchContainer); } else if (isEditable) { // 创建输入框 const inputContainer = document.createElement('div'); inputContainer.className = 'input-container'; const inputElement = document.createElement('input'); // 根据不同的设置项设置输入框类型和属性 if (key === '等级') { inputElement.type = 'number'; inputElement.min = 0; inputElement.max = 128; inputElement.value = value; } else if (key === '过期时间') { inputElement.type = 'datetime-local'; // 将时间戳转换为datetime-local格式 const date = new Date(value * 1000); const year = date.getFullYear(); const month = String(date.getMonth() + 1).padStart(2, '0'); const day = String(date.getDate()).padStart(2, '0'); const hours = String(date.getHours()).padStart(2, '0'); const minutes = String(date.getMinutes()).padStart(2, '0'); inputElement.value = `${year}-${month}-${day}T${hours}:${minutes}`; } else { inputElement.type = 'text'; inputElement.value = value; } inputElement.className = 'setting-input'; // 添加保存按钮 const saveButton = document.createElement('button'); saveButton.textContent = '保存'; saveButton.className = 'save-btn'; // 保存按钮点击事件 saveButton.addEventListener('click', () => { let newValue = inputElement.value; // 验证和转换不同类型的输入 if (key === '等级') { newValue = parseInt(newValue); if (isNaN(newValue) || newValue < 0 || newValue > 128) { alert('等级必须在 0-128 之间'); return; } } else if (key === '过期时间') { // 将datetime-local格式转换为时间戳 const date = new Date(newValue); if (isNaN(date.getTime())) { alert('请输入有效的日期时间'); return; } newValue = Math.floor(date.getTime() / 1000); } else if (key === '头像' && newValue && !newValue.match(/^https?:\/\/.+/)) { if (!confirm('头像URL似乎不是有效的HTTP/HTTPS地址,是否继续保存?')) { return; } } // 更新配置 switch (key) { case '用户名': user.name = newValue; GM_setValue('name', newValue); break; case '头像': user.photo = newValue; GM_setValue('photo', newValue); break; case '等级': user.level = newValue; GM_setValue('level', newValue); break; case '过期时间': user.endtime = newValue; GM_setValue('endtime', newValue); break; } // 显示保存成功提示 saveButton.textContent = '已保存'; saveButton.classList.add('saved'); setTimeout(() => { saveButton.textContent = '保存'; saveButton.classList.remove('saved'); location.reload(); }, 1500); }); inputContainer.appendChild(inputElement); inputContainer.appendChild(saveButton); content.appendChild(inputContainer); } else { // 非编辑项的显示 const valueElement = document.createElement('div'); valueElement.className = 'setting-value'; valueElement.textContent = key === '过期时间' ? new Date(value * 1000).toLocaleString() : value; content.appendChild(valueElement); } item.appendChild(content); if (comment) { const commentElement = document.createElement('div'); commentElement.className = 'setting-comment'; commentElement.textContent = comment; item.appendChild(commentElement); } return item; }; function createSettingsPanel() { // 检查是否已存在面板 if (document.getElementById('vip-settings-panel')) { return; } // 创建面板容器 const panel = document.createElement('div'); panel.id = 'vip-settings-panel'; panel.className = 'settings-panel'; // 创建标题栏 const header = document.createElement('div'); header.className = 'panel-header'; // 创建标题容器 const titleContainer = document.createElement('div'); titleContainer.className = 'title-container'; const title = document.createElement('h3'); title.textContent = '123云盘脚本设置'; titleContainer.appendChild(title); // 添加GitHub图标 const githubIcon = document.createElement('a'); githubIcon.href = 'https://github.com/QingJ01/123pan_unlock'; githubIcon.target = '_blank'; githubIcon.className = 'github-icon'; githubIcon.innerHTML = ` `; githubIcon.title = '访问GitHub项目'; titleContainer.appendChild(githubIcon); header.appendChild(titleContainer); // 添加关闭按钮 const closeButton = document.createElement('button'); closeButton.className = 'close-btn'; closeButton.innerHTML = '×'; closeButton.addEventListener('click', () => panel.remove()); header.appendChild(closeButton); panel.appendChild(header); // 创建设置列表 const settingsList = document.createElement('div'); settingsList.className = 'settings-list'; // 添加所有设置项 const settings = [ { key: 'VIP状态', value: user.vip, comment: '会员修改总开关' }, { key: 'SVIP显示', value: user.svip, comment: '显示为超级会员 (关闭将自动关闭长期会员)' }, { key: '长期会员显示', value: user.pvip, comment: '显示为长期会员 (开启将自动开启 SVIP 显示)' }, { key: '广告控制', value: user.ad, comment: '关闭广告' }, { key: '秒传功能', value: FastLinkConfig.enabled, comment: '启用秒传链接生成和保存功能' }, { key: '用户名', value: user.name, comment: '自定义用户名(支持中文、英文、数字)' }, { key: '头像', value: user.photo, comment: '自定义头像URL(建议使用HTTPS地址)' }, { key: '等级', value: user.level, comment: '成长容量等级(0-128,数字越大容量越大)' }, { key: '过期时间', value: user.endtime, comment: '会员过期时间(可自定义任意时间)' }, { key: '调试模式', value: user.debug, comment: '调试信息显示级别' } ]; settings.forEach(setting => { settingsList.appendChild(formatSetting(setting.key, setting.value, setting.comment)); }); panel.appendChild(settingsList); // 添加交流群按钮 const groupButton = document.createElement('a'); groupButton.href = 'http://qm.qq.com/cgi-bin/qm/qr?_wv=1027&k=GGU3-kUsPnz1bq-jwN7e8D41yxZ-DyI2&authKey=ujGsFKDnF5zD3j1z9krJR5xHlWWAKHOJV2oarfAgNmqZAl0xmTb45QwsqgYPPF7e&noverify=0&group_code=1035747022'; groupButton.target = '_blank'; groupButton.className = 'group-btn'; groupButton.innerHTML = ` 加入交流群 `; panel.appendChild(groupButton); document.body.appendChild(panel); } // 秒传功能UI函数 function showFastLinkToast(message, type = 'info', duration = 3000) { const toast = document.createElement('div'); toast.className = 'fastlink-toast'; toast.style.cssText = ` position: fixed; top: 20px; right: 20px; background: #fff; color: #333; padding: 12px 20px; border-radius: 8px; box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15); z-index: 10002; font-size: 14px; max-width: 300px; animation: fastlinkToastSlideIn 0.3s ease-out; `; // 根据类型设置边框颜色 const borderColors = { success: '#4CAF50', error: '#f44336', warning: '#ff9800', info: '#2196F3' }; toast.style.borderLeft = `4px solid ${borderColors[type] || borderColors.info}`; toast.textContent = message; document.body.appendChild(toast); setTimeout(() => { toast.style.animation = 'fastlinkToastSlideOut 0.3s ease-out forwards'; setTimeout(() => { if (toast.parentNode) { toast.parentNode.removeChild(toast); } }, 300); }, duration); } function showFastLinkProgress(title, percent, desc) { let modal = document.getElementById('fastlink-progress-modal'); const taskQueueHtml = taskList.length > 0 ? ` - 队列 ${taskList.length}` : ''; if (!modal) { modal = document.createElement('div'); modal.id = 'fastlink-progress-modal'; modal.className = 'fastlink-modal-overlay'; modal.innerHTML = ` `; document.body.appendChild(modal); // 绑定最小化按钮 const minimizeBtn = modal.querySelector('#minimize-progress-btn'); if (minimizeBtn && !minimizeBtn.dataset.bound) { minimizeBtn.dataset.bound = 'true'; minimizeBtn.addEventListener('click', (e) => { e.stopPropagation(); isProgressMinimized = true; modal.remove(); updateMinimizedWidget(title, percent, desc); }); } } else { modal.querySelector('#fastlink-title').textContent = title + taskQueueHtml; modal.querySelector('#fastlink-progress').style.width = percent + '%'; modal.querySelector('#fastlink-percent').textContent = percent + '%'; modal.querySelector('#fastlink-desc').textContent = desc; } } function hideFastLinkProgress() { const modal = document.getElementById('fastlink-progress-modal'); if (modal) modal.remove(); const widget = document.getElementById(minimizeWidgetId); if (widget) widget.remove(); isProgressMinimized = false; } function showFastLinkCopyModal(shareLink) { const fileListHtml = shareLinkManager.fileInfoList.length > 0 ? `
文件列表(共${shareLinkManager.fileInfoList.length}个):
${shareLinkManager.fileInfoList.map(f => `
${f.path}
`).join('')}
` : ''; const modal = document.createElement('div'); modal.className = 'fastlink-modal-overlay'; modal.id = 'fastlink-copy-modal'; modal.innerHTML = ` `; // 事件绑定 modal.querySelector('#close-copy-modal').addEventListener('click', () => modal.remove()); modal.querySelector('#close-modal-btn').addEventListener('click', () => modal.remove()); // 复制按钮和下拉菜单 const dropdown = modal.querySelector('.copy-dropdown'); const dropdownMenu = modal.querySelector('.copy-dropdown-menu'); const mainBtn = modal.querySelector('#copy-main-btn'); mainBtn.addEventListener('click', () => copyContent('text')); mainBtn.addEventListener('mouseenter', () => dropdownMenu.style.display = 'block'); dropdown.addEventListener('mouseleave', () => dropdownMenu.style.display = 'none'); modal.querySelectorAll('.copy-dropdown-item').forEach(item => { item.addEventListener('click', () => { copyContent(item.dataset.type); dropdownMenu.style.display = 'none'; }); }); // 导出JSON modal.querySelector('#export-json-btn').addEventListener('click', () => exportJson()); modal.addEventListener('click', (e) => { if (e.target === modal) modal.remove(); }); document.body.appendChild(modal); function copyContent(type) { const inputField = document.querySelector('#fastlink-copy-text'); if (!inputField) return; let contentToCopy = inputField.value; if (type === 'json') { try { const jsonData = shareLinkManager.shareLinkToJson(contentToCopy); contentToCopy = JSON.stringify(jsonData, null, 2); } catch (error) { showFastLinkToast('转换JSON失败: ' + error.message, 'error'); return; } } navigator.clipboard.writeText(contentToCopy).then(() => { showFastLinkToast(`已成功复制${type === 'json' ? 'JSON' : '纯文本'}到剪贴板 📋`, 'success'); }).catch(err => { showFastLinkToast(`复制失败: ${err.message || '请手动复制内容'}`, 'error'); }); } function exportJson() { const inputField = document.querySelector('#fastlink-copy-text'); if (!inputField) return; const shareLink = inputField.value; if (!shareLink.trim()) { showFastLinkToast('没有内容可导出', 'warning'); return; } try { const jsonData = shareLinkManager.shareLinkToJson(shareLink); const jsonContent = JSON.stringify(jsonData, null, 2); const filename = getExportFilename(shareLink); downloadJsonFile(jsonContent, filename); showFastLinkToast('JSON文件导出成功 📁', 'success'); } catch (error) { showFastLinkToast('导出失败: ' + error.message, 'error'); } } function downloadJsonFile(content, filename) { const blob = new Blob([content], { type: 'application/json' }); const url = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = url; a.download = filename; document.body.appendChild(a); a.click(); document.body.removeChild(a); URL.revokeObjectURL(url); } function getExportFilename(shareLink) { if (shareLinkManager.commonPath) { const commonPath = shareLinkManager.commonPath.replace(/\/$/, ''); return `${commonPath}.json`; } return '秒传链接.json'; } } function showFastLinkInputModal() { const modal = document.createElement('div'); modal.className = 'fastlink-modal-overlay'; modal.id = 'fastlink-input-modal'; modal.innerHTML = ` `; const textarea = modal.querySelector('#fastlink-input-text'); const fileInput = modal.querySelector('#json-file-input'); const selectFileBtn = modal.querySelector('#select-file-btn'); // 文件拖拽功能 textarea.addEventListener('dragover', (e) => { e.preventDefault(); textarea.style.borderColor = '#4CAF50'; textarea.style.background = '#f0f8f0'; }); textarea.addEventListener('dragleave', (e) => { e.preventDefault(); textarea.style.borderColor = ''; textarea.style.background = ''; }); textarea.addEventListener('drop', (e) => { e.preventDefault(); textarea.style.borderColor = ''; textarea.style.background = ''; const files = e.dataTransfer.files; if (files.length > 0) { readJsonFile(files[0], textarea); } }); // 文件选择功能 selectFileBtn.addEventListener('click', () => { fileInput.click(); }); fileInput.addEventListener('change', (e) => { const files = e.target.files; if (files.length > 0) { readJsonFile(files[0], textarea); } }); // 保存按钮 modal.querySelector('#fastlink-save-btn').addEventListener('click', async () => { const content = textarea.value; if (!content.trim()) { showFastLinkToast("请输入秒传链接或导入JSON文件", 'warning'); return; } modal.remove(); await addAndRunTask('save', { content }); }); // 关闭按钮 modal.querySelector('#close-input-modal').addEventListener('click', () => modal.remove()); modal.querySelector('#cancel-input-btn').addEventListener('click', () => modal.remove()); modal.addEventListener('click', (e) => { if (e.target === modal) modal.remove(); }); document.body.appendChild(modal); setTimeout(() => { if (textarea) textarea.focus(); }, 100); function readJsonFile(file, textarea) { if (!file.name.toLowerCase().endsWith('.json')) { showFastLinkToast('请选择JSON文件', 'warning'); return; } const reader = new FileReader(); reader.onload = (e) => { try { const jsonContent = e.target.result; const jsonData = JSON.parse(jsonContent); if (shareLinkManager.validateJson(jsonData)) { textarea.value = jsonContent; showFastLinkToast('JSON文件导入成功 ✅', 'success'); } else { showFastLinkToast('无效的JSON格式', 'error'); } } catch (error) { showFastLinkToast('JSON文件解析失败: ' + error.message, 'error'); } }; reader.readAsText(file); } } function showFastLinkResult(result) { const totalCount = result.success.length + result.failed.length; const successCount = result.success.length; const failedCount = result.failed.length; const failedListHtml = failedCount > 0 ? `
失败文件列表:
${result.failed.map(fileInfo => `
${fileInfo.fileName} ${fileInfo.error ? `(${fileInfo.error})` : ''}
`).join('')}
` : ''; const modal = document.createElement('div'); modal.className = 'fastlink-modal-overlay'; modal.id = 'fastlink-result-modal'; modal.innerHTML = ` `; // 事件绑定 modal.querySelector('#close-result-modal').addEventListener('click', () => modal.remove()); modal.querySelector('#confirm-result-btn').addEventListener('click', () => modal.remove()); // 重试按钮和下拉菜单 if (failedCount > 0) { const dropdown = modal.querySelector('.copy-dropdown'); const dropdownMenu = modal.querySelector('.copy-dropdown-menu'); const retryBtn = modal.querySelector('#retry-btn'); retryBtn.addEventListener('click', () => { modal.remove(); addAndRunTask('retry', { fileList: result.failed }); }); retryBtn.addEventListener('mouseenter', () => dropdownMenu.style.display = 'block'); dropdown.addEventListener('mouseleave', () => dropdownMenu.style.display = 'none'); modal.querySelectorAll('.copy-dropdown-item').forEach(item => { item.addEventListener('click', () => { const action = item.dataset.action; if (action === 'retry') { modal.remove(); addAndRunTask('retry', { fileList: result.failed }); } else if (action === 'export') { const shareLink = shareLinkManager.buildShareLink(result.failed, result.commonPath || ''); modal.remove(); showFastLinkCopyModal(shareLink); } dropdownMenu.style.display = 'none'; }); }); } modal.addEventListener('click', (e) => { if (e.target === modal) modal.remove(); }); document.body.appendChild(modal); } async function executeGenerateShareLink() { if (!shareLinkManager) { console.error('[123云盘解锁] shareLinkManager未初始化'); alert('秒传功能未启用'); return; } if (!tableRowSelector) { console.error('[123云盘解锁] tableRowSelector未初始化'); alert('文件选择器未初始化'); return; } const fileSelection = tableRowSelector.getSelection(); if (!fileSelection.isSelectAll && fileSelection.selectedRowKeys.length === 0) { showFastLinkToast('请先选择文件', 'warning'); return; } shareLinkManager.progress = 0; const poll = setInterval(() => { updateFastLinkProgress("生成秒传链接", shareLinkManager.progress, shareLinkManager.progressDesc); if (shareLinkManager.progress >= 100) { clearInterval(poll); } }, 500); const shareLink = await shareLinkManager.generateShareLink(fileSelection); shareLinkManager.taskCancel = false; clearInterval(poll); hideFastLinkProgress(); if (!shareLink) { showFastLinkToast("没有选择文件", 'warning'); return; } showFastLinkCopyModal(shareLink); } async function executeSaveShareLink(content, retry = false) { if (!shareLinkManager) { alert('秒传功能未启用'); return; } shareLinkManager.progress = 0; const poll = setInterval(() => { updateFastLinkProgress("保存秒传文件", shareLinkManager.progress, shareLinkManager.progressDesc); if (shareLinkManager.progress >= 100) { clearInterval(poll); } }, 100); let result; if (retry) { result = await shareLinkManager.retrySaveFailed(content); } else { result = await shareLinkManager.saveShareLink(content); } shareLinkManager.taskCancel = false; clearInterval(poll); hideFastLinkProgress(); showFastLinkResult(result); // 刷新文件列表 try { const renewButton = document.querySelector('.layout-operate-icon.mfy-tooltip svg'); if (renewButton) { renewButton.click(); } } catch(e) {} } // 任务队列系统 function addAndRunTask(taskType, params = {}) { const taskId = ++taskIdCounter; if (taskType === 'generate') { const fileSelectInfo = tableRowSelector.getSelection(); if (!fileSelectInfo || (!fileSelectInfo.isSelectAll && fileSelectInfo.selectedRowKeys.length === 0)) { showFastLinkToast("请先选择文件", 'warning'); return; } taskList.push({ id: taskId, type: 'generate', params: { fileSelectInfo } }); } else if (taskType === 'save') { taskList.push({ id: taskId, type: 'save', params: { content: params.content } }); } else if (taskType === 'retry') { taskList.push({ id: taskId, type: 'retry', params: { fileList: params.fileList } }); } runNextTask(); } function runNextTask() { if (isTaskRunning) { return; } if (taskList.length === 0) { return; } const task = taskList[0]; currentTask = task; isTaskRunning = true; setTimeout(async () => { try { if (task.type === 'generate') { await executeGenerateShareLink(); } else if (task.type === 'save') { await executeSaveShareLink(task.params.content); } else if (task.type === 'retry') { await executeSaveShareLink(task.params.fileList, true); } } catch (error) { console.error('[123云盘解锁] 任务执行失败:', error); showFastLinkToast('任务执行失败: ' + error.message, 'error'); } isTaskRunning = false; taskList = taskList.filter(t => t.id !== task.id); currentTask = null; runNextTask(); }, 100); } function cancelCurrentTask() { if (shareLinkManager) { shareLinkManager.taskCancel = true; } return true; } function updateFastLinkProgress(title, percent, desc) { if (isProgressMinimized) { updateMinimizedWidget(title, percent, desc); } else { showFastLinkProgress(title, percent, desc); } } function updateMinimizedWidget(title, percent, desc) { let widget = document.getElementById(minimizeWidgetId); const taskCountHtml = taskList.length > 0 ? `
${taskList.length}
` : ''; const html = ` ${taskCountHtml}
${title}${taskList.length > 0 ? ` - 队列 ${taskList.length}` : ''}
${Math.ceil(percent)}%
`; if (!widget) { widget = document.createElement('div'); widget.id = minimizeWidgetId; widget.style = 'position:fixed;right:20px;bottom:20px;width:220px;background:#fff;border-radius:12px;box-shadow:0 8px 24px rgba(0,0,0,0.18);padding:10px 12px;z-index:10005;display:flex;align-items:center;gap:10px;cursor:pointer;'; widget.innerHTML = html; widget.addEventListener('click', () => { isProgressMinimized = false; widget.remove(); updateFastLinkProgress(title, percent, desc); }); document.body.appendChild(widget); } else { widget.innerHTML = html; } } function addFastLinkButton() { if (!FastLinkConfig.enabled) { return; } const checkAndAddButton = () => { const existingButton = document.getElementById('fastlink-trigger'); if (existingButton && document.body.contains(existingButton)) { return; } const isFilePage = window.location.pathname === "/" && !window.location.search.includes("sharekey=") && !window.location.pathname.includes("/account"); if (!isFilePage) { return; } if (!document.body) { return; } const trigger = document.createElement('button'); trigger.id = 'fastlink-trigger'; trigger.innerHTML = ` `; trigger.addEventListener('click', (e) => { toggleFastLinkMenu(e); }); document.body.appendChild(trigger); // 添加下拉菜单 const menu = document.createElement('div'); menu.id = 'fastlink-menu'; menu.className = 'fastlink-menu'; menu.innerHTML = ` `; menu.querySelectorAll('.fastlink-menu-item').forEach(item => { item.addEventListener('click', (e) => { e.stopPropagation(); const action = item.dataset.action; menu.style.display = 'none'; if (action === 'generate') { addAndRunTask('generate'); } else if (action === 'save') { showFastLinkInputModal(); } }); }); document.body.appendChild(menu); }; // 立即尝试添加按钮 checkAndAddButton(); // 定期检查按钮是否还在 setInterval(() => { const btn = document.getElementById('fastlink-trigger'); const isFilePage = window.location.pathname === "/" && !window.location.search.includes("sharekey=") && !window.location.pathname.includes("/account"); if (isFilePage && (!btn || !document.body.contains(btn))) { checkAndAddButton(); } }, 3000); } function toggleFastLinkMenu(e) { e.stopPropagation(); const menu = document.getElementById('fastlink-menu'); if (!menu) { return; } const trigger = document.getElementById('fastlink-trigger'); const rect = trigger.getBoundingClientRect(); if (menu.style.display === 'block') { menu.style.display = 'none'; } else { menu.style.right = (window.innerWidth - rect.right) + 'px'; menu.style.bottom = (window.innerHeight - rect.top + 5) + 'px'; menu.style.left = 'auto'; menu.style.display = 'block'; } } function addTriggerButton() { const trigger = document.createElement('button'); trigger.id = 'settings-trigger'; trigger.innerHTML = ` `; trigger.addEventListener('click', createSettingsPanel); document.body.appendChild(trigger); } // 添加样式 - 修复版本 function addStyles() { // 先移除可能存在的旧样式 const existingStyle = document.getElementById('vip-settings-style'); if (existingStyle) { existingStyle.remove(); } const style = document.createElement('style'); style.id = 'vip-settings-style'; style.textContent = ` /* 全局样式 */ .settings-panel { position: fixed !important; top: 50% !important; left: 50% !important; transform: translate(-50%, -50%) !important; background: rgba(255, 255, 255, 0.95) !important; backdrop-filter: blur(20px) !important; -webkit-backdrop-filter: blur(20px) !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; border-radius: 16px !important; box-shadow: 0 20px 50px rgba(0, 0, 0, 0.15), 0 0 0 1px rgba(255, 255, 255, 0.1) inset !important; z-index: 10000 !important; width: 90% !important; max-width: 500px !important; max-height: 80vh !important; overflow: hidden !important; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, sans-serif !important; color: #333 !important; animation: panelFadeIn 0.3s ease !important; -webkit-font-smoothing: antialiased !important; -moz-osx-font-smoothing: grayscale !important; text-rendering: optimizeLegibility !important; } @keyframes panelFadeIn { from { opacity: 0; transform: translate(-50%, -48%); } to { opacity: 1; transform: translate(-50%, -50%); } } .panel-header { display: flex !important; justify-content: space-between !important; align-items: center !important; padding: 16px 20px !important; border-bottom: 1px solid rgba(238, 238, 238, 0.6) !important; background: rgba(248, 249, 250, 0.8) !important; backdrop-filter: blur(10px) !important; -webkit-backdrop-filter: blur(10px) !important; border-radius: 16px 16px 0 0 !important; } .title-container { display: flex !important; align-items: center !important; gap: 10px !important; -webkit-font-smoothing: antialiased !important; -moz-osx-font-smoothing: grayscale !important; } .panel-header h3 { margin: 0 !important; font-size: 18px !important; font-weight: 600 !important; color: #1a73e8 !important; text-rendering: optimizeLegibility !important; -webkit-font-smoothing: antialiased !important; -moz-osx-font-smoothing: grayscale !important; letter-spacing: 0.01em !important; } .github-icon { display: flex !important; align-items: center !important; justify-content: center !important; width: 30px !important; height: 30px !important; background: rgba(26, 115, 232, 0.1) !important; border: 1px solid rgba(26, 115, 232, 0.2) !important; border-radius: 6px !important; color: #1a73e8 !important; text-decoration: none !important; transition: all 0.3s ease !important; backdrop-filter: blur(5px) !important; -webkit-backdrop-filter: blur(5px) !important; image-rendering: -webkit-optimize-contrast !important; image-rendering: crisp-edges !important; shape-rendering: geometricPrecision !important; } .github-icon:hover { background: rgba(26, 115, 232, 0.15) !important; border-color: rgba(26, 115, 232, 0.4) !important; transform: scale(1.05) !important; color: #1557b0 !important; } .close-btn { background: none !important; border: none !important; font-size: 24px !important; cursor: pointer !important; color: #70757a !important; padding: 0 !important; width: 30px !important; height: 30px !important; display: flex !important; align-items: center !important; justify-content: center !important; border-radius: 50% !important; transition: background 0.2s !important; } .close-btn:hover { background: #f1f3f4 !important; color: #d93025 !important; } .settings-list { padding: 16px 20px !important; overflow-y: auto !important; max-height: calc(80vh - 180px) !important; padding-bottom: 20px !important; } .setting-item { margin-bottom: 16px !important; padding: 12px !important; background: rgba(248, 249, 250, 0.7) !important; backdrop-filter: blur(5px) !important; -webkit-backdrop-filter: blur(5px) !important; border-radius: 12px !important; border: 1px solid rgba(232, 234, 237, 0.6) !important; transition: all 0.3s ease !important; } .setting-item:hover { background: rgba(248, 249, 250, 0.9) !important; border-color: rgba(26, 115, 232, 0.3) !important; transform: translateY(-1px) !important; box-shadow: 0 4px 12px rgba(0, 0, 0, 0.08) !important; } .setting-content { display: flex !important; justify-content: space-between !important; align-items: center !important; margin-bottom: 8px !important; } .setting-key { font-weight: 500 !important; flex: 1 !important; } .setting-value { color: #1a73e8 !important; font-weight: 500 !important; text-align: right !important; } .setting-comment { font-size: 12px !important; color: #70757a !important; line-height: 1.4 !important; } /* 开关样式 */ .switch { position: relative !important; display: inline-block !important; width: 40px !important; height: 20px !important; } .switch input { opacity: 0 !important; width: 0 !important; height: 0 !important; } .slider { position: absolute !important; cursor: pointer !important; top: 0 !important; left: 0 !important; right: 0 !important; bottom: 0 !important; background-color: #ccc !important; transition: .3s !important; } .slider:before { position: absolute !important; content: "" !important; height: 16px !important; width: 16px !important; left: 2px !important; bottom: 2px !important; background-color: white !important; transition: .3s !important; } input:checked + .slider { background-color: #1a73e8 !important; } input:checked + .slider:before { transform: translateX(20px) !important; } .slider.round { border-radius: 34px !important; } .slider.round:before { border-radius: 50% !important; } /* 输入框和按钮样式 */ .input-container { display: flex !important; gap: 8px !important; } .setting-input { padding: 6px 10px !important; border: 1px solid #dadce0 !important; border-radius: 4px !important; font-size: 14px !important; flex: 1 !important; min-width: 0 !important; } .setting-input:focus { outline: none !important; border-color: #1a73e8 !important; } .save-btn { padding: 6px 12px !important; background: #1a73e8 !important; color: white !important; border: none !important; border-radius: 4px !important; cursor: pointer !important; font-size: 12px !important; transition: background 0.2s !important; white-space: nowrap !important; } .save-btn:hover { background: #1557b0 !important; } .save-btn.saved { background: #188038 !important; } /* 交流群按钮 */ .group-btn { display: flex !important; align-items: center !important; justify-content: center !important; gap: 8px !important; position: sticky !important; bottom: 0 !important; margin: 8px 20px 20px 20px !important; padding: 12px 16px !important; background: linear-gradient(135deg, rgba(26, 115, 232, 0.9), rgba(21, 87, 176, 0.9)) !important; backdrop-filter: blur(15px) !important; -webkit-backdrop-filter: blur(15px) !important; color: white !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; border-radius: 12px !important; cursor: pointer !important; text-decoration: none !important; font-weight: 500 !important; transition: all 0.3s ease !important; box-shadow: 0 4px 12px rgba(26, 115, 232, 0.2), 0 -2px 10px rgba(0, 0, 0, 0.1) !important; z-index: 10 !important; } .group-btn:hover { transform: translateY(-2px) !important; box-shadow: 0 8px 20px rgba(26, 115, 232, 0.4), 0 -4px 15px rgba(0, 0, 0, 0.15) !important; background: linear-gradient(135deg, rgba(26, 115, 232, 1), rgba(21, 87, 176, 1)) !important; } /* 触发按钮 */ #settings-trigger { position: fixed !important; bottom: 20px !important; right: 20px !important; width: 54px !important; height: 54px !important; background: rgba(26, 115, 232, 0.9) !important; backdrop-filter: blur(15px) !important; -webkit-backdrop-filter: blur(15px) !important; color: white !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; border-radius: 50% !important; cursor: pointer !important; z-index: 9999 !important; box-shadow: 0 6px 20px rgba(26, 115, 232, 0.3), 0 0 0 1px rgba(255, 255, 255, 0.1) inset !important; display: flex !important; align-items: center !important; justify-content: center !important; transition: all 0.3s ease !important; } #settings-trigger:hover { background: rgba(21, 87, 176, 0.95) !important; transform: scale(1.05) rotate(90deg) !important; box-shadow: 0 8px 25px rgba(26, 115, 232, 0.4), 0 0 0 1px rgba(255, 255, 255, 0.2) inset !important; } /* 响应式设计 */ @media (max-width: 600px) { .settings-panel { width: 95% !important; max-height: 85vh !important; } .panel-header { padding: 14px 16px !important; } .settings-list { padding: 12px 16px !important; max-height: calc(85vh - 160px) !important; padding-bottom: 16px !important; } .setting-content { flex-direction: column !important; align-items: flex-start !important; gap: 8px !important; } .input-container { width: 100% !important; } .group-btn { margin: 8px 16px 16px 16px !important; padding: 10px 14px !important; } .title-container { gap: 8px !important; } .github-icon { width: 28px !important; height: 28px !important; } .github-icon svg { width: 18px !important; height: 18px !important; } #settings-trigger { bottom: 16px !important; right: 16px !important; width: 44px !important; height: 44px !important; } } /* 秒传功能样式 */ #fastlink-trigger { position: fixed !important; bottom: 90px !important; right: 20px !important; width: 54px !important; height: 54px !important; background: rgba(76, 175, 80, 0.9) !important; backdrop-filter: blur(15px) !important; -webkit-backdrop-filter: blur(15px) !important; color: white !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; border-radius: 50% !important; cursor: pointer !important; z-index: 9999 !important; box-shadow: 0 6px 20px rgba(76, 175, 80, 0.3), 0 0 0 1px rgba(255, 255, 255, 0.1) inset !important; display: flex !important; align-items: center !important; justify-content: center !important; transition: all 0.3s ease !important; } #fastlink-trigger:hover { background: rgba(67, 160, 71, 0.95) !important; transform: scale(1.05) !important; box-shadow: 0 8px 25px rgba(76, 175, 80, 0.4), 0 0 0 1px rgba(255, 255, 255, 0.2) inset !important; } .fastlink-menu { position: fixed !important; display: none; background: rgba(255, 255, 255, 0.95) !important; backdrop-filter: blur(15px) !important; -webkit-backdrop-filter: blur(15px) !important; border: 1px solid rgba(0, 0, 0, 0.1) !important; border-radius: 10px !important; box-shadow: 0 4px 16px rgba(0, 0, 0, 0.15) !important; overflow: hidden !important; z-index: 10000 !important; min-width: 140px !important; } .fastlink-menu-item { padding: 12px 16px !important; cursor: pointer !important; font-size: 14px !important; color: #333 !important; transition: all 0.2s !important; border-bottom: 1px solid rgba(0, 0, 0, 0.05) !important; } .fastlink-menu-item:last-child { border-bottom: none !important; } .copy-dropdown { position: relative; display: inline-block; } .copy-dropdown-menu { position: absolute; background: #fff; border: 1px solid #e1e5e9; border-radius: 10px; box-shadow: 0 4px 16px rgba(0,0,0,0.12); min-width: 120px; z-index: 10001; padding: 0; } .copy-dropdown:hover .copy-dropdown-menu { display: block !important; } .copy-dropdown-item { padding: 10px 18px; cursor: pointer; font-size: 14px; border-bottom: 1px solid #f0f0f0; background: #fff; transition: background 0.2s; } .copy-dropdown-item:first-child { border-radius: 10px 10px 0 0; } .copy-dropdown-item:last-child { border-bottom: none; border-radius: 0 0 10px 10px; } .copy-dropdown-item:hover { background: #e8f5e9; color: #388e3c; } .copy-btn { background: linear-gradient(135deg, #4CAF50 0%, #45a049 100%); color: white; border: none; padding: 14px 24px; cursor: pointer; border-radius: 8px; font-size: 16px; font-weight: 500; min-width: 100px; transition: all 0.3s ease; box-shadow: 0 4px 12px rgba(76, 175, 80, 0.3); } .copy-btn:hover { transform: translateY(-2px); box-shadow: 0 8px 20px rgba(76, 175, 80, 0.4); } .copy-btn:active { transform: translateY(0); box-shadow: 0 2px 8px rgba(76, 175, 80, 0.3); } .export-btn { background: linear-gradient(135deg, #2196F3 0%, #1976D2 100%); color: white; border: none; padding: 14px 24px; cursor: pointer; border-radius: 8px; font-size: 16px; font-weight: 500; min-width: 100px; transition: all 0.3s ease; box-shadow: 0 4px 12px rgba(33, 150, 243, 0.3); } .export-btn:hover { transform: translateY(-2px); box-shadow: 0 8px 20px rgba(33, 150, 243, 0.4); } .export-btn:active { transform: translateY(0); box-shadow: 0 2px 8px rgba(33, 150, 243, 0.3); } .file-input-btn { background: linear-gradient(135deg, #FF9800 0%, #F57C00 100%); color: white; border: none; padding: 14px 24px; cursor: pointer; border-radius: 8px; font-size: 16px; font-weight: 500; min-width: 100px; transition: all 0.3s ease; box-shadow: 0 4px 12px rgba(255, 152, 0, 0.3); } .file-input-btn:hover { transform: translateY(-2px); box-shadow: 0 8px 20px rgba(255, 152, 0, 0.4); } .file-input-btn:active { transform: translateY(0); box-shadow: 0 2px 8px rgba(255, 152, 0, 0.3); } .close-btn { position: absolute; top: 16px; right: 16px; background: transparent; border: none; font-size: 24px; color: #999; cursor: pointer; width: 32px; height: 32px; border-radius: 50%; display: flex; align-items: center; justify-content: center; transition: all 0.2s ease; } .close-btn:hover { background: rgba(244, 67, 54, 0.1); color: #f44336; transform: scale(1.1); } .fastlink-menu-item:hover { background: rgba(76, 175, 80, 0.1) !important; color: #4CAF50 !important; } @keyframes fastlinkToastSlideIn { from { transform: translateX(400px); opacity: 0; } to { transform: translateX(0); opacity: 1; } } @keyframes fastlinkToastSlideOut { from { transform: translateX(0); opacity: 1; } to { transform: translateX(400px); opacity: 0; } } .fastlink-modal-overlay { display: flex !important; position: fixed !important; top: 0 !important; left: 0 !important; width: 100% !important; height: 100% !important; background: rgba(0, 0, 0, 0.6) !important; backdrop-filter: blur(4px) !important; justify-content: center !important; align-items: center !important; z-index: 10000 !important; animation: fadeIn 0.3s ease-out !important; } .fastlink-modal { background: rgba(255, 255, 255, 0.95) !important; backdrop-filter: blur(20px) !important; -webkit-backdrop-filter: blur(20px) !important; padding: 32px !important; border-radius: 16px !important; box-shadow: 0 20px 60px rgba(0, 0, 0, 0.15) !important; text-align: center !important; width: 480px !important; max-width: 90vw !important; max-height: 90vh !important; overflow: hidden !important; position: relative !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; } .fastlink-modal h3 { margin: 0 0 20px 0 !important; font-size: 20px !important; color: #333 !important; } .fastlink-modal textarea { width: 100% !important; padding: 16px !important; margin: 0 0 24px 0 !important; border: 2px solid #e1e5e9 !important; border-radius: 12px !important; resize: vertical !important; min-height: 120px !important; font-family: 'Consolas', 'Monaco', 'Courier New', monospace !important; font-size: 14px !important; line-height: 1.5 !important; background: rgba(250, 251, 252, 0.7) !important; transition: all 0.3s ease !important; box-sizing: border-box !important; outline: none !important; } .fastlink-modal textarea:focus { border-color: #4CAF50 !important; background: rgba(255, 255, 255, 0.9) !important; box-shadow: 0 0 0 3px rgba(76, 175, 80, 0.1) !important; } .fastlink-progress-bar { background: #eee !important; border-radius: 8px !important; overflow: hidden !important; height: 18px !important; margin: 20px 0 !important; } #fastlink-progress { background: linear-gradient(90deg, #4CAF50, #66BB6A) !important; height: 100% !important; transition: width 0.3s !important; } #fastlink-title { margin-bottom: 16px !important; font-size: 18px !important; color: #333 !important; word-wrap: break-word !important; word-break: break-all !important; white-space: pre-wrap !important; line-height: 1.4 !important; } #fastlink-percent { margin: 8px 0 !important; font-size: 14px !important; color: #666 !important; font-weight: 500 !important; } #fastlink-desc { margin-top: 8px !important; font-size: 13px !important; color: #888 !important; word-wrap: break-word !important; word-break: break-all !important; white-space: pre-wrap !important; line-height: 1.4 !important; min-height: 20px !important; } `; document.head.appendChild(style); } // 注册菜单命令 GM_registerMenuCommand('⚙️ 打开设置面板', createSettingsPanel); // 等待页面加载完成 function waitForBody() { if (document.body) { addStyles(); // 先添加样式 addTriggerButton(); // 添加设置按钮 // 初始化秒传功能 if (FastLinkConfig.enabled) { initFastLink(); addFastLinkButton(); // 添加秒传按钮 // 点击其他地方关闭菜单 document.addEventListener('click', (e) => { const menu = document.getElementById('fastlink-menu'); const trigger = document.getElementById('fastlink-trigger'); if (menu && trigger && !trigger.contains(e.target) && !menu.contains(e.target)) { menu.style.display = 'none'; } }); console.log('[123云盘解锁] 秒传菜单关闭监听器已添加'); } } else { setTimeout(waitForBody, 100); } } // 页面加载完成后初始化 if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', waitForBody); } else { waitForBody(); } // 输出版本信息 console.log('[123云盘解锁+秒传] v1.2.0 已加载完成'); if (FastLinkConfig.enabled) { console.log('[123云盘解锁] 秒传功能已启用'); } })();