您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Greasy Fork is available in English.
自动复制,鼠标选中,直接复制,省去你的左手。
当前为
// ==UserScript== // @name 自动复制 // @name:zh-CN 自动复制 // @name:en Auto-Copy // @name:ja オートコピー // @name:ko 자동 복사 // @name:es Copia Automática // @namespace gura8390/copy/1 // @license MIT // @version 2.2 // @description 自动复制,鼠标选中,直接复制,省去你的左手。 // @description:en Automatically copy selected text to clipboard // @description:ja 選択したテキストを自動的にコピーします // @description:ko 선택한 텍스트를 자동으로 복사합니다 // @description:es Copia automáticamente el texto seleccionado // @author lbihhe // @icon  // @match *://*/* // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @grant GM_addStyle // ==/UserScript== (function() { 'use strict'; const STORAGE_KEY = 'auto_copy_enabled'; let isEnabled = GM_getValue(STORAGE_KEY, true); let menuCommands = []; // ======================== // 多语言支持系统 // ======================== const i18n = { 'en': { panelTitle: 'Auto Copy Control', toggleBtn: isEnabled => isEnabled ? '🟢 Enabled' : '🔴 Disabled', statusText: isEnabled => `Status: ${isEnabled ? 'Working' : 'Paused'}`, copied: 'Text copied', copyFailed: 'Copy failed', enableNotify: 'Auto copy enabled', disableNotify: 'Auto copy disabled', openPanel: 'Open Control Panel', closePanel: 'Close Control Panel', menuEnable: 'Enable Auto Copy', menuDisable: 'Disable Auto Copy' }, 'zh-CN': { panelTitle: '智能复制控制', toggleBtn: isEnabled => isEnabled ? '已启用' : '已禁用', statusText: isEnabled => `当前状态: ${isEnabled ? '✓ 运行中' : '✗ 已暂停'}`, copied: '文本已复制', copyFailed: '复制失败', enableNotify: '自动复制功能已启用', disableNotify: '自动复制功能已禁用', openPanel: '打开控制面板', closePanel: '关闭控制面板', menuEnable: '启用自动复制', menuDisable: '禁用自动复制' }, 'ja': { panelTitle: '自動コピー制御', toggleBtn: isEnabled => isEnabled ? '有効' : '無効', statusText: isEnabled => `状態: ${isEnabled ? '✓ 動作中' : '✗ 停止中'}`, copied: 'コピーされました', copyFailed: 'コピー失敗', enableNotify: '自動コピーが有効になりました', disableNotify: '自動コピーが無効になりました', openPanel: 'コントロールパネルを開く', closePanel: 'コントロールパネルを閉じる', menuEnable: '自動コピーを有効にする', menuDisable: '自動コピーを無効にする' }, 'ko': { panelTitle: '자동 복사 제어', toggleBtn: isEnabled => isEnabled ? '활성화됨' : '비활성화됨', statusText: isEnabled => `상태: ${isEnabled ? '✓ 작동 중' : '✗ 일시 중지됨'}`, copied: '텍스트가 복사되었습니다', copyFailed: '복사 실패', enableNotify: '자동 복사가 활성화되었습니다', disableNotify: '자동 복사가 비활성화되었습니다', openPanel: '제어판 열기', closePanel: '제어판 닫기', menuEnable: '자동 복사 활성화', menuDisable: '자동 복사 비활성화' }, 'es': { panelTitle: 'Control de Copia Automática', toggleBtn: isEnabled => isEnabled ? 'Activado' : 'Desactivado', statusText: isEnabled => `Estado: ${isEnabled ? '✓ Trabajando' : '✗ Pausado'}`, copied: 'Texto copiado', copyFailed: 'Error al copiar', enableNotify: 'Copia automática activada', disableNotify: 'Copia automática desactivada', openPanel: 'Abrir Panel de Control', closePanel: 'Cerrar Panel de Control', menuEnable: 'Activar Copia Automática', menuDisable: 'Desactivar Copia Automática' } }; // 自动检测浏览器语言 const getBrowserLang = () => { const lang = navigator.languages?.[0] || navigator.language || 'en'; return lang.split('-')[0]; }; const currentLang = Object.keys(i18n).find(l => l.startsWith(getBrowserLang())) || 'en'; const t = i18n[currentLang] || i18n.en; // ======================== // 视觉反馈系统 // ======================== const createFeedback = (textKey, e) => { const feedback = document.createElement('div'); feedback.innerHTML = ` <div class="feedback-container"> <svg class="feedback-icon" viewBox="0 0 24 24"> ${textKey === 'copied' ? '<path fill="currentColor" d="M9 16.17L4.83 12l-1.42 1.41L9 19 21 7l-1.41-1.41z"/>' : '<path fill="currentColor" d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"/>'} </svg> <span class="feedback-text">${t[textKey]}</span> </div> `; Object.assign(feedback.style, { position: 'fixed', left: `${Math.min(e.clientX + 10, window.innerWidth - 200)}px`, top: `${e.clientY + 20}px`, zIndex: 2147483647, pointerEvents: 'none' }); document.body.appendChild(feedback); setTimeout(() => feedback.remove(), 2000); }; // ======================== // 状态指示系统 // ======================== const createStatusIcon = () => { const icon = document.createElement('div'); icon.className = 'status-icon'; icon.innerHTML = ` <svg viewBox="0 0 24 24"> <circle cx="12" cy="12" r="10" class="status-base"/> <path class="status-indicator" d="M8 12l3 3 5-6"/> </svg> `; Object.assign(icon.style, { position: 'fixed', bottom: '20px', right: '-10px', width: '40px', height: '40px', cursor: 'pointer', zIndex: 2147483647, transition: 'right 0.3s ease, transform 0.3s ease, opacity 0.3s ease', opacity: '0.8' }); icon.addEventListener('click', () => { const panel = document.querySelector('.control-panel'); if (panel) { panel.remove(); updateMenuCommands(); } else { toggleFeature(); } }); icon.addEventListener('mouseenter', () => { icon.style.transition = 'none'; icon.style.right = '20px'; icon.style.transform = 'scale(1.1)'; icon.style.opacity = '1'; }); icon.addEventListener('mouseleave', () => { icon.style.transition = 'right 0.3s ease, transform 0.3s ease, opacity 0.3s ease'; icon.style.right = '-10px'; icon.style.transform = 'scale(1)'; icon.style.opacity = '0.8'; }); updateIconStyle(icon); return icon; }; const updateIconStyle = (icon) => { const color = isEnabled ? '#4CAF50' : '#F44336'; icon.style.filter = `drop-shadow(0 2px 4px ${color}40)`; icon.querySelector('.status-indicator').style.stroke = color; icon.querySelector('.status-base').style.stroke = color; }; // ======================== // 控制面板系统 // ======================== const createControlPanel = () => { const panel = document.createElement('div'); panel.className = 'control-panel'; panel.innerHTML = ` <div class="panel-header"> <h3>${t.panelTitle}</h3> <div class="close-btn"> <svg viewBox="0 0 24 24" width="24" height="24"> <path fill="currentColor" d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"/> </svg> </div> </div> <div class="panel-body"> <button class="toggle-btn ${isEnabled ? 'active' : ''}"> ${t.toggleBtn(isEnabled)} </button> <div class="status-text"> ${t.statusText(isEnabled)} </div> </div> `; panel.querySelector('.toggle-btn').addEventListener('click', toggleFeature); panel.querySelector('.close-btn').addEventListener('click', () => { panel.style.transform = 'translateY(100%)'; setTimeout(() => { panel.remove(); updateMenuCommands(); }, 300); }); return panel; }; // ======================== // 核心功能逻辑 // ======================== const handleSelection = (e) => { if (!isEnabled) return; const selection = window.getSelection().toString().trim(); if (!selection) return; navigator.clipboard.writeText(selection) .then(() => createFeedback('copied', e)) .catch(err => { createFeedback('copyFailed', e); console.error('Copy failed:', err); }); }; const toggleFeature = () => { isEnabled = !isEnabled; GM_setValue(STORAGE_KEY, isEnabled); updateIconStyle(statusIcon); showGlobalNotification(); if (document.querySelector('.control-panel')) refreshControlPanel(); }; // ======================== // 辅助功能 // ======================== const showGlobalNotification = () => { const notification = document.createElement('div'); notification.className = 'global-notification'; notification.textContent = isEnabled ? t.enableNotify : t.disableNotify; document.body.appendChild(notification); setTimeout(() => notification.remove(), 1500); }; const refreshControlPanel = () => { document.querySelector('.control-panel')?.remove(); document.body.appendChild(createControlPanel()); updateMenuCommands(); }; // ======================== // 菜单管理系统 // ======================== const updateMenuCommands = () => { // 清除旧菜单 menuCommands.forEach(cmd => GM_unregisterMenuCommand(cmd)); menuCommands = []; // 注册新菜单 if (document.querySelector('.control-panel')) { menuCommands.push(GM_registerMenuCommand(t.closePanel, () => { document.querySelector('.control-panel')?.remove(); updateMenuCommands(); })); } else { menuCommands.push(GM_registerMenuCommand(t.openPanel, () => { document.body.appendChild(createControlPanel()); updateMenuCommands(); })); } menuCommands.push(GM_registerMenuCommand( isEnabled ? t.menuDisable : t.menuEnable, toggleFeature )); }; // ======================== // 样式系统 // ======================== GM_addStyle(` .feedback-container { background: linear-gradient(145deg, #f8f9fa, #ffffff); border-radius: 12px; padding: 12px 16px; box-shadow: 0 4px 12px rgba(0,0,0,0.1); display: flex; align-items: center; gap: 8px; animation: slideIn 0.3s ease, fadeOut 0.5s ease 1.5s forwards; } .feedback-icon { width: 20px; height: 20px; color: ${isEnabled ? '#4CAF50' : '#F44336'}; } .feedback-text { color: #2d3436; font-family: -apple-system, system-ui; font-size: 14px; } .status-icon:hover { transform: scale(1.1); opacity: 1; } .control-panel { position: fixed; bottom: 80px; right: 20px; background: rgba(255,255,255,0.98); border-radius: 16px; box-shadow: 0 8px 32px rgba(0,0,0,0.15); padding: 24px; width: 280px; transform: translateY(0); transition: transform 0.3s cubic-bezier(0.4, 0, 0.2, 1); backdrop-filter: blur(16px); border: 1px solid rgba(0,0,0,0.1); z-index: 2147483647; } .panel-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 16px; } .panel-header h3 { margin: 0; font-size: 18px; font-weight: 600; color: #2d3436; } .close-btn { cursor: pointer; padding: 8px; border-radius: 50%; transition: all 0.2s ease; display: flex; color: rgba(0,0,0,0.5); } .close-btn:hover { background: rgba(0,0,0,0.05); color: rgba(0,0,0,0.8); transform: rotate(90deg); } .toggle-btn { width: 100%; padding: 16px; border: none; border-radius: 12px; background: linear-gradient(145deg, #f0f0f0, #ffffff); color: #666; font-weight: 500; cursor: pointer; transition: all 0.3s ease; box-shadow: 0 4px 12px rgba(0,0,0,0.1); } .toggle-btn.active { background: linear-gradient(145deg, #4CAF50, #45a049); color: white; box-shadow: 0 4px 16px rgba(76,175,80,0.3); } .status-text { margin-top: 16px; text-align: center; color: #666; font-size: 13px; } .status-text span { color: ${isEnabled ? '#4CAF50' : '#F44336'}; font-weight: bold; } .global-notification { position: fixed; top: 20px; left: 50%; transform: translateX(-50%); background: #333; color: #fff; padding: 8px 16px; border-radius: 8px; z-index: 2147483647; opacity: 0.9; } @keyframes slideIn { from { transform: translateY(20px); opacity: 0; } to { transform: translateY(0); opacity: 1; } } @keyframes fadeOut { from { opacity: 1; } to { opacity: 0; } } `); // ======================== // 初始化系统 // ======================== const statusIcon = createStatusIcon(); document.body.appendChild(statusIcon); document.addEventListener('mouseup', handleSelection); updateMenuCommands(); })();