310 lines
23 KiB
JavaScript
310 lines
23 KiB
JavaScript
// ==UserScript==
|
|
// @name Gemini Watermark Remover
|
|
// @namespace https://github.com/journey-ad
|
|
// @version 1.0.0
|
|
// @description Automatically removes watermarks from Gemini AI generated images
|
|
// @author journey-ad
|
|
// @match https://gemini.google.com/app/*
|
|
// @grant GM_xmlhttpRequest
|
|
// @run-at document-start
|
|
// ==/UserScript==
|
|
|
|
(() => {
|
|
// ../js/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;
|
|
}
|
|
|
|
// ../js/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)));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ../js/core/watermarkEngine.js
|
|
var BG_48_PATH = "../assets/bg_48.png";
|
|
var BG_96_PATH = "../assets/bg_96.png";
|
|
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_PATH;
|
|
}),
|
|
new Promise((resolve, reject) => {
|
|
bg96.onload = resolve;
|
|
bg96.onerror = reject;
|
|
bg96.src = BG_96_PATH;
|
|
})
|
|
]);
|
|
return new _WatermarkEngine({ bg48, bg96 });
|
|
}
|
|
/**
|
|
* 从背景捕获图像获取 alpha map
|
|
* @param {number} size - 水印尺寸 (48 或 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;
|
|
}
|
|
/**
|
|
* 移除图像上的水印
|
|
* @param {HTMLImageElement|HTMLCanvasElement} image - 输入图像
|
|
* @returns {Promise<HTMLCanvasElement>} 处理后的 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;
|
|
}
|
|
/**
|
|
* 获取水印信息(用于显示)
|
|
* @param {number} imageWidth - 图像宽度
|
|
* @param {number} imageHeight - 图像高度
|
|
* @returns {Object} 水印信息
|
|
*/
|
|
getWatermarkInfo(imageWidth, imageHeight) {
|
|
const config = detectWatermarkConfig(imageWidth, imageHeight);
|
|
const position = calculateWatermarkPosition(imageWidth, imageHeight, config);
|
|
return {
|
|
size: config.logoSize,
|
|
position,
|
|
config
|
|
};
|
|
}
|
|
};
|
|
|
|
// ../assets/bg_48.png
|
|
var bg_48_default = "";
|
|
|
|
// ../assets/bg_96.png
|
|
var bg_96_default = "";
|
|
|
|
// index.js
|
|
function debounce(func, wait) {
|
|
let timeout;
|
|
return function(...args) {
|
|
clearTimeout(timeout);
|
|
timeout = setTimeout(() => func.apply(this, args), wait);
|
|
};
|
|
}
|
|
function findGeminiImages() {
|
|
return [...document.querySelectorAll('img[src*="googleusercontent.com"]')].filter(isValidGeminiImage);
|
|
}
|
|
function isValidGeminiImage(img) {
|
|
if (/=s\d+\-rj/.test(img.src)) return true;
|
|
return img.naturalWidth >= 256 && img.naturalHeight >= 256;
|
|
}
|
|
var engine = null;
|
|
var processingQueue = /* @__PURE__ */ new Set();
|
|
async function processImage(imgElement) {
|
|
if (!engine || processingQueue.has(imgElement)) return;
|
|
try {
|
|
imgElement.dataset.watermarkProcessed = "processing";
|
|
processingQueue.add(imgElement);
|
|
if (!isValidGeminiImage(imgElement)) {
|
|
imgElement.dataset.watermarkProcessed = "skipped";
|
|
return;
|
|
}
|
|
const src = imgElement.src;
|
|
imgElement.src = "";
|
|
imgElement.dataset.src = src;
|
|
const blob = await new Promise((resolve, reject) => {
|
|
GM_xmlhttpRequest({
|
|
method: "GET",
|
|
url: src.replace(/=s\d+.+$/, "=s0"),
|
|
responseType: "blob",
|
|
onload: (response) => resolve(response.response),
|
|
onerror: reject
|
|
});
|
|
});
|
|
const blobUrl = URL.createObjectURL(blob);
|
|
const tempImg = new Image();
|
|
await new Promise((resolve, reject) => {
|
|
tempImg.onload = resolve;
|
|
tempImg.onerror = reject;
|
|
tempImg.src = blobUrl;
|
|
});
|
|
const resultCanvas = await engine.removeWatermarkFromImage(tempImg);
|
|
const processedBlob = await new Promise((resolve) => {
|
|
resultCanvas.toBlob(resolve, "image/png");
|
|
});
|
|
URL.revokeObjectURL(blobUrl);
|
|
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";
|
|
} finally {
|
|
processingQueue.delete(imgElement);
|
|
}
|
|
}
|
|
async function processAllImages() {
|
|
const images = findGeminiImages();
|
|
console.log(`[Gemini Watermark Remover] Found ${images.length} images to process`);
|
|
for (const img of images) {
|
|
processImage(img);
|
|
}
|
|
}
|
|
function setupMutationObserver() {
|
|
const observer = new MutationObserver(debounce(() => {
|
|
processAllImages();
|
|
}, 100));
|
|
observer.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 = new Image();
|
|
await new Promise((resolve, reject) => {
|
|
img.onload = resolve;
|
|
img.onerror = reject;
|
|
img.src = blobUrl;
|
|
});
|
|
const resultCanvas = await engine.removeWatermarkFromImage(img);
|
|
URL.revokeObjectURL(blobUrl);
|
|
return new Promise((resolve) => {
|
|
resultCanvas.toBlob(resolve, "image/png");
|
|
});
|
|
}
|
|
async function getImage(base64) {
|
|
const img = new Image();
|
|
img.src = base64;
|
|
await new Promise((resolve, reject) => {
|
|
img.onload = resolve;
|
|
img.onerror = reject;
|
|
});
|
|
return img;
|
|
}
|
|
var { fetch: origFetch } = unsafeWindow;
|
|
unsafeWindow.fetch = async (...args) => {
|
|
const url = typeof args[0] === "string" ? args[0] : args[0]?.url;
|
|
if (/^https:\/\/lh3\.googleusercontent\.com\/rd-gg(-dl)?\//.test(url)) {
|
|
console.log("[Gemini Watermark Remover] Intercepting:", url);
|
|
const origUrl = url.replace(/=s\d+.+$/, "=s0");
|
|
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 blob = await response.blob();
|
|
const processedBlob = await processImageBlob(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 = new WatermarkEngine({
|
|
bg48: await getImage(bg_48_default),
|
|
bg96: await getImage(bg_96_default)
|
|
});
|
|
await processAllImages();
|
|
setupMutationObserver();
|
|
console.log("[Gemini Watermark Remover] Ready");
|
|
} catch (error) {
|
|
console.error("[Gemini Watermark Remover] Initialization failed:", error);
|
|
}
|
|
}
|
|
init();
|
|
})();
|