Greasy Fork

拖拽打开链接

拖拽链接时在新标签页打开

// ==UserScript==
// @name         拖拽打开链接
// @namespace    http://tampermonkey.net/
// @version      1.0
// @description  拖拽链接时在新标签页打开
// @license MIT
// @icon         
// @author       zm
// @match        http://*/*
// @match        https://*/*
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// ==/UserScript==

(async function() {
    'use strict';

    // 全局变量保存当前菜单标题
    let currentMenuTitle = null;
    let refreshTimeout = null;
    
    // 初始化配置项(首次运行时创建)
    const DEFAULT_CONFIG = {
        openInNewTab: false // 默认不启用新标签页打开
    };

    // 配置管理器
    const ConfigManager = {
        async init() {
            const savedConfig = await GM_getValue('drag_open_config');
            this.config = savedConfig ? JSON.parse(savedConfig) : {...DEFAULT_CONFIG};
        },
        async set(key, value) {
            this.config[key] = value;
            await GM_setValue('drag_open_config', JSON.stringify(this.config));
        },
        get(key) {
            return this.config[key];
        }
    };

    // 初始化配置
    await ConfigManager.init();

    // 注册菜单命令
    registerConfigMenu(ConfigManager);

    // 添加状态提示
    function showStatusNotification(enabled) {
        const msg = enabled ? '✅ 已启用:拖拽链接将在新标签打开' : '❌ 已禁用:拖拽链接不会在新标签打开';
        const toast = document.createElement('div');
        toast.style.cssText = `
            position: fixed;
            right: 30px;
            bottom: 30px;
            background: rgba(0,0,0,0.8);
            color: white;
            padding: 12px 24px;
            border-radius: 6px;
            z-index: 99999;
            font-family: sans-serif;
            transition: opacity 0.5s ease-out;
        `;
        toast.textContent = msg;
        document.body.appendChild(toast);
        setTimeout(() => {
            toast.style.opacity = '0';
            setTimeout(() => toast.remove(), 500);
        }, 2000);
    }

    // 菜单注册函数
    function registerConfigMenu(configManager) {
        const newTitle = `拖拽链接新标签页: ${configManager.get('openInNewTab') ? '✅ 启用' : '❌ 禁用'}`;
        
        if (currentMenuTitle === newTitle) return; // 完全相同直接返回

        // 防抖 300ms 避免连续快速调用
        // 清除之前的防抖定时器
        if (refreshTimeout) clearTimeout(refreshTimeout);
        // 设置新的防抖定时器,避免频繁刷新菜单
        refreshTimeout = setTimeout(async () => {
            try {
                // 如果已有菜单项,先注销
                if (currentMenuTitle) {
                    await GM_unregisterMenuCommand(currentMenuTitle);
                }
                // 注册新的菜单项,并在回调中处理点击逻辑
                await GM_registerMenuCommand(newTitle, async () => {
                    const newValue = !configManager.get('openInNewTab');
                    await configManager.set('openInNewTab', newValue);
                    showStatusNotification(newValue); // 显示状态提示
                    registerConfigMenu(configManager); // 递归刷新菜单
                });

                currentMenuTitle = newTitle;
            } catch (e) {
                console.error('菜单刷新失败:', e);
            }
        }, 300);

    }


    // 监听拖拽开始事件
    document.addEventListener('dragstart', function(e) {
         // 1. 阻止所有图片的拖拽事件(含链接包裹的图片)
        // if (e.target.tagName === 'IMG' || e.target.closest('a img')) {
        //     e.preventDefault();// 阻止默认拖拽行为
        //     e.stopImmediatePropagation();// 阻止其他监听器
        //     return;
        // }

        // 2. 保留文字链接的拖拽高亮功能
        if (e.target.tagName === 'A' && e.target.href) {
            // 添加高亮样式
            e.target.style.border = '2px dashed orange';
            e.target.style.padding = '2px';
            
            // 清除拖拽数据中的图片信息(防止Lens识别)
            e.dataTransfer.setData('text/plain', e.target.href);
        }
    },true);

    // 监听拖拽结束事件
    document.addEventListener('dragend', function(e) {
        // 移除高亮样式
        if (e.target.tagName === 'A' && e.target.href) {
            e.target.style.border = '';
            e.target.style.padding = '';
        }
    });

    // 监听拖拽释放事件(可选)
    document.addEventListener('drop', function(e) {
        e.preventDefault();
        // 仅允许纯文本链接释放
        const link = e.dataTransfer.getData('text/plain');
        if (link && isValidUrl(link.trim()) && ConfigManager.get('openInNewTab')) {
            window.open(link, '_blank');
        }

    });
    
    function isValidUrl(url) {
        try {
            // 自动补全缺少协议的URL(如 example.com => http://example.com)
            const fullUrl = url.includes('://') ? url : 'http://' + url;
            const parsed = new URL(fullUrl);
            return ['http:', 'https:'].includes(parsed.protocol);
        } catch (e) {
            return false;
        }
    }

    // 防止默认拖拽行为(必须启用),因为浏览器默认会阻止 drop 事件的触发,从而导致拖拽效果无效。
    document.addEventListener('dragover', function(e) {
        e.preventDefault();
    });

})();