您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Replace Google logo with custom logo, apply grayscale to service logos, supports dark mode
// ==UserScript== // @name Google Zen Color & Logo // @namespace http://tampermonkey.net/ // @version 0.8 // @description Replace Google logo with custom logo, apply grayscale to service logos, supports dark mode // @author djshigel // @license MIT // @match https://www.google.com/* // @match https://www.google.com/webhp* // @match https://www.google.com/search* // @match https://google.com/* // @match https://myaccount.google.com/* // @match https://maps.google.com/* // @match https://news.google.com/* // @match https://mail.google.com/* // @match https://meet.google.com/* // @match https://chat.google.com/* // @match https://contacts.google.com/* // @match https://drive.google.com/* // @match https://calendar.google.com/* // @match https://play.google.com/* // @match https://translate.google.com/* // @match https://photos.google.com/* // @match https://www.google.com/shopping* // @grant none // @run-at document-start // ==/UserScript== (function() { 'use strict'; // ===================== Configuration ===================== const CONFIG = { // Control replacement when Doodle is displayed REPLACE_ON_DOODLE: false, // true: always replace, false: keep Doodle // Logo Base64 data (500x150px) - paste actual data here LOGO_BASE64: '', // Check interval for logo replacement (milliseconds) CHECK_INTERVAL: 500, // Throttle for MutationObserver callbacks MUTATION_THROTTLE: 100 }; // ===================== State Management ===================== const state = { lastMutationTime: 0, intervalId: null, fastIntervalId: null, forceReplaceCount: 0, maxForceReplace: 15, pageLoadTime: Date.now(), stylesInjected: false }; // ===================== Utility Functions ===================== // Check if dark mode is enabled function isDarkMode() { return window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches; } // Get filter for dark mode (invert brightness) function getDarkModeFilter() { return isDarkMode() ? 'invert(1)' : 'none'; } // Check if Doodle is present function hasDoodle() { const pictureElement = document.querySelector('picture'); if (pictureElement) { const img = pictureElement.querySelector('img#hplogo'); if (img && img.src && !img.src.startsWith('data:')) return true; } const logoLink = document.querySelector('a#logo > img'); if (logoLink && logoLink.src && !logoLink.src.includes('googlelogo') && !logoLink.src.startsWith('data:')) { return true; } return false; } // Normalize tagName for comparison (always uppercase) function normalizeTagName(element) { return element.tagName ? element.tagName.toUpperCase() : ''; } // Create cropped logo container for icon use function createCroppedLogoContainer(size) { const container = document.createElement('div'); container.className = 'custom-logo-cropped'; container.style.width = size + 'px'; container.style.height = size + 'px'; container.style.overflow = 'hidden'; container.style.position = 'relative'; container.style.display = 'inline-block'; container.style.verticalAlign = 'middle'; const img = document.createElement('img'); img.src = CONFIG.LOGO_BASE64; img.style.position = 'absolute'; img.style.left = '0'; img.style.top = '0'; img.style.width = Math.round(500 * (size / 120)) + 'px'; img.style.height = Math.round(150 * (size / 120)) + 'px'; img.style.filter = getDarkModeFilter(); img.style.imageRendering = 'auto'; container.appendChild(img); return container; } // ===================== Style Injection ===================== // Inject CSS styles for service logos function injectServiceStyles() { if (state.stylesInjected) return; const styleElement = document.createElement('style'); styleElement.id = 'google-zen-styles'; styleElement.textContent = ` /* Google logo when scrolling */ div.minidiv #hplogo { width: 120px !important; margin-top: -6px; } div.minidiv #logo { top: -6px; right: 7px; } div.minidiv > div:not(:has(div)) { background: ${isDarkMode() ? 'rgb(31, 31, 31) !important' : ''}; } div.minidiv >[role="navigation"] { background: ${isDarkMode() ? '#202124 !important' : ''}; } /* AI mode logo replacement*/ [data-xid="aim-zero-state"] div:has(div > [role="heading"]) > div > div:not(:has([role="heading"])) { display: none; } [data-xid="aim-zero-state"] div:has(div > [role="heading"]) > div > div:has([role="heading"])::before { content: url(${CONFIG.LOGO_BASE64}) !important; zoom: 0.544; background-size: contain; filter: ${getDarkModeFilter()} !important; margin-bottom: 47px; transition: opacity 1s ease-in-out !important; } body > header > header > div > a > div > img { opacity: 0; transition: opacity 1s ease-in-out !important; } body:has([data-xid="aim-mars-turn-root"] [data-processed="true"]) header > header > div > a > div > img { opacity: 1 !important; } body > header > header > div > a > div > img:hover { transition: opacity .2s ease-in-out !important; opacity: 1 !important; } /* Mobile */ #gb-main img#hplogo { width: 163px !important; } header > div > a[aria-label="Google"] > svg { display: none; } header > div > a[aria-label="Google"]::before { content: url(${CONFIG.LOGO_BASE64}) !important; filter: ${getDarkModeFilter()} !important; zoom: 0.2; } /* Common Bento menu iframe grayscale */ iframe[src*="ogs.google.com"] { filter: grayscale(100%)contrast(150%) !important; } /* Hide Chrome recommendations */ body > c-wiz:has(a[href^="https://www.google.com/url?q=https://www.google.com/chrome/"]) { display: none !important; } /* Footer opacity control on homepage */ div[role="contentinfo"], #fbar { opacity: 0 !important; transition: opacity .2s ease-in-out !important; } div[role="contentinfo"]:hover, #fbar:hover { opacity: 1 !important; } /* Pattern 1 - Services with span[role="presentation"] */ #gb > div > div > div > div > a:has(span[role="presentation"]) > span:not([role="presentation"]) { font-family: serif !important; font-weight: bold !important; } #gb > div > div > div > div > a > span[role="presentation"] { background-image: url(${CONFIG.LOGO_BASE64}) !important; background-size: contain !important; background-repeat: no-repeat !important; background-position: center !important; filter: ${getDarkModeFilter()} !important; } #gb > div > div > div > div > a > span[role="presentation"]::before { content: "" !important; background-image: none !important; } /* Pattern 2 - Services with img[role="presentation"] */ #gb > div > div > div > div > a:has(img[role="presentation"]) > span:not([role="presentation"]), #gb > div > div > div > div > span:has(img[role="presentation"]) > span:not([role="presentation"]) { font-family: serif !important; font-weight: bold !important; } #gb > div > div > div > div > a > img[role="presentation"], #gb > div > div > div > div > span > img[role="presentation"] { filter: grayscale(100%)contrast(150%) !important; width: 40px !important;; object-fit: cover; object-position: left; } /* Google Maps specific */ #settings > div > div > ul > div > div > img { content: url(${CONFIG.LOGO_BASE64}) !important; filter: ${getDarkModeFilter()} !important; } /* Google Play specific */ body > c-wiz > header > nav > a { filter: grayscale(100%)contrast(150%) !important; } body > c-wiz > header > nav > a > span:not([role="presentation"]) { font-family: serif !important; font-weight: bold !important; } /* Google Photos specific */ body > div > div > c-wiz > c-wiz > div > div > div > div > div > div > a > span:not([title="Google"]) { font-family: serif !important; font-weight: bold !important; } body > div > div > c-wiz > c-wiz > div > div > div > div > div > div > a > span[title="Google"] { background-image: url(${CONFIG.LOGO_BASE64}) !important; background-size: contain !important; background-repeat: no-repeat !important; background-position: center !important; filter: ${getDarkModeFilter()} !important; } body > div > div > c-wiz > c-wiz > div > div > div > div > div > div > a > span[title="Google"]::before { content: "" !important; background-image: none !important; } /* Google Shopping specific */ body > div > div > div > img { filter: grayscale(100%)contrast(150%) !important; } `; if (document.head) { document.head.appendChild(styleElement); state.stylesInjected = true; console.log('[Google Zen] Styles injected'); } else { // Wait for head element setTimeout(injectServiceStyles, 100); } } // ===================== Logo Replacement Functions ===================== // Replace main page logo function replaceMainPageLogo() { // Multiple selectors for main page logo - svg[aria-label="Google"]を優先 const selectors = [ 'svg[aria-label="Google"]', 'img#hplogo', 'picture > img#hplogo', 'picture > svg#hplogo', 'div[data-hveid] img[alt="Google"]', 'img[alt="Google"][height="92"]' ]; let hplogo = null; for (const selector of selectors) { hplogo = document.querySelector(selector); if (hplogo) break; } if (!hplogo) return false; const tagName = normalizeTagName(hplogo); // Check if already replaced if (tagName === 'IMG' && hplogo.src === CONFIG.LOGO_BASE64) { // Ensure parent styles const pictureElement = hplogo.closest('picture'); if (pictureElement) { const parentDiv = pictureElement.parentElement?.parentElement; if (parentDiv && normalizeTagName(parentDiv) === 'DIV') { parentDiv.style.height = 'auto'; } } return false; } if (!CONFIG.REPLACE_ON_DOODLE && hasDoodle()) { console.log('[Google Zen] Doodle detected, skipping main logo'); return false; } if (tagName === 'SVG' || tagName === 'IMG') { const img = document.createElement('img'); img.id = 'hplogo'; img.src = CONFIG.LOGO_BASE64; img.style.width = '272px'; img.style.height = 'auto'; img.style.setProperty('filter', getDarkModeFilter(), 'important'); img.dataset.customLogo = 'true'; img.alt = 'Google'; hplogo.parentElement.replaceChild(img, hplogo); // Set parent styles const pictureElement = img.closest('picture'); if (pictureElement) { const parentDiv = pictureElement.parentElement?.parentElement; if (parentDiv && normalizeTagName(parentDiv) === 'DIV') { parentDiv.style.height = 'auto'; } } console.log('[Google Zen] Main page logo replaced'); return true; } return false; } // Replace search results page header logo function replaceSearchPageLogo() { const selectors = [ 'a#logo', ]; let logoLink = null; for (const selector of selectors) { logoLink = document.querySelector(selector); if (logoLink && logoLink.id === 'logo') break; if (logoLink) { const hasLogoContent = logoLink.querySelector('svg, img') || logoLink.children.length > 0; if (hasLogoContent) break; } } if (!logoLink) return false; const existingCustom = logoLink.querySelector('img[data-custom-logo="true"]'); const needsReplacement = !existingCustom || existingCustom.src !== CONFIG.LOGO_BASE64 || existingCustom.style.width !== '120px' || state.forceReplaceCount > 0; if (needsReplacement) { const targetElement = logoLink.querySelector('svg, img:not([data-custom-logo])') || logoLink.children[0]; if (!CONFIG.REPLACE_ON_DOODLE && targetElement && normalizeTagName(targetElement) === 'IMG' && hasDoodle()) { console.log('[Google Zen] Doodle detected in search page'); return false; } const img = document.createElement('img'); img.src = CONFIG.LOGO_BASE64; img.style.width = '120px'; img.style.height = '36px'; img.style.objectFit = 'contain'; img.style.filter = getDarkModeFilter(); img.style.display = 'block'; img.dataset.customLogo = 'true'; logoLink.innerHTML = ''; logoLink.appendChild(img); console.log('[Google Zen] Search page logo replaced'); return true; } return false; } // Replace AI mode (udm=50) single character logo - 簡易化されたセレクタ function replaceAIModeLogo() { if (!window.location.search.includes('udm=50')) { return false; } const selectors = [ 'body > header > header > div > a:not([role="button"])', 'body > header > header > a:has(svg[focusable="false"])' ]; let aiLogoLink = null; for (const selector of selectors) { const elem = document.querySelector(selector); if (elem) { // 簡易化された条件:子要素があるか確認するだけ if (elem.querySelector('svg, img, div > img') || elem.children.length > 0) { aiLogoLink = elem; break; } } } if (!aiLogoLink) return false; // Apply styles to parent div const parentDiv = aiLogoLink.parentElement?.parentElement; if (parentDiv && normalizeTagName(parentDiv) === 'DIV') { parentDiv.style.left = '21px'; parentDiv.style.top = '33px'; //parentDiv.style.position = 'absolute'; } // Apply styles to link aiLogoLink.style.display = 'inline-block'; aiLogoLink.style.width = '34px'; aiLogoLink.style.height = '34px'; aiLogoLink.style.verticalAlign = 'middle'; aiLogoLink.style.position = 'relative'; const existingCustom = aiLogoLink.querySelector('.custom-logo-cropped'); const existingImg = existingCustom?.querySelector('img'); const needsReplacement = !existingCustom || !existingImg || existingImg.src !== CONFIG.LOGO_BASE64 || aiLogoLink.style.width !== '34px' || state.forceReplaceCount > 0; if (needsReplacement) { const targetElement = aiLogoLink.querySelector('svg, img:not(.custom-logo-cropped img), div > img') || aiLogoLink.children[0]; if (!CONFIG.REPLACE_ON_DOODLE && targetElement && normalizeTagName(targetElement) === 'IMG' && targetElement.width > 34) { console.log('[Google Zen] Doodle detected in AI mode'); return false; } const croppedLogo = createCroppedLogoContainer(34); croppedLogo.style.position = 'relative'; croppedLogo.style.verticalAlign = 'middle'; aiLogoLink.innerHTML = ''; aiLogoLink.appendChild(croppedLogo); aiLogoLink.style.display = 'inline-block'; aiLogoLink.style.width = '34px'; aiLogoLink.style.height = '34px'; aiLogoLink.style.verticalAlign = 'middle'; aiLogoLink.style.position = 'relative'; console.log('[Google Zen] AI mode logo replaced with cropped image'); return true; } else if (existingCustom) { aiLogoLink.style.display = 'inline-block'; aiLogoLink.style.width = '34px'; aiLogoLink.style.height = '34px'; aiLogoLink.style.verticalAlign = 'middle'; aiLogoLink.style.position = 'relative'; } return false; } // Replace Google Maps logo function replaceGoogleMapsLogo() { if (!window.location.hostname.includes('maps.google.com')) return false; const logoImg = document.querySelector('#settings > div > div > ul > div > div > img'); if (logoImg && logoImg.src !== CONFIG.LOGO_BASE64) { logoImg.src = CONFIG.LOGO_BASE64; logoImg.style.filter = getDarkModeFilter(); console.log('[Google Zen] Google Maps logo replaced'); return true; } return false; } // Replace all logos function replaceAllLogos() { if (state.forceReplaceCount > 0) { state.forceReplaceCount--; console.log(`[Google Zen] Force replace mode: ${state.forceReplaceCount} remaining`); } const mainResult = replaceMainPageLogo(); const searchResult = replaceSearchPageLogo(); const aiResult = replaceAIModeLogo(); const mapsResult = replaceGoogleMapsLogo(); if (state.forceReplaceCount > 0 || (searchResult || aiResult || mapsResult)) { console.log(`[Google Zen] Status - Main: ${mainResult}, Search: ${searchResult}, AI: ${aiResult}, Maps: ${mapsResult}`); } if (!searchResult && document.querySelector('a#logo, div.logo')) { setTimeout(() => { replaceSearchPageLogo(); }, 100); } if (!aiResult && window.location.search.includes('udm=50')) { setTimeout(() => { replaceAIModeLogo(); }, 100); } } // ===================== Execution Control ===================== // Initialize logo replacer with DOM monitoring function initLogoReplacer() { // Inject styles injectServiceStyles(); // Reset force replacement counter state.forceReplaceCount = state.maxForceReplace; // Initial execution replaceAllLogos(); // Set up fast checking for first 3 seconds if (state.fastIntervalId) { clearInterval(state.fastIntervalId); } state.fastIntervalId = setInterval(() => { replaceAllLogos(); }, 200); setTimeout(() => { if (state.fastIntervalId) { clearInterval(state.fastIntervalId); state.fastIntervalId = null; console.log('[Google Zen] Fast checking stopped'); } }, 3000); // Set up regular periodic checking if (state.intervalId) { clearInterval(state.intervalId); } state.intervalId = setInterval(replaceAllLogos, CONFIG.CHECK_INTERVAL); // Monitor DOM changes with MutationObserver const observer = new MutationObserver((mutations) => { const now = Date.now(); if (now - state.lastMutationTime < CONFIG.MUTATION_THROTTLE) { return; } state.lastMutationTime = now; let shouldReplace = false; let navigationDetected = false; for (const mutation of mutations) { if (mutation.type === 'childList') { const addedNodes = Array.from(mutation.addedNodes); for (const node of addedNodes) { if (node.nodeType === 1) { const hasLogoElements = (node.id === 'hplogo' || node.id === 'logo') || (node.querySelector && ( node.querySelector('#hplogo') || node.querySelector('#logo') || node.querySelector('a#logo') || node.querySelector('picture') || node.querySelector('svg[aria-label="Google"]') )); const nodeTag = normalizeTagName(node); if (nodeTag === 'HEADER' || nodeTag === 'MAIN' || node.classList?.contains('logo')) { navigationDetected = true; } if (hasLogoElements) { shouldReplace = true; break; } } } } if (shouldReplace) break; } if (navigationDetected) { state.forceReplaceCount = state.maxForceReplace; console.log('[Google Zen] Navigation detected, resetting force replace'); if (state.fastIntervalId) { clearInterval(state.fastIntervalId); } state.fastIntervalId = setInterval(() => { replaceAllLogos(); }, 200); setTimeout(() => { if (state.fastIntervalId) { clearInterval(state.fastIntervalId); state.fastIntervalId = null; } }, 3000); } if (shouldReplace || navigationDetected) { setTimeout(replaceAllLogos, 50); } }); if (document.body) { observer.observe(document.body, { childList: true, subtree: true }); } else { const bodyObserver = new MutationObserver(() => { if (document.body) { observer.observe(document.body, { childList: true, subtree: true }); bodyObserver.disconnect(); replaceAllLogos(); } }); bodyObserver.observe(document.documentElement, { childList: true, subtree: true }); } } // Monitor dark mode changes if (window.matchMedia) { window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', (e) => { console.log('[Google Zen] Dark mode changed:', e.matches); state.forceReplaceCount = state.maxForceReplace; if (state.fastIntervalId) { clearInterval(state.fastIntervalId); } state.fastIntervalId = setInterval(() => { replaceAllLogos(); }, 200); setTimeout(() => { if (state.fastIntervalId) { clearInterval(state.fastIntervalId); state.fastIntervalId = null; } }, 3000); replaceAllLogos(); }); } // Execute when DOM is ready if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', () => { state.forceReplaceCount = state.maxForceReplace; replaceAllLogos(); }); } else { state.forceReplaceCount = state.maxForceReplace; replaceAllLogos(); } // Also monitor readyState changes document.addEventListener('readystatechange', () => { if (document.readyState === 'interactive' || document.readyState === 'complete') { console.log(`[Google Zen] Document ready state: ${document.readyState}`); setTimeout(replaceAllLogos, 100); } }); // Clean up on page unload window.addEventListener('unload', () => { if (state.intervalId) { clearInterval(state.intervalId); } if (state.fastIntervalId) { clearInterval(state.fastIntervalId); } }); // Handle browser back/forward navigation window.addEventListener('popstate', () => { console.log('[Google Zen] Browser navigation detected'); state.forceReplaceCount = state.maxForceReplace; if (state.fastIntervalId) { clearInterval(state.fastIntervalId); } state.fastIntervalId = setInterval(() => { replaceAllLogos(); }, 200); setTimeout(() => { if (state.fastIntervalId) { clearInterval(state.fastIntervalId); state.fastIntervalId = null; } }, 3000); setTimeout(replaceAllLogos, 100); }); // Initialize initLogoReplacer(); // Monitor URL changes (for SPA navigation) let lastUrl = location.href; new MutationObserver(() => { const url = location.href; if (url !== lastUrl) { lastUrl = url; console.log('[Google Zen] URL changed, resetting force replace'); state.forceReplaceCount = state.maxForceReplace; if (state.fastIntervalId) { clearInterval(state.fastIntervalId); } state.fastIntervalId = setInterval(() => { replaceAllLogos(); }, 200); setTimeout(() => { if (state.fastIntervalId) { clearInterval(state.fastIntervalId); state.fastIntervalId = null; } }, 3000); setTimeout(replaceAllLogos, 100); } }).observe(document, {subtree: true, childList: true}); console.log('[Google Zen] Script initialized v0.3.1'); })();