// ==UserScript==
// @name Google Zen Color, Font & Logo
// @namespace http://tampermonkey.net/
// @version 1.1
// @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 / 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 */
* {
font-family: 'Noto Serif JP', serif !important;
}
a > h3 {
font-family: 'Noto Serif JP', serif !important;
font-weight: bold !important;
filter: grayscale(50%);
}
/* 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) {
div:has(div > div.save-components-async) > div {
grid-template-columns: minmax(130px,140px) repeat(12,36px) minmax(0,1fr) !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: .21;
margin-left: -30px;
}
/* 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 Image Search specific */
body > div > div > div > img[data-defe="1"] {
visibility: hidden;
}
body > div > div:has(div > img[data-defe="1"]) > div {
background-image: url(${CONFIG.LOGO_BASE64}) !important;
filter: ${getDarkModeFilter()} !important;
background-size: contain;
width: 300px;
}
body > div > 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(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);
}
}
// 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();
// Reset force replacement counter
state.forceReplaceCount = state.maxForceReplace;
// Initial execution
replaceAllLogos();
// Set up Google Fonts
loadGoogleFonts();
// 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');
})();