// ==UserScript== // @name EdgeDL // @namespace https://github.com/Chumor/EdgeDL // @version 1.0.0 // @description EdgeDL:为 Android Edge 浏览器自动检测下载链接并调用外部下载管理器 // @icon  // @author Chumor // @match *://*/* // @grant none // @license Apache-2.0 // @run-at document-start // @downloadURL https://raw.githubusercontent.com/Chumor/EdgeDL/main/dist/EdgeDL.user.js // @updateURL https://raw.githubusercontent.com/Chumor/EdgeDL/main/dist/EdgeDL.user.js // ==/UserScript== (function () { 'use strict'; // 下载器用户配置 const DOWNLOADERS = { IDM: 'idm.internet.download.manager.plus', ADM: 'com.dv.adm' }; // 默认下载器,用户可选 .IDM / .ADM let selectedDownloader = DOWNLOADERS.IDM; // 下载链接关键字匹配 const EXTENSIONS = [ '.apk','.apks','.xapk','.apkm','.ipa','.obb','.aab', '.zip','.rar','.7z','.tar','.gz','.tgz','.bz2','.xz', '.iso','.cab','.jar','.z', '.mp4','.mkv','.avi','.mov','.flv','.wmv','.webm', '.m4v','.3gp','.ts','.mpg','.mpeg','.vob', '.mp3','.flac','.wav','.ogg','.m4a','.aac','.wma','.ape', '.pdf','.epub','.mobi','.azw3','.djvu', '.doc','.docx','.xls','.xlsx','.ppt','.pptx', '.exe','.msi','.bin','.dat','.dmg','.bat','.sh','.img', '.torrent' ]; // 下载链接后缀匹配 const KEYWORDS = [ '/down/','/download/','/downloads/','/dl/','/get/','/fetch/', '/files/','/file/','/attach/','/attachment/','/media/','/static/', '/assets/','/cdn/','/dist/','/repo/','/backup/','/upload/', '/releases/download/','/binary/','/pkg/', '?file=','&file=','?filename=','&filename=','?f=','&f=', 'download?','&download=','?download=','&download=', 'token=','auth_key=','download_token=','sig=','signature=', 'force_download','response-content-disposition=', 'content-disposition=attachment' ]; // 下载链接检测 function isDownloadLink(url){ if(!url || !url.startsWith('http')) return false; const lowerUrl = url.toLowerCase(); // 排除非下载页面 if( lowerUrl.includes('/login')||lowerUrl.includes('/register')|| lowerUrl.includes('/signin')||lowerUrl.includes('/signup')|| lowerUrl.includes('/logout')|| lowerUrl.includes('/account/')||lowerUrl.includes('/user/')|| lowerUrl.includes('/blob/')||lowerUrl.includes('/src/')|| lowerUrl.includes('/tree/') ) return false; // 后缀匹配 try{ const path = new URL(url).pathname.toLowerCase(); if(EXTENSIONS.some(ext=>path.endsWith(ext))) return true; }catch(e){ const path = lowerUrl.split('?')[0].split('#')[0]; if(EXTENSIONS.some(ext=>path.endsWith(ext))) return true; } // 关键字匹配 return KEYWORDS.some(kw=>lowerUrl.includes(kw)); } function openIDM(url) { const scheme = url.startsWith('https') ? 'https' : 'http'; const cleanLink = url.replace(/^https?:\/\//, ''); const intentUrl = `intent://${cleanLink}#Intent;scheme=${scheme};package=${DOWNLOADERS.IDM};type=*/*;end`; window.location.href = intentUrl; } function openADM(url) { const scheme = url.startsWith('https') ? 'https' : 'http'; const cleanLink = url.replace(/^https?:\/\//, ''); const intentUrl = `intent://${cleanLink}#Intent;scheme=${scheme};package=${DOWNLOADERS.ADM};type=*/*;end`; window.location.href = intentUrl; } // 根据用户选择唤起下载器 function openDownload(url){ switch(selectedDownloader){ case DOWNLOADERS.IDM: return openIDM(url); case DOWNLOADERS.ADM: return openADM(url); default: return openIDM(url); } } function showToast(message, duration = 1500) { try { const downloaderName = selectedDownloader === DOWNLOADERS.IDM ? 'IDM+' : 'ADM'; const toastMessage = message || `⚡ ${downloaderName} 正在唤起...`; const toast = document.createElement('div'); toast.textContent = toastMessage; toast.style.cssText = ` position: fixed; bottom: 15%; left: 50%; transform: translateX(-50%); background: rgba(0,0,0,0.85); color: #fff; font-weight: 500; font-size: 13px; padding: 10px 20px; border-radius: 12px; box-shadow: 0 4px 12px rgba(0,0,0,0.3); z-index: 999999; pointer-events: none; opacity: 0; transition: opacity 0.3s ease-in-out; `; document.body.appendChild(toast); requestAnimationFrame(() => { toast.style.opacity = '1'; }); setTimeout(() => { toast.style.opacity = '0'; toast.addEventListener('transitionend', () => toast.remove(), { once: true }); }, duration); } catch (err) { console.warn('Toast 创建失败', err); } } // 全局点击拦截下载 document.addEventListener('click', e=>{ let target = e.target; while(target && target.tagName!=='A') target=target.parentElement; if(!target) return; const url = target.href; if(isDownloadLink(url)){ e.preventDefault();e.stopPropagation();e.stopImmediatePropagation(); showToast(`⚡ ${selectedDownloader} 正在唤起`); openDownload(url); } }, true); })();