|
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 = {} |
|
|
|
|
|
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_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) |
|
|
|
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 |
|
|
|
|
|
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%") |
|
|