123云盘解锁

专业的123云盘增强脚本 - 完美解锁会员功能、突破下载限制、去广告、支持自定义用户信息。整合秒传链接功能,支持生成和保存秒传文件,快速分享和保存文件。界面精美,功能强大,让你的123云盘体验更美好!

安装此脚本
作者推荐脚本

您可能也喜欢搜索引擎净化

安装此脚本
// ==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 = `
            <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="currentColor" style="image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; shape-rendering: geometricPrecision;">
                <path d="M12 0c-6.626 0-12 5.373-12 12 0 5.302 3.438 9.8 8.207 11.387.599.111.793-.261.793-.577v-2.234c-3.338.726-4.033-1.416-4.033-1.416-.546-1.387-1.333-1.756-1.333-1.756-1.089-.745.083-.729.083-.729 1.205.084 1.839 1.237 1.839 1.237 1.07 1.834 2.807 1.304 3.492.997.107-.775.418-1.305.762-1.604-2.665-.305-5.467-1.334-5.467-5.931 0-1.311.469-2.381 1.236-3.221-.124-.303-.535-1.524.117-3.176 0 0 1.008-.322 3.301 1.23.957-.266 1.983-.399 3.003-.404 1.02.005 2.047.138 3.006.404 2.291-1.552 3.297-1.23 3.297-1.23.653 1.653.242 2.874.118 3.176.77.84 1.235 1.911 1.235 3.221 0 4.609-2.807 5.624-5.479 5.921.43.372.823 1.102.823 2.222v3.293c0 .319.192.694.801.576 4.765-1.589 8.199-6.086 8.199-11.386 0-6.627-5.373-12-12-12z"/>
            </svg>
        `;
        githubIcon.title = '访问GitHub项目';
        titleContainer.appendChild(githubIcon);

        header.appendChild(titleContainer);

        // 添加关闭按钮
        const closeButton = document.createElement('button');
        closeButton.className = 'close-btn';
        closeButton.innerHTML = '&times;';
        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 = `
            <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
                <path d="M21 11.5a8.38 8.38 0 0 1-.9 3.8 8.5 8.5 0 0 1-7.6 4.7 8.38 8.38 0 0 1-3.8-.9L3 21l1.9-5.7a8.38 8.38 0 0 1-.9-3.8 8.5 8.5 0 0 1 4.7-7.6 8.38 8.38 0 0 1 3.8-.9h.5a8.48 8.48 0 0 1 8 8v.5z"></path>
            </svg>
            <span>加入交流群</span>
        `;
        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 = `
                <div class="fastlink-modal" style="position:relative;">
                    <button id="minimize-progress-btn" style="position:absolute;right:12px;top:12px;width:32px;height:32px;border-radius:50%;background:#ffc504;color:#000;border:none;display:flex;align-items:center;justify-content:center;font-weight:700;cursor:pointer;box-shadow:0 2px 6px rgba(0,0,0,0.15);z-index:10003;transition:all 0.2s ease;" title="最小化">−</button>
                    <h3 id="fastlink-title">${title}${taskQueueHtml}</h3>
                    <div class="fastlink-progress-bar">
                        <div id="fastlink-progress" style="width: ${percent}%"></div>
                    </div>
                    <div id="fastlink-percent">${percent}%</div>
                    <div id="fastlink-desc">${desc}</div>
                </div>
            `;
            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 ? `
            <div style="max-height:120px;overflow-y:auto;background:rgba(248,249,250,0.7);border-radius:8px;padding:10px;margin-bottom:16px;font-size:13px;">
                <div style='color:#888;margin-bottom:6px;'>文件列表(共${shareLinkManager.fileInfoList.length}个):</div>
                ${shareLinkManager.fileInfoList.map(f => `<div style='color:#333;word-break:break-all;margin:2px 0;'>${f.path}</div>`).join('')}
            </div>
        ` : '';

        const modal = document.createElement('div');
        modal.className = 'fastlink-modal-overlay';
        modal.id = 'fastlink-copy-modal';
        modal.innerHTML = `
            <div class="fastlink-modal">
                <button class="close-btn" id="close-copy-modal">×</button>
                <h3>🚀 秒传链接</h3>
                ${fileListHtml}
                <textarea id="fastlink-copy-text" placeholder="秒传链接已生成...">${shareLink}</textarea>
                <div style="display: flex; gap: 12px; justify-content: center;">
                    <div class="copy-dropdown" style="position: relative;">
                        <button class="copy-btn" id="copy-main-btn">复制 ▼</button>
                        <div class="copy-dropdown-menu" style="display: none; position: absolute; bottom: 100%; left: 0; 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; margin-bottom: 6px; padding: 0;">
                            <div class="copy-dropdown-item" data-type="text">复制纯文本</div>
                            <div class="copy-dropdown-item" data-type="json">复制JSON</div>
                        </div>
                    </div>
                    <button class="export-btn" id="export-json-btn">导出JSON</button>
                    <button class="export-btn" id="close-modal-btn">关闭</button>
                </div>
            </div>
        `;
        
        // 事件绑定
        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 = `
            <div class="fastlink-modal">
                <button class="close-btn" id="close-input-modal">×</button>
                <h3>📥 保存秒传文件</h3>
                <textarea id="fastlink-input-text" placeholder="请输入或粘贴秒传链接,或拖入JSON文件..."></textarea>
                <div style="display: flex; gap: 12px; justify-content: center;">
                    <button class="copy-btn" id="fastlink-save-btn">保存</button>
                    <button class="file-input-btn" id="select-file-btn">选择JSON</button>
                    <input type="file" id="json-file-input" accept=".json" style="display: none;">
                    <button class="export-btn" id="cancel-input-btn">取消</button>
                </div>
            </div>
        `;
        
        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 ? `
            <div style="margin-top: 12px; color: #f44336; font-size: 14px;">
                <div style="margin-bottom: 6px;">失败文件列表:</div>
                <div style="max-height: 160px; overflow-y: auto; background: rgba(245,245,245,0.7); border-radius: 6px; padding: 8px;">
                    ${result.failed.map(fileInfo => `<div style="font-size: 13px; margin: 2px 0;">${fileInfo.fileName} ${fileInfo.error ? `<span style="color: #f44336;">(${fileInfo.error})</span>` : ''}</div>`).join('')}
                </div>
            </div>
        ` : '';

        const modal = document.createElement('div');
        modal.className = 'fastlink-modal-overlay';
        modal.id = 'fastlink-result-modal';
        modal.innerHTML = `
            <div class="fastlink-modal">
                <button class="close-btn" id="close-result-modal">×</button>
                <h3>📊 保存结果</h3>
                <div style="margin: 20px 0;">
                    <div style="font-size: 16px; margin-bottom: 12px;">
                        <span style="color: #666;">总计:</span><strong>${totalCount}</strong> 个文件
                    </div>
                    <div style="font-size: 16px; margin-bottom: 8px; color: #4CAF50;">
                        ✅ 成功:<strong>${successCount}</strong> 个
                    </div>
                    <div style="font-size: 16px; margin-bottom: 8px; color: ${failedCount > 0 ? '#f44336' : '#666'};">
                        ${failedCount > 0 ? '❌' : '✅'} 失败:<strong>${failedCount}</strong> 个
                    </div>
                    ${failedListHtml}
                </div>
                <div style="display: flex; gap: 12px; justify-content: center;">
                    <button class="copy-btn" id="confirm-result-btn">确定</button>
                    ${failedCount > 0 ? `
                        <div class="copy-dropdown" style="position: relative;">
                            <button class="file-input-btn" id="retry-btn">重试失败 ▼</button>
                            <div class="copy-dropdown-menu" style="display: none; position: absolute; bottom: 100%; left: 0; 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; margin-bottom: 6px; padding: 0;">
                                <div class="copy-dropdown-item" data-action="retry">重试失败</div>
                                <div class="copy-dropdown-item" data-action="export">导出失败链接</div>
                            </div>
                        </div>
                    ` : ''}
                </div>
            </div>
        `;
        
        // 事件绑定
        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 ? 
            `<div style="position:absolute;left:-8px;top:-8px;width:22px;height:22px;background:#f44336;color:#fff;border-radius:50%;display:flex;align-items:center;justify-content:center;font-size:13px;font-weight:700;z-index:2;box-shadow:0 2px 6px rgba(0,0,0,0.12);">${taskList.length}</div>` : '';
        
        const html = `
            ${taskCountHtml}
            <div style="flex:1;">
                <div style="font-size:12px;color:#333;margin-bottom:6px;">${title}${taskList.length > 0 ? ` - 队列 ${taskList.length}` : ''}</div>
                <div style="height:8px;background:#eee;border-radius:6px;overflow:hidden;">
                    <div style="height:100%;background:#4CAF50;width:${percent}%;transition:width 0.2s;"></div>
                </div>
            </div>
            <div style="font-size:12px;color:#666;width:36px;text-align:right;">${Math.ceil(percent)}%</div>
        `;
        
        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 = `
                <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 1024 1024" fill="currentColor">
                    <path d="M395.765333 586.570667h-171.733333c-22.421333 0-37.888-22.442667-29.909333-43.381334L364.768 95.274667A32 32 0 0 1 394.666667 74.666667h287.957333c22.72 0 38.208 23.018667 29.632 44.064l-99.36 243.882666h187.050667c27.509333 0 42.186667 32.426667 24.042666 53.098667l-458.602666 522.56c-22.293333 25.408-63.626667 3.392-54.976-29.28l85.354666-322.421333z"/>
                </svg>
            `;
            
            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 = `
                <div class="fastlink-menu-item" data-action="generate">生成秒传链接</div>
                <div class="fastlink-menu-item" data-action="save">保存秒传文件</div>
            `;
            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 = `
            <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
                <circle cx="12" cy="12" r="3"></circle>
                <path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z"></path>
            </svg>
        `;
        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云盘解锁] 秒传功能已启用');
    }
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址