Spaces:
Sleeping
Sleeping
| from PIL import Image | |
| import os | |
| from typing import Literal, Optional | |
| from io import BytesIO | |
| def compress_image_file( | |
| input_path: str, | |
| output_path: str, | |
| quality: int = 85, | |
| format: Literal["JPEG", "PNG", "WEBP"] = "JPEG", | |
| max_width: Optional[int] = None, | |
| max_height: Optional[int] = None | |
| ) -> str: | |
| """ | |
| Compress an image file from disk. | |
| """ | |
| try: | |
| if not os.path.splitext(output_path)[1]: | |
| extension_map = {"JPEG": ".jpg", "PNG": ".png", "WEBP": ".webp"} | |
| output_path = output_path + extension_map[format] | |
| with Image.open(input_path) as img: | |
| if format == "JPEG" and img.mode in ("RGBA", "P"): | |
| img = img.convert("RGB") | |
| if max_width or max_height: | |
| img.thumbnail((max_width or img.width, max_height or img.height), Image.Resampling.LANCZOS) | |
| save_kwargs = {"format": format, "optimize": True} | |
| if format in ["JPEG", "WEBP"]: | |
| save_kwargs["quality"] = quality | |
| img.save(output_path, **save_kwargs) | |
| original_size = os.path.getsize(input_path) / 1024 / 1024 | |
| compressed_size = os.path.getsize(output_path) / 1024 / 1024 | |
| reduction = (1 - compressed_size/original_size) * 100 | |
| return f"✅ Compressed successfully!\nOriginal: {original_size:.2f}MB → Compressed: {compressed_size:.2f}MB\nReduction: {reduction:.1f}%" | |
| except Exception as e: | |
| return f"❌ Error: {str(e)}" | |
| def compress_image_memory(image: Image.Image, quality: int = 80, format: str = "JPEG") -> Image.Image: | |
| """ | |
| Compress an image in memory and return the compressed image. | |
| """ | |
| if format == "JPEG" and image.mode in ("RGBA", "P"): | |
| image = image.convert("RGB") | |
| output = BytesIO() | |
| save_kwargs = {"format": format, "optimize": True} | |
| if format in ["JPEG", "WEBP"]: | |
| save_kwargs["quality"] = quality | |
| image.save(output, **save_kwargs) | |
| output.seek(0) | |
| return Image.open(output) | |