Spaces:
Sleeping
Sleeping
| from fastapi import FastAPI, File, UploadFile, Form | |
| from fastapi.responses import HTMLResponse | |
| from fastapi.staticfiles import StaticFiles | |
| import numpy as np | |
| from PIL import Image | |
| from io import BytesIO | |
| import requests | |
| import base64 | |
| import os | |
| from tkinter import Tk, Label, Button, Radiobutton, IntVar | |
| app = FastAPI() | |
| # Mount the static folder for CSS and other assets | |
| app.mount("/static", StaticFiles(directory="static"), name="static") | |
| # Function to add padding | |
| def fill_rectangle_cropper(img, padding_type): | |
| # Calculate the average color of the image | |
| avg_color_per_row = np.average(np.array(img), axis=0) | |
| avg_color = np.average(avg_color_per_row, axis=0) | |
| if padding_type == "top_bottom": | |
| # Increase height to create a rectangle | |
| new_height = int(img.width * (4/3)) # Example: height = width * 4/3 | |
| newimg = Image.new( | |
| 'RGB', | |
| (img.width, new_height), | |
| (round(avg_color[0]), round(avg_color[1]), round(avg_color[2])) | |
| ) | |
| padding_top = (new_height - img.height) // 2 | |
| newimg.paste(img, (0, padding_top)) # Center the image vertically | |
| return newimg | |
| elif padding_type == "left_right": | |
| # Increase width to create a rectangle | |
| new_width = int(img.height * (4/3)) # Example: width = height * 4/3 | |
| newimg = Image.new( | |
| 'RGB', | |
| (new_width, img.height), | |
| (round(avg_color[0]), round(avg_color[1]), round(avg_color[2])) | |
| ) | |
| padding_left = (new_width - img.width) // 2 | |
| newimg.paste(img, (padding_left, 0)) # Center the image horizontally | |
| return newimg | |
| # Home Page | |
| def home_page(): | |
| return """ | |
| <html> | |
| <head> | |
| <title>Part of Idoia's Developer Portfolio - Innovating the Web</title> | |
| <link rel="stylesheet" href="/static/styles/style.css"> | |
| <!-- Meta Tags for SEO --> | |
| <meta charset="UTF-8"> | |
| <meta name="viewport" content="width=device-width, initial-scale=1.0"> | |
| <meta name="description" content="Explore the developer portfolio of Idoia, showcasing expertise in FastAPI, web development, and cutting-edge applications."> | |
| <meta name="keywords" content="Idoia, Developer, FastAPI, Web Development, Python Projects, Image Processing, Online Portfolio"> | |
| <meta name="author" content="Idoia"> | |
| <!-- Open Graph Meta Tags --> | |
| <meta property="og:title" content="Idoia's Developer Portfolio - Innovating the Web"> | |
| <meta property="og:description" content="Showcasing FastAPI projects, web apps, and image processing expertise. Explore Idoia's developer journey."> | |
| <meta property="og:image" content="/static/images/banner.jpg"> | |
| <meta property="og:url" content="https://webdevserv.github.io/html_bites/dev/webdev.html"> | |
| <meta property="og:type" content="website"> | |
| <!-- Twitter Card Meta Tags --> | |
| <meta name="twitter:card" content="summary_large_image"> | |
| <meta name="twitter:title" content="Idoia's Developer Portfolio - Innovating the Web"> | |
| <meta name="twitter:description" content="Discover the developer profile of Idoia. Dive into FastAPI-powered web apps and creative Python projects."> | |
| <meta name="twitter:image" content="/static/images/banner.jpg"> | |
| <link rel="icon" href="/static/images/6464.ico" type="image/x-icon"> | |
| <!-- Google Fonts (Optional for Styling) --> | |
| <link href="https://fonts.googleapis.com/css2?family=Roboto:wght@400;700&display=swap" rel="stylesheet"> | |
| <!-- Schema.org JSON-LD (Optional for Rich Snippets) --> | |
| <script type="application/ld+json"> | |
| { | |
| "@context": "https://schema.org", | |
| "@type": "Person", | |
| "name": "Idoia", | |
| "jobTitle": "Web Developer", | |
| "url": "https://webdevserv.github.io/html_bites/dev/webdev.html", | |
| "image": "https://idoia-dev-portfolio.com/static/images/banner.jpg", | |
| "description": "Experienced web developer with a focus on Streamlit, HF, Python, and modern web applications." | |
| } | |
| </script> | |
| </head> | |
| <body> | |
| <img class="banner" src="/static/images/banner.jpg" alt="Banner" width="100%"> | |
| <h2>Rectangle and Fill Image App</h2> | |
| <p>Please select an option below:</p> | |
| <ul> | |
| <li><a href="/demo">Demo</a></li> | |
| <li><a href="/application">Application</a></li> | |
| </ul> | |
| <div id="credit">Image creations by | |
| <a href="https://stock.adobe.com/es/contributor/212598146/UMAMI%20LAB" target="_blank">Adobe Stock User Umami Lab</a> | |
| and | |
| <a href="https://www.shutterstock.com/g/Idoia+Lerchundi?rid=430751957" target="_blank">Shutterstock User PhoArt101</a>. | |
| </div> | |
| </body> | |
| </html> | |
| """ | |
| # Demo Page | |
| def demo_page(): | |
| # URLs for demo images | |
| url1 = "https://raw.githubusercontent.com/webdevserv/images_video/main/squareportrait.png" | |
| url2 = "https://raw.githubusercontent.com/webdevserv/images_video/main/squarelandscape.png" | |
| # Process the first image | |
| response = requests.get(url1) | |
| img1 = Image.open(BytesIO(response.content)).convert("RGB") | |
| rectangled_img1 = fill_rectangle_cropper(img1, "top_bottom") | |
| output1 = BytesIO() | |
| rectangled_img1.save(output1, format="JPEG") | |
| encoded_img1 = base64.b64encode(output1.getvalue()).decode("utf-8") | |
| # Process the second image | |
| response = requests.get(url2) | |
| img2 = Image.open(BytesIO(response.content)).convert("RGB") | |
| rectangled_img2 = fill_rectangle_cropper(img2, "left_right") | |
| output2 = BytesIO() | |
| rectangled_img2.save(output2, format="JPEG") | |
| encoded_img2 = base64.b64encode(output2.getvalue()).decode("utf-8") | |
| return f""" | |
| <html> | |
| <head> | |
| <title>Part of Idoia's Developer Portfolio - Innovating the Web</title> | |
| <link rel="stylesheet" href="/static/styles/style.css"> | |
| </head> | |
| <body> | |
| <img class="banner" src="/static/images/banner.jpg" alt="Banner" width="100%"> | |
| <h2>Rectangle Image Demo</h2> | |
| <p>Image will be rectangled with color filler where applicable.</p> | |
| <h3>Result 1:</h3> | |
| <img src="data:image/jpeg;base64,{encoded_img1}" /> | |
| <h3>Result 2:</h3> | |
| <img src="data:image/jpeg;base64,{encoded_img2}" /> | |
| <p><a href="/">Back</a></p> | |
| <div id="credit">Image creations by | |
| <a href="https://stock.adobe.com/es/contributor/212598146/UMAMI%20LAB" target="_blank">Adobe Stock User Umami Lab</a> | |
| and | |
| <a href="https://www.shutterstock.com/g/Idoia+Lerchundi?rid=430751957" target="_blank">Shutterstock User PhoArt101</a>. | |
| </div> | |
| </body> | |
| </html> | |
| """ | |
| # Application Page | |
| def application_page(): | |
| return """ | |
| <html> | |
| <head> | |
| <title>Part of Idoia's Developer Portfolio - Innovating the Web</title> | |
| <link rel="stylesheet" href="/static/styles/style.css"> | |
| </head> | |
| <body> | |
| <img class="banner" src="/static/images/banner.jpg" alt="Banner" width="100%"> | |
| <h2>Rectangle Image Application</h2> | |
| <p>Upload a JPG image to rectangle and fill with color filler.</p> | |
| <form action="/upload/" enctype="multipart/form-data" method="post"> | |
| <label for="file">Upload your image:</label> | |
| <input name="file" type="file" required><br><br> | |
| <label>Choose the padding direction:</label><br> | |
| <input type="radio" id="top_bottom" name="padding_type" value="top_bottom" checked> | |
| <label for="top_bottom">Top/Bottom</label><br> | |
| <input type="radio" id="left_right" name="padding_type" value="left_right"> | |
| <label for="left_right">Left/Right</label><br><br> | |
| <input type="submit" value="Rectangle It"> | |
| </form> | |
| <a href="/">Back</a> | |
| <div id="credit">Image creations by | |
| <a href="https://stock.adobe.com/es/contributor/212598146/UMAMI%20LAB" target="_blank">Adobe Stock User Umami Lab</a> | |
| and | |
| <a href="https://www.shutterstock.com/g/Idoia+Lerchundi?rid=430751957" target="_blank">Shutterstock User PhoArt101</a>. | |
| </div> | |
| </body> | |
| </html> | |
| """ | |
| async def upload_file(file: UploadFile = File(...), padding_type: str = Form(...)): | |
| try: | |
| # Await file upload | |
| contents = await file.read() | |
| img = Image.open(BytesIO(contents)).convert("RGB") | |
| # Apply padding based on user's choice | |
| rectangled_img = fill_rectangle_cropper(img,padding_type) | |
| # Save the rectangle image (original size) | |
| output = BytesIO() | |
| rectangled_img.save(output, format="JPEG") | |
| output.seek(0) | |
| # Encode the full-size image for download | |
| full_size_encoded_img = base64.b64encode(output.getvalue()).decode("utf-8") | |
| # Resize the image for display (512px by 512px) | |
| display_img = rectangled_img.copy() | |
| desired_width = 512 | |
| aspect_ratio = display_img.height / display_img.width | |
| desired_height = int(desired_width * aspect_ratio) | |
| display_img.thumbnail((desired_width, desired_height)) | |
| display_output = BytesIO() | |
| display_img.save(display_output, format="JPEG") | |
| display_output.seek(0) | |
| # Encode the resized display image | |
| display_encoded_img = base64.b64encode(display_output.getvalue()).decode("utf-8") | |
| # Return the HTML response | |
| return HTMLResponse( | |
| content=f""" | |
| <html> | |
| <head> | |
| <title>Part of Idoia's Developer Portfolio - Innovating the Web</title> | |
| <link rel="stylesheet" href="/static/styles/style.css"> | |
| </head> | |
| <body> | |
| <img class="banner" src="/static/images/banner.jpg" alt="Banner" width="100%"> | |
| <h2>Image successfully rectangled!</h2> | |
| <!--<img src='_encoded_img'/>--> | |
| <img src='data:image/jpeg;base64,{display_encoded_img}' width="512"/> | |
| <p><a href="data:image/jpeg;base64,{full_size_encoded_img}" download="rectangled_image.jpg"> | |
| Download Full-Size Image</a></p> | |
| <p><a href="/">Back</a></p> | |
| <div id="credit">Image creations by | |
| <a href="https://stock.adobe.com/es/contributor/212598146/UMAMI%20LAB" target="_blank">Adobe Stock User Umami Lab</a> | |
| and | |
| <a href="https://www.shutterstock.com/g/Idoia+Lerchundi?rid=430751957" target="_blank">Shutterstock User PhoArt101</a>. | |
| </div> | |
| </body> | |
| </html> | |
| """, | |
| media_type="text/html" | |
| ) | |
| except Exception as e: | |
| return HTMLResponse(content=f"<h3>An error occurred: {e}</h3>", media_type="text/html") | |
| if __name__ == "__main__": | |
| import uvicorn | |
| uvicorn.run(app, host="0.0.0.0", port=int(os.environ.get("PORT", 7860))) |