EdgeDL

EdgeDL:让 Android Edge 浏览器使用外部下载器下载文件

スクリプトをインストールするには、Tampermonkey, GreasemonkeyViolentmonkey のような拡張機能のインストールが必要です。

You will need to install an extension such as Tampermonkey to install this script.

スクリプトをインストールするには、TampermonkeyViolentmonkey のような拡張機能のインストールが必要です。

スクリプトをインストールするには、TampermonkeyUserscripts のような拡張機能のインストールが必要です。

このスクリプトをインストールするには、Tampermonkeyなどの拡張機能をインストールする必要があります。

このスクリプトをインストールするには、ユーザースクリプト管理ツールの拡張機能をインストールする必要があります。

(ユーザースクリプト管理ツールは設定済みなのでインストール!)

このスタイルをインストールするには、Stylusなどの拡張機能をインストールする必要があります。

このスタイルをインストールするには、Stylus などの拡張機能をインストールする必要があります。

このスタイルをインストールするには、Stylus tなどの拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

(ユーザースタイル管理ツールは設定済みなのでインストール!)

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください
// ==UserScript==
// @name         EdgeDL
// @namespace    https://github.com/Chumor/EdgeDL
// @version      1.2.0
// @description  EdgeDL:让 Android Edge 浏览器使用外部下载器下载文件
// @icon         
// @author       Chumor
// @match        *://*/*
// @grant        GM_registerMenuCommand
// @license      Apache-2.0
// @run-at       document-start
// ==/UserScript==


