Greasy Fork is available in English.
自动移除 Gemini AI 生成图像中的水印
// ==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
// ==/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<Float32Array>} 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<HTMLCanvasElement>} 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);
}
})();
})();