Greasy Fork

Greasy Fork is available in English.

Gemini NanoBanana 图片水印移除

自动移除 Gemini AI 生成图像中的水印

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==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);
    }
  })();
})();