(function () {
    'use strict';

    // 下载器用户配置
    const DOWNLOADERS = {
        IDM: 'idm.internet.download.manager.plus',
        ADM: 'com.dv.adm',
    };

    // 默认下载器
    localStorage.getItem('edgedl-default-downloader') || 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, packageName) {
        const scheme = url.startsWith('https') ? 'https' : 'http';
        const cleanLink = url.replace(/^https?:\/\//, '');
        const intentUrl = `intent://${cleanLink}#Intent;scheme=${scheme};package=${packageName};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 showToast(message, duration = 1500) {
        try {
            const toast = document.createElement('div');
            toast.textContent = message;
            toast.style.cssText = `
            position: fixed;
            bottom: 15%;
            left: 50%;
            transform: translateX(-50%);
            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;
        `;

            if (window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches) {
                toast.style.background = 'rgba(255,255,255,0.12)';
                toast.style.color = '#fff';
            } else {
                toast.style.background = 'rgba(0,0,0,0.85)';
                toast.style.color = '#fff';
            }

            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);
        }
    }

    // 调用指定下载器下载
    function openDownload(url, downloader) {
        switch (downloader) {
            case 'idm.internet.download.manager':
                showToast('⚡ 1DM 正在唤起');
                openIDM(url, 'idm.internet.download.manager');
                break;

            case DOWNLOADERS.IDM:
                showToast('⚡ 1DM+ 正在唤起');
                openIDM(url, DOWNLOADERS.IDM);
                break;

            case DOWNLOADERS.ADM:
                showToast('⚡ ADM 正在唤起');
                openADM(url);
                break;

            case 'edge':
            default:
                showToast('⚡ Edge 内置下载');
                window.open(url, '_blank');
                break;
        }
    }

    function showDownloadPicker(url, callback) {
        if(document.getElementById('edgedl-picker')) return;

        const idmIcon = '';
        const idmPlusIcon = '';
        const admIcon = '';
        const edgeIcon = '';

        const picker = document.createElement('div');
        picker.id = 'edgedl-picker';
        picker.innerHTML = `
        <div class="edgedl-bg"></div>
        <div class="edgedl-card">
            <h3>选择下载器</h3>
            <div class="edgedl-options">
                <button data-pkg="${DOWNLOADERS.IDM}">
                    <img src="${idmIcon}" /> 1DM
                </button>
                <button data-pkg="${DOWNLOADERS.IDM}">
                    <img src="${idmPlusIcon}" /> 1DM+
                </button>
                <button data-pkg="${DOWNLOADERS.ADM}">
                    <img src="${admIcon}" /> ADM
                </button>
                <button data-pkg="edge">
                    <img src="${edgeIcon}" /> Edge
                </button>
            </div>
            <label style="margin-top: 12px; display: flex; align-items: center; gap: 6px; font-size: 13px;">
                <input type="checkbox" id="edgedl-set-default" /> 设为默认下载器
            </label>
        </div>
    `;

        document.documentElement.appendChild(picker);

        // 注入下载器选择弹窗样式
        const style = document.createElement('style');
        style.textContent = `
        #edgedl-picker { position: fixed; inset: 0; display: flex; justify-content: center; align-items: center; z-index: 2147483647; }
        #edgedl-picker .edgedl-bg { position: absolute; inset:0; background: rgba(0,0,0,0.45); backdrop-filter: blur(6px); animation: edgedl-fade-in .18s ease-out; }
        #edgedl-picker .edgedl-card { position: relative; background: #fff; border-radius: 24px; padding: 20px; width: 260px; box-shadow: 0 10px 28px rgba(0,0,0,0.25); display: flex; flex-direction: column; align-items: center; animation: edgedl-slide-up .22s ease-out; }
        #edgedl-picker h3 { margin: 0 0 16px 0; font-weight: 500; font-size: 16px; }
        #edgedl-picker .edgedl-options { display: flex; flex-direction: column; width: 100%; gap: 12px; }
        #edgedl-picker .edgedl-options button { display: flex; align-items: center; gap: 10px; padding: 10px; border: none; border-radius: 12px; background: #f2f2f2; font-weight: 500; cursor: pointer; transition: background 0.2s; }
        #edgedl-picker .edgedl-options button:hover { background: #e0e0e0; }
        #edgedl-picker .edgedl-options img { width: 24px; height: 24px; }
        @media (prefers-color-scheme: dark) {
            #edgedl-picker .edgedl-card { background: #1E1E1E; color: #FFFFFF; box-shadow: 0 4px 16px rgba(0,0,0,0.6); border-radius: 24px; }
            #edgedl-picker .edgedl-options button { background: #2C2C2C; color: #FFFFFF; transition: background 0.2s, box-shadow 0.2s; }
            #edgedl-picker .edgedl-options button:hover { background: #3A3A3A; box-shadow: 0 2px 8px rgba(0,0,0,0.4); }
            #edgedl-picker .edgedl-options button:active { background: #4A4A4A; box-shadow: 0 1px 4px rgba(0,0,0,0.5); }
            #edgedl-picker .edgedl-options img { filter: brightness(1.5) contrast(1.2); }
            #edgedl-picker .edgedl-options button { background: #2C2C2C; color: #FFFFFF; }
        }

        @keyframes edgedl-fade-in {
            from { opacity: 0; }
            to { opacity: 1; }
        }

        @keyframes edgedl-slide-up {
            from { opacity: 0; transform: translateY(18px) scale(.98); }
            to { opacity: 1; transform: translateY(0) scale(1); }
        }
    `;
        document.head.appendChild(style);

        // 读取默认下载器
        const defaultDownloader = localStorage.getItem('edgedl-default-downloader');
        const defaultCheckbox = picker.querySelector('#edgedl-set-default');
        // 明确设置 checkbox 状态(存在则选中,否则不选中)
        defaultCheckbox.checked = !!defaultDownloader;
        if (defaultDownloader) {
            // 高亮默认下载器按钮
            const defaultBtn = picker.querySelector(`button[data-pkg="${defaultDownloader}"]`);
            if (defaultBtn) defaultBtn.classList.add('selected');
        }

        // 当复选框变化时立即保存或清除“待设置”标志(保证能在点击按钮时写入正确的包名)
        defaultCheckbox.addEventListener('change', () => {
            if (defaultCheckbox.checked) {
                // 标记为“用户希望设为默认”,但具体包名留到点击按钮时写入
                localStorage.setItem('edgedl-default-pending', '1');
            } else {
                // 取消:移除已保存的默认和待设置标志
                localStorage.removeItem('edgedl-default-downloader');
                localStorage.removeItem('edgedl-default-pending');
            }
        });
        
        // 按钮高亮样式
        style.textContent += `
        #edgedl-picker .edgedl-options button.selected {
            outline: 2px solid #4CAF50;
        }
    `;
        
        // 点击唤起
        picker.querySelectorAll('button').forEach(btn => {
            btn.addEventListener('click', () => {
                const pkg = btn.dataset.pkg;

                // 若复选框已勾选或之前标记为“待设置”,保存为默认;否则清除默认(确保下次继续弹窗)
                if (defaultCheckbox.checked || localStorage.getItem('edgedl-default-pending')) {
                    localStorage.setItem('edgedl-default-downloader', pkg);
                    localStorage.removeItem('edgedl-default-pending');
                } else {
                    localStorage.removeItem('edgedl-default-downloader');
                }

                // 调用回调唤起下载器
                if (typeof callback === 'function') callback(pkg);

                picker.remove();
                
                // 通知外部(例如油猴菜单注册器)弹窗已关闭,方便重注册菜单
                window.dispatchEvent(new CustomEvent('edgedl:picker-closed'));
            });
        });

        // 点击背景关闭
        picker.querySelector('.edgedl-bg').addEventListener('click', () => {
            picker.remove();
            // 通知外部(例如油猴菜单注册器)弹窗已关闭,方便重注册菜单
            window.dispatchEvent(new CustomEvent('edgedl:picker-closed'));
        });
    }

    let menuRegistered = false;

    function registerMenu() {
        if (menuRegistered) return;
        if (typeof GM_registerMenuCommand !== 'function') return;

        GM_registerMenuCommand('更改默认下载器', () => {
            showDownloadPicker('', pkg => {
                localStorage.setItem('edgedl-default-downloader', pkg);
            });
        });

        menuRegistered = true;
    }

    registerMenu();

    // 获取当前默认下载器
    function getDefaultDownloader() {
        return localStorage.getItem('edgedl-default-downloader');
    }

    // 全局点击拦截下载
    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();

            const defaultDownloader = getDefaultDownloader();
            if (defaultDownloader) {
                openDownload(url, defaultDownloader);
            } else {
                // 弹出下载选择器
                showDownloadPicker(url, selected => {
                    openDownload(url, selected);
                });
            }
        }
    }, true);

})();