// ==UserScript== // @name Gemini NanoBanana Watermark Remover // @name:zh-CN Gemini NanoBanana 图片水印移除 // @namespace https://github.com/journey-ad // @version 0.1.5 // @description Automatically removes watermarks from Gemini AI generated images // @description:zh-CN 自动移除 Gemini AI 生成图像中的水印 // @icon https://www.google.com/s2/favicons?domain=gemini.google.com // @author journey-ad // @license MIT // @match https://gemini.google.com/* // @grant GM_xmlhttpRequest // @run-at document-end // @downloadURL https://update.greasyfork.icu/scripts/559574/Gemini%20NanoBanana%20Watermark%20Remover.user.js // @updateURL https://update.greasyfork.icu/scripts/559574/Gemini%20NanoBanana%20Watermark%20Remover.meta.js // ==/UserScript== (() => { // src/core/alphaMap.js function calculateAlphaMap(bgCaptureImageData) { const { width, height, data } = bgCaptureImageData; const alphaMap = new Float32Array(width * height); for (let i = 0; i < alphaMap.length; i++) { const idx = i * 4; const r = data[idx]; const g = data[idx + 1]; const b = data[idx + 2]; const maxChannel = Math.max(r, g, b); alphaMap[i] = maxChannel / 255; } return alphaMap; } // src/core/blendModes.js var ALPHA_THRESHOLD = 2e-3; var MAX_ALPHA = 0.99; var LOGO_VALUE = 255; function removeWatermark(imageData, alphaMap, position) { const { x, y, width, height } = position; for (let row = 0; row < height; row++) { for (let col = 0; col < width; col++) { const imgIdx = ((y + row) * imageData.width + (x + col)) * 4; const alphaIdx = row * width + col; let alpha = alphaMap[alphaIdx]; if (alpha < ALPHA_THRESHOLD) { continue; } alpha = Math.min(alpha, MAX_ALPHA); const oneMinusAlpha = 1 - alpha; for (let c = 0; c < 3; c++) { const watermarked = imageData.data[imgIdx + c]; const original = (watermarked - alpha * LOGO_VALUE) / oneMinusAlpha; imageData.data[imgIdx + c] = Math.max(0, Math.min(255, Math.round(original))); } } } } // src/assets/bg_48.png var bg_48_default = ""; // src/assets/bg_96.png var bg_96_default = ""; // src/core/watermarkEngine.js function detectWatermarkConfig(imageWidth, imageHeight) { if (imageWidth > 1024 && imageHeight > 1024) { return { logoSize: 96, marginRight: 64, marginBottom: 64 }; } else { return { logoSize: 48, marginRight: 32, marginBottom: 32 }; } } function calculateWatermarkPosition(imageWidth, imageHeight, config) { const { logoSize, marginRight, marginBottom } = config; return { x: imageWidth - marginRight - logoSize, y: imageHeight - marginBottom - logoSize, width: logoSize, height: logoSize }; } var WatermarkEngine = class _WatermarkEngine { constructor(bgCaptures) { this.bgCaptures = bgCaptures; this.alphaMaps = {}; } static async create() { const bg48 = new Image(); const bg96 = new Image(); await Promise.all([ new Promise((resolve, reject) => { bg48.onload = resolve; bg48.onerror = reject; bg48.src = bg_48_default; }), new Promise((resolve, reject) => { bg96.onload = resolve; bg96.onerror = reject; bg96.src = bg_96_default; }) ]); return new _WatermarkEngine({ bg48, bg96 }); } /** * Get alpha map from background captured image based on watermark size * @param {number} size - Watermark size (48 or 96) * @returns {Promise} Alpha map */ async getAlphaMap(size) { if (this.alphaMaps[size]) { return this.alphaMaps[size]; } const bgImage = size === 48 ? this.bgCaptures.bg48 : this.bgCaptures.bg96; const canvas = document.createElement("canvas"); canvas.width = size; canvas.height = size; const ctx = canvas.getContext("2d"); ctx.drawImage(bgImage, 0, 0); const imageData = ctx.getImageData(0, 0, size, size); const alphaMap = calculateAlphaMap(imageData); this.alphaMaps[size] = alphaMap; return alphaMap; } /** * Remove watermark from image based on watermark size * @param {HTMLImageElement|HTMLCanvasElement} image - Input image * @returns {Promise} Processed canvas */ async removeWatermarkFromImage(image) { const canvas = document.createElement("canvas"); canvas.width = image.width; canvas.height = image.height; const ctx = canvas.getContext("2d"); ctx.drawImage(image, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const config = detectWatermarkConfig(canvas.width, canvas.height); const position = calculateWatermarkPosition(canvas.width, canvas.height, config); const alphaMap = await this.getAlphaMap(config.logoSize); removeWatermark(imageData, alphaMap, position); ctx.putImageData(imageData, 0, 0); return canvas; } /** * Get watermark information (for display) * @param {number} imageWidth - Image width * @param {number} imageHeight - Image height * @returns {Object} Watermark information {size, position, config} */ getWatermarkInfo(imageWidth, imageHeight) { const config = detectWatermarkConfig(imageWidth, imageHeight); const position = calculateWatermarkPosition(imageWidth, imageHeight, config); return { size: config.logoSize, position, config }; } }; // src/userscript/index.js var engine = null; var processingQueue = /* @__PURE__ */ new Set(); var debounce = (func, wait) => { let timeout; return (...args) => { clearTimeout(timeout); timeout = setTimeout(() => func(...args), wait); }; }; var loadImage = (src) => new Promise((resolve, reject) => { const img = new Image(); img.onload = () => resolve(img); img.onerror = reject; img.src = src; }); var canvasToBlob = (canvas, type = "image/png") => new Promise((resolve) => canvas.toBlob(resolve, type)); var isValidGeminiImage = (img) => img.closest("generated-image,.generated-image-container") !== null; var findGeminiImages = () => [...document.querySelectorAll('img[src*="googleusercontent.com"]')].filter(isValidGeminiImage); var fetchBlob = (url) => new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: "GET", url, responseType: "blob", onload: (response) => resolve(response.response), onerror: reject }); }); var replaceWithNormalSize = (src) => { return src.replace(/=s\d+(?=[-?#]|$)/, "=s0"); }; async function processImage(imgElement) { if (!engine || processingQueue.has(imgElement)) return; processingQueue.add(imgElement); imgElement.dataset.watermarkProcessed = "processing"; const originalSrc = imgElement.src; try { imgElement.src = ""; const normalSizeBlob = await fetchBlob(replaceWithNormalSize(originalSrc)); const normalSizeBlobUrl = URL.createObjectURL(normalSizeBlob); const normalSizeImg = await loadImage(normalSizeBlobUrl); const processedCanvas = await engine.removeWatermarkFromImage(normalSizeImg); const processedBlob = await canvasToBlob(processedCanvas); URL.revokeObjectURL(normalSizeBlobUrl); imgElement.src = URL.createObjectURL(processedBlob); imgElement.dataset.watermarkProcessed = "true"; console.log("[Gemini Watermark Remover] Processed image"); } catch (error) { console.warn("[Gemini Watermark Remover] Failed to process image:", error); imgElement.dataset.watermarkProcessed = "failed"; imgElement.src = originalSrc; } finally { processingQueue.delete(imgElement); } } var processAllImages = () => { const images = findGeminiImages(); if (images.length === 0) return; console.log(`[Gemini Watermark Remover] Found ${images.length} images to process`); images.forEach(processImage); }; var setupMutationObserver = () => { new MutationObserver(debounce(processAllImages, 100)).observe(document.body, { childList: true, subtree: true }); console.log("[Gemini Watermark Remover] MutationObserver active"); }; async function processImageBlob(blob) { const blobUrl = URL.createObjectURL(blob); const img = await loadImage(blobUrl); const canvas = await engine.removeWatermarkFromImage(img); URL.revokeObjectURL(blobUrl); return canvasToBlob(canvas); } var GEMINI_URL_PATTERN = /^https:\/\/lh3\.googleusercontent\.com\/rd-gg(?:-dl)?\/.+=s(?!0-d\?).*/; var { fetch: origFetch } = unsafeWindow; unsafeWindow.fetch = async (...args) => { const url = typeof args[0] === "string" ? args[0] : args[0]?.url; if (GEMINI_URL_PATTERN.test(url)) { console.log("[Gemini Watermark Remover] Intercepting:", url); const origUrl = replaceWithNormalSize(url); if (typeof args[0] === "string") args[0] = origUrl; else if (args[0]?.url) args[0].url = origUrl; const response = await origFetch(...args); if (!engine || !response.ok) return response; try { const processedBlob = await processImageBlob(await response.blob()); return new Response(processedBlob, { status: response.status, statusText: response.statusText, headers: response.headers }); } catch (error) { console.warn("[Gemini Watermark Remover] Processing failed:", error); return response; } } return origFetch(...args); }; (async function init() { try { console.log("[Gemini Watermark Remover] Initializing..."); engine = await WatermarkEngine.create(); processAllImages(); setupMutationObserver(); console.log("[Gemini Watermark Remover] Ready"); } catch (error) { console.error("[Gemini Watermark Remover] Initialization failed:", error); } })(); })();