Spaces:
Paused
Paused
Delete app-backup6.py
Browse files- app-backup6.py +0 -887
app-backup6.py
DELETED
@@ -1,887 +0,0 @@
|
|
1 |
-
import os
|
2 |
-
import re
|
3 |
-
import random
|
4 |
-
import time
|
5 |
-
import html
|
6 |
-
import base64
|
7 |
-
import string
|
8 |
-
import json
|
9 |
-
import asyncio
|
10 |
-
import requests
|
11 |
-
import anthropic
|
12 |
-
import openai
|
13 |
-
|
14 |
-
from http import HTTPStatus
|
15 |
-
from typing import Dict, List, Optional, Tuple
|
16 |
-
from functools import partial
|
17 |
-
|
18 |
-
import gradio as gr
|
19 |
-
import modelscope_studio.components.base as ms
|
20 |
-
import modelscope_studio.components.legacy as legacy
|
21 |
-
import modelscope_studio.components.antd as antd
|
22 |
-
|
23 |
-
|
24 |
-
# ------------------------
|
25 |
-
# 1) DEMO_LIST 및 SystemPrompt
|
26 |
-
# ------------------------
|
27 |
-
|
28 |
-
DEMO_LIST = [
|
29 |
-
{"description": "블록이 위에서 떨어지는 클래식 테트리스 게임을 개발해주세요. 화살표 키로 조작하며, 가로줄이 채워지면 해당 줄이 제거되고 점수가 올라가는 메커니즘이 필요합니다. 난이도는 시간이 지날수록 블록이 빨라지도록 구현하고, 게임오버 조건과 점수 표시 기능을 포함해주세요."},
|
30 |
-
{"description": "두 명이 번갈아가며 플레이할 수 있는 체스 게임을 만들어주세요. 기본적인 체스 규칙(킹, 퀸, 룩, 비숍, 나이트, 폰의 이동 규칙)을 구현하고, 체크와 체크메이트 감지 기능이 필요합니다. 드래그 앤 드롭으로 말을 움직일 수 있게 하며, 이동 기록도 표시해주세요."},
|
31 |
-
{"description": "짝을 맞추는 메모리 카드 게임을 개발해주세요. 카드를 뒤집으면 그림이 나타나고, 같은 그림의 카드 두 장을 찾으면 점수를 얻는 방식입니다. 카드 뒤집기 애니메이션과 함께 시도 횟수를 기록하는 점수 시스템, 그리고 쉬움/보통/어려움 난이도 선택 기능(카드 수 변경)도 구현해주세요."},
|
32 |
-
{"description": "플레이어가 우주선을 조종하여 적 우주선을 파괴하는 슈팅 게임을 만들어주세요. 키보드 방향키로 움직이고 스페이스바로 발사하며, 다양한 적 웨이브가 공격해오는 구조입니다. 충돌 감지 시스템과 함께 파워업 아이템(방패, 다중 발사, 속도 증가 등)을 구현하고, 난이도가 점진적으로 증가하는 시스템을 추가해주세요."},
|
33 |
-
{"description": "3x3 또는 4x4 크기의 슬라이드 퍼즐 게임을 만들어주세요. 숫자나 이미지 조각을 섞은 후, 빈 칸을 이용해 조각들을 올바른 위치로 밀어 맞추는 게임입니다. 섞기 기능과 이동 횟수 카운터, 완성 시 축하 메시지를 표시하고, 난이도 설정(크기 변경)도 구현해주세요."},
|
34 |
-
{"description": "고전적인 뱀 게임을 구현해주세요. 플레이어는 방향키로 뱀을 조종하여 필드에 랜덤하게 생성되는 먹이를 먹으며, 먹이를 먹을 때마다 뱀의 길이가 늘어납니다. 자신의 몸에 부딪히거나 벽에 부딪히면 게임이 종료되며, 점수는 먹은 먹이의 수에 비례합니다. 시간이 지날수록 뱀의 이동 속도가 빨라지는 난이도 조절 기능도 추가해주세요."},
|
35 |
-
{"description": "화면 상단에 여러 줄의 벽돌이 배치된 브레이크아웃 게임을 만들어주세요. 플레이어는 화면 하단의 패들을 좌우로 움직여 공을 튕겨내어 벽돌을 깨야 합니다. 벽돌을 모두 깨면 스테이지 클리어, 공이 바닥에 떨어지면 생명이 감소합니다. 공의 속도는 시간이 지날수록 증가하며, 특수 벽돌(추가 생명, 패들 확장 등)도 구현해주세요."},
|
36 |
-
{"description": "길을 따라 이동하는 적들을 방어하는 타워 디펜스 게임을 개발해주세요. 플레이어는 맵의 특정 위치에 다양한 타워(기본 공격, 범위 공격, 감속 효과 등)를 설치하여 적을 물리쳐야 합니다. 웨이브 시스템으로 난이도가 점진적으로 증가하며, 적을 처치하면 자원을 얻어 타워를 업그레이드하거나 새 타워를 건설할 수 있는 경제 시스템을 구현해주세요."},
|
37 |
-
{"description": "캐릭터가 끝없이 달리며 장애물을 뛰어넘는 엔드리스 러너 게임을 만들어주세요. 스페이스바나 마우스 클릭으로 점프하여 다가오는 장애물(바위, 구덩이, 적 등)을 피해야 합니다. 거리에 따라 점수가 증가하며, 코인 등의 수집품을 모으는 요소와 파워업(일시적 무적, 자석 효과 등)도 추가해주세요. 시간이 지날수록 게임 속도가 빨라지는 난이도 시스템도 구현해주세요."},
|
38 |
-
{"description": "2D 플랫포머 게임을 개발해주세요. 플레이어는 방향키로 캐릭터를 조종하여 발판 위를 이동하고, 스페이스바로 점프하며 코인이나 보석 같은 아이템을 수집합니다. 적 캐릭터(간단한 AI로 움직임)와 함정(가시, 떨어지는 발판 등)을 피해 목표 지점까지 도달하는 레벨 기반 구조로 만들어주세요. 체력 시스템과 체크포인트 기능도 구현해주세요."},
|
39 |
-
{"description": "매번 새로운 미로를 자동 생성하는 미로 게임을 만들어주세요. 플���이어는 시작점에서 출발하여 방향키로 캐릭터를 조종해 출구를 찾아야 합니다. 미로 생성 알고리즘(예: 깊이 우선 탐색, 프림 알고리즘 등)을 활용하여 다양한 크기와 복잡도의 미로를 만들고, 타이머로 시간을 측정하며, 선택적으로 최단 경로를 보여주는 힌트 기능도 구현해주세요."},
|
40 |
-
{"description": "간단한 턴제 RPG 게임을 개발해주세요. 플레이어는 탑다운 뷰에서 타일 기반으로 이동하며, 몬스터와 마주치면 턴제 전투가 시작됩니다. 기본 공격, 특수 스킬, 아이템 사용 등의 전투 옵션과 함께 레벨업 시스템(경험치, 능력치 상승)을 구현해주세요. 또한 전투에서 승리하면 골드와 아이템을 획득할 수 있으며, 상점에서 장비를 구매하는 기능도 추가해주세요."},
|
41 |
-
{"description": "같은 색상/모양의 아이템 3개 이상을 일렬로 맞추는 매치-3 퍼즐 게임을 만들어주세요. 아이템을 스와이프하여 위치를 바꾸고, 매치되면 아이템이 사라지며 점수를 얻는 방식입니다. 특수 매치(4개 이상, T자 모양 등)는 특수 아이템을 생성하며, 연속 매치(콤보)는 추가 점수나 보너스 효과를 제공합니다. 목표 점수 또는 제한 시간/이동 횟수 모드를 구현해주세요."},
|
42 |
-
{"description": "플래피 버드 스타일의 게임을 개발해주세요. 플레이어는 스페이스바나 마우스 클릭으로 새를 점프시켜 위아래로 움직이는 파이프 사이를 통과해야 합니다. 파이프에 부딪히거나 화면 상단/하단에 닿으면 게임 오버이며, 통과한 파이프 쌍마다 점수가 1점씩 증가합니다. 파이프 간격은 랜덤하게 생성되며, 최고 점수를 로컬 스토리지에 저장하는 기능도 구현해주세요."},
|
43 |
-
{"description": "두 개의 유사한 이미지에서 차이점을 찾는 게임을 만들어주세요. 5-10개의 차이점이 있는 이미지 쌍을 준비하고, 플레이어가 차이점을 클릭하면 표시되도록 합니다. 제한 시간 내에 모든 차이점을 찾아야 하며, 오답 클릭 시 시간 패널티가 부과됩니다. 힌트 시스템(차이점 하나를 자동으로 표시)과 난이도 선택(쉬움: 차이점이 명확, 어려움: 미묘한 차이)도 구현해주세요."},
|
44 |
-
{"description": "화면 상단에서 단어가 떨어지는 타이핑 게임을 개발해주세요. 플레이어는 키보드로 해당 단어를 정확히 입력하여 단어가 바닥에 닿기 전에 제거해야 합니다. 정확히 입력한 단어는 사라지고 점수를 얻으며, 난이도에 따라 단어의 길이와 떨어지는 속도가 조절됩니다. 특수 단어(빨간색 등)는 보너스 점수나 시간 추가 등의 효과를 제공하며, 일정 시간/점수마다 난이도가 상승하는 시스템도 구현해주세요."},
|
45 |
-
{"description": "물리 엔진 기반의 미니 골프 게임을 만들어주세요. 플레이어는 마우스 드래그로 공을 치는 방향과 세기를 조절하여 홀에 공을 넣어야 합니다. 다양한 장애물(모래 함정, 물웅덩이, 경사로 등)이 있는 여러 개의 코스를 구현하고, 각 홀마다 타수를 기록하여 최종 점수를 계산합니다. 바람 방향/세기 같은 환경 요소와 함께 궤적 미리보기 기능도 추가해주세요."},
|
46 |
-
{"description": "플레이어가 낚시를 즐기는 시뮬레이션 게임을 개발해주세요. 마우스 클릭으로 낚싯줄을 던지고, 물고기가 물면 타이밍 맞추기 미니게임으로 물고기를 낚아야 합니다. 다양한 종류의 물고기(희귀도별 점수 차등)를 구현하고, 낚은 물고기에 따라 골드를 획득하여 더 좋은 낚싯대, 미끼 등을 구매할 수 있는 업그레이드 시스템을 추가해주세요. 시간대나 날씨에 따라 출현하는 물고기가 달라지는 기능도 구현해주세요."},
|
47 |
-
{"description": "1인용 또는 AI 대전 빙고 게임을 만들어주세요. 5x5 그리드에 1-25 숫자를 무작위로 배치하고, 번갈아가며 숫자를 선택하여 해당 칸을 마킹합니다. 가로, 세로, 대각선으로 5개의 연속된 마킹이 완성되면 빙고가 되며, 먼저 3빙고를 달성하는 쪽이 승리합니다. 컴퓨터 AI는 랜덤하게 또는 전략적으로(빙고에 가까운 라인 우선) 숫자를 선택하도록 구현하고, 타이머와 승/패 기록 시스템도 추가해주세요."},
|
48 |
-
{"description": "화면 하단에서 상단으로 노트가 올라오면 정확한 타이밍에 키를 눌러 점수를 얻는 리듬 게임을 개발해주세요. 4개의 레인(D, F, J, K 키)에 노트가 등장하며, 타이밍 정확도에 따라 Perfect, Good, Miss 등급이 표시됩니다. 배경 음악에 맞춰 노트가 생성되며, 연속 성공 시 콤보 시스템으로 추가 점수를 제공합니다. 난이도 선택(노트 속도와 밀도 조절)과 함께 최종 결과 화면(정확도, 콤보, 등급)도 구현해주세요."},
|
49 |
-
{"description": "탑다운 뷰의 2D 레이싱 게임을 만들어주세요. 플레이어는 방향키로 자동차를 조종하여 트랙을 따라 주행하며, 트랙 이탈 시 감속되는 메커니즘을 구현합니다. 여러 AI 경쟁자들과 경쟁하며 3바퀴를 가장 빨리 완주하는 게임 모드와 함께, 시간 제한 내에 체크포인트를 통과하는 타임 어택 모드도 구현해주세요. 다양한 차량 선택지(속도와 핸들링 특성 차등)와 부스트 아이템, 장애물 등도 추가해주세요."},
|
50 |
-
{"description": "다양한 카테고리의 퀴즈를 풀어나가는 게임을 개발해주세요. 주어진 질문에 4개의 보기 중 정답을 선택하는 방식으로, 정답 시 점수를 획득하고 오답 시 생명이 감소합니다. 30초 제한 시간 내에 답을 선택해야 하며, 난이도에 따라 질문의 복잡도와 제한 시간이 조절됩니다. 50:50 힌트(오답 2개 제거), 시간 추가 등의 도움 아이템과 함께 최종 결과 요약(정답률, 카테고리별 성적)도 구현해주세요."},
|
51 |
-
{"description": "움직이는 표적을 맞추는 사격 갤러리 게임을 만들어주세요. 마우스 클릭으로 발사하며, 다양한 속도와 패턴으로 움직이는 표적(오리, 병, 풍선 등)을 맞추면 점수를 획득합니다. 제한된 시간과 총알 수 안에 최대한 많은 점수를 얻는 것이 목표이며, 특수 표적(황금 표적 등)은 보너스 점수나 추가 시간/총알을 제공합니다. 연속 명중 시 점수 배율이 증가하는 콤보 시스템과 함께 다양한 난이도 레벨(표적 속도/수 증가)도 구현해주세요."},
|
52 |
-
{"description": "가상 주사위를 굴려 보드판을 돌아다니는 보드 게임을 개발해주세요. 플레이어는 차례대로 1-6 주사위를 굴려 말을 이동시키며, 도착한 칸에 따라 다양한 이벤트(앞으로/뒤로 이동, 한 턴 쉬기, 미니게임 등)가 발생합니다. 특수 아이템(추가 주사위, 이벤트 회피 등)을 수집하고 사용할 수 있으며, 먼저 결승점에 도달하거나 가장 많은 포인트를 모은 플레이어가 승리합니다. 1-4명의 로컬 멀티플레이어를 지원하며, AI 플레이어도 구현해주세요."},
|
53 |
-
{"description": "탑다운 뷰의 좀비 서바이벌 게임을 만들어주세요. WASD로 이동하고 마우스로 조준/발사하며, 끊임없이 몰려오는 좀비 웨이브를 최대한 오래 생존하는 것이 목표입니다. 다양한 무기(권총, 샷건, 기관총 등)와 제한된 탄약, 그리고 체력 회복 아이템과 폭탄 같은 특수 아이템을 맵에서 획득할 수 있습니다. 시간이 지날수록 좀비의 수와 속도가 증가하며, 특수 좀비(탱커, 러너 등)도 등장하는 난이도 시스템을 구현해주세요."},
|
54 |
-
{"description": "축구 페널티킥 게임을 개발해주세요. 공격 시에는 방향과 파워를 조절하여 슛을 날리고, 수비 시에는 골키퍼를 좌/중앙/우 중 한 방향으로 다이빙시켜 공을 막아야 합니다. 5번의 키커-골키퍼 대결 후 더 많은 골을 넣은 쪽이 승리하며, 동점일 경우 서든데스로 승부를 가립니다. 슛의 정확도와 파워에 따라 결과가 달라지며, 골키퍼 AI는 패턴 학습을 통해 플레이어의 경향성을 파악하도록 구현해주세요. 1인 플레이와 2인 로컬 대전 모드를 모두 지원해주세요."},
|
55 |
-
{"description": "클래식한 지뢰찾기 게임을 구현해주세요. NxN 크기의 그리드에 M개의 지뢰가 무작위로 배치되며, 플레이어는 좌클릭으로 칸을 열고 우클릭으로 지뢰 위치에 깃발을 표시합니다. 열린 칸에는 주변 8칸의 지뢰 수가 표시되며, 주변에 지뢰가 없는 칸을 열면 연쇄적으로 주변 칸들이 열립니다. 지뢰가 있는 칸을 열면 게임 오버, 지뢰가 아닌 모든 칸을 열면 승리입니다. 난이도 설정(쉬움: 9x9/10개, 중간: 16x16/40개, 어려움: 30x16/99개)과 함께 첫 클릭은 항상 안전하도록 구현해주세요."},
|
56 |
-
{"description": "두 플레이어가 번갈아가며 7x6 그리드에 색깔 디스크를 떨어뜨려 가로, 세로, 대각선으로 4개의 연속된 디스크를 만드는 Connect Four 게임을 개발해주세요. 플레이어는 열을 클릭하여 디스크를 해당 열의 가장 아래 빈 칸에 배치합니다. 4개의 연속된 디스크를 먼저 만드는 플레이어가 승리하며, 모든 칸이 차면 무승부입니다. 1인 플레이(AI 대전)과 2인 로컬 대전 모드를 구현하고, AI는 최소한 1단계 앞을 내다보는 논리로 작동하도록 해주세요."},
|
57 |
-
{"description": "글자 타일을 배치하여 단어를 만드는 스크래블 스타일의 단어 게임을 만들어주세요. 각 플레이어는 7개의 글자 타일을 받고, 이를 보드에 배치하여 가로나 세로로 단어를 형성합니다. 새 단어는 기존 단어와 반드시 연결되어야 하며, 각 타일에는 점수가 있어 단어의 총점이 계산됩니다. 특수 칸(2배 글자 점수, 3배 단어 점수 등)을 활용한 전략적 배치가 가능하며, 사전 검증 기능으로 유효한 단어만 허용합니다. 1-4인 로컬 멀티플레이어와 AI 대전을 지원해주세요."},
|
58 |
-
{"description": "2D 환경에서 진행되는 탱크 전투 게임을 개발해주세요. 플레이어는 WASD로 탱크를 조종하고, 마우스로 포탑을 조준하여 클릭으로 발사합니다. 파괴 가능한 지형(벽돌, 나무 등)과 파괴 불가능한 장애물(강철, 물 등)이 있는 맵에서 적 탱크들과 전투를 벌입니다. 다양한 무기(기본 포탄, 확산탄, 레이저 등)와 아이템(속도 증가, 방어력 강화, 추가 생명 등)을 구현하고, 스테이지별로 증가하는 적 AI 난이도와 보스 전투도 추가해주세요."},
|
59 |
-
{"description": "3개 이상의 같은 보석을 맞추어 제거하는 퍼즐 게임을 만들어주세요. 인접한 두 보석을 스왑하여 매치를 만들며, 매치된 보석이 사라지면 위의 보석들이 떨어지고 새 보석이 채워집니다. 4개 이상 매치 시 특수 보석(가로/세로 폭발, 주변 9칸 폭발 등)이 생성되며, 연쇄 매치가 발생하면 콤보 점수가 추가됩니다. 제한 시간 또는 제한 이동 횟수 내에 목표 점수를 달성하는 레벨 기반 진행 구조와 함께, 특수 미션(특정 색상 N개 제거, 장애물 파괴 등)도 구현해주세요."},
|
60 |
-
{"description": "단일 타워가 끊임없이 몰려오는 적들을 격퇴하는 타워 디펜스 게임을 개발해주세요. 화면 중앙의 타워는 자동으로 가장 가까운 적을 향해 발사하며, 플레이어는 웨이브 사이에 획득한 자원으로 타워를 업그레이드(공격력, 공격 속도, 범위 등)할 수 있습니다. 시간이 지날수록 더 강력하고 다양한 적(빠른 적, 방어력 높은 적, 분열하는 적 등)이 등장하며, 타워의 체력이 0이 되면 게임 오버입니다. 특수 능력(범위 공격, 일시 정지, 즉시 회복 등)과 함께 생존한 웨이브 수에 따른 랭킹 시스템도 구현해주세요."},
|
61 |
-
{"description": "캐릭터가 끝없이 달리며 좀비와 장애물을 피하는 사이드 스크롤링 러너 게임을 만들어주세요. 스페이스바로 점프, S키로 슬라이딩하여 다양한 장애물(웅덩이, 장벽, 좀비 무리 등)을 피해야 합니다. 코인과 파워업(일시적 무적, 자석 효과, 속도 감소 등)을 수집하며, 특정 구간마다 미니 보스 좀비와의 간단한 전투도 포함됩니다. 거리에 따라 점수가 증가하고, 코인으로 캐릭터 업그레이드(더블 점프, 체력 증가 등)를 구매할 수 있는 시스템도 구현해주세요."},
|
62 |
-
{"description": "탑다운 뷰의 간단한 액션 RPG 게임을 개발해주세요. WASD로 이동하고, 마우스 클릭으로 기본 공격, 1-4 키로 특수 스킬을 사용합니다. 플레이어는 몬스터를 처치하며 경험치와 아이템을 획득하고, 레벨업 시 능력치(공격력, 체력, 속도 등)를 향상시킵니다. 다양한 무기와 방어구를 착용할 수 있으며, 스킬 트리 시스템으로 캐릭터를 특화시킬 수 있습니다. 여러 지역과 보스 몬스터, 간단한 퀘스트 시스템도 구현해주세요."},
|
63 |
-
]
|
64 |
-
|
65 |
-
SystemPrompt = """
|
66 |
-
# GameCraft 시스템 프롬프트
|
67 |
-
|
68 |
-
## 1. 기본 정보 및 역할
|
69 |
-
당신의 이름은 'GameCraft'입니다. 당신은 게임플레이 메커니즘, 인터랙티브 디자인, 성능 최적화에 뛰어난 웹 게임 개발 전문가입니다. HTML, JavaScript, CSS를 활용하여 간결하고 효율적인 웹 기반 게임을 제작하는 것이 당신의 임무입니다.
|
70 |
-
|
71 |
-
## 2. 핵심 기술 스택
|
72 |
-
- **프론트엔드**: HTML5, CSS3, JavaScript(ES6+)
|
73 |
-
- **렌더링 방식**: 브라우저에서 직접 렌더링 가능한 코드 생성
|
74 |
-
- **코드 스타일**: 바닐라 JavaScript 우선, 외부 라이브러리 최소화
|
75 |
-
|
76 |
-
## 3. 게임 유형별 특화 지침
|
77 |
-
### 3.1 아케이드/액션 게임
|
78 |
-
- 간결한 충돌 감지 시스템 구현
|
79 |
-
- 키보드/터치 입력 최적화
|
80 |
-
- 기본적인 점수 시스템
|
81 |
-
|
82 |
-
### 3.2 퍼즐 게임
|
83 |
-
- 명확한 게임 규칙 및 승리 조건
|
84 |
-
- 기본 난이도 구현
|
85 |
-
- 핵심 게임 메커니즘에 집중
|
86 |
-
|
87 |
-
### 3.3 카드/보드 게임
|
88 |
-
- 간소화된 턴 기반 시스템
|
89 |
-
- 기본 게임 규칙 자동화
|
90 |
-
- 핵심 게임 로직 중심
|
91 |
-
|
92 |
-
### 3.4 시뮬레이션 게임
|
93 |
-
- 효율적인 상태 관리
|
94 |
-
- 가장 중요한 상호작용 구현
|
95 |
-
- 핵심 요소만 포함
|
96 |
-
|
97 |
-
## 4. 이모지 활용 지침 🎮
|
98 |
-
- 게임 UI 요소에 이모지 활용 (예: 생명력 ❤️, 코인 💰, 시간 ⏱️)
|
99 |
-
- 이모지 사용은 핵심 요소에만 집중
|
100 |
-
|
101 |
-
## 5. 기술적 구현 가이드라인
|
102 |
-
### 5.1 코드 구조
|
103 |
-
- **간결성 중시**: 코드는 최대한 간결하게 작성하고, 주석은 최소화
|
104 |
-
- **모듈화**: 코드 기능별로 분리하되 불필요한 추상화 지양
|
105 |
-
- **최적화**: 게임 루프와 렌더링 최적화에 집중
|
106 |
-
- **코드 크기 제한**: 전체 코드는 200줄을 넘지 않도록 함
|
107 |
-
|
108 |
-
### 5.2 성능 최적화
|
109 |
-
- DOM 조작 최소화
|
110 |
-
- 불필요한 변수와 함수 제거
|
111 |
-
- 메모리 관리에 주의
|
112 |
-
|
113 |
-
### 5.3 반응형 디자인
|
114 |
-
- 기본적인 반응형 지원
|
115 |
-
- 핵심 기능에 집중한 심플한 UI
|
116 |
-
|
117 |
-
## 6. 외부 라이브러리
|
118 |
-
- 라이브러리 사용은 최소화하고, 필요한 경우에만 사용
|
119 |
-
- 라이브러리 사용 시 CDN으로 가져올 것
|
120 |
-
|
121 |
-
## 7. 접근성 및 포용성
|
122 |
-
- 핵심 접근성 기능에만 집중
|
123 |
-
|
124 |
-
## 8. 제약사항 및 유의사항
|
125 |
-
- 외부 API 호출 금지
|
126 |
-
- 코드 크기 최소화에 우선순위 (200줄 이내)
|
127 |
-
- 주석 최소화 - 필수적인 설명만 포함
|
128 |
-
- 불필요한 기능 구현 지양
|
129 |
-
|
130 |
-
## 9. 출력 형식
|
131 |
-
- HTML 코드 블록으로만 코드 반환
|
132 |
-
- 추가 설명 없이 즉시 실행 가능한 코드만 제공
|
133 |
-
- 모든 코드는 단일 HTML 파일에 인라인으로 포함
|
134 |
-
|
135 |
-
## 10. 코드 품질 기준
|
136 |
-
- 효율성과 간결함이 최우선
|
137 |
-
- 핵심 게임플레이 메커니즘에만 집중
|
138 |
-
- 복잡한 기능보다 작동하는 기본 기능 우선
|
139 |
-
- 불필요한 주석이나 장황한 코드 지양
|
140 |
-
- 단일 파일에 모든 코드 포함
|
141 |
-
- 코드 길이 제한: 완성된 게임 코드는 200줄 이내로 작성
|
142 |
-
|
143 |
-
## 11. 중요: 코드 생성 제한
|
144 |
-
- 게임 코드는 반드시 200줄 이내로 제한
|
145 |
-
- 불필요한 설명이나 주석 제외
|
146 |
-
- 핵심 기능만 구현하고 부가 기능은 생략
|
147 |
-
- 코드 크기가 커질 경우 기능을 간소화하거나 생략할 것
|
148 |
-
"""
|
149 |
-
|
150 |
-
|
151 |
-
# ------------------------
|
152 |
-
# 2) 공통 상수, 함수, 클래스
|
153 |
-
# ------------------------
|
154 |
-
|
155 |
-
class Role:
|
156 |
-
SYSTEM = "system"
|
157 |
-
USER = "user"
|
158 |
-
ASSISTANT = "assistant"
|
159 |
-
|
160 |
-
History = List[Tuple[str, str]]
|
161 |
-
Messages = List[Dict[str, str]]
|
162 |
-
|
163 |
-
IMAGE_CACHE = {}
|
164 |
-
|
165 |
-
def get_image_base64(image_path):
|
166 |
-
if image_path in IMAGE_CACHE:
|
167 |
-
return IMAGE_CACHE[image_path]
|
168 |
-
try:
|
169 |
-
with open(image_path, "rb") as image_file:
|
170 |
-
encoded_string = base64.b64encode(image_file.read()).decode()
|
171 |
-
IMAGE_CACHE[image_path] = encoded_string
|
172 |
-
return encoded_string
|
173 |
-
except:
|
174 |
-
return IMAGE_CACHE.get('default.png', '')
|
175 |
-
|
176 |
-
def history_to_messages(history: History, system: str) -> Messages:
|
177 |
-
messages = [{'role': Role.SYSTEM, 'content': system}]
|
178 |
-
for h in history:
|
179 |
-
messages.append({'role': Role.USER, 'content': h[0]})
|
180 |
-
messages.append({'role': Role.ASSISTANT, 'content': h[1]})
|
181 |
-
return messages
|
182 |
-
|
183 |
-
def messages_to_history(messages: Messages) -> History:
|
184 |
-
assert messages[0]['role'] == Role.SYSTEM
|
185 |
-
history = []
|
186 |
-
for q, r in zip(messages[1::2], messages[2::2]):
|
187 |
-
history.append([q['content'], r['content']])
|
188 |
-
return history
|
189 |
-
|
190 |
-
|
191 |
-
# ------------------------
|
192 |
-
# 3) API 연동 설정
|
193 |
-
# ------------------------
|
194 |
-
|
195 |
-
YOUR_ANTHROPIC_TOKEN = os.getenv('ANTHROPIC_API_KEY', '').strip()
|
196 |
-
YOUR_OPENAI_TOKEN = os.getenv('OPENAI_API_KEY', '').strip()
|
197 |
-
|
198 |
-
claude_client = anthropic.Anthropic(api_key=YOUR_ANTHROPIC_TOKEN)
|
199 |
-
openai_client = openai.OpenAI(api_key=YOUR_OPENAI_TOKEN)
|
200 |
-
|
201 |
-
async def try_claude_api(system_message, claude_messages, timeout=15):
|
202 |
-
try:
|
203 |
-
system_message_with_limit = system_message + "\n\n추가 중요 지침: 생성하는 코드는 절대로 200줄을 넘지 마세요..."
|
204 |
-
start_time = time.time()
|
205 |
-
with claude_client.messages.stream(
|
206 |
-
model="claude-3-7-sonnet-20250219",
|
207 |
-
max_tokens=19800,
|
208 |
-
system=system_message_with_limit,
|
209 |
-
messages=claude_messages,
|
210 |
-
temperature=0.3
|
211 |
-
) as stream:
|
212 |
-
collected_content = ""
|
213 |
-
for chunk in stream:
|
214 |
-
current_time = time.time()
|
215 |
-
if current_time - start_time > timeout:
|
216 |
-
raise TimeoutError("Claude API timeout")
|
217 |
-
if chunk.type == "content_block_delta":
|
218 |
-
collected_content += chunk.delta.text
|
219 |
-
yield collected_content
|
220 |
-
await asyncio.sleep(0)
|
221 |
-
start_time = current_time
|
222 |
-
except Exception as e:
|
223 |
-
raise e
|
224 |
-
|
225 |
-
async def try_openai_api(openai_messages):
|
226 |
-
try:
|
227 |
-
if openai_messages and openai_messages[0]["role"] == "system":
|
228 |
-
openai_messages[0]["content"] += "\n\n추가 중요 지침: 생성하는 코드는 절대로 200줄을 넘지 마세요..."
|
229 |
-
|
230 |
-
stream = openai_client.chat.completions.create(
|
231 |
-
model="o3",
|
232 |
-
messages=openai_messages,
|
233 |
-
stream=True,
|
234 |
-
max_tokens=19800,
|
235 |
-
temperature=0.2
|
236 |
-
)
|
237 |
-
collected_content = ""
|
238 |
-
for chunk in stream:
|
239 |
-
if chunk.choices[0].delta.content is not None:
|
240 |
-
collected_content += chunk.choices[0].delta.content
|
241 |
-
yield collected_content
|
242 |
-
except Exception as e:
|
243 |
-
raise e
|
244 |
-
|
245 |
-
|
246 |
-
# ------------------------
|
247 |
-
# 4) 템플릿(하나로 통합)
|
248 |
-
# ------------------------
|
249 |
-
|
250 |
-
def load_json_data():
|
251 |
-
data_list = []
|
252 |
-
for i, demo in enumerate(DEMO_LIST):
|
253 |
-
game_type = ""
|
254 |
-
desc = demo["description"]
|
255 |
-
|
256 |
-
# 게임 타입 추정
|
257 |
-
if "테트리스" in desc or "블록" in desc:
|
258 |
-
game_type = "[퍼즐] 테트리스"
|
259 |
-
elif "체스" in desc:
|
260 |
-
game_type = "[보드] 체스"
|
261 |
-
elif "메모리 카드" in desc or "짝을 맞추는" in desc:
|
262 |
-
game_type = "[카드] 메모리 게임"
|
263 |
-
elif "슈팅" in desc or "우주선" in desc:
|
264 |
-
game_type = "[액션] 슈팅 게임"
|
265 |
-
elif "슬라이드 퍼즐" in desc:
|
266 |
-
game_type = "[퍼즐] 슬라이드 퍼즐"
|
267 |
-
elif "뱀 게임" in desc:
|
268 |
-
game_type = "[아케이드] 스네이크"
|
269 |
-
elif "브레이크아웃" in desc or "벽돌" in desc:
|
270 |
-
game_type = "[아케이드] 브레이크아웃"
|
271 |
-
elif "타워 디펜스" in desc:
|
272 |
-
game_type = "[전략] 타워 디펜스"
|
273 |
-
elif "엔드리스 러너" in desc or "달리며 장애물" in desc:
|
274 |
-
game_type = "[액션] 러너 게임"
|
275 |
-
elif "플랫포머" in desc:
|
276 |
-
game_type = "[액션] 플랫포머"
|
277 |
-
elif "미로" in desc:
|
278 |
-
game_type = "[퍼즐] 미로 게임"
|
279 |
-
elif "RPG" in desc or "턴제" in desc:
|
280 |
-
game_type = "[RPG] 턴제 전투"
|
281 |
-
elif "매치-3" in desc or "같은 색상" in desc:
|
282 |
-
game_type = "[퍼즐] 매치3 게임"
|
283 |
-
elif "플래피 버드" in desc:
|
284 |
-
game_type = "[아케이드] 플래피 버드"
|
285 |
-
elif "차이점" in desc:
|
286 |
-
game_type = "[퍼즐] 차이 찾기"
|
287 |
-
elif "타이핑" in desc:
|
288 |
-
game_type = "[교육] 타이핑 게임"
|
289 |
-
elif "골프" in desc:
|
290 |
-
game_type = "[스포츠] 미니 골프"
|
291 |
-
elif "낚시" in desc:
|
292 |
-
game_type = "[시뮬레이션] 낚시"
|
293 |
-
elif "빙고" in desc:
|
294 |
-
game_type = "[보드] 빙고"
|
295 |
-
elif "리듬" in desc:
|
296 |
-
game_type = "[음악] 리듬 게임"
|
297 |
-
elif "레이싱" in desc:
|
298 |
-
game_type = "[레이싱] 탑다운 드라이빙"
|
299 |
-
elif "퀴즈" in desc:
|
300 |
-
game_type = "[교육] 퀴즈"
|
301 |
-
elif "사격" in desc:
|
302 |
-
game_type = "[아케이드] 사격 갤러리"
|
303 |
-
elif "보드 게임" in desc or "주사위" in desc:
|
304 |
-
game_type = "[보드] 룰렛 게임"
|
305 |
-
elif "좀비" in desc and "서바이벌" in desc:
|
306 |
-
game_type = "[액션] 좀비 서바이벌"
|
307 |
-
elif "페널티킥" in desc or "축구" in desc:
|
308 |
-
game_type = "[스포츠] 페널티킥"
|
309 |
-
elif "지뢰찾기" in desc:
|
310 |
-
game_type = "[퍼즐] 지뢰찾기"
|
311 |
-
elif "Connect Four" in desc:
|
312 |
-
game_type = "[보드] 사목게임"
|
313 |
-
elif "스크래블" in desc or "단어" in desc:
|
314 |
-
game_type = "[보드] 단어 게임"
|
315 |
-
elif "탱크" in desc:
|
316 |
-
game_type = "[액션] 탱크 배틀"
|
317 |
-
elif "보석" in desc:
|
318 |
-
game_type = "[퍼즐] 보석 매치"
|
319 |
-
else:
|
320 |
-
game_type = f"[게임] 예제 {i+1}"
|
321 |
-
|
322 |
-
# 설명 요약 (너무 길면 축약)
|
323 |
-
short_desc = desc[:150] + "..." if len(desc) > 150 else desc
|
324 |
-
|
325 |
-
data_list.append({
|
326 |
-
"name": game_type,
|
327 |
-
"prompt": desc
|
328 |
-
})
|
329 |
-
|
330 |
-
return data_list
|
331 |
-
|
332 |
-
def create_template_html(title, items):
|
333 |
-
import html as html_lib
|
334 |
-
html_content = r"""
|
335 |
-
<style>
|
336 |
-
.template-title {
|
337 |
-
font-size: 1.1rem;
|
338 |
-
font-weight: 600;
|
339 |
-
margin-bottom: 16px;
|
340 |
-
color: #0066cc;
|
341 |
-
text-align: center;
|
342 |
-
}
|
343 |
-
.prompt-grid {
|
344 |
-
display: grid;
|
345 |
-
grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
|
346 |
-
gap: 16px;
|
347 |
-
padding: 12px;
|
348 |
-
}
|
349 |
-
.prompt-card {
|
350 |
-
background: white;
|
351 |
-
border: 1px solid #e9ecef;
|
352 |
-
border-radius: 12px;
|
353 |
-
padding: 15px;
|
354 |
-
cursor: pointer;
|
355 |
-
box-shadow: 0 4px 6px rgba(0,0,0,0.05);
|
356 |
-
transition: all 0.3s ease;
|
357 |
-
height: 180px;
|
358 |
-
display: flex;
|
359 |
-
flex-direction: column;
|
360 |
-
}
|
361 |
-
.prompt-card:hover {
|
362 |
-
transform: translateY(-4px);
|
363 |
-
box-shadow: 0 8px 15px rgba(0,0,0,0.1);
|
364 |
-
border-color: #66a3ff;
|
365 |
-
}
|
366 |
-
.card-name {
|
367 |
-
font-weight: 600;
|
368 |
-
margin-bottom: 10px;
|
369 |
-
font-size: 14px;
|
370 |
-
color: #2c3e50;
|
371 |
-
padding: 5px 10px;
|
372 |
-
background: #f7f9fc;
|
373 |
-
border-radius: 6px;
|
374 |
-
display: inline-block;
|
375 |
-
}
|
376 |
-
.card-prompt {
|
377 |
-
font-size: 12px;
|
378 |
-
line-height: 1.5;
|
379 |
-
color: #546e7a;
|
380 |
-
display: -webkit-box;
|
381 |
-
-webkit-line-clamp: 6;
|
382 |
-
-webkit-box-orient: vertical;
|
383 |
-
overflow: hidden;
|
384 |
-
flex-grow: 1;
|
385 |
-
}
|
386 |
-
</style>
|
387 |
-
<div class="template-title">🎮 다양한 게임 템플릿을 선택해 보세요</div>
|
388 |
-
<div class="prompt-grid">
|
389 |
-
"""
|
390 |
-
for item in items:
|
391 |
-
card_html = f"""
|
392 |
-
<div class="prompt-card" onclick="copyToInput(this)" data-prompt="{html_lib.escape(item.get('prompt', ''))}">
|
393 |
-
<div class="card-name">{html_lib.escape(item.get('name', ''))}</div>
|
394 |
-
<div class="card-prompt">{html_lib.escape(item.get('prompt', ''))}</div>
|
395 |
-
</div>
|
396 |
-
"""
|
397 |
-
html_content += card_html
|
398 |
-
html_content += r"""
|
399 |
-
<script>
|
400 |
-
function copyToInput(card) {
|
401 |
-
const prompt = card.dataset.prompt;
|
402 |
-
const textarea = document.querySelector('.ant-input-textarea-large textarea');
|
403 |
-
if (textarea) {
|
404 |
-
textarea.value = prompt;
|
405 |
-
textarea.dispatchEvent(new Event('input', { bubbles: true }));
|
406 |
-
document.querySelector('.session-drawer .close-btn').click();
|
407 |
-
}
|
408 |
-
}
|
409 |
-
</script>
|
410 |
-
</div>
|
411 |
-
"""
|
412 |
-
return gr.HTML(value=html_content)
|
413 |
-
|
414 |
-
def load_all_templates():
|
415 |
-
return create_template_html("🎮 모든 게임 템플릿", load_json_data())
|
416 |
-
|
417 |
-
# ------------------------
|
418 |
-
# 5) 배포/부스트/기타 유틸
|
419 |
-
# ------------------------
|
420 |
-
|
421 |
-
def remove_code_block(text):
|
422 |
-
pattern = r'```html\s*([\s\S]+?)\s*```'
|
423 |
-
match = re.search(pattern, text, re.DOTALL)
|
424 |
-
if match:
|
425 |
-
return match.group(1).strip()
|
426 |
-
|
427 |
-
pattern = r'```(?:\w+)?\s*([\s\S]+?)\s*```'
|
428 |
-
match = re.search(pattern, text, re.DOTALL)
|
429 |
-
if match:
|
430 |
-
return match.group(1).strip()
|
431 |
-
|
432 |
-
text = re.sub(r'```html\s*', '', text)
|
433 |
-
text = re.sub(r'\s*```', '', text)
|
434 |
-
return text.strip()
|
435 |
-
|
436 |
-
def optimize_code(code: str) -> str:
|
437 |
-
if not code or len(code.strip()) == 0:
|
438 |
-
return code
|
439 |
-
lines = code.split('\n')
|
440 |
-
if len(lines) <= 200:
|
441 |
-
return code
|
442 |
-
|
443 |
-
comment_patterns = [
|
444 |
-
r'/\*[\s\S]*?\*/',
|
445 |
-
r'//.*?$',
|
446 |
-
r'<!--[\s\S]*?-->'
|
447 |
-
]
|
448 |
-
cleaned_code = code
|
449 |
-
for pattern in comment_patterns:
|
450 |
-
cleaned_code = re.sub(pattern, '', cleaned_code, flags=re.MULTILINE)
|
451 |
-
|
452 |
-
cleaned_lines = []
|
453 |
-
empty_line_count = 0
|
454 |
-
for line in cleaned_code.split('\n'):
|
455 |
-
if line.strip() == '':
|
456 |
-
empty_line_count += 1
|
457 |
-
if empty_line_count <= 1:
|
458 |
-
cleaned_lines.append('')
|
459 |
-
else:
|
460 |
-
empty_line_count = 0
|
461 |
-
cleaned_lines.append(line)
|
462 |
-
cleaned_code = '\n'.join(cleaned_lines)
|
463 |
-
cleaned_code = re.sub(r'console\.log\(.*?\);', '', cleaned_code, flags=re.MULTILINE)
|
464 |
-
cleaned_code = re.sub(r' {2,}', ' ', cleaned_code)
|
465 |
-
return cleaned_code
|
466 |
-
|
467 |
-
def send_to_sandbox(code):
|
468 |
-
clean_code = remove_code_block(code)
|
469 |
-
clean_code = optimize_code(clean_code)
|
470 |
-
if clean_code.startswith('```html'):
|
471 |
-
clean_code = clean_code[7:].strip()
|
472 |
-
if clean_code.endswith('```'):
|
473 |
-
clean_code = clean_code[:-3].strip()
|
474 |
-
|
475 |
-
if not clean_code.strip().startswith('<!DOCTYPE') and not clean_code.strip().startswith('<html'):
|
476 |
-
clean_code = f"""<!DOCTYPE html>
|
477 |
-
<html>
|
478 |
-
<head>
|
479 |
-
<meta charset="UTF-8">
|
480 |
-
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
481 |
-
<title>Game Preview</title>
|
482 |
-
</head>
|
483 |
-
<body>
|
484 |
-
{clean_code}
|
485 |
-
</body>
|
486 |
-
</html>"""
|
487 |
-
encoded_html = base64.b64encode(clean_code.encode('utf-8')).decode('utf-8')
|
488 |
-
data_uri = f"data:text/html;charset=utf-8;base64,{encoded_html}"
|
489 |
-
return f'<iframe src="{data_uri}" width="100%" height="920px" style="border:none;"></iframe>'
|
490 |
-
|
491 |
-
def boost_prompt(prompt: str) -> str:
|
492 |
-
if not prompt:
|
493 |
-
return ""
|
494 |
-
# (생략)
|
495 |
-
# 이 부분은 동일
|
496 |
-
|
497 |
-
def handle_boost(prompt: str):
|
498 |
-
try:
|
499 |
-
boosted_prompt = boost_prompt(prompt)
|
500 |
-
return boosted_prompt, gr.update(active_key="empty")
|
501 |
-
except Exception:
|
502 |
-
return prompt, gr.update(active_key="empty")
|
503 |
-
|
504 |
-
def history_render(history: History):
|
505 |
-
return gr.update(open=True), history
|
506 |
-
|
507 |
-
def execute_code(query: str):
|
508 |
-
if not query or query.strip() == '':
|
509 |
-
return None, gr.update(active_key="empty")
|
510 |
-
try:
|
511 |
-
clean_code = remove_code_block(query)
|
512 |
-
if clean_code.startswith('```html'):
|
513 |
-
clean_code = clean_code[7:].strip()
|
514 |
-
if clean_code.endswith('```'):
|
515 |
-
clean_code = clean_code[:-3].strip()
|
516 |
-
if not clean_code.strip().startswith('<!DOCTYPE') and not clean_code.strip().startswith('<html'):
|
517 |
-
if not ('<body' in clean_code and '</body>' in clean_code):
|
518 |
-
clean_code = f"""<!DOCTYPE html>
|
519 |
-
<html>
|
520 |
-
<head>
|
521 |
-
<meta charset="UTF-8">
|
522 |
-
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
523 |
-
<title>Game Preview</title>
|
524 |
-
</head>
|
525 |
-
<body>
|
526 |
-
{clean_code}
|
527 |
-
</body>
|
528 |
-
</html>"""
|
529 |
-
return send_to_sandbox(clean_code), gr.update(active_key="render")
|
530 |
-
except Exception as e:
|
531 |
-
print(f"Execute code error: {str(e)}")
|
532 |
-
return None, gr.update(active_key="empty")
|
533 |
-
|
534 |
-
|
535 |
-
# ------------------------
|
536 |
-
# 6) 데모 클래스
|
537 |
-
# ------------------------
|
538 |
-
|
539 |
-
class Demo:
|
540 |
-
def __init__(self):
|
541 |
-
pass
|
542 |
-
|
543 |
-
async def generation_code(self, query: Optional[str], _setting: Dict[str, str], _history: Optional[History]):
|
544 |
-
if not query or query.strip() == '':
|
545 |
-
query = random.choice(DEMO_LIST)['description']
|
546 |
-
|
547 |
-
if _history is None:
|
548 |
-
_history = []
|
549 |
-
|
550 |
-
query = f"""
|
551 |
-
다음 게임을 제작해주세요.
|
552 |
-
중요 요구사항:
|
553 |
-
1. 코드는 가능한 한 간결하게 작성할 것
|
554 |
-
2. 불필요한 주석이나 설명은 제외할 것
|
555 |
-
3. 코드는 600줄을 넘지 않을 것
|
556 |
-
4. 모든 코드는 하나의 HTML 파일에 통합할 것
|
557 |
-
5. 핵심 기능만 구현하고 부가 기능은 생략할 것
|
558 |
-
게임 요청: {query}
|
559 |
-
"""
|
560 |
-
messages = history_to_messages(_history, _setting['system'])
|
561 |
-
system_message = messages[0]['content']
|
562 |
-
|
563 |
-
claude_messages = [
|
564 |
-
{"role": msg["role"] if msg["role"] != "system" else "user", "content": msg["content"]}
|
565 |
-
for msg in messages[1:] + [{'role': Role.USER, 'content': query}]
|
566 |
-
if msg["content"].strip() != ''
|
567 |
-
]
|
568 |
-
|
569 |
-
openai_messages = [{"role": "system", "content": system_message}]
|
570 |
-
for msg in messages[1:]:
|
571 |
-
openai_messages.append({"role": msg["role"], "content": msg["content"]})
|
572 |
-
openai_messages.append({"role": "user", "content": query})
|
573 |
-
|
574 |
-
try:
|
575 |
-
yield [
|
576 |
-
"Generating code...",
|
577 |
-
_history,
|
578 |
-
None,
|
579 |
-
gr.update(active_key="loading"),
|
580 |
-
gr.update(open=True)
|
581 |
-
]
|
582 |
-
await asyncio.sleep(0)
|
583 |
-
collected_content = None
|
584 |
-
try:
|
585 |
-
async for content in try_claude_api(system_message, claude_messages):
|
586 |
-
yield [
|
587 |
-
content,
|
588 |
-
_history,
|
589 |
-
None,
|
590 |
-
gr.update(active_key="loading"),
|
591 |
-
gr.update(open=True)
|
592 |
-
]
|
593 |
-
await asyncio.sleep(0)
|
594 |
-
collected_content = content
|
595 |
-
except Exception:
|
596 |
-
async for content in try_openai_api(openai_messages):
|
597 |
-
yield [
|
598 |
-
content,
|
599 |
-
_history,
|
600 |
-
None,
|
601 |
-
gr.update(active_key="loading"),
|
602 |
-
gr.update(open=True)
|
603 |
-
]
|
604 |
-
await asyncio.sleep(0)
|
605 |
-
collected_content = content
|
606 |
-
|
607 |
-
if collected_content:
|
608 |
-
clean_code = remove_code_block(collected_content)
|
609 |
-
code_lines = clean_code.count('\n') + 1
|
610 |
-
if code_lines > 700:
|
611 |
-
warning_msg = f"""
|
612 |
-
⚠️ **경고: 생성된 코드가 너무 깁니다 ({code_lines}줄)** ...
|
613 |
-
"""
|
614 |
-
yield [warning_msg, _history, None, gr.update(active_key="empty"), gr.update(open=True)]
|
615 |
-
else:
|
616 |
-
_history = messages_to_history([
|
617 |
-
{'role': Role.SYSTEM, 'content': system_message}
|
618 |
-
] + claude_messages + [{
|
619 |
-
'role': Role.ASSISTANT, 'content': collected_content
|
620 |
-
}])
|
621 |
-
yield [
|
622 |
-
collected_content,
|
623 |
-
_history,
|
624 |
-
send_to_sandbox(clean_code),
|
625 |
-
gr.update(active_key="render"),
|
626 |
-
gr.update(open=True)
|
627 |
-
]
|
628 |
-
else:
|
629 |
-
raise ValueError("No content was generated from either API")
|
630 |
-
except Exception as e:
|
631 |
-
raise ValueError(f'Error calling APIs: {str(e)}')
|
632 |
-
|
633 |
-
def clear_history(self):
|
634 |
-
return []
|
635 |
-
|
636 |
-
|
637 |
-
# ------------------------
|
638 |
-
# 7) Gradio / Modelscope UI 빌드
|
639 |
-
# ------------------------
|
640 |
-
|
641 |
-
demo_instance = Demo()
|
642 |
-
theme = gr.themes.Soft(
|
643 |
-
primary_hue="blue",
|
644 |
-
secondary_hue="purple",
|
645 |
-
neutral_hue="slate",
|
646 |
-
spacing_size=gr.themes.sizes.spacing_md,
|
647 |
-
radius_size=gr.themes.sizes.radius_md,
|
648 |
-
text_size=gr.themes.sizes.text_md,
|
649 |
-
)
|
650 |
-
|
651 |
-
with gr.Blocks(css_paths=["app.css"], theme=theme) as demo:
|
652 |
-
# 헤더 영역 (스크립트 제거 - 순수 HTML만 남김)
|
653 |
-
header_html = gr.HTML("""
|
654 |
-
<div class="app-header">
|
655 |
-
<h1>🎮 Vibe Game Craft</h1>
|
656 |
-
<p>설명을 입력하면 웹 기반 HTML5, JavaScript, CSS 게임을 생성합니다. 직관적인 인터페이스로 쉽게 게임을 만들고, 실시간으로 미리보기를 확인하세요.</p>
|
657 |
-
</div>
|
658 |
-
""")
|
659 |
-
|
660 |
-
history = gr.State([])
|
661 |
-
setting = gr.State({"system": SystemPrompt})
|
662 |
-
|
663 |
-
with ms.Application() as app:
|
664 |
-
with antd.ConfigProvider():
|
665 |
-
|
666 |
-
# code Drawer
|
667 |
-
with antd.Drawer(open=False, title="코드 보기", placement="left", width="750px") as code_drawer:
|
668 |
-
code_output = legacy.Markdown()
|
669 |
-
|
670 |
-
# history Drawer
|
671 |
-
with antd.Drawer(open=False, title="히스토리", placement="left", width="900px") as history_drawer:
|
672 |
-
history_output = legacy.Chatbot(
|
673 |
-
show_label=False, flushing=False, height=960, elem_classes="history_chatbot"
|
674 |
-
)
|
675 |
-
|
676 |
-
# templates Drawer
|
677 |
-
with antd.Drawer(
|
678 |
-
open=False,
|
679 |
-
title="게임 템플릿",
|
680 |
-
placement="right",
|
681 |
-
width="900px",
|
682 |
-
elem_classes="session-drawer"
|
683 |
-
) as session_drawer:
|
684 |
-
with antd.Flex(vertical=True, gap="middle"):
|
685 |
-
gr.Markdown("### 사용 가능한 게임 템플릿")
|
686 |
-
session_history = gr.HTML(elem_classes="session-history")
|
687 |
-
close_btn = antd.Button("닫기", type="default", elem_classes="close-btn")
|
688 |
-
|
689 |
-
with antd.Row(gutter=[32, 12], align="top", elem_classes="equal-height-container") as layout:
|
690 |
-
|
691 |
-
# 좌측 미리보기
|
692 |
-
with antd.Col(span=24, md=16, elem_classes="equal-height-col"):
|
693 |
-
# style 속성 대신 CSS 클래스만 사용
|
694 |
-
with ms.Div(elem_classes="right_panel panel fixed-height"):
|
695 |
-
gr.HTML(r"""
|
696 |
-
<div class="render_header">
|
697 |
-
<span class="header_btn"></span>
|
698 |
-
<span class="header_btn"></span>
|
699 |
-
<span class="header_btn"></span>
|
700 |
-
</div>
|
701 |
-
""")
|
702 |
-
with antd.Tabs(active_key="empty", render_tab_bar="() => null") as state_tab:
|
703 |
-
with antd.Tabs.Item(key="empty"):
|
704 |
-
empty = antd.Empty(description="게임을 만들려면 설명을 입력하세요", elem_classes="right_content fixed-content-height")
|
705 |
-
with antd.Tabs.Item(key="loading"):
|
706 |
-
loading = antd.Spin(
|
707 |
-
True, tip="게임 코드 생성 중...", size="large", elem_classes="right_content fixed-content-height"
|
708 |
-
)
|
709 |
-
with antd.Tabs.Item(key="render"):
|
710 |
-
sandbox = gr.HTML(elem_classes="html_content fixed-content-height")
|
711 |
-
|
712 |
-
# 우측 입력/버튼/배포
|
713 |
-
with antd.Col(span=24, md=8, elem_classes="equal-height-col"):
|
714 |
-
with antd.Flex(vertical=True, gap="small", elem_classes="right-top-buttons"):
|
715 |
-
with antd.Flex(gap="small", elem_classes="setting-buttons", justify="space-between"):
|
716 |
-
codeBtn = antd.Button("🧑💻 코드 보기", type="default", elem_classes="code-btn")
|
717 |
-
historyBtn = antd.Button("📜 히스토리", type="default", elem_classes="history-btn")
|
718 |
-
template_btn = antd.Button("🎮 템플릿", type="default", elem_classes="template-btn")
|
719 |
-
|
720 |
-
with antd.Flex(gap="small", justify="space-between", elem_classes="action-buttons"):
|
721 |
-
btn = antd.Button("전송", type="primary", size="large", elem_classes="send-btn")
|
722 |
-
boost_btn = antd.Button("증강", type="default", size="large", elem_classes="boost-btn")
|
723 |
-
execute_btn = antd.Button("코드", type="default", size="large", elem_classes="execute-btn")
|
724 |
-
deploy_btn = antd.Button("배포", type="default", size="large", elem_classes="deploy-btn")
|
725 |
-
clear_btn = antd.Button("클리어", type="default", size="large", elem_classes="clear-btn")
|
726 |
-
|
727 |
-
# style 속성 대신 CSS 클래스 사용
|
728 |
-
with antd.Flex(vertical=True, gap="middle", wrap=True, elem_classes="input-panel fixed-content-height"):
|
729 |
-
# 입력창의 높이를 CSS로 조정
|
730 |
-
input_text = antd.InputTextarea(
|
731 |
-
size="large",
|
732 |
-
allow_clear=True,
|
733 |
-
placeholder=random.choice(DEMO_LIST)['description'],
|
734 |
-
max_length=100000,
|
735 |
-
elem_classes="fixed-input-height"
|
736 |
-
)
|
737 |
-
gr.HTML('<div class="help-text">💡 원하는 게임의 설명을 입력하세요. 예: "테트리스 게임 제작해줘."</div>')
|
738 |
-
|
739 |
-
# 가장 기본적인 방식의 배포 결과 표시
|
740 |
-
deploy_result = gr.Markdown("", label="배포 결과", elem_classes="deploy-result", elem_id="deploy-result")
|
741 |
-
|
742 |
-
# 더 이상 자바스크립트 삽입 없음
|
743 |
-
js_trigger = gr.HTML(elem_id="js-trigger", visible=False)
|
744 |
-
|
745 |
-
# ─── 버튼 / Drawer 이벤트 ───
|
746 |
-
codeBtn.click(
|
747 |
-
lambda: gr.update(open=True),
|
748 |
-
inputs=[],
|
749 |
-
outputs=[code_drawer]
|
750 |
-
)
|
751 |
-
code_drawer.close(
|
752 |
-
lambda: gr.update(open=False),
|
753 |
-
inputs=[],
|
754 |
-
outputs=[code_drawer]
|
755 |
-
)
|
756 |
-
|
757 |
-
historyBtn.click(
|
758 |
-
history_render,
|
759 |
-
inputs=[history],
|
760 |
-
outputs=[history_drawer, history_output]
|
761 |
-
)
|
762 |
-
history_drawer.close(
|
763 |
-
lambda: gr.update(open=False),
|
764 |
-
inputs=[],
|
765 |
-
outputs=[history_drawer]
|
766 |
-
)
|
767 |
-
|
768 |
-
template_btn.click(
|
769 |
-
fn=lambda: (gr.update(open=True), load_all_templates()),
|
770 |
-
outputs=[session_drawer, session_history],
|
771 |
-
queue=False
|
772 |
-
)
|
773 |
-
session_drawer.close(
|
774 |
-
lambda: (gr.update(open=False), gr.HTML("")),
|
775 |
-
outputs=[session_drawer, session_history]
|
776 |
-
)
|
777 |
-
close_btn.click(
|
778 |
-
lambda: (gr.update(open=False), gr.HTML("")),
|
779 |
-
outputs=[session_drawer, session_history]
|
780 |
-
)
|
781 |
-
|
782 |
-
btn.click(
|
783 |
-
demo_instance.generation_code,
|
784 |
-
inputs=[input_text, setting, history],
|
785 |
-
outputs=[code_output, history, sandbox, state_tab, code_drawer]
|
786 |
-
)
|
787 |
-
|
788 |
-
clear_btn.click(
|
789 |
-
demo_instance.clear_history,
|
790 |
-
inputs=[],
|
791 |
-
outputs=[history]
|
792 |
-
)
|
793 |
-
|
794 |
-
boost_btn.click(
|
795 |
-
fn=handle_boost,
|
796 |
-
inputs=[input_text],
|
797 |
-
outputs=[input_text, state_tab]
|
798 |
-
)
|
799 |
-
|
800 |
-
execute_btn.click(
|
801 |
-
fn=execute_code,
|
802 |
-
inputs=[input_text],
|
803 |
-
outputs=[sandbox, state_tab]
|
804 |
-
)
|
805 |
-
|
806 |
-
# 매우 단순화된 배포 함수
|
807 |
-
def simple_deploy(code):
|
808 |
-
if not code or code.strip() == "":
|
809 |
-
return "⚠️ **배포 실패**: 배포할 코드가 없습니다. 먼저 게임을 생성해주세요."
|
810 |
-
|
811 |
-
try:
|
812 |
-
clean_code = remove_code_block(code)
|
813 |
-
|
814 |
-
token = "A8IFZmgW2cqA4yUNlLPnci0N"
|
815 |
-
if not token:
|
816 |
-
return "⚠️ **배포 실패**: Vercel 토큰이 설정되지 않았습니다."
|
817 |
-
|
818 |
-
project_name = ''.join(random.choice(string.ascii_lowercase) for i in range(6))
|
819 |
-
deploy_url = "https://api.vercel.com/v13/deployments"
|
820 |
-
headers = {
|
821 |
-
"Authorization": f"Bearer {token}",
|
822 |
-
"Content-Type": "application/json"
|
823 |
-
}
|
824 |
-
package_json = {
|
825 |
-
"name": project_name,
|
826 |
-
"version": "1.0.0",
|
827 |
-
"private": True,
|
828 |
-
"dependencies": {"vite": "^5.0.0"},
|
829 |
-
"scripts": {
|
830 |
-
"dev": "vite",
|
831 |
-
"build": "echo 'No build needed' && mkdir -p dist && cp index.html dist/",
|
832 |
-
"preview": "vite preview"
|
833 |
-
}
|
834 |
-
}
|
835 |
-
files = [
|
836 |
-
{"file": "index.html", "data": clean_code},
|
837 |
-
{"file": "package.json", "data": json.dumps(package_json, indent=2)}
|
838 |
-
]
|
839 |
-
project_settings = {
|
840 |
-
"buildCommand": "npm run build",
|
841 |
-
"outputDirectory": "dist",
|
842 |
-
"installCommand": "npm install",
|
843 |
-
"framework": None
|
844 |
-
}
|
845 |
-
deploy_data = {
|
846 |
-
"name": project_name,
|
847 |
-
"files": files,
|
848 |
-
"target": "production",
|
849 |
-
"projectSettings": project_settings
|
850 |
-
}
|
851 |
-
|
852 |
-
deploy_response = requests.post(deploy_url, headers=headers, json=deploy_data)
|
853 |
-
|
854 |
-
if deploy_response.status_code != 200:
|
855 |
-
return f"⚠️ **배포 실패**: API 오류가 발생했습니다. (상태 코드: {deploy_response.status_code})"
|
856 |
-
|
857 |
-
time.sleep(5) # 배포 완료 대기
|
858 |
-
|
859 |
-
deployment_url = f"https://{project_name}.vercel.app"
|
860 |
-
return f"""✅ **배포 완료!**
|
861 |
-
|
862 |
-
게임은 다음 주소에서 확인할 수 있습니다:
|
863 |
-
[{deployment_url}]({deployment_url})
|
864 |
-
|
865 |
-
이 링크를 클릭하여 게임을 플레이하세요."""
|
866 |
-
|
867 |
-
except Exception as e:
|
868 |
-
return f"⚠️ **배포 실패**: {str(e)}"
|
869 |
-
|
870 |
-
# 배포 버튼 이벤트 - 가장 단순한 방식
|
871 |
-
deploy_btn.click(
|
872 |
-
fn=simple_deploy,
|
873 |
-
inputs=[code_output],
|
874 |
-
outputs=[deploy_result]
|
875 |
-
)
|
876 |
-
|
877 |
-
# ------------------------
|
878 |
-
# 8) 실제 실행
|
879 |
-
# ------------------------
|
880 |
-
|
881 |
-
if __name__ == "__main__":
|
882 |
-
try:
|
883 |
-
demo_instance = Demo()
|
884 |
-
demo.queue(default_concurrency_limit=20).launch(ssr_mode=False)
|
885 |
-
except Exception as e:
|
886 |
-
print(f"Initialization error: {e}")
|
887 |
-
raise
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|