Google Zen Color, Font & Logo

Replace Google logo with custom logo, apply grayscale to service logos, supports dark mode

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         Google Zen Color, Font & Logo
// @namespace    http://tampermonkey.net/
// @version      1.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: 2000, // Increased from 500ms to reduce CPU usage
        
        // Throttle for MutationObserver callbacks  
        MUTATION_THROTTLE: 300, // Increased from 100ms to reduce frequency
        
        // Service-specific settings
        SERVICES: {
            search: { enabled: true, fastCheck: true },
            drive: { enabled: true, fastCheck: false },
            maps: { enabled: true, fastCheck: false },
            mail: { enabled: true, fastCheck: false },
            photos: { enabled: true, fastCheck: false },
            calendar: { enabled: true, fastCheck: false },
            meet: { enabled: true, fastCheck: false },
            default: { enabled: true, fastCheck: false }
        }
    };

    // ===================== State Management =====================
    const state = {
        lastMutationTime: 0,
        intervalId: null,
        fastIntervalId: null,
        forceReplaceCount: 0,
        maxForceReplace: 5, 
        pageLoadTime: Date.now(),
        stylesInjected: false,
        currentService: null,
        progressiveInterval: 500, // Starting interval
        progressiveStep: 200, // Increment by 200ms each time
        maxProgressiveInterval: 1000, // Maximum interval
        darkModeCache: null, // Cache dark mode result
        lastDarkModeCheck: 0 // Timestamp of last check
    };

    // ===================== Utility Functions =====================
    
    // Get current Google service
    function getCurrentService() {
        const hostname = window.location.hostname;
        if (hostname.includes('drive.google.com')) return 'drive';
        if (hostname.includes('maps.google.com')) return 'maps';
        if (hostname.includes('mail.google.com')) return 'mail';
        if (hostname.includes('photos.google.com')) return 'photos';
        if (hostname.includes('calendar.google.com')) return 'calendar';
        if (hostname.includes('meet.google.com')) return 'meet';
        if (hostname.includes('google.com') && window.location.pathname.includes('/search')) return 'search';
        return 'default';
    }
    
    // Check if dark mode is enabled by analyzing actual page colors
    function isDarkMode() {
        // console.log('[Google Zen] isDarkMode() called, readyState:', document.readyState);
        
        // Use cache if available and body exists, but only if we have a reliable result
        const now = Date.now();
        if (state.darkModeCache !== null && document.body && (now - state.lastDarkModeCheck < 5000)) {
            // console.log('[Google Zen] Using cached dark mode result:', state.darkModeCache);
            return state.darkModeCache;
        }
        
        // Clear cache if document became ready but we only have media query result
        if (document.readyState !== 'loading' && document.body && state.darkModeCache !== null) {
            // console.log('[Google Zen] Document ready, clearing cache to re-check with body');
            state.darkModeCache = null;
        }
        
        // First check body background color
        const body = document.body;
        if (!body) {
            // console.log('[Google Zen] Body not found, waiting for DOM... readyState:', document.readyState);
            
            // Fallback to media query when body is not available
            const mediaQueryResult = window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches;
            // console.log('[Google Zen] Using media query result:', mediaQueryResult);
            return mediaQueryResult;
        }
        
        // Get computed style of body
        const bodyStyle = window.getComputedStyle(body);
        const bgColor = bodyStyle.backgroundColor;
        
        // Debug: log the background color
        // console.log('[Google Zen] Body background color:', bgColor);
        
        // Parse RGB values (handle both rgb() and rgba() formats)
        const match = bgColor.match(/rgba?\s*\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)/);
        if (match) {
            const r = parseInt(match[1]);
            const g = parseInt(match[2]);
            const b = parseInt(match[3]);
            
            // Calculate brightness (0-255)
            const brightness = (r * 299 + g * 587 + b * 114) / 1000;
            
            // Debug: log brightness calculation
            // console.log('[Google Zen] RGB:', r, g, b, 'Brightness:', brightness, 'Dark mode:', brightness < 128);
            
            // Consider it dark mode if brightness is below 128
            const result = brightness < 128;
            
            // Cache the result
            state.darkModeCache = result;
            state.lastDarkModeCheck = Date.now();
            
            return result;
        }
        
        // If background is transparent or not set, check parent elements
        let currentElement = body.parentElement;
        while (currentElement && currentElement !== document.documentElement) {
            const elementStyle = window.getComputedStyle(currentElement);
            const elementBgColor = elementStyle.backgroundColor;
            
            if (elementBgColor && elementBgColor !== 'rgba(0, 0, 0, 0)' && elementBgColor !== 'transparent') {
                const parentMatch = elementBgColor.match(/rgba?\s*\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)/);
                if (parentMatch) {
                    const r = parseInt(parentMatch[1]);
                    const g = parseInt(parentMatch[2]);
                    const b = parseInt(parentMatch[3]);
                    const brightness = (r * 299 + g * 587 + b * 114) / 1000;
                    const result = brightness < 128;
                    
                    // Cache the result
                    state.darkModeCache = result;
                    state.lastDarkModeCheck = Date.now();
                    
                    return result;
                }
            }
            currentElement = currentElement.parentElement;
        }
        
        // Check for specific dark theme classes or attributes
        const htmlElement = document.documentElement;
        if (htmlElement.classList.contains('dark') || 
            htmlElement.getAttribute('data-theme') === 'dark' ||
            body.classList.contains('dark-theme') ||
            body.classList.contains('night-mode')) {
            return true;
        }
        
        // Fallback to media query
        const mediaResult = window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches;
        // console.log('[Google Zen] Final fallback to media query:', mediaResult);
        
        // Cache the result
        state.darkModeCache = mediaResult;
        state.lastDarkModeCheck = Date.now();
        
        return mediaResult;
    }

    // Get filter for dark mode (invert brightness)
    function getDarkModeFilter() {
        // console.log('[Google Zen] getDarkModeFilter() called');
        const darkMode = isDarkMode();
        // console.log('[Google Zen] Dark mode result:', darkMode);
        return darkMode ? 'invert(.9)' : 'none';
    }

    // Force update all existing logo filters
    function updateExistingLogoFilters() {
        // console.log('[Google Zen] Updating existing logo filters');
        const filter = getDarkModeFilter();
        
        // Update all custom logo images
        document.querySelectorAll('img[src^=""]').forEach(img => {
            img.style.setProperty('filter', filter, 'important');
        });
        
        // Update logo containers
        document.querySelectorAll('.custom-logo-cropped img').forEach(img => {
            img.style.setProperty('filter', filter, 'important');
        });
        
        // Update any other logos that might have the filter applied
        document.querySelectorAll('img[style*="filter"]').forEach(img => {
            if (img.src.startsWith('')) {
                img.style.setProperty('filter', filter, 'important');
            }
        });
    }

    // 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 / 141)) + 'px';
        img.style.height = Math.round(150 * (size / 141)) + '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 = `
            /* Prefer Roman Font */
            span:not(.google-symbols):not(.material-icons-extended):not(:has(.google-symbols)),
            div:not(.google-symbols):not(.material-icons-extended):not(:has(.google-symbols)) {
                font-family: 'Noto Serif JP', serif !important;
            }
            
            [data-viewfamily="EVENT_EDIT"] [role="toolbar"] [role="button"] span > span > span,
            [data-viewfamily="EVENT_EDIT"] [role="tabpanel"] [jsshadow] > div > [aria-hidden="true"] .google-material-icons span {
                font-family: "Material Icons Extended" !important;
            }

            a:link:not(.google-symbols) > h3:not(.google-symbols) {
                font-family: 'Noto Serif JP', serif !important;
                color: ${isDarkMode() ? '#94a6bf' : '#3e5778' };
            }
            
            a:visited:not(.google-symbols) > h3:not(.google-symbols) {
                font-family: 'Noto Serif JP', serif !important;
                color: ${isDarkMode() ? '#c4b1d6' : '#6e5983'};
            }
            
            /* 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: .544;
                background-size: contain;
                filter: ${getDarkModeFilter()} !important;
                margin-top: 47px;
                margin-bottom: 47px;
                transition: opacity .7s ease-in-out !important;
            }
            
            div:has(div > [data-xid="aim-zero-state"]):not(:has([data-xid="aim-mars-turn-root"])) {
                margin-left: -38px;
            }
            
            body > header > header > div > a > div > img {
                visibility: hidden;
                opacity: 0;
                transition: opacity .7s ease-in-out !important;
            }
            
            body:has([data-xid="aim-mars-turn-root"] [data-processed="true"]) header > header > div > a > div > img {
                opacity: 1 !important;
                visibility: visible !important;
            }
            
            div:has(div > div > #aim-lhs-panel-threads-view-container[style="display: none;"]) > div:has(button),
            div:has(div > div > #aim-lhs-panel-threads-view-container[style="display: none; overflow: auto;"]) > div:has(button) {
                background: transparent;
            }
            
            @media(max-width:1419px) {
                body:has([data-xid="aim-mars-turn-root"] [data-processed="true"]) header > header > div > a > div > img {
                    zoom: 0.65;
                    margin-top: 10px;
                }
            }
            
            /* 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: .21;
                margin-left: -30px;
            }
            
            /* Common Bento menu iframe grayscale */
            iframe[src*="ogs.google.com"] {
                filter: grayscale(100%)contrast(170%) !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(170%) !important;
                width: 40px !important;
                object-fit: cover;
                object-position: left;
            }
            
            /* Google Image Search specific */
            body > div > div > div > img[data-defe="1"] {
                visibility: hidden;
            }
            
            body:not(:has([data-doodle="1"])) > div:not(#main) > div:has(div > img[data-defe="1"]) > div {
                background-image: url(${CONFIG.LOGO_BASE64}) !important;
                filter: ${getDarkModeFilter()} !important;
                background-size: contain;
                width: 300px;
            }
            
            body:not(:has([data-doodle="1"])) > div:not(#main) > div:has(div > img[data-defe="1"]) > div > span {
                filter: ${getDarkModeFilter()} !important;
            }
            
            /* 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(170%) !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(170%) !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() {
        const shouldLog = state.forceReplaceCount > 0 || state.progressiveInterval <= 200;
        
        if (state.forceReplaceCount > 0) {
            state.forceReplaceCount--;
            if (shouldLog) {
                console.log(`[Google Zen] Force replace mode: ${state.forceReplaceCount} remaining (interval: ${state.progressiveInterval}ms)`);
            }
        }
        
        const mainResult = replaceMainPageLogo();
        const searchResult = replaceSearchPageLogo();
        const aiResult = replaceAIModeLogo();
        const mapsResult = replaceGoogleMapsLogo();
        
        if (shouldLog && (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);
        }
    }
    
    // Set up Google Fonts
    function loadGoogleFonts() {
       const head = document.head || document.getElementsByTagName('head')[0];
    
       const preconnect1 = document.createElement('link');
       preconnect1.rel = 'preconnect';
       preconnect1.href = 'https://fonts.googleapis.com';
       head.appendChild(preconnect1);
    
       const preconnect2 = document.createElement('link');
       preconnect2.rel = 'preconnect';
       preconnect2.href = 'https://fonts.gstatic.com';
       preconnect2.setAttribute('crossorigin', '');
       head.appendChild(preconnect2);
    
       const stylesheet = document.createElement('link');
       stylesheet.rel = 'stylesheet';
       stylesheet.href = 'https://fonts.googleapis.com/css2?family=Noto+Serif+JP:wght@400;700&display=swap';
       head.appendChild(stylesheet);
   }


    
    // ===================== Execution Control =====================
    
    // Initialize logo replacer with DOM monitoring
    function initLogoReplacer() {
        // Inject styles
        injectServiceStyles();
        
        // Determine current service
        state.currentService = getCurrentService();
        const serviceConfig = CONFIG.SERVICES[state.currentService] || CONFIG.SERVICES.default;
        
        // Reset counters
        state.forceReplaceCount = state.maxForceReplace;
        state.progressiveInterval = 500; // Reset to initial interval
        
        // Initial execution
        replaceAllLogos();
        
        // Set up Google Fonts
        loadGoogleFonts();
        
        // Progressive interval implementation
        function runProgressiveCheck() {
            replaceAllLogos();
            
            // Increase interval progressively
            if (state.progressiveInterval < state.maxProgressiveInterval && state.forceReplaceCount > 0) {
                state.progressiveInterval = Math.min(state.progressiveInterval + state.progressiveStep, state.maxProgressiveInterval);
                
                // Schedule next check with new interval
                if (state.fastIntervalId) {
                    clearTimeout(state.fastIntervalId);
                }
                state.fastIntervalId = setTimeout(runProgressiveCheck, state.progressiveInterval);
            } else if (state.forceReplaceCount <= 0) {
                // Stop progressive checks when force replace count reaches 0
                if (state.fastIntervalId) {
                    clearTimeout(state.fastIntervalId);
                    state.fastIntervalId = null;
                    console.log('[Google Zen] Progressive checking completed');
                }
            }
        }
        
        // Start progressive checking only for services that need it
        if (serviceConfig.fastCheck) {
            state.fastIntervalId = setTimeout(runProgressiveCheck, state.progressiveInterval);
        }
        
        // 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;
                state.progressiveInterval = 500; // Reset to initial interval
                console.log('[Google Zen] Navigation detected, resetting progressive check');
                
                // Clear any existing interval
                if (state.fastIntervalId) {
                    clearTimeout(state.fastIntervalId);
                    state.fastIntervalId = null;
                }
                
                // Restart progressive checking for services that need it
                const serviceConfig = CONFIG.SERVICES[state.currentService] || CONFIG.SERVICES.default;
                if (serviceConfig.fastCheck) {
                    function runProgressiveCheck() {
                        replaceAllLogos();
                        
                        if (state.progressiveInterval < state.maxProgressiveInterval && state.forceReplaceCount > 0) {
                            state.progressiveInterval = Math.min(state.progressiveInterval + state.progressiveStep, state.maxProgressiveInterval);
                            state.fastIntervalId = setTimeout(runProgressiveCheck, state.progressiveInterval);
                        }
                    }
                    state.fastIntervalId = setTimeout(runProgressiveCheck, state.progressiveInterval);
                }
            }
            
            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();
    
    // Force dark mode recheck when document becomes ready
    function recheckDarkMode() {
        if (document.readyState !== 'loading' && document.body) {
            // console.log('[Google Zen] Document ready, forcing dark mode recheck');
            const oldCache = state.darkModeCache;
            state.darkModeCache = null;
            state.lastDarkModeCheck = 0;
            
            // Get new dark mode result
            const newDarkMode = isDarkMode();
            // console.log('[Google Zen] Dark mode changed from cached to actual:', oldCache, '->', newDarkMode);
            
            // Force re-injection of styles if dark mode changed
            if (oldCache !== newDarkMode) {
                console.log('[Google Zen] Dark mode changed, re-injecting styles');
                state.stylesInjected = false;
                injectServiceStyles();
                
                // Force update all existing logos and filters
                updateExistingLogoFilters();
                replaceAllLogos();
            }
        }
    }
    
    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', recheckDarkMode);
        document.addEventListener('readystatechange', recheckDarkMode);
    } else {
        recheckDarkMode();
    }
    
    // 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');

})();