Spaces:
Runtime error
Runtime error
Update main.py
Browse files
main.py
CHANGED
|
@@ -84,6 +84,16 @@ def get_gemini_response(prompt: str, images: list):
|
|
| 84 |
else:
|
| 85 |
return f"Gemini API 請求失敗:{response.status_code}, {response.text}"
|
| 86 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 87 |
@line_handler.add(MessageEvent, message=TextMessage)
|
| 88 |
def handle_text_message(event):
|
| 89 |
user_id = event.source.user_id
|
|
@@ -142,14 +152,14 @@ def handle_text_message(event):
|
|
| 142 |
user_states[user_id] = defaultdict(lambda: {"upper_body_images": [], "lower_body_images": [], "current_mode": None, "is_ready_for_outfit": False, "is_ready_for_photo": False, "user_info": {}, "personal_photo": None})[user_id]
|
| 143 |
line_bot_api.reply_message(
|
| 144 |
reply_token,
|
| 145 |
-
TextSendMessage(text="
|
| 146 |
)
|
| 147 |
return
|
| 148 |
|
| 149 |
# 如果沒有進入任何模式,給予提示
|
| 150 |
line_bot_api.reply_message(
|
| 151 |
reply_token,
|
| 152 |
-
TextSendMessage(text="
|
| 153 |
)
|
| 154 |
|
| 155 |
@line_handler.add(MessageEvent, message=ImageMessage)
|
|
@@ -161,12 +171,12 @@ def handle_image_message(event):
|
|
| 161 |
if not user_states[user_id]["is_ready_for_outfit"]:
|
| 162 |
line_bot_api.reply_message(
|
| 163 |
reply_token,
|
| 164 |
-
TextSendMessage(text="
|
| 165 |
)
|
| 166 |
return
|
| 167 |
|
| 168 |
-
#
|
| 169 |
-
if not user_states[user_id]["personal_photo"]:
|
| 170 |
try:
|
| 171 |
image_id = event.message.id
|
| 172 |
base64_img = get_base64_image(image_id)
|
|
@@ -178,22 +188,30 @@ def handle_image_message(event):
|
|
| 178 |
TextSendMessage(text="已收到您的個人照片,正在為您準備穿搭... 請稍候。")
|
| 179 |
)
|
| 180 |
|
| 181 |
-
#
|
| 182 |
user_info = user_states[user_id]["user_info"]
|
| 183 |
occasion = user_info["occasion"]
|
| 184 |
|
| 185 |
prompt = (
|
| 186 |
-
f"
|
| 187 |
"請根據這些衣物,為我推薦一套最適合的穿搭,並詳細說明為何這套搭配適合這個場合。請以繁體中文回答。"
|
| 188 |
-
"
|
| 189 |
)
|
| 190 |
|
| 191 |
all_images = user_states[user_id]["upper_body_images"] + user_states[user_id]["lower_body_images"] + [user_states[user_id]["personal_photo"]]
|
| 192 |
|
| 193 |
response_text = get_gemini_response(prompt, all_images)
|
| 194 |
|
| 195 |
-
#
|
| 196 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 197 |
|
| 198 |
# 發送 Gemini 的文字建議
|
| 199 |
line_bot_api.push_message(
|
|
@@ -261,7 +279,7 @@ def handle_image_message(event):
|
|
| 261 |
else:
|
| 262 |
line_bot_api.reply_message(
|
| 263 |
reply_token,
|
| 264 |
-
TextSendMessage(text="
|
| 265 |
)
|
| 266 |
|
| 267 |
except Exception as e:
|
|
|
|
| 84 |
else:
|
| 85 |
return f"Gemini API 請求失敗:{response.status_code}, {response.text}"
|
| 86 |
|
| 87 |
+
def get_virtual_tryon_image(user_photo, upper_body_image, lower_body_image):
|
| 88 |
+
"""
|
| 89 |
+
模擬虛擬試穿 API,返回一個試穿後的圖片 URL。
|
| 90 |
+
在實際應用中,這裡會呼叫一個運行 IDM-VTON 的獨立服務。
|
| 91 |
+
"""
|
| 92 |
+
# 這裡的邏輯需要您自行實現,可以是呼叫一個公開的 API,
|
| 93 |
+
# 或者將圖片傳輸到一個運行 IDM-VTON 的伺服器。
|
| 94 |
+
# 由於沒有實際的 IDM-VTON 服務,我們使用佔位符圖片來展示功能。
|
| 95 |
+
return "https://placehold.co/1024x1024?text=Virtual+Try-On+Outfit"
|
| 96 |
+
|
| 97 |
@line_handler.add(MessageEvent, message=TextMessage)
|
| 98 |
def handle_text_message(event):
|
| 99 |
user_id = event.source.user_id
|
|
|
|
| 152 |
user_states[user_id] = defaultdict(lambda: {"upper_body_images": [], "lower_body_images": [], "current_mode": None, "is_ready_for_outfit": False, "is_ready_for_photo": False, "user_info": {}, "personal_photo": None})[user_id]
|
| 153 |
line_bot_api.reply_message(
|
| 154 |
reply_token,
|
| 155 |
+
TextSendMessage(text="狀態已重置。請重新輸入個人資訊,格式為:身高,胸圍,腰圍,臀圍,場合")
|
| 156 |
)
|
| 157 |
return
|
| 158 |
|
| 159 |
# 如果沒有進入任何模式,給予提示
|
| 160 |
line_bot_api.reply_message(
|
| 161 |
reply_token,
|
| 162 |
+
TextSendMessage(text="請先依照格式輸入個人資訊,格式為:身高,胸圍,腰圍,臀圍,場合。")
|
| 163 |
)
|
| 164 |
|
| 165 |
@line_handler.add(MessageEvent, message=ImageMessage)
|
|
|
|
| 171 |
if not user_states[user_id]["is_ready_for_outfit"]:
|
| 172 |
line_bot_api.reply_message(
|
| 173 |
reply_token,
|
| 174 |
+
TextSendMessage(text="請先輸入個人資訊,格式為:身高,胸圍,腰圍,臀圍,場合。")
|
| 175 |
)
|
| 176 |
return
|
| 177 |
|
| 178 |
+
# 處理個人照片上傳
|
| 179 |
+
if len(user_states[user_id]["upper_body_images"]) == MAX_IMAGES_PER_TYPE and len(user_states[user_id]["lower_body_images"]) == MAX_IMAGES_PER_TYPE and not user_states[user_id]["personal_photo"]:
|
| 180 |
try:
|
| 181 |
image_id = event.message.id
|
| 182 |
base64_img = get_base64_image(image_id)
|
|
|
|
| 188 |
TextSendMessage(text="已收到您的個人照片,正在為您準備穿搭... 請稍候。")
|
| 189 |
)
|
| 190 |
|
| 191 |
+
# 準備 Gemini 提示
|
| 192 |
user_info = user_states[user_id]["user_info"]
|
| 193 |
occasion = user_info["occasion"]
|
| 194 |
|
| 195 |
prompt = (
|
| 196 |
+
f"我提供了三件上衣圖片和三件下半身圖片,以及使用者的一張個人照片。使用者資訊:身高 {user_info['height']}cm,三圍 {user_info['bust']}-{user_info['waist']}-{user_info['hip']},場合是「{occasion}」。"
|
| 197 |
"請根據這些衣物,為我推薦一套最適合的穿搭,並詳細說明為何這套搭配適合這個場合。請以繁體中文回答。"
|
| 198 |
+
"請在描述中以圖片連結的形式(例如:'https://example.com/outfit1.jpg')顯示你搭配好的衣物組合。"
|
| 199 |
)
|
| 200 |
|
| 201 |
all_images = user_states[user_id]["upper_body_images"] + user_states[user_id]["lower_body_images"] + [user_states[user_id]["personal_photo"]]
|
| 202 |
|
| 203 |
response_text = get_gemini_response(prompt, all_images)
|
| 204 |
|
| 205 |
+
# 從 Gemini 建議中找出最佳搭配的衣物圖片(這裡需要更複雜的邏輯,我們使用佔位符)
|
| 206 |
+
# 在實際應用中,您可以設計一個提示,讓 Gemini 返回最佳搭配的圖片索引
|
| 207 |
+
best_upper_index = 0
|
| 208 |
+
best_lower_index = 0
|
| 209 |
+
|
| 210 |
+
best_upper_body_image = user_states[user_id]["upper_body_images"][best_upper_index]
|
| 211 |
+
best_lower_body_image = user_states[user_id]["lower_body_images"][best_lower_index]
|
| 212 |
+
|
| 213 |
+
# 呼叫虛擬試穿 API,並取得結果圖片 URL
|
| 214 |
+
virtual_try_on_url = get_virtual_tryon_image(user_states[user_id]["personal_photo"], best_upper_body_image, best_lower_body_image)
|
| 215 |
|
| 216 |
# 發送 Gemini 的文字建議
|
| 217 |
line_bot_api.push_message(
|
|
|
|
| 279 |
else:
|
| 280 |
line_bot_api.reply_message(
|
| 281 |
reply_token,
|
| 282 |
+
TextSendMessage(text="已收到所有衣物圖片!接下來,請上傳一張您個人的全身照片,以便進行虛擬試穿。")
|
| 283 |
)
|
| 284 |
|
| 285 |
except Exception as e:
|