Greasy Fork is available in English.
在检测到密码字段时,提供生成、填充和复制密码的功能
// ==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 
// ==/UserScript==
(function() {
'use strict';
// ==================== 配置和常量 ====================
const PASSWORD_KEYWORDS = ['password', '密码', 'пароль', 'contraseña', 'senha', 'mot de passe', 'passwort'];
// 内联SVG图标(钥匙图标)
const ICON_SVG = `data:image/svg+xml,${encodeURIComponent(`
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="#4CAF50">
<path d="M12.65 10C11.83 7.67 9.61 6 7 6c-3.31 0-6 2.69-6 6s2.69 6 6 6c2.61 0 4.83-1.67 5.65-4H17v4h4v-4h2v-4H12.65zM7 14c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2z"/>
</svg>
`)}`;
// ==================== 全局变量 ====================
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 = `
<h3>🔐 自动密码生成器 设置</h3>
<div class="ap-setting-row">
<label>
<input type="checkbox" id="ap-include-symbols" ${settings.includeSymbols ? 'checked' : ''}>
包含特殊符号
</label>
</div>
<div class="ap-setting-row">
<label>
密码长度:
<input type="number" id="ap-password-length" value="${settings.length}" min="8" max="64">
</label>
</div>
<div class="ap-password-preview" id="ap-preview-password">${previewPassword}</div>
<div class="ap-btn-row">
<button class="ap-btn-secondary" id="ap-regenerate">重新生成</button>
<button class="ap-btn-primary" id="ap-copy">复制密码</button>
</div>
<div class="ap-blacklist-section">
<strong>黑名单管理</strong>
<p style="font-size: 12px; color: #666; margin: 8px 0 !important;">以下网站不会显示密码生成按钮:</p>
<div class="ap-blacklist-list" id="ap-blacklist-container">
${blacklist.length === 0 ? '<div style="padding: 12px; color: #999; text-align: center;">黑名单为空</div>' :
blacklist.map(domain => `
<div class="ap-blacklist-item">
<span>${domain}</span>
<button data-domain="${domain}">移除</button>
</div>
`).join('')}
</div>
<button class="ap-btn-danger" id="ap-add-current-site" style="margin-top: 8px; width: 100%;">
将当前网站加入黑名单
</button>
</div>
<div class="ap-btn-row" style="margin-top: 20px;">
<button class="ap-btn-secondary" id="ap-close-settings">关闭</button>
</div>
`;
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 + ' 加入黑名单');
});
})();