// ==UserScript== // @name 自动密码生成器 // @namespace http://tampermonkey.net/ // @version 1.1 // @description 在检测到密码字段时,提供生成、填充和复制密码的功能 // @author jwze // @match *://*/* // @grant GM_setValue // @grant GM_getValue // @grant GM_setClipboard // @grant GM_registerMenuCommand // @grant GM_addStyle // @run-at document-end // @license MIT // @icon  // @downloadURL https://update.greasyfork.icu/scripts/558418/%E8%87%AA%E5%8A%A8%E5%AF%86%E7%A0%81%E7%94%9F%E6%88%90%E5%99%A8.user.js // @updateURL https://update.greasyfork.icu/scripts/558418/%E8%87%AA%E5%8A%A8%E5%AF%86%E7%A0%81%E7%94%9F%E6%88%90%E5%99%A8.meta.js // ==/UserScript== (function() { 'use strict'; // ==================== 配置和常量 ==================== const PASSWORD_KEYWORDS = ['password', '密码', 'пароль', 'contraseña', 'senha', 'mot de passe', 'passwort']; // 内联SVG图标(钥匙图标) const ICON_SVG = `data:image/svg+xml,${encodeURIComponent(` `)}`; // ==================== 全局变量 ==================== let lastFocusedPasswordField = null; let currentLogoButton = null; let isPasswordFieldActive = false; let generatedPassword = ''; // ==================== 样式注入 ==================== GM_addStyle(` #ap-logo-button { position: absolute !important; width: 20px !important; height: 20px !important; cursor: pointer !important; z-index: 2147483647 !important; background-size: contain !important; background-repeat: no-repeat !important; background-position: center !important; opacity: 0.8 !important; transition: opacity 0.2s !important; border: none !important; outline: none !important; padding: 0 !important; margin: 0 !important; } #ap-logo-button:hover { opacity: 1 !important; } /* 设置面板样式 */ #ap-settings-panel { position: fixed !important; top: 50% !important; left: 50% !important; transform: translate(-50%, -50%) !important; background-color: #ffffff !important; border: 1px solid #dcdcdc !important; border-radius: 12px !important; box-shadow: 0 8px 32px rgba(0,0,0,0.2) !important; padding: 20px !important; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif !important; font-size: 14px !important; line-height: 1.5 !important; color: #333 !important; min-width: 320px !important; max-width: 400px !important; box-sizing: border-box !important; z-index: 2147483647 !important; } #ap-settings-panel h3 { margin: 0 0 16px 0 !important; font-size: 18px !important; font-weight: 600 !important; color: #333 !important; border-bottom: 1px solid #eee !important; padding-bottom: 12px !important; } #ap-settings-panel .ap-setting-row { display: flex !important; align-items: center !important; margin-bottom: 12px !important; padding: 8px 0 !important; } #ap-settings-panel label { display: flex !important; align-items: center !important; cursor: pointer !important; color: #555 !important; font-weight: normal !important; } #ap-settings-panel input[type="checkbox"] { width: 16px !important; height: 16px !important; margin-right: 10px !important; cursor: pointer !important; } #ap-settings-panel input[type="number"] { width: 60px !important; padding: 6px 8px !important; border: 1px solid #ccc !important; border-radius: 4px !important; font-size: 14px !important; margin-left: 10px !important; } #ap-settings-panel .ap-password-preview { background-color: #f5f5f5 !important; padding: 12px !important; border-radius: 6px !important; margin: 16px 0 !important; word-break: break-all !important; font-family: monospace !important; font-size: 14px !important; color: #007bff !important; text-align: center !important; } #ap-settings-panel .ap-btn-row { display: flex !important; gap: 10px !important; margin-top: 16px !important; } #ap-settings-panel button { flex: 1 !important; padding: 10px 16px !important; border: none !important; border-radius: 6px !important; cursor: pointer !important; font-size: 14px !important; font-weight: 500 !important; transition: background-color 0.2s !important; } #ap-settings-panel .ap-btn-primary { background-color: #4CAF50 !important; color: white !important; } #ap-settings-panel .ap-btn-primary:hover { background-color: #45a049 !important; } #ap-settings-panel .ap-btn-secondary { background-color: #f0f0f0 !important; color: #333 !important; } #ap-settings-panel .ap-btn-secondary:hover { background-color: #e0e0e0 !important; } #ap-settings-panel .ap-btn-danger { background-color: #f44336 !important; color: white !important; } #ap-settings-panel .ap-btn-danger:hover { background-color: #d32f2f !important; } /* 黑名单管理样式 */ #ap-settings-panel .ap-blacklist-section { margin-top: 16px !important; padding-top: 16px !important; border-top: 1px solid #eee !important; } #ap-settings-panel .ap-blacklist-list { max-height: 150px !important; overflow-y: auto !important; border: 1px solid #eee !important; border-radius: 4px !important; margin: 8px 0 !important; } #ap-settings-panel .ap-blacklist-item { display: flex !important; justify-content: space-between !important; align-items: center !important; padding: 8px 12px !important; border-bottom: 1px solid #eee !important; } #ap-settings-panel .ap-blacklist-item:last-child { border-bottom: none !important; } #ap-settings-panel .ap-blacklist-item button { flex: none !important; padding: 4px 8px !important; font-size: 12px !important; background-color: #ff6b6b !important; color: white !important; } #ap-settings-panel .ap-overlay { position: fixed !important; top: 0 !important; left: 0 !important; right: 0 !important; bottom: 0 !important; background-color: rgba(0,0,0,0.5) !important; z-index: 2147483646 !important; } `); // ==================== 存储管理 ==================== function getSettings() { return { includeSymbols: GM_getValue('includeSymbols', true), length: GM_getValue('passwordLength', 16) }; } function saveSettings(settings) { GM_setValue('includeSymbols', settings.includeSymbols); GM_setValue('passwordLength', settings.length); } function getBlacklist() { return GM_getValue('blacklist', []); } function addToBlacklist(domain) { const blacklist = getBlacklist(); const mainDomain = domain.split('.').slice(-2).join('.'); if (!blacklist.includes(mainDomain)) { blacklist.push(mainDomain); GM_setValue('blacklist', blacklist); } } function removeFromBlacklist(domain) { let blacklist = getBlacklist(); blacklist = blacklist.filter(item => item !== domain); GM_setValue('blacklist', blacklist); } function isSiteBlacklisted() { const hostname = window.location.hostname; if (!hostname) return true; const blacklist = getBlacklist(); return blacklist.some(domain => hostname.endsWith(domain)); } // ==================== 密码生成 ==================== function generateRandomPassword(options = {}) { const settings = getSettings(); const includeSymbols = options.includeSymbols !== undefined ? options.includeSymbols : settings.includeSymbols; const length = options.length || settings.length; const alphaNum = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"; const symbols = "!@#$%^&*()_+~`|}{[]:;?><,./-="; const charset = includeSymbols ? alphaNum + symbols : alphaNum; let password = ""; const array = new Uint32Array(length); crypto.getRandomValues(array); for (let i = 0; i < length; i++) { password += charset.charAt(array[i] % charset.length); } return password; } // ==================== 密码框检测 ==================== function checkAndMarkPasswordField(input) { if (!input || input.tagName !== 'INPUT' || input.readOnly) return false; if (input.getAttribute('data-ap-processed') === 'true') { return input.getAttribute('data-ap-is-password') === 'true'; } input.setAttribute('data-ap-processed', 'true'); const inputType = (input.type || 'text').toLowerCase(); if (inputType === 'password') { input.setAttribute('data-ap-is-password', 'true'); return true; } if (inputType === 'text') { const lowerPlaceholder = (input.placeholder || '').toLowerCase(); const lowerAriaLabel = (input.getAttribute('aria-label') || '').toLowerCase(); if (PASSWORD_KEYWORDS.some(kw => lowerPlaceholder.includes(kw) || lowerAriaLabel.includes(kw))) { input.setAttribute('data-ap-is-password', 'true'); return true; } const label = input.id ? document.querySelector(`label[for="${input.id}"]`) : input.closest('label'); if (label) { const lowerLabelText = (label.textContent || '').toLowerCase(); if (PASSWORD_KEYWORDS.some(kw => lowerLabelText.includes(kw))) { input.setAttribute('data-ap-is-password', 'true'); return true; } } } return false; } function scanForPasswordFields() { const inputs = document.querySelectorAll('input:not([data-ap-processed])'); inputs.forEach(input => checkAndMarkPasswordField(input)); } // ==================== Logo按钮 ==================== function showLogoButton(inputField) { if (isSiteBlacklisted()) return; hideLogoButton(); const button = document.createElement('div'); button.id = 'ap-logo-button'; button.title = '点击生成并填充密码'; button.style.backgroundImage = `url("${ICON_SVG}")`; button.addEventListener('click', (e) => { e.preventDefault(); e.stopPropagation(); // 生成密码 generatedPassword = generateRandomPassword(); // 填充到输入框 inputField.value = generatedPassword; inputField.dispatchEvent(new InputEvent('input', { bubbles: true })); inputField.dispatchEvent(new Event('change', { bubbles: true })); // 复制到剪贴板 - 使用 navigator.clipboard 作为主要方法 if (navigator.clipboard && navigator.clipboard.writeText) { navigator.clipboard.writeText(generatedPassword).then(() => { console.log('密码已复制到剪贴板'); }).catch(err => { console.error('使用 navigator.clipboard 复制失败:', err); fallbackCopy(); }); } else { fallbackCopy(); } function fallbackCopy() { try { GM_setClipboard(generatedPassword, 'text'); console.log('使用 GM_setClipboard 复制成功'); } catch (err) { console.error('GM_setClipboard 复制失败:', err); // 最后的降级方案 const textArea = document.createElement('textarea'); textArea.value = generatedPassword; textArea.style.position = 'fixed'; textArea.style.opacity = '0'; textArea.style.left = '-9999px'; document.body.appendChild(textArea); textArea.select(); textArea.focus(); const success = document.execCommand('copy'); document.body.removeChild(textArea); if (success) { console.log('使用 execCommand 复制成功'); } else { console.error('所有复制方法都失败了'); } } } // 隐藏按钮 hideLogoButton(); }); document.body.appendChild(button); currentLogoButton = button; positionLogoButton(inputField); } function positionLogoButton(inputField) { if (!currentLogoButton || !inputField) return; const rect = inputField.getBoundingClientRect(); const scrollX = window.scrollX || window.pageXOffset; const scrollY = window.scrollY || window.pageYOffset; currentLogoButton.style.left = `${rect.right + scrollX - 26}px`; currentLogoButton.style.top = `${rect.top + scrollY + (rect.height - 20) / 2}px`; } function hideLogoButton() { if (currentLogoButton) { currentLogoButton.remove(); currentLogoButton = null; } } // ==================== 设置面板 ==================== function showSettingsPanel() { // 移除已存在的面板 const existingPanel = document.getElementById('ap-settings-panel'); const existingOverlay = document.querySelector('.ap-overlay'); if (existingPanel) existingPanel.remove(); if (existingOverlay) existingOverlay.remove(); const settings = getSettings(); const blacklist = getBlacklist(); const previewPassword = generateRandomPassword(); // 创建遮罩 const overlay = document.createElement('div'); overlay.className = 'ap-overlay'; // 创建面板 const panel = document.createElement('div'); panel.id = 'ap-settings-panel'; panel.innerHTML = `

🔐 自动密码生成器 设置

${previewPassword}
黑名单管理

以下网站不会显示密码生成按钮:

${blacklist.length === 0 ? '
黑名单为空
' : blacklist.map(domain => `
${domain}
`).join('')}
`; document.body.appendChild(overlay); document.body.appendChild(panel); // 事件绑定 const includeSymbolsCheckbox = panel.querySelector('#ap-include-symbols'); const passwordLengthInput = panel.querySelector('#ap-password-length'); const previewEl = panel.querySelector('#ap-preview-password'); const regenerateBtn = panel.querySelector('#ap-regenerate'); const copyBtn = panel.querySelector('#ap-copy'); const closeBtn = panel.querySelector('#ap-close-settings'); const addCurrentSiteBtn = panel.querySelector('#ap-add-current-site'); function updatePreview() { const newSettings = { includeSymbols: includeSymbolsCheckbox.checked, length: parseInt(passwordLengthInput.value, 10) || 16 }; saveSettings(newSettings); previewEl.textContent = generateRandomPassword(newSettings); } includeSymbolsCheckbox.addEventListener('change', updatePreview); passwordLengthInput.addEventListener('change', updatePreview); regenerateBtn.addEventListener('click', updatePreview); copyBtn.addEventListener('click', () => { GM_setClipboard(previewEl.textContent, 'text'); copyBtn.textContent = '已复制!'; setTimeout(() => { copyBtn.textContent = '复制密码'; }, 1500); }); addCurrentSiteBtn.addEventListener('click', () => { addToBlacklist(window.location.hostname); hideLogoButton(); showSettingsPanel(); // 刷新面板 }); // 黑名单移除按钮 panel.querySelectorAll('.ap-blacklist-item button').forEach(btn => { btn.addEventListener('click', () => { removeFromBlacklist(btn.dataset.domain); showSettingsPanel(); // 刷新面板 }); }); // 关闭 const closePanel = () => { panel.remove(); overlay.remove(); }; closeBtn.addEventListener('click', closePanel); overlay.addEventListener('click', closePanel); } // ==================== 事件监听 ==================== document.addEventListener('focusin', (event) => { const target = event.target; if (checkAndMarkPasswordField(target)) { lastFocusedPasswordField = target; showLogoButton(target); isPasswordFieldActive = true; } }, true); document.addEventListener('focusout', (event) => { if (isPasswordFieldActive) { setTimeout(() => { if (document.activeElement && !checkAndMarkPasswordField(document.activeElement)) { isPasswordFieldActive = false; hideLogoButton(); } }, 150); } }, true); // 监听滚动和窗口调整 window.addEventListener('scroll', () => { if (currentLogoButton && lastFocusedPasswordField) { positionLogoButton(lastFocusedPasswordField); } }, true); window.addEventListener('resize', () => { if (currentLogoButton && lastFocusedPasswordField) { positionLogoButton(lastFocusedPasswordField); } }); // ==================== 初始化 ==================== // 扫描页面 scanForPasswordFields(); // MutationObserver 监视 DOM 变化 const observer = new MutationObserver(() => scanForPasswordFields()); observer.observe(document.body, { childList: true, subtree: true }); // 注册油猴菜单命令 GM_registerMenuCommand('⚙️ 打开设置', showSettingsPanel); GM_registerMenuCommand('🔑 生成并复制密码', () => { const password = generateRandomPassword(); GM_setClipboard(password, 'text'); alert('密码已生成并复制到剪贴板:\n\n' + password); }); GM_registerMenuCommand('🚫 将当前网站加入黑名单', () => { addToBlacklist(window.location.hostname); hideLogoButton(); alert('已将 ' + window.location.hostname + ' 加入黑名单'); }); })();