Compare commits

...

55 Commits

Author SHA1 Message Date
Moyasee
47e5999125 fix: add support for vikingfile alternative domain in downloader logic 2026-01-29 19:59:51 +02:00
Chubby Granny Chaser
53bd31dec0 Merge pull request #1941 from hydralauncher/dependabot/npm_and_yarn/npm_and_yarn-bb754c2437
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
chore(deps): bump tar from 7.5.2 to 7.5.3 in the npm_and_yarn group across 1 directory
2026-01-26 15:30:07 +00:00
Chubby Granny Chaser
9f00df3fb0 Merge branch 'main' into dependabot/npm_and_yarn/npm_and_yarn-bb754c2437 2026-01-26 15:30:00 +00:00
Chubby Granny Chaser
ce66710dac Merge pull request #1960 from hydralauncher/feat/LBX-467
Feat: Changing order of queued downloads and adding games directly to queue
2026-01-26 15:28:27 +00:00
Moyasee
884aabbfc0 Merge branch 'feat/LBX-467' of https://github.com/hydralauncher/hydra into feat/LBX-467 2026-01-25 10:41:26 +02:00
Moyasee
d4bd8f7bf1 refactor: remove unnecessary download key deletion in game installer actions and simplify file size handling in download manager 2026-01-25 10:40:38 +02:00
Chubby Granny Chaser
ad812c393d Merge branch 'main' into feat/LBX-467 2026-01-25 06:14:21 +00:00
Zamitto
57c2e74013 chore: update ww sdk
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
2026-01-24 20:54:54 -03:00
Chubby Granny Chaser
17528e74bb Merge pull request #1944 from anderlli0053/main
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
Added Slovenian translation
2026-01-24 18:10:02 +00:00
Chubby Granny Chaser
355993b954 Merge branch 'main' into main 2026-01-24 18:09:44 +00:00
Chubby Granny Chaser
ed79b8faeb Merge branch 'main' into dependabot/npm_and_yarn/npm_and_yarn-bb754c2437 2026-01-24 18:09:03 +00:00
Chubby Granny Chaser
4aaaecbee5 Merge pull request #1939 from Wkeynhk/main
Fix RU translation
2026-01-24 18:08:56 +00:00
Chubby Granny Chaser
802b4bd26b Merge branch 'main' into main 2026-01-24 18:03:20 +00:00
Moyasee
c9afd65536 refactor: simplify game entry preparation in download processes by consolidating logic into a helper function 2026-01-24 19:56:55 +02:00
Moyasee
d448a699da refactor: streamline error handling in download processes by utilizing a dedicated error handler 2026-01-24 19:50:25 +02:00
Moyasee
eea7148108 refactor: enhance download management by validating URLs and adding file size handling 2026-01-24 19:38:00 +02:00
Moyasee
fb1380356e feat: add functionality to manage download queue with new actions and translations 2026-01-24 18:46:07 +02:00
dependabot[bot]
3f82b13b1f chore(deps): bump tar in the npm_and_yarn group across 1 directory
Bumps the npm_and_yarn group with 1 update in the / directory: [tar](https://github.com/isaacs/node-tar).


Updates `tar` from 7.5.2 to 7.5.3
- [Release notes](https://github.com/isaacs/node-tar/releases)
- [Changelog](https://github.com/isaacs/node-tar/blob/main/CHANGELOG.md)
- [Commits](https://github.com/isaacs/node-tar/compare/v7.5.2...v7.5.3)

---
updated-dependencies:
- dependency-name: tar
  dependency-version: 7.5.3
  dependency-type: direct:production
  dependency-group: npm_and_yarn
...

Signed-off-by: dependabot[bot] <support@github.com>
2026-01-21 21:28:17 +00:00
Zamitto
baa2c8471a chore: bump ww version
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
2026-01-21 18:26:17 -03:00
Moyase
63bb5ca511 Merge pull request #1948 from hydralauncher/fix/LBX-454
refactor: improve notification handling in SidebarProfile component
2026-01-21 21:05:46 +02:00
Moyase
073d3f25e3 Merge branch 'main' into fix/LBX-454 2026-01-21 18:24:44 +02:00
Moyase
066185e6ee Merge pull request #1947 from hydralauncher/feat/LBX-452
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
feat: implement dynamic port discovery for Python RPC service
2026-01-21 18:24:25 +02:00
Moyase
8ce4b59294 Merge branch 'main' into feat/LBX-452 2026-01-21 18:09:16 +02:00
Moyase
90b62e4e8d Merge pull request #1946 from hydralauncher/feat/download-option-availability
feat: enhance repack availability status display with orb displaying availability
2026-01-21 18:09:03 +02:00
Moyase
8a447f683a Merge branch 'main' into feat/download-option-availability 2026-01-21 18:04:15 +02:00
Moyasee
5ddfd88ef7 refactor: remove polling to notifications count api 2026-01-21 11:33:42 +02:00
Moyasee
569ad1c862 chore: add get-port dependency and refactor Python RPC port handling 2026-01-21 11:28:14 +02:00
Chubby Granny Chaser
039df43123 Merge pull request #1945 from hydralauncher/feat/LBX-399
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
feat: add automatic executable path binding upon download finish
2026-01-21 09:20:17 +00:00
Moyasee
50bafbb7f6 refactor: improve notification handling in SidebarProfile component 2026-01-20 19:41:24 +02:00
Moyasee
46154fa49a fix: correct error handling in Python RPC process exit code 2026-01-20 19:34:04 +02:00
Moyasee
aae35b591d feat: implement dynamic port discovery for Python RPC service 2026-01-20 19:25:32 +02:00
Moyasee
10ac6c9d9c style: update repacks modal styles for improved layout and positioning 2026-01-20 19:03:35 +02:00
Moyasee
9ca6a114b1 feat: enhance repack availability status display with new UI elements and translations 2026-01-20 18:52:52 +02:00
Moyasee
2108a523bc refactor: streamline game scanning logic and enhance notification handling 2026-01-19 18:01:55 +02:00
Moyasee
fbbb2520e0 feat: enhance game scanning notifications and UI updates 2026-01-19 17:57:49 +02:00
Moyasee
049a989e85 fix: deleted unnecessary import and fixed assertion 2026-01-19 15:19:50 +02:00
Moyasee
88b2581797 feat: add scan installed games functionality with UI integration 2026-01-19 15:17:27 +02:00
Moyasee
c9801644ac fix: prevent processing downloads without a folder name 2026-01-19 04:22:44 +02:00
Moyasee
98cfe7be98 feat: add automatic executable path binding upon download finish 2026-01-19 04:01:21 +02:00
Andrew Poženel
128f864ca7 Reformat the translation.json file 2026-01-18 23:02:46 +01:00
Andrew Poženel
a2b993bb9b Added Slovenian translation 2026-01-18 22:36:30 +01:00
Zamitto
7293afb618 Merge branch 'release/v3.8.0'
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
2026-01-15 08:43:06 -03:00
Zamitto
194e7918ca feat: dont setup ww feedback widget if user has no token
Some checks failed
Build Renderer / build (push) Has been cancelled
Release / build (ubuntu-latest) (push) Has been cancelled
Release / build (windows-2022) (push) Has been cancelled
2026-01-15 08:42:33 -03:00
Zamitto
979958aca6 feat: update ww webRequest interceptor
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
2026-01-14 19:37:17 -03:00
Wkeynhk
cafa536f79 Fix RU translation 2026-01-14 16:55:55 +03:00
Zamitto
6e92e0f79f fix: getLibrary throwing error
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
2026-01-14 00:37:22 -03:00
Zamitto
aef069d4c7 Merge branch 'release/v3.8.1' 2026-01-14 00:07:53 -03:00
Zamitto
1f447cc478 chore: add sentry var to build-renderer action
Some checks failed
Build Renderer / build (push) Has been cancelled
Release / build (ubuntu-latest) (push) Has been cancelled
Release / build (windows-2022) (push) Has been cancelled
2026-01-14 00:05:55 -03:00
Zamitto
5d2dc3616c Merge pull request #1938 from hydralauncher/release/v3.8.1
sync main
2026-01-13 23:43:48 -03:00
Zamitto
1f9972f74e Merge pull request #1937 from hydralauncher/chore/add-sentry
chore: add sentry
2026-01-13 23:43:16 -03:00
Zamitto
3344f68408 feat: add semver for sentry 2026-01-13 23:42:22 -03:00
Zamitto
65be11cc07 chore: add sentry 2026-01-13 23:34:09 -03:00
Chubby Granny Chaser
7e78a0f9f1 chore: update version to 3.8.1 and enhance translations
Some checks failed
Build / build (ubuntu-latest) (push) Has been cancelled
Build / build (windows-2022) (push) Has been cancelled
- Bumped version number in package.json to 3.8.1.
- Added new translation keys for notifications and loading states in Spanish, Portuguese, and Russian.
- Improved UI elements in download group with updated styles for buttons and layout adjustments.
2026-01-11 19:25:11 +00:00
Chubby Granny Chaser
d56cc8695b Merge pull request #1928 from hydralauncher/feat/LBX-367
feat: implement native HTTP downloader option
2026-01-11 18:43:47 +00:00
Zamitto
96140e614c Merge pull request #1917 from hydralauncher/fix/friends-box-display
Some checks failed
Build Renderer / build (push) Has been cancelled
Release / build (ubuntu-latest) (push) Has been cancelled
Release / build (windows-2022) (push) Has been cancelled
hotfix: add empty state for friends box and new translation key
2026-01-04 02:59:53 -03:00
51 changed files with 2328 additions and 279 deletions

View File

@@ -42,6 +42,7 @@ jobs:
run: yarn build run: yarn build
env: env:
RENDERER_VITE_EXTERNAL_RESOURCES_URL: ${{ vars.EXTERNAL_RESOURCES_URL }} RENDERER_VITE_EXTERNAL_RESOURCES_URL: ${{ vars.EXTERNAL_RESOURCES_URL }}
RENDERER_VITE_SENTRY_DSN: ${{ vars.SENTRY_DSN }}
- name: Deploy to Cloudflare Pages - name: Deploy to Cloudflare Pages
env: env:

View File

@@ -1,6 +1,6 @@
{ {
"name": "hydralauncher", "name": "hydralauncher",
"version": "3.8.0", "version": "3.8.1",
"description": "Hydra", "description": "Hydra",
"main": "./out/main/index.js", "main": "./out/main/index.js",
"author": "Los Broxas", "author": "Los Broxas",
@@ -40,6 +40,7 @@
"@primer/octicons-react": "^19.9.0", "@primer/octicons-react": "^19.9.0",
"@radix-ui/react-dropdown-menu": "^2.1.16", "@radix-ui/react-dropdown-menu": "^2.1.16",
"@reduxjs/toolkit": "^2.2.3", "@reduxjs/toolkit": "^2.2.3",
"@sentry/react": "^10.33.0",
"@tiptap/extension-bold": "^3.6.2", "@tiptap/extension-bold": "^3.6.2",
"@tiptap/extension-italic": "^3.6.2", "@tiptap/extension-italic": "^3.6.2",
"@tiptap/extension-link": "^3.6.2", "@tiptap/extension-link": "^3.6.2",
@@ -63,6 +64,7 @@
"embla-carousel-react": "^8.6.0", "embla-carousel-react": "^8.6.0",
"file-type": "^20.5.0", "file-type": "^20.5.0",
"framer-motion": "^12.15.0", "framer-motion": "^12.15.0",
"get-port": "^7.1.0",
"hls.js": "^1.5.12", "hls.js": "^1.5.12",
"i18next": "^23.11.2", "i18next": "^23.11.2",
"i18next-browser-languagedetector": "^7.2.1", "i18next-browser-languagedetector": "^7.2.1",
@@ -86,12 +88,12 @@
"sound-play": "^1.1.0", "sound-play": "^1.1.0",
"steam-shortcut-editor": "https://github.com/hydralauncher/steam-shortcut-editor", "steam-shortcut-editor": "https://github.com/hydralauncher/steam-shortcut-editor",
"sudo-prompt": "^9.2.1", "sudo-prompt": "^9.2.1",
"tar": "^7.5.2", "tar": "^7.5.4",
"tough-cookie": "^5.1.1", "tough-cookie": "^5.1.1",
"user-agents": "^1.1.387", "user-agents": "^1.1.387",
"uuid": "^13.0.0", "uuid": "^13.0.0",
"winreg": "^1.2.5", "winreg": "^1.2.5",
"workwonders-sdk": "0.0.10", "workwonders-sdk": "0.1.1",
"ws": "^8.18.1", "ws": "^8.18.1",
"yaml": "^2.6.1", "yaml": "^2.6.1",
"yup": "^1.5.0" "yup": "^1.5.0"

View File

@@ -108,7 +108,17 @@
"search_results": "Search results", "search_results": "Search results",
"settings": "Settings", "settings": "Settings",
"version_available_install": "Version {{version}} available. Click here to restart and install.", "version_available_install": "Version {{version}} available. Click here to restart and install.",
"version_available_download": "Version {{version}} available. Click here to download." "version_available_download": "Version {{version}} available. Click here to download.",
"scan_games_tooltip": "Scan PC for installed games",
"scan_games_title": "Scan PC for installed games",
"scan_games_description": "This will scan your disks for known game executables. This may take several minutes.",
"scan_games_start": "Start Scan",
"scan_games_cancel": "Cancel",
"scan_games_result": "Found {{found}} of {{total}} games without executable path",
"scan_games_no_results": "We couldn't find any installed games.",
"scan_games_in_progress": "Scanning your disks for installed games...",
"scan_games_close": "Close",
"scan_games_scan_again": "Scan Again"
}, },
"bottom_panel": { "bottom_panel": {
"no_downloads_in_progress": "No downloads in progress", "no_downloads_in_progress": "No downloads in progress",
@@ -175,6 +185,7 @@
"repacks_modal_description": "Choose the repack you want to download", "repacks_modal_description": "Choose the repack you want to download",
"select_folder_hint": "To change the default folder, go to the <0>Settings</0>", "select_folder_hint": "To change the default folder, go to the <0>Settings</0>",
"download_now": "Download now", "download_now": "Download now",
"add_to_queue": "Add to queue",
"loading": "Loading...", "loading": "Loading...",
"no_shop_details": "Could not retrieve shop details.", "no_shop_details": "Could not retrieve shop details.",
"download_options": "Download options", "download_options": "Download options",
@@ -372,6 +383,9 @@
"audio": "Audio", "audio": "Audio",
"filter_by_source": "Filter by source", "filter_by_source": "Filter by source",
"no_repacks_found": "No sources found for this game", "no_repacks_found": "No sources found for this game",
"source_online": "Source is online",
"source_partial": "Some links are offline",
"source_offline": "Source is offline",
"delete_review": "Delete review", "delete_review": "Delete review",
"remove_review": "Remove Review", "remove_review": "Remove Review",
"delete_review_modal_title": "Are you sure you want to delete your review?", "delete_review_modal_title": "Are you sure you want to delete your review?",
@@ -434,7 +448,9 @@
"yes": "Yes", "yes": "Yes",
"no": "No", "no": "No",
"network": "NETWORK", "network": "NETWORK",
"peak": "PEAK" "peak": "PEAK",
"move_up": "Move up",
"move_down": "Move down"
}, },
"settings": { "settings": {
"downloads_path": "Downloads path", "downloads_path": "Downloads path",
@@ -619,7 +635,11 @@
"game_extracted": "{{title}} extracted successfully", "game_extracted": "{{title}} extracted successfully",
"friend_started_playing_game": "{{displayName}} started playing a game", "friend_started_playing_game": "{{displayName}} started playing a game",
"test_achievement_notification_title": "This is a test notification", "test_achievement_notification_title": "This is a test notification",
"test_achievement_notification_description": "Pretty cool, huh?" "test_achievement_notification_description": "Pretty cool, huh?",
"scan_games_complete_title": "Scanning for games finished successfully",
"scan_games_complete_description": "Found {{count}} games without executable path set",
"scan_games_no_results_title": "Scanning for games finished",
"scan_games_no_results_description": "No installed games were found"
}, },
"system_tray": { "system_tray": {
"open": "Open Hydra", "open": "Open Hydra",

View File

@@ -26,6 +26,7 @@
"game_has_no_executable": "El juego no tiene un ejecutable seleccionado", "game_has_no_executable": "El juego no tiene un ejecutable seleccionado",
"sign_in": "Iniciar Sesión", "sign_in": "Iniciar Sesión",
"friends": "Amigos", "friends": "Amigos",
"notifications": "Notificaciones",
"need_help": "¿Necesitás ayuda?", "need_help": "¿Necesitás ayuda?",
"favorites": "Favoritos", "favorites": "Favoritos",
"playable_button_title": "Solo mostrar juegos que podés jugar en este momento", "playable_button_title": "Solo mostrar juegos que podés jugar en este momento",
@@ -115,6 +116,7 @@
"downloading": "Descargando {{title}}… ({{percentage}} completado) - Restante {{eta}} - {{speed}}", "downloading": "Descargando {{title}}… ({{percentage}} completado) - Restante {{eta}} - {{speed}}",
"calculating_eta": "Descargando {{title}}… ({{percentage}} completado) - Comprobando tiempo restante…", "calculating_eta": "Descargando {{title}}… ({{percentage}} completado) - Comprobando tiempo restante…",
"checking_files": "Revisando archivos de {{title}}… ({{percentage}} completado)", "checking_files": "Revisando archivos de {{title}}… ({{percentage}} completado)",
"extracting": "Extrayendo {{title}}… ({{percentage}} completado)",
"installing_common_redist": "{{log}}…", "installing_common_redist": "{{log}}…",
"installation_complete": "Instalación completada", "installation_complete": "Instalación completada",
"installation_complete_message": "Common redistributables instalados correctamente" "installation_complete_message": "Common redistributables instalados correctamente"
@@ -173,6 +175,7 @@
"repacks_modal_description": "Elegí el repack que querés descargar", "repacks_modal_description": "Elegí el repack que querés descargar",
"select_folder_hint": "Si querés cambiar la carpeta por defecto, andá a <0>Ajustes</0>", "select_folder_hint": "Si querés cambiar la carpeta por defecto, andá a <0>Ajustes</0>",
"download_now": "Descargar ahora", "download_now": "Descargar ahora",
"loading": "Cargando...",
"no_shop_details": "No se pudieron obtener detalles de la tienda.", "no_shop_details": "No se pudieron obtener detalles de la tienda.",
"download_options": "Opciones de descarga", "download_options": "Opciones de descarga",
"download_path": "Ruta de descarga", "download_path": "Ruta de descarga",
@@ -206,6 +209,7 @@
"danger_zone_section_description": "Remover este juego de tu librería o los archivos descargados por Hydra", "danger_zone_section_description": "Remover este juego de tu librería o los archivos descargados por Hydra",
"download_in_progress": "Descarga en progreso", "download_in_progress": "Descarga en progreso",
"download_paused": "Descarga pausada", "download_paused": "Descarga pausada",
"extracting": "Extrayendo",
"last_downloaded_option": "Última opción de descarga", "last_downloaded_option": "Última opción de descarga",
"new_download_option": "Nuevo", "new_download_option": "Nuevo",
"create_steam_shortcut": "Crear atajo de Steam", "create_steam_shortcut": "Crear atajo de Steam",
@@ -400,6 +404,10 @@
"completed": "Completado", "completed": "Completado",
"removed": "No descargado", "removed": "No descargado",
"cancel": "Cancelar", "cancel": "Cancelar",
"cancel_download": "¿Cancelar descarga?",
"cancel_download_description": "¿Estás seguro de que querés cancelar esta descarga? Todos los archivos descargados serán eliminados.",
"keep_downloading": "No, seguir descargando",
"yes_cancel": "Sí, cancelar",
"filter": "Filtrar juegos descargados", "filter": "Filtrar juegos descargados",
"remove": "Remover", "remove": "Remover",
"downloading_metadata": "Descargando metadatos…", "downloading_metadata": "Descargando metadatos…",
@@ -420,7 +428,13 @@
"resume_seeding": "Continuar sembrando", "resume_seeding": "Continuar sembrando",
"options": "Administrar", "options": "Administrar",
"extract": "Extraer archivos", "extract": "Extraer archivos",
"extracting": "Extrayendo archivos…" "extracting": "Extrayendo archivos…",
"delete_archive_title": "¿Querés eliminar {{fileName}}?",
"delete_archive_description": "El archivo se extrajo exitosamente y ya no es necesario.",
"yes": "Sí",
"no": "No",
"network": "RED",
"peak": "PICO"
}, },
"settings": { "settings": {
"downloads_path": "Ruta de descarga", "downloads_path": "Ruta de descarga",
@@ -544,6 +558,7 @@
"show_download_speed_in_megabytes": "Mostrar velocidad de descarga en megabytes por segundo", "show_download_speed_in_megabytes": "Mostrar velocidad de descarga en megabytes por segundo",
"extract_files_by_default": "Extraer archivos por defecto después de descargar", "extract_files_by_default": "Extraer archivos por defecto después de descargar",
"enable_steam_achievements": "Habilitar búsqueda de logros de Steam", "enable_steam_achievements": "Habilitar búsqueda de logros de Steam",
"enable_new_download_options_badges": "Mostrar badges de nuevas opciones de descarga",
"achievement_custom_notification_position": "Posición de notificación de logros", "achievement_custom_notification_position": "Posición de notificación de logros",
"top-left": "Superior Izquierda", "top-left": "Superior Izquierda",
"top-center": "Superior Centro", "top-center": "Superior Centro",
@@ -570,20 +585,10 @@
"debrid_description": "Los servicios Debrid son descargadores premium sin restricciones que te dejan descargar más rápido archivos alojados en servicios de alojamiento siendo que la única limitación es tu velocidad de internet.", "debrid_description": "Los servicios Debrid son descargadores premium sin restricciones que te dejan descargar más rápido archivos alojados en servicios de alojamiento siendo que la única limitación es tu velocidad de internet.",
"enable_friend_start_game_notifications": "Cuando un amigo está jugando un juego", "enable_friend_start_game_notifications": "Cuando un amigo está jugando un juego",
"autoplay_trailers_on_game_page": "Reproducir trailers automáticamente en la página del juego", "autoplay_trailers_on_game_page": "Reproducir trailers automáticamente en la página del juego",
"change_achievement_sound": "Cambiar sonido de logro", "hide_to_tray_on_game_start": "Ocultar Hydra en la bandeja al iniciar un juego",
"download_source_already_exists": "Esta fuente de descarga URL ya existe.", "downloads": "Descargas",
"download_source_failed": "Error", "use_native_http_downloader": "Usar descargador HTTP nativo (experimental)",
"download_source_matched": "Actualizado", "cannot_change_downloader_while_downloading": "No se puede cambiar esta configuración mientras una descarga está en progreso"
"download_source_matching": "Actualizando",
"download_source_no_information": "Sin información disponible",
"download_source_pending_matching": "Actualizando pronto",
"download_sources_synced_successfully": "Todas las fuentes de descarga están sincronizadas",
"failed_add_download_source": "Error al añadir la fuente de descarga. Por favor intentá de nuevo.",
"hydra_cloud": "Hydra Cloud",
"preview_sound": "Vista previa de sonido",
"remove_achievement_sound": "Eliminar sonido de logros",
"removed_all_download_sources": "Todas las fuentes de descarga eliminadas",
"hide_to_tray_on_game_start": "Ocultar Hydra en la bandeja al iniciar un juego"
}, },
"notifications": { "notifications": {
"download_complete": "Descarga completada", "download_complete": "Descarga completada",
@@ -675,6 +680,7 @@
"blocked_users": "Usuarios bloqueados", "blocked_users": "Usuarios bloqueados",
"unblock": "Desbloquear", "unblock": "Desbloquear",
"no_friends_added": "No tenés amistades añadidas", "no_friends_added": "No tenés amistades añadidas",
"no_friends_yet": "Aún no has agregado ningún amigo",
"view_all": "Ver todo", "view_all": "Ver todo",
"load_more": "Cargar más", "load_more": "Cargar más",
"loading": "Cargando", "loading": "Cargando",

View File

@@ -26,6 +26,7 @@
"game_has_no_executable": "Jogo não possui executável selecionado", "game_has_no_executable": "Jogo não possui executável selecionado",
"sign_in": "Login", "sign_in": "Login",
"friends": "Amigos", "friends": "Amigos",
"notifications": "Notificações",
"need_help": "Precisa de ajuda?", "need_help": "Precisa de ajuda?",
"favorites": "Favoritos", "favorites": "Favoritos",
"playable_button_title": "Mostrar apenas jogos que você pode jogar agora", "playable_button_title": "Mostrar apenas jogos que você pode jogar agora",
@@ -163,6 +164,7 @@
"repacks_modal_description": "Escolha o repack do jogo que deseja baixar", "repacks_modal_description": "Escolha o repack do jogo que deseja baixar",
"select_folder_hint": "Para trocar o diretório padrão, acesse a <0>Tela de Ajustes</0>", "select_folder_hint": "Para trocar o diretório padrão, acesse a <0>Tela de Ajustes</0>",
"download_now": "Iniciar download", "download_now": "Iniciar download",
"loading": "Carregando...",
"no_shop_details": "Não foi possível obter os detalhes da loja.", "no_shop_details": "Não foi possível obter os detalhes da loja.",
"download_options": "Opções de download", "download_options": "Opções de download",
"download_path": "Diretório de download", "download_path": "Diretório de download",
@@ -368,6 +370,7 @@
"show_translation": "Mostrar tradução", "show_translation": "Mostrar tradução",
"show_original_translated_from": "Mostrar original (traduzido do {{language}})", "show_original_translated_from": "Mostrar original (traduzido do {{language}})",
"hide_original": "Ocultar original", "hide_original": "Ocultar original",
"vote_failed": "Falha ao registrar seu voto. Por favor, tente novamente.",
"rating_count": "Avaliação", "rating_count": "Avaliação",
"review_from_blocked_user": "Avaliação de usuário bloqueado", "review_from_blocked_user": "Avaliação de usuário bloqueado",
"show": "Mostrar", "show": "Mostrar",
@@ -390,6 +393,10 @@
"completed": "Concluído", "completed": "Concluído",
"removed": "Cancelado", "removed": "Cancelado",
"cancel": "Cancelar", "cancel": "Cancelar",
"cancel_download": "Cancelar download?",
"cancel_download_description": "Tem certeza de que deseja cancelar este download? Todos os arquivos baixados serão excluídos.",
"keep_downloading": "Não, continuar baixando",
"yes_cancel": "Sim, cancelar",
"filter": "Filtrar jogos baixados", "filter": "Filtrar jogos baixados",
"remove": "Remover", "remove": "Remover",
"downloading_metadata": "Baixando metadados…", "downloading_metadata": "Baixando metadados…",
@@ -463,6 +470,7 @@
"download_sources_synced_successfully": "Fontes de download sincronizadas", "download_sources_synced_successfully": "Fontes de download sincronizadas",
"removed_download_source": "Fonte removida", "removed_download_source": "Fonte removida",
"removed_download_sources": "Fontes removidas", "removed_download_sources": "Fontes removidas",
"removed_all_download_sources": "Todas as fontes de download removidas",
"cancel_button_confirmation_delete_all_sources": "Não", "cancel_button_confirmation_delete_all_sources": "Não",
"confirm_button_confirmation_delete_all_sources": "Sim, excluir tudo", "confirm_button_confirmation_delete_all_sources": "Sim, excluir tudo",
"title_confirmation_delete_all_sources": "Remover todas as fontes de download", "title_confirmation_delete_all_sources": "Remover todas as fontes de download",
@@ -488,6 +496,7 @@
"blocked_users": "Usuários bloqueados", "blocked_users": "Usuários bloqueados",
"user_unblocked": "Usuário desbloqueado", "user_unblocked": "Usuário desbloqueado",
"enable_achievement_notifications": "Quando uma conquista é desbloqueada", "enable_achievement_notifications": "Quando uma conquista é desbloqueada",
"hydra_cloud": "Hydra Cloud",
"launch_minimized": "Iniciar o Hydra minimizado", "launch_minimized": "Iniciar o Hydra minimizado",
"disable_nsfw_alert": "Desativar alerta de conteúdo inapropriado", "disable_nsfw_alert": "Desativar alerta de conteúdo inapropriado",
"seed_after_download_complete": "Semear após a conclusão do download", "seed_after_download_complete": "Semear após a conclusão do download",
@@ -550,6 +559,7 @@
"show_download_speed_in_megabytes": "Exibir taxas de download em megabytes por segundo", "show_download_speed_in_megabytes": "Exibir taxas de download em megabytes por segundo",
"extract_files_by_default": "Extrair arquivos automaticamente após o download", "extract_files_by_default": "Extrair arquivos automaticamente após o download",
"enable_steam_achievements": "Habilitar busca por conquistas da Steam", "enable_steam_achievements": "Habilitar busca por conquistas da Steam",
"enable_new_download_options_badges": "Mostrar badges de novas opções de download",
"enable_achievement_custom_notifications": "Habilitar notificações customizadas de conquistas", "enable_achievement_custom_notifications": "Habilitar notificações customizadas de conquistas",
"top-left": "Superior esquerdo", "top-left": "Superior esquerdo",
"top-center": "Superior central", "top-center": "Superior central",
@@ -567,6 +577,9 @@
"test_notification": "Testar notificação", "test_notification": "Testar notificação",
"achievement_sound_volume": "Volume do som de conquista", "achievement_sound_volume": "Volume do som de conquista",
"select_achievement_sound": "Selecionar som de conquista", "select_achievement_sound": "Selecionar som de conquista",
"change_achievement_sound": "Alterar som de conquista",
"remove_achievement_sound": "Remover som de conquista",
"preview_sound": "Reproduzir som",
"select": "Selecionar", "select": "Selecionar",
"preview": "Reproduzir", "preview": "Reproduzir",
"remove": "Remover", "remove": "Remover",
@@ -574,7 +587,10 @@
"notification_preview": "Prévia da Notificação de Conquistas", "notification_preview": "Prévia da Notificação de Conquistas",
"enable_friend_start_game_notifications": "Quando um amigo iniciar um jogo", "enable_friend_start_game_notifications": "Quando um amigo iniciar um jogo",
"autoplay_trailers_on_game_page": "Reproduzir trailers automaticamente na página do jogo", "autoplay_trailers_on_game_page": "Reproduzir trailers automaticamente na página do jogo",
"hide_to_tray_on_game_start": "Ocultar o Hydra na bandeja ao iniciar um jogo" "hide_to_tray_on_game_start": "Ocultar o Hydra na bandeja ao iniciar um jogo",
"downloads": "Downloads",
"use_native_http_downloader": "Usar downloader HTTP nativo (experimental)",
"cannot_change_downloader_while_downloading": "Não é possível alterar esta configuração enquanto um download estiver em andamento"
}, },
"notifications": { "notifications": {
"download_complete": "Download concluído", "download_complete": "Download concluído",
@@ -680,6 +696,7 @@
"blocked_users": "Usuários bloqueados", "blocked_users": "Usuários bloqueados",
"unblock": "Desbloquear", "unblock": "Desbloquear",
"no_friends_added": "Você ainda não possui amigos adicionados", "no_friends_added": "Você ainda não possui amigos adicionados",
"no_friends_yet": "Você ainda não adicionou nenhum amigo",
"view_all": "Ver todos", "view_all": "Ver todos",
"load_more": "Carregar mais", "load_more": "Carregar mais",
"loading": "Carregando", "loading": "Carregando",

View File

@@ -26,6 +26,7 @@
"game_has_no_executable": "Файл запуска игры не выбран", "game_has_no_executable": "Файл запуска игры не выбран",
"sign_in": "Войти", "sign_in": "Войти",
"friends": "Друзья", "friends": "Друзья",
"notifications": "Уведомления",
"need_help": "Нужна помощь?", "need_help": "Нужна помощь?",
"favorites": "Избранное", "favorites": "Избранное",
"playable_button_title": "Показать только установленные игры.", "playable_button_title": "Показать только установленные игры.",
@@ -115,6 +116,7 @@
"downloading": "Загрузка {{title}}… ({{percentage}} завершено) - Окончание {{eta}} - {{speed}}", "downloading": "Загрузка {{title}}… ({{percentage}} завершено) - Окончание {{eta}} - {{speed}}",
"calculating_eta": "Загрузка {{title}}… ({{percentage}} завершено) - Подсчёт оставшегося времени…", "calculating_eta": "Загрузка {{title}}… ({{percentage}} завершено) - Подсчёт оставшегося времени…",
"checking_files": "Проверка файлов {{title}}… ({{percentage}} завершено)", "checking_files": "Проверка файлов {{title}}… ({{percentage}} завершено)",
"extracting": "Распаковка {{title}}… ({{percentage}} завершено)",
"installing_common_redist": "{{log}}…", "installing_common_redist": "{{log}}…",
"installation_complete": "Установка завершена", "installation_complete": "Установка завершена",
"installation_complete_message": "Библиотеки успешно установлены" "installation_complete_message": "Библиотеки успешно установлены"
@@ -148,7 +150,7 @@
"filter": "Поиск репаков", "filter": "Поиск репаков",
"requirements": "Системные требования", "requirements": "Системные требования",
"minimum": "Минимальные", "minimum": "Минимальные",
"recommended": "Рекомендуемые", "recommended": "Рекомендованные",
"paused": "Приостановлено", "paused": "Приостановлено",
"release_date": "Выпущено {{date}}", "release_date": "Выпущено {{date}}",
"publisher": "Издатель {{publisher}}", "publisher": "Издатель {{publisher}}",
@@ -173,6 +175,7 @@
"repacks_modal_description": "Выберите репак для загрузки", "repacks_modal_description": "Выберите репак для загрузки",
"select_folder_hint": "Чтобы изменить папку загрузок по умолчанию, откройте <0>Настройки</0>", "select_folder_hint": "Чтобы изменить папку загрузок по умолчанию, откройте <0>Настройки</0>",
"download_now": "Загрузить сейчас", "download_now": "Загрузить сейчас",
"loading": "Загрузка...",
"no_shop_details": "Не удалось получить описание", "no_shop_details": "Не удалось получить описание",
"download_options": "Источники", "download_options": "Источники",
"download_path": "Путь для загрузок", "download_path": "Путь для загрузок",
@@ -186,7 +189,6 @@
"downloader_not_configured": "Доступен, но не настроен", "downloader_not_configured": "Доступен, но не настроен",
"downloader_offline": "Ссылка недоступна", "downloader_offline": "Ссылка недоступна",
"downloader_not_available": "Недоступно", "downloader_not_available": "Недоступно",
"recommended": "Рекомендуется",
"go_to_settings": "Перейти в настройки", "go_to_settings": "Перейти в настройки",
"select_executable": "Выбрать", "select_executable": "Выбрать",
"no_executable_selected": "Файл не выбран", "no_executable_selected": "Файл не выбран",
@@ -208,6 +210,7 @@
"danger_zone_section_description": "Вы можете удалить эту игру из вашей библиотеки или файлы скачанные из Hydra", "danger_zone_section_description": "Вы можете удалить эту игру из вашей библиотеки или файлы скачанные из Hydra",
"download_in_progress": "Идёт загрузка", "download_in_progress": "Идёт загрузка",
"download_paused": "Загрузка приостановлена", "download_paused": "Загрузка приостановлена",
"extracting": "Распаковка",
"last_downloaded_option": "Последний вариант загрузки", "last_downloaded_option": "Последний вариант загрузки",
"new_download_option": "Новый", "new_download_option": "Новый",
"create_steam_shortcut": "Создать ярлык Steam", "create_steam_shortcut": "Создать ярлык Steam",
@@ -239,11 +242,11 @@
"show_more": "Показать больше", "show_more": "Показать больше",
"show_less": "Показать меньше", "show_less": "Показать меньше",
"reviews": "Отзывы", "reviews": "Отзывы",
"review_played_for": "Играли",
"leave_a_review": "Оставить отзыв", "leave_a_review": "Оставить отзыв",
"write_review_placeholder": "Поделитесь своими мыслями об этой игре...", "write_review_placeholder": "Поделитесь своими мыслями об этой игре...",
"sort_newest": "Сначала новые", "sort_newest": "Сначала новые",
"no_reviews_yet": "Пока нет отзывов", "no_reviews_yet": "Пока нет отзывов",
"review_played_for": "Играли",
"be_first_to_review": "Станьте первым, кто поделится своими мыслями об этой игре!", "be_first_to_review": "Станьте первым, кто поделится своими мыслями об этой игре!",
"sort_oldest": "Сначала старые", "sort_oldest": "Сначала старые",
"sort_highest_score": "Высший балл", "sort_highest_score": "Высший балл",
@@ -368,8 +371,6 @@
"audio": "Аудио", "audio": "Аудио",
"filter_by_source": "Фильтр по источнику", "filter_by_source": "Фильтр по источнику",
"no_repacks_found": "Источники для этой игры не найдены", "no_repacks_found": "Источники для этой игры не найдены",
"show": "Показать",
"hide": "Скрыть",
"delete_review": "Удалить отзыв", "delete_review": "Удалить отзыв",
"remove_review": "Удалить отзыв", "remove_review": "Удалить отзыв",
"delete_review_modal_title": "Вы уверены, что хотите удалить свой отзыв?", "delete_review_modal_title": "Вы уверены, что хотите удалить свой отзыв?",
@@ -381,7 +382,9 @@
"show_translation": "Показать перевод", "show_translation": "Показать перевод",
"show_original_translated_from": "Показать оригинал (переведено с {{language}})", "show_original_translated_from": "Показать оригинал (переведено с {{language}})",
"hide_original": "Скрыть оригинал", "hide_original": "Скрыть оригинал",
"review_from_blocked_user": "Отзыв от заблокированного пользователя" "review_from_blocked_user": "Отзыв от заблокированного пользователя",
"show": "Показать",
"hide": "Скрыть"
}, },
"activation": { "activation": {
"title": "Активировать Hydra", "title": "Активировать Hydra",
@@ -400,6 +403,10 @@
"completed": "Завершено", "completed": "Завершено",
"removed": "Не скачано", "removed": "Не скачано",
"cancel": "Отмена", "cancel": "Отмена",
"cancel_download": "Отменить загрузку?",
"cancel_download_description": "Вы уверены, что хотите отменить эту загрузку? Все загруженные файлы будут удалены.",
"keep_downloading": "Нет, продолжить загрузку",
"yes_cancel": "Да, отменить",
"filter": "Поиск загруженных игр", "filter": "Поиск загруженных игр",
"remove": "Удалить", "remove": "Удалить",
"downloading_metadata": "Загрузка метаданных…", "downloading_metadata": "Загрузка метаданных…",
@@ -420,7 +427,13 @@
"resume_seeding": "Продолжить раздачу", "resume_seeding": "Продолжить раздачу",
"options": "Управлять", "options": "Управлять",
"extract": "Распаковать файлы", "extract": "Распаковать файлы",
"extracting": "Распаковка файлов…" "extracting": "Распаковка файлов…",
"delete_archive_title": "Хотите удалить {{fileName}}?",
"delete_archive_description": "Файл был успешно распакован и больше не нужен.",
"yes": "Да",
"no": "Нет",
"network": "СЕТЬ",
"peak": "ПИК"
}, },
"settings": { "settings": {
"downloads_path": "Путь загрузок", "downloads_path": "Путь загрузок",
@@ -556,6 +569,7 @@
"show_download_speed_in_megabytes": "Показать скорость загрузки в мегабайтах в секунду", "show_download_speed_in_megabytes": "Показать скорость загрузки в мегабайтах в секунду",
"extract_files_by_default": "Извлекать файлы по умолчанию после загрузки", "extract_files_by_default": "Извлекать файлы по умолчанию после загрузки",
"enable_steam_achievements": "Включить поиск достижений Steam", "enable_steam_achievements": "Включить поиск достижений Steam",
"enable_new_download_options_badges": "Показывать значки новых вариантов загрузки",
"achievement_custom_notification_position": "Позиция уведомлений достижений", "achievement_custom_notification_position": "Позиция уведомлений достижений",
"top-left": "Верхний левый угол", "top-left": "Верхний левый угол",
"top-center": "Верхний центр", "top-center": "Верхний центр",
@@ -573,6 +587,9 @@
"test_notification": "Тестовое уведомление", "test_notification": "Тестовое уведомление",
"achievement_sound_volume": "Громкость звука достижения", "achievement_sound_volume": "Громкость звука достижения",
"select_achievement_sound": "Выбрать звук достижения", "select_achievement_sound": "Выбрать звук достижения",
"change_achievement_sound": "Изменить звук достижения",
"remove_achievement_sound": "Удалить звук достижения",
"preview_sound": "Предпросмотр звука",
"select": "Выбрать", "select": "Выбрать",
"preview": "Предпросмотр", "preview": "Предпросмотр",
"remove": "Удалить", "remove": "Удалить",
@@ -580,7 +597,10 @@
"notification_preview": "Предварительный просмотр уведомления о достижении", "notification_preview": "Предварительный просмотр уведомления о достижении",
"enable_friend_start_game_notifications": "Когда друг начинает играть в игру", "enable_friend_start_game_notifications": "Когда друг начинает играть в игру",
"autoplay_trailers_on_game_page": "Автоматически начинать воспроизведение трейлеров на странице игры", "autoplay_trailers_on_game_page": "Автоматически начинать воспроизведение трейлеров на странице игры",
"hide_to_tray_on_game_start": "Скрывать Hydra в трей при запуске игры" "hide_to_tray_on_game_start": "Скрывать Hydra в трей при запуске игры",
"downloads": "Загрузки",
"use_native_http_downloader": "Использовать встроенный HTTP-загрузчик (экспериментально)",
"cannot_change_downloader_while_downloading": "Нельзя изменить эту настройку во время загрузки"
}, },
"notifications": { "notifications": {
"download_complete": "Загрузка завершена", "download_complete": "Загрузка завершена",
@@ -675,6 +695,7 @@
"blocked_users": "Заблокированные пользователи", "blocked_users": "Заблокированные пользователи",
"unblock": "Разблокировать", "unblock": "Разблокировать",
"no_friends_added": "Вы ещё не добавили ни одного друга", "no_friends_added": "Вы ещё не добавили ни одного друга",
"no_friends_yet": "Вы ещё не добавили ни одного друга",
"view_all": "Показать все", "view_all": "Показать все",
"load_more": "Загрузить еще", "load_more": "Загрузить еще",
"loading": "Загрузка", "loading": "Загрузка",
@@ -700,7 +721,7 @@
"report_reason_spam": "Спам", "report_reason_spam": "Спам",
"report_reason_other": "Другое", "report_reason_other": "Другое",
"profile_reported": "Жалоба на профиль отправлена", "profile_reported": "Жалоба на профиль отправлена",
"your_friend_code": "Код вашего друга:", "your_friend_code": "Ваш код друга:",
"copy_friend_code": "Копировать код друга", "copy_friend_code": "Копировать код друга",
"copied": "Скопировано!", "copied": "Скопировано!",
"upload_banner": "Загрузить баннер", "upload_banner": "Загрузить баннер",
@@ -729,10 +750,30 @@
"karma": "Карма", "karma": "Карма",
"karma_count": "карма", "karma_count": "карма",
"user_reviews": "Отзывы", "user_reviews": "Отзывы",
"delete_review": "Удалить отзыв",
"loading_reviews": "Загрузка отзывов...", "loading_reviews": "Загрузка отзывов...",
"wrapped_2025": "Wrapped 2025", "wrapped_2025": "Wrapped 2025"
"no_reviews": "Пока нет отзывов", },
"delete_review": "Удалить отзыв" "library": {
"library": "Библиотека",
"play": "Играть",
"download": "Скачать",
"downloading": "Скачивание",
"game": "игра",
"games": "игры",
"grid_view": "Вид сетки",
"compact_view": "Компактный вид",
"large_view": "Большой вид",
"no_games_title": "Ваша библиотека пуста",
"no_games_description": "Добавьте игры из каталога или скачайте их, чтобы начать",
"amount_hours": "{{amount}} часов",
"amount_minutes": "{{amount}} минут",
"amount_hours_short": "{{amount}}ч",
"amount_minutes_short": "{{amount}}м",
"manual_playtime_tooltip": "Время игры было обновлено вручную",
"all_games": "Все игры",
"recently_played": "Недавно сыгранные",
"favorites": "Избранное"
}, },
"achievement": { "achievement": {
"achievement_unlocked": "Достижение разблокировано", "achievement_unlocked": "Достижение разблокировано",
@@ -763,27 +804,6 @@
"learn_more": "Подробнее", "learn_more": "Подробнее",
"debrid_description": "Скачивайте в 4 раза быстрее с Nimbus" "debrid_description": "Скачивайте в 4 раза быстрее с Nimbus"
}, },
"library": {
"library": "Библиотека",
"play": "Играть",
"download": "Скачать",
"downloading": "Скачивание",
"game": "игра",
"games": "игры",
"grid_view": "Вид сетки",
"compact_view": "Компактный вид",
"large_view": "Большой вид",
"no_games_title": "Ваша библиотека пуста",
"no_games_description": "Добавьте игры из каталога или скачайте их, чтобы начать",
"amount_hours": "{{amount}} часов",
"amount_minutes": "{{amount}} минут",
"amount_hours_short": "{{amount}}ч",
"amount_minutes_short": "{{amount}}м",
"manual_playtime_tooltip": "Время игры было обновлено вручную",
"all_games": "Все игры",
"recently_played": "Недавно сыгранные",
"favorites": "Избранное"
},
"notifications_page": { "notifications_page": {
"title": "Уведомления", "title": "Уведомления",
"mark_all_as_read": "Отметить все как прочитанные", "mark_all_as_read": "Отметить все как прочитанные",

View File

@@ -0,0 +1,844 @@
{
"language_name": "Slovenščina",
"app": {
"successfully_signed_in": "Uspešno ste se prijavili"
},
"home": {
"surprise_me": "Preseneti me",
"no_results": "Ni najdenih rezultatov",
"start_typing": "Začnite tipkati za iskanje...",
"hot": "Trenutno vroče",
"weekly": "📅 Najboljše igre tedna",
"achievements": "🏆 Igre za premagati"
},
"sidebar": {
"catalogue": "Katalog",
"library": "Knjižnica",
"downloads": "Prenosi",
"settings": "Nastavitve",
"my_library": "Moja knjižnica",
"downloading_metadata": "{{title}} (Prenos metapodatkov…)",
"paused": "{{title}} (V premoru)",
"downloading": "{{title}} ({{percentage}} - Prenos…)",
"filter": "Filtriraj knjižnico",
"home": "Domov",
"queued": "{{title}} (V čakalni vrsti)",
"game_has_no_executable": "Igra nima izbrane izvršljive datoteke",
"sign_in": "Prijavite se",
"friends": "Prijatelji",
"notifications": "Obvestila",
"need_help": "Potrebujete pomoč?",
"favorites": "Priljubljene",
"playable_button_title": "Pokaži le igre, ki jih lahko igrate zdaj",
"add_custom_game_tooltip": "Dodaj igro po meri",
"show_playable_only_tooltip": "Pokaži samo igrljive",
"custom_game_modal": "Dodaj igro po meri",
"custom_game_modal_description": "Dodajte igro po meri v vašo knjižnico z izbiro izvršljive datoteke",
"custom_game_modal_executable_path": "Pot do izvršljive datoteke",
"custom_game_modal_select_executable": "Izberite izvršljivo datoteko",
"custom_game_modal_title": "Naslov",
"custom_game_modal_enter_title": "Vnesite naslov",
"custom_game_modal_browse": "Brskaj",
"custom_game_modal_cancel": "Prekliči",
"custom_game_modal_add": "Dodaj igro",
"custom_game_modal_adding": "Dodajanje igre...",
"custom_game_modal_success": "Igra po meri je bila uspešno dodana",
"custom_game_modal_failed": "Dodajanje igre po meri ni uspelo",
"custom_game_modal_executable": "Izvršljiva datoteka",
"edit_game_modal": "Prilagodi sredstva",
"edit_game_modal_description": "Prilagodite sredstva in podrobnosti igre",
"edit_game_modal_title": "Naslov",
"edit_game_modal_enter_title": "Vnesite naslov",
"edit_game_modal_image": "Slika",
"edit_game_modal_select_image": "Izberite sliko",
"edit_game_modal_browse": "Brskaj",
"edit_game_modal_image_preview": "Predogled slike",
"edit_game_modal_icon": "Ikona",
"edit_game_modal_select_icon": "Izberite ikono",
"edit_game_modal_icon_preview": "Predogled ikone",
"edit_game_modal_logo": "Logotip",
"edit_game_modal_select_logo": "Izberite logotip",
"edit_game_modal_logo_preview": "Predogled logotipa",
"edit_game_modal_hero": "Hero knjižnice",
"edit_game_modal_select_hero": "Izberite sliko hero knjižnice",
"edit_game_modal_hero_preview": "Predogled hero slike knjižnice",
"edit_game_modal_cancel": "Prekliči",
"edit_game_modal_update": "Posodobi",
"edit_game_modal_updating": "Posodabljanje...",
"edit_game_modal_fill_required": "Prosimo, izpolnite vsa obvezna polja",
"edit_game_modal_success": "Sredstva so bila uspešno posodobljena",
"edit_game_modal_failed": "Posodabljanje sredstev ni uspelo",
"edit_game_modal_image_filter": "Slika",
"edit_game_modal_icon_resolution": "Priporočena resolucija: 256x256px",
"edit_game_modal_logo_resolution": "Priporočena resolucija: 640x360px",
"edit_game_modal_hero_resolution": "Priporočena resolucija: 1920x620px",
"edit_game_modal_assets": "Sredstva",
"edit_game_modal_drop_icon_image_here": "Spustite ikono tukaj",
"edit_game_modal_drop_logo_image_here": "Spustite logotip tukaj",
"edit_game_modal_drop_hero_image_here": "Spustite hero sliko tukaj",
"edit_game_modal_drop_to_replace_icon": "Spustite za zamenjavo ikone",
"edit_game_modal_drop_to_replace_logo": "Spustite za zamenjavo logotipa",
"edit_game_modal_drop_to_replace_hero": "Spustite za zamenjavo hero slike",
"install_decky_plugin": "Namesti Decky vtičnik",
"update_decky_plugin": "Posodobi Decky vtičnik",
"decky_plugin_installed_version": "Decky vtičnik (v{{version}})",
"install_decky_plugin_title": "Namesti Hydra Decky vtičnik",
"install_decky_plugin_message": "To bo preneslo in namestilo Hydra vtičnik za Decky Loader. To lahko zahteva povišane pravice. Nadaljujem?",
"update_decky_plugin_title": "Posodobi Hydra Decky vtičnik",
"update_decky_plugin_message": "Na voljo je nova različica Hydra Decky vtičnika. Ali želite posodobiti zdaj?",
"decky_plugin_installed": "Decky vtičnik v{{version}} je bil uspešno nameščen",
"decky_plugin_installation_failed": "Namestitev Decky vtičnika ni uspela: {{error}}",
"decky_plugin_installation_error": "Napaka pri nameščanju Decky vtičnika: {{error}}",
"confirm": "Potrdi",
"cancel": "Prekliči"
},
"header": {
"search": "Išči igre",
"search_library": "Išči v knjižnici",
"recent_searches": "Nedavna iskanja",
"suggestions": "Predlogi",
"clear_history": "Počisti",
"remove_from_history": "Odstrani iz zgodovine",
"loading": "Nalaganje...",
"no_results": "Ni rezultatov",
"home": "Domov",
"catalogue": "Katalog",
"library": "Knjižnica",
"downloads": "Prenosi",
"search_results": "Rezultati iskanja",
"settings": "Nastavitve",
"version_available_install": "Različica {{version}} je na voljo. Kliknite tukaj za ponovni zagon in namestitev.",
"version_available_download": "Različica {{version}} je na voljo. Kliknite tukaj za prenos."
},
"bottom_panel": {
"no_downloads_in_progress": "Ni prenosa v teku",
"downloading_metadata": "Prenos metapodatkov {{title}}…",
"downloading": "Prenos {{title}}… ({{percentage}} končano) - Čas {{eta}} - {{speed}}",
"calculating_eta": "Prenos {{title}}… ({{percentage}} končano) - Izračun preostalega časa…",
"checking_files": "Preverjanje datotek {{title}}… ({{percentage}} končano)",
"extracting": "Razpakiranje {{title}}… ({{percentage}} končano)",
"installing_common_redist": "{{log}}…",
"installation_complete": "Namestitev zaključena",
"installation_complete_message": "Skupni redistributables so bili uspešno nameščeni"
},
"catalogue": {
"search": "Filtriraj…",
"developers": "Razvijalci",
"genres": "Žanri",
"tags": "Oznake",
"publishers": "Izdajatelji",
"download_sources": "Viri prenosa",
"result_count": "{{resultCount}} rezultatov",
"filter_count": "{{filterCount}} na voljo",
"clear_filters": "Počisti {{filterCount}} izbranih"
},
"game_details": {
"open_download_options": "Odpri možnosti prenosa",
"download_options_zero": "Ni možnosti prenosa",
"download_options_one": "{{count}} možnost prenosa",
"download_options_other": "{{count}} možnosti prenosa",
"updated_at": "Posodobljeno {{updated_at}}",
"install": "Namesti",
"resume": "Nadaljuj",
"pause": "Premor",
"cancel": "Prekliči",
"remove": "Odstrani",
"space_left_on_disk": "{{space}} prosto na disku",
"eta": "Zaključek {{eta}}",
"calculating_eta": "Izračun preostalega časa…",
"downloading_metadata": "Prenos metapodatkov…",
"filter": "Filtriraj repake",
"requirements": "Sistemske zahteve",
"minimum": "Minimum",
"recommended": "Priporočeno",
"paused": "V premoru",
"release_date": "Izid dne {{date}}",
"publisher": "Objavljeno s strani {{publisher}}",
"hours": "ur",
"minutes": "minut",
"amount_hours": "{{amount}} ur",
"amount_minutes": "{{amount}} minut",
"accuracy": "{{accuracy}}% natančnost",
"add_to_library": "Dodaj v knjižnico",
"already_in_library": "Že v knjižnici",
"remove_from_library": "Odstrani iz knjižnice",
"no_downloads": "Ni razpoložljivih prenosov",
"play_time": "Odigrano {{amount}}",
"last_time_played": "Nazadnje igrano {{period}}",
"not_played_yet": "Še niste igrali {{title}}",
"next_suggestion": "Naslednji predlog",
"play": "Igraj",
"deleting": "Brisanje namestitvenega programa…",
"close": "Zapri",
"playing_now": "Trenutno igranje",
"change": "Spremeni",
"repacks_modal_description": "Izberite repak, ki ga želite prenesti",
"select_folder_hint": "Za spremembo privzete mape pojdite v <0>Nastavitve</0>",
"download_now": "Prenesi zdaj",
"loading": "Nalaganje...",
"no_shop_details": "Podatkov o trgovini ni bilo mogoče pridobiti.",
"download_options": "Možnosti prenosa",
"download_path": "Pot prenosa",
"previous_screenshot": "Prejšnji posnetek zaslona",
"next_screenshot": "Naslednji posnetek zaslona",
"screenshot": "Posnetek zaslona {{number}}",
"open_screenshot": "Odpri posnetek zaslona {{number}}",
"download_settings": "Nastavitve prenosa",
"downloader": "Prenosnik",
"downloader_online": "Spletno",
"downloader_not_configured": "Na voljo, vendar ni nastavljeno",
"downloader_offline": "Povezava je brez povezave",
"downloader_not_available": "Ni na voljo",
"recommended": "Priporočeno",
"go_to_settings": "Pojdi v nastavitve",
"select_executable": "Izberi",
"no_executable_selected": "Ni izbrane izvršljive datoteke",
"open_folder": "Odpri mapo",
"open_download_location": "Poglej prenesene datoteke",
"create_shortcut": "Ustvari bližnjico na namizju",
"create_shortcut_simple": "Ustvari bližnjico",
"clear": "Počisti",
"remove_files": "Odstrani datoteke",
"remove_from_library_title": "Ali ste prepričani?",
"remove_from_library_description": "To bo odstranilo {{game}} iz vaše knjižnice",
"options": "Možnosti",
"properties": "Lastnosti",
"executable_section_title": "Izvršljiva datoteka",
"executable_section_description": "Pot do datoteke, ki se bo izvedla ob kliku na \"Igraj\"",
"downloads_section_title": "Prenosi",
"downloads_section_description": "Preverite posodobitve ali druge različice te igre",
"danger_zone_section_title": "Nevarno območje",
"danger_zone_section_description": "Odstranite to igro iz knjižnice ali datoteke, ki jih je prenesel Hydra",
"download_in_progress": "Prenos v teku",
"download_paused": "Prenos v premoru",
"extracting": "Razpakiranje",
"last_downloaded_option": "Zadnja prenesena možnost",
"new_download_option": "Novo",
"create_steam_shortcut": "Ustvari Steam bližnjico",
"create_shortcut_success": "Bližnjica je bila uspešno ustvarjena",
"you_might_need_to_restart_steam": "Morda boste morali ponovno zagnati Steam, da vidite spremembe",
"create_shortcut_error": "Napaka pri ustvarjanju bližnjice",
"add_to_favorites": "Dodaj med priljubljene",
"remove_from_favorites": "Odstrani iz priljubljenih",
"failed_update_favorites": "Posodabljanje priljubljenih ni uspelo",
"game_removed_from_library": "Igra odstranjena iz knjižnice",
"failed_remove_from_library": "Odstranjevanje iz knjižnice ni uspelo",
"files_removed_success": "Datoteke so bile uspešno odstranjene",
"failed_remove_files": "Odstranjevanje datotek ni uspelo",
"nsfw_content_title": "Ta igra vsebuje neprimerno vsebino",
"nsfw_content_description": "{{title}} vsebuje vsebino, ki morda ni primerna za vse starosti. Ali ste prepričani, da želite nadaljevati?",
"allow_nsfw_content": "Nadaljuj",
"refuse_nsfw_content": "Nazaj",
"stats": "Statistika",
"download_count": "Prenosi",
"player_count": "Aktivni igralci",
"rating_count": "Ocena",
"download_error": "Ta možnost prenosa ni na voljo",
"download": "Prenesi",
"executable_path_in_use": "Izvršljiva datoteka že uporablja \"{{game}}\"",
"warning": "Opozorilo:",
"hydra_needs_to_remain_open": "Za ta prenos mora Hydra ostati odprta, dokler ni končana. Če se Hydra zapre pred končanim prenosom, boste izgubili napredek.",
"achievements": "Dosežki",
"achievements_count": "Dosežki {{unlockedCount}}/{{achievementsCount}}",
"show_more": "Pokaži več",
"show_less": "Pokaži manj",
"reviews": "Mnenja",
"review_played_for": "Odigrano za",
"leave_a_review": "Oddajte mnenje",
"write_review_placeholder": "Delite svoje misli o tej igri...",
"sort_newest": "Najnovejše",
"no_reviews_yet": "Ni še mnenj",
"be_first_to_review": "Bodite prvi, ki delite svoje misli o tej igri!",
"sort_oldest": "Najstarejše",
"sort_highest_score": "Najvišja ocena",
"sort_lowest_score": "Najnižja ocena",
"sort_most_voted": "Največ glasov",
"rating": "Ocena",
"rating_stats": "Ocena",
"rating_very_negative": "Zelo negativno",
"rating_negative": "Negativno",
"rating_neutral": "Nevtralno",
"rating_positive": "Pozitivno",
"rating_very_positive": "Zelo pozitivno",
"submit_review": "Pošlji",
"submitting": "Pošiljanje...",
"review_submitted_successfully": "Mnenje je bilo uspešno poslano!",
"review_submission_failed": "Pošiljanje mnenja ni uspelo. Prosimo, poskusite znova.",
"review_cannot_be_empty": "Polje mnenja ne sme biti prazno.",
"review_deleted_successfully": "Mnenje je bilo uspešno izbrisano.",
"review_deletion_failed": "Brisanje mnenja ni uspelo. Prosimo, poskusite znova.",
"loading_reviews": "Nalagam mnenja...",
"loading_more_reviews": "Nalagam več mnenj...",
"load_more_reviews": "Naloži več mnenj",
"you_seemed_to_enjoy_this_game": "Zdi se, da uživate v tej igri",
"would_you_recommend_this_game": "Bi radi oddali mnenje o tej igri?",
"yes": "Da",
"maybe_later": "Mogoče kasneje",
"cloud_save": "Shranjevanje v oblaku",
"cloud_save_description": "Shranjujte napredek v oblak in nadaljujte igranje na katerikoli napravi",
"backups": "Varnostne kopije",
"install_backup": "Namesti",
"delete_backup": "Izbriši",
"create_backup": "Nova varnostna kopija",
"last_backup_date": "Zadnja varnostna kopija {{date}}",
"no_backup_preview": "Ni shranjenih iger za ta naslov",
"restoring_backup": "Obnavljanje varnostne kopije ({{progress}} končano)…",
"uploading_backup": "Nalaganje varnostne kopije…",
"no_backups": "Za to igro še niste ustvarili varnostnih kopij",
"backup_uploaded": "Varnostna kopija naložena",
"backup_failed": "Varnostna kopija ni uspela",
"backup_deleted": "Varnostna kopija izbrisana",
"backup_restored": "Varnostna kopija obnovljena",
"see_all_achievements": "Poglej vse dosežke",
"sign_in_to_see_achievements": "Prijavite se za ogled dosežkov",
"mapping_method_automatic": "Samodejno",
"mapping_method_manual": "Ročno",
"mapping_method_label": "Način mapiranja",
"files_automatically_mapped": "Datoteke so samodejno preslikane",
"no_backups_created": "Za to igro ni ustvarjenih varnostnih kopij",
"manage_files": "Upravljaj datoteke",
"loading_save_preview": "Iskanje shranjenih iger…",
"wine_prefix": "Wine predpona",
"wine_prefix_description": "Wine predpona, uporabljena za zagon te igre",
"launch_options": "Možnosti zagona",
"launch_options_description": "Napredni uporabniki lahko vpišejo spremembe v možnosti zagona (eksperimentalna funkcija)",
"launch_options_placeholder": "Ni določenega parametra",
"no_download_option_info": "Ni razpoložljivih informacij",
"backup_deletion_failed": "Brisanje varnostne kopije ni uspelo",
"max_number_of_artifacts_reached": "Doseženo je največje število varnostnih kopij za to igro",
"achievements_not_sync": "Oglejte si, kako sinhronizirati svoje dosežke",
"manage_files_description": "Upravljajte, katere datoteke bodo varnostno kopirane in obnovljene",
"select_folder": "Izberite mapo",
"backup_from": "Varnostna kopija od {{date}}",
"automatic_backup_from": "Samodejna varnostna kopija od {{date}}",
"enable_automatic_cloud_sync": "Omogoči samodejno sinhronizacijo v oblaku",
"custom_backup_location_set": "Nastavljena je po meri lokacija varnostne kopije",
"no_directory_selected": "Ni izbrane mape",
"no_write_permission": "V to mapo ni mogoče prenesti. Kliknite tukaj za več informacij.",
"reset_achievements": "Ponastavi dosežke",
"reset_achievements_description": "To bo ponastavilo vse dosežke za {{game}}",
"reset_achievements_title": "Ali ste prepričani?",
"reset_achievements_success": "Dosežki so bili uspešno ponastavljeni",
"reset_achievements_error": "Ponastavitev dosežkov ni uspela",
"download_error_gofile_quota_exceeded": "Presegli ste mesečno kvoto Gofile. Prosimo, počakajte, da se kvota ponastavi.",
"download_error_real_debrid_account_not_authorized": "Vaš račun Real-Debrid ni pooblaščen za nove prenose. Preverite nastavitve računa in poskusite znova.",
"download_error_not_cached_on_real_debrid": "Ta prenos ni na voljo v Real-Debrid in preverjanje statusa prenosa iz Real-Debrid še ni na voljo.",
"update_playtime_title": "Posodobi čas igranja",
"update_playtime_description": "Ročno posodobite čas igranja za {{game}}",
"update_playtime": "Posodobi čas igranja",
"update_playtime_success": "Čas igranja je bil uspešno posodobljen",
"update_playtime_error": "Posodabljanje časa igranja ni uspelo",
"update_game_playtime": "Posodobi čas igranja igre",
"manual_playtime_warning": "Vaše ure bodo označene kot ročno posodobljene, tega ni mogoče razveljaviti.",
"manual_playtime_tooltip": "Ta čas igranja je bil ročno posodobljen",
"download_error_not_cached_on_torbox": "Ta prenos ni na voljo v TorBox in preverjanje statusa prenosa iz TorBox še ni na voljo.",
"download_error_not_cached_on_hydra": "Ta prenos ni na voljo v Nimbus.",
"game_removed_from_favorites": "Igra odstranjena iz priljubljenih",
"game_added_to_favorites": "Igra dodana med priljubljene",
"game_removed_from_pinned": "Igra odstranjena iz pripetih",
"game_added_to_pinned": "Igra pripeta",
"automatically_extract_downloaded_files": "Samodejno razpakiraj prenesene datoteke",
"create_start_menu_shortcut": "Ustvari bližnjico v Start meniju",
"invalid_wine_prefix_path": "Neveljavna pot Wine predpone",
"invalid_wine_prefix_path_description": "Pot do Wine predpone je neveljavna. Preverite pot in poskusite znova.",
"missing_wine_prefix": "Wine predpona je potrebna za ustvarjanje varnostne kopije na Linuxu",
"artifact_renamed": "Varnostna kopija je bila uspešno preimenovana",
"rename_artifact": "Preimenuj varnostno kopijo",
"rename_artifact_description": "Preimenujte varnostno kopijo v bolj opisno ime",
"artifact_name_label": "Ime varnostne kopije",
"artifact_name_placeholder": "Vnesite ime varnostne kopije",
"save_changes": "Shrani spremembe",
"required_field": "To polje je obvezno",
"max_length_field": "To polje mora biti krajše od {{length}} znakov",
"freeze_backup": "Pripni, da jo samodejne varnostne kopije ne prepišejo",
"unfreeze_backup": "Odpni",
"backup_frozen": "Varnostna kopija je pripeta",
"backup_unfrozen": "Varnostna kopija je odprijeta",
"backup_freeze_failed": "Pripenjanje varnostne kopije ni uspelo",
"backup_freeze_failed_description": "Morate pustiti vsaj en prost prostor za samodejne varnostne kopije",
"edit_game_modal_button": "Prilagodi sredstva igre",
"game_details": "Podrobnosti igre",
"currency_symbol": "$",
"currency_country": "us",
"prices": "Cene",
"no_prices_found": "Ni najdenih cen",
"view_all_prices": "Kliknite za ogled vseh cen",
"retail_price": "Maloprodajna cena",
"keyshop_price": "Cena v trgovini s ključki",
"historical_retail": "Zgodovinska maloprodajna cena",
"historical_keyshop": "Zgodovinska cena v trgovini s ključki",
"language": "Jezik",
"caption": "Naslov",
"audio": "Zvok",
"filter_by_source": "Filtriraj po viru",
"no_repacks_found": "Za to igro ni najdenih virov",
"delete_review": "Izbriši mnenje",
"remove_review": "Odstrani mnenje",
"delete_review_modal_title": "Ali ste prepričani, da želite izbrisati svoje mnenje?",
"delete_review_modal_description": "Tega dejanja ni mogoče razveljaviti.",
"delete_review_modal_delete_button": "Izbriši",
"delete_review_modal_cancel_button": "Prekliči",
"vote_failed": "Glasovanja ni uspelo. Prosimo, poskusite znova.",
"show_original": "Pokaži original",
"show_translation": "Pokaži prevod",
"show_original_translated_from": "Pokaži original (prevedeno iz {{language}})",
"hide_original": "Skrij original",
"review_from_blocked_user": "Mnenje blokiranega uporabnika",
"show": "Pokaži",
"hide": "Skrij"
},
"activation": {
"title": "Aktiviraj Hydra",
"installation_id": "ID namestitve:",
"enter_activation_code": "Vnesite aktivacijsko kodo",
"message": "Če ne veste, kje naj to pridobite, potem tega ne bi smeli imeti.",
"activate": "Aktiviraj",
"loading": "Nalaganje…"
},
"downloads": {
"resume": "Nadaljuj",
"pause": "Premor",
"eta": "Zaključek {{eta}}",
"paused": "V premoru",
"verifying": "Preverjanje…",
"completed": "Dokončano",
"removed": "Ni preneseno",
"cancel": "Prekliči",
"cancel_download": "Prekliči prenos?",
"cancel_download_description": "Ali ste prepričani, da želite prekiniti ta prenos? Vse prenesene datoteke bodo izbrisane.",
"keep_downloading": "Ne, nadaljuj prenos",
"yes_cancel": "Da, prekliči",
"filter": "Filtriraj prenesene igre",
"remove": "Odstrani",
"downloading_metadata": "Prenos metapodatkov…",
"deleting": "Brisanje namestitvenega programa…",
"delete": "Odstrani namestitveni program",
"delete_modal_title": "Ali ste prepričani?",
"delete_modal_description": "To bo odstranilo vse namestitvene datoteke z računalnika",
"install": "Namesti",
"download_in_progress": "V teku",
"queued_downloads": "Prenosi v čakalni vrsti",
"downloads_completed": "Dokončano",
"queued": "V čakalni vrsti",
"no_downloads_title": "Tako prazno",
"no_downloads_description": "Še niste prenesli ničesar z Hydra, a nikoli ni prepozno začeti.",
"checking_files": "Preverjanje datotek…",
"seeding": "Sejanje",
"stop_seeding": "Ustavi sejanje",
"resume_seeding": "Nadaljuj sejanje",
"options": "Upravljaj",
"extract": "Razpakiraj datoteke",
"extracting": "Razpakiranje datotek…",
"delete_archive_title": "Ali želite izbrisati {{fileName}}?",
"delete_archive_description": "Datoteka je bila uspešno razpakirana in ni več potrebna.",
"yes": "Da",
"no": "Ne",
"network": "OMREŽJE",
"peak": "VRH"
},
"settings": {
"downloads_path": "Pot prenosa",
"change": "Posodobi",
"notifications": "Obvestila",
"enable_download_notifications": "Ko je prenos končan",
"enable_repack_list_notifications": "Ko je dodan nov repack",
"real_debrid_api_token_label": "Real-Debrid API žeton",
"quit_app_instead_hiding": "Ne skrij Hydre pri zapiranju",
"launch_with_system": "Zaženi Hydra ob zagonu sistema",
"general": "Splošno",
"behavior": "Obnašanje",
"download_sources": "Viri prenosa",
"language": "Jezik",
"api_token": "API žeton",
"enable_real_debrid": "Omogoči Real-Debrid",
"real_debrid_description": "Real-Debrid je neomejen prenašalnik, ki vam omogoča hitro prenašanje datotek, omejeno le s hitrostjo vašega interneta.",
"debrid_invalid_token": "Neveljaven API žeton",
"debrid_api_token_hint": "Žeton API lahko dobite <0>tukaj</0>",
"real_debrid_free_account_error": "Račun \"{{username}}\" je brezplačen. Prosimo, naročite se na Real-Debrid",
"debrid_linked_message": "Račun \"{{username}}\" povezan",
"save_changes": "Shrani spremembe",
"changes_saved": "Spremembe uspešno shranjene",
"download_sources_description": "Hydra bo pridobila povezave za prenos iz teh virov. URL vira mora biti neposredna povezava do .json datoteke, ki vsebuje povezave za prenos.",
"validate_download_source": "Preveri",
"remove_download_source": "Odstrani",
"add_download_source": "Dodaj vir",
"adding": "Dodajanje…",
"failed_add_download_source": "Dodajanje vira za prenos ni uspelo. Poskusite znova.",
"download_source_already_exists": "Ta URL vira za prenos že obstaja.",
"download_count_zero": "Ni možnosti prenosa",
"download_count_one": "{{countFormatted}} možnost prenosa",
"download_count_other": "{{countFormatted}} možnosti prenosa",
"download_source_url": "URL vira za prenos",
"add_download_source_description": "Vstavite URL .json datoteke",
"download_source_up_to_date": "Posodobljeno",
"download_source_errored": "Napaka",
"download_source_pending_matching": "Kmalu posodobljeno",
"download_source_matched": "Posodobljeno",
"download_source_matching": "Posodabljanje",
"download_source_failed": "Napaka",
"download_source_no_information": "Ni podatkov na voljo",
"sync_download_sources": "Sinhroniziraj vire",
"removed_download_source": "Vir prenosa odstranjen",
"removed_download_sources": "Viri prenosa odstranjeni",
"removed_all_download_sources": "Vsi viri prenosa odstranjeni",
"download_sources_synced_successfully": "Vsi viri prenosa so sinhronizirani",
"cancel_button_confirmation_delete_all_sources": "Ne",
"confirm_button_confirmation_delete_all_sources": "Da, izbriši vse",
"title_confirmation_delete_all_sources": "Izbriši vse vire prenosa",
"description_confirmation_delete_all_sources": "Izbrišete vse vire prenosa",
"button_delete_all_sources": "Odstrani vse",
"added_download_source": "Vir prenosa dodan",
"download_sources_synced": "Vsi viri prenosa so sinhronizirani",
"insert_valid_json_url": "Vnesite veljaven JSON URL",
"found_download_option_zero": "Ni možnosti prenosa",
"found_download_option_one": "Najdena {{countFormatted}} možnost prenosa",
"found_download_option_other": "Najdenih {{countFormatted}} možnosti prenosa",
"import": "Uvozi",
"importing": "Uvažanje...",
"public": "Javno",
"private": "Zasebno",
"friends_only": "Samo prijatelji",
"privacy": "Zasebnost",
"profile_visibility": "Vidnost profila",
"profile_visibility_description": "Izberite, kdo lahko vidi vaš profil in knjižnico",
"required_field": "To polje je obvezno",
"source_already_exists": "Ta vir je že bil dodan",
"must_be_valid_url": "Vir mora biti veljaven URL",
"blocked_users": "Blokirani uporabniki",
"user_unblocked": "Uporabnik je odblokiran",
"enable_achievement_notifications": "Ko je dosežek odklenjen",
"launch_minimized": "Zaženi Hydra minimizirano",
"disable_nsfw_alert": "Onemogoči opozorilo NSFW",
"seed_after_download_complete": "Sejanje po končanem prenosu",
"show_hidden_achievement_description": "Pokaži opis skritih dosežkov pred njihovim odklepanjem",
"account": "Račun",
"hydra_cloud": "Hydra Cloud",
"no_users_blocked": "Nimate blokiranih uporabnikov",
"subscription_active_until": "Vaš Hydra Cloud je aktiven do {{date}}",
"manage_subscription": "Upravljaj naročnino",
"update_email": "Posodobi e-pošto",
"update_password": "Posodobi geslo",
"current_email": "Trenutna e-pošta:",
"no_email_account": "Še niste nastavili e-pošte",
"account_data_updated_successfully": "Podatki računa so bili uspešno posodobljeni",
"renew_subscription": "Obnovi Hydra Cloud",
"subscription_expired_at": "Vaša naročnina je potekla {{date}}",
"no_subscription": "Uživajte v Hydri na najboljši način",
"become_subscriber": "Postanite Hydra Cloud uporabnik",
"subscription_renew_cancelled": "Samodejno podaljševanje je onemogočeno",
"subscription_renews_on": "Vaša naročnina se podaljša {{date}}",
"bill_sent_until": "Naslednji račun bo poslan do tega dne",
"no_themes": "Zdi se, da še nimate tem, vendar brez skrbi, kliknite tukaj, da ustvarite svojo prvo mojstrovino.",
"editor_tab_code": "Koda",
"editor_tab_info": "Info",
"editor_tab_save": "Shrani",
"web_store": "Spletna trgovina",
"clear_themes": "Počisti",
"create_theme": "Ustvari",
"create_theme_modal_title": "Ustvari prilagojeno temo",
"create_theme_modal_description": "Ustvarite novo temo za prilagajanje videza Hydre",
"theme_name": "Ime",
"insert_theme_name": "Vstavite ime teme",
"set_theme": "Nastavi temo",
"unset_theme": "Odstrani temo",
"delete_theme": "Izbriši temo",
"edit_theme": "Uredi temo",
"delete_all_themes": "Izbriši vse teme",
"delete_all_themes_description": "To bo izbrisalo vse vaše prilagojene teme",
"delete_theme_description": "To bo izbrisalo temo {{theme}}",
"cancel": "Prekliči",
"appearance": "Videz",
"debrid": "Debrid",
"debrid_description": "Debrid storitve so premium neomejeni prenašalniki, ki vam omogočajo hitro prenašanje datotek, gostovanih na različnih storitvah za gostovanje datotek, omejeno le s hitrostjo vašega interneta.",
"enable_torbox": "Omogoči TorBox",
"torbox_description": "TorBox je vaša premium seedbox storitev, ki se lahko kosuje tudi najboljšim strežnikom na trgu.",
"torbox_account_linked": "TorBox račun povezan",
"create_real_debrid_account": "Kliknite tukaj, če še nimate Real-Debrid računa",
"create_torbox_account": "Kliknite tukaj, če še nimate TorBox računa",
"real_debrid_account_linked": "Real-Debrid račun povezan",
"name_min_length": "Ime teme mora imeti vsaj 3 znake",
"import_theme": "Uvozi temo",
"import_theme_description": "Uvožili boste {{theme}} iz trgovine tem",
"error_importing_theme": "Napaka pri uvozu teme",
"theme_imported": "Tema uspešno uvožena",
"enable_friend_request_notifications": "Ko je prejet prijateljski zahtevek",
"enable_auto_install": "Samodejno prenesi posodobitve",
"common_redist": "Skupni redistributable-ji",
"common_redist_description": "Skupni redistributable-ji so potrebni za zagon nekaterih iger. Priporočamo njihovo namestitev, da se izognete težavam.",
"install_common_redist": "Namesti",
"installing_common_redist": "Nameščanje…",
"show_download_speed_in_megabytes": "Pokaži hitrost prenosa v megabajtih na sekundo",
"extract_files_by_default": "Privzeto razpakiraj datoteke po prenosu",
"enable_steam_achievements": "Omogoči iskanje po Steam dosežkih",
"enable_new_download_options_badges": "Pokaži značke novih možnosti prenosa",
"achievement_custom_notification_position": "Lastna pozicija obvestil o dosežkih",
"top-left": "Zgoraj levo",
"top-center": "Zgoraj na sredini",
"top-right": "Zgoraj desno",
"bottom-left": "Spodaj levo",
"bottom-center": "Spodaj na sredini",
"bottom-right": "Spodaj desno",
"enable_achievement_custom_notifications": "Omogoči lastna obvestila o dosežkih",
"alignment": "Poravnava",
"variation": "Variacija",
"default": "Privzeto",
"rare": "Redko",
"platinum": "Platinasto",
"hidden": "Skrito",
"test_notification": "Preizkusno obvestilo",
"achievement_sound_volume": "Glasnost zvoka dosežka",
"select_achievement_sound": "Izberite zvok dosežka",
"change_achievement_sound": "Spremeni zvok dosežka",
"remove_achievement_sound": "Odstrani zvok dosežka",
"preview_sound": "Predogled zvoka",
"select": "Izberi",
"preview": "Predogled",
"remove": "Odstrani",
"no_sound_file_selected": "Nobena zvočna datoteka ni izbrana",
"notification_preview": "Predogled obvestila o dosežku",
"enable_friend_start_game_notifications": "Ko prijatelj začne igrati igro",
"autoplay_trailers_on_game_page": "Samodejno predvajaj napovednike na strani igre",
"hide_to_tray_on_game_start": "Skrij Hydreo v sistemsko vrstico ob zagonu igre",
"downloads": "Prenosi",
"use_native_http_downloader": "Uporabi izvorni HTTP prenašalnik (eksperimentalno)",
"cannot_change_downloader_while_downloading": "Nastavitve ni mogoče spremeniti med prenosom",
"notifications": {
"download_complete": "Prenos končan",
"game_ready_to_install": "{{title}} je pripravljen za namestitev",
"repack_list_updated": "Seznam repackov posodobljen",
"repack_count_one": "{{count}} repack dodan",
"repack_count_other": "{{count}} repackov dodanih",
"new_update_available": "Različica {{version}} na voljo",
"restart_to_install_update": "Znova zaženite Hydreo za namestitev posodobitve",
"notification_achievement_unlocked_title": "Dosežek odklenjen za {{game}}",
"notification_achievement_unlocked_body": "{{achievement}} in drugi {{count}} so bili odklenjeni",
"new_friend_request_description": "{{displayName}} vam je poslal prijateljsko zahtevo",
"new_friend_request_title": "Nova prijateljska zahteva",
"extraction_complete": "Razpakiranje končano",
"game_extracted": "{{title}} je bil uspešno razpakiran",
"friend_started_playing_game": "{{displayName}} je začel igrati igro",
"test_achievement_notification_title": "To je preizkusno obvestilo",
"test_achievement_notification_description": "Kar kul, kajne?"
},
"system_tray": {
"open": "Odpri Hydreo",
"quit": "Izhod"
},
"game_card": {
"available_one": "Na voljo",
"available_other": "Na voljo",
"no_downloads": "Ni razpoložljivih prenosov",
"calculating": "Računam"
},
"binary_not_found_modal": {
"title": "Programi niso nameščeni",
"description": "Izvajalniki Wine ali Lutris niso bili najdeni na vašem sistemu",
"instructions": "Preverite pravi način za namestitev katerega od njih na vašo Linux distribucijo, da bi igra lahko normalno tekla"
},
"modal": {
"close": "Zapri gumb"
},
"forms": {
"toggle_password_visibility": "Preklopi vidnost gesla"
},
"user_profile": {
"amount_hours": "{{amount}} ur",
"amount_minutes": "{{amount}} minut",
"amount_hours_short": "{{amount}}h",
"amount_minutes_short": "{{amount}}m",
"last_time_played": "Nazadnje igrano {{period}}",
"activity": "Nedavna dejavnost",
"library": "Knjižnica",
"pinned": "Pripeto",
"sort_by": "Razvrsti po:",
"achievements_earned": "Odklenjeni dosežki",
"played_recently": "Nazadnje igrano",
"playtime": "Čas igranja",
"total_play_time": "Skupni čas igranja",
"manual_playtime_tooltip": "Ta čas igranja je bil ročno posodobljen",
"no_recent_activity_title": "Hmmm… nič tukaj",
"no_recent_activity_description": "Niste igrali nobene igre v zadnjem času. Čas je, da to spremenite!",
"display_name": "Prikazno ime",
"saving": "Shranjevanje",
"save": "Shrani",
"edit_profile": "Uredi profil",
"saved_successfully": "Uspešno shranjeno",
"try_again": "Prosimo, poskusite znova",
"sign_out_modal_title": "Ste prepričani?",
"cancel": "Prekliči",
"successfully_signed_out": "Uspešno odjavljeni",
"sign_out": "Odjavi se",
"playing_for": "Igra za {{amount}}",
"sign_out_modal_text": "Vaša knjižnica je povezana s trenutnim računom. Ob odjavi knjižnica ne bo več vidna, napredek pa se ne bo shranil. Nadaljujete z odjavo?",
"add_friends": "Dodaj prijatelje",
"add": "Dodaj",
"friend_code": "Koda prijatelja",
"see_profile": "Poglej profil",
"sending": "Pošiljanje",
"friend_request_sent": "Zahteva za prijateljstvo poslana",
"friends": "Prijatelji",
"badges": "Značke",
"friends_list": "Seznam prijateljev",
"user_not_found": "Uporabnik ni najden",
"block_user": "Blokiraj uporabnika",
"add_friend": "Dodaj prijatelja",
"request_sent": "Zahteva poslana",
"request_received": "Zahteva prejeta",
"accept_request": "Sprejmi zahtevo",
"ignore_request": "Ignoriraj zahtevo",
"cancel_request": "Prekliči zahtevo",
"undo_friendship": "Razveljavi prijateljstvo",
"friendship_removed": "Prijatelj odstranjen",
"request_accepted": "Zahteva sprejeta",
"user_blocked_successfully": "Uporabnik uspešno blokiran",
"user_block_modal_text": "To bo blokiralo {{displayName}}",
"blocked_users": "Blokirani uporabniki",
"unblock": "Odblokiraj",
"no_friends_added": "Nimate dodanih prijateljev",
"no_friends_yet": "Še niste dodali prijateljev",
"view_all": "Poglej vse",
"load_more": "Naloži več",
"loading": "Nalaganje",
"pending": "V teku",
"no_pending_invites": "Nimate čakajočih povabil",
"no_blocked_users": "Nimate blokiranih uporabnikov",
"friend_code_copied": "Koda prijatelja kopirana",
"undo_friendship_modal_text": "To bo razveljavilo vaše prijateljstvo z {{displayName}}",
"privacy_hint": "Za prilagoditev, kdo to vidi, pojdite na <0>Nastavitve</0>",
"locked_profile": "Ta profil je zaseben",
"image_process_failure": "Napaka pri obdelavi slike",
"required_field": "To polje je obvezno",
"displayname_min_length": "Prikazno ime mora biti dolgo vsaj 3 znake",
"displayname_max_length": "Prikazno ime mora imeti največ 50 znakov",
"report_profile": "Prijavi ta profil",
"report_reason": "Zakaj prijavljate ta profil?",
"report_description": "Dodatne informacije",
"report_description_placeholder": "Dodatne informacije",
"report": "Prijavi",
"report_reason_hate": "Sovražni govor",
"report_reason_sexual_content": "Seksualna vsebina",
"report_reason_violence": "Nasilje",
"report_reason_spam": "Spam",
"report_reason_other": "Drugo",
"profile_reported": "Profil prijavljen",
"your_friend_code": "Vaša koda prijatelja:",
"copy_friend_code": "Kopiraj kodo prijatelja",
"copied": "Kopirano!",
"upload_banner": "Naloži banner",
"uploading_banner": "Nalaganje bannerja…",
"change_banner": "Spremeni banner",
"replace_banner": "Zamenjaj banner",
"remove_banner": "Odstrani banner",
"remove_banner_modal_title": "Odstrani banner?",
"remove_banner_confirmation": "Ali ste prepričani, da želite odstraniti banner? Kadarkoli lahko izberete novega.",
"remove": "Odstrani",
"background_image_updated": "Pozadinska slika posodobljena",
"stats": "Statistika",
"achievements": "dosežki",
"games": "Igre",
"top_percentile": "Top {{percentile}}%",
"ranking_updated_weekly": "Uvrstitev se posodablja tedensko",
"playing": "Igra {{game}}",
"achievements_unlocked": "Dosežki odklenjeni",
"earned_points": "Zaslužene točke",
"show_achievements_on_profile": "Pokaži vaše dosežke na profilu",
"show_points_on_profile": "Pokaži vaše zaslužene točke na profilu",
"error_adding_friend": "Zahteve za prijatelja ni bilo mogoče poslati. Preverite kodo prijatelja",
"friend_code_length_error": "Koda prijatelja mora vsebovati 8 znakov",
"game_removed_from_pinned": "Igra odstranjena iz pripetih",
"game_added_to_pinned": "Igra dodana med pripete",
"karma": "Karma",
"karma_count": "karma",
"user_reviews": "Mnenja",
"delete_review": "Izbriši mnenje",
"loading_reviews": "Nalaganje mnenj...",
"wrapped_2025": "Wrapped 2025"
},
"library": {
"library": "Knjižnica",
"play": "Igraj",
"download": "Prenesi",
"downloading": "Prenašanje",
"game": "igra",
"games": "igre",
"grid_view": "Mrežni pogled",
"compact_view": "Kompaktni pogled",
"large_view": "Velik pogled",
"no_games_title": "Vaša knjižnica je prazna",
"no_games_description": "Dodajte igre iz kataloga ali jih prenesite, da začnete",
"amount_hours": "{{amount}} ur",
"amount_minutes": "{{amount}} minut",
"amount_hours_short": "{{amount}}h",
"amount_minutes_short": "{{amount}}m",
"manual_playtime_tooltip": "Ta čas igranja je bil ročno posodobljen",
"all_games": "Vse igre",
"recently_played": "Nedavno igrane",
"favorites": "Priljubljene"
},
"achievement": {
"achievement_unlocked": "Dosežek odklenjen",
"user_achievements": "Dosežki uporabnika {{displayName}}",
"your_achievements": "Vaši dosežki",
"unlocked_at": "Odklenjeno: {{date}}",
"subscription_needed": "Naročnina na Hydra Cloud je potrebna za ogled te vsebine",
"new_achievements_unlocked": "Odklenili ste {{achievementCount}} novih dosežkov iz {{gameCount}} iger",
"achievement_progress": "{{unlockedCount}}/{{totalCount}} dosežkov",
"achievements_unlocked_for_game": "Odklenili ste {{achievementCount}} novih dosežkov za {{gameTitle}}",
"hidden_achievement_tooltip": "To je skriti dosežek",
"achievement_earn_points": "Z zaslužite {{points}} točk s tem dosežkom",
"earned_points": "Zaslužene točke:",
"available_points": "Razpoložljive točke:",
"how_to_earn_achievements_points": "Kako zaslužiti točke za dosežke?"
},
"hydra_cloud": {
"subscription_tour_title": "Naročnina Hydra Cloud",
"subscribe_now": "Naroči se zdaj",
"cloud_saving": "Shranjevanje v oblak",
"cloud_achievements": "Shrani svoje dosežke v oblak",
"animated_profile_picture": "Animirane profilne slike",
"premium_support": "Premium podpora",
"show_and_compare_achievements": "Pokaži in primerjaj svoje dosežke z drugimi uporabniki",
"animated_profile_banner": "Animirani profilni banner",
"hydra_cloud": "Hydra Cloud",
"hydra_cloud_feature_found": "Pravkar ste odkrili funkcijo Hydra Cloud!",
"learn_more": "Več informacij",
"debrid_description": "Prenesite do 4x hitreje z Nimbusom"
},
"notifications_page": {
"title": "Obvestila",
"mark_all_as_read": "Označi vse kot prebrano",
"clear_all": "Počisti vse",
"loading": "Nalagam...",
"empty_title": "Ni obvestil",
"empty_description": "Ste na tekočem! Preverite kasneje za nove posodobitve.",
"empty_filter_description": "Nobeno obvestilo ne ustreza tem filtram.",
"filter_all": "Vse",
"filter_unread": "Neprebrano",
"filter_friends": "Prijatelji",
"filter_badges": "Značke",
"filter_upvotes": "Glasovi za všečkanje",
"filter_local": "Lokalno",
"load_more": "Naloži več",
"dismiss": "Opusti",
"accept": "Sprejmi",
"refuse": "Zavrni",
"notification": "Obvestilo",
"friend_request_received_title": "Nova prijateljska zahteva!",
"friend_request_received_description": "{{displayName}} želi biti vaš prijatelj",
"friend_request_accepted_title": "Zahteva za prijateljstvo sprejeta!",
"friend_request_accepted_description": "{{displayName}} je sprejel vašo zahtevo",
"badge_received_title": "Prejeli ste novo značko!",
"badge_received_description": "{{badgeName}}",
"review_upvote_title": "Vaša recenzija za {{gameTitle}} je dobila glasove!",
"review_upvote_description": "Vaša recenzija je dobila {{count}} novih glasov",
"marked_all_as_read": "Vsa obvestila označena kot prebrana",
"failed_to_mark_as_read": "Neuspešno označevanje obvestil kot prebranih",
"cleared_all": "Vsa obvestila izbrisana",
"failed_to_clear": "Neuspešno brisanje obvestil",
"failed_to_load": "Neuspešno nalaganje obvestil",
"failed_to_dismiss": "Neuspešno opustitev obvestila",
"friend_request_accepted": "Zahteva za prijateljstvo sprejeta",
"friend_request_refused": "Zahteva za prijateljstvo zavrnjena"
}
}
}

View File

@@ -15,14 +15,7 @@ const deleteGameFolder = async (
const downloadKey = levelKeys.game(shop, objectId); const downloadKey = levelKeys.game(shop, objectId);
const download = await downloadsSublevel.get(downloadKey); const download = await downloadsSublevel.get(downloadKey);
if (!download?.folderName) return; if (!download) return;
const folderPath = path.join(
download.downloadPath ?? (await getDownloadsPath()),
download.folderName
);
const metaPath = `${folderPath}.meta`;
const deleteFile = async (filePath: string, isDirectory = false) => { const deleteFile = async (filePath: string, isDirectory = false) => {
if (fs.existsSync(filePath)) { if (fs.existsSync(filePath)) {
@@ -47,8 +40,18 @@ const deleteGameFolder = async (
} }
}; };
if (download.folderName) {
const folderPath = path.join(
download.downloadPath ?? (await getDownloadsPath()),
download.folderName
);
const metaPath = `${folderPath}.meta`;
await deleteFile(folderPath, true); await deleteFile(folderPath, true);
await deleteFile(metaPath); await deleteFile(metaPath);
}
await downloadsSublevel.del(downloadKey); await downloadsSublevel.del(downloadKey);
}; };

View File

@@ -22,7 +22,6 @@ const getGameInstallerActionType = async (
); );
if (!fs.existsSync(gamePath)) { if (!fs.existsSync(gamePath)) {
await downloadsSublevel.del(downloadKey);
return "open-folder"; return "open-folder";
} }

View File

@@ -25,7 +25,7 @@ const getLibrary = async (): Promise<LibraryGame[]> => {
const achievements = await gameAchievementsSublevel.get(key); const achievements = await gameAchievementsSublevel.get(key);
unlockedAchievementCount = unlockedAchievementCount =
achievements?.unlockedAchievements.length ?? 0; achievements?.unlockedAchievements?.length ?? 0;
} }
return { return {

View File

@@ -24,6 +24,7 @@ import "./remove-game-from-favorites";
import "./remove-game-from-library"; import "./remove-game-from-library";
import "./remove-game"; import "./remove-game";
import "./reset-game-achievements"; import "./reset-game-achievements";
import "./scan-installed-games";
import "./select-game-wine-prefix"; import "./select-game-wine-prefix";
import "./toggle-automatic-cloud-sync"; import "./toggle-automatic-cloud-sync";
import "./toggle-game-pin"; import "./toggle-game-pin";

View File

@@ -38,7 +38,6 @@ const openGameInstaller = async (
); );
if (!fs.existsSync(gamePath)) { if (!fs.existsSync(gamePath)) {
await downloadsSublevel.del(downloadKey);
return true; return true;
} }

View File

@@ -0,0 +1,143 @@
import path from "node:path";
import fs from "node:fs";
import { t } from "i18next";
import { registerEvent } from "../register-event";
import { gamesSublevel } from "@main/level";
import {
GameExecutables,
LocalNotificationManager,
logger,
WindowManager,
} from "@main/services";
const SCAN_DIRECTORIES = [
String.raw`C:\Games`,
String.raw`D:\Games`,
String.raw`C:\Program Files (x86)\Steam\steamapps\common`,
String.raw`C:\Program Files\Steam\steamapps\common`,
String.raw`C:\Program Files (x86)\DODI-Repacks`,
];
interface FoundGame {
title: string;
executablePath: string;
}
interface ScanResult {
foundGames: FoundGame[];
total: number;
}
async function searchInDirectories(
executableNames: Set<string>
): Promise<string | null> {
for (const scanDir of SCAN_DIRECTORIES) {
if (!fs.existsSync(scanDir)) continue;
const foundPath = await findExecutableInFolder(scanDir, executableNames);
if (foundPath) return foundPath;
}
return null;
}
async function publishScanNotification(foundCount: number): Promise<void> {
const hasFoundGames = foundCount > 0;
await LocalNotificationManager.createNotification(
"SCAN_GAMES_COMPLETE",
t(
hasFoundGames
? "scan_games_complete_title"
: "scan_games_no_results_title",
{ ns: "notifications" }
),
t(
hasFoundGames
? "scan_games_complete_description"
: "scan_games_no_results_description",
{ ns: "notifications", count: foundCount }
),
{ url: "/library?openScanModal=true" }
);
}
const scanInstalledGames = async (
_event: Electron.IpcMainInvokeEvent
): Promise<ScanResult> => {
const games = await gamesSublevel
.iterator()
.all()
.then((results) =>
results
.filter(
([_key, game]) => game.isDeleted === false && game.shop !== "custom"
)
.map(([key, game]) => ({ key, game }))
);
const foundGames: FoundGame[] = [];
const gamesToScan = games.filter((g) => !g.game.executablePath);
for (const { key, game } of gamesToScan) {
const executableNames = GameExecutables.getExecutablesForGame(
game.objectId
);
if (!executableNames || executableNames.length === 0) continue;
const normalizedNames = new Set(
executableNames.map((name) => name.toLowerCase())
);
const foundPath = await searchInDirectories(normalizedNames);
if (foundPath) {
await gamesSublevel.put(key, { ...game, executablePath: foundPath });
logger.info(
`[ScanInstalledGames] Found executable for ${game.objectId}: ${foundPath}`
);
foundGames.push({ title: game.title, executablePath: foundPath });
}
}
WindowManager.mainWindow?.webContents.send("on-library-batch-complete");
await publishScanNotification(foundGames.length);
return { foundGames, total: gamesToScan.length };
};
async function findExecutableInFolder(
folderPath: string,
executableNames: Set<string>
): Promise<string | null> {
try {
const entries = await fs.promises.readdir(folderPath, {
withFileTypes: true,
recursive: true,
});
for (const entry of entries) {
if (!entry.isFile()) continue;
const fileName = entry.name.toLowerCase();
if (executableNames.has(fileName)) {
const parentPath =
"parentPath" in entry ? entry.parentPath : folderPath;
return path.join(parentPath, entry.name);
}
}
} catch (err) {
logger.error(
`[ScanInstalledGames] Error reading folder ${folderPath}:`,
err
);
}
return null;
}
registerEvent("scanInstalledGames", scanInstalledGames);

View File

@@ -0,0 +1,78 @@
import { registerEvent } from "../register-event";
import type { Download, StartGameDownloadPayload } from "@types";
import { DownloadManager, HydraApi, logger } from "@main/services";
import { createGame } from "@main/services/library-sync";
import { downloadsSublevel, gamesSublevel, levelKeys } from "@main/level";
import { parseBytes } from "@shared";
import { handleDownloadError, prepareGameEntry } from "@main/helpers";
const addGameToQueue = async (
_event: Electron.IpcMainInvokeEvent,
payload: StartGameDownloadPayload
) => {
const {
objectId,
title,
shop,
downloadPath,
downloader,
uri,
automaticallyExtract,
fileSize,
} = payload;
const gameKey = levelKeys.game(shop, objectId);
const download: Download = {
shop,
objectId,
status: "paused",
progress: 0,
bytesDownloaded: 0,
downloadPath,
downloader,
uri,
folderName: null,
fileSize: parseBytes(fileSize ?? null),
shouldSeed: false,
timestamp: Date.now(),
queued: true,
extracting: false,
automaticallyExtract,
extractionProgress: 0,
};
try {
await DownloadManager.validateDownloadUrl(download);
} catch (err: unknown) {
logger.error("Failed to validate download URL for queue", err);
return handleDownloadError(err, downloader);
}
await prepareGameEntry({ gameKey, title, objectId, shop });
try {
await downloadsSublevel.put(gameKey, download);
const updatedGame = await gamesSublevel.get(gameKey);
await Promise.all([
createGame(updatedGame!).catch(() => {}),
HydraApi.post(`/games/${shop}/${objectId}/download`, null, {
needsAuth: false,
}).catch(() => {}),
]);
return { ok: true };
} catch (err: unknown) {
logger.error("Failed to add game to queue", err);
if (err instanceof Error) {
return { ok: false, error: err.message };
}
return { ok: false };
}
};
registerEvent("addGameToQueue", addGameToQueue);

View File

@@ -1,3 +1,4 @@
import "./add-game-to-queue";
import "./cancel-game-download"; import "./cancel-game-download";
import "./check-debrid-availability"; import "./check-debrid-availability";
import "./pause-game-download"; import "./pause-game-download";
@@ -5,3 +6,4 @@ import "./pause-game-seed";
import "./resume-game-download"; import "./resume-game-download";
import "./resume-game-seed"; import "./resume-game-seed";
import "./start-game-download"; import "./start-game-download";
import "./update-download-queue-position";

View File

@@ -2,14 +2,8 @@ import { registerEvent } from "../register-event";
import type { Download, StartGameDownloadPayload } from "@types"; import type { Download, StartGameDownloadPayload } from "@types";
import { DownloadManager, HydraApi, logger } from "@main/services"; import { DownloadManager, HydraApi, logger } from "@main/services";
import { createGame } from "@main/services/library-sync"; import { createGame } from "@main/services/library-sync";
import { Downloader, DownloadError } from "@shared"; import { downloadsSublevel, gamesSublevel, levelKeys } from "@main/level";
import { import { handleDownloadError, prepareGameEntry } from "@main/helpers";
downloadsSublevel,
gamesShopAssetsSublevel,
gamesSublevel,
levelKeys,
} from "@main/level";
import { AxiosError } from "axios";
const startGameDownload = async ( const startGameDownload = async (
_event: Electron.IpcMainInvokeEvent, _event: Electron.IpcMainInvokeEvent,
@@ -38,30 +32,7 @@ const startGameDownload = async (
} }
} }
const game = await gamesSublevel.get(gameKey); await prepareGameEntry({ gameKey, title, objectId, shop });
const gameAssets = await gamesShopAssetsSublevel.get(gameKey);
await downloadsSublevel.del(gameKey);
if (game) {
await gamesSublevel.put(gameKey, {
...game,
isDeleted: false,
});
} else {
await gamesSublevel.put(gameKey, {
title,
iconUrl: gameAssets?.iconUrl ?? null,
libraryHeroImageUrl: gameAssets?.libraryHeroImageUrl ?? null,
logoImageUrl: gameAssets?.logoImageUrl ?? null,
objectId,
shop,
remoteId: null,
playTimeInMilliseconds: 0,
lastTimePlayed: null,
isDeleted: false,
});
}
await DownloadManager.cancelDownload(gameKey); await DownloadManager.cancelDownload(gameKey);
@@ -101,68 +72,7 @@ const startGameDownload = async (
return { ok: true }; return { ok: true };
} catch (err: unknown) { } catch (err: unknown) {
logger.error("Failed to start download", err); logger.error("Failed to start download", err);
return handleDownloadError(err, downloader);
if (err instanceof AxiosError) {
if (err.response?.status === 429 && downloader === Downloader.Gofile) {
return { ok: false, error: DownloadError.GofileQuotaExceeded };
}
if (
err.response?.status === 403 &&
downloader === Downloader.RealDebrid
) {
return {
ok: false,
error: DownloadError.RealDebridAccountNotAuthorized,
};
}
if (downloader === Downloader.TorBox) {
return { ok: false, error: err.response?.data?.detail };
}
}
if (err instanceof Error) {
if (downloader === Downloader.Buzzheavier) {
if (err.message.includes("Rate limit")) {
return {
ok: false,
error: "Buzzheavier: Rate limit exceeded",
};
}
if (
err.message.includes("not found") ||
err.message.includes("deleted")
) {
return {
ok: false,
error: "Buzzheavier: File not found",
};
}
}
if (downloader === Downloader.FuckingFast) {
if (err.message.includes("Rate limit")) {
return {
ok: false,
error: "FuckingFast: Rate limit exceeded",
};
}
if (
err.message.includes("not found") ||
err.message.includes("deleted")
) {
return {
ok: false,
error: "FuckingFast: File not found",
};
}
}
return { ok: false, error: err.message };
}
return { ok: false };
} }
}; };

View File

@@ -0,0 +1,67 @@
import { registerEvent } from "../register-event";
import { downloadsSublevel, levelKeys } from "@main/level";
import { GameShop } from "@types";
import { orderBy } from "lodash-es";
const updateDownloadQueuePosition = async (
_event: Electron.IpcMainInvokeEvent,
shop: GameShop,
objectId: string,
direction: "up" | "down"
) => {
const gameKey = levelKeys.game(shop, objectId);
const download = await downloadsSublevel.get(gameKey);
if (!download || !download.queued || download.status !== "paused") {
return false;
}
const allDownloads = await downloadsSublevel.values().all();
const queuedDownloads = orderBy(
allDownloads.filter((d) => d.status === "paused" && d.queued),
"timestamp",
"desc"
);
const currentIndex = queuedDownloads.findIndex(
(d) => d.shop === shop && d.objectId === objectId
);
if (currentIndex === -1) {
return false;
}
const targetIndex = direction === "up" ? currentIndex - 1 : currentIndex + 1;
if (targetIndex < 0 || targetIndex >= queuedDownloads.length) {
return false;
}
const currentDownload = queuedDownloads[currentIndex];
const adjacentDownload = queuedDownloads[targetIndex];
const currentKey = levelKeys.game(
currentDownload.shop,
currentDownload.objectId
);
const adjacentKey = levelKeys.game(
adjacentDownload.shop,
adjacentDownload.objectId
);
const tempTimestamp = currentDownload.timestamp;
await downloadsSublevel.put(currentKey, {
...currentDownload,
timestamp: adjacentDownload.timestamp,
});
await downloadsSublevel.put(adjacentKey, {
...adjacentDownload,
timestamp: tempTimestamp,
});
return true;
};
registerEvent("updateDownloadQueuePosition", updateDownloadQueuePosition);

View File

@@ -0,0 +1,51 @@
import { AxiosError } from "axios";
import { Downloader, DownloadError } from "@shared";
export const handleDownloadError = (
err: unknown,
downloader: Downloader
): { ok: false; error?: string } => {
if (err instanceof AxiosError) {
if (err.response?.status === 429 && downloader === Downloader.Gofile) {
return { ok: false, error: DownloadError.GofileQuotaExceeded };
}
if (err.response?.status === 403 && downloader === Downloader.RealDebrid) {
return { ok: false, error: DownloadError.RealDebridAccountNotAuthorized };
}
if (downloader === Downloader.TorBox) {
return { ok: false, error: err.response?.data?.detail };
}
}
if (err instanceof Error) {
if (downloader === Downloader.Buzzheavier) {
if (err.message.includes("Rate limit")) {
return { ok: false, error: "Buzzheavier: Rate limit exceeded" };
}
if (
err.message.includes("not found") ||
err.message.includes("deleted")
) {
return { ok: false, error: "Buzzheavier: File not found" };
}
}
if (downloader === Downloader.FuckingFast) {
if (err.message.includes("Rate limit")) {
return { ok: false, error: "FuckingFast: Rate limit exceeded" };
}
if (
err.message.includes("not found") ||
err.message.includes("deleted")
) {
return { ok: false, error: "FuckingFast: File not found" };
}
}
return { ok: false, error: err.message };
}
return { ok: false };
};

View File

@@ -0,0 +1,45 @@
import {
downloadsSublevel,
gamesShopAssetsSublevel,
gamesSublevel,
} from "@main/level";
import type { GameShop } from "@types";
interface PrepareGameEntryParams {
gameKey: string;
title: string;
objectId: string;
shop: GameShop;
}
export const prepareGameEntry = async ({
gameKey,
title,
objectId,
shop,
}: PrepareGameEntryParams): Promise<void> => {
const game = await gamesSublevel.get(gameKey);
const gameAssets = await gamesShopAssetsSublevel.get(gameKey);
await downloadsSublevel.del(gameKey);
if (game) {
await gamesSublevel.put(gameKey, {
...game,
isDeleted: false,
});
} else {
await gamesSublevel.put(gameKey, {
title,
iconUrl: gameAssets?.iconUrl ?? null,
libraryHeroImageUrl: gameAssets?.libraryHeroImageUrl ?? null,
logoImageUrl: gameAssets?.logoImageUrl ?? null,
objectId,
shop,
remoteId: null,
playTimeInMilliseconds: 0,
lastTimePlayed: null,
isDeleted: false,
});
}
};

View File

@@ -94,3 +94,5 @@ export const getThemeSoundPath = (
}; };
export * from "./reg-parser"; export * from "./reg-parser";
export * from "./download-error-handler";
export * from "./download-game-helper";

View File

@@ -21,7 +21,7 @@ import { RealDebridClient } from "./real-debrid";
import path from "node:path"; import path from "node:path";
import { logger } from "../logger"; import { logger } from "../logger";
import { db, downloadsSublevel, gamesSublevel, levelKeys } from "@main/level"; import { db, downloadsSublevel, gamesSublevel, levelKeys } from "@main/level";
import { sortBy } from "lodash-es"; import { orderBy } from "lodash-es";
import { TorBoxClient } from "./torbox"; import { TorBoxClient } from "./torbox";
import { GameFilesManager } from "../game-files-manager"; import { GameFilesManager } from "../game-files-manager";
import { HydraDebridClient } from "./hydra-debrid"; import { HydraDebridClient } from "./hydra-debrid";
@@ -323,7 +323,8 @@ export class DownloadManager {
this.sendProgressUpdate(progress, status, game); this.sendProgressUpdate(progress, status, game);
if (progress === 1) { const isComplete = progress === 1 || download.status === "complete";
if (isComplete) {
await this.handleDownloadCompletion(download, game, gameId); await this.handleDownloadCompletion(download, game, gameId);
} }
} }
@@ -362,6 +363,11 @@ export class DownloadManager {
if (download.automaticallyExtract) { if (download.automaticallyExtract) {
this.handleExtraction(download, game); this.handleExtraction(download, game);
} else {
// For downloads without extraction (e.g., torrents with ready-to-play files),
// search for executable in the download folder
const gameFilesManager = new GameFilesManager(game.shop, game.objectId);
gameFilesManager.searchAndBindExecutable();
} }
await this.processNextQueuedDownload(); await this.processNextQueuedDownload();
@@ -417,10 +423,10 @@ export class DownloadManager {
.values() .values()
.all() .all()
.then((games) => .then((games) =>
sortBy( orderBy(
games.filter((game) => game.status === "paused" && game.queued), games.filter((game) => game.status === "paused" && game.queued),
"timestamp", ["timestamp"],
"DESC" ["desc"]
) )
); );
@@ -494,6 +500,9 @@ export class DownloadManager {
} }
static async cancelDownload(downloadKey = this.downloadingGameId) { static async cancelDownload(downloadKey = this.downloadingGameId) {
const isActiveDownload = downloadKey === this.downloadingGameId;
if (isActiveDownload) {
if (this.usingJsDownloader && this.jsDownloader) { if (this.usingJsDownloader && this.jsDownloader) {
logger.log("[DownloadManager] Cancelling JS download"); logger.log("[DownloadManager] Cancelling JS download");
this.jsDownloader.cancelDownload(); this.jsDownloader.cancelDownload();
@@ -505,7 +514,6 @@ export class DownloadManager {
.catch((err) => logger.error("Failed to cancel game download", err)); .catch((err) => logger.error("Failed to cancel game download", err));
} }
if (downloadKey === this.downloadingGameId) {
WindowManager.mainWindow?.setProgressBar(-1); WindowManager.mainWindow?.setProgressBar(-1);
WindowManager.mainWindow?.webContents.send("on-download-progress", null); WindowManager.mainWindow?.webContents.send("on-download-progress", null);
this.downloadingGameId = null; this.downloadingGameId = null;
@@ -927,6 +935,20 @@ export class DownloadManager {
} }
} }
static async validateDownloadUrl(download: Download): Promise<void> {
const useJsDownloader = await this.shouldUseJsDownloader();
const isHttp = this.isHttpDownloader(download.downloader);
if (useJsDownloader && isHttp) {
const options = await this.getJsDownloadOptions(download);
if (!options) {
throw new Error("Failed to validate download URL");
}
} else if (isHttp) {
await this.getDownloadPayload(download);
}
}
static async startDownload(download: Download) { static async startDownload(download: Download) {
const useJsDownloader = await this.shouldUseJsDownloader(); const useJsDownloader = await this.shouldUseJsDownloader();
const isHttp = this.isHttpDownloader(download.downloader); const isHttp = this.isHttpDownloader(download.downloader);

View File

@@ -320,10 +320,17 @@ export class JsHttpDownloader {
return null; return null;
} }
let progress = 0;
if (this.status === "complete") {
progress = 1;
} else if (this.fileSize > 0) {
progress = this.bytesDownloaded / this.fileSize;
}
return { return {
folderName: this.folderName, folderName: this.folderName,
fileSize: this.fileSize, fileSize: this.fileSize,
progress: this.fileSize > 0 ? this.bytesDownloaded / this.fileSize : 0, progress,
downloadSpeed: this.downloadSpeed, downloadSpeed: this.downloadSpeed,
numPeers: 0, numPeers: 0,
numSeeds: 0, numSeeds: 0,

View File

@@ -0,0 +1,13 @@
import { gameExecutables } from "./process-watcher";
export class GameExecutables {
static getExecutablesForGame(objectId: string): string[] | null {
const executables = gameExecutables[objectId];
if (!executables || executables.length === 0) {
return null;
}
return executables.map((exe) => exe.exe);
}
}

View File

@@ -7,6 +7,7 @@ import { SevenZip, ExtractionProgress } from "./7zip";
import { WindowManager } from "./window-manager"; import { WindowManager } from "./window-manager";
import { publishExtractionCompleteNotification } from "./notifications"; import { publishExtractionCompleteNotification } from "./notifications";
import { logger } from "./logger"; import { logger } from "./logger";
import { GameExecutables } from "./game-executables";
const PROGRESS_THROTTLE_MS = 1000; const PROGRESS_THROTTLE_MS = 1000;
@@ -151,6 +152,100 @@ export class GameFilesManager {
if (publishNotification && game) { if (publishNotification && game) {
publishExtractionCompleteNotification(game); publishExtractionCompleteNotification(game);
} }
await this.searchAndBindExecutable();
}
async searchAndBindExecutable(): Promise<void> {
try {
const [download, game] = await Promise.all([
downloadsSublevel.get(this.gameKey),
gamesSublevel.get(this.gameKey),
]);
if (!download || !game || game.executablePath) {
return;
}
const executableNames = GameExecutables.getExecutablesForGame(
this.objectId
);
if (!executableNames || executableNames.length === 0) {
return;
}
if (!download.folderName) {
return;
}
const gameFolderPath = path.join(
download.downloadPath,
download.folderName
);
if (!fs.existsSync(gameFolderPath)) {
return;
}
const foundExePath = await this.findExecutableInFolder(
gameFolderPath,
executableNames
);
if (foundExePath) {
logger.info(
`[GameFilesManager] Auto-detected executable for ${this.objectId}: ${foundExePath}`
);
await gamesSublevel.put(this.gameKey, {
...game,
executablePath: foundExePath,
});
WindowManager.mainWindow?.webContents.send("on-library-batch-complete");
}
} catch (err) {
logger.error(
`[GameFilesManager] Error searching for executable: ${this.objectId}`,
err
);
}
}
private async findExecutableInFolder(
folderPath: string,
executableNames: string[]
): Promise<string | null> {
const normalizedNames = new Set(
executableNames.map((name) => name.toLowerCase())
);
try {
const entries = await fs.promises.readdir(folderPath, {
withFileTypes: true,
recursive: true,
});
for (const entry of entries) {
if (!entry.isFile()) continue;
const fileName = entry.name.toLowerCase();
if (normalizedNames.has(fileName)) {
const parentPath =
"parentPath" in entry
? entry.parentPath
: (entry as unknown as { path?: string }).path || folderPath;
return path.join(parentPath, entry.name);
}
}
} catch {
// Silently fail if folder cannot be read
}
return null;
} }
async extractDownloadedFile() { async extractDownloadedFile() {

View File

@@ -10,6 +10,7 @@ export * from "./ludusavi";
export * from "./cloud-sync"; export * from "./cloud-sync";
export * from "./7zip"; export * from "./7zip";
export * from "./game-files-manager"; export * from "./game-files-manager";
export * from "./game-executables";
export * from "./common-redist-manager"; export * from "./common-redist-manager";
export * from "./aria2"; export * from "./aria2";
export * from "./ws"; export * from "./ws";

View File

@@ -69,7 +69,7 @@ const getGameExecutables = async () => {
return gameExecutables; return gameExecutables;
}; };
const gameExecutables = await getGameExecutables(); export const gameExecutables = await getGameExecutables();
const findGamePathByProcess = async ( const findGamePathByProcess = async (
processMap: Map<string, Set<string>>, processMap: Map<string, Set<string>>,

View File

@@ -1,5 +1,6 @@
import axios from "axios"; import axios from "axios";
import http from "node:http"; import http from "node:http";
import getPort, { portNumbers } from "get-port";
import cp from "node:child_process"; import cp from "node:child_process";
import fs from "node:fs"; import fs from "node:fs";
@@ -27,11 +28,17 @@ const binaryNameByPlatform: Partial<Record<NodeJS.Platform, string>> = {
win32: "hydra-python-rpc.exe", win32: "hydra-python-rpc.exe",
}; };
const RPC_PORT_RANGE_START = 8080;
const RPC_PORT_RANGE_END = 9000;
const DEFAULT_RPC_PORT = 8084;
const HEALTH_CHECK_INTERVAL_MS = 100;
const HEALTH_CHECK_TIMEOUT_MS = 10000;
export class PythonRPC { export class PythonRPC {
public static readonly BITTORRENT_PORT = "5881"; public static readonly BITTORRENT_PORT = "5881";
public static readonly RPC_PORT = "8084";
public static readonly rpc = axios.create({ public static readonly rpc = axios.create({
baseURL: `http://localhost:${this.RPC_PORT}`, baseURL: `http://localhost:${DEFAULT_RPC_PORT}`,
httpAgent: new http.Agent({ httpAgent: new http.Agent({
family: 4, // Force IPv4 family: 4, // Force IPv4
}), }),
@@ -62,15 +69,46 @@ export class PythonRPC {
return newPassword; return newPassword;
} }
private static async waitForHealthCheck(): Promise<void> {
const startTime = Date.now();
while (Date.now() - startTime < HEALTH_CHECK_TIMEOUT_MS) {
try {
const response = await this.rpc.get("/healthcheck", { timeout: 1000 });
if (response.status === 200) {
pythonRpcLogger.log("RPC health check passed");
return;
}
} catch {
// Server not ready yet, continue polling
}
await new Promise((resolve) =>
setTimeout(resolve, HEALTH_CHECK_INTERVAL_MS)
);
}
throw new Error("RPC health check timed out");
}
public static async spawn( public static async spawn(
initialDownload?: GamePayload, initialDownload?: GamePayload,
initialSeeding?: GamePayload[] initialSeeding?: GamePayload[]
) { ) {
const rpcPassword = await this.getRPCPassword(); const rpcPassword = await this.getRPCPassword();
const port = await getPort({
port: [
DEFAULT_RPC_PORT,
...portNumbers(RPC_PORT_RANGE_START, RPC_PORT_RANGE_END),
],
});
this.rpc.defaults.baseURL = `http://localhost:${port}`;
pythonRpcLogger.log(`Using RPC port: ${port}`);
const commonArgs = [ const commonArgs = [
this.BITTORRENT_PORT, this.BITTORRENT_PORT,
this.RPC_PORT, String(port),
rpcPassword, rpcPassword,
initialDownload ? JSON.stringify(initialDownload) : "", initialDownload ? JSON.stringify(initialDownload) : "",
initialSeeding ? JSON.stringify(initialSeeding) : "", initialSeeding ? JSON.stringify(initialSeeding) : "",
@@ -91,6 +129,7 @@ export class PythonRPC {
); );
app.quit(); app.quit();
return;
} }
const childProcess = cp.spawn(binaryPath, commonArgs, { const childProcess = cp.spawn(binaryPath, commonArgs, {
@@ -99,7 +138,6 @@ export class PythonRPC {
}); });
this.logStderr(childProcess.stderr); this.logStderr(childProcess.stderr);
this.pythonProcess = childProcess; this.pythonProcess = childProcess;
} else { } else {
const scriptPath = path.join( const scriptPath = path.join(
@@ -115,11 +153,23 @@ export class PythonRPC {
}); });
this.logStderr(childProcess.stderr); this.logStderr(childProcess.stderr);
this.pythonProcess = childProcess; this.pythonProcess = childProcess;
} }
this.rpc.defaults.headers.common["x-hydra-rpc-password"] = rpcPassword; this.rpc.defaults.headers.common["x-hydra-rpc-password"] = rpcPassword;
try {
await this.waitForHealthCheck();
pythonRpcLogger.log(`Python RPC started successfully on port ${port}`);
} catch (err) {
pythonRpcLogger.log(`Failed to start Python RPC: ${err}`);
dialog.showErrorBox(
"RPC Error",
`Failed to start download service.\n\nThe service did not respond in time. Please try restarting Hydra.`
);
this.kill();
throw err;
}
} }
public static kill() { public static kill() {

View File

@@ -138,12 +138,21 @@ export class WindowManager {
(details, callback) => { (details, callback) => {
if ( if (
details.webContentsId !== this.mainWindow?.webContents.id || details.webContentsId !== this.mainWindow?.webContents.id ||
details.url.includes("chatwoot") || details.url.includes("chatwoot")
details.url.includes("workwonders")
) { ) {
return callback(details); return callback(details);
} }
if (details.url.includes("workwonders")) {
return callback({
...details,
requestHeaders: {
Origin: "https://workwonders.app",
...details.requestHeaders,
},
});
}
const userAgent = new UserAgent(); const userAgent = new UserAgent();
callback({ callback({

View File

@@ -27,6 +27,8 @@ contextBridge.exposeInMainWorld("electron", {
/* Torrenting */ /* Torrenting */
startGameDownload: (payload: StartGameDownloadPayload) => startGameDownload: (payload: StartGameDownloadPayload) =>
ipcRenderer.invoke("startGameDownload", payload), ipcRenderer.invoke("startGameDownload", payload),
addGameToQueue: (payload: StartGameDownloadPayload) =>
ipcRenderer.invoke("addGameToQueue", payload),
cancelGameDownload: (shop: GameShop, objectId: string) => cancelGameDownload: (shop: GameShop, objectId: string) =>
ipcRenderer.invoke("cancelGameDownload", shop, objectId), ipcRenderer.invoke("cancelGameDownload", shop, objectId),
pauseGameDownload: (shop: GameShop, objectId: string) => pauseGameDownload: (shop: GameShop, objectId: string) =>
@@ -37,6 +39,17 @@ contextBridge.exposeInMainWorld("electron", {
ipcRenderer.invoke("pauseGameSeed", shop, objectId), ipcRenderer.invoke("pauseGameSeed", shop, objectId),
resumeGameSeed: (shop: GameShop, objectId: string) => resumeGameSeed: (shop: GameShop, objectId: string) =>
ipcRenderer.invoke("resumeGameSeed", shop, objectId), ipcRenderer.invoke("resumeGameSeed", shop, objectId),
updateDownloadQueuePosition: (
shop: GameShop,
objectId: string,
direction: "up" | "down"
) =>
ipcRenderer.invoke(
"updateDownloadQueuePosition",
shop,
objectId,
direction
),
onDownloadProgress: (cb: (value: DownloadProgress | null) => void) => { onDownloadProgress: (cb: (value: DownloadProgress | null) => void) => {
const listener = ( const listener = (
_event: Electron.IpcRendererEvent, _event: Electron.IpcRendererEvent,
@@ -241,6 +254,7 @@ contextBridge.exposeInMainWorld("electron", {
ipcRenderer.invoke("changeGamePlayTime", shop, objectId, playtime), ipcRenderer.invoke("changeGamePlayTime", shop, objectId, playtime),
extractGameDownload: (shop: GameShop, objectId: string) => extractGameDownload: (shop: GameShop, objectId: string) =>
ipcRenderer.invoke("extractGameDownload", shop, objectId), ipcRenderer.invoke("extractGameDownload", shop, objectId),
scanInstalledGames: () => ipcRenderer.invoke("scanInstalledGames"),
getDefaultWinePrefixSelectionPath: () => getDefaultWinePrefixSelectionPath: () =>
ipcRenderer.invoke("getDefaultWinePrefixSelectionPath"), ipcRenderer.invoke("getDefaultWinePrefixSelectionPath"),
createSteamShortcut: (shop: GameShop, objectId: string) => createSteamShortcut: (shop: GameShop, objectId: string) =>

View File

@@ -1,6 +1,6 @@
import { useCallback, useEffect, useRef, useState } from "react"; import { useCallback, useEffect, useRef, useState } from "react";
import { Sidebar, BottomPanel, Header, Toast } from "@renderer/components"; import { Sidebar, BottomPanel, Header, Toast } from "@renderer/components";
import { WorkWondersSdk } from "workwonders-sdk"; import { WorkWonders } from "workwonders-sdk";
import { import {
useAppDispatch, useAppDispatch,
useAppSelector, useAppSelector,
@@ -52,7 +52,7 @@ export function App() {
const { clearDownload, setLastPacket } = useDownload(); const { clearDownload, setLastPacket } = useDownload();
const workwondersRef = useRef<WorkWondersSdk | null>(null); const workwondersRef = useRef<WorkWonders | null>(null);
const { const {
hasActiveSubscription, hasActiveSubscription,
@@ -125,16 +125,19 @@ export function App() {
const parsedLocale = const parsedLocale =
possibleLocales.find((l) => l === locale?.slice(0, 2)) ?? "en"; possibleLocales.find((l) => l === locale?.slice(0, 2)) ?? "en";
workwondersRef.current = new WorkWondersSdk(); workwondersRef.current = new WorkWonders();
await workwondersRef.current.init({ await workwondersRef.current.init({
organization: "hydra", organization: "hydra",
token, token,
locale: parsedLocale, locale: parsedLocale,
}); });
await workwondersRef.current.initChangelogWidget(); await workwondersRef.current.changelog.initChangelogWidget();
workwondersRef.current.initChangelogWidgetMini(); workwondersRef.current.changelog.initChangelogWidgetMini();
workwondersRef.current.initFeedbackWidget();
if (token) {
workwondersRef.current.feedback.initFeedbackWidget();
}
}, },
[workwondersRef] [workwondersRef]
); );

View File

@@ -61,10 +61,26 @@
cursor: pointer; cursor: pointer;
transition: all ease 0.2s; transition: all ease 0.2s;
padding: globals.$spacing-unit; padding: globals.$spacing-unit;
display: flex;
align-items: center;
justify-content: center;
&:hover { &:hover {
color: #dadbe1; color: #dadbe1;
} }
&--scanning svg {
animation: spin 2s linear infinite;
}
}
@keyframes spin {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
} }
&__section { &__section {

View File

@@ -1,7 +1,13 @@
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import { useEffect, useMemo, useRef, useState } from "react"; import { useEffect, useId, useMemo, useRef, useState } from "react";
import { useLocation, useNavigate } from "react-router-dom"; import { useLocation, useNavigate, useSearchParams } from "react-router-dom";
import { ArrowLeftIcon, SearchIcon, XIcon } from "@primer/octicons-react"; import {
ArrowLeftIcon,
SearchIcon,
SyncIcon,
XIcon,
} from "@primer/octicons-react";
import { Tooltip } from "react-tooltip";
import { import {
useAppDispatch, useAppDispatch,
@@ -12,6 +18,7 @@ import {
import "./header.scss"; import "./header.scss";
import { AutoUpdateSubHeader } from "./auto-update-sub-header"; import { AutoUpdateSubHeader } from "./auto-update-sub-header";
import { ScanGamesModal } from "./scan-games-modal";
import { setFilters, setLibrarySearchQuery } from "@renderer/features"; import { setFilters, setLibrarySearchQuery } from "@renderer/features";
import cn from "classnames"; import cn from "classnames";
import { SearchDropdown } from "@renderer/components"; import { SearchDropdown } from "@renderer/components";
@@ -29,9 +36,11 @@ const pathTitle: Record<string, string> = {
export function Header() { export function Header() {
const inputRef = useRef<HTMLInputElement>(null); const inputRef = useRef<HTMLInputElement>(null);
const searchContainerRef = useRef<HTMLDivElement>(null); const searchContainerRef = useRef<HTMLDivElement>(null);
const scanButtonTooltipId = useId();
const navigate = useNavigate(); const navigate = useNavigate();
const location = useLocation(); const location = useLocation();
const [searchParams, setSearchParams] = useSearchParams();
const { headerTitle, draggingDisabled } = useAppSelector( const { headerTitle, draggingDisabled } = useAppSelector(
(state) => state.window (state) => state.window
@@ -61,6 +70,12 @@ export function Header() {
x: 0, x: 0,
y: 0, y: 0,
}); });
const [showScanModal, setShowScanModal] = useState(false);
const [isScanning, setIsScanning] = useState(false);
const [scanResult, setScanResult] = useState<{
foundGames: { title: string; executablePath: string }[];
total: number;
} | null>(null);
const { t } = useTranslation("header"); const { t } = useTranslation("header");
@@ -224,6 +239,25 @@ export function Header() {
setActiveIndex(-1); setActiveIndex(-1);
}; };
const handleStartScan = async () => {
if (isScanning) return;
setIsScanning(true);
setScanResult(null);
setShowScanModal(false);
try {
const result = await window.electron.scanInstalledGames();
setScanResult(result);
} finally {
setIsScanning(false);
}
};
const handleClearScanResult = () => {
setScanResult(null);
};
useEffect(() => { useEffect(() => {
if (!isDropdownVisible) return; if (!isDropdownVisible) return;
@@ -235,6 +269,14 @@ export function Header() {
return () => window.removeEventListener("resize", handleResize); return () => window.removeEventListener("resize", handleResize);
}, [isDropdownVisible]); }, [isDropdownVisible]);
useEffect(() => {
if (searchParams.get("openScanModal") === "true") {
setShowScanModal(true);
searchParams.delete("openScanModal");
setSearchParams(searchParams, { replace: true });
}
}, [searchParams, setSearchParams]);
return ( return (
<> <>
<header <header
@@ -265,6 +307,21 @@ export function Header() {
</section> </section>
<section className="header__section"> <section className="header__section">
{isOnLibraryPage && window.electron.platform === "win32" && (
<button
type="button"
className={cn("header__action-button", {
"header__action-button--scanning": isScanning,
})}
onClick={() => setShowScanModal(true)}
data-tooltip-id={scanButtonTooltipId}
data-tooltip-content={t("scan_games_tooltip")}
data-tooltip-place="bottom"
>
<SyncIcon size={16} />
</button>
)}
<div <div
ref={searchContainerRef} ref={searchContainerRef}
className={cn("header__search", { className={cn("header__search", {
@@ -304,6 +361,11 @@ export function Header() {
</div> </div>
</section> </section>
</header> </header>
{isOnLibraryPage && window.electron.platform === "win32" && (
<Tooltip id={scanButtonTooltipId} style={{ zIndex: 1 }} />
)}
<AutoUpdateSubHeader /> <AutoUpdateSubHeader />
<SearchDropdown <SearchDropdown
@@ -327,6 +389,15 @@ export function Header() {
currentQuery={searchValue} currentQuery={searchValue}
searchContainerRef={searchContainerRef} searchContainerRef={searchContainerRef}
/> />
<ScanGamesModal
visible={showScanModal}
onClose={() => setShowScanModal(false)}
isScanning={isScanning}
scanResult={scanResult}
onStartScan={handleStartScan}
onClearResult={handleClearScanResult}
/>
</> </>
); );
} }

View File

@@ -0,0 +1,107 @@
@use "../../scss/globals.scss";
.scan-games-modal {
display: flex;
flex-direction: column;
gap: calc(globals.$spacing-unit * 3);
min-width: 400px;
&__description {
color: globals.$muted-color;
font-size: 14px;
line-height: 1.5;
margin: 0;
}
&__results {
display: flex;
flex-direction: column;
gap: calc(globals.$spacing-unit * 2);
}
&__result {
color: globals.$body-color;
font-size: 14px;
margin: 0;
}
&__no-results {
color: globals.$muted-color;
font-size: 14px;
margin: 0;
text-align: center;
padding: calc(globals.$spacing-unit * 2) 0;
}
&__scanning {
display: flex;
flex-direction: column;
align-items: center;
gap: calc(globals.$spacing-unit * 2);
padding: calc(globals.$spacing-unit * 3) 0;
}
&__spinner {
color: globals.$muted-color;
animation: spin 2s linear infinite;
}
&__scanning-text {
color: globals.$muted-color;
font-size: 14px;
margin: 0;
}
@keyframes spin {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
}
&__games-list {
list-style: none;
margin: 0;
max-height: 200px;
overflow-y: auto;
display: flex;
flex-direction: column;
gap: globals.$spacing-unit;
background-color: globals.$dark-background-color;
border-radius: 4px;
padding: calc(globals.$spacing-unit * 2);
}
&__game-item {
display: flex;
flex-direction: column;
gap: 4px;
padding-bottom: globals.$spacing-unit;
border-bottom: 1px solid globals.$border-color;
&:last-child {
border-bottom: none;
padding-bottom: 0;
}
}
&__game-title {
color: globals.$body-color;
font-size: 14px;
font-weight: 500;
}
&__game-path {
color: globals.$muted-color;
font-size: 12px;
word-break: break-all;
}
&__actions {
display: flex;
justify-content: flex-end;
gap: calc(globals.$spacing-unit * 2);
}
}

View File

@@ -0,0 +1,126 @@
import { useTranslation } from "react-i18next";
import { SyncIcon } from "@primer/octicons-react";
import { Button, Modal } from "@renderer/components";
import "./scan-games-modal.scss";
interface FoundGame {
title: string;
executablePath: string;
}
interface ScanResult {
foundGames: FoundGame[];
total: number;
}
export interface ScanGamesModalProps {
visible: boolean;
onClose: () => void;
isScanning: boolean;
scanResult: ScanResult | null;
onStartScan: () => void;
onClearResult: () => void;
}
export function ScanGamesModal({
visible,
onClose,
isScanning,
scanResult,
onStartScan,
onClearResult,
}: Readonly<ScanGamesModalProps>) {
const { t } = useTranslation("header");
const handleClose = () => {
onClose();
};
const handleStartScan = () => {
onStartScan();
};
const handleScanAgain = () => {
onClearResult();
onStartScan();
};
return (
<Modal
visible={visible}
title={t("scan_games_title")}
onClose={handleClose}
clickOutsideToClose={!isScanning}
>
<div className="scan-games-modal">
{!scanResult && !isScanning && (
<p className="scan-games-modal__description">
{t("scan_games_description")}
</p>
)}
{isScanning && !scanResult && (
<div className="scan-games-modal__scanning">
<SyncIcon size={24} className="scan-games-modal__spinner" />
<p className="scan-games-modal__scanning-text">
{t("scan_games_in_progress")}
</p>
</div>
)}
{scanResult && (
<div className="scan-games-modal__results">
{scanResult.foundGames.length > 0 ? (
<>
<p className="scan-games-modal__result">
{t("scan_games_result", {
found: scanResult.foundGames.length,
total: scanResult.total,
})}
</p>
<ul className="scan-games-modal__games-list">
{scanResult.foundGames.map((game) => (
<li
key={game.executablePath}
className="scan-games-modal__game-item"
>
<span className="scan-games-modal__game-title">
{game.title}
</span>
<span className="scan-games-modal__game-path">
{game.executablePath}
</span>
</li>
))}
</ul>
</>
) : (
<p className="scan-games-modal__no-results">
{t("scan_games_no_results")}
</p>
)}
</div>
)}
<div className="scan-games-modal__actions">
<Button theme="outline" onClick={handleClose}>
{scanResult ? t("scan_games_close") : t("scan_games_cancel")}
</Button>
{!scanResult && (
<Button onClick={handleStartScan} disabled={isScanning}>
{t("scan_games_start")}
</Button>
)}
{scanResult && (
<Button onClick={handleScanAgain}>
{t("scan_games_scan_again")}
</Button>
)}
</div>
</div>
</Modal>
);
}

View File

@@ -1,7 +1,7 @@
import { useNavigate } from "react-router-dom"; import { useNavigate } from "react-router-dom";
import { BellIcon } from "@primer/octicons-react"; import { BellIcon } from "@primer/octicons-react";
import { useAppSelector, useUserDetails } from "@renderer/hooks"; import { useAppSelector, useUserDetails } from "@renderer/hooks";
import { useCallback, useEffect, useMemo, useState } from "react"; import { useCallback, useEffect, useMemo, useState, useRef } from "react";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import SteamLogo from "@renderer/assets/steam-logo.svg?react"; import SteamLogo from "@renderer/assets/steam-logo.svg?react";
import { Avatar } from "../avatar/avatar"; import { Avatar } from "../avatar/avatar";
@@ -20,51 +20,60 @@ export function SidebarProfile() {
const { gameRunning } = useAppSelector((state) => state.gameRunning); const { gameRunning } = useAppSelector((state) => state.gameRunning);
const [notificationCount, setNotificationCount] = useState(0); const [notificationCount, setNotificationCount] = useState(0);
const apiNotificationCountRef = useRef(0);
const hasFetchedInitialCount = useRef(false);
const fetchNotificationCount = useCallback(async () => { const fetchLocalNotificationCount = useCallback(async () => {
try { try {
// Always fetch local notification count
const localCount = await window.electron.getLocalNotificationsCount(); const localCount = await window.electron.getLocalNotificationsCount();
setNotificationCount(localCount + apiNotificationCountRef.current);
} catch (error) {
logger.error("Failed to fetch local notification count", error);
}
}, []);
// Fetch API notification count only if logged in const fetchApiNotificationCount = useCallback(async () => {
let apiCount = 0;
if (userDetails) {
try { try {
const response = const response =
await window.electron.hydraApi.get<NotificationCountResponse>( await window.electron.hydraApi.get<NotificationCountResponse>(
"/profile/notifications/count", "/profile/notifications/count",
{ needsAuth: true } { needsAuth: true }
); );
apiCount = response.count; apiNotificationCountRef.current = response.count;
} catch { } catch {
// Ignore API errors // Ignore API errors
} }
} fetchLocalNotificationCount();
}, [fetchLocalNotificationCount]);
setNotificationCount(localCount + apiCount);
} catch (error) {
logger.error("Failed to fetch notification count", error);
}
}, [userDetails]);
// Initial fetch on mount (only once)
useEffect(() => { useEffect(() => {
fetchNotificationCount(); fetchLocalNotificationCount();
}, [fetchLocalNotificationCount]);
const interval = setInterval(fetchNotificationCount, 60000); // Fetch API count when user logs in (only if not already fetched)
return () => clearInterval(interval); useEffect(() => {
}, [fetchNotificationCount]); if (userDetails && !hasFetchedInitialCount.current) {
hasFetchedInitialCount.current = true;
fetchApiNotificationCount();
} else if (!userDetails) {
hasFetchedInitialCount.current = false;
apiNotificationCountRef.current = 0;
fetchLocalNotificationCount();
}
}, [userDetails, fetchApiNotificationCount, fetchLocalNotificationCount]);
useEffect(() => { useEffect(() => {
const unsubscribe = window.electron.onLocalNotificationCreated(() => { const unsubscribe = window.electron.onLocalNotificationCreated(() => {
fetchNotificationCount(); fetchLocalNotificationCount();
}); });
return () => unsubscribe(); return () => unsubscribe();
}, [fetchNotificationCount]); }, [fetchLocalNotificationCount]);
useEffect(() => { useEffect(() => {
const handleNotificationsChange = () => { const handleNotificationsChange = () => {
fetchNotificationCount(); fetchLocalNotificationCount();
}; };
window.addEventListener("notificationsChanged", handleNotificationsChange); window.addEventListener("notificationsChanged", handleNotificationsChange);
@@ -74,15 +83,18 @@ export function SidebarProfile() {
handleNotificationsChange handleNotificationsChange
); );
}; };
}, [fetchNotificationCount]); }, [fetchLocalNotificationCount]);
useEffect(() => { useEffect(() => {
const unsubscribe = window.electron.onSyncNotificationCount(() => { const unsubscribe = window.electron.onSyncNotificationCount(
fetchNotificationCount(); (notification) => {
}); apiNotificationCountRef.current = notification.notificationCount;
fetchLocalNotificationCount();
}
);
return () => unsubscribe(); return () => unsubscribe();
}, [fetchNotificationCount]); }, [fetchLocalNotificationCount]);
const handleProfileClick = () => { const handleProfileClick = () => {
if (userDetails === null) { if (userDetails === null) {

View File

@@ -225,6 +225,16 @@ export function GameDetailsContextProvider({
}; };
}, [game?.id, isGameRunning, updateGame]); }, [game?.id, isGameRunning, updateGame]);
useEffect(() => {
const unsubscribe = window.electron.onLibraryBatchComplete(() => {
updateGame();
});
return () => {
unsubscribe();
};
}, [updateGame]);
useEffect(() => { useEffect(() => {
const handler = (ev: Event) => { const handler = (ev: Event) => {
try { try {

View File

@@ -47,11 +47,19 @@ declare global {
startGameDownload: ( startGameDownload: (
payload: StartGameDownloadPayload payload: StartGameDownloadPayload
) => Promise<{ ok: boolean; error?: string }>; ) => Promise<{ ok: boolean; error?: string }>;
addGameToQueue: (
payload: StartGameDownloadPayload
) => Promise<{ ok: boolean; error?: string }>;
cancelGameDownload: (shop: GameShop, objectId: string) => Promise<void>; cancelGameDownload: (shop: GameShop, objectId: string) => Promise<void>;
pauseGameDownload: (shop: GameShop, objectId: string) => Promise<void>; pauseGameDownload: (shop: GameShop, objectId: string) => Promise<void>;
resumeGameDownload: (shop: GameShop, objectId: string) => Promise<void>; resumeGameDownload: (shop: GameShop, objectId: string) => Promise<void>;
pauseGameSeed: (shop: GameShop, objectId: string) => Promise<void>; pauseGameSeed: (shop: GameShop, objectId: string) => Promise<void>;
resumeGameSeed: (shop: GameShop, objectId: string) => Promise<void>; resumeGameSeed: (shop: GameShop, objectId: string) => Promise<void>;
updateDownloadQueuePosition: (
shop: GameShop,
objectId: string,
direction: "up" | "down"
) => Promise<boolean>;
onDownloadProgress: ( onDownloadProgress: (
cb: (value: DownloadProgress | null) => void cb: (value: DownloadProgress | null) => void
) => () => Electron.IpcRenderer; ) => () => Electron.IpcRenderer;
@@ -211,6 +219,10 @@ declare global {
minimized: boolean; minimized: boolean;
}) => Promise<void>; }) => Promise<void>;
extractGameDownload: (shop: GameShop, objectId: string) => Promise<boolean>; extractGameDownload: (shop: GameShop, objectId: string) => Promise<boolean>;
scanInstalledGames: () => Promise<{
foundGames: { title: string; executablePath: string }[];
total: number;
}>;
onExtractionComplete: ( onExtractionComplete: (
cb: (shop: GameShop, objectId: string) => void cb: (shop: GameShop, objectId: string) => void
) => () => Electron.IpcRenderer; ) => () => Electron.IpcRenderer;

View File

@@ -38,6 +38,14 @@ export function useDownload() {
return response; return response;
}; };
const addGameToQueue = async (payload: StartGameDownloadPayload) => {
const response = await window.electron.addGameToQueue(payload);
if (response.ok) updateLibrary();
return response;
};
const pauseDownload = async (shop: GameShop, objectId: string) => { const pauseDownload = async (shop: GameShop, objectId: string) => {
await window.electron.pauseGameDownload(shop, objectId); await window.electron.pauseGameDownload(shop, objectId);
await updateLibrary(); await updateLibrary();
@@ -61,10 +69,16 @@ export function useDownload() {
}; };
const cancelDownload = async (shop: GameShop, objectId: string) => { const cancelDownload = async (shop: GameShop, objectId: string) => {
await window.electron.cancelGameDownload(shop, objectId); const gameId = `${shop}:${objectId}`;
dispatch(clearDownload()); const isActiveDownload = lastPacket?.gameId === gameId;
updateLibrary();
await window.electron.cancelGameDownload(shop, objectId);
if (isActiveDownload) {
dispatch(clearDownload());
}
updateLibrary();
removeGameInstaller(shop, objectId); removeGameInstaller(shop, objectId);
}; };
@@ -113,6 +127,7 @@ export function useDownload() {
lastPacket, lastPacket,
eta: calculateETA(), eta: calculateETA(),
startDownload, startDownload,
addGameToQueue,
pauseDownload, pauseDownload,
resumeDownload, resumeDownload,
cancelDownload, cancelDownload,

View File

@@ -21,6 +21,7 @@ import resources from "@locales";
import { logger } from "./logger"; import { logger } from "./logger";
import { addCookieInterceptor } from "./cookies"; import { addCookieInterceptor } from "./cookies";
import * as Sentry from "@sentry/react";
import { levelDBService } from "./services/leveldb.service"; import { levelDBService } from "./services/leveldb.service";
import Catalogue from "./pages/catalogue/catalogue"; import Catalogue from "./pages/catalogue/catalogue";
import Home from "./pages/home/home"; import Home from "./pages/home/home";
@@ -36,6 +37,18 @@ import { AchievementNotification } from "./pages/achievements/notification/achie
console.log = logger.log; console.log = logger.log;
Sentry.init({
dsn: import.meta.env.RENDERER_VITE_SENTRY_DSN,
integrations: [
Sentry.browserTracingIntegration(),
Sentry.replayIntegration(),
],
tracesSampleRate: 0.5,
replaysSessionSampleRate: 0,
replaysOnErrorSampleRate: 0,
release: "hydra-launcher@" + (await window.electron.getVersion()),
});
const isStaging = await window.electron.isStaging(); const isStaging = await window.electron.isStaging();
addCookieInterceptor(isStaging); addCookieInterceptor(isStaging);

View File

@@ -427,7 +427,7 @@
padding: 0; padding: 0;
cursor: pointer; cursor: pointer;
text-align: left; text-align: left;
width: 100%; width: fit-content;
transition: opacity 0.2s ease; transition: opacity 0.2s ease;
&:focus, &:focus,
@@ -509,6 +509,7 @@
&__simple-menu-btn { &__simple-menu-btn {
padding: calc(globals.$spacing-unit); padding: calc(globals.$spacing-unit);
min-height: unset; min-height: unset;
border-radius: 8px;
} }
&__simple-action-btn { &__simple-action-btn {
@@ -516,6 +517,7 @@
min-height: unset; min-height: unset;
gap: calc(globals.$spacing-unit); gap: calc(globals.$spacing-unit);
min-width: 120px; min-width: 120px;
border-radius: 8px;
} }
&__progress-wrapper { &__progress-wrapper {

View File

@@ -26,6 +26,8 @@ import {
DropdownMenuItem, DropdownMenuItem,
} from "@renderer/components/dropdown-menu/dropdown-menu"; } from "@renderer/components/dropdown-menu/dropdown-menu";
import { import {
ArrowDownIcon,
ArrowUpIcon,
ClockIcon, ClockIcon,
ColumnsIcon, ColumnsIcon,
DownloadIcon, DownloadIcon,
@@ -40,6 +42,44 @@ import {
import { MoreVertical, Folder } from "lucide-react"; import { MoreVertical, Folder } from "lucide-react";
import { average } from "color.js"; import { average } from "color.js";
function hexToRgb(hex: string): [number, number, number] {
let h = hex.replace("#", "");
if (h.length === 3) {
h = h[0] + h[0] + h[1] + h[1] + h[2] + h[2];
}
const r = Number.parseInt(h.substring(0, 2), 16) || 0;
const g = Number.parseInt(h.substring(2, 4), 16) || 0;
const b = Number.parseInt(h.substring(4, 6), 16) || 0;
return [r, g, b];
}
function isTooCloseRGB(a: string, b: string, threshold: number): boolean {
const [r1, g1, b1] = hexToRgb(a);
const [r2, g2, b2] = hexToRgb(b);
const distance = Math.sqrt(
Math.pow(r1 - r2, 2) + Math.pow(g1 - g2, 2) + Math.pow(b1 - b2, 2)
);
return distance < threshold;
}
const CHART_BACKGROUND_COLOR = "#1a1a1a";
const COLOR_DISTANCE_THRESHOLD = 28;
const FALLBACK_CHART_COLOR = "#fff";
function pickChartColor(dominant?: string): string {
if (!dominant || typeof dominant !== "string" || !dominant.startsWith("#")) {
return FALLBACK_CHART_COLOR;
}
if (
isTooCloseRGB(dominant, CHART_BACKGROUND_COLOR, COLOR_DISTANCE_THRESHOLD)
) {
return FALLBACK_CHART_COLOR;
}
return dominant;
}
interface AnimatedPercentageProps { interface AnimatedPercentageProps {
value: number; value: number;
} }
@@ -442,6 +482,7 @@ export interface DownloadGroupProps {
openDeleteGameModal: (shop: GameShop, objectId: string) => void; openDeleteGameModal: (shop: GameShop, objectId: string) => void;
openGameInstaller: (shop: GameShop, objectId: string) => void; openGameInstaller: (shop: GameShop, objectId: string) => void;
seedingStatus: SeedingStatus[]; seedingStatus: SeedingStatus[];
queuedGameIds?: string[];
} }
export function DownloadGroup({ export function DownloadGroup({
@@ -450,6 +491,7 @@ export function DownloadGroup({
openDeleteGameModal, openDeleteGameModal,
openGameInstaller, openGameInstaller,
seedingStatus, seedingStatus,
queuedGameIds = [],
}: Readonly<DownloadGroupProps>) { }: Readonly<DownloadGroupProps>) {
const { t } = useTranslation("downloads"); const { t } = useTranslation("downloads");
const { t: tGameDetails } = useTranslation("game_details"); const { t: tGameDetails } = useTranslation("game_details");
@@ -690,6 +732,18 @@ export function DownloadGroup({
setGameToCancelObjectId(null); setGameToCancelObjectId(null);
}, []); }, []);
const handleMoveInQueue = useCallback(
async (shop: GameShop, objectId: string, direction: "up" | "down") => {
await window.electron.updateDownloadQueuePosition(
shop,
objectId,
direction
);
updateLibrary();
},
[updateLibrary]
);
const getGameActions = (game: LibraryGame): DropdownMenuItem[] => { const getGameActions = (game: LibraryGame): DropdownMenuItem[] => {
const download = lastPacket?.download; const download = lastPacket?.download;
const isGameDownloading = isGameDownloadingMap[game.id]; const isGameDownloading = isGameDownloadingMap[game.id];
@@ -698,14 +752,6 @@ export function DownloadGroup({
if (game.download?.progress === 1) { if (game.download?.progress === 1) {
const actions = [ const actions = [
{
label: t("install"),
disabled: deleting,
onClick: () => {
openGameInstaller(game.shop, game.objectId);
},
icon: <DownloadIcon />,
},
{ {
label: t("extract"), label: t("extract"),
disabled: game.download.extracting, disabled: game.download.extracting,
@@ -773,7 +819,12 @@ export function DownloadGroup({
(download?.downloader === Downloader.TorBox && (download?.downloader === Downloader.TorBox &&
!userPreferences?.torBoxApiToken); !userPreferences?.torBoxApiToken);
return [ const queueIndex = queuedGameIds.indexOf(game.id);
const isFirstInQueue = queueIndex === 0;
const isLastInQueue = queueIndex === queuedGameIds.length - 1;
const isInQueue = queueIndex !== -1;
const actions = [
{ {
label: t("resume"), label: t("resume"),
disabled: isResumeDisabled, disabled: isResumeDisabled,
@@ -782,6 +833,22 @@ export function DownloadGroup({
}, },
icon: <PlayIcon />, icon: <PlayIcon />,
}, },
{
label: t("move_up"),
show: isInQueue && !isFirstInQueue,
onClick: () => {
handleMoveInQueue(game.shop, game.objectId, "up");
},
icon: <ArrowUpIcon />,
},
{
label: t("move_down"),
show: isInQueue && !isLastInQueue,
onClick: () => {
handleMoveInQueue(game.shop, game.objectId, "down");
},
icon: <ArrowDownIcon />,
},
{ {
label: t("cancel"), label: t("cancel"),
onClick: () => { onClick: () => {
@@ -790,6 +857,8 @@ export function DownloadGroup({
icon: <XCircleIcon />, icon: <XCircleIcon />,
}, },
]; ];
return actions.filter((action) => action.show !== false);
}; };
const downloadInfo = useMemo( const downloadInfo = useMemo(
@@ -871,7 +940,7 @@ export function DownloadGroup({
currentProgress = lastPacket.progress; currentProgress = lastPacket.progress;
} }
const dominantColor = dominantColors[game.id] || "#fff"; const dominantColor = pickChartColor(dominantColors[game.id]);
return ( return (
<> <>

View File

@@ -103,18 +103,26 @@ export default function Downloads() {
}; };
}, [library, lastPacket?.gameId, extraction?.visibleId]); }, [library, lastPacket?.gameId, extraction?.visibleId]);
const queuedGameIds = useMemo(
() => libraryGroup.queued.map((game) => game.id),
[libraryGroup.queued]
);
const downloadGroups = [ const downloadGroups = [
{ {
title: t("download_in_progress"), title: t("download_in_progress"),
library: libraryGroup.downloading, library: libraryGroup.downloading,
queuedGameIds: [] as string[],
}, },
{ {
title: t("queued_downloads"), title: t("queued_downloads"),
library: libraryGroup.queued, library: libraryGroup.queued,
queuedGameIds,
}, },
{ {
title: t("downloads_completed"), title: t("downloads_completed"),
library: libraryGroup.complete, library: libraryGroup.complete,
queuedGameIds: [] as string[],
}, },
]; ];
@@ -142,10 +150,11 @@ export default function Downloads() {
<DownloadGroup <DownloadGroup
key={group.title} key={group.title}
title={group.title} title={group.title}
library={orderBy(group.library, ["updatedAt"], ["desc"])} library={group.library}
openDeleteGameModal={handleOpenDeleteGameModal} openDeleteGameModal={handleOpenDeleteGameModal}
openGameInstaller={handleOpenGameInstaller} openGameInstaller={handleOpenGameInstaller}
seedingStatus={seedingStatus} seedingStatus={seedingStatus}
queuedGameIds={group.queuedGameIds}
/> />
))} ))}
</div> </div>

View File

@@ -37,7 +37,7 @@ export default function GameDetails() {
const fromRandomizer = searchParams.get("fromRandomizer"); const fromRandomizer = searchParams.get("fromRandomizer");
const gameTitle = searchParams.get("title"); const gameTitle = searchParams.get("title");
const { startDownload } = useDownload(); const { startDownload, addGameToQueue } = useDownload();
const { t } = useTranslation("game_details"); const { t } = useTranslation("game_details");
@@ -100,9 +100,11 @@ export default function GameDetails() {
repack: GameRepack, repack: GameRepack,
downloader: Downloader, downloader: Downloader,
downloadPath: string, downloadPath: string,
automaticallyExtract: boolean automaticallyExtract: boolean,
addToQueueOnly = false
) => { ) => {
const response = await startDownload({ const response = addToQueueOnly
? await addGameToQueue({
objectId: objectId!, objectId: objectId!,
title: gameTitle, title: gameTitle,
downloader, downloader,
@@ -110,6 +112,17 @@ export default function GameDetails() {
downloadPath, downloadPath,
uri: selectRepackUri(repack, downloader), uri: selectRepackUri(repack, downloader),
automaticallyExtract: automaticallyExtract, automaticallyExtract: automaticallyExtract,
fileSize: repack.fileSize,
})
: await startDownload({
objectId: objectId!,
title: gameTitle,
downloader,
shop,
downloadPath,
uri: selectRepackUri(repack, downloader),
automaticallyExtract: automaticallyExtract,
fileSize: repack.fileSize,
}); });
if (response.ok) { if (response.ok) {

View File

@@ -12,11 +12,17 @@ import {
DownloadIcon, DownloadIcon,
SyncIcon, SyncIcon,
CheckCircleFillIcon, CheckCircleFillIcon,
PlusIcon,
} from "@primer/octicons-react"; } from "@primer/octicons-react";
import { Downloader, formatBytes, getDownloadersForUri } from "@shared"; import { Downloader, formatBytes, getDownloadersForUri } from "@shared";
import type { GameRepack } from "@types"; import type { GameRepack } from "@types";
import { DOWNLOADER_NAME } from "@renderer/constants"; import { DOWNLOADER_NAME } from "@renderer/constants";
import { useAppSelector, useFeature, useToast } from "@renderer/hooks"; import {
useAppSelector,
useDownload,
useFeature,
useToast,
} from "@renderer/hooks";
import { motion } from "framer-motion"; import { motion } from "framer-motion";
import { Tooltip } from "react-tooltip"; import { Tooltip } from "react-tooltip";
import { RealDebridInfoModal } from "./real-debrid-info-modal"; import { RealDebridInfoModal } from "./real-debrid-info-modal";
@@ -29,7 +35,8 @@ export interface DownloadSettingsModalProps {
repack: GameRepack, repack: GameRepack,
downloader: Downloader, downloader: Downloader,
downloadPath: string, downloadPath: string,
automaticallyExtract: boolean automaticallyExtract: boolean,
addToQueueOnly?: boolean
) => Promise<{ ok: boolean; error?: string }>; ) => Promise<{ ok: boolean; error?: string }>;
repack: GameRepack | null; repack: GameRepack | null;
} }
@@ -46,8 +53,11 @@ export function DownloadSettingsModal({
(state) => state.userPreferences.value (state) => state.userPreferences.value
); );
const { lastPacket } = useDownload();
const { showErrorToast } = useToast(); const { showErrorToast } = useToast();
const hasActiveDownload = lastPacket !== null;
const [diskFreeSpace, setDiskFreeSpace] = useState<number | null>(null); const [diskFreeSpace, setDiskFreeSpace] = useState<number | null>(null);
const [selectedPath, setSelectedPath] = useState(""); const [selectedPath, setSelectedPath] = useState("");
const [downloadStarting, setDownloadStarting] = useState(false); const [downloadStarting, setDownloadStarting] = useState(false);
@@ -220,7 +230,8 @@ export function DownloadSettingsModal({
repack, repack,
selectedDownloader!, selectedDownloader!,
selectedPath, selectedPath,
automaticExtractionEnabled automaticExtractionEnabled,
hasActiveDownload
); );
if (response.ok) { if (response.ok) {
@@ -456,6 +467,11 @@ export function DownloadSettingsModal({
<SyncIcon className="download-settings-modal__loading-spinner" /> <SyncIcon className="download-settings-modal__loading-spinner" />
{t("loading")} {t("loading")}
</> </>
) : hasActiveDownload ? (
<>
<PlusIcon />
{t("add_to_queue")}
</>
) : ( ) : (
<> <>
<DownloadIcon /> <DownloadIcon />

View File

@@ -40,6 +40,34 @@
gap: calc(globals.$spacing-unit * 1); gap: calc(globals.$spacing-unit * 1);
color: globals.$body-color; color: globals.$body-color;
padding: calc(globals.$spacing-unit * 2); padding: calc(globals.$spacing-unit * 2);
padding-right: calc(globals.$spacing-unit * 4);
position: relative;
}
&__availability-orb {
position: absolute;
top: calc(globals.$spacing-unit * 1.5);
right: calc(globals.$spacing-unit * 1.5);
width: 8px;
height: 8px;
border-radius: 50%;
flex-shrink: 0;
&--online {
background-color: #22c55e;
box-shadow: 0 0 6px rgba(34, 197, 94, 0.5);
}
&--partial {
background-color: #eab308;
box-shadow: 0 0 6px rgba(234, 179, 8, 0.5);
}
&--offline {
background-color: #ef4444;
opacity: 0.7;
box-shadow: 0 0 6px rgba(239, 68, 68, 0.4);
}
} }
&__repack-title { &__repack-title {

View File

@@ -6,6 +6,7 @@ import {
ChevronDownIcon, ChevronDownIcon,
ChevronUpIcon, ChevronUpIcon,
} from "@primer/octicons-react"; } from "@primer/octicons-react";
import { Tooltip } from "react-tooltip";
import { import {
Badge, Badge,
@@ -38,7 +39,8 @@ export interface RepacksModalProps {
repack: GameRepack, repack: GameRepack,
downloader: Downloader, downloader: Downloader,
downloadPath: string, downloadPath: string,
automaticallyExtract: boolean automaticallyExtract: boolean,
addToQueueOnly?: boolean
) => Promise<{ ok: boolean; error?: string }>; ) => Promise<{ ok: boolean; error?: string }>;
onClose: () => void; onClose: () => void;
} }
@@ -185,6 +187,20 @@ export function RepacksModal({
); );
}, [repacks, hashesInDebrid]); }, [repacks, hashesInDebrid]);
const getRepackAvailabilityStatus = (
repack: GameRepack
): "online" | "partial" | "offline" => {
const unavailableSet = new Set(repack.unavailableUris ?? []);
const availableCount = repack.uris.filter(
(uri) => !unavailableSet.has(uri)
).length;
const unavailableCount = repack.uris.length - availableCount;
if (unavailableCount === 0) return "online";
if (availableCount === 0) return "offline";
return "partial";
};
useEffect(() => { useEffect(() => {
const term = filterTerm.trim().toLowerCase(); const term = filterTerm.trim().toLowerCase();
@@ -363,6 +379,8 @@ export function RepacksModal({
filteredRepacks.map((repack) => { filteredRepacks.map((repack) => {
const isLastDownloadedOption = const isLastDownloadedOption =
checkIfLastDownloadedOption(repack); checkIfLastDownloadedOption(repack);
const availabilityStatus = getRepackAvailabilityStatus(repack);
const tooltipId = `availability-orb-${repack.id}`;
return ( return (
<Button <Button
@@ -371,6 +389,13 @@ export function RepacksModal({
onClick={() => handleRepackClick(repack)} onClick={() => handleRepackClick(repack)}
className="repacks-modal__repack-button" className="repacks-modal__repack-button"
> >
<span
className={`repacks-modal__availability-orb repacks-modal__availability-orb--${availabilityStatus}`}
data-tooltip-id={tooltipId}
data-tooltip-content={t(`source_${availabilityStatus}`)}
/>
<Tooltip id={tooltipId} />
<p className="repacks-modal__repack-title"> <p className="repacks-modal__repack-title">
{repack.title} {repack.title}
{userPreferences?.enableNewDownloadOptionsBadges !== {userPreferences?.enableNewDownloadOptionsBadges !==

View File

@@ -58,6 +58,8 @@ export function LocalNotificationItem({
return <SyncIcon size={24} />; return <SyncIcon size={24} />;
case "ACHIEVEMENT_UNLOCKED": case "ACHIEVEMENT_UNLOCKED":
return <TrophyIcon size={24} />; return <TrophyIcon size={24} />;
case "SCAN_GAMES_COMPLETE":
return <SyncIcon size={24} />;
default: default:
return <DownloadIcon size={24} />; return <DownloadIcon size={24} />;
} }

View File

@@ -14,10 +14,6 @@
&__section { &__section {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
&:not(:last-child) {
margin-bottom: calc(globals.$spacing-unit * 2);
}
} }
&__section-header { &__section-header {

View File

@@ -51,6 +51,25 @@ export const formatBytes = (bytes: number): string => {
return `${Math.trunc(formatedByte * 10) / 10} ${FORMAT[base]}`; return `${Math.trunc(formatedByte * 10) / 10} ${FORMAT[base]}`;
}; };
export const parseBytes = (sizeString: string | null): number | null => {
if (!sizeString) return null;
const regex = /^([\d.,]+)\s*([A-Za-z]+)$/;
const match = regex.exec(sizeString.trim());
if (!match) return null;
const value = Number.parseFloat(match[1].replaceAll(",", "."));
const unit = match[2].toUpperCase();
if (Number.isNaN(value)) return null;
const unitIndex = FORMAT.indexOf(unit);
if (unitIndex === -1) return null;
const byteKBase = 1024;
return Math.round(value * Math.pow(byteKBase, unitIndex));
};
export const formatBytesToMbps = (bytesPerSecond: number): string => { export const formatBytesToMbps = (bytesPerSecond: number): string => {
const bitsPerSecond = bytesPerSecond * 8; const bitsPerSecond = bytesPerSecond * 8;
const mbps = bitsPerSecond / (1024 * 1024); const mbps = bitsPerSecond / (1024 * 1024);
@@ -123,7 +142,10 @@ export const getDownloadersForUri = (uri: string) => {
if (uri.startsWith("https://fuckingfast.co")) { if (uri.startsWith("https://fuckingfast.co")) {
return [Downloader.FuckingFast]; return [Downloader.FuckingFast];
} }
if (uri.startsWith("https://vikingfile.com")) { if (
uri.startsWith("https://vikingfile.com") ||
uri.startsWith("https://vik1ngfile.site")
) {
return [Downloader.VikingFile]; return [Downloader.VikingFile];
} }
if (uri.startsWith("https://www.rootz.so")) { if (uri.startsWith("https://www.rootz.so")) {

View File

@@ -118,6 +118,7 @@ export interface StartGameDownloadPayload {
downloadPath: string; downloadPath: string;
downloader: Downloader; downloader: Downloader;
automaticallyExtract: boolean; automaticallyExtract: boolean;
fileSize?: string | null;
} }
export interface UserFriend { export interface UserFriend {
@@ -330,7 +331,8 @@ export type LocalNotificationType =
| "EXTRACTION_COMPLETE" | "EXTRACTION_COMPLETE"
| "DOWNLOAD_COMPLETE" | "DOWNLOAD_COMPLETE"
| "UPDATE_AVAILABLE" | "UPDATE_AVAILABLE"
| "ACHIEVEMENT_UNLOCKED"; | "ACHIEVEMENT_UNLOCKED"
| "SCAN_GAMES_COMPLETE";
export interface Notification { export interface Notification {
id: string; id: string;

View File

@@ -2174,6 +2174,60 @@
resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.52.4.tgz#5b2dd648a960b8fa00d76f2cc4eea2f03daa80f4" resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.52.4.tgz#5b2dd648a960b8fa00d76f2cc4eea2f03daa80f4"
integrity sha512-bf9PtUa0u8IXDVxzRToFQKsNCRz9qLYfR/MpECxl4mRoWYjAeFjgxj1XdZr2M/GNVpT05p+LgQOHopYDlUu6/w== integrity sha512-bf9PtUa0u8IXDVxzRToFQKsNCRz9qLYfR/MpECxl4mRoWYjAeFjgxj1XdZr2M/GNVpT05p+LgQOHopYDlUu6/w==
"@sentry-internal/browser-utils@10.33.0":
version "10.33.0"
resolved "https://registry.yarnpkg.com/@sentry-internal/browser-utils/-/browser-utils-10.33.0.tgz#4a5d98352267b63fcc449efe14627c0fc082089e"
integrity sha512-nDJFHAfiFifBfJB0OF6DV6BIsIV5uah4lDsV4UBAgPBf+YAHclO10y1gi2U/JMh58c+s4lXi9p+PI1TFXZ0c6w==
dependencies:
"@sentry/core" "10.33.0"
"@sentry-internal/feedback@10.33.0":
version "10.33.0"
resolved "https://registry.yarnpkg.com/@sentry-internal/feedback/-/feedback-10.33.0.tgz#5865b4a68d607bb48d8159a100464ae640a638e7"
integrity sha512-sN/VLWtEf0BeV6w6wldIpTxUQxNVc9o9tjLRQa8je1ZV2FCgXA124Iff/zsowsz82dLqtg7qp6GA5zYXVq+JMA==
dependencies:
"@sentry/core" "10.33.0"
"@sentry-internal/replay-canvas@10.33.0":
version "10.33.0"
resolved "https://registry.yarnpkg.com/@sentry-internal/replay-canvas/-/replay-canvas-10.33.0.tgz#9ea15b320618ad220e5d8f7c804a0d9ca55b04af"
integrity sha512-MTmP6uoAVzw4CCPeqCgCLsRSiOfGLxgyMFjGTCW3E7t62MJ9S0H5sLsQ34sHxXUa1gFU9UNAjEvRRpZ0JvWrPw==
dependencies:
"@sentry-internal/replay" "10.33.0"
"@sentry/core" "10.33.0"
"@sentry-internal/replay@10.33.0":
version "10.33.0"
resolved "https://registry.yarnpkg.com/@sentry-internal/replay/-/replay-10.33.0.tgz#8cfe3a353731fcd81e7afb646b6befeb0f9feb0f"
integrity sha512-UOU9PYxuXnPop3HoQ3l4Q7SZUXJC3Vmfm0Adgad8U03UcrThWIHYc5CxECSrVzfDFNOT7w9o7HQgRAgWxBPMXg==
dependencies:
"@sentry-internal/browser-utils" "10.33.0"
"@sentry/core" "10.33.0"
"@sentry/browser@10.33.0":
version "10.33.0"
resolved "https://registry.yarnpkg.com/@sentry/browser/-/browser-10.33.0.tgz#33284952a1cdf43cdac15ac144c85e81e7cbaa93"
integrity sha512-iWiPjik9zetM84jKfk01UveW1J0+X7w8XmJ8+IrhTyNDBVUWCRJWD8FrksiN1dRSg5mFWgfMRzKMz27hAScRwg==
dependencies:
"@sentry-internal/browser-utils" "10.33.0"
"@sentry-internal/feedback" "10.33.0"
"@sentry-internal/replay" "10.33.0"
"@sentry-internal/replay-canvas" "10.33.0"
"@sentry/core" "10.33.0"
"@sentry/core@10.33.0":
version "10.33.0"
resolved "https://registry.yarnpkg.com/@sentry/core/-/core-10.33.0.tgz#ea4964fbec290503b419ccaf1a313924d30ad1c8"
integrity sha512-ehH1VSUclIHZKEZVdv+klofsFIh8FFzqA6AAV23RtLepptzA8wqQzUGraEuSN25sYcNmYJ0jti5U0Ys+WZv5Dw==
"@sentry/react@^10.33.0":
version "10.33.0"
resolved "https://registry.yarnpkg.com/@sentry/react/-/react-10.33.0.tgz#89a3be88d43e49de90943ad2ac86ee1664048097"
integrity sha512-iMdC2Iw54ibAccatJ5TjoLlIy3VotFteied7JFvOudgj1/2eBBeWthRobZ5p6/nAOpj4p9vJk0DeLrc012sd2g==
dependencies:
"@sentry/browser" "10.33.0"
"@sentry/core" "10.33.0"
"@sindresorhus/is@^4.0.0": "@sindresorhus/is@^4.0.0":
version "4.6.0" version "4.6.0"
resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-4.6.0.tgz#3c7c9c46e678feefe7a2e5bb609d3dbd665ffb3f" resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-4.6.0.tgz#3c7c9c46e678feefe7a2e5bb609d3dbd665ffb3f"
@@ -5533,6 +5587,11 @@ get-nonce@^1.0.0:
resolved "https://registry.yarnpkg.com/get-nonce/-/get-nonce-1.0.1.tgz#fdf3f0278073820d2ce9426c18f07481b1e0cdf3" resolved "https://registry.yarnpkg.com/get-nonce/-/get-nonce-1.0.1.tgz#fdf3f0278073820d2ce9426c18f07481b1e0cdf3"
integrity sha512-FJhYRoDaiatfEkUK8HKlicmu/3SGFD51q3itKDGoSTysQJBnfOcxU5GxnhE1E6soB76MbT0MBtnKJuXyAx+96Q== integrity sha512-FJhYRoDaiatfEkUK8HKlicmu/3SGFD51q3itKDGoSTysQJBnfOcxU5GxnhE1E6soB76MbT0MBtnKJuXyAx+96Q==
get-port@^7.1.0:
version "7.1.0"
resolved "https://registry.yarnpkg.com/get-port/-/get-port-7.1.0.tgz#d5a500ebfc7aa705294ec2b83cc38c5d0e364fec"
integrity sha512-QB9NKEeDg3xxVwCCwJQ9+xycaz6pBB6iQ76wiWMl1927n0Kir6alPiP+yuiICLLU4jpMe08dXfpebuQppFA2zw==
get-proto@^1.0.0, get-proto@^1.0.1: get-proto@^1.0.0, get-proto@^1.0.1:
version "1.0.1" version "1.0.1"
resolved "https://registry.yarnpkg.com/get-proto/-/get-proto-1.0.1.tgz#150b3f2743869ef3e851ec0c49d15b1d14d00ee1" resolved "https://registry.yarnpkg.com/get-proto/-/get-proto-1.0.1.tgz#150b3f2743869ef3e851ec0c49d15b1d14d00ee1"
@@ -6405,10 +6464,10 @@ keyv@^4.0.0, keyv@^4.5.3:
dependencies: dependencies:
json-buffer "3.0.1" json-buffer "3.0.1"
ky@^1.11.0: ky@^1.14.2:
version "1.14.1" version "1.14.2"
resolved "https://registry.yarnpkg.com/ky/-/ky-1.14.1.tgz#16f20b3bf3939abcc04e2a9613f47360fe5f64c9" resolved "https://registry.yarnpkg.com/ky/-/ky-1.14.2.tgz#385d6d05d2825502e68898ace125124e6fe7357d"
integrity sha512-hYje4L9JCmpEQBtudo+v52X5X8tgWXUYyPcxKSuxQNboqufecl9VMWjGiucAFH060AwPXHZuH+WB2rrqfkmafw== integrity sha512-q3RBbsO5A5zrPhB6CaCS8ZUv+NWCXv6JJT4Em0i264G9W0fdPB8YRfnnEi7Dm7X7omAkBIPojzYJ2D1oHTHqug==
language-subtag-registry@^0.3.20: language-subtag-registry@^0.3.20:
version "0.3.23" version "0.3.23"
@@ -8622,10 +8681,10 @@ tar@^6.0.5, tar@^6.1.11, tar@^6.1.12, tar@^6.2.1:
mkdirp "^1.0.3" mkdirp "^1.0.3"
yallist "^4.0.0" yallist "^4.0.0"
tar@^7.5.2: tar@^7.5.4:
version "7.5.2" version "7.5.4"
resolved "https://registry.yarnpkg.com/tar/-/tar-7.5.2.tgz#115c061495ec51ff3c6745ff8f6d0871c5b1dedc" resolved "https://registry.yarnpkg.com/tar/-/tar-7.5.4.tgz#18b53b44f939a7e03ed874f1fafe17d29e306c81"
integrity sha512-7NyxrTE4Anh8km8iEy7o0QYPs+0JKBTj5ZaqHg6B39erLg0qYXN3BijtShwbsNSvQ+LN75+KV+C4QR/f6Gwnpg== integrity sha512-AN04xbWGrSTDmVwlI4/GTlIIwMFk/XEv7uL8aa57zuvRy6s4hdBed+lVq2fAZ89XDa7Us3ANXcE3Tvqvja1kTA==
dependencies: dependencies:
"@isaacs/fs-minipass" "^4.0.0" "@isaacs/fs-minipass" "^4.0.0"
chownr "^3.0.0" chownr "^3.0.0"
@@ -9179,12 +9238,12 @@ word-wrap@^1.2.5:
resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.5.tgz#d2c45c6dd4fbce621a66f136cbe328afd0410b34" resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.5.tgz#d2c45c6dd4fbce621a66f136cbe328afd0410b34"
integrity sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA== integrity sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==
workwonders-sdk@0.0.10: workwonders-sdk@0.1.1:
version "0.0.10" version "0.1.1"
resolved "https://registry.yarnpkg.com/workwonders-sdk/-/workwonders-sdk-0.0.10.tgz#377167370a39c905c5228f8972c37c19004b7b21" resolved "https://registry.yarnpkg.com/workwonders-sdk/-/workwonders-sdk-0.1.1.tgz#7ac0eb3d9ef0a5a8cc5ca4e6f5e387e29875faa9"
integrity sha512-bnswhlLRz1TCiqGV8l+VEOBej7u1SAkzLMEv6A60Sp0+S4j4pnmSve92KeOts/GYtUeNDuNM7fLPwZwMKY3sAg== integrity sha512-PEsl33QCeiBlYed/MmnX1unnd4Kn7vzVIza00HQ/5Zsan89nqnwWx9vqgJnNipXkkmIWl8oDL9bGRNjtL4XZ4Q==
dependencies: dependencies:
ky "^1.11.0" ky "^1.14.2"
"wrap-ansi-cjs@npm:wrap-ansi@^7.0.0": "wrap-ansi-cjs@npm:wrap-ansi@^7.0.0":
version "7.0.0" version "7.0.0"