|
from PIL import Image, ImageDraw, ImageFont |
|
import os |
|
from typing import Dict, Any |
|
import cv2 |
|
import numpy as np |
|
|
|
def add_watermark(image: Image.Image, watermark_text: str, opacity: float = 0.5) -> Image.Image: |
|
""" |
|
Add a semi-transparent text watermark directly to a PIL Image. |
|
|
|
Args: |
|
image: PIL Image object to watermark |
|
watermark_text: Text to use as watermark |
|
opacity: Opacity of the watermark (0.1-1.0) |
|
|
|
Returns: |
|
PIL Image with watermark added |
|
""" |
|
from PIL import ImageDraw, ImageFont |
|
|
|
overlay = Image.new('RGBA', image.size, (255, 255, 255, 0)) |
|
draw = ImageDraw.Draw(overlay) |
|
|
|
try: |
|
font_size = min(image.width, image.height) // 20 |
|
font = ImageFont.truetype("arial.ttf", font_size) |
|
except: |
|
font = ImageFont.load_default() |
|
|
|
bbox = draw.textbbox((0, 0), watermark_text, font=font) |
|
text_width = bbox[2] - bbox[0] |
|
text_height = bbox[3] - bbox[1] |
|
|
|
x = (image.width - text_width) // 2 |
|
y = (image.height - text_height) // 2 |
|
|
|
alpha_value = int(255 * opacity) |
|
text_color = (255, 255, 255, alpha_value) |
|
shadow_color = (0, 0, 0, int(alpha_value * 0.5)) |
|
|
|
draw.text((x-2, y-2), watermark_text, fill=shadow_color, font=font) |
|
draw.text((x, y), watermark_text, fill=text_color, font=font) |
|
|
|
watermarked = Image.alpha_composite(image.convert('RGBA'), overlay) |
|
return watermarked.convert('RGB') |
|
|
|
def remove_watermark(image_path: str, alpha: float = 2.0, beta: float = -160) -> Dict[str, Any]: |
|
""" |
|
Attempt to remove watermarks from an image using contrast and brightness adjustment. |
|
|
|
Args: |
|
image_path: The path to the input image file. |
|
alpha: Contrast control (1.0-3.0, default 2.0). Higher values increase contrast. |
|
beta: Brightness control (-255 to 255, default -160). Negative values decrease brightness. |
|
|
|
Returns: |
|
A dictionary containing success status, file paths, and operation details. |
|
On success: success=True, input_path, output_path, output_size_bytes, alpha, beta, message. |
|
On failure: success=False, error message, input_path, output_path=None. |
|
""" |
|
try: |
|
img = cv2.imread(image_path) |
|
|
|
if img is None: |
|
raise ValueError("Could not load image") |
|
|
|
new = alpha * img + beta |
|
new = np.clip(new, 0, 255).astype(np.uint8) |
|
|
|
base_dir = os.path.dirname(image_path) |
|
base_name, ext = os.path.splitext(os.path.basename(image_path)) |
|
new_filename = f"{base_name}_cleaned{ext}" |
|
new_path = os.path.join(base_dir, new_filename) |
|
|
|
cv2.imwrite(new_path, new) |
|
output_size = os.path.getsize(new_path) |
|
|
|
return { |
|
"success": True, |
|
"message": "Watermark removal attempted successfully", |
|
"input_path": image_path, |
|
"output_path": new_path, |
|
"output_size_bytes": output_size, |
|
"alpha": alpha, |
|
"beta": beta |
|
} |
|
|
|
except Exception as e: |
|
return { |
|
"success": False, |
|
"error": str(e), |
|
"input_path": image_path, |
|
"output_path": None |
|
} |