Hamed744 commited on
Commit
8cd1eaf
·
verified ·
1 Parent(s): c4ad66a

Update app.py

Browse files
Files changed (1) hide show
  1. app.py +109 -56
app.py CHANGED
@@ -7,10 +7,6 @@ import threading
7
  import os
8
 
9
  # --- دیکشنری زبان‌ها و صداها با کلیدهای فارسی (نمونه) ---
10
- # توجه: این فقط یک نمونه کوچک است. شما باید کل دیکشنری را به این شکل فارسی کنید.
11
- # برای سادگی، من فقط چند مورد اول را تغییر می دهم و بقیه را انگلیسی نگه می دارم.
12
- # شما باید برای هر کلید، نام زبان و جنسیت را به فارسی ترجمه کنید.
13
-
14
  language_dict_persian_keys = {
15
  'انگلیسی - جنی (زن)': 'en-US-JennyNeural',
16
  'انگلیسی - گای (مرد)': 'en-US-GuyNeural',
@@ -21,7 +17,7 @@ language_dict_persian_keys = {
21
  'انگلیسی - میشل (زن)': 'en-US-MichelleNeural',
22
  'انگلیسی - راجر (مرد)': 'en-US-RogerNeural',
23
  'اسپانیایی (مکزیک) - دالیا (زن)': 'es-MX-DaliaNeural',
24
- 'اسپانیایی (مکزیک) - خورخه (مرد)': 'es-MX-JorgeNeural', # نام خورخه ممکن است دقیق نباشد
25
  'کره‌ای - سان-هی (زن)': 'ko-KR-SunHiNeural',
26
  'کره‌ای - این‌جون (مرد)': 'ko-KR-InJoonNeural',
27
  'تایلندی - پرموادی (زن)': 'th-TH-PremwadeeNeural',
@@ -43,13 +39,13 @@ language_dict_persian_keys = {
43
  'هلندی - کولت (زن)': 'nl-NL-ColetteNeural',
44
  'هلندی - فنا (زن)': 'nl-NL-FennaNeural',
45
  'هلندی - مارتن (مرد)': 'nl-NL-MaartenNeural',
46
- 'مالایی - عثمان (مرد)': 'ms-MY-OsmanNeural', # "Malese" به "مالایی"
47
  'مالایی - یاسمین (زن)': 'ms-MY-YasminNeural',
48
  'نروژی - پرنیل (زن)': 'nb-NO-PernilleNeural',
49
  'نروژی - فین (مرد)': 'nb-NO-FinnNeural',
50
  'سوئدی - سوفی (زن)': 'sv-SE-SofieNeural',
51
  'سوئدی - ماتیاس (مرد)': 'sv-SE-MattiasNeural',
52
- 'عربی (عربستان) - حامد (مرد)': 'ar-SA-HamedNeural', # "عربی" به "عربی (عربستان)"
53
  'عربی (عربستان) - زاریا (زن)': 'ar-SA-ZariyahNeural',
54
  'یونانی - آتنا (زن)': 'el-GR-AthinaNeural',
55
  'یونانی - نستوراس (مرد)': 'el-GR-NestorasNeural',
@@ -57,9 +53,9 @@ language_dict_persian_keys = {
57
  'آلمانی - آمالا (زن)': 'de-DE-AmalaNeural',
58
  'آلمانی - کنراد (مرد)': 'de-DE-ConradNeural',
59
  'آلمانی - کیلیان (مرد)': 'de-DE-KillianNeural',
60
- 'آفریقایی - آدری (زن)': 'af-ZA-AdriNeural', # "Afrikaans" به "آفریقایی"
61
  'آفریقایی - ویلم (مرد)': 'af-ZA-WillemNeural',
62
- 'اتیوپیایی - آمه‌ها (مرد)': 'am-ET-AmehaNeural', # "Ethiopian" به "اتیوپیایی"
63
  'اتیوپیایی - مکدس (زن)': 'am-ET-MekdesNeural',
64
  'عربی (امارات) - فاطمه (زن)': 'ar-AE-FatimaNeural',
65
  'عربی (امارات) - حمدان (مرد)': 'ar-AE-HamdanNeural',
@@ -69,9 +65,9 @@ language_dict_persian_keys = {
69
  'عربی (مصر) - سلما (زن)': 'ar-EG-SalmaNeural',
70
  'عربی (مصر) - شاکر (مرد)': 'ar-EG-ShakirNeural',
71
  'عربی (عراق) - باسل (مرد)': 'ar-IQ-BasselNeural',
72
- 'عربی (عراق) - رعنا (زن)': 'ar-IQ-RanaNeural', # "Rana" به "رعنا"
73
  'عربی (اردن) - سانا (زن)': 'ar-JO-SanaNeural',
74
- 'عربی (اردن) - تایم (مرد)': 'ar-JO-TaimNeural', # "Taim"
75
  'عربی (کویت) - فهد (مرد)': 'ar-KW-FahedNeural',
76
  'عربی (کویت) - نورا (زن)': 'ar-KW-NouraNeural',
77
  'عربی (لبنان) - لیلا (زن)': 'ar-LB-LaylaNeural',
@@ -82,11 +78,11 @@ language_dict_persian_keys = {
82
  'عربی (مراکش) - مونا (زن)': 'ar-MA-MounaNeural',
83
  'عربی (عمان) - عبدالله (مرد)': 'ar-OM-AbdullahNeural',
84
  'عربی (عمان) - عایشه (زن)': 'ar-OM-AyshaNeural',
85
- 'عربی (قطر) - امل (زن)': 'ar-QA-AmalNeural', # "Amal"
86
  'عربی (قطر) - معاذ (مرد)': 'ar-QA-MoazNeural',
87
  'عربی (سوریه) - امانی (زن)': 'ar-SY-AmanyNeural',
88
  'عربی (سوریه) - لیث (مرد)': 'ar-SY-LaithNeural',
89
- 'عربی (تونس) - هادی (مرد)': 'ar-TN-HediNeural', # "Hedi"
90
  'عربی (تونس) - ریم (زن)': 'ar-TN-ReemNeural',
91
  'عربی (یمن) - مریم (زن)': 'ar-YE-MaryamNeural',
92
  'عربی (یمن) - صالح (مرد)': 'ar-YE-SalehNeural',
@@ -97,14 +93,14 @@ language_dict_persian_keys = {
97
  'بنگالی (بنگلادش) - نابانیتا (زن)': 'bn-BD-NabanitaNeural',
98
  'بنگالی (بنگلادش) - پرادیپ (مرد)': 'bn-BD-PradeepNeural',
99
  'بنگالی (هند) - باشکار (مرد)': 'bn-IN-BashkarNeural',
100
- 'بنگالی (هند) - تانیشا (زن)': 'bn-IN-TanishaaNeural', # "Tanishaa"
101
- 'بوسنیایی - گوران (مرد)': 'bs-BA-GoranNeural', # "Bosnian" به "بوسنیایی"
102
  'بوسنیایی - وسنا (زن)': 'bs-BA-VesnaNeural',
103
- 'کاتالان (اسپانیا) - جوآنا (زن)': 'ca-ES-JoanaNeural', # "Catalan"
104
  'کاتالان (اسپانیا) - انریک (مرد)': 'ca-ES-EnricNeural',
105
- 'چکی - آنتونین (مرد)': 'cs-CZ-AntoninNeural', # "Czech" به "چکی"
106
  'چکی - ولاستا (زن)': 'cs-CZ-VlastaNeural',
107
- 'ولزی (بریتانیا) - آلد (مرد)': 'cy-GB-AledNeural', # "Welsh"
108
  'ولزی (بریتانیا) - نیا (زن)': 'cy-GB-NiaNeural',
109
  'دانمارکی - کریستل (زن)': 'da-DK-ChristelNeural',
110
  'دانمارکی - یپه (مرد)': 'da-DK-JeppeNeural',
@@ -175,13 +171,13 @@ language_dict_persian_keys = {
175
  'اسپانیایی (پاراگوئه) - تانیا (زن)': 'es-PY-TaniaNeural',
176
  'اسپانیایی (السالوادور) - لورنا (زن)': 'es-SV-LorenaNeural',
177
  'اسپانیایی (السالوادور) - رودریگو (مرد)': 'es-SV-RodrigoNeural',
178
- 'اسپانیایی (آمریکا) - آلونسو (مرد)': 'es-US-AlonsoNeural', # "United States" به "آمریکا"
179
  'اسپانیایی (آمریکا) - پالوما (زن)': 'es-US-PalomaNeural',
180
  'اسپانیایی (اروگوئه) - ماتئو (مرد)': 'es-UY-MateoNeural',
181
  'اسپانیایی (اروگوئه) - والنتینا (زن)': 'es-UY-ValentinaNeural',
182
  'اسپانیایی (ونزوئلا) - پائولا (زن)': 'es-VE-PaolaNeural',
183
  'اسپانیایی (ونزوئلا) - سباستین (مرد)': 'es-VE-SebastianNeural',
184
- 'استونیایی - آنو (زن)': 'et-EE-AnuNeural', # "Estonian"
185
  'استونیایی - کرت (مرد)': 'et-EE-KertNeural',
186
  'فارسی (ایران) - دل‌آرا (زن)': 'fa-IR-DilaraNeural',
187
  'فارسی (ایران) - فرید (مرد)': 'fa-IR-FaridNeural',
@@ -196,7 +192,7 @@ language_dict_persian_keys = {
196
  'فرانسوی (سوئیس) - فابریس (مرد)': 'fr-CH-FabriceNeural',
197
  'ایرلندی - کلم (مرد)': 'ga-IE-ColmNeural',
198
  'ایرلندی - اورلا (زن)': 'ga-IE-OrlaNeural',
199
- 'گالیسی (اسپانیا) - روی (مرد)': 'gl-ES-RoiNeural', # "Galician"
200
  'گالیسی (اسپانیا) - سابلا (زن)': 'gl-ES-SabelaNeural',
201
  'گجراتی (هند) - دوانی (زن)': 'gu-IN-DhwaniNeural',
202
  'گجراتی (هند) - نیرانجان (مرد)': 'gu-IN-NiranjanNeural',
@@ -204,7 +200,7 @@ language_dict_persian_keys = {
204
  'عبری (اسرائیل) - هیلا (زن)': 'he-IL-HilaNeural',
205
  'هندی (هند) - مادور (مرد)': 'hi-IN-MadhurNeural',
206
  'هندی (هند) - سوارا (زن)': 'hi-IN-SwaraNeural',
207
- 'کروات - گابریلا (زن)': 'hr-HR-GabrijelaNeural', # "Croatian"
208
  'کروات - سرچکو (مرد)': 'hr-HR-SreckoNeural',
209
  'مجاری - نوئمی (زن)': 'hu-HU-NoemiNeural',
210
  'مجاری - تاماش (مرد)': 'hu-HU-TamasNeural',
@@ -212,15 +208,15 @@ language_dict_persian_keys = {
212
  'ارمنی - هایک (مرد)': 'hy-AM-HaykNeural',
213
  'ایسلندی - گودرون (زن)': 'is-IS-GudrunNeural',
214
  'ایسلندی - گونار (مرد)': 'is-IS-GunnarNeural',
215
- 'جاوه‌ای (اندونزی) - دیماس (مرد)': 'jv-ID-DimasNeural', # "Javanese"
216
  'جاوه‌ای (اندونزی) - سیتی (زن)': 'jv-ID-SitiNeural',
217
  'گرجی - اکا (زن)': 'ka-GE-EkaNeural',
218
  'گرجی - گیورگی (مرد)': 'ka-GE-GiorgiNeural',
219
  'قزاقی - آیگول (زن)': 'kk-KZ-AigulNeural',
220
  'قزاقی - دولت (مرد)': 'kk-KZ-DauletNeural',
221
- 'خمر (کامبوج) - پیست (مرد)': 'km-KH-PisethNeural', # "Khmer"
222
  'خمر (کامبوج) - سری‌مم (زن)': 'km-KH-SreymomNeural',
223
- 'کانادایی (هند) - گاگان (مرد)': 'kn-IN-GaganNeural', # "Kannada"
224
  'کانادایی (هند) - ساپنا (زن)': 'kn-IN-SapnaNeural',
225
  'لائوسی - چانتاونگ (مرد)': 'lo-LA-ChanthavongNeural',
226
  'لائوسی - کئومانی (زن)': 'lo-LA-KeomanyNeural',
@@ -236,7 +232,7 @@ language_dict_persian_keys = {
236
  'مغولی - یسوی (زن)': 'mn-MN-YesuiNeural',
237
  'مراتی (هند) - آروهی (زن)': 'mr-IN-AarohiNeural',
238
  'مراتی (هند) - مانوهار (مرد)': 'mr-IN-ManoharNeural',
239
- 'مالتی (مالت) - گریس (زن)': 'mt-MT-GraceNeural', # "Maltese"
240
  'مالتی (مالت) - جوزف (مرد)': 'mt-MT-JosephNeural',
241
  'برمه‌ای (میانمار) - نیلار (زن)': 'my-MM-NilarNeural',
242
  'برمه‌ای (میانمار) - تیها (مرد)': 'my-MM-ThihaNeural',
@@ -254,7 +250,7 @@ language_dict_persian_keys = {
254
  'رومانیایی - امیل (مرد)': 'ro-RO-EmilNeural',
255
  'روسی - دیمیتری (مرد)': 'ru-RU-DmitryNeural',
256
  'روسی - سوتلانا (زن)': 'ru-RU-SvetlanaNeural',
257
- 'سینهالی (سریلانکا) - دینوکا (مرد)': 'si-LK-DinukaNeural', # "Sinhala"
258
  'سینهالی (سریلانکا) - تیلینی (زن)': 'si-LK-ThiliniNeural',
259
  'اسلواک - لوکاش (مرد)': 'sk-SK-LukasNeural',
260
  'اسلواک - ویکتوریا (زن)': 'sk-SK-ViktoriaNeural',
@@ -266,7 +262,7 @@ language_dict_persian_keys = {
266
  'آلبانیایی - ایلیر (مرد)': 'sq-AL-IlirNeural',
267
  'صربی - نیکولا (مرد)': 'sr-RS-NikolaNeural',
268
  'صربی - سوفی (زن)': 'sr-RS-SophieNeural',
269
- 'سوندانی (اندونزی) - جاجانگ (مرد)': 'su-ID-JajangNeural', # "Sundanese"
270
  'سوندانی (اندونزی) - توتی (زن)': 'su-ID-TutiNeural',
271
  'سواحیلی (کنیا) - رفیقی (مرد)': 'sw-KE-RafikiNeural',
272
  'سواحیلی (کنیا) - زوری (زن)': 'sw-KE-ZuriNeural',
@@ -302,11 +298,10 @@ language_dict_persian_keys = {
302
  'زولو (آفریقای جنوبی) - تمبا (مرد)': 'zu-ZA-ThembaNeural',
303
  }
304
 
305
- # --- توابع تبدیل متن به گفتار و wrapper (همانند قبل) ---
306
- async def text_to_speech_edge_async(text, language_code_persian, rate, volume, pitch): # language_code_persian نام پارامتر تغییر کرد
307
  try:
308
  if not text: return "خطا: لطفاً متنی را برای تبدیل وارد کنید.", None
309
- # دریافت voice_id از دیکشنری با کلیدهای فارسی
310
  voice_id = language_dict_persian_keys.get(language_code_persian)
311
  if voice_id is None: return f"خطا: مدل صدای انتخاب شده ('{language_code_persian}') یافت نشد.", None
312
  rate_str, volume_str, pitch_str = f"{int(rate):+g}%", f"{int(volume):+g}%", f"{int(pitch):+g}Hz"
@@ -318,7 +313,7 @@ async def text_to_speech_edge_async(text, language_code_persian, rate, volume, p
318
  error_msg = f"خطا: صدایی برای متن و صدای انتخاب شده دریافت نشد (صدا: {voice_id})."
319
  return error_msg, None
320
  except ValueError as ve:
321
- error_msg = f"خطا در پارامترهای ورودی: {ve}" # حذف اشاره به edge-tts
322
  return error_msg, None
323
  except Exception as e:
324
  return f"خطای غیرمنتظره در سرور: {type(e).__name__}", None
@@ -331,17 +326,38 @@ def _get_or_create_event_loop():
331
  return _event_loops_by_thread[thread_id]
332
 
333
  def text_to_speech_edge_sync_wrapper(text, language_code_persian, rate, volume, pitch):
 
334
  try:
335
- loop = _get_or_create_event_loop(); asyncio.set_event_loop(loop)
336
- result = loop.run_until_complete(text_to_speech_edge_async(text, language_code_persian, rate, volume, pitch))
 
 
 
337
  except RuntimeError as e:
338
  if "no current event loop" in str(e).lower() or "cannot be called from a running event loop" in str(e).lower():
339
- new_loop = asyncio.new_event_loop(); asyncio.set_event_loop(new_loop)
340
- try: result = new_loop.run_until_complete(text_to_speech_edge_async(text, language_code_persian, rate, volume, pitch))
341
- finally: new_loop.close()
342
- else: return f"خطای اجرایی: {e}", None
343
- except Exception as e: return f"خطای غیرمنتظره: {type(e).__name__}", None
344
- return result
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
345
 
346
  # --- تعریف تم و CSS ---
347
  app_theme = gr.themes.Soft(
@@ -364,10 +380,9 @@ body { font-family: 'Vazirmatn', 'Arial', sans-serif; direction: rtl; }
364
  text-align: center; padding: 20px 10px; background: #34495e; color: white;
365
  border-radius: 12px; margin-bottom: 1.5rem;
366
  }
367
- .app-header img.logo {
368
- width: 50px; height: auto; margin-bottom: 5px;
369
- animation: float_soft 4s ease-in-out infinite alternate;
370
- }
371
  .app-header h1 {
372
  color: white !important; font-size: 1.6em !important; font-weight: 600 !important;
373
  margin: 5px 0;
@@ -412,6 +427,25 @@ div[data-testid="flag"] { display: none !important; }
412
  button[title="Flag"], button[aria-label="Flag"] {display: none !important; }
413
  .footer-utils { display: none !important; visibility: hidden !important; }
414
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
415
  @keyframes float_soft {
416
  0% { transform: translatey(0px) scale(1); }
417
  50% { transform: translatey(-5px) scale(1.05); }
@@ -426,18 +460,15 @@ button[title="Flag"], button[aria-label="Flag"] {display: none !important; }
426
  .main-content-row > .gr-column:nth-child(1) { flex-basis: 60%; }
427
  .main-content-row > .gr-column:nth-child(2) { flex-basis: 40%; }
428
  .gr-button.lg.primary { width: auto !important; }
 
429
  }
430
  """
431
 
432
  # انتخاب صدای پیش فرض فارسی
433
  default_voice_key_persian = 'فارسی (ایران) - فرید (مرد)'
434
  if default_voice_key_persian not in language_dict_persian_keys:
435
- # اگر به هر دلیلی صدای پیش فرض فارسی ما در لیست نبود، اولین مورد لیست را انتخاب کن
436
  default_voice_key_persian = list(language_dict_persian_keys.keys())[0] if language_dict_persian_keys else None
437
 
438
- # LOGO_URL دیگر استفاده نمی‌شود چون لوگو حذف شده است، اما برای حفظ سایر بخش‌های کد دست نخورده، آن را نگه می‌داریم
439
- LOGO_URL = "https://www.gstatic.com/lamda/images/gemini/google_bard_logo_150_v2_dark_color_1x.png"
440
-
441
  with gr.Blocks(theme=app_theme, css=custom_css) as demo:
442
  with gr.Row():
443
  gr.HTML(f"""
@@ -448,14 +479,14 @@ with gr.Blocks(theme=app_theme, css=custom_css) as demo:
448
  """)
449
 
450
  with gr.Row(elem_classes="main-content-row"):
451
- with gr.Column(scale=3):
452
  input_text = gr.Textbox(
453
  lines=5,
454
  label="📝 متن خود را برای تبدیل وارد نمایید",
455
  placeholder="اینجا بنویسید...",
456
  value=""
457
  )
458
- language_dropdown = gr.Dropdown( # نام متغیر برای وضوح بیشتر
459
  choices=list(language_dict_persian_keys.keys()),
460
  value=default_voice_key_persian,
461
  label="🗣️ زبان و گوینده را انتخاب کنید"
@@ -468,9 +499,14 @@ with gr.Blocks(theme=app_theme, css=custom_css) as demo:
468
 
469
  submit_button = gr.Button("🔊 تولید و پخش صدا", variant="primary")
470
 
471
- with gr.Column(scale=2):
472
  output_text_status = gr.Textbox(label="📊 وضعیت عملیات", interactive=False, lines=1, placeholder="نتیجه اینجا نمایش داده می‌شود...")
473
  output_audio = gr.Audio(type="filepath", label="🎧 فایل صوتی خروجی", interactive=False)
 
 
 
 
 
474
 
475
  gr.HTML("<hr style='margin-top: 1rem; margin-bottom: 1rem; border: 0; border-top: 1px solid #dee2e6;'>")
476
 
@@ -480,17 +516,34 @@ with gr.Blocks(theme=app_theme, css=custom_css) as demo:
480
  ["This is a test of the speech synthesis system.", 'انگلیسی - جنی (زن)', +5, 0, 0],
481
  ["تجربه کاربری در این سامانه بسیار روان است.", 'فارسی (ایران) - فرید (مرد)', -5, 0, 0],
482
  ],
483
- inputs=[input_text, language_dropdown, rate_slider, volume_slider, pitch_slider], # استفاده از نام متغیرهای جدید
484
- outputs=[output_text_status, output_audio],
485
  fn=text_to_speech_edge_sync_wrapper,
486
- cache_examples=False,
487
  label="💡 چند نمونه برای شروع"
488
  )
489
 
490
  submit_button.click(
491
  fn=text_to_speech_edge_sync_wrapper,
492
- inputs=[input_text, language_dropdown, rate_slider, volume_slider, pitch_slider], # استفاده از نام متغیرهای جدید
493
- outputs=[output_text_status, output_audio],
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
494
  )
495
 
496
  demo.launch()
 
7
  import os
8
 
9
  # --- دیکشنری زبان‌ها و صداها با کلیدهای فارسی (نمونه) ---
 
 
 
 
10
  language_dict_persian_keys = {
11
  'انگلیسی - جنی (زن)': 'en-US-JennyNeural',
12
  'انگلیسی - گای (مرد)': 'en-US-GuyNeural',
 
17
  'انگلیسی - میشل (زن)': 'en-US-MichelleNeural',
18
  'انگلیسی - راجر (مرد)': 'en-US-RogerNeural',
19
  'اسپانیایی (مکزیک) - دالیا (زن)': 'es-MX-DaliaNeural',
20
+ 'اسپانیایی (مکزیک) - خورخه (مرد)': 'es-MX-JorgeNeural',
21
  'کره‌ای - سان-هی (زن)': 'ko-KR-SunHiNeural',
22
  'کره‌ای - این‌جون (مرد)': 'ko-KR-InJoonNeural',
23
  'تایلندی - پرموادی (زن)': 'th-TH-PremwadeeNeural',
 
39
  'هلندی - کولت (زن)': 'nl-NL-ColetteNeural',
40
  'هلندی - فنا (زن)': 'nl-NL-FennaNeural',
41
  'هلندی - مارتن (مرد)': 'nl-NL-MaartenNeural',
42
+ 'مالایی - عثمان (مرد)': 'ms-MY-OsmanNeural',
43
  'مالایی - یاسمین (زن)': 'ms-MY-YasminNeural',
44
  'نروژی - پرنیل (زن)': 'nb-NO-PernilleNeural',
45
  'نروژی - فین (مرد)': 'nb-NO-FinnNeural',
46
  'سوئدی - سوفی (زن)': 'sv-SE-SofieNeural',
47
  'سوئدی - ماتیاس (مرد)': 'sv-SE-MattiasNeural',
48
+ 'عربی (عربستان) - حامد (مرد)': 'ar-SA-HamedNeural',
49
  'عربی (عربستان) - زاریا (زن)': 'ar-SA-ZariyahNeural',
50
  'یونانی - آتنا (زن)': 'el-GR-AthinaNeural',
51
  'یونانی - نستوراس (مرد)': 'el-GR-NestorasNeural',
 
53
  'آلمانی - آمالا (زن)': 'de-DE-AmalaNeural',
54
  'آلمانی - کنراد (مرد)': 'de-DE-ConradNeural',
55
  'آلمانی - کیلیان (مرد)': 'de-DE-KillianNeural',
56
+ 'آفریقایی - آدری (زن)': 'af-ZA-AdriNeural',
57
  'آفریقایی - ویلم (مرد)': 'af-ZA-WillemNeural',
58
+ 'اتیوپیایی - آمه‌ها (مرد)': 'am-ET-AmehaNeural',
59
  'اتیوپیایی - مکدس (زن)': 'am-ET-MekdesNeural',
60
  'عربی (امارات) - فاطمه (زن)': 'ar-AE-FatimaNeural',
61
  'عربی (امارات) - حمدان (مرد)': 'ar-AE-HamdanNeural',
 
65
  'عربی (مصر) - سلما (زن)': 'ar-EG-SalmaNeural',
66
  'عربی (مصر) - شاکر (مرد)': 'ar-EG-ShakirNeural',
67
  'عربی (عراق) - باسل (مرد)': 'ar-IQ-BasselNeural',
68
+ 'عربی (عراق) - رعنا (زن)': 'ar-IQ-RanaNeural',
69
  'عربی (اردن) - سانا (زن)': 'ar-JO-SanaNeural',
70
+ 'عربی (اردن) - تایم (مرد)': 'ar-JO-TaimNeural',
71
  'عربی (کویت) - فهد (مرد)': 'ar-KW-FahedNeural',
72
  'عربی (کویت) - نورا (زن)': 'ar-KW-NouraNeural',
73
  'عربی (لبنان) - لیلا (زن)': 'ar-LB-LaylaNeural',
 
78
  'عربی (مراکش) - مونا (زن)': 'ar-MA-MounaNeural',
79
  'عربی (عمان) - عبدالله (مرد)': 'ar-OM-AbdullahNeural',
80
  'عربی (عمان) - عایشه (زن)': 'ar-OM-AyshaNeural',
81
+ 'عربی (قطر) - امل (زن)': 'ar-QA-AmalNeural',
82
  'عربی (قطر) - معاذ (مرد)': 'ar-QA-MoazNeural',
83
  'عربی (سوریه) - امانی (زن)': 'ar-SY-AmanyNeural',
84
  'عربی (سوریه) - لیث (مرد)': 'ar-SY-LaithNeural',
85
+ 'عربی (تونس) - هادی (مرد)': 'ar-TN-HediNeural',
86
  'عربی (تونس) - ریم (زن)': 'ar-TN-ReemNeural',
87
  'عربی (یمن) - مریم (زن)': 'ar-YE-MaryamNeural',
88
  'عربی (یمن) - صالح (مرد)': 'ar-YE-SalehNeural',
 
93
  'بنگالی (بنگلادش) - نابانیتا (زن)': 'bn-BD-NabanitaNeural',
94
  'بنگالی (بنگلادش) - پرادیپ (مرد)': 'bn-BD-PradeepNeural',
95
  'بنگالی (هند) - باشکار (مرد)': 'bn-IN-BashkarNeural',
96
+ 'بنگالی (هند) - تانیشا (زن)': 'bn-IN-TanishaaNeural',
97
+ 'بوسنیایی - گوران (مرد)': 'bs-BA-GoranNeural',
98
  'بوسنیایی - وسنا (زن)': 'bs-BA-VesnaNeural',
99
+ 'کاتالان (اسپانیا) - جوآنا (زن)': 'ca-ES-JoanaNeural',
100
  'کاتالان (اسپانیا) - انریک (مرد)': 'ca-ES-EnricNeural',
101
+ 'چکی - آنتونین (مرد)': 'cs-CZ-AntoninNeural',
102
  'چکی - ولاستا (زن)': 'cs-CZ-VlastaNeural',
103
+ 'ولزی (بریتانیا) - آلد (مرد)': 'cy-GB-AledNeural',
104
  'ولزی (بریتانیا) - نیا (زن)': 'cy-GB-NiaNeural',
105
  'دانمارکی - کریستل (زن)': 'da-DK-ChristelNeural',
106
  'دانمارکی - یپه (مرد)': 'da-DK-JeppeNeural',
 
171
  'اسپانیایی (پاراگوئه) - تانیا (زن)': 'es-PY-TaniaNeural',
172
  'اسپانیایی (السالوادور) - لورنا (زن)': 'es-SV-LorenaNeural',
173
  'اسپانیایی (السالوادور) - رودریگو (مرد)': 'es-SV-RodrigoNeural',
174
+ 'اسپانیایی (آمریکا) - آلونسو (مرد)': 'es-US-AlonsoNeural',
175
  'اسپانیایی (آمریکا) - پالوما (زن)': 'es-US-PalomaNeural',
176
  'اسپانیایی (اروگوئه) - ماتئو (مرد)': 'es-UY-MateoNeural',
177
  'اسپانیایی (اروگوئه) - والنتینا (زن)': 'es-UY-ValentinaNeural',
178
  'اسپانیایی (ونزوئلا) - پائولا (زن)': 'es-VE-PaolaNeural',
179
  'اسپانیایی (ونزوئلا) - سباستین (مرد)': 'es-VE-SebastianNeural',
180
+ 'استونیایی - آنو (زن)': 'et-EE-AnuNeural',
181
  'استونیایی - کرت (مرد)': 'et-EE-KertNeural',
182
  'فارسی (ایران) - دل‌آرا (زن)': 'fa-IR-DilaraNeural',
183
  'فارسی (ایران) - فرید (مرد)': 'fa-IR-FaridNeural',
 
192
  'فرانسوی (سوئیس) - فابریس (مرد)': 'fr-CH-FabriceNeural',
193
  'ایرلندی - کلم (مرد)': 'ga-IE-ColmNeural',
194
  'ایرلندی - اورلا (زن)': 'ga-IE-OrlaNeural',
195
+ 'گالیسی (اسپانیا) - روی (مرد)': 'gl-ES-RoiNeural',
196
  'گالیسی (اسپانیا) - سابلا (زن)': 'gl-ES-SabelaNeural',
197
  'گجراتی (هند) - دوانی (زن)': 'gu-IN-DhwaniNeural',
198
  'گجراتی (هند) - نیرانجان (مرد)': 'gu-IN-NiranjanNeural',
 
200
  'عبری (اسرائیل) - هیلا (زن)': 'he-IL-HilaNeural',
201
  'هندی (هند) - مادور (مرد)': 'hi-IN-MadhurNeural',
202
  'هندی (هند) - سوارا (زن)': 'hi-IN-SwaraNeural',
203
+ 'کروات - گابریلا (زن)': 'hr-HR-GabrijelaNeural',
204
  'کروات - سرچکو (مرد)': 'hr-HR-SreckoNeural',
205
  'مجاری - نوئمی (زن)': 'hu-HU-NoemiNeural',
206
  'مجاری - تاماش (مرد)': 'hu-HU-TamasNeural',
 
208
  'ارمنی - هایک (مرد)': 'hy-AM-HaykNeural',
209
  'ایسلندی - گودرون (زن)': 'is-IS-GudrunNeural',
210
  'ایسلندی - گونار (مرد)': 'is-IS-GunnarNeural',
211
+ 'جاوه‌ای (اندونزی) - دیماس (مرد)': 'jv-ID-DimasNeural',
212
  'جاوه‌ای (اندونزی) - سیتی (زن)': 'jv-ID-SitiNeural',
213
  'گرجی - اکا (زن)': 'ka-GE-EkaNeural',
214
  'گرجی - گیورگی (مرد)': 'ka-GE-GiorgiNeural',
215
  'قزاقی - آیگول (زن)': 'kk-KZ-AigulNeural',
216
  'قزاقی - دولت (مرد)': 'kk-KZ-DauletNeural',
217
+ 'خمر (کامبوج) - پیست (مرد)': 'km-KH-PisethNeural',
218
  'خمر (کامبوج) - سری‌مم (زن)': 'km-KH-SreymomNeural',
219
+ 'کانادایی (هند) - گاگان (مرد)': 'kn-IN-GaganNeural',
220
  'کانادایی (هند) - ساپنا (زن)': 'kn-IN-SapnaNeural',
221
  'لائوسی - چانتاونگ (مرد)': 'lo-LA-ChanthavongNeural',
222
  'لائوسی - کئومانی (زن)': 'lo-LA-KeomanyNeural',
 
232
  'مغولی - یسوی (زن)': 'mn-MN-YesuiNeural',
233
  'مراتی (هند) - آروهی (زن)': 'mr-IN-AarohiNeural',
234
  'مراتی (هند) - مانوهار (مرد)': 'mr-IN-ManoharNeural',
235
+ 'مالتی (مالت) - گریس (زن)': 'mt-MT-GraceNeural',
236
  'مالتی (مالت) - جوزف (مرد)': 'mt-MT-JosephNeural',
237
  'برمه‌ای (میانمار) - نیلار (زن)': 'my-MM-NilarNeural',
238
  'برمه‌ای (میانمار) - تیها (مرد)': 'my-MM-ThihaNeural',
 
250
  'رومانیایی - امیل (مرد)': 'ro-RO-EmilNeural',
251
  'روسی - دیمیتری (مرد)': 'ru-RU-DmitryNeural',
252
  'روسی - سوتلانا (زن)': 'ru-RU-SvetlanaNeural',
253
+ 'سینهالی (سریلانکا) - دینوکا (مرد)': 'si-LK-DinukaNeural',
254
  'سینهالی (سریلانکا) - تیلینی (زن)': 'si-LK-ThiliniNeural',
255
  'اسلواک - لوکاش (مرد)': 'sk-SK-LukasNeural',
256
  'اسلواک - ویکتوریا (زن)': 'sk-SK-ViktoriaNeural',
 
262
  'آلبانیایی - ایلیر (مرد)': 'sq-AL-IlirNeural',
263
  'صربی - نیکولا (مرد)': 'sr-RS-NikolaNeural',
264
  'صربی - سوفی (زن)': 'sr-RS-SophieNeural',
265
+ 'سوندانی (اندونزی) - جاجانگ (مرد)': 'su-ID-JajangNeural',
266
  'سوندانی (اندونزی) - توتی (زن)': 'su-ID-TutiNeural',
267
  'سواحیلی (کنیا) - رفیقی (مرد)': 'sw-KE-RafikiNeural',
268
  'سواحیلی (کنیا) - زوری (زن)': 'sw-KE-ZuriNeural',
 
298
  'زولو (آفریقای جنوبی) - تمبا (مرد)': 'zu-ZA-ThembaNeural',
299
  }
300
 
301
+ # --- توابع تبدیل متن به گفتار ---
302
+ async def text_to_speech_edge_async(text, language_code_persian, rate, volume, pitch):
303
  try:
304
  if not text: return "خطا: لطفاً متنی را برای تبدیل وارد کنید.", None
 
305
  voice_id = language_dict_persian_keys.get(language_code_persian)
306
  if voice_id is None: return f"خطا: مدل صدای انتخاب شده ('{language_code_persian}') یافت نشد.", None
307
  rate_str, volume_str, pitch_str = f"{int(rate):+g}%", f"{int(volume):+g}%", f"{int(pitch):+g}Hz"
 
313
  error_msg = f"خطا: صدایی برای متن و صدای انتخاب شده دریافت نشد (صدا: {voice_id})."
314
  return error_msg, None
315
  except ValueError as ve:
316
+ error_msg = f"خطا در پارامترهای ورودی: {ve}"
317
  return error_msg, None
318
  except Exception as e:
319
  return f"خطای غیرمنتظره در سرور: {type(e).__name__}", None
 
326
  return _event_loops_by_thread[thread_id]
327
 
328
  def text_to_speech_edge_sync_wrapper(text, language_code_persian, rate, volume, pitch):
329
+ status_msg, audio_path = None, None
330
  try:
331
+ loop = _get_or_create_event_loop()
332
+ asyncio.set_event_loop(loop)
333
+ status_msg, audio_path = loop.run_until_complete(
334
+ text_to_speech_edge_async(text, language_code_persian, rate, volume, pitch)
335
+ )
336
  except RuntimeError as e:
337
  if "no current event loop" in str(e).lower() or "cannot be called from a running event loop" in str(e).lower():
338
+ new_loop = asyncio.new_event_loop()
339
+ asyncio.set_event_loop(new_loop)
340
+ try:
341
+ status_msg, audio_path = new_loop.run_until_complete(
342
+ text_to_speech_edge_async(text, language_code_persian, rate, volume, pitch)
343
+ )
344
+ finally:
345
+ new_loop.close()
346
+ else:
347
+ status_msg = f"خطای اجرایی: {e}"
348
+ audio_path = None
349
+ except Exception as e:
350
+ status_msg = f"خطای غیرمنتظره: {type(e).__name__}"
351
+ audio_path = None
352
+
353
+ if audio_path:
354
+ return status_msg, audio_path, gr.update(visible=True), audio_path
355
+ else:
356
+ # اگر status_msg از قبل توسط exception ها مقداردهی نشده، یک پیام عمومی بگذار
357
+ if status_msg is None:
358
+ status_msg = "خطا در تولید صدا."
359
+ return status_msg, None, gr.update(visible=False), None
360
+
361
 
362
  # --- تعریف تم و CSS ---
363
  app_theme = gr.themes.Soft(
 
380
  text-align: center; padding: 20px 10px; background: #34495e; color: white;
381
  border-radius: 12px; margin-bottom: 1.5rem;
382
  }
383
+ /* لوگو دیگر نمایش داده نمی‌شود، اما کلاس آن را نگه می‌داریم تا سایر استایل‌ها به هم نریزد */
384
+ .app-header img.logo { display: none; }
385
+
 
386
  .app-header h1 {
387
  color: white !important; font-size: 1.6em !important; font-weight: 600 !important;
388
  margin: 5px 0;
 
427
  button[title="Flag"], button[aria-label="Flag"] {display: none !important; }
428
  .footer-utils { display: none !important; visibility: hidden !important; }
429
 
430
+ .download-button-custom {
431
+ background: #2ecc71 !important; /* Green color */
432
+ color: white !important;
433
+ font-weight: 500 !important;
434
+ border-radius: 8px !important;
435
+ padding: 10px 15px !important;
436
+ width: 100% !important;
437
+ font-size: 0.95em !important;
438
+ transition: all 0.2s ease !important;
439
+ box-shadow: 0 3px 6px rgba(46, 204, 113, 0.25) !important;
440
+ border: none !important;
441
+ margin-top: 10px !important; /* Space above the button */
442
+ }
443
+ .download-button-custom:hover {
444
+ background: #27ae60 !important; /* Darker green */
445
+ transform: translateY(-2px) !important;
446
+ box-shadow: 0 5px 10px rgba(46, 204, 113, 0.35) !important;
447
+ }
448
+
449
  @keyframes float_soft {
450
  0% { transform: translatey(0px) scale(1); }
451
  50% { transform: translatey(-5px) scale(1.05); }
 
460
  .main-content-row > .gr-column:nth-child(1) { flex-basis: 60%; }
461
  .main-content-row > .gr-column:nth-child(2) { flex-basis: 40%; }
462
  .gr-button.lg.primary { width: auto !important; }
463
+ .download-button-custom { width: auto !important; } /* Adjust for wider screens */
464
  }
465
  """
466
 
467
  # انتخاب صدای پیش فرض فارسی
468
  default_voice_key_persian = 'فارسی (ایران) - فرید (مرد)'
469
  if default_voice_key_persian not in language_dict_persian_keys:
 
470
  default_voice_key_persian = list(language_dict_persian_keys.keys())[0] if language_dict_persian_keys else None
471
 
 
 
 
472
  with gr.Blocks(theme=app_theme, css=custom_css) as demo:
473
  with gr.Row():
474
  gr.HTML(f"""
 
479
  """)
480
 
481
  with gr.Row(elem_classes="main-content-row"):
482
+ with gr.Column(scale=3): # ستون ورودی‌ها
483
  input_text = gr.Textbox(
484
  lines=5,
485
  label="📝 متن خود را برای تبدیل وارد نمایید",
486
  placeholder="اینجا بنویسید...",
487
  value=""
488
  )
489
+ language_dropdown = gr.Dropdown(
490
  choices=list(language_dict_persian_keys.keys()),
491
  value=default_voice_key_persian,
492
  label="🗣️ زبان و گوینده را انتخاب کنید"
 
499
 
500
  submit_button = gr.Button("🔊 تولید و پخش صدا", variant="primary")
501
 
502
+ with gr.Column(scale=2): # ستون خروجی‌ها
503
  output_text_status = gr.Textbox(label="📊 وضعیت عملیات", interactive=False, lines=1, placeholder="نتیجه اینجا نمایش داده می‌شود...")
504
  output_audio = gr.Audio(type="filepath", label="🎧 فایل صوتی خروجی", interactive=False)
505
+
506
+ audio_file_path_state = gr.State(value=None) # برای نگهداری مسیر فایل صوتی
507
+ download_button = gr.Button("📥 دانلود صدا", visible=False, elem_classes="download-button-custom")
508
+ # کامپوننت HTML نامرئی برای اجرای جاوااسکریپت جهت باز کردن لینک دانلود
509
+ download_trigger_html = gr.HTML(visible=False)
510
 
511
  gr.HTML("<hr style='margin-top: 1rem; margin-bottom: 1rem; border: 0; border-top: 1px solid #dee2e6;'>")
512
 
 
516
  ["This is a test of the speech synthesis system.", 'انگلیسی - جنی (زن)', +5, 0, 0],
517
  ["تجربه کاربری در این سامانه بسیار روان است.", 'فارسی (ایران) - فرید (مرد)', -5, 0, 0],
518
  ],
519
+ inputs=[input_text, language_dropdown, rate_slider, volume_slider, pitch_slider],
520
+ outputs=[output_text_status, output_audio, download_button, audio_file_path_state], # دکمه دانلود و استیت هم اضافه شد
521
  fn=text_to_speech_edge_sync_wrapper,
522
+ cache_examples=False, # برای اطمینان از اجرای تابع در هر بار
523
  label="💡 چند نمونه برای شروع"
524
  )
525
 
526
  submit_button.click(
527
  fn=text_to_speech_edge_sync_wrapper,
528
+ inputs=[input_text, language_dropdown, rate_slider, volume_slider, pitch_slider],
529
+ outputs=[output_text_status, output_audio, download_button, audio_file_path_state], # دکمه دانلود و استیت هم اضافه شد
530
+ )
531
+
532
+ # تابع برای مدیریت کلیک دکمه دانلود
533
+ def trigger_download_action(filepath_from_state):
534
+ if filepath_from_state:
535
+ # Gradio فایل‌ها را از طریق '/file=' سرویس‌دهی می‌کند
536
+ # استفاده از window.open برای باز کردن در تب جدید
537
+ # افزودن یک پارامتر اضافی مثل 'download' به URL ممکن است به برخی مرورگرها برای دانلود مستقیم کمک کند
538
+ # اما استانداردترین روش، باز کردن مستقیم فایل است که مرورگر تصمیم می‌گیرد چگونه آن را مدیریت کند.
539
+ escaped_filepath = filepath_from_state.replace("'", "\\'") # برای جلوگیری از خطای JS اگر نام فایل ' داشت
540
+ return f"<script>window.open('/file={escaped_filepath}', '_blank');</script>"
541
+ return "" # اگر مسیری وجود نداشت، هیچ کاری نکن
542
+
543
+ download_button.click(
544
+ fn=trigger_download_action,
545
+ inputs=[audio_file_path_state],
546
+ outputs=[download_trigger_html] # خروجی به کامپوننت HTML نامرئی برای اجرای اسکریپت
547
  )
548
 
549
  demo.launch()