zenityx's picture
Update app.py
0f8d476 verified
import gradio as gr
from transformers import MarianMTModel, MarianTokenizer
import torch
import random
# ตรวจสอบว่าใช้งาน GPU หรือไม่ (ในกรณีนี้ไม่มี GPU)
device = torch.device("cpu")
# เลือกโมเดลแปลภาษาที่มีขนาดเล็กลง
# คุณสามารถค้นหาโมเดลที่เล็กกว่าได้ที่ Hugging Face Model Hub
# ในตัวอย่างนี้ยังคงใช้ "Helsinki-NLP/opus-mt-th-en" แต่คุณสามารถเปลี่ยนได้ตามต้องการ
model_name = "Helsinki-NLP/opus-mt-th-en"
tokenizer = MarianTokenizer.from_pretrained(model_name)
model = MarianMTModel.from_pretrained(model_name).to(device)
model.eval() # ตั้งค่าโมเดลเป็นโหมดประเมินผล
# แคชสำหรับการแปลข้อความที่ไม่อยู่ในรายการคงที่
translation_cache = {}
def translate_batch(th_texts):
"""แปลข้อความแบบแบทช์เพื่อเพิ่มความเร็ว"""
inputs = tokenizer(th_texts, return_tensors="pt", padding=True, truncation=True, max_length=512).to(device)
with torch.no_grad():
translated = model.generate(**inputs, max_length=512)
en_texts = [tokenizer.decode(t, skip_special_tokens=True) for t in translated]
return en_texts
def translate_th_to_en(th_text: str) -> str:
th_text = th_text.strip()
if not th_text:
return ""
# ตรวจสอบแคชก่อน
if th_text in translation_cache:
return translation_cache[th_text]
# แปลข้อความ
en_text = translate_batch([th_text])[0]
# เก็บในแคช
translation_cache[th_text] = en_text
return en_text
# ตัวเลือก (ภาษาไทย) สำหรับ Dropdown
body_shapes_th = [
"ตัวเหนียวขยุกขยิก", "ตัวปุกปุยกลม", "ตัวเต็มไปด้วยหนาม",
"ตัวเล็กเหมือนแมลง", "ตัวโปร่งแสงลอยได้", "ตัวโคลนยืดหยุ่น",
"ตัวหินแข็งแรง", "ตัวทรงพีระมิด", "ตัวระยิบระยับ", "ตัวเหมือนฟองสบู่"
]
heads_th = [
"หัวตาเดียว", "หัวสองเขาเหมือนมังกร", "หัวมีเขาใหญ่มาก",
"หัวผีแสยะยิ้ม", "หัวกระต่ายน่ารัก", "หัวปลาหมึกสุดแปลก",
"หัวเพลิงลุก", "หัวงูพ่นพิษ", "หัววุ้นใส", "หัวเมฆหมอก"
]
arms_legs_th = [
"แขนหนวดปลาหมึกและขาปลาหมึก", "แขนกลและเท้าสเก็ต",
"ปีกขนนกและอุ้งเท้าสิงโต", "แขนเรืองแสงและขายานโฮเวอร์",
"แขนไม้หุ่นเชิดและขายักษ์แมงมุม", "แขนโลหะ",
"แขนเหล็กข้อต่อ", "ปีกค้างคาว", "ขาไก่ยักษ์", "แขนขนนุ่ม"
]
skin_patterns_th = [
"ลายสีรุ้ง", "จุดม่วงเรืองแสง", "เกล็ดเหล็ก",
"ขนนกวิบวับ", "จุดกลมกากเพชร", "ลายหินอ่อน",
"ลายไฟลุก", "ลายฟ้าแลบ", "ลายสเกลปลา", "ลายหมอก"
]
abilities_th = [
"พ่นไฟ", "ปล่อยฟองสบู่", "ควบคุมสายฟ้า",
"ร้องเพลงกล่อม", "วาร์ประยะสั้น", "เสกสิ่งของ",
"ปล่อยพลังงานแสง", "ควบคุมแรงโน้มถ่วง", "เปลี่ยนรูปร่าง", "เสกภาพลวงตา"
]
moods_th = [
"เป็นมิตร", "ขี้หงุดหงิด", "ตลกโปกฮา",
"ซนเป็นลิง", "ขี้อาย", "ขี้เล่น",
"เศร้าสร้อย", "จริงจัง", "มีความสุข", "ขี้ตกใจ"
]
# แปลล่วงหน้าสำหรับรายการคงที่
# คุณสามารถแปลข้อความเหล่านี้ล่วงหน้านอกโปรแกรมและใส่ค่าลงไปตรงนี้เพื่อความรวดเร็ว
body_shapes_en = {
"ตัวเหนียวขยุกขยิก": "a squishy and wobbly body",
"ตัวปุกปุยกลม": "a round and fluffy body",
"ตัวเต็มไปด้วยหนาม": "a spiky body",
"ตัวเล็กเหมือนแมลง": "a small, insect-like body",
"ตัวโปร่งแสงลอยได้": "a translucent, floating body",
"ตัวโคลนยืดหยุ่น": "a flexible, slime-like body",
"ตัวหินแข็งแรง": "a strong, rocky body",
"ตัวทรงพีระมิด": "a pyramidal body",
"ตัวระยิบระยับ": "a sparkling body",
"ตัวเหมือนฟองสบู่": "a bubble-like body"
}
heads_en = {
"หัวตาเดียว": "a single-eyed head",
"หัวสองเขาเหมือนมังกร": "a two-horned dragon-like head",
"หัวมีเขาใหญ่มาก": "a head with very large horns",
"หัวผีแสยะยิ้ม": "a smiling ghostly head",
"หัวกระต่ายน่ารัก": "a cute rabbit head",
"หัวปลาหมึกสุดแปลก": "a bizarre squid head",
"หัวเพลิงลุก": "a flaming head",
"หัวงูพ่นพิษ": "a venomous snake head",
"หัววุ้นใส": "a transparent jelly head",
"หัวเมฆหมอก": "a misty cloud head"
}
arms_legs_en = {
"แขนหนวดปลาหมึกและขาปลาหมึก": "octopus-like tentacles and legs",
"แขนกลและเท้าสเก็ต": "mechanical arms and skating feet",
"ปีกขนนกและอุ้งเท้าสิงโต": "feathered wings and lion-like paws",
"แขนเรืองแสงและขายานโฮเวอร์": "glowing arms and hover jets",
"แขนไม้หุ่นเชิดและขายักษ์แมงมุม": "puppet wooden arms and giant spider legs",
"แขนโลหะ": "metal arms",
"แขนเหล็กข้อต่อ": "jointed iron arms",
"ปีกค้างคาว": "bat wings",
"ขาไก่ยักษ์": "giant chicken legs",
"แขนขนนุ่ม": "soft feathered arms"
}
skin_patterns_en = {
"ลายสีรุ้ง": "rainbow patterns",
"จุดม่วงเรืองแสง": "glowing purple spots",
"เกล็ดเหล็ก": "iron scales",
"ขนนกวิบวับ": "shimmering feathers",
"จุดกลมกากเพชร": "diamond-like round dots",
"ลายหินอ่อน": "marble patterns",
"ลายไฟลุก": "flaming patterns",
"ลายฟ้าแลบ": "lightning patterns",
"ลายสเกลปลา": "fish scale patterns",
"ลายหมอก": "misty patterns"
}
abilities_en = {
"พ่นไฟ": "breathe fire",
"ปล่อยฟองสบู่": "release soap bubbles",
"ควบคุมสายฟ้า": "control lightning",
"ร้องเพลงกล่อม": "sing a lullaby",
"วาร์ประยะสั้น": "short-range warp",
"เสกสิ่งของ": "conjure objects",
"ปล่อยพลังงานแสง": "emit light energy",
"ควบคุมแรงโน้มถ่วง": "control gravity",
"เปลี่ยนรูปร่าง": "shape-shift",
"เสกภาพลวงตา": "create illusions"
}
moods_en = {
"เป็นมิตร": "friendly",
"ขี้หงุดหงิด": "irritable",
"ตลกโปกฮา": "funny",
"ซนเป็นลิง": "mischievous like a monkey",
"ขี้อาย": "shy",
"ขี้เล่น": "playful",
"เศร้าสร้อย": "sorrowful",
"จริงจัง": "serious",
"มีความสุข": "happy",
"ขี้ตกใจ": "easily startled"
}
def generate_monster_lab(
body_dd, body_tb,
head_dd, head_tb,
arms_dd, arms_tb,
skin_dd, skin_tb,
ability_dd, ability_tb,
mood_dd, mood_tb,
):
# ถ้า textbox มีข้อความ -> ใช้ค่านั้น, ถ้าว่าง -> ใช้ dropdown
body_th = body_tb.strip() if body_tb.strip() else body_dd
head_th = head_tb.strip() if head_tb.strip() else head_dd
arms_th = arms_tb.strip() if arms_tb.strip() else arms_dd
skin_th = skin_tb.strip() if skin_tb.strip() else skin_dd
ability_th = ability_tb.strip() if ability_tb.strip() else ability_dd
mood_th = mood_tb.strip() if mood_tb.strip() else mood_dd
desc_th = (
f"มอนสเตอร์อารมณ์{mood_th} มี{body_th} "
f"พร้อมด้วย{head_th}, {arms_th}, "
f"ผิว/ลายแบบ{skin_th}, และพลังพิเศษคือ{ability_th}!"
)
# แปลเป็นภาษาอังกฤษ โดยใช้การแปลล่วงหน้าสำหรับรายการคงที่
body_en = body_shapes_en.get(body_th, translate_th_to_en(body_th))
head_en = heads_en.get(head_th, translate_th_to_en(head_th))
arms_en = arms_legs_en.get(arms_th, translate_th_to_en(arms_th))
skin_en = skin_patterns_en.get(skin_th, translate_th_to_en(skin_th))
ability_en = abilities_en.get(ability_th, translate_th_to_en(ability_th))
mood_en = moods_en.get(mood_th, translate_th_to_en(mood_th))
prompt_en = (
f"This is a {mood_en} monster with {body_en}, "
f"a {head_en}, {arms_en}, covered in {skin_en} skin, "
f"and it can {ability_en}!"
)
return desc_th, prompt_en
def random_monster():
bd = random.choice(body_shapes_th)
hd = random.choice(heads_th)
ar = random.choice(arms_legs_th)
sk = random.choice(skin_patterns_th)
ab = random.choice(abilities_th)
md = random.choice(moods_th)
# Clear textbox
return (bd, "", hd, "", ar, "", sk, "", ab, "", md, "")
# ---- CSS ----
css_code = """
body {
background-color: #FFF7EA;
font-family: "Kanit", sans-serif;
}
#title {
color: #FF6F91;
text-align: center;
font-size: 2rem;
margin-top: 20px;
margin-bottom: 10px;
font-weight: bold;
}
.game-desc {
margin: 0 auto;
width: 80%;
background-color: #FFF2F0;
border: 2px dashed #FAB1A0;
border-radius: 10px;
padding: 15px;
color: #333;
margin-bottom: 20px;
}
/* ปุ่มหลัก (สร้างมอนสเตอร์) */
.btn-main {
background-color: #FFC107;
border: 2px solid #FFA000;
font-weight: bold;
font-size: 1.1rem;
/* กำหนดความสูงและความยาว (horizontal padding) เยอะๆ */
padding: 10px 40px;
border-radius: 10px;
margin-right: 10px;
}
/* ปุ่มสุ่ม (รอง) => ตัวเล็กลงในความยาว (horizontal padding) แต่ความสูงเท่ากัน */
.btn-random {
background-color: #FFE08E;
border: 2px solid #FFC107;
font-weight: normal;
font-size: 1.1rem;
/* ใช้ padding vertical เท่ากัน แต่ลด horizontal padding ลง */
padding: 10px 15px;
border-radius: 8px;
}
.btn-random:hover {
background-color: #FFF3C4;
}
#desc-th, #prompt-en {
background-color: #FFFAE6;
border: 2px solid #FFE082;
border-radius: 10px;
padding: 10px;
margin-bottom: 20px;
}
"""
def initial_text():
return "ยังไม่ได้สร้างมอนสเตอร์ ลองเลือกหรือพิมพ์ แล้วกด 'สร้างมอนสเตอร์!' หรือจะกดสุ่มดูก็ได้จ้า"
# ปุ่ม copy
copy_button_html = """
<button style="background-color: #F06292; border: 2px solid #E91E63; font-weight: bold;
font-size: 1.1rem; padding: 10px 20px; border-radius: 10px;"
onclick="copyPromptText()">
Copy Prompt
</button>
<script>
function copyPromptText() {
const promptBox = document.querySelector('#prompt-en textarea');
if (!promptBox) {
alert('ไม่พบข้อความ Prompt!');
return;
}
const promptText = promptBox.value;
navigator.clipboard.writeText(promptText);
alert('คัดลอก Prompt แล้ว!');
}
</script>
"""
with gr.Blocks(css=css_code) as demo:
gr.Markdown("<h1 id='title'>ZenityX Monster Lab</h1>")
gr.Markdown("""
<div class="game-desc">
<p>หนูน้อยจ๊ะ เลือกได้ว่าจะใช้ค่าใน <strong>Dropdown</strong> หรือจะ <strong>พิมพ์เอง</strong></p>
<p>หากอยากลุ้นโชค กดปุ่ม (ขวามือ) <strong>"สุ่มมอนสเตอร์สุดเซอร์ไพรส์!"</strong></p>
<p>เมื่อพร้อมแล้ว กดปุ่ม (ซ้าย) <strong>"สร้างมอนสเตอร์!"</strong> เพื่อดูรายละเอียดและ Prompt</p>
</div>
""")
with gr.Row():
with gr.Column():
body_dd = gr.Dropdown(body_shapes_th, label="รูปร่าง", value=body_shapes_th[0])
body_tb = gr.Textbox(label="หรือพิมพ์เอง", placeholder="ตัวโคลนยืดหยุ่น...")
head_dd = gr.Dropdown(heads_th, label="หัว/ใบหน้า", value=heads_th[0])
head_tb = gr.Textbox(label="หรือพิมพ์เอง", placeholder="หัวปลาหมึกสุดแปลก...")
with gr.Column():
arms_dd = gr.Dropdown(arms_legs_th, label="แขนขา", value=arms_legs_th[0])
arms_tb = gr.Textbox(label="หรือพิมพ์เอง", placeholder="แขนออกรากไม้...")
skin_dd = gr.Dropdown(skin_patterns_th, label="ผิว/ลวดลาย", value=skin_patterns_th[0])
skin_tb = gr.Textbox(label="หรือพิมพ์เอง", placeholder="ลายทางสีทอง...")
with gr.Column():
ability_dd = gr.Dropdown(abilities_th, label="พลังพิเศษ", value=abilities_th[0])
ability_tb = gr.Textbox(label="หรือพิมพ์เอง", placeholder="สร้างภาพลวงตา...")
mood_dd = gr.Dropdown(moods_th, label="อารมณ์", value=moods_th[0])
mood_tb = gr.Textbox(label="หรือพิมพ์เอง", placeholder="ขี้เล่นเป็นพิเศษ...")
with gr.Row():
# ปุ่มหลัก (สร้างมอนสเตอร์) -> อยู่ซ้าย
create_btn = gr.Button("สร้างมอนสเตอร์!", elem_classes="btn-main")
# ปุ่มรอง (สุ่ม) -> อยู่ขวา, ตัวสั้นลง
random_btn = gr.Button("สุ่มมอนสเตอร์สุดเซอร์ไพรส์!", elem_classes="btn-random")
monster_desc_th = gr.Textbox(label="รายละเอียด (ภาษาไทย)", interactive=False, elem_id="desc-th")
monster_prompt_en = gr.Textbox(label="Prompt (English)", interactive=False, elem_id="prompt-en")
gr.HTML(copy_button_html)
# ปุ่มสุ่ม
random_btn.click(
fn=random_monster,
inputs=[],
outputs=[
body_dd, body_tb,
head_dd, head_tb,
arms_dd, arms_tb,
skin_dd, skin_tb,
ability_dd, ability_tb,
mood_dd, mood_tb
]
)
# ปุ่มสร้าง
create_btn.click(
fn=generate_monster_lab,
inputs=[
body_dd, body_tb,
head_dd, head_tb,
arms_dd, arms_tb,
skin_dd, skin_tb,
ability_dd, ability_tb,
mood_dd, mood_tb
],
outputs=[monster_desc_th, monster_prompt_en]
)
demo.load(fn=initial_text, inputs=None, outputs=monster_desc_th)
demo.launch()