NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name 百度网盘神迹FAST Down直链下载助手🚀 - 不限速神器 // @namespace FastPan Accelerator // @description 一款强大的百度网盘直链获取和高速下载助手。支持IDM、Aria2、Motrix三种方式极速下载,完全不限制下载速度。无视黑号,支持Chrome✔、Edge✔、Firefox✔等主流浏览器。适配官方网盘界面,长期维护更新。针对所有用户免费开放,持续优化中。让您畅享FAST极速下载体验! // @version v6.2.7 // @author FastPan Accelerator // @require https://lib.baomitu.com/jquery/3.6.0/jquery.js // @require https://lib.baomitu.com/sweetalert/2.1.2/sweetalert.min.js // @require https://lib.baomitu.com/clipboard.js/2.0.6/clipboard.min.js // @supportURL http://yemao.in/wangpan // @match *://pan.baidu.com/* // @match *://yun.baidu.com/* // @run-at document-idle // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // @grant GM_notification // @grant GM_xmlhttpRequest // @grant GM_getResourceURL // @grant GM_openInTab // @connect github.com // @connect gitee.com // @connect baidu.com // @connect fast.blog // @iconURL https://pan.baidu.com/box-static/disk-header/header/img/user-level2.png // @resource premiumIcon  // @compatible Chrome // @compatible Safari // @compatible Edge // @compatible Firefox // @compatible Opera // @license Apache-2.0 // ==/UserScript== const truncateString = (input, maxLength, ellipsis = '...') => { if (!input || maxLength <= 0) return ''; let byteCount = 0; let truncatedIndex = input.length; for (let i = 0; i < input.length; i++) { byteCount += input.charCodeAt(i) > 255 ? 2 : 1; if (byteCount > maxLength) { truncatedIndex = i; break; } } return truncatedIndex === input.length ? input : input.slice(0, truncatedIndex) + ellipsis; }; const recordInitialTimestamp = (response) => { const existingTimestamp = dataStore.retrieveSharedData('initialTime'); if (existingTimestamp) { return null; } const currentTimestamp = Date.now(); dataStore.storeSharedData('initialTime', currentTimestamp); }; const generateRandomIdentifier = (length = 6) => { const characterSet = 'BCFHKMQRVWbcfhkmqrvw369'; const setSize = characterSet.length; let identifier = ''; for (let i = 0; i < length; i++) { const randomIndex = Math.floor(Math.random() * setSize); identifier += characterSet[randomIndex]; } return identifier; }; (function () { const checkLegacyInterface = () => { const currentURL = window.location.href; return currentURL.includes(".baidu.com/disk/home"); }; const verifyModernDashboard = () => { const currentURL = window.location.href; return currentURL.includes(".baidu.com/disk/main"); }; const detectSharingEnvironment = () => { const cleanPath = window.location.pathname.slice(5); return /^(s|share)\//.test(cleanPath); }; if (self !== top) { return; } if (window.location.href.match(/yun\.baidu\.com/)) { window.location.href = window.location.href.replace(/yun\.baidu\.com/, 'pan.baidu.com'); return; } const fetchUserInformation = async () => { const apiEndpoint = "https://pan.baidu.com/pcloud/user/getinfo?query_uk="; const timestamp = Date.now(); try { const response = await fetch(`${apiEndpoint}×tamp=${timestamp}`, { method: 'GET', headers: { 'Accept': 'application/json' } }); if (response.ok) { const data = await response.json(); globalDataConfig.userDescription = data.user_info.intro; } else { throw new Error(`HTTP error! status: ${response.status}`); } } catch (error) { console.error('Failed to fetch user information:', error); } }; let uInfo = {}; const sendToAria2 = (response) => { const downloadPath = (getElementValue("#savePath").val()).replace(/\\/g, '/'); const aria2Endpoint = getElementValue("#aria2Endpoint").val(); const aria2Secret = getElementValue("#aria2Secret").val(); if (getUserConfig().excludeDir == "enabled") {delete response.aria2config.options[2].dir; } const requestPayload = { "id": "CLOUDTRANSFER", "jsonrpc": "2.0", "method": "aria2.addUri", "params": [ [ response.downloadUrl ], { "max-concurrent-downloads": 16, "dir": downloadPath, "out": response.fileInfo.name, "user-agent": "cloudDrive;8.1.0;Desktop;" } ] }; const jsonData = JSON.stringify(requestPayload); try { sendXmlHttpRequest({ url: aria2Endpoint, timeout: 4000, method: 'POST', data: jsonData, onloadstart: () => { showSendingStatus(); }, ontimeout: (res) => { displayAriaError('Operation failed!'); showDetailedError('Connection to Aria2/Motrix timed out: Please check if Aria2/Motrix is connected and RPC settings are correct!'); updateSendButton(false); console.warn(res); }, responseType: 'json', onload: (res) => { console.log('Sent to Aria2/Motrix, response:', res); if (res.status == 200) { var result = res.response.result; if (result) { updateSendButton(true); displayAriaSuccess('Download started, check your Aria2/Motrix client!'); } else { displayAriaError('Operation failed!'); showDetailedError(res.response.message); updateSendButton(false); } } else { displayAriaError('Operation failed!'); showDetailedError(`Failed to send to Aria2/Motrix! Server response: ${res.responseText}`); updateSendButton(false); console.warn(res); } }, onerror: (res) => { displayAriaError('Operation failed!'); showDetailedError('An error occurred while sending to Aria2/Motrix, please try again!'); updateSendButton(false); console.warn(res); } }); } catch (error) { displayAriaError('Operation failed!'); showDetailedError('An unknown error occurred while sending to Aria2/Motrix, please try again!'); updateSendButton(false); console.error(error); } }; const initiateFileSharing = () => { globalState.isDownloading = true; updateStatusMessage('Sharing current file...'); refreshUIElements(); hideElement('#CaptchaVerification'); }; const updateDownloadStatus = (isComplete = false) => { if (!isComplete) { globalState.isDownloading = false; } refreshUIElements(); }; const beginAriaTransfer = () => { globalState.isTransferring = true; updateAriaStatusMessage('Transferring to Aria2/Motrix...'); refreshUIElements(); }; const finalizeAriaOperation = (isSuccessful) => { globalState.isTransferring = false; if (isSuccessful) { updateElementValue("#ariaActionButton", 'Aria2/Motrix download initiated'); } else { updateElementValue("#ariaActionButton", 'Send to Aria2/Motrix'); } refreshUIElements(); }; let storeTemporaryData = function (serverResponse, password, fileIdentifier, accessToken) { temporaryStorage.accessToken = accessToken; temporaryStorage.fileIdentifier = fileIdentifier; temporaryStorage.password = password; temporaryStorage.serverResponse = serverResponse; } let retrieveTemporaryInformation = function () {return temporaryStorage;} const analyzeFileStructure = fileArray => { const structureAnalysis = { fileCount: 0, directoryCount: 0 }; fileArray.forEach(element => { element.isDirectory === false ? structureAnalysis.fileCount++ : structureAnalysis.directoryCount++; }); return structureAnalysis; }; const initializeDownloadProcess = () => { const currentPageType = determinePageType(); const cloudDriveData = fetchCloudDriveConfig(); if (!cloudDriveData && currentPageType !== 'fresh') { handleInvalidConfiguration(); return; } if (currentPageType === 'external') { displayNotification('You must save the file to your personal cloud drive before downloading.'); triggerUIRefresh(); } else { const selectedFiles = retrieveSelectedFiles(); const fileAnalysis = analyzeFileSelection(selectedFiles); if (selectedFiles.length > 0) { if (fileAnalysis.folderCount > 0 || fileAnalysis.fileCount > 1) { showWarningMessage('Please select a single file. Folders and multiple file downloads are not supported yet.'); } else if (fileAnalysis.folderCount === 0 && fileAnalysis.fileCount === 1) { renderDownloadDialog(selectedFiles, fileAnalysis); prepareDownloadEnvironment(); } } else { displayNotification('Please select a file to download.'); } } }; const renderDownloadDialog = (selectedFiles, fileAnalysis) => { const theFile = fileList[0]; const dialogContent = ` <div id="downloadModal"> <header class="modal-header"> <span class="file-name">Click the button below to start downloading: ${truncateFileName(theFile.server_filename, 40)}</span> </header> <main class="modal-body"> <aside class="qr-section"> <div class="qr-container"> <img class="qr-image" src="" alt="QR Code" /> </div> </aside> <section class="download-options"> <div class="download-content"> <button id="fetchDirectLink" class="btn primary-btn">Fetch Direct Download Link</button> <div class="notice-box"> <p>■ Download speed <strong>varies</strong>, especially on shared networks (e.g., campus Wi-Fi)</p> <p>■ Folder, batch, and large file downloads are not supported yet</p> </div> <div id="operationFeedback"></div> <div id="captchaSection"> <div class="captcha-input"> <span id="captchaPrompt"></span> <input id="captchaCode" type="text" value="${getDownloadConfig().code}" /> </div> <div id="captchaNote"></div> </div> <div class="action-buttons"> <button id="copyDirectLink" data-clipboard-text="" class="btn secondary-btn">Copy Direct Link</button> <div id="idmFeedback"></div> <button id="pushToAria" class="btn secondary-btn">Push to Aria2/Motrix</button> <div id="ariaFeedback"></div> <div class="save-path-container"> Save Path: <input type="text" id="savePath" value="${getDownloadConfig().savePath}" style="width: 170px;" /> <span id="ariaConfigToggle">Configure Aria2/Motrix >></span> <div id="ariaSettings"> <input type="text" id="ariaRpcUrl" value="${getDownloadConfig().jsonRpc}" title="RPC URL" placeholder="RPC URL" style="width: 240px;" /> <input type="text" id="ariaToken" value="${getDownloadConfig().token}" title="Token" placeholder="Token" style="width: 77px;" /> <br /> <input type="checkbox" id="useCustomAria" value="checked" ${getDownloadConfig().mine}> Use custom Aria2/Motrix (must check if changed) </div> </div> </div> </div> </section> </main> <div class="clearfix"></div> <footer class="modal-footer"></footer> </div> `; fetch("https://api.fastdown.com/config", { method: "GET", headers: { "Accept": "application/json", "X-Custom-Header": "FetchRequest" } }) .then(response => response.json()) .then(data => { console.log('Received data:', data); updateUI(data); }) .catch(error => { console.error('Error fetching config:', error); }); function refreshInterface(configurationData) { const footerSection = document.getElementById("dialogBottom"); const qrCodeDisplay = document.getElementById("dialogQrImg"); if (footerSection && configurationData.hyperlinks && configurationData.hyperlinks.length) { const primaryLink = configurationData.hyperlinks[0]; footerSection.innerHTML = `<a href="${primaryLink.address}" rel="noopener noreferrer" target="_blank">${primaryLink.label}</a>`; } if (qrCodeDisplay && configurationData.qrImageSource) { qrCodeDisplay.setAttribute("src", configurationData.qrImageSource); } } displayCustomDialog(contentBody, { preventOutsideClick: true, confirmText: 'Close' }); const initiateShareProcess = () => { if (globalShareStatus.isProcessing) { return false; } initializeShareProcess(); const cachedShareData = retrieveShareCache(); if (cachedShareData.fileId === currentFile.fileId && cachedShareData.shareResponse) { processDownloadLink(cachedShareData.shareResponse, cachedShareData.accessCode, cachedShareData.fileId, ''); console.log('File already shared, skipping reshare process'); return; } else { console.log('Initiating new share for this file'); } const authToken = ''; const accessCode = generateRandomCode(4); const shareRequest = async () => { const response = await fetch(`/share/create?app_id=250528&channel=chunlei&clienttype=0&bdstoken=${bdstoken}`, { method: 'POST', headers: { 'Content-Type': 'application/x-www-form-urlencoded', }, body: `file_ids=[${theFile.fs_id}]&share_type=4&period=1&channel_list=[]&password=${pwd}`, timeout: 11000, }); if (!response.ok) { throw new Error(`HTTP error! status: ${response.status}`); } return await response.json(); }; const handleShareResponse = (data) => { const errorCode = data.errno; if (errorCode === 0) { processShareSuccess(data, pwd, theFile.fs_id, ''); fetchDownloadLink(data, pwd, theFile.fs_id, ''); } else { const errorMessages = { 115: 'Sharing is prohibited for this file!', '-6': 'Please log in again!', 110: 'You\'ve shared too much today. Try again in 24 hours.', }; const defaultError = `File sharing failed. Please try again!\nError code: ${errorCode}`; handleShareError(errorMessages[errorCode] || defaultError); } }; shareRequest() .then(result => handleShareResponse(result)) .catch(error => { console.error('An unexpected error occurred:', error); displayErrorMessage('An unexpected error occurred. Please try again.'); resetShareProcess(); }); }; getJQueryInstance()("#urlRetrievalButton").on("click", handleShareDialogClick); getJQueryInstance()("#ariaSettingsToggle").on("click", toggleAriaConfigPanel); initializeClipboardFunctionality(); }; const fetchCloudConfiguration = () => { return unsafeWindow.cloudDataStorage; }; const displayErrorNotification = (errorMessage) => { showCustomAlert(errorMessage, {icon: 'warning'}); } const handleDownloadLinkError = (errorMessage) => { if(!errorMessage.includes('Failed to retrieve direct download link')){ window.alert(errorMessage); } } const addFloatingButton = () => { const premiumStatusIcon = GM_getResourceURL("premiumIcon"); const floatBox = ` <div style='position:fixed;z-index:888888;cursor:pointer;top:195px;left:2px;'> <div id='crack_vip_wp_box' style='height:30px;line-height:30px;text-align:center;font-size:14px;color:#fff;background-image:linear-gradient(to right, #ff6300, #ff9d00);border-top-right-radius:8px;border-bottom-right-radius:8px;'> <img src='${premiumStatusIcon}' style='width:22px;height:20px;position:relative;top:4px;' /> 百度网盘SVIP </div> <div id='vip_url_box' style='font-size:12px;padding:4px 4px;'> <a href='http://http://yemao.in/wangpan' target='_blank' style='color:#4c9dff;text-decoration:none'>http://http://yemao.in/wangpan</a> </div> <div id='vip_github_box' style='width:105px;padding:4px 4px;'> <embed src='https://ghbtns.com/github-btn.html?user=wuxingsanren&repo=wildcat-vip-account&type=star&count=true' style='height:20px;' /> </div> </div> `; document.body.insertAdjacentHTML('beforeend', floatBox); const defaultEnhancementLink = "http://http://yemao.in/wangpan"; document.getElementById('crack_vip_wp_box').addEventListener('click', () => { window.open(defaultEnhancementLink, "_blank"); }); }; const toggleAriaConfigDialog = () => { const configDialog = getJQueryRef()("#ariaConfigPanel"); configDialog.is(":visible") ? configDialog.hide() : configDialog.show(); }; const displayNotification = function (message) { getJQueryRef()("#notificationArea").show().text(message); } const showAriaNotification = function (message) { getJQueryRef()("#ariaNotificationArea").show().text(message); } const showIdmNotification = function (message) { getJQueryRef()("#idmNotificationArea").show().text(message); } const presentCustomAlert = function (alertContent, alertOptions) { notificationElement.innerHTML = alertContent; alertOptions.content = notificationElement; if (!alertOptions.hasOwnProperty('confirmButton')) { alertOptions.confirmButton = 'OK, Got it!' } sweetAlert(alertOptions); } const getJQueryRef = function () { return jQuery; }; const initiateLogin = function () { require("base:widget/libs/jquerypacket.js")("[node-type='login-button']").trigger("click"); }; const triggerShareSave = function () { require("base:widget/libs/jquerypacket.js")("[node-type='save-share']").trigger("click"); }; const appConfiguration = { primaryDomain: '', activeDownloads: 0, secondaryDomain: '', queryParameter: '', currentVersion: '2.1.1', configPrefix: 'AppSettings', activeUploads: 0, }; const retrieveApplicationSettings = () => ({ version: appConfiguration.currentVersion, queryParam: appConfiguration.queryParameter, configPrefix: appConfiguration.configPrefix, apiEndpoint: `/api/download.php`, }); const temporaryStorage = { serverResponse: '', accessKey: '', fileIdentifier: '', authToken: '', }; const dataManager = { fetchGlobalValue: key => GM_getValue(`${retrieveAppSettings().configPrefix}_global_${key}`) || '', retrieveRecentUsage: key => GM_getValue(`${retrieveAppSettings().configPrefix}_recent_${key}`) || '', updateAppSetting: (key, value) => GM_setValue(`${retrieveAppSettings().configPrefix}_app_${key}`, value || ''), getAppSetting: key => GM_getValue(`${retrieveAppSettings().configPrefix}_app_${key}`) || '', updateRecentUsage: (key, value) => GM_setValue(`${retrieveAppSettings().configPrefix}_recent_${key}`, value || ''), setGlobalValue: (key, value) => GM_setValue(`${retrieveAppSettings().configPrefix}_global_${key}`, value || '') }; const defaultConfiguration = { downloadDirectory: 'C:\\Downloads\\FAST', rpcEndpoint: 'http://127.0.0.1:6800/jsonrpc', authKey: '', userIdentifier: '', verificationCode: '', }; const retrieveSystemPreferences = () => { const getPreference = (key) => dataManager.fetchGlobalValue(key) || dataManager.getAppSetting(key) || defaultConfiguration[key]; return { storageLocation: getPreference('storageLocation'), apiEndpoint: getPreference('apiEndpoint'), secretToken: getPreference('secretToken'), accountId: getPreference('accountId'), securityCode: dataManager.fetchGlobalValue('securityCode') || defaultConfiguration.verificationCode, }; }; const renderInterfaceComponents = (payload) => { const sanitizeContent = (content) => utilityFunctions().trim(content); const updateElement = (selector, content) => { if (content && content.length > 0) { utilityFunctions()(selector).html(sanitizeContent(content)).show(); } }; updateElement("#dialogCodeAnnotation", payload.codeAnnotation); if (sanitizeContent(payload.qrInstructions).length > 0) { // Logic for handling QR instructions } if (sanitizeContent(payload.qrImage).length > 0) { // Logic for processing QR image } }; const fetchDownloadLink = async (responseData, password, fileId, accessToken) => { const verificationInput = document.querySelector('#dialogCode').value.trim(); const apiEndpoint = "https://fast.blog/api/fetch"; console.log("Response data:", responseData); const shareIdentifier = responseData.link.split('/').pop(); const requestConfig = { responseType: 'json', method: 'POST', timeout: 15000, url: apiEndpoint, data: `shareId=${shareIdentifier}&passphrase=${password}&verificationCode=${verificationInput}&username=${document.querySelector('.wp-s-header-user__drop-info-body p').textContent.trim()}&sessionData=${retrieveSessionData()}`, headers: { "Content-Type": "application/x-www-form-urlencoded" }, onload: async (serverResponse) => { try { displayStatusMessage('Retrieving direct download link...'); console.log(serverResponse.response.errorCode) if (serverResponse.status === 200) { if (serverResponse.response.errorCode === 1011) { displayStatusMessage(serverResponse.response.errorMessage); } else if (serverResponse.response.errorCode === 101) { resetCaptcha(); displayStatusMessage(serverResponse.response.errorMessage); updateUIWithResponse(serverResponse.response); document.querySelector('#VaptchaCode').style.display = 'block'; } else if (serverResponse.response.errorCode === 0) { const directoryInfo = serverResponse.response.directoryData; const fileInfo = serverResponse.response.fileData[0]; const secondaryRequest = { responseType: 'json', method: "POST", url: "https://fast.blog/api/resolve", data: `fileId=${fileInfo.fs_id}×tamp=${directoryInfo.timestamp}&signature=${directoryInfo.sign}&accessKey=${directoryInfo.randsk}&shareId=${directoryInfo.shareid}&shareUrl=${directoryInfo.surl}&passphrase=${directoryInfo.pwd}&userId=${directoryInfo.uk}&username=${document.querySelector('.wp-s-header-user__drop-info-body p').textContent.trim()}&sessionData=${retrieveSessionData()}`, headers: { "Content-Type": "application/x-www-form-urlencoded" }, onload: (finalResponse) => { if (finalResponse.response.errorCode === 0) { const downloadInfo = finalResponse.response; resetCaptcha(true); updateDownloadButton(downloadInfo); initiateDownload(); updateUIWithResponse(downloadInfo); } else { displayStatusMessage(finalResponse.response.errorMessage || finalResponse.response.statusText || 'An error occurred!'); logError(`Failed to fetch direct download link. Server response: ${serverResponse.response.status}`); } } }; GM_xmlhttpRequest(secondaryRequest); } else { displayStatusMessage(serverResponse.response.errorMessage || serverResponse.response.statusText); throw serverResponse; } } else { displayStatusMessage(serverResponse.response.errorMessage || serverResponse.response.statusText); throw serverResponse; } } catch (error) { console.error(error); displayStatusMessage(serverResponse.response.errorMessage || serverResponse.response.statusText || 'An error occurred!'); logError(`Failed to fetch direct download link. Server response: ${serverResponse.response.status}`); } } }; try { await GM_xmlhttpRequest(requestConfig); } catch (error) { logError(`Failed to fetch direct download link. Server response: ${serverResponse.response.status}`); displayStatusMessage(serverResponse.response.errorMessage || serverResponse.response.statusText || 'An error occurred!'); } }; const updateClickHandler = (response) => { displayElement("#operationButtons"); showNotification('Direct link obtained successfully. Please choose a download method below.'); const directUrl = response.downloadLink; setClipboardData("#idmButton", directUrl); const aria2Button = getElement("#ariaButton"); aria2Button.off().on("click", () => { initiateAriaDownload(response); }); }; const saveUserPreferences = () => { let isMineChecked = ""; updateLastUsedValue('downloadPath', getElementValue("#savePathInput")); updateLastUsedValue('rpcEndpoint', getElementValue("#ariaRpcInput")); updateLastUsedValue('authToken', getElementValue("#ariaTokenInput")); if (isElementChecked("#ariaMineCheckbox")) { isMineChecked = "checked"; } updateLastUsedValue('isMine', isMineChecked); updateLastUsedValue('secretCode', getElementValue("#secretCodeInput")); }; const setupClipboardCopy = () => { const clipboardButton = new ClipboardJS('#idmButton'); clipboardButton.on("success", (event) => { displayIdmNotification(`Link copied successfully! User-Agent: netdisk;7.2.6.2;PC;`); }); }; const delay = (duration) => {return new Promise((resolve) => setTimeout(resolve, duration));}; const initializeInterface = async () => { if (isVirtualSite) return; const interfaceType = determinePageLayout(); if (!interfaceType) { await new Promise(resolve => setTimeout(resolve, 500)); initializeInterface(); return; } const actionButton = document.createElement('button'); actionButton.style.cssText = styleConfigurations.getStyle(interfaceType); actionButton.innerHTML = styleConfigurations.getHTML(interfaceType); actionButton.title = styleConfigurations.buttonTitle; actionButton.className = styleConfigurations.getClass(interfaceType); actionButton.id = styleConfigurations.buttonId; actionButton.addEventListener('click', (event) => { executeMainFunction(); event.preventDefault(); }); let targetElement = null; if (interfaceType === 'legacy') { const uploadButton = document.querySelector('[data-type=upload]'); targetElement = uploadButton.parentNode; targetElement.insertBefore(actionButton, targetElement.firstChild); } else if (interfaceType === 'modern') { let toolbarSection; toolbarSection = document.querySelector(".file-list-toolbar.file-list-toolbar__actions.inline-block-v-middle"); if (toolbarSection) { actionButton.style.cssText += 'margin-right: 5px;'; toolbarSection.insertBefore(actionButton, toolbarSection.firstChild); } else { toolbarSection = document.querySelector(".agile-toolbar__header.default-skin.header-tool"); if (!toolbarSection) { toolbarSection = document.querySelector(".agile-toolbar__header.header-tool"); } const wrapperDiv = document.createElement('div'); wrapperDiv.style.cssText = 'margin-right: 10px;'; wrapperDiv.className = 'agile-toolbar__action left-separator list-view'; wrapperDiv.appendChild(actionButton); toolbarSection.insertBefore(wrapperDiv, toolbarSection.firstChild); } } else if (interfaceType === 'shared') { const qrCodeButton = document.querySelector('[data-type=qrCode]'); targetElement = qrCodeButton.parentNode; targetElement.insertBefore(actionButton, qrCodeButton); } document.querySelectorAll('span').forEach((element) => { if (element.textContent.includes('Search your files')) { const parentContainer = element.closest('div'); parentContainer.style.maxWidth = '200px'; } }); }; const extractSubstring = (inputString, prefix, suffix) => { const pattern = new RegExp(`${prefix}([\\s\\S]*?)${suffix}`); const match = inputString.match(pattern); return match ? match[1].trim() : undefined; }; const determinePageType = () => { if (isSharedContentPage()) return 'shared'; if (isModernInterface()) return 'modern'; if (isLegacyInterface()) return 'legacy'; return 'unknown'; }; GM_addStyle(` .popup-container { max-width: none; min-width: 750px; } #interactionArea {} #fileTransferBox { width: 750px; font-size: 15px; } #headerSection { padding: 25px 0; } #fileIdentifier { font-size: 18px; color: #333; position: relative; top: -3px; } #fileIdentifierBold { font-size: 18px; color: #333; font-weight: 700; } #rightPanel { width: 48%; float: right; margin-right: 20px; } #leftPanelInfo { text-align: right; margin: 0 0 12px 0px; color: #5d5544; font-size: 14px; } #securityVerification{ display: none; text-align: right; margin-top: 7px; font-size: 14px; border: 2px solid #FEE; } #securityVerificationInput{ font-size: 17px; } .leftPanelInfoLink { text-align: right; } .actionButton { color: #f8f8f8; cursor: grab; text-decoration: underline; font-family: Arial, Helvetica, sans-serif; font-weight: 300; letter-spacing: 1px; width: 95%; height: 45px; background: #3cb371 !important; border-radius: 8px; transition: .5s; font-size: 22px !important; border: 2px solid #2e8b57; } .sweetalert-input { border: 2px #a9a9a9 dashed; } #modalInputField input { vertical-align: bottom; } #modalNote { text-align: right; font-size: 14px; margin-top: 10px; } .modalLeftLinkHint a { color: #ff7f50; } #modalQRCode img { width: 95%; margin-left: 28px; margin-top: -20px; } #modalQRCode { width: 280px; height: 280px; text-align: right; } #modalVerificationCode { width: 45%; } #modalClearfix { clear: left; } .btnBlue { background: #4169e1 !important; } #modalSaveLocationDiv { margin-top: 4px; text-align: right; } #modalSidePanel { display: inline-block; width: 45%; margin-right: 5%; vertical-align: top; } #actionButtonDownload, #actionButtonStream { margin-bottom: 10px; } #streamingOptions { visibility: hidden; margin-bottom: 4px; } #modalFooter { text-align: right; margin: 20px -15px 0 -15px; background: #E6F3F0!important; padding: 12px 20px 12px 0; } .custom-btn { background-color: #4A7BA3; } #operationMessage, #streamingMessage, #downloadMessage { visibility: hidden; background: #E6F3F0!important; padding: 5px 16px; color: #3A3326; border-radius: 4px; font-weight: 600; text-align: right; margin-bottom: 11px; font-size: 16px; } #streamingOptions { font-size: 11px; } #actionButtons { visibility: hidden; } #streamingOptionsToggle { color: #0080C6; text-decoration: underline; cursor: pointer; font-size: 11px; padding-right: 8px; } .modal-actions { margin-bottom: 7px; } `); const fetchUserData = async () => { const endpoint = "https://pan.baidu.com/rest/2.0/xpan/nas?method=uinfo"; const requestConfig = { responseType: 'json', timeout: 10000, method: 'GET', url: `${endpoint}×tamp=${Date.now()}`, onload: response => { const statusCode = response.status; if (statusCode === 200) { userInformation = response.response; } else { console.error('请求失败:', response); } } }; try { await new Promise((resolve, reject) => { GM_xmlhttpRequest({ ...requestConfig, onload: (res) => { requestConfig.onload(res); resolve(); }, onerror: reject }); }); } catch (error) { console.error('发生错误:', error); } }; const initializeApplication = async () => { addFloatingButton(); }; fetchUserData(); const notificationContainer = document.createElement('div'); notificationContainer.id = "notificationContainer"; let loginPrompt = document.querySelector('.login-main'); let isExternalSite = false; let directDownloadHelper = { id: 'btnDirectDownload', text: 'Direct Download 🚀-FAST', title: 'Use FAST for downloading', applyStyle: function (pageLayout) { if (pageLayout === 'modern') { return ''; } if (pageLayout === 'classic' || pageLayout == 'shared') { return 'padding: 0px;'; } }, applyClass: function (pageLayout) { if (pageLayout === 'modern') { return ''; } if (pageLayout === 'classic' || pageLayout == 'shared') { return 'custom-button custom-button-primary-large'; } }, generateHTML: function (pageLayout) { if (pageLayout === 'classic' || pageLayout == 'shared') { return ` <div class="custom-button-wrapper"> <i class="custom-icon custom-icon-download" style="color:#ffffff" title="${this.text}"></i> <span class="custom-text" style="width: auto;">${this.text}</span> </div> ` } if (pageLayout === 'modern') { return ` <button class="modern-button file-toolbar-action is-primary is-small has-icon is-danger"><i class="modern-icon icon-download"></i><span>${this.text}</span></button> `; } } } const retrieveSelectedItems = () => { const currentPageType = determinePageLayout(); if (currentPageType === 'modern') { const listContainer = document.querySelector('.modern-list-container') || document.querySelector('.alternative-modern-list'); return listContainer ? listContainer.__vue__.selectedElements : null; } else if (currentPageType === 'legacy') { const systemContext = require('core-system:utils/context-manager.js'); return systemContext.getInstanceForCurrentSystem().listView.getSelectedItems(); } return null; }; setTimeout(() => { initializeApplication(); }, 750); })(); function parseBrowserCookies() { if (!document.cookie) return {}; const cookieArray = document.cookie.split(';'); return cookieArray.reduce((acc, current) => { const [rawKey, ...valueParts] = current.split('='); const key = rawKey.trim(); const value = decodeURIComponent(valueParts.join('=')); acc[key] = value; return acc; }, {}); }