您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
拖拽链接时在新标签页打开
// ==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(); }); })();