您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Blurs inactive WhatsApp chats/messages, unblurs on hover, enables messaging without saving numbers, and toggles blur for unread messages.
当前为
// ==UserScript== // @name WhatsApp Web Blur & Quick Message // @namespace http://tampermonkey.net/ // @version 1.2.11 // @description Blurs inactive WhatsApp chats/messages, unblurs on hover, enables messaging without saving numbers, and toggles blur for unread messages. // @author Noushad Bhuiyan // @match https://web.whatsapp.com/* // @icon https://www.google.com/s2/favicons?sz=64&domain=whatsapp.com // @grant GM_setValue // @grant GM_getValue // @license MIT // ==/UserScript== (() => { // Load blur state for unread messages (default: true, blur enabled) let blurUnread = GM_getValue('blurUnread', true); // Data URLs for toggle icons const toggleOffIcon = ''; const toggleOnIcon = ''; const applyBlur = el => { // Reset existing blur state if (el.hasAttribute('is-blurred')) { el.style.filter = 'none'; el.style.transition = 'none'; el.removeAttribute('is-blurred'); el.removeEventListener('mouseenter', () => {}); el.removeEventListener('mouseleave', () => {}); } // Apply blur if conditions are met if (!el.hasAttribute('aria-selected')) { const isUnread = el.querySelector('[aria-label*="unread"]'); if (!isUnread || (isUnread && blurUnread)) { el.style.filter = 'blur(8px)'; el.style.transition = 'filter 0.3s ease-in-out'; el.setAttribute('is-blurred', 'true'); el.addEventListener('mouseenter', () => el.style.filter = 'none'); el.addEventListener('mouseleave', () => el.style.filter = 'blur(8px)'); } } }; const clearBlurFromActiveChat = () => document.querySelectorAll('[aria-label="Chat list"] [aria-selected="true"]').forEach(el => { const listItem = el.closest('[role="listitem"]'); if (listItem?.hasAttribute('is-blurred')) { listItem.style.filter = 'none'; listItem.style.transition = 'none'; listItem.removeAttribute('is-blurred'); listItem.removeEventListener('mouseenter', () => {}); listItem.removeEventListener('mouseleave', () => {}); } }); const blurElements = () => { document.querySelectorAll('[aria-label="Chat list"] [role="listitem"], [role="application"] [data-testid="conversation-panel-messages"]').forEach(applyBlur); clearBlurFromActiveChat(); }; const addButtons = () => { const h1 = document.querySelector('h1'); if (h1 && !document.getElementById('quick-message-btn')) { // Quick Message Button const msgBtn = Object.assign(document.createElement('button'), { id: 'quick-message-btn', title: 'Send message to a number without saving', style: 'margin-left:10px;cursor:pointer;background:none;border:none;vertical-align:middle' }); msgBtn.innerHTML = '<img src="" alt="Message Icon" style="width:20px;height:20px">'; msgBtn.onclick = showModal; h1.parentElement.parentElement.insertBefore(msgBtn, h1.parentElement.nextSibling); // Toggle Blur Button const toggleBtn = Object.assign(document.createElement('button'), { id: 'toggle-blur-btn', title: blurUnread ? 'Disable blur for unread messages' : 'Enable blur for unread messages', style: 'margin-left:10px;cursor:pointer;background:none;border:none;vertical-align:middle' }); toggleBtn.innerHTML = `<img src="${blurUnread ? toggleOnIcon : toggleOffIcon}" alt="Toggle Blur Icon" style="width:20px;height:20px">`; toggleBtn.onclick = () => { blurUnread = !blurUnread; GM_setValue('blurUnread', blurUnread); toggleBtn.title = blurUnread ? 'Disable blur for unread messages' : 'Enable blur for unread messages'; toggleBtn.innerHTML = `<img src="${blurUnread ? toggleOnIcon : toggleOffIcon}" alt="Toggle Blur Icon" style="width:20px;height:20px">`; blurElements(); }; h1.parentElement.parentElement.insertBefore(toggleBtn, msgBtn.nextSibling); } }; const showModal = () => { document.getElementById('quick-message-modal')?.remove(); const isDark = window.matchMedia('(prefers-color-scheme: dark)').matches; const modal = Object.assign(document.createElement('div'), { id: 'quick-message-modal', style: 'position:fixed;top:0;left:0;width:100%;height:100%;background:' + (isDark ? 'rgba(0,0,0,0.7)' : 'rgba(0,0,0,0.5)') + ';backdrop-filter:blur(5px);display:flex;align-items:center;justify-content:center;z-index:1000' }); const content = Object.assign(document.createElement('div'), { style: 'background:' + (isDark ? '#1a1a1a' : '#fff') + ';color:' + (isDark ? '#fff' : '#000') + ';padding:20px;border-radius:8px;max-width:400px;width:100%;box-shadow:0 4px 8px rgba(0,0,0,0.2)' }); content.onclick = e => e.stopPropagation(); const input = Object.assign(document.createElement('input'), { type: 'text', placeholder: 'Enter phone number', style: 'width:-webkit-fill-available;padding:10px;margin:10px 0;border-radius:4px;background:rgb(51,51,51);color:rgb(255,255,255)', autofocus: true }); const note = Object.assign(document.createElement('div'), { style: 'font-size:12px;color:' + (isDark ? '#bbb' : '#555') + ';margin-bottom:10px', innerHTML: 'Please include the country code (e.g., +1234567890)' }); const btnRow = Object.assign(document.createElement('div'), { style: 'display:flex;gap:10px' }); const checkBtn = Object.assign(document.createElement('button'), { textContent: 'Check Number', style: 'background:#25D366;color:#fff;padding:10px;border:none;border-radius:4px;cursor:pointer;flex:1' }); checkBtn.onclick = () => { const num = input.value.trim().replace(/[\s-+]/g, ''); if (num) { window.location.href = `https://web.whatsapp.com/send/?phone=${num}`; modal.remove(); } }; const closeBtn = Object.assign(document.createElement('button'), { textContent: 'Close', style: 'background:' + (isDark ? '#444' : '#ccc') + ';color:' + (isDark ? '#fff' : '#000') + ';padding:10px;border:none;border-radius:4px;cursor:pointer;flex:1' }); closeBtn.onclick = () => modal.remove(); modal.onclick = e => e.target === modal && modal.remove(); btnRow.append(checkBtn, closeBtn); content.append(input, note, btnRow); modal.appendChild(content); document.body.appendChild(modal); }; blurElements(); addButtons(); const observer = new MutationObserver(mutations => mutations.forEach(m => m.addedNodes.length && (blurElements(), addButtons()))); const target = document.querySelector('body'); target && observer.observe(target, { childList: true, subtree: true }); window.addEventListener('unload', () => observer.disconnect()); })();