codelion's picture
Update app.py
392146d verified
raw
history blame
4.82 kB
import gradio as gr
import PIL
from PIL import Image
import os
import tempfile
import zipfile
import shutil
from pathlib import Path
import numpy as np
# Define target sizes for different viewports
SIZES = [
(480, 423),
(696, 613),
(960, 846),
(1095, 965),
(1280, 1128),
(1440, 1269),
(1670, 1472),
(1870, 1648),
(2048, 1805)
]
def resize_image(img, target_size):
"""Resize image maintaining aspect ratio"""
img_copy = img.copy()
return img_copy.resize(target_size, PIL.Image.Resampling.LANCZOS)
def generate_html_snippet(image_path_prefix):
"""Generate HTML code snippet with srcset"""
srcset_items = [f"{image_path_prefix}-{w}x{h}.jpg {w}w" for w, h in SIZES]
srcset_str = ",\n ".join(srcset_items)
sizes_items = [
f"(max-width: {w}px) {w}px" for w, _ in SIZES[:-1]
] + [f"{SIZES[-1][0]}px"]
sizes_str = ",\n ".join(sizes_items)
# Use the middle size as default src
default_size = SIZES[4] # 1280x1128
html = f'''<!-- Responsive Image -->
<img
src="{image_path_prefix}-{default_size[0]}x{default_size[1]}.jpg"
srcset="
{srcset_str}
"
sizes="
{sizes_str}
"
alt="Your image description"
style="max-width: 100%; height: auto;"
>'''
return html
def process_image(input_img):
"""Main processing function"""
if input_img is None:
return None, None, "Please upload an image"
# Create temporary directory for processed images
temp_dir = tempfile.mkdtemp()
zip_path = os.path.join(temp_dir, "responsive_images.zip")
processed_paths = [] # Initialize the list outside try block
try:
# Get original filename
base_filename = "image" # Default filename
if isinstance(input_img, dict) and "name" in input_img:
original_name = input_img["name"]
base_filename = os.path.splitext(original_name)[0]
# Get image data
if isinstance(input_img, dict) and "image" in input_img:
img_data = input_img["image"]
else:
img_data = input_img
# Convert to PIL Image if needed
if isinstance(img_data, np.ndarray):
img = Image.fromarray(img_data)
else:
img = img_data
# Convert to RGB if necessary
if img.mode != 'RGB':
img = img.convert('RGB')
# Process each size
for width, height in SIZES:
resized = resize_image(img, (width, height))
output_path = os.path.join(temp_dir, f"{base_filename}-{width}x{height}.jpg")
resized.save(output_path, "JPEG", quality=90)
processed_paths.append(output_path)
# Create zip file
with zipfile.ZipFile(zip_path, 'w') as zf:
for path in processed_paths:
zf.write(path, os.path.basename(path))
# Generate HTML snippet using original filename
html_snippet = generate_html_snippet(f"images/{base_filename}")
return zip_path, html_snippet, "Processing completed successfully!"
except Exception as e:
if os.path.exists(zip_path):
try:
os.remove(zip_path)
except:
pass
return None, None, f"Error processing image: {str(e)}"
finally:
# Clean up temporary files except zip
for path in processed_paths:
try:
os.remove(path)
except:
pass
# Try to remove the temp directory if it's empty
try:
os.rmdir(temp_dir)
except:
pass
# Create Gradio interface
with gr.Blocks(title="Responsive Image Generator") as app:
gr.Markdown("""
# Responsive Image Generator
Upload an image to generate optimized versions for different viewport sizes.
You'll receive:
1. A ZIP file containing all sized versions
2. HTML code snippet with proper srcset attributes
""")
with gr.Row():
with gr.Column():
input_image = gr.Image(
label="Upload Original Image",
type="pil" # Changed to PIL type
)
process_btn = gr.Button("Process Image")
with gr.Column():
output_zip = gr.File(label="Download Processed Images")
output_html = gr.Code(
label="HTML Code Snippet",
language="html"
)
output_message = gr.Textbox(label="Status")
process_btn.click(
fn=process_image,
inputs=[input_image], # Now only passing single input
outputs=[output_zip, output_html, output_message]
)
app.launch()