nnrrjf4b / app.py
ssboost's picture
Update app.py
9bdda18 verified
raw
history blame
7.42 kB
import os
import logging
from PIL import Image
import gradio as gr
from dotenv import load_dotenv
from db_examples import product_background_examples
# ํ™˜๊ฒฝ ๋ณ€์ˆ˜ ๋กœ๋“œ
load_dotenv()
# ๋กœ๊น… ์„ค์ •
logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s - %(message)s')
logger = logging.getLogger(__name__)
# ์ด๋ฏธ์ง€ ์บ์‹œ
IMAGE_CACHE = {}
# CSS (์›๋ณธ ์Šคํƒ€์ผ ์œ ์ง€ + ์˜ˆ์‹œ ํฌ๊ธฐ ์ถ•์†Œ + ํ•˜๋‹จ ์—ฌ๋ฐฑ ์ œ๊ฑฐ)
custom_css = """
:root {
--primary-color: #FB7F0D;
--secondary-color: #ff9a8b;
--accent-color: #FF6B6B;
--background-color: #FFFFFF;
--card-bg: #ffffff;
--text-color: #334155;
--border-radius: 18px;
--shadow: 0 8px 30px rgba(251, 127, 13, 0.08);
}
.example-gallery {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(120px, 1fr));
gap: 20px;
padding: 20px;
}
.example-item {
cursor: pointer;
border: 1px solid rgba(0, 0, 0, 0.1);
border-radius: var(--border-radius);
overflow: hidden;
transition: all 0.3s ease;
background: white;
}
.example-item:hover {
box-shadow: 0 10px 20px rgba(0, 0, 0, 0.1);
transform: translateY(-2px);
}
.example-item img {
width: 80%;
height: 80px;
object-fit: cover;
}
.image-container {
border-radius: var(--border-radius);
overflow: hidden;
border: 1px solid rgba(0, 0, 0, 0.08);
transition: all 0.3s ease;
background-color: white;
aspect-ratio: 1 / 1;
}
.image-container:hover {
box-shadow: var(--shadow);
}
.image-container img {
width: 100%;
height: 100%;
object-fit: contain;
}
.section-title {
display: flex;
align-items: center;
font-size: 20px;
font-weight: 700;
color: #333333;
margin-bottom: 10px;
padding-bottom: 5px;
border-bottom: 2px solid #FB7F0D;
font-family: 'Pretendard', 'Noto Sans KR', -apple-system, BlinkMacSystemFont, sans-serif;
}
.section-title img {
margin-right: 10px;
width: 24px;
height: 24px;
}
body {
font-family: 'Pretendard', 'Noto Sans KR', -apple-system, BlinkMacSystemFont, sans-serif;
background-color: var(--background-color);
color: var(--text-color);
line-height: 1.6;
margin: 0;
padding: 0;
}
footer { visibility: hidden; }
.gradio-container {
padding-bottom: 0px !important;
margin-bottom: 0px !important;
}
.gr-block {
margin-bottom: 0px !important;
}
"""
# FontAwesome ์•„์ด์ฝ˜ ๋งํฌ
fontawesome_link = """
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.0.0/css/all.min.css" crossorigin="anonymous" referrerpolicy="no-referrer" />
"""
# ์ด๋ฏธ์ง€ ๋กœ๋“œ ๋ฐ ์บ์‹œ
def load_image_cached(path):
if path not in IMAGE_CACHE:
try:
img = Image.open(path)
# ์ตœ๋Œ€ ํฌ๊ธฐ 1000px๋กœ ์ถ•์†Œ
max_dim = max(img.size)
if max_dim > 1000:
ratio = 1000 / max_dim
img = img.resize((int(img.width*ratio), int(img.height*ratio)), Image.Resampling.LANCZOS)
IMAGE_CACHE[path] = img
except Exception as e:
logger.error(f"์ด๋ฏธ์ง€ ๋กœ๋“œ ์‹คํŒจ: {path} - {e}")
return None
return IMAGE_CACHE[path]
# ์˜ˆ์‹œ ๋ฏธ๋ฆฌ ๋กœ๋“œ
def preload_example_images():
for ex in product_background_examples:
load_image_cached(ex[0])
load_image_cached(ex[5])
# ์˜ˆ์‹œ ์„ ํƒ ํ•ธ๋“ค๋Ÿฌ
def load_example(evt: gr.SelectData):
ex = product_background_examples[evt.index]
return ex[0], ex[1], ex[2], ex[3], ex[4] or "(์—†์Œ)", ex[5]
# ์ฒซ ๋ฒˆ์งธ ์˜ˆ์‹œ ์ž๋™ ๋กœ๋“œ
def load_first_example():
if product_background_examples:
ex = product_background_examples[0]
return ex[0], ex[1], ex[2], ex[3], ex[4] or "(์—†์Œ)", ex[5]
return None, "", "", "", "", None
# Gradio ์•ฑ ์ƒ์„ฑ
def create_app():
with gr.Blocks(css=custom_css, theme=gr.themes.Default(
primary_hue="orange", secondary_hue="orange",
font=[gr.themes.GoogleFont("Noto Sans KR"), "ui-sans-serif", "system-ui"]
)) as demo:
gr.HTML(fontawesome_link)
# ๋ฉ”์ธ ๋ทฐ: ์ฒซ ๋ฒˆ์งธ ์˜ˆ์‹œ
with gr.Column(elem_classes="custom-section-group custom-frame"):
gr.HTML('<div class="section-title"><img src="https://cdn-icons-png.flaticon.com/512/681/681443.png"> ๋ฐฐ๊ฒฝ ์œ ํ˜•๋ณ„ ์˜ˆ์‹œ</div>')
with gr.Row():
example_input_image = gr.Image(
label="์ž…๋ ฅ ์ด๋ฏธ์ง€",
elem_classes="image-container",
interactive=False,
show_download_button=False,
show_share_button=False,
container=False
)
with gr.Column():
example_bg_type = gr.Textbox(label="๋ฐฐ๊ฒฝ ์œ ํ˜•", interactive=False)
example_bg_option = gr.Textbox(label="๋ฐฐ๊ฒฝ ์„ ํƒ", interactive=False)
example_product_name = gr.Textbox(label="์ƒํ’ˆ๋ช…", interactive=False)
example_additional_info = gr.Textbox(label="์ถ”๊ฐ€ ์š”์ฒญ์‚ฌํ•ญ", interactive=False)
example_output_image = gr.Image(
label="๊ฒฐ๊ณผ ์ด๋ฏธ์ง€",
elem_classes="image-container",
interactive=False,
show_download_button=False,
show_share_button=False,
container=False
)
# ๊ฐค๋Ÿฌ๋ฆฌ ๋ทฐ: ์œ ํ˜•๋ณ„ ๊ทธ๋ฃนํ™”
with gr.Column(elem_classes="custom-section-group"):
bg_groups = {}
for idx, ex in enumerate(product_background_examples):
bg_groups.setdefault(ex[1], []).append((idx, ex))
for bg_type, items in bg_groups.items():
gr.HTML(f'<div class="section-title">{bg_type}</div>')
with gr.Row(elem_classes="example-gallery"):
for idx, ex in items:
item = gr.Image(
value=ex[5],
show_label=False,
elem_classes="example-item",
interactive=False,
show_download_button=False,
show_share_button=False,
container=False
)
item.select(
fn=load_example,
inputs=None,
outputs=[
example_input_image,
example_bg_type,
example_bg_option,
example_product_name,
example_additional_info,
example_output_image
]
)
# ์ฒซ ๋ฒˆ์งธ ์˜ˆ์‹œ ์ž๋™ ๋กœ๋“œ
demo.load(
fn=load_first_example,
outputs=[
example_input_image,
example_bg_type,
example_bg_option,
example_product_name,
example_additional_info,
example_output_image
]
)
return demo
if __name__ == "__main__":
preload_example_images()
app = create_app()
app.launch(share=False, inbrowser=True, width="100%")