mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-10-25 03:43:46 +02:00 
			
		
		
		
	Compare commits
	
		
			170 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 1e03194765 | ||
|  | 10c558faaf | ||
|  | dcf6957842 | ||
|  | 77bebc48bc | ||
|  | cad14dd6a4 | ||
|  | 5d72af5b20 | ||
|  | 1fcb923469 | ||
|  | f89bc41ac1 | ||
|  | 70988a91fd | ||
|  | f3481f94dd | ||
|  | 204eadd809 | ||
|  | 69f2ec1777 | ||
|  | b98c23e369 | ||
|  | 64779fb55a | ||
|  | 845cf0a035 | ||
|  | 9dba240202 | ||
|  | 8aa682028f | ||
|  | cb80b80efb | ||
|  | eb1d997413 | ||
|  | 986badc653 | ||
|  | 2432db0b31 | ||
|  | 5be2c9b51d | ||
|  | d801994fed | ||
|  | 19bb9a0331 | ||
|  | b8dc6f9a53 | ||
|  | 3a7ba00486 | ||
|  | 86416df7dd | ||
|  | 704f209e4d | ||
|  | e1bed6f0cf | ||
|  | 61c88596f0 | ||
|  | a07c51a04a | ||
|  | 0e46a83843 | ||
|  | 1a4b9658c3 | ||
|  | 9b922abe48 | ||
|  | 6f0f883d88 | ||
|  | aafcc4d96e | ||
|  | e019c8027b | ||
|  | b6acb43055 | ||
|  | d9f6e3328e | ||
|  | 0e50132e13 | ||
|  | c74e8456d2 | ||
|  | 33c60c9d4f | ||
|  | 151e085655 | ||
|  | b1d641ae24 | ||
|  | 7ddb35f933 | ||
|  | b05a2ed14f | ||
|  | b22a82d11b | ||
|  | ee051153ca | ||
|  | 46871191c4 | ||
|  | 0cb518d434 | ||
|  | 03778fd743 | ||
|  | 514ca3e28b | ||
|  | 37e90dabe7 | ||
|  | ee99f2b71f | ||
|  | ca9289f104 | ||
|  | aefe7eaa5f | ||
|  | e979f7c6f3 | ||
|  | e2118a3b53 | ||
|  | d3a5d3b0fa | ||
|  | bc7c70aca6 | ||
|  | 6cde454bd9 | ||
|  | 78e20984ed | ||
|  | f963996a19 | ||
|  | 1ddf2869a6 | ||
|  | f9b97b4469 | ||
|  | 6f706ab493 | ||
|  | ed2e6e2e9e | ||
|  | f00758cd83 | ||
|  | 25bd61f289 | ||
|  | 25d19d22cf | ||
|  | 9ffb91266f | ||
|  | 8d921cfbf1 | ||
|  | 3ed6b168e1 | ||
|  | 4a45c900c3 | ||
|  | 7e9e6958a3 | ||
|  | 9de082f684 | ||
|  | b40afc7329 | ||
|  | 4fb3da45e9 | ||
|  | 1f8e051ed6 | ||
|  | adc4e7c2b4 | ||
|  | 325d2f48dc | ||
|  | aa00f42a23 | ||
|  | 62dbffab52 | ||
|  | fd38444836 | ||
|  | 71cc49a30f | ||
|  | e4187c8e17 | ||
|  | 15ae98d426 | ||
|  | 7b4fa2269e | ||
|  | 656e14793d | ||
|  | d23381147b | ||
|  | cfe184c6d5 | ||
|  | 9c16f24a08 | ||
|  | ce200403e0 | ||
|  | caca84f84d | ||
|  | ac2d7b9639 | ||
|  | 78069a9b26 | ||
|  | 3e23fffaea | ||
|  | 624b9fb6dc | ||
|  | fbd6189721 | ||
|  | 6a44fe227c | ||
|  | d1cb2688c6 | ||
|  | b43b2f9740 | ||
|  | 6288a9cb8d | ||
|  | 18c6f75054 | ||
|  | 1f8d187b84 | ||
|  | 77618ad1ff | ||
|  | 75efd335e9 | ||
|  | c4438de200 | ||
|  | e3c9a227d3 | ||
|  | 548f859349 | ||
|  | db413badec | ||
|  | dfac3af3f5 | ||
|  | 5eceaeecde | ||
|  | 62c23004f7 | ||
|  | cd153e7d11 | ||
|  | 3b494511d7 | ||
|  | 6c806a44d4 | ||
|  | c5bac43bfd | ||
|  | 1636281d6d | ||
|  | 0f4feb2da6 | ||
|  | c32f34c116 | ||
|  | d391c5cfc2 | ||
|  | bb45a157b3 | ||
|  | c90a571f89 | ||
|  | 1278407c90 | ||
|  | dff1b4cf39 | ||
|  | 105e70a814 | ||
|  | 2938cea419 | ||
|  | 9b6b7780d8 | ||
|  | f53a4f3827 | ||
|  | 9b0d672553 | ||
|  | 9d14145ac2 | ||
|  | 9948797b25 | ||
|  | a80d9e3623 | ||
|  | 37ecb057f9 | ||
|  | 06cbe74c6c | ||
|  | f5769b85fe | ||
|  | 875868af47 | ||
|  | 24ea15d600 | ||
|  | 87cdc3dcef | ||
|  | c2f976d7f4 | ||
|  | ebc46bfd3f | ||
|  | e674f7e89d | ||
|  | 86d29b163c | ||
|  | a849919799 | ||
|  | d8c805a6b3 | ||
|  | 8acbd3ef78 | ||
|  | b81088d767 | ||
|  | 7071e34a74 | ||
|  | 6a73ade359 | ||
|  | 6c5e5043a4 | ||
|  | 5edaf1306d | ||
|  | 5bf7fdb94e | ||
|  | 7808bc5ccb | ||
|  | 06a079e452 | ||
|  | de509737e6 | ||
|  | 08a3ba8d13 | ||
|  | 2b27902d5f | ||
|  | 62185127c2 | ||
|  | 9c46e3f88c | ||
|  | daa4de921d | ||
|  | bc977e2a5a | ||
|  | 1e3815ca20 | ||
|  | 0e2fa96b9f | ||
|  | 389aebe54e | ||
|  | fbfeaf2a91 | ||
|  | 485812d076 | ||
|  | 68e98ec719 | ||
|  | cbe41de734 | ||
|  | abb8641105 | 
							
								
								
									
										2
									
								
								.flutter
									
									
									
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										2
									
								
								.flutter
									
									
									
									
									
								
							 Submodule .flutter updated: 54e66469a9...2663184aa7
									
								
							
							
								
								
									
										14
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,10 @@ on: | ||||
|       beta: | ||||
|         type: boolean | ||||
|         description: Is beta? | ||||
|      | ||||
|       draft: | ||||
|         type: boolean | ||||
|         description: Is draft? | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     runs-on: ubuntu-latest | ||||
| @@ -35,13 +38,7 @@ jobs: | ||||
|       - name: Check submodule | ||||
|         id: check_submodule | ||||
|         run: | | ||||
|           SUBMODULE_COMMIT_LONG="$(git submodule status | head -1 | tail -c +2 | awk '{print $1}')" | ||||
|           FLUTTER_COMMIT_SHORT="$(flutter --version | head -2 | tail -1 | awk '{print $4}')" | ||||
|           echo "SUBMODULE_COMMIT_LONG=$SUBMODULE_COMMIT_LONG, FLUTTER_COMMIT_SHORT=$FLUTTER_COMMIT_SHORT" | ||||
|           if ! [[ "$SUBMODULE_COMMIT_LONG" =~ ^$FLUTTER_COMMIT_SHORT ]]; then | ||||
|             echo "Your submodule has not been updated!" | ||||
|             exit 1 | ||||
|           fi | ||||
|           git checkout ${{ inputs.checkout }} | ||||
|  | ||||
|       - name: Extract Version | ||||
|         id: extract_version       | ||||
| @@ -92,5 +89,6 @@ jobs: | ||||
|           token: ${{ secrets.GH_ACCESS_TOKEN }} | ||||
|           tag: "${{ steps.extract_version.outputs.tag }}" | ||||
|           prerelease: "${{ steps.extract_version.outputs.beta }}" | ||||
|           draft: "${{ inputs.draft }}" | ||||
|           artifacts: ./build/app/outputs/flutter-apk/*-release*.apk* | ||||
|           generateReleaseNotes: true | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| [](https://techforpalestine.org/learn-more) | ||||
|  | ||||
| Get Android App Updates Directly From the Source. | ||||
| Get Android app updates straight from the source. | ||||
|  | ||||
| Obtainium allows you to install and update apps directly from their releases pages, and receive notifications when new releases are made available. | ||||
|  | ||||
| @@ -16,7 +16,7 @@ Currently supported App sources: | ||||
| - Open Source - General: | ||||
|   - [GitHub](https://github.com/) | ||||
|   - [GitLab](https://gitlab.com/) | ||||
|   - [Codeberg](https://codeberg.org/) | ||||
|   - [Forgejo](https://forgejo.org/) ([Codeberg](https://codeberg.org/)) | ||||
|   - [F-Droid](https://f-droid.org/) | ||||
|   - Third Party F-Droid Repos | ||||
|   - [IzzyOnDroid](https://android.izzysoft.de/) | ||||
| @@ -25,9 +25,10 @@ Currently supported App sources: | ||||
|   - [APKPure](https://apkpure.net/) | ||||
|   - [Aptoide](https://aptoide.com/) | ||||
|   - [Uptodown](https://uptodown.com/) | ||||
|   - [APKMirror](https://apkmirror.com/) (Track-Only) | ||||
|   - [Huawei AppGallery](https://appgallery.huawei.com/) | ||||
|   - [Tencent App Store](https://sj.qq.com/) | ||||
|   - Jenkins Jobs | ||||
|   - [APKMirror](https://apkmirror.com/) (Track-Only) | ||||
| - Open Source - App-Specific: | ||||
|   - [Signal](https://signal.org/) | ||||
|   - [VLC](https://videolan.org/) | ||||
|   | ||||
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								assets/fonts/WixMadeforDisplay-Regular.otf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								assets/fonts/WixMadeforDisplay-Regular.otf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -24,7 +24,7 @@ | ||||
|     "colour": "Boja", | ||||
|     "standard": "Standard", | ||||
|     "custom": "Custom", | ||||
|     "useMaterialYou": "Use Material You", | ||||
|     "useMaterialYou": "Koristi Material You temu", | ||||
|     "githubStarredRepos": "GitHub repo-i sa zvjezdicom", | ||||
|     "uname": "Korisničko ime", | ||||
|     "wrongArgNum": "Naveden je pogrešan broj argumenata", | ||||
| @@ -113,7 +113,7 @@ | ||||
|     "dark": "Tamna", | ||||
|     "light": "Svijetla", | ||||
|     "followSystem": "Pratite sistem", | ||||
|     "followSystemThemeExplanation": "Following system theme is possible only by using third-party applications", | ||||
|     "followSystemThemeExplanation": "Praćenje sistemske teme je moguće jedino koristeći aplikacije treće strane", | ||||
|     "useBlackTheme": "Koristite čisto crnu tamnu temu", | ||||
|     "appSortBy": "Aplikacije sortirane po", | ||||
|     "authorName": "Autor/Ime", | ||||
| @@ -147,10 +147,10 @@ | ||||
|     "noNewUpdates": "Nema novih ažuriranja.", | ||||
|     "xHasAnUpdate": "{} ima ažuriranje.", | ||||
|     "appsUpdated": "Aplikacije su ažurirane", | ||||
|     "appsNotUpdated": "Failed to update applications", | ||||
|     "appsNotUpdated": "Neuspješno ažuriranje aplikacija", | ||||
|     "appsUpdatedNotifDescription": "Obavještava korisnika da su u pozadini primijenjena ažuriranja na jednu ili više aplikacija", | ||||
|     "xWasUpdatedToY": "{} je ažuriran na {}.", | ||||
|     "xWasNotUpdatedToY": "Failed to update {} to {}.", | ||||
|     "xWasNotUpdatedToY": "Neuspješno ažuriranje {} na {}.", | ||||
|     "errorCheckingUpdates": "Greška pri provjeri ažuriranja", | ||||
|     "errorCheckingUpdatesNotifDescription": "Obavijest koja se prikazuje kada provjera sigurnosnog ažuriranja ne uspije", | ||||
|     "appsRemoved": "Aplikacije su uklonjene", | ||||
| @@ -191,7 +191,7 @@ | ||||
|     "downloadingX": "Preuzimanje {}", | ||||
|     "downloadX": "Download {}", | ||||
|     "downloadedX": "Downloaded {}", | ||||
|     "releaseAsset": "Release Asset", | ||||
|     "releaseAsset": "Fajlovi verzije", | ||||
|     "downloadNotifDescription": "Obavještava korisnika o napretku u preuzimanju aplikacije", | ||||
|     "noAPKFound": "APK nije pronađen", | ||||
|     "noVersionDetection": "Nema detekcije verzije", | ||||
| @@ -253,12 +253,14 @@ | ||||
|     "verifyLatestTag": "Provjerite 'posljednu' ('latest') oznaku", | ||||
|     "intermediateLinkRegex": "Filter za 'srednju' vezu za posjetu", | ||||
|     "filterByLinkText": "Filtriraj linkove prema tekstu linka", | ||||
|     "intermediateLinkNotFound": "Intermediate link nije nađen", | ||||
|     "intermediateLink": "srednja karika", | ||||
|     "intermediateLinkNotFound": "Intermediate veza nije nađena", | ||||
|     "intermediateLink": "Intermediate veza", | ||||
|     "exemptFromBackgroundUpdates": "Izuzmi iz ažuriranja u pozadini (ako su uključeni)", | ||||
|     "bgUpdatesOnWiFiOnly": "Isključite ažuriranje u pozadini kada niste na WiFi-ju", | ||||
|     "autoSelectHighestVersionCode": "Automatski izaberite najveću (verziju) versionCode APK-a", | ||||
|     "versionExtractionRegEx": "RegEx ekstrakcija verzije", | ||||
|     "trimVersionString": "Trim Version String With RegEx", | ||||
|     "matchGroupToUseForX": "Match Group to Use for \"{}\"", | ||||
|     "matchGroupToUse": "Podjesite grupu za upotebu", | ||||
|     "highlightTouchTargets": "Istaknite manje vidljive touch mete", | ||||
|     "pickExportDir": "Izaberite datoteku za izvoz", | ||||
| @@ -288,11 +290,11 @@ | ||||
|     "supportFixedAPKURL": "Podržite fiksne APK URL-ove", | ||||
|     "selectX": "Izaberite {}", | ||||
|     "parallelDownloads": "Dozvoli paralelna preuzimanja", | ||||
|     "useShizuku": "Use Shizuku or Sui to install", | ||||
|     "shizukuBinderNotFound": "Shizuku is not running", | ||||
|     "shizukuOld": "Old Shizuku version (<11) - update it", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku running on Android < 8.1 with ADB - update Android or use Sui instead", | ||||
|     "shizukuPretendToBeGooglePlay": "Set Google Play as the installation source (if Shizuku is used)", | ||||
|     "useShizuku": "Koristi Shizuku ili Sui za instaliranje", | ||||
|     "shizukuBinderNotFound": "Shizuku nije pokrenut", | ||||
|     "shizukuOld": "Stara Shizuku verzija (<11) - ažurirajte je", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku pokrenut na Android-u < 8.1 pomoću ADB-a - ažurirajte Android ili koristite Sui", | ||||
|     "shizukuPretendToBeGooglePlay": "Postavi Google Play kao izvor instalacije (samo ako je Shizuku u upotrebi)", | ||||
|     "useSystemFont": "Koristite sistemski font", | ||||
|     "useVersionCodeAsOSVersion": "Koristite kod verzije aplikacije kao verziju koju je otkrio OS", | ||||
|     "requestHeader": "Zaglavlje zahtjeva", | ||||
| @@ -305,13 +307,14 @@ | ||||
|     "installed": "Instalirano", | ||||
|     "latest": "Najnoviji", | ||||
|     "invertRegEx": "Obrni regularni izraz", | ||||
|     "note": "Note", | ||||
|     "selfHostedNote": "The \"{}\" dropdown can be used to reach self-hosted/custom instances of any source.", | ||||
|     "badDownload": "The APK could not be parsed (incompatible or partial download)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Share new Apps with AppVerifier (if available)", | ||||
|     "appVerifierInstructionToast": "Share to AppVerifier, then return here when ready.", | ||||
|     "wiki": "Help/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", | ||||
|     "note": "Zabelješke", | ||||
|     "selfHostedNote": "\"{}\" padajući meni se može koristiti da dosegnete vlastite/prilagođene instance bilo kojeg izvora.", | ||||
|     "badDownload": "APK ne može biti raščlanjen (nekomaptibilno ili delimično preuzimanje)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Dijeli nove aplikacije sa AppVerifier-om (ako je dostupno)", | ||||
|     "appVerifierInstructionToast": "Dijeli sa AppVerifier-om, zatim se vratite kada ste spremni.", | ||||
|     "wiki": "Pomoć/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Konfiguracije aplikacije obezbeđene pomoću velikog broja ljudi (crowdsourcing) (koristite na svoju odgovornost)", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Želite li ukloniti aplikaciju?", | ||||
|         "other": "Želite li ukloniti aplikacije?" | ||||
| @@ -370,6 +373,6 @@ | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APKs" | ||||
|         "other": "{} APK-a" | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Deaktivovat aktualizace na pozadí, pokud není k dispozici Wi-Fi", | ||||
|     "autoSelectHighestVersionCode": "Automaticky vybrat nejvyšší verzi APK", | ||||
|     "versionExtractionRegEx": "Extrakce verze pomocí RegEx", | ||||
|     "trimVersionString": "Oříznutí řetězce verze pomocí příkazu RegEx", | ||||
|     "matchGroupToUseForX": "Skupina shody, která se použije pro \"{}\"", | ||||
|     "matchGroupToUse": "Odpovídá použité skupině", | ||||
|     "highlightTouchTargets": "Zvýraznit méně zjevné cíle dotyku", | ||||
|     "pickExportDir": "Vybrat adresář pro export", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "Sdílejte do aplikace AppVerifier a po dokončení se sem vraťte.", | ||||
|     "wiki": "Nápověda/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Konfigurace aplikací s využitím crowdsourcingu (použití na vlastní nebezpečí)", | ||||
|     "allowInsecure": "Povolení nezabezpečených požadavků HTTP", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Odstranit Apku?", | ||||
|         "other": "Odstranit Apky?" | ||||
|   | ||||
							
								
								
									
										378
									
								
								assets/translations/da.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										378
									
								
								assets/translations/da.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,378 @@ | ||||
| { | ||||
|     "invalidURLForSource": "Ikke et gyldigt {} App-URL", | ||||
|     "noReleaseFound": "Kunne ikke finde en passende udgivelse", | ||||
|     "noVersionFound": "Kunne ikke afgøre udgivelsesversion", | ||||
|     "urlMatchesNoSource": "URL'en matcher ikke en kendt kilde", | ||||
|     "cantInstallOlderVersion": "Kan ikke installere en ældre version af en app", | ||||
|     "appIdMismatch": "Hentet pakke-ID matcher ikke eksisterende app-ID", | ||||
|     "functionNotImplemented": "Denne klasse har ikke implementeret denne funktion", | ||||
|     "placeholder": "Pladsholder", | ||||
|     "someErrors": "Nogle fejl opstod", | ||||
|     "unexpectedError": "Uventet fejl", | ||||
|     "ok": "Okay", | ||||
|     "and": "og", | ||||
|     "githubPATLabel": "GitHub Personlig Adgangstoken (øger hastighedsgrænse)", | ||||
|     "includePrereleases": "Inkluder forudgivelser", | ||||
|     "fallbackToOlderReleases": "Fallback til ældre udgivelser", | ||||
|     "filterReleaseTitlesByRegEx": "Filtrer udgivelsestitler efter regulært udtryk", | ||||
|     "invalidRegEx": "Ugyldigt regulært udtryk", | ||||
|     "noDescription": "Ingen beskrivelse", | ||||
|     "cancel": "Annuller", | ||||
|     "continue": "Fortsæt", | ||||
|     "requiredInBrackets": "(Påkrævet)", | ||||
|     "dropdownNoOptsError": "FEJL: RULLEMENU SKAL HAVE MINDST ÉT TILVALG", | ||||
|     "colour": "Farve", | ||||
|     "standard": "Standard", | ||||
|     "custom": "Brugerdefineret", | ||||
|     "useMaterialYou": "Brug Material You", | ||||
|     "githubStarredRepos": "Stjernemarkeret GitHub-repos", | ||||
|     "uname": "Brugernavn", | ||||
|     "wrongArgNum": "Forkert antal argumenter angivet", | ||||
|     "xIsTrackOnly": "{} kan kun følges", | ||||
|     "source": "Kilde", | ||||
|     "app": "App", | ||||
|     "appsFromSourceAreTrackOnly": "Apps fra denne kilde er 'Følg Kun'.", | ||||
|     "youPickedTrackOnly": "Du har valgt 'Følg Kun'-indstillingen.", | ||||
|     "trackOnlyAppDescription": "Appen tjekkes for opdateringer, men Obtainium kan ikke hente eller installere den.", | ||||
|     "cancelled": "Annulleret", | ||||
|     "appAlreadyAdded": "Appen er allerede tilføjet", | ||||
|     "alreadyUpToDateQuestion": "Appen er allerede opdateret?", | ||||
|     "addApp": "Tilføj app", | ||||
|     "appSourceURL": "URL til app-kilde", | ||||
|     "error": "Fejl", | ||||
|     "add": "Tilføj", | ||||
|     "searchSomeSourcesLabel": "Søg (kun visse kilder)", | ||||
|     "search": "Søg", | ||||
|     "additionalOptsFor": "Yderligere indstillinger for {}", | ||||
|     "supportedSources": "Understøttede kilder", | ||||
|     "trackOnlyInBrackets": "(Følg Kun)", | ||||
|     "searchableInBrackets": "(Kan Søges)", | ||||
|     "appsString": "Apps", | ||||
|     "noApps": "Ingen apps", | ||||
|     "noAppsForFilter": "Ingen apps til filter", | ||||
|     "byX": "Af {}", | ||||
|     "percentProgress": "Fremskridt: {}%", | ||||
|     "pleaseWait": "Vent venligst", | ||||
|     "updateAvailable": "Opdatering tilgængelig", | ||||
|     "notInstalled": "Ikke installeret", | ||||
|     "pseudoVersion": "pseudo-version", | ||||
|     "selectAll": "Vælg alle", | ||||
|     "deselectX": "Fravælg {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} fjernes fra Obtainium, men forbliver installeret på enheden.", | ||||
|     "removeSelectedAppsQuestion": "Fjern valgte apps?", | ||||
|     "removeSelectedApps": "Fjern valgte apps", | ||||
|     "updateX": "Opdater {}", | ||||
|     "installX": "Installer {}", | ||||
|     "markXTrackOnlyAsUpdated": "Markér {}\n(Følg Kun)\nsom opdateret", | ||||
|     "changeX": "Skift {}", | ||||
|     "installUpdateApps": "Installer/Opdater apps", | ||||
|     "installUpdateSelectedApps": "Installer/Opdater valgte apps", | ||||
|     "markXSelectedAppsAsUpdated": "Markér {} valgte apps som opdateret?", | ||||
|     "no": "Nej", | ||||
|     "yes": "Ja", | ||||
|     "markSelectedAppsUpdated": "Markér valgte apps som opdateret", | ||||
|     "pinToTop": "Fastgør til toppen", | ||||
|     "unpinFromTop": "Frigør fra toppen", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Nulstil installationsstatus for valgte apps?", | ||||
|     "installStatusOfXWillBeResetExplanation": "Installationsstatus for alle valgte apps nulstilles.\n\nDette kan hjælpe, når den app-version, der vises i Obtainium, er forkert grundet mislykkede opdateringer eller andre problemer.", | ||||
|     "customLinkMessage": "Disse links virker på enheder med Obtainium installeret", | ||||
|     "shareAppConfigLinks": "Del app-konfiguration som HTML-link", | ||||
|     "shareSelectedAppURLs": "Del valgte app-URL'er", | ||||
|     "resetInstallStatus": "Nulstil installationsstatus", | ||||
|     "more": "Mere", | ||||
|     "removeOutdatedFilter": "Fjern forældet app-filter", | ||||
|     "showOutdatedOnly": "Vis kun forældet apps", | ||||
|     "filter": "Filtrer", | ||||
|     "filterApps": "Filtrer Apps", | ||||
|     "appName": "Appnavn", | ||||
|     "author": "Udvikler", | ||||
|     "upToDateApps": "Opdaterede apps", | ||||
|     "nonInstalledApps": "Ikke-installerede apps", | ||||
|     "importExport": "Import/Eksport", | ||||
|     "settings": "Indstillinger", | ||||
|     "exportedTo": "Eksportér til {}", | ||||
|     "obtainiumExport": "Obtainium-eksport", | ||||
|     "invalidInput": "Ugyldigt input", | ||||
|     "importedX": "Importerede {}", | ||||
|     "obtainiumImport": "Obtainium-import", | ||||
|     "importFromURLList": "Importér fra URL-liste", | ||||
|     "searchQuery": "Søgning", | ||||
|     "appURLList": "Liste over app-URL'er", | ||||
|     "line": "Linje", | ||||
|     "searchX": "Søg {}", | ||||
|     "noResults": "Ingen resultater fundet", | ||||
|     "importX": "Importér {}", | ||||
|     "importedAppsIdDisclaimer": "Importerede apps vises muligvis forkert som \"Ikke installeret\".\nFor at løse dette, geninstaller dem via Obtainium.\nDette bør ikke påvirke app-data.\n\nPåvirker kun URL- og tredjepartsimportmetoder.", | ||||
|     "importErrors": "Importfejl", | ||||
|     "importedXOfYApps": "{} af {} app importeret.", | ||||
|     "followingURLsHadErrors": "Følgende URL'er havde fejl:", | ||||
|     "selectURL": "Vælg URL", | ||||
|     "selectURLs": "Vælg URL'er", | ||||
|     "pick": "Vælg", | ||||
|     "theme": "Tema", | ||||
|     "dark": "Mørk", | ||||
|     "light": "Lys", | ||||
|     "followSystem": "Følg system", | ||||
|     "followSystemThemeExplanation": "Det er kun muligt at følge systemtemaet ved brug af tredjepartsapplikationer", | ||||
|     "useBlackTheme": "Brug rent sort, mørkt tema", | ||||
|     "appSortBy": "Sortér apps efter:", | ||||
|     "authorName": "Udvikler/Navn", | ||||
|     "nameAuthor": "Navn/Udvikler", | ||||
|     "asAdded": "Som tilføjet", | ||||
|     "appSortOrder": "Sorteringsrækkefølge for apps", | ||||
|     "ascending": "Stigende", | ||||
|     "descending": "Faldende", | ||||
|     "bgUpdateCheckInterval": "Kontrolinterval for baggrundsopdatering", | ||||
|     "neverManualOnly": "Aldrig - Kun manuelt", | ||||
|     "appearance": "Udseende", | ||||
|     "showWebInAppView": "Vis kildewebsiden i appvisning", | ||||
|     "pinUpdates": "Fastgør opdateringer til toppen af appvisning", | ||||
|     "updates": "Opdateringer", | ||||
|     "sourceSpecific": "Kildespecifik", | ||||
|     "appSource": "App-kilde", | ||||
|     "noLogs": "Ingen logs", | ||||
|     "appLogs": "App-logs", | ||||
|     "close": "Luk", | ||||
|     "share": "Del", | ||||
|     "appNotFound": "App ikke fundet", | ||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-eksport", | ||||
|     "pickAnAPK": "Vælg en APK", | ||||
|     "appHasMoreThanOnePackage": "{} har mere end én pakke:", | ||||
|     "deviceSupportsXArch": "Din enhed understøtter {} CPU-arkitekturen.", | ||||
|     "deviceSupportsFollowingArchs": "Din enhed understøtter følgende CPU-arkitekturer:", | ||||
|     "warning": "Advarsel", | ||||
|     "sourceIsXButPackageFromYPrompt": "App-kilden er '{}', men udgivelsespakken kommer fra '{}'. Fortsæt?", | ||||
|     "updatesAvailable": "Opdateringer tilgængelige", | ||||
|     "updatesAvailableNotifDescription": "Underretter brugeren om, at opdateringer er tilgængelige for en eller flere apps, der spores af Obtainium", | ||||
|     "noNewUpdates": "Ingen nye opdateringer.", | ||||
|     "xHasAnUpdate": "{} har en opdatering.", | ||||
|     "appsUpdated": "Apps opdateret", | ||||
|     "appsNotUpdated": "Kunne ikke opdatere applikationerne", | ||||
|     "appsUpdatedNotifDescription": "Underretter brugeren om, at opdateringer til en eller flere apps blev udført i baggrunden", | ||||
|     "xWasUpdatedToY": "{} blev opdateret til {}.", | ||||
|     "xWasNotUpdatedToY": "Kunne ikke opdatere {} til {}.", | ||||
|     "errorCheckingUpdates": "Fejl ved tjek for opdateringer", | ||||
|     "errorCheckingUpdatesNotifDescription": "En meddelelse, der vises, når opdateringstjek i baggrunden mislykkes", | ||||
|     "appsRemoved": "Apps fjernet", | ||||
|     "appsRemovedNotifDescription": "Underretter brugeren om, at en eller flere apps blev fjernet grundet fejl under indlæsning af dem", | ||||
|     "xWasRemovedDueToErrorY": "{} blev fjernet grundet denne fejl: {}", | ||||
|     "completeAppInstallation": "Færdiggør app-installation", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium skal være åben for at installere apps", | ||||
|     "completeAppInstallationNotifDescription": "Beder brugeren om at vende tilbage til Obtainium for at afslutte installationen af en app", | ||||
|     "checkingForUpdates": "Tjekker for opdateringer", | ||||
|     "checkingForUpdatesNotifDescription": "Kortvarig meddelelse, der vises ved tjek for opdateringer", | ||||
|     "pleaseAllowInstallPerm": "Tillad venligst Obtainium at installere apps", | ||||
|     "trackOnly": "Følg Kun", | ||||
|     "errorWithHttpStatusCode": "Fejl {}", | ||||
|     "versionCorrectionDisabled": "Versionskorrigering deaktiveret (plugin ser ikke ud til at virke)", | ||||
|     "unknown": "Ukendt", | ||||
|     "none": "Ingen", | ||||
|     "never": "Aldrig", | ||||
|     "latestVersionX": "Seneste: {}", | ||||
|     "installedVersionX": "Installeret: {}", | ||||
|     "lastUpdateCheckX": "Sidste opdateringstjek: {}", | ||||
|     "remove": "Fjern", | ||||
|     "yesMarkUpdated": "Ja, markér som opdateret", | ||||
|     "fdroid": "F-Droid Officiel", | ||||
|     "appIdOrName": "App-ID eller navn", | ||||
|     "appId": "App-ID", | ||||
|     "appWithIdOrNameNotFound": "Ingen app med det ID eller navn blev fundet", | ||||
|     "reposHaveMultipleApps": "Repos kan indeholde flere apps", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Tredjeparts-repo", | ||||
|     "steamMobile": "Steam Mobil", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Installer", | ||||
|     "markInstalled": "Markér som installeret", | ||||
|     "update": "Opdater", | ||||
|     "markUpdated": "Markér som opdateret", | ||||
|     "additionalOptions": "Yderligere indstillinger", | ||||
|     "disableVersionDetection": "Deaktivér versionsregistrering", | ||||
|     "noVersionDetectionExplanation": "Denne indstilling bør kun bruges til apps, hvor versionsregistrering ikke virker korrekt.", | ||||
|     "downloadingX": "Henter {}", | ||||
|     "downloadX": "Hent {}", | ||||
|     "downloadedX": "Hentede {}", | ||||
|     "releaseAsset": "Udgivelsesressource", | ||||
|     "downloadNotifDescription": "Underretter brugeren om fremskridt i hentning af en app", | ||||
|     "noAPKFound": "Ingen APK fundet", | ||||
|     "noVersionDetection": "Ingen versionsregistrering", | ||||
|     "categorize": "Kategoriser", | ||||
|     "categories": "Kategorier", | ||||
|     "category": "Kategori", | ||||
|     "noCategory": "Ingen kategori", | ||||
|     "noCategories": "Ingen kategorier", | ||||
|     "deleteCategoriesQuestion": "Slet kategorier?", | ||||
|     "categoryDeleteWarning": "Alle apps i slettede kategorier indstilles til ukategoriseret.", | ||||
|     "addCategory": "Tilføj kategori", | ||||
|     "label": "Etiket", | ||||
|     "language": "Sprog", | ||||
|     "copiedToClipboard": "Kopieret til udklipsholder", | ||||
|     "storagePermissionDenied": "Lagringstilladelse nægtet", | ||||
|     "selectedCategorizeWarning": "Dette erstatter alle eksisterende kategoriindstillinger for de valgte apps.", | ||||
|     "filterAPKsByRegEx": "Filtrer APK'er efter regulært udtryk", | ||||
|     "removeFromObtainium": "Fjern fra Obtainium", | ||||
|     "uninstallFromDevice": "Afinstaller fra enhed", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Virker kun for apps med versionsregistrering deaktiveret.", | ||||
|     "releaseDateAsVersion": "Brug udgivelsesdato som versionsstreng", | ||||
|     "releaseDateAsVersionExplanation": "Denne indstilling bør kun bruges til apps, hvor versionsregistrering ikke virker korrekt, men hvor en udgivelsesdato er tilgængelig.", | ||||
|     "changes": "Ændringer", | ||||
|     "releaseDate": "Udgivelsesdato", | ||||
|     "importFromURLsInFile": "Importér fra URL'er i fil (som OPML)", | ||||
|     "versionDetectionExplanation": "Afstem versionsstreng med versionen registreret fra OS", | ||||
|     "versionDetection": "Versionsregistrering", | ||||
|     "standardVersionDetection": "Standard versionsregistrering", | ||||
|     "groupByCategory": "Gruppér efter kategori", | ||||
|     "autoApkFilterByArch": "Forsøg at filtrere APK'er efter CPU-arkitektur, hvis muligt", | ||||
|     "overrideSource": "Tilsidesæt kilde", | ||||
|     "dontShowAgain": "Vis ikke denne igen", | ||||
|     "dontShowTrackOnlyWarnings": "Vis ikke 'Følg Kun'-advarsler", | ||||
|     "dontShowAPKOriginWarnings": "Vis ikke advarsler om APK-oprindelse", | ||||
|     "moveNonInstalledAppsToBottom": "Flyt ikke-installerede apps til bunden af appvisning", | ||||
|     "gitlabPATLabel": "GitLab Personlig Adgangstoken", | ||||
|     "about": "Om", | ||||
|     "requiresCredentialsInSettings": "{} kræver yderligere legitimation (i Indstillinger)", | ||||
|     "checkOnStart": "Tjek for opdateringer ved opstart", | ||||
|     "tryInferAppIdFromCode": "Forsøg at udlede app-ID fra kildekode", | ||||
|     "removeOnExternalUninstall": "Fjern automatisk eksternt afinstallerede apps", | ||||
|     "pickHighestVersionCode": "Auto-vælg højeste versionKode af APK", | ||||
|     "checkUpdateOnDetailPage": "Tjek for opdateringer ved åbning af appens detaljeside", | ||||
|     "disablePageTransitions": "Deaktivér sideovergangsanimationer", | ||||
|     "reversePageTransitions": "Omvendte sideovergangsanimationer", | ||||
|     "minStarCount": "Minimum antal stjerner", | ||||
|     "addInfoBelow": "Tilføj denne info nedenfor.", | ||||
|     "addInfoInSettings": "Tilføj denne info i indstillingerne.", | ||||
|     "githubSourceNote": "GitHubs hastighedsbegrænsning kan undgås med en API-nøgle.", | ||||
|     "sortByLastLinkSegment": "Sortér kun efter det sidste segment af linket", | ||||
|     "filterReleaseNotesByRegEx": "Filtrer udgivelsesnoter efter regulært udtryk", | ||||
|     "customLinkFilterRegex": "Brugerdefineret APK-linkfilter efter regulært udtryk (standard '.apk$')", | ||||
|     "appsPossiblyUpdated": "App-opdateringer forsøgt", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Underretter brugeren om, at opdateringer til en eller flere apps potentielt blev udført i baggrunden", | ||||
|     "xWasPossiblyUpdatedToY": "{} er muligvis blevet opdateret til {}.", | ||||
|     "enableBackgroundUpdates": "Aktivér baggrundsopdateringer", | ||||
|     "backgroundUpdateReqsExplanation": "Baggrundsopdateringer er muligvis ikke mulige for alle apps.", | ||||
|     "backgroundUpdateLimitsExplanation": "Om en baggrundsinstallation er vellykket, kan kun afgøres, når Obtainium åbnes.", | ||||
|     "verifyLatestTag": "Verificer 'seneste'-tagget", | ||||
|     "intermediateLinkRegex": "Filtrer efter et 'mellemliggende' link at besøge", | ||||
|     "filterByLinkText": "Filtrer links efter linktekst", | ||||
|     "intermediateLinkNotFound": "Mellemliggende link ikke fundet", | ||||
|     "intermediateLink": "Mellemliggende link", | ||||
|     "exemptFromBackgroundUpdates": "Undtag fra baggrundsopdateringer (hvis aktiveret)", | ||||
|     "bgUpdatesOnWiFiOnly": "Deaktivér baggrundsopdateringer, når du ikke er på WiFi", | ||||
|     "autoSelectHighestVersionCode": "Auto-vælg højeste versionKode af APK", | ||||
|     "versionExtractionRegEx": "RegEx for versionsstrengsudtrækning", | ||||
|     "trimVersionString": "Trim versionsstrengen med RegEx", | ||||
|     "matchGroupToUseForX": "Matchgruppe til brug for \"{}\"", | ||||
|     "matchGroupToUse": "Match gruppe til brug til RegEx for versionsstrengsudtrækning", | ||||
|     "highlightTouchTargets": "Fremhæv mindre åbenlyse berøringsmål", | ||||
|     "pickExportDir": "Vælg eksportmappe", | ||||
|     "autoExportOnChanges": "Auto-eksport ved ændringer", | ||||
|     "includeSettings": "Inkluder indstillinger", | ||||
|     "filterVersionsByRegEx": "Filtrer versioner efter regulært udtryk", | ||||
|     "trySelectingSuggestedVersionCode": "Forsøg at vælge den foreslåede versionKode af APK", | ||||
|     "dontSortReleasesList": "Behold udgivelsesordre fra API", | ||||
|     "reverseSort": "Omvendt sortering", | ||||
|     "takeFirstLink": "Tag første link", | ||||
|     "skipSort": "Spring sortering over", | ||||
|     "debugMenu": "Fejlfindingsmenu", | ||||
|     "bgTaskStarted": "Baggrundsopgave startet - tjek logfiler.", | ||||
|     "runBgCheckNow": "Kør baggrundsopdateringstjek nu", | ||||
|     "versionExtractWholePage": "Anvend RegEx for versionsstrengsudtrækning for hele siden", | ||||
|     "installing": "Installerer", | ||||
|     "skipUpdateNotifications": "Spring opdateringsmeddelelser over", | ||||
|     "updatesAvailableNotifChannel": "Opdateringer tilgængelige", | ||||
|     "appsUpdatedNotifChannel": "Apps opdateret", | ||||
|     "appsPossiblyUpdatedNotifChannel": "App-opdateringer forsøgt", | ||||
|     "errorCheckingUpdatesNotifChannel": "Fejl ved opdateringstjek", | ||||
|     "appsRemovedNotifChannel": "Apps fjernet", | ||||
|     "downloadingXNotifChannel": "Henter {}", | ||||
|     "completeAppInstallationNotifChannel": "Færdiggør app-installation", | ||||
|     "checkingForUpdatesNotifChannel": "Tjekker for opdateringer", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Tjek kun installeret og 'Følg Kun'-apps for opdateringer", | ||||
|     "supportFixedAPKURL": "Understøt fikserede APK-URL'er", | ||||
|     "selectX": "Vælg {}", | ||||
|     "parallelDownloads": "Tillad samtidige overførsler", | ||||
|     "useShizuku": "Brug Shizuku eller Sui til at installere", | ||||
|     "shizukuBinderNotFound": "Shizuku-tjeneste kører ikke", | ||||
|     "shizukuOld": "Forældet Shizuku-version (<11). Opdater den", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku kører på Android <8.1 med ADB. Opdater Android eller brug Sui i stedet", | ||||
|     "shizukuPretendToBeGooglePlay": "Indstil Google Play som installationskilde (hvis Shizuku bruges)", | ||||
|     "useSystemFont": "Brug systemskrifttype", | ||||
|     "useVersionCodeAsOSVersion": "Brug app versionKode som OS-registreret version", | ||||
|     "requestHeader": "Anmod overskrift", | ||||
|     "useLatestAssetDateAsReleaseDate": "Brug seneste ressourceupload som udgivelsesdato", | ||||
|     "defaultPseudoVersioningMethod": "Standard pseudo-versioneringsmetode", | ||||
|     "partialAPKHash": "Delvis APK-hash", | ||||
|     "APKLinkHash": "Hash for APK-link", | ||||
|     "directAPKLink": "Direkte APK-link", | ||||
|     "pseudoVersionInUse": "En pseudo-version er i brug", | ||||
|     "installed": "Installeret", | ||||
|     "latest": "Seneste", | ||||
|     "invertRegEx": "Inverter regulært udtryk", | ||||
|     "note": "Bemærk", | ||||
|     "selfHostedNote": "Rullemenuen \"{}\" kan bruges til at nå selvhostede/brugerdefinerede instanser af enhver kilde.", | ||||
|     "badDownload": "APK'en kunne ikke analyseres (inkompatibel eller delvis hentning)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Del nye apps med AppVerifier (hvis tilgængelig)", | ||||
|     "appVerifierInstructionToast": "Del til AppVerifier, og vend tilbage hertil, når du er klar.", | ||||
|     "wiki": "Hjælp/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourcede app-konfigurationer (brug på egen risiko)", | ||||
|     "allowInsecure": "Tillad usikre HTTP-anmodninger", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Fjern app?", | ||||
|         "other": "Fjern apps?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "For mange anmodninger (begrænset hastighed). Prøv igen om {} minut", | ||||
|         "other": "For mange anmodninger (begrænset hastighed). Prøv igen om {} minutter" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "Baggrundsopdateringstjek stødte på en {}. Planlægger et nyt tjek om {} minut", | ||||
|         "other": "Baggrundsopdateringstjek stødte på en {}. Planlægger et nyt tjek om {} minutter" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "Baggrundsopdateringstjek fandt {} opdatering. Underretter brugeren, hvis nødvendigt", | ||||
|         "other": "Baggrundsopdateringstjek fandt {} opdateringer. Underretter brugeren, hvis nødvendigt" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} App", | ||||
|         "other": "{} Apps" | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URL'er" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} Minut", | ||||
|         "other": "{} Minutter" | ||||
|     }, | ||||
|     "hour": { | ||||
|         "one": "{} Time", | ||||
|         "other": "{} Timer" | ||||
|     }, | ||||
|     "day": { | ||||
|         "one": "{} Dag", | ||||
|         "other": "{} Dage" | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "Ryddet {n} log (før = {before}, efter = {after})", | ||||
|         "other": "Ryddet {n} logs (før = {before}, efter = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} og 1 anden app har opdateringer.", | ||||
|         "other": "{} og {} andre apps har opdateringer." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} og 1 anden app blev opdateret.", | ||||
|         "other": "{} og {} andre apps blev opdateret." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Kunne ikke opdatere {} og 1 anden app.", | ||||
|         "other": "Kunne ikke opdatere {} og {} andre apps." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} og 1 anden app blev muligvis opdateret.", | ||||
|         "other": "{} og {} andre apps blev muligvis opdateret." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APK'er" | ||||
|     } | ||||
| } | ||||
| @@ -11,7 +11,7 @@ | ||||
|     "unexpectedError": "Unerwarteter Fehler", | ||||
|     "ok": "OK", | ||||
|     "and": "und", | ||||
|     "githubPATLabel": "GitHub Personal Access Token (Erhöht das Ratenlimit)", | ||||
|     "githubPATLabel": "Persönlicher Zugangstoken für GitHub (erhöht das Ratenlimit)", | ||||
|     "includePrereleases": "Vorabversionen einbeziehen", | ||||
|     "fallbackToOlderReleases": "Fallback auf ältere Versionen", | ||||
|     "filterReleaseTitlesByRegEx": "Release-Titel nach regulärem Ausdruck\nfiltern", | ||||
| @@ -24,15 +24,15 @@ | ||||
|     "colour": "Farbe", | ||||
|     "standard": "Standard", | ||||
|     "custom": "Benutzerdefiniert", | ||||
|     "useMaterialYou": "Verwende Material You", | ||||
|     "useMaterialYou": "Material You verwenden", | ||||
|     "githubStarredRepos": "GitHub Starred Repos", | ||||
|     "uname": "Benutzername", | ||||
|     "wrongArgNum": "Falsche Anzahl von Argumenten (Parametern) übermittelt", | ||||
|     "xIsTrackOnly": "{} ist nur zur Nachverfolgung", | ||||
|     "source": "Quelle", | ||||
|     "app": "App", | ||||
|     "appsFromSourceAreTrackOnly": "Apps aus dieser Quelle sind nur zum Nachverfolgen.", | ||||
|     "youPickedTrackOnly": "Sie haben die Option „Nur Nachverfolgen“ gewählt.", | ||||
|     "appsFromSourceAreTrackOnly": "Apps aus dieser Quelle sind nur zur Nachverfolgung.", | ||||
|     "youPickedTrackOnly": "Sie haben die Option „Nur nachverfolgen“ gewählt.", | ||||
|     "trackOnlyAppDescription": "Die App wird auf Aktualisierungen überwacht, aber Obtainium wird sie nicht herunterladen oder installieren.", | ||||
|     "cancelled": "Abgebrochen", | ||||
|     "appAlreadyAdded": "App bereits hinzugefügt", | ||||
| @@ -45,7 +45,7 @@ | ||||
|     "search": "Suchen", | ||||
|     "additionalOptsFor": "Zusatzoptionen für {}", | ||||
|     "supportedSources": "Unterstützte Quellen", | ||||
|     "trackOnlyInBrackets": "(Nur Nachverfolgen)", | ||||
|     "trackOnlyInBrackets": "(Nur nachverfolgen)", | ||||
|     "searchableInBrackets": "(Durchsuchbar)", | ||||
|     "appsString": "Apps", | ||||
|     "noApps": "Keine Apps", | ||||
| @@ -61,22 +61,22 @@ | ||||
|     "xWillBeRemovedButRemainInstalled": "{} wird aus Obtainium entfernt, bleibt aber auf dem Gerät installiert.", | ||||
|     "removeSelectedAppsQuestion": "Ausgewählte Apps entfernen?", | ||||
|     "removeSelectedApps": "Ausgewählte Apps entfernen", | ||||
|     "updateX": "Aktualisiere {}", | ||||
|     "installX": "Installiere {}", | ||||
|     "markXTrackOnlyAsUpdated": "Markiere {}\n(Nur Nachverfolgen)\nals aktualisiert", | ||||
|     "changeX": "Ändere {}", | ||||
|     "updateX": "{} aktualisieren", | ||||
|     "installX": "{} installieren", | ||||
|     "markXTrackOnlyAsUpdated": "Markiere {}\n(Nur nachverfolgen)\nals aktualisiert", | ||||
|     "changeX": "{} ändern", | ||||
|     "installUpdateApps": "Apps installieren/aktualisieren", | ||||
|     "installUpdateSelectedApps": "Ausgewählte Apps installieren/aktualisieren", | ||||
|     "markXSelectedAppsAsUpdated": "Markiere {} ausgewählte Apps als aktuell?", | ||||
|     "markXSelectedAppsAsUpdated": "{} ausgewählte Apps als aktuell markieren?", | ||||
|     "no": "Nein", | ||||
|     "yes": "Ja", | ||||
|     "markSelectedAppsUpdated": "Markiere ausgewählte Apps als aktuell", | ||||
|     "markSelectedAppsUpdated": "Ausgewählte Apps als aktuell markieren", | ||||
|     "pinToTop": "Oben anheften", | ||||
|     "unpinFromTop": "„Oben anheften“ aufheben", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Installationsstatus für ausgewählte Apps zurücksetzen?", | ||||
|     "installStatusOfXWillBeResetExplanation": "Der Installationsstatus der ausgewählten Apps wird zurückgesetzt. Dies kann hilfreich sein, wenn die in Obtainium angezeigte App-Version aufgrund fehlgeschlagener Aktualisierungen oder anderer Probleme falsch ist.", | ||||
|     "customLinkMessage": "Diese Links funktionieren auf Geräten, wo Obtainium installiert ist", | ||||
|     "shareAppConfigLinks": "Teile die Appkonfiguration als HTML-Link", | ||||
|     "shareAppConfigLinks": "App-Konfiguration als HTML-Link teilen", | ||||
|     "shareSelectedAppURLs": "Ausgewählte App-URLs teilen", | ||||
|     "resetInstallStatus": "Installationsstatus zurücksetzen", | ||||
|     "more": "Mehr", | ||||
| @@ -84,25 +84,25 @@ | ||||
|     "showOutdatedOnly": "Nur nicht aktuelle Apps anzeigen", | ||||
|     "filter": "Filter", | ||||
|     "filterApps": "Apps filtern", | ||||
|     "appName": "App Name", | ||||
|     "appName": "App-Name", | ||||
|     "author": "Autor", | ||||
|     "upToDateApps": "Apps mit aktueller Version", | ||||
|     "nonInstalledApps": "Nicht installierte Apps", | ||||
|     "importExport": "Import/Export", | ||||
|     "settings": "Einstellungen", | ||||
|     "exportedTo": "Exportiert zu {}", | ||||
|     "exportedTo": "Exportiert nach {}", | ||||
|     "obtainiumExport": "Obtainium-Export", | ||||
|     "invalidInput": "Ungültige Eingabe", | ||||
|     "importedX": "Importiert {}", | ||||
|     "importedX": "{} importiert", | ||||
|     "obtainiumImport": "Obtainium-Import", | ||||
|     "importFromURLList": "Importieren aus URL-Liste", | ||||
|     "importFromURLList": "Aus URL-Liste importieren", | ||||
|     "searchQuery": "Suchanfrage", | ||||
|     "appURLList": "App URL-Liste", | ||||
|     "appURLList": "App-URL-Liste", | ||||
|     "line": "Linie", | ||||
|     "searchX": "Suche {}", | ||||
|     "searchX": "{} suchen", | ||||
|     "noResults": "Keine Ergebnisse gefunden", | ||||
|     "importX": "Importieren {}", | ||||
|     "importedAppsIdDisclaimer": "Importierte Apps werden möglicherweise fälschlicherweise als \"Nicht installiert\" angezeigt. Um dies zu beheben, installieren Sie sie erneut über Obtainium. Dies hat keine Auswirkungen auf App-Daten. Es betrifft nur URL- und Drittanbieter-Importmethoden.", | ||||
|     "importX": "{} importieren", | ||||
|     "importedAppsIdDisclaimer": "Importierte Apps werden manchmal fälschlicherweise als „Nicht installiert“ angezeigt. Um dies zu beheben, installieren Sie sie erneut über Obtainium. Dies hat keine Auswirkungen auf App-Daten. Es betrifft nur URL- und Drittanbieter-Importmethoden.", | ||||
|     "importErrors": "Importfehler", | ||||
|     "importedXOfYApps": "{} von {} Apps importiert.", | ||||
|     "followingURLsHadErrors": "Bei folgenden URLs traten Fehler auf:", | ||||
| @@ -113,8 +113,8 @@ | ||||
|     "dark": "Dunkel", | ||||
|     "light": "Hell", | ||||
|     "followSystem": "System folgen", | ||||
|     "followSystemThemeExplanation": "Das Folgen des Systemthemes ist unter Android <10 nur mit Hilfe von Drittanbieteranwendungen möglich", | ||||
|     "useBlackTheme": "Verwende Pure Black Dark Theme", | ||||
|     "followSystemThemeExplanation": "Das Folgen des Systemthemes ist unter Android < 10 nur mit Hilfe von Drittanbieterapps möglich", | ||||
|     "useBlackTheme": "Pure Black Dark Theme verwenden", | ||||
|     "appSortBy": "App sortieren nach", | ||||
|     "authorName": "Autor/Name", | ||||
|     "nameAuthor": "Name/Autor", | ||||
| @@ -146,8 +146,8 @@ | ||||
|     "updatesAvailableNotifDescription": "Benachrichtigt den Nutzer, dass Aktualisierungen für eine oder mehrere von Obtainium verfolgte Apps verfügbar sind", | ||||
|     "noNewUpdates": "Keine neuen Aktualisierungen.", | ||||
|     "xHasAnUpdate": "{} hat eine Aktualisierung.", | ||||
|     "appsUpdated": "Apps aktualisiert", | ||||
|     "appsNotUpdated": "Aktualisierung der Anwendungen fehlgeschlagen", | ||||
|     "appsUpdated": "App wurde aktualisiert", | ||||
|     "appsNotUpdated": "Aktualisierung der Apps fehlgeschlagen", | ||||
|     "appsUpdatedNotifDescription": "Benachrichtigt den Benutzer, dass Aktualisierungen für eine oder mehrere Apps im Hintergrund durchgeführt wurden", | ||||
|     "xWasUpdatedToY": "{} wurde auf {} aktualisiert.", | ||||
|     "xWasNotUpdatedToY": "Die Aktualisierung von {} auf {} ist fehlgeschlagen.", | ||||
| @@ -156,13 +156,13 @@ | ||||
|     "appsRemoved": "Apps entfernt", | ||||
|     "appsRemovedNotifDescription": "Benachrichtigt den Benutzer, dass eine oder mehrere Apps aufgrund von Fehlern beim Laden entfernt wurden", | ||||
|     "xWasRemovedDueToErrorY": "{} wurde aufgrund des folgenden Fehlers entfernt: {}", | ||||
|     "completeAppInstallation": "App Installation abschließen", | ||||
|     "completeAppInstallation": "App-Installation abschließen", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium muss geöffnet sein, um Apps zu installieren", | ||||
|     "completeAppInstallationNotifDescription": "Aufforderung an den Benutzer, zu Obtainium zurückzukehren, um die Installation einer App abzuschließen", | ||||
|     "checkingForUpdates": "Nach Aktualisierungen suchen", | ||||
|     "checkingForUpdatesNotifDescription": "Vorübergehende Benachrichtigung, die bei der Suche nach Aktualisierungen angezeigt wird", | ||||
|     "pleaseAllowInstallPerm": "Bitte erlauben Sie Obtainium die Installation von Apps", | ||||
|     "trackOnly": "Nur Nachverfolgen", | ||||
|     "trackOnly": "Nur nachverfolgen", | ||||
|     "errorWithHttpStatusCode": "Fehler {}", | ||||
|     "versionCorrectionDisabled": "Versionskorrektur deaktiviert (Plugin scheint nicht zu funktionieren)", | ||||
|     "unknown": "Unbekannt", | ||||
| @@ -173,25 +173,25 @@ | ||||
|     "lastUpdateCheckX": "Letzte Aktualisierungsprüfung: {}", | ||||
|     "remove": "Entfernen", | ||||
|     "yesMarkUpdated": "Ja, als aktualisiert markieren", | ||||
|     "fdroid": "offizielles F-Droid-Repo", | ||||
|     "appIdOrName": "App ID oder Name", | ||||
|     "appId": "App ID", | ||||
|     "fdroid": "Offizielles F-Droid-Repo", | ||||
|     "appIdOrName": "App-ID oder Name", | ||||
|     "appId": "App-ID", | ||||
|     "appWithIdOrNameNotFound": "Es wurde keine App mit dieser ID oder diesem Namen gefunden", | ||||
|     "reposHaveMultipleApps": "Repos können mehrere Apps enthalten", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Drittparteienrepo", | ||||
|     "fdroidThirdPartyRepo": "F-Droid-Drittanbieter-Repo", | ||||
|     "steamMobile": "Steam Mobile", | ||||
|     "steamChat": "Steam-Chat", | ||||
|     "install": "Installieren", | ||||
|     "markInstalled": "Als Installiert markieren", | ||||
|     "markInstalled": "Als installiert markieren", | ||||
|     "update": "Aktualisieren", | ||||
|     "markUpdated": "Als Aktuell markieren", | ||||
|     "markUpdated": "Als aktuell markieren", | ||||
|     "additionalOptions": "Zusätzliche Optionen", | ||||
|     "disableVersionDetection": "Versionsermittlung deaktivieren", | ||||
|     "disableVersionDetection": "Versionserkennung deaktivieren", | ||||
|     "noVersionDetectionExplanation": "Diese Option sollte nur für Apps verwendet werden, bei denen die Versionserkennung nicht korrekt funktioniert.", | ||||
|     "downloadingX": "Lade {} herunter", | ||||
|     "downloadingX": "{} wird heruntergeladen", | ||||
|     "downloadX": "{} herunterladen", | ||||
|     "downloadedX": "{} heruntergeladen", | ||||
|     "releaseAsset": "release Asset", | ||||
|     "releaseAsset": "Release-Asset", | ||||
|     "downloadNotifDescription": "Benachrichtigt den Nutzer über den Fortschritt beim Herunterladen einer App", | ||||
|     "noAPKFound": "Keine APK gefunden", | ||||
|     "noVersionDetection": "Keine Versionserkennung", | ||||
| @@ -216,50 +216,52 @@ | ||||
|     "releaseDateAsVersionExplanation": "Diese Option sollte nur für Apps verwendet werden, bei denen die Versionserkennung nicht korrekt funktioniert, aber ein Veröffentlichungsdatum verfügbar ist.", | ||||
|     "changes": "Änderungen", | ||||
|     "releaseDate": "Veröffentlichungsdatum", | ||||
|     "importFromURLsInFile": "Importieren von URLs aus Datei (z. B. OPML)", | ||||
|     "importFromURLsInFile": "URLs aus Datei importieren (z. B. OPML)", | ||||
|     "versionDetectionExplanation": "Abgleich der Versionsnummer mit der vom Betriebssystem erkannten Version", | ||||
|     "versionDetection": "Versionserkennung", | ||||
|     "standardVersionDetection": "Standardversionserkennung", | ||||
|     "groupByCategory": "Nach Kategorie gruppieren", | ||||
|     "autoApkFilterByArch": "Nach Möglichkeit versuchen, APKs nach CPU-Architektur zu filtern", | ||||
|     "overrideSource": "Quelle überschreiben", | ||||
|     "dontShowAgain": "Nicht noch einmal zeigen", | ||||
|     "dontShowTrackOnlyWarnings": "Warnung für 'Nur Nachverfolgen' nicht anzeigen", | ||||
|     "dontShowAgain": "Nicht noch einmal anzeigen", | ||||
|     "dontShowTrackOnlyWarnings": "Warnung für 'Nur nachverfolgen' nicht anzeigen", | ||||
|     "dontShowAPKOriginWarnings": "Warnung für APK-Herkunft nicht anzeigen", | ||||
|     "moveNonInstalledAppsToBottom": "Nicht installierte Apps ans Ende der Apps Ansicht verschieben", | ||||
|     "gitlabPATLabel": "GitLab Personal Access Token", | ||||
|     "moveNonInstalledAppsToBottom": "Nicht installierte Apps ans Ende der App-Ansicht verschieben", | ||||
|     "gitlabPATLabel": "Persönlicher Zugangstoken für GitLab", | ||||
|     "about": "Über", | ||||
|     "requiresCredentialsInSettings": "{}: Benötigt zusätzliche Anmeldedaten (in den Einstellungen)", | ||||
|     "checkOnStart": "Überprüfe einmalig beim Start", | ||||
|     "tryInferAppIdFromCode": "Versuche, die App-ID aus dem Quellcode zu ermitteln", | ||||
|     "checkOnStart": "Einmalig beim Start überprüfen", | ||||
|     "tryInferAppIdFromCode": "Versuchen, die App-ID aus dem Quellcode zu ermitteln", | ||||
|     "removeOnExternalUninstall": "Automatisches Entfernen von extern deinstallierten Apps", | ||||
|     "pickHighestVersionCode": "Automatische Auswahl des APK mit höchstem Versionscode", | ||||
|     "checkUpdateOnDetailPage": "Nach Aktualisierungen suchen, wenn eine App-Detailseite geöffnet wird", | ||||
|     "disablePageTransitions": "Animationen für Seitenübergänge deaktivieren", | ||||
|     "reversePageTransitions": "Umgekehrte Animationen für Seitenübergänge", | ||||
|     "minStarCount": "Minimale Anzahl von Sternen", | ||||
|     "addInfoBelow": "Fügen Sie diese Informationen unten hinzu.", | ||||
|     "addInfoInSettings": "Fügen Sie diese Info in den Einstellungen hinzu.", | ||||
|     "addInfoBelow": "Diese Information unten hinzufügen.", | ||||
|     "addInfoInSettings": "Diese Information in den Einstellungen hinzufügen.", | ||||
|     "githubSourceNote": "Die GitHub-Ratenbegrenzung kann mit einem API-Schlüssel umgangen werden.", | ||||
|     "sortByLastLinkSegment": "Sortiere nur nach dem letzten Teil des Links", | ||||
|     "sortByLastLinkSegment": "Nur nach dem letzten Teil des Links sortieren", | ||||
|     "filterReleaseNotesByRegEx": "Versionshinweise nach regulärem Ausdruck filtern", | ||||
|     "customLinkFilterRegex": "Benutzerdefinierter APK Link Filter nach Regulärem Ausdruck (Standard '.apk$')", | ||||
|     "appsPossiblyUpdated": "App Aktualisierungen wurden versucht", | ||||
|     "customLinkFilterRegex": "Benutzerdefinierter APK-Linkfilter durch regulären Ausdruck (Standard '.apk$')", | ||||
|     "appsPossiblyUpdated": "App-Aktualisierungen wurden versucht", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Benachrichtigt den Benutzer, dass Aktualisierungen für eine oder mehrere Apps möglicherweise im Hintergrund durchgeführt wurden", | ||||
|     "xWasPossiblyUpdatedToY": "{} wurde möglicherweise aktualisiert auf {}.", | ||||
|     "enableBackgroundUpdates": "Aktiviere Hintergrundaktualisierungen", | ||||
|     "enableBackgroundUpdates": "Hintergrundaktualisierungen aktivieren", | ||||
|     "backgroundUpdateReqsExplanation": "Die Hintergrundaktualisierung ist möglicherweise nicht für alle Apps möglich.", | ||||
|     "backgroundUpdateLimitsExplanation": "Der Erfolg einer Hintergrundinstallation kann nur festgestellt werden, wenn Obtainium geöffnet wird.", | ||||
|     "verifyLatestTag": "Überprüfe das „latest“ Tag", | ||||
|     "verifyLatestTag": "„Latest“-Tag überprüfen", | ||||
|     "intermediateLinkRegex": "Filter für einen „Zwischen“-Link, der zuerst besucht werden soll", | ||||
|     "filterByLinkText": "Filtere Links durch Linktext", | ||||
|     "filterByLinkText": "Links durch Linktext filtern", | ||||
|     "intermediateLinkNotFound": "„Zwischen“-Link nicht gefunden", | ||||
|     "intermediateLink": "„Zwischen“-Link", | ||||
|     "exemptFromBackgroundUpdates": "Ausschluss von Hintergrundaktualisierungen (falls aktiviert)", | ||||
|     "bgUpdatesOnWiFiOnly": "Hintergrundaktualisierungen deaktivieren, wenn kein WLAN vorhanden ist", | ||||
|     "autoSelectHighestVersionCode": "Automatisch höchste APK-Version auswählen", | ||||
|     "versionExtractionRegEx": "Versions-Extraktion per RegEx", | ||||
|     "matchGroupToUse": "zu verwendende Gruppe abgleichen", | ||||
|     "versionExtractionRegEx": "Versionsextraktion per RegEx", | ||||
|     "trimVersionString": "Versionszeichenfolge mit RegEx kürzen", | ||||
|     "matchGroupToUseForX": "Zu verwendende Abgleichsgruppe für „{}“", | ||||
|     "matchGroupToUse": "Zu verwendende Gruppe abgleichen", | ||||
|     "highlightTouchTargets": "Weniger offensichtliche Touch-Ziele hervorheben", | ||||
|     "pickExportDir": "Export-Verzeichnis wählen", | ||||
|     "autoExportOnChanges": "Automatischer Export bei Änderung(en)", | ||||
| @@ -268,57 +270,58 @@ | ||||
|     "trySelectingSuggestedVersionCode": "Versuchen, den vorgeschlagenen APK-Versionscode auszuwählen", | ||||
|     "dontSortReleasesList": "Freigaberelease von der API ordern", | ||||
|     "reverseSort": "Umgekehrtes Sortieren", | ||||
|     "takeFirstLink": "Verwende den ersten Link", | ||||
|     "skipSort": "Überspringe Sortieren", | ||||
|     "takeFirstLink": "Ersten Link verwenden", | ||||
|     "skipSort": "Sortieren überspringen", | ||||
|     "debugMenu": "Debug-Menü", | ||||
|     "bgTaskStarted": "Hintergrundaufgabe gestartet – Logs prüfen.", | ||||
|     "runBgCheckNow": "Hintergrundaktualisierungsprüfung jetzt durchführen", | ||||
|     "versionExtractWholePage": "Versions-Extraktion per RegEx auf die gesamte Seite anwenden", | ||||
|     "installing": "Installiere", | ||||
|     "skipUpdateNotifications": "Keine Benachrichtigung zu App-Aktualisierungen geben", | ||||
|     "versionExtractWholePage": "Versionsextraktion per RegEx auf die gesamte Seite anwenden", | ||||
|     "installing": "Installieren", | ||||
|     "skipUpdateNotifications": "Update-Benachrichtigungen überspringen", | ||||
|     "updatesAvailableNotifChannel": "Aktualisierungen verfügbar", | ||||
|     "appsUpdatedNotifChannel": "Apps aktualisiert", | ||||
|     "appsPossiblyUpdatedNotifChannel": "App Aktualisierungen wurden versucht", | ||||
|     "appsPossiblyUpdatedNotifChannel": "App-Aktualisierungen wurden versucht", | ||||
|     "errorCheckingUpdatesNotifChannel": "Fehler beim Prüfen auf Aktualisierungen", | ||||
|     "appsRemovedNotifChannel": "Apps entfernt", | ||||
|     "downloadingXNotifChannel": "Lade {} herunter", | ||||
|     "completeAppInstallationNotifChannel": "App Installation abschließen", | ||||
|     "downloadingXNotifChannel": "{} herunterladen", | ||||
|     "completeAppInstallationNotifChannel": "App-Installation abschließen", | ||||
|     "checkingForUpdatesNotifChannel": "Nach Aktualisierungen suchen", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Überprüfe nur installierte und mit „nur Nachverfolgen“ markierte Apps auf Aktualisierungen", | ||||
|     "supportFixedAPKURL": "neuere Version anhand der ersten dreißig Zahlen der Checksumme der APK URL erraten, wenn anderweitig nicht unterstützt", | ||||
|     "selectX": "Wähle {}", | ||||
|     "parallelDownloads": "Erlaube parallele Downloads", | ||||
|     "useShizuku": "Verwenden Sie Shizuku oder Sui zur Installation", | ||||
|     "shizukuBinderNotFound": "Kompatibler Shizukudienst wurde nicht gefunden", | ||||
|     "shizukuOld": "Alte Shizuku-Version (<11) - aktualisieren Sie sie", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Nur installierte und mit „Nur nachverfolgen“ markierte Apps auf Aktualisierungen prüfen", | ||||
|     "supportFixedAPKURL": "Unterstützung von festen APK-URLs", | ||||
|     "selectX": "{} wählen", | ||||
|     "parallelDownloads": "Parallele Downloads erlauben", | ||||
|     "useShizuku": "Shizuku oder Sui zur Installation verwenden", | ||||
|     "shizukuBinderNotFound": "Kompatibler Shizuku-Dienst wurde nicht gefunden", | ||||
|     "shizukuOld": "Alte Shizuku-Version (< 11) - aktualisieren Sie sie", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku läuft auf Android < 8.1 mit ADB - aktualisieren Sie Android oder verwenden Sie stattdessen Sui", | ||||
|     "shizukuPretendToBeGooglePlay": "Google Play als Installationsquelle festlegen (wenn Shizuku verwendet wird)", | ||||
|     "useSystemFont": "Verwende die Systemschriftart", | ||||
|     "useVersionCodeAsOSVersion": "Verwende die Appversion als erkannte Version vom Betriebssystem", | ||||
|     "requestHeader": "Request Header", | ||||
|     "useLatestAssetDateAsReleaseDate": "Den letzten Asset-Upload als Veröffentlichungsdatum verwenden", | ||||
|     "useSystemFont": "Systemschriftart verwenden", | ||||
|     "useVersionCodeAsOSVersion": "App-Version als erkannte Version vom Betriebssystem verwenden", | ||||
|     "requestHeader": "Kopfzeile anfordern", | ||||
|     "useLatestAssetDateAsReleaseDate": "Letzten Asset-Upload als Veröffentlichungsdatum verwenden", | ||||
|     "defaultPseudoVersioningMethod": "Standardmäßiges Verfahren zur Pseudo-Versionierung", | ||||
|     "partialAPKHash": "partieller APK-Hash", | ||||
|     "partialAPKHash": "Partieller APK-Hash", | ||||
|     "APKLinkHash": "APK-Link-Hash", | ||||
|     "directAPKLink": "Direkter APK-Link", | ||||
|     "pseudoVersionInUse": "Es werden Pseudoversionen verwendet", | ||||
|     "installed": "Installiert", | ||||
|     "latest": "Neueste Version", | ||||
|     "invertRegEx": "Regulären Ausdruck  invertieren", | ||||
|     "invertRegEx": "Regulären Ausdruck invertieren", | ||||
|     "note": "Hinweis", | ||||
|     "selfHostedNote": "Das „{}“-Dropdown-Menü kann verwendet werden, um selbst gehostete/angepasste Instanzen einer beliebigen Quelle zu erreichen.", | ||||
|     "selfHostedNote": "Das „{}“-Drop-down-Menü kann verwendet werden, um selbst gehostete/angepasste Instanzen einer beliebigen Quelle zu erreichen.", | ||||
|     "badDownload": "Die APK konnte nicht geparst werden (inkompatibler oder teilweiser Download)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Neue Apps mit AppVerifier teilen (falls verfügbar)", | ||||
|     "appVerifierInstructionToast": "Geben Sie die Daten an AppVerifier weiter und kehren Sie dann hierher zurück, wenn Sie fertig sind.", | ||||
|     "wiki": "Hilfe/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (Verwendung auf eigene Gefahr)", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App-Konfigurationen (Verwendung auf eigene Gefahr)", | ||||
|     "allowInsecure": "Unsichere HTTP-Anfragen zulassen", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "App entfernen?", | ||||
|         "other": "Apps entfernen?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "Zu viele Anfragen (Rate begrenzt) – versuche es in {} Minute erneut", | ||||
|         "other": "Zu viele Anfragen (Rate begrenzt) – versuche es in {} Minuten erneut" | ||||
|         "one": "Zu viele Anfragen (Rate begrenzt) – versuchen Sie es in {} Minute erneut", | ||||
|         "other": "Zu viele Anfragen (Rate begrenzt) – versuchen Sie es in {} Minuten erneut" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minute geplant", | ||||
| @@ -329,12 +332,12 @@ | ||||
|         "other": "Die Hintergrundaktualisierungsprüfung fand {} Aktualisierungen – benachrichtigt den Benutzer, falls erforderlich" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "eine": "{} App", | ||||
|         "andere": "{} Apps" | ||||
|         "one": "{} App", | ||||
|         "other": "{} Apps" | ||||
|     }, | ||||
|     "url": { | ||||
|         "eine": "{} URL", | ||||
|         "andere": "{} URLs" | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URLs" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} Minute", | ||||
| @@ -357,16 +360,16 @@ | ||||
|         "other": "{} und {} weitere Apps haben Aktualisierungen." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} und 1 weitere Anwendung wurden aktualisiert.", | ||||
|         "other": "{} und {} weitere Anwendungen wurden aktualisiert." | ||||
|         "one": "{} und 1 weitere App wurden aktualisiert.", | ||||
|         "other": "{} und {} weitere Apps wurden aktualisiert." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Aktualisierung fehlgeschlagen {} und 1 weitere Anwendung.", | ||||
|         "other": "Die Aktualisierung von {} und {} weiteren Anwendungen ist fehlgeschlagen." | ||||
|         "one": "Die Aktualisierung von {} und 1 weiteren App ist fehlgeschlagen.", | ||||
|         "other": "Die Aktualisierung von {} und {} weiteren Apps ist fehlgeschlagen." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} und 1 weitere Anwendung wurden möglicherweise aktualisiert.", | ||||
|         "other": "{} und {} weitere Anwendungen wurden möglicherweise aktualisiert." | ||||
|         "one": "{} und 1 weitere App wurden möglicherweise aktualisiert.", | ||||
|         "other": "{} und {} weitere Apps wurden möglicherweise aktualisiert." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|   | ||||
							
								
								
									
										378
									
								
								assets/translations/en-EO.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										378
									
								
								assets/translations/en-EO.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,378 @@ | ||||
| { | ||||
|     "invalidURLForSource": "Nevalida apo-URL {}", | ||||
|     "noReleaseFound": "Ne eblas trovi taŭgan eldonon", | ||||
|     "noVersionFound": "Ne eblas determini la eldonversion", | ||||
|     "urlMatchesNoSource": "La URL ne konformas al konata fonto", | ||||
|     "cantInstallOlderVersion": "Ne eblas instali malnovan version de la Apo", | ||||
|     "appIdMismatch": "La identigilo de la elŝutita pakaĵo ne konformas al la identigilo de la ekzistanta apo", | ||||
|     "functionNotImplemented": "Tiu klaso ne fasonadas tiun funkcion", | ||||
|     "placeholder": "Rezervilo", | ||||
|     "someErrors": "Eraroj okazis", | ||||
|     "unexpectedError": "Neatendita eraro", | ||||
|     "ok": "Okej", | ||||
|     "and": "kaj", | ||||
|     "githubPATLabel": "Github persona atingoĵetono (pliigas la kvantolimon)", | ||||
|     "includePrereleases": "Inkluzivi la pra-eldonojn", | ||||
|     "fallbackToOlderReleases": "Reveno al malnovaj versioj", | ||||
|     "filterReleaseTitlesByRegEx": "Filtri la eldontitoljn per regula esprimo", | ||||
|     "invalidRegEx": "Nevalida regula esprimo", | ||||
|     "noDescription": "Neniu priskribo", | ||||
|     "cancel": "Nuligi", | ||||
|     "continue": "Daŭrigi", | ||||
|     "requiredInBrackets": "(Neprigata)", | ||||
|     "dropdownNoOptsError": "ERARO: LA MALVOLVANTA LISTO DEVAS HAVI ALMENAŬ UNU OPCION", | ||||
|     "colour": "Koloro", | ||||
|     "standard": "Norma", | ||||
|     "custom": "Agordita", | ||||
|     "useMaterialYou": "Uzi Material You", | ||||
|     "githubStarredRepos": "Stelaj GitHub-deponejoj", | ||||
|     "uname": "Uzantnomo", | ||||
|     "wrongArgNum": "Malĝusta nombro da provizitaj argumentoj", | ||||
|     "xIsTrackOnly": "{} estas nur sekvitaj", | ||||
|     "source": "Fonto", | ||||
|     "app": "Apo", | ||||
|     "appsFromSourceAreTrackOnly": "Apoj el tiu fonto estas 'Nur sekvitaj'.", | ||||
|     "youPickedTrackOnly": "Vi selektis la opcion 'Nur sekvitaj'.", | ||||
|     "trackOnlyAppDescription": "La apo estas sekvota por ĝisdatigoj, sed Obtainium ne povos elŝuti aŭ instali ĝin.", | ||||
|     "cancelled": "Nuligita", | ||||
|     "appAlreadyAdded": "Jam aldonita apo", | ||||
|     "alreadyUpToDateQuestion": "Ĉu la apo estas ĝisdata?", | ||||
|     "addApp": "Aldoni apon", | ||||
|     "appSourceURL": "URL de la apofonto", | ||||
|     "error": "Eraro", | ||||
|     "add": "Aldoni", | ||||
|     "searchSomeSourcesLabel": "Serĉi (nur kelkaj fontoj)", | ||||
|     "search": "Serĉi", | ||||
|     "additionalOptsFor": "Kromaj opcioj por {}", | ||||
|     "supportedSources": "Taskiĝata fontoj", | ||||
|     "trackOnlyInBrackets": "(Nur sekvita)", | ||||
|     "searchableInBrackets": "(Serĉebla)", | ||||
|     "appsString": "Apoj", | ||||
|     "noApps": "Neniu apo", | ||||
|     "noAppsForFilter": "Neniu apo por filtri", | ||||
|     "byX": "Laŭ {}", | ||||
|     "percentProgress": "Progreso: {}%", | ||||
|     "pleaseWait": "Bonvolu atendi", | ||||
|     "updateAvailable": "Havebla ĝisdatigo", | ||||
|     "notInstalled": "Neinstalita", | ||||
|     "pseudoVersion": "Pseŭdoversio", | ||||
|     "selectAll": "Selekti ĉion", | ||||
|     "deselectX": "Malselekti {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} estas farigota de Obtainium sed restos instalata en la disponaĵo.", | ||||
|     "removeSelectedAppsQuestion": "Farigi la selektitajn apojn?", | ||||
|     "removeSelectedApps": "Farigi la selektitajn apojn", | ||||
|     "updateX": "Ĝisdatigi {}", | ||||
|     "installX": "Instali {}", | ||||
|     "markXTrackOnlyAsUpdated": "Marki {}\n(Nur sekvita)\nkiel ĝisdata", | ||||
|     "changeX": "Ŝanĝi {}", | ||||
|     "installUpdateApps": "Instali/ĝisdatigi apojn", | ||||
|     "installUpdateSelectedApps": "Instali/ĝisdatigi la selektitajn apojn", | ||||
|     "markXSelectedAppsAsUpdated": "Marki la {} selektitajn apojn kiel ĝisdataj?", | ||||
|     "no": "Ne", | ||||
|     "yes": "Jes", | ||||
|     "markSelectedAppsUpdated": "Marki la selektitajn apojn kiel ĝisdataj", | ||||
|     "pinToTop": "Alpingli supre", | ||||
|     "unpinFromTop": "Malalpingli el supro", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Rekomenci la instaladostaton de la selektitaj apoj?", | ||||
|     "installStatusOfXWillBeResetExplanation": "Instaladostato de ĉiuj selektitaj apoj estas rekomencota.\n\nTio povas utili kiam la versio de la montrata application en Obtainium estas malĝusta kaŭze de malsukcesaj ĝidatigoj aŭ aliaj problemoj.", | ||||
|     "customLinkMessage": "Tiuj ligiloj funkcias en disponaĵoj kun Obtainium instalita", | ||||
|     "shareAppConfigLinks": "Diskonigi la apo-agordojn kiel HTML ligilo", | ||||
|     "shareSelectedAppURLs": "Diskonigi la selektitajn apo-URLjn", | ||||
|     "resetInstallStatus": "Rekomenci la instaladostaton de la apoj", | ||||
|     "more": "Pli", | ||||
|     "removeOutdatedFilter": "Forigi la filtron de malmodernaj apoj", | ||||
|     "showOutdatedOnly": "Nur montri malmodernajn apojn", | ||||
|     "filter": "Filtro", | ||||
|     "filterApps": "Filtri la apojn", | ||||
|     "appName": "Aponomo", | ||||
|     "author": "Aŭtoro", | ||||
|     "upToDateApps": "Ĝisdataj apoj", | ||||
|     "nonInstalledApps": "Neinstalita apoj", | ||||
|     "importExport": "Importi/Eksporti", | ||||
|     "settings": "Agordoj", | ||||
|     "exportedTo": "Eksportita en {}", | ||||
|     "obtainiumExport": "Eksporti Obtainium-n", | ||||
|     "invalidInput": "Nevalida enigo", | ||||
|     "importedX": "Importita {}", | ||||
|     "obtainiumImport": "Importi Obtainium-n", | ||||
|     "importFromURLList": "Eksporti el URL-listo", | ||||
|     "searchQuery": "Informmendo de serĉo", | ||||
|     "appURLList": "Listo de URLj de la apo", | ||||
|     "line": "Horizontalo", | ||||
|     "searchX": "Serĉi {}", | ||||
|     "noResults": "Neniu rezulto", | ||||
|     "importX": "Importi {}", | ||||
|     "importedAppsIdDisclaimer": "La importitaj apoj povas montriĝi malĝuste kiel \"Neinstalitaj\".\nPor solvi tion, reinstalu ilin per Obtainium.\nTiu ne afekcios la apodatumoj.\n\nAkefcias nur la URL-ajn lak triajn importmetodojn.", | ||||
|     "importErrors": "Eraroj de importado", | ||||
|     "importedXOfYApps": "{} apoj el {} importitaj.", | ||||
|     "followingURLsHadErrors": "La sekvantaj URLj havis erarojn:", | ||||
|     "selectURL": "Selekti URLn", | ||||
|     "selectURLs": "Selekti URLjn", | ||||
|     "pick": "Elekti", | ||||
|     "theme": "Temo", | ||||
|     "dark": "Malhela", | ||||
|     "light": "Hela", | ||||
|     "followSystem": "Konformi al sistemo", | ||||
|     "followSystemThemeExplanation": "Konformi al sistemtemo nur eblas kun triaj aplikaĵoj.", | ||||
|     "useBlackTheme": "Uzi puran nigran temon", | ||||
|     "appSortBy": "Klasi apojn laŭ", | ||||
|     "authorName": "Aŭtoro/Nomo", | ||||
|     "nameAuthor": "Nomo/Aŭtoro", | ||||
|     "asAdded": "Aldondato", | ||||
|     "appSortOrder": "Ordo de la apoklasifiko", | ||||
|     "ascending": "Kreskanta", | ||||
|     "descending": "Malkreskanta", | ||||
|     "bgUpdateCheckInterval": "Serĉo-intertempo de fonaj ĝisdatigoj", | ||||
|     "neverManualOnly": "Neniam - Nur Malaŭtomata", | ||||
|     "appearance": "Aspekto", | ||||
|     "showWebInAppView": "Montri la fonta retpaĝo en la langeto 'Apoj'", | ||||
|     "pinUpdates": "Alpingli ĝisatigojn en la supro de la langeto 'Apoj'", | ||||
|     "updates": "Ĝisdatigoj", | ||||
|     "sourceSpecific": "Fontspecifa", | ||||
|     "appSource": "Apofonto", | ||||
|     "noLogs": "Neniu protokolo", | ||||
|     "appLogs": "Apoprotokoloj", | ||||
|     "close": "Fermi", | ||||
|     "share": "Diskonigi", | ||||
|     "appNotFound": "Netrovebla apo", | ||||
|     "obtainiumExportHyphenatedLowercase": "eksporto-obtainium", | ||||
|     "pickAnAPK": "Selektu APKn", | ||||
|     "appHasMoreThanOnePackage": "{} havas pli ol unu pakaĵon:", | ||||
|     "deviceSupportsXArch": "Via disponaĵo taskiĝas la CPU-arkitekturon {}.", | ||||
|     "deviceSupportsFollowingArchs": "Via disponaĵo taskiĝas la sekvantajn CPU-arkitekturojn:", | ||||
|     "warning": "Avertu", | ||||
|     "sourceIsXButPackageFromYPrompt": "La apofonto estas '{}' sed la pakaĵo de ĝisdatigo devenas de '{}'. Daŭrigi?", | ||||
|     "updatesAvailable": "Haveblaj ĝisdatigoj", | ||||
|     "updatesAvailableNotifDescription": "Sciigas al la uzanto, ke ĝisdatigoj haveblas por unu aŭ pluraj apoj sekvitaj fare de Obtainium.", | ||||
|     "noNewUpdates": "Neniu nova ĝisdatigo.", | ||||
|     "xHasAnUpdate": "{} havas ĝisdatigon.", | ||||
|     "appsUpdated": "Ĝisdataj apoj", | ||||
|     "appsNotUpdated": "Malsukcesis ĝisdatigi aplikaĵojn", | ||||
|     "appsUpdatedNotifDescription": "Sciigas al la uzanto, ke ĝisdatigoj de unu aŭ pluraj apoj estas instalitaj fone.", | ||||
|     "xWasUpdatedToY": "{} estas ĝidatigita en {}.", | ||||
|     "xWasNotUpdatedToY": "Malsukcesas ĝisdatigi de {} en {}.", | ||||
|     "errorCheckingUpdates": "Eraro serĉante ĝisdatigojn", | ||||
|     "errorCheckingUpdatesNotifDescription": "Sciigas al la uzanto, kiam la serĉo de fonaj ĝisdatigoj malsukcesas.", | ||||
|     "appsRemoved": "Forigitaj Apoj", | ||||
|     "appsRemovedNotifDescription": "Sciigas al la uzanto, ke unu aŭ pluraj apoj estas forigitaj kaŭze de eraroj dum ilia ŝarĝado.", | ||||
|     "xWasRemovedDueToErrorY": "{} estas forigita kaŭze de tiu eraro: {}", | ||||
|     "completeAppInstallation": "Kompleta instalado de la apo", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium devas esti malfermita por instali apojn", | ||||
|     "completeAppInstallationNotifDescription": "Petas la uzanton reiri en Obtainium por fini la instaladon de apo", | ||||
|     "checkingForUpdates": "Serĉante ĝisdatigojn", | ||||
|     "checkingForUpdatesNotifDescription": "Dumtempa sciigo kiu aperas dum la serĉo de ĝisdatigoj", | ||||
|     "pleaseAllowInstallPerm": "Bonvolu permesi al Obtainium instali apojn", | ||||
|     "trackOnly": "Nur sekvita", | ||||
|     "errorWithHttpStatusCode": "Eraro {}", | ||||
|     "versionCorrectionDisabled": "Versiokorekto malaktivigita (ŝajnas, ke la kromaĵo ne funkcias)", | ||||
|     "unknown": "Nekonata", | ||||
|     "none": "Nenio", | ||||
|     "never": "Neniam", | ||||
|     "latestVersionX": "Lasta versio: {}", | ||||
|     "installedVersionX": "Instalita versio: {}", | ||||
|     "lastUpdateCheckX": "Lasta serĉo de ĝisdatigoj: {}", | ||||
|     "remove": "Forigi", | ||||
|     "yesMarkUpdated": "Jes, marki kiel ĝisdata", | ||||
|     "fdroid": "Oficiala F-Droid", | ||||
|     "appIdOrName": "Identigilo aŭ nomo de la apo", | ||||
|     "appId": "Identigilo de la apo", | ||||
|     "appWithIdOrNameNotFound": "Neniu apo estas trovita kun tiu identigilo aŭ nomo", | ||||
|     "reposHaveMultipleApps": "Deponejoj povas enhavi plurajn apojn", | ||||
|     "fdroidThirdPartyRepo": "Tria deponejo de F-Droid", | ||||
|     "steamMobile": "Telefona Steam", | ||||
|     "steamChat": "Steam Babilejo", | ||||
|     "install": "Instali", | ||||
|     "markInstalled": "Marki kiel instalita", | ||||
|     "update": "Ĝisdatigi", | ||||
|     "markUpdated": "Marki kiel ĝisdata", | ||||
|     "additionalOptions": "Kromaj opcioj", | ||||
|     "disableVersionDetection": "Malaktivigi la versiodetekto", | ||||
|     "noVersionDetectionExplanation": "Tiu opcio devas esti uzata nur por apoj, kie la versiodetekto ne funkcias ĝuste.", | ||||
|     "downloadingX": "Elŝutante {}", | ||||
|     "downloadX": "Elŝuti {}", | ||||
|     "downloadedX": "Elŝutita {}", | ||||
|     "releaseAsset": "Aktiva versio", | ||||
|     "downloadNotifDescription": "Sciigas al la uzanto pri la progreso de apo-elŝuton", | ||||
|     "noAPKFound": "Neniu trovita APK", | ||||
|     "noVersionDetection": "Neniu versiodetekto", | ||||
|     "categorize": "Kategorii", | ||||
|     "categories": "Kategorioj", | ||||
|     "category": "Kategorio", | ||||
|     "noCategory": "Neniu kategorio", | ||||
|     "noCategories": "Neniuj kategorioj", | ||||
|     "deleteCategoriesQuestion": "Forigi la kategoriojn?", | ||||
|     "categoryDeleteWarning": "Ĉiuj apoj el forigitaj kategorioj iĝos nekategoriitaj.", | ||||
|     "addCategory": "Aldoni kategorion", | ||||
|     "label": "Etikedo", | ||||
|     "language": "Lingvo", | ||||
|     "copiedToClipboard": "Kopiita en la tondujo", | ||||
|     "storagePermissionDenied": "Permeso de stoko malkonsentita", | ||||
|     "selectedCategorizeWarning": "Tio substituos ĉiujn difinitajn kategoriojn de la selektitaj apoj.", | ||||
|     "filterAPKsByRegEx": "Filtri APKj per regula esprimo", | ||||
|     "removeFromObtainium": "Forigi el Obtainium", | ||||
|     "uninstallFromDevice": "Malinstali el la disponaĵo", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Funkcias nur por apoj kun malaktiva versiodetekto.", | ||||
|     "releaseDateAsVersion": "Uzi eldondato kiel versioĉeno", | ||||
|     "releaseDateAsVersionExplanation": "Tiu opcio devas esti uzata nur por apoj, por kiu la versiodetekto ne funkcias ĝuste, sed eldondato estas havebla.", | ||||
|     "changes": "Modifoj", | ||||
|     "releaseDate": "Eldondato", | ||||
|     "importFromURLsInFile": "Importi el URLj en dosiero (kiel OPML)", | ||||
|     "versionDetectionExplanation": "Akordigi la versioĉeno laŭ la versio detektita fare de la operaciumo", | ||||
|     "versionDetection": "Versiodetekto", | ||||
|     "standardVersionDetection": "Norma versiodetekto", | ||||
|     "groupByCategory": "Grupigi per kategorio", | ||||
|     "autoApkFilterByArch": "Provi filtri APKj per CPU-arkitekturo se ebla", | ||||
|     "overrideSource": "Substitui la fonton", | ||||
|     "dontShowAgain": "Ne plu montri", | ||||
|     "dontShowTrackOnlyWarnings": "Ne plu montri 'Nur sekvita' avertojn", | ||||
|     "dontShowAPKOriginWarnings": "Ne plu montri avertojn pri APK-origino", | ||||
|     "moveNonInstalledAppsToBottom": "Movi neinstalatajn apojn al malsupro de la langeto Apoj", | ||||
|     "gitlabPATLabel": "GitLab persona atingoĵetono", | ||||
|     "about": "Pri", | ||||
|     "requiresCredentialsInSettings": "{} bezonas kromajn informojn (en Agordoj)", | ||||
|     "checkOnStart": "Serĉi ĝisdatigojn starte", | ||||
|     "tryInferAppIdFromCode": "Provi dedukti la identigilon de la apo el la fontkodo", | ||||
|     "removeOnExternalUninstall": "Aŭtomate forigi ekstere malinstalitajn apojn", | ||||
|     "pickHighestVersionCode": "Aŭtomate selekti la plej ĵusan version el APK-kodo", | ||||
|     "checkUpdateOnDetailPage": "Serĉi ĝisdatigojn dum la malfermo de la detala apopaĝo", | ||||
|     "disablePageTransitions": "Malaktivigi la animaciojn de paĝotransiro", | ||||
|     "reversePageTransitions": "Inversigi la animaciojn de paĝotransiro", | ||||
|     "minStarCount": "Minimuma nombro da steloj", | ||||
|     "addInfoBelow": "Aldonu ĉi tiu informo ĉi-suba.", | ||||
|     "addInfoInSettings": "Aldonu tiu informo en la agordoj.", | ||||
|     "githubSourceNote": "La kvantolimo de GitHub povas esti evitata danke al API-ŝlosilo.", | ||||
|     "sortByLastLinkSegment": "Klasi laŭ la lasta segmento de la ligilo", | ||||
|     "filterReleaseNotesByRegEx": "Filtri versionotojn per regula esprimo", | ||||
|     "customLinkFilterRegex": "Filtro de APK-ligilo agordita per regula esprimo (defaŭlte '.apk$')", | ||||
|     "appsPossiblyUpdated": "Provitaj apoĝisdatigoj", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Sciigas al la uzanto, ke ĝisdatigoj de unu aŭ pluraj apoj eble estas aplikitaj fone", | ||||
|     "xWasPossiblyUpdatedToY": "{} povas esti ĝisdata kiel {}.", | ||||
|     "enableBackgroundUpdates": "Aktivigi fonajn ĝisdatigojn", | ||||
|     "backgroundUpdateReqsExplanation": "Fonaj ĝisdatigoj ne eblas por ĉiuj apoj.", | ||||
|     "backgroundUpdateLimitsExplanation": "La sukceso de fona instalado povas esti nur determinata dum la malfermo de Obtainium.", | ||||
|     "verifyLatestTag": "Inspekti la etikedon 'latest'", | ||||
|     "intermediateLinkRegex": "Filtri por 'pera' vizitota ligilo", | ||||
|     "filterByLinkText": "Filtri ligilojn laŭ ligiloteksto", | ||||
|     "intermediateLinkNotFound": "Netrovebla pera ligilo", | ||||
|     "intermediateLink": "Pera ligilo", | ||||
|     "exemptFromBackgroundUpdates": "Escepti el la fonaj ĝisdatigoj (se aktiva)", | ||||
|     "bgUpdatesOnWiFiOnly": "Malaktivigi fonajn ĝisdatigojn se sen vifio", | ||||
|     "autoSelectHighestVersionCode": "Aŭtomate selekti la plej ĵusan version de la APK-kodo", | ||||
|     "versionExtractionRegEx": "Ekstrakti la versioĉenon per regula esprimo", | ||||
|     "trimVersionString": "Mallongigi la versioĉenon per regula esprimo", | ||||
|     "matchGroupToUseForX": "Konformecogrupo uzota por \"{}\"", | ||||
|     "matchGroupToUse": "Grupo de konformeco uzota por la eltiraĵo de la versio per regula esprimo", | ||||
|     "highlightTouchTargets": "Emfazi malpi evidentajn klavojn", | ||||
|     "pickExportDir": "Selekti la dosierujon de eksporto", | ||||
|     "autoExportOnChanges": "Aŭtomate eksporti dum modifoj", | ||||
|     "includeSettings": "Inkluzivi la agordojn", | ||||
|     "filterVersionsByRegEx": "Filtri versiojn per regula esprimo", | ||||
|     "trySelectingSuggestedVersionCode": "Provi selekti la sugestitan version fare de la APK-kodo", | ||||
|     "dontSortReleasesList": "Konservi la ordo de la API-versio", | ||||
|     "reverseSort": "Inversigi klasifikon", | ||||
|     "takeFirstLink": "Uzi la unuan ligilon", | ||||
|     "skipSort": "Salti la klasifikon", | ||||
|     "debugMenu": "Menu de eraroserĉado", | ||||
|     "bgTaskStarted": "Fona tasko startita - inspektu la protokolojn.", | ||||
|     "runBgCheckNow": "Ruli la serĉo de fonaj ĝisdatigoj nun", | ||||
|     "versionExtractWholePage": "Apliki eltiraĵon de la versioĉeno per regula esprimo al la tuta paĝo", | ||||
|     "installing": "Instalante", | ||||
|     "skipUpdateNotifications": "Salti la sciigojn de ĝisdatigo", | ||||
|     "updatesAvailableNotifChannel": "Haveblaj ĝisdatigoj", | ||||
|     "appsUpdatedNotifChannel": "Ĝisdataj aplikaĵoj", | ||||
|     "appsPossiblyUpdatedNotifChannel": "Provitaj apoĝisdatigoj", | ||||
|     "errorCheckingUpdatesNotifChannel": "Eraro serĉante ĝisdatigojn", | ||||
|     "appsRemovedNotifChannel": "Forigitaj aplikaĵoj", | ||||
|     "downloadingXNotifChannel": "Elŝutante {}", | ||||
|     "completeAppInstallationNotifChannel": "Kompleta apo instalado", | ||||
|     "checkingForUpdatesNotifChannel": "Serĉante ĝisdatigojn", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Nur serĉi la ĝisdatigojn de instalitaj kaj 'nur sekvitaj' apoj", | ||||
|     "supportFixedAPKURL": "Taskiĝo de la fiksaj APK URLj", | ||||
|     "selectX": "Selekti {}", | ||||
|     "parallelDownloads": "Permesi dumajn elŝutojn", | ||||
|     "useShizuku": "Uzi Shizuku aŭ Sui por instali", | ||||
|     "shizukuBinderNotFound": "La servo Shizuku estas netrovebla", | ||||
|     "shizukuOld": "Malnova versio de Shizuku (<11) - bonvolu ĝisdatigi ĝin", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku funkcias en Android < 8.1 kun ADB - bonvolu ĝisdatigi Android aŭ uzu Sui anstataŭe", | ||||
|     "shizukuPretendToBeGooglePlay": "Determini Google Play kiel instalfonto (se Shizuku estas uzata)", | ||||
|     "useSystemFont": "Uzi la sistema tiparo", | ||||
|     "useVersionCodeAsOSVersion": "Uzi versiokodo de la apo kiel versio detektita fare de la operaciumo", | ||||
|     "requestHeader": "Titolo de la informmendo", | ||||
|     "useLatestAssetDateAsReleaseDate": "Uzi la lastan publikigitan elementon kiel eldondato", | ||||
|     "defaultPseudoVersioningMethod": "Defaŭlta metodo de fikcia versio", | ||||
|     "partialAPKHash": "Parta APKa haketo", | ||||
|     "APKLinkHash": "Haketo de la APKa ligilo", | ||||
|     "directAPKLink": "Rekta APKa ligilo", | ||||
|     "pseudoVersionInUse": "Pseŭdoversio estas uzata", | ||||
|     "installed": "Instalita", | ||||
|     "latest": "Lasta versio", | ||||
|     "invertRegEx": "Inversigi la regula esprimo", | ||||
|     "note": "Noto", | ||||
|     "selfHostedNote": "La malvolvanta listo \"{}\" povas esti uzata por aliri al memgastigataj/agordata instancoj el ajna fonto.", | ||||
|     "badDownload": "La APK ne povis esti analizita (neakordigebla aŭ nekompleta elŝuto)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Diskonigi novajn aplikaĵojn kun AppVerifier (se havebla)", | ||||
|     "appVerifierInstructionToast": "Diskonigu kun AppVerifier, poste revenu ĉi tie kiam preta.", | ||||
|     "wiki": "Helpo/Vikio", | ||||
|     "crowdsourcedConfigsLabel": "Komunumaj apo-agordoj (uzu kun singardo)", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Forigi la aplikaĵon?", | ||||
|         "other": "Forigi la aplikaĵojn?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "Tro da informmendoj (limigita kvanto) - reprovu en {} minuto", | ||||
|         "other": "Tro da informmendoj (limigita kvanto) - reprovu en {} minutoj" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "La serĉo de fonaj ĝisdatigoj renkontis al {}, nova programota provo en {} minuto", | ||||
|         "other": "La serĉo de fonaj ĝisdatigoj renkontis al {}, nova programota provo en {} minutoj" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "La serĉo de fonaj ĝisdatigoj trovis {} ĝisdatigon - la uzanto estas sciigota se necesa", | ||||
|         "other": "La serĉo de fonaj ĝisdatigoj trovis {} ĝisdatigojn - la uzanto estas sciigota se necesa" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} Apo", | ||||
|         "other": "{} Apoj" | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URLj" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} Minuto", | ||||
|         "other": "{} Minutoj" | ||||
|     }, | ||||
|     "hour": { | ||||
|         "one": "{} Horo", | ||||
|         "other": "{} Horoj" | ||||
|     }, | ||||
|     "day": { | ||||
|         "one": "{} Tago", | ||||
|         "other": "{} Tagoj" | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "Purigis {n} protokolon (antaŭe = {before}, malantaŭe = {after})", | ||||
|         "other": "Purigis {n} protokolojn (antaŭe = {before}, malantaŭe = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} kaj 1 alia apo havas ĝisdatigojn.", | ||||
|         "other": "{} kaj {} aliaj apoj havas ĝisdatigojn." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} kaj 1 alia apo ĝisdatiĝis.", | ||||
|         "other": "{} kaj {} aliaj apoj ĝisdatiĝis." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Ne eblas ĝisdatigi {} kaj 1 alian apon.", | ||||
|         "other": "Ne eblas ĝisdatigi {} et {} aliajn apojn." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} kaj 1 alia apo eble ĝisdatiĝis.", | ||||
|         "other": "{} kaj {} aliaj apoj eble ĝisdatiĝis." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APKj" | ||||
|     } | ||||
| } | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Disable background updates when not on WiFi", | ||||
|     "autoSelectHighestVersionCode": "Auto-select highest versionCode APK", | ||||
|     "versionExtractionRegEx": "Version String Extraction RegEx", | ||||
|     "trimVersionString": "Trim Version String With RegEx", | ||||
|     "matchGroupToUseForX": "Match Group to Use for \"{}\"", | ||||
|     "matchGroupToUse": "Match Group to Use for Version String Extraction RegEx", | ||||
|     "highlightTouchTargets": "Highlight less obvious touch targets", | ||||
|     "pickExportDir": "Pick Export Directory", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "Share to AppVerifier, then return here when ready.", | ||||
|     "wiki": "Help/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Remove App?", | ||||
|         "other": "Remove Apps?" | ||||
|   | ||||
| @@ -21,15 +21,15 @@ | ||||
|     "continue": "Continuar", | ||||
|     "requiredInBrackets": "(Requerido)", | ||||
|     "dropdownNoOptsError": "ERROR: EL DESPLEGABLE DEBE TENER AL MENOS UNA OPCIÓN", | ||||
|     "colour": "Color", | ||||
|     "colour": "color", | ||||
|     "standard": "Estándar", | ||||
|     "custom": "A medida", | ||||
|     "useMaterialYou": "Use 'Material You'", | ||||
|     "githubStarredRepos": "Repositorios favoritos en GitHub", | ||||
|     "useMaterialYou": "Aplicar 'Material You'", | ||||
|     "githubStarredRepos": "repositorios favoritos en GitHub", | ||||
|     "uname": "Nombre de usuario", | ||||
|     "wrongArgNum": "Número de argumentos provistos inválido", | ||||
|     "xIsTrackOnly": "{} es de 'sólo seguimiento'", | ||||
|     "source": "Origen", | ||||
|     "source": "origen", | ||||
|     "app": "Aplicación", | ||||
|     "appsFromSourceAreTrackOnly": "Las aplicaciones de este origen son solo para seguimiento.", | ||||
|     "youPickedTrackOnly": "Debe seleccionar la opción de 'solo para seguimiento'.", | ||||
| @@ -122,14 +122,14 @@ | ||||
|     "appSortOrder": "Orden de Clasificación", | ||||
|     "ascending": "Ascendente", | ||||
|     "descending": "Descendente", | ||||
|     "bgUpdateCheckInterval": "Comprobación actualizaciones en segundo plano", | ||||
|     "bgUpdateCheckInterval": "Comprobar actualizaciones en segundo plano", | ||||
|     "neverManualOnly": "Nunca, solo manual", | ||||
|     "appearance": "Apariencia", | ||||
|     "showWebInAppView": "Mostrar vista de la web de origen", | ||||
|     "pinUpdates": "Anclar actualizaciones al principio", | ||||
|     "updates": "Actualizaciones", | ||||
|     "sourceSpecific": "Fuente específica", | ||||
|     "appSource": "Obtainium en GitHub", | ||||
|     "appSource": "Filtrar por fuente", | ||||
|     "noLogs": "Ningún registro", | ||||
|     "appLogs": "Registros", | ||||
|     "close": "Cerrar", | ||||
| @@ -220,11 +220,11 @@ | ||||
|     "versionDetectionExplanation": "Conciliar la cadena de versión con la versión detectada desde el sistema operativo", | ||||
|     "versionDetection": "Detección de versiones", | ||||
|     "standardVersionDetection": "Por versión", | ||||
|     "groupByCategory": "Agrupar por categoría", | ||||
|     "groupByCategory": "Agrupar por categorías", | ||||
|     "autoApkFilterByArch": "Filtrar APK por arquitectura del procesador (si es posible)", | ||||
|     "overrideSource": "Forzar desde la fuente", | ||||
|     "dontShowAgain": "No mostrar de nuevo", | ||||
|     "dontShowTrackOnlyWarnings": "No mostrar avisos sobre apps 'solo para seguimiento", | ||||
|     "dontShowTrackOnlyWarnings": "No mostrar avisos sobre apps 'solo para seguimiento'", | ||||
|     "dontShowAPKOriginWarnings": "No mostrar avisos sobre las fuentes de las APKs", | ||||
|     "moveNonInstalledAppsToBottom": "Mover apps no instaladas al final", | ||||
|     "gitlabPATLabel": "Token de acceso personal a GitLab", | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Deshabilitar las actualizaciones en segundo plano sin WiFi", | ||||
|     "autoSelectHighestVersionCode": "Auto selección del paquete APK con versión más reciente", | ||||
|     "versionExtractionRegEx": "Versión de extracción regex", | ||||
|     "trimVersionString": "Recortar cadena de versión con RegEx", | ||||
|     "matchGroupToUseForX": "Grupo de coincidencia a utilizar para \"{}\"", | ||||
|     "matchGroupToUse": "Grupo a usar para versión de extracción regex", | ||||
|     "highlightTouchTargets": "Resaltar objetivos menos obvios", | ||||
|     "pickExportDir": "Directorio para exportar", | ||||
| @@ -288,7 +290,7 @@ | ||||
|     "supportFixedAPKURL": "Soporte para URLs fijas de APK", | ||||
|     "selectX": "Elija {}", | ||||
|     "parallelDownloads": "Permitir descargas paralelas", | ||||
|     "useShizuku": "Use Shizuku o Sui para instalar", | ||||
|     "useShizuku": "Usar Shizuku o Sui para instalar", | ||||
|     "shizukuBinderNotFound": "Shizuku no funciona", | ||||
|     "shizukuOld": "Versión antigua de Shizuku (<11) - actualícela", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku corriendo en Android < 8.1 con ADB - actualiza Android o usa Sui en su lugar", | ||||
| @@ -308,10 +310,11 @@ | ||||
|     "note": "Nota", | ||||
|     "selfHostedNote": "El desplegable «{}» puede usarse para acceder a instancias autoalojadas/personalizadas de cualquier fuente.", | ||||
|     "badDownload": "No se ha podido analizar el APK (incompatible o descarga parcial)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Compartir nuevas aplicaciones con AppVerifier (si está disponible)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Compartir aplicaciones nuevas con AppVerifier (si está disponible)", | ||||
|     "appVerifierInstructionToast": "Comparta con AppVerifier y vuelva aquí cuando esté listo.", | ||||
|     "wiki": "Ayuda/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (uso bajo su propia responsabilidad)", | ||||
|     "allowInsecure": "Permitir peticiones HTTP inseguras", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "¿Eliminar aplicación?", | ||||
|         "other": "¿Eliminar aplicaciones?" | ||||
| @@ -333,8 +336,8 @@ | ||||
|         "other": "{} Aplicaciones" | ||||
|     }, | ||||
|     "url": { | ||||
|         "uno": "{} URL", | ||||
|         "otro": "{} URL" | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URL" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} minuto", | ||||
|   | ||||
| @@ -113,7 +113,7 @@ | ||||
|     "dark": "تاریک", | ||||
|     "light": "روشن", | ||||
|     "followSystem": "هماهنگ با سیستم", | ||||
|     "followSystemThemeExplanation": "Following system theme is possible only by using third-party applications", | ||||
|     "followSystemThemeExplanation": "دنبال کردن تم سیستم فقط با استفاده از برنامه های شخص ثالث امکان پذیر است", | ||||
|     "useBlackTheme": "استفاده از تم تیره سیاه خالص", | ||||
|     "appSortBy": "مرتب سازی برنامه بر اساس", | ||||
|     "authorName": "سازنده/اسم", | ||||
| @@ -147,10 +147,10 @@ | ||||
|     "noNewUpdates": "به روز رسانی جدیدی وجود ندارد.", | ||||
|     "xHasAnUpdate": "{} یک به روز رسانی دارد.", | ||||
|     "appsUpdated": "برنامه ها به روز شدند", | ||||
|     "appsNotUpdated": "Failed to update applications", | ||||
|     "appsNotUpdated": "به روز رسانی برنامه ها ناموفق بود", | ||||
|     "appsUpdatedNotifDescription": "به کاربر اطلاع می دهد که به روز رسانی یک یا چند برنامه در پس زمینه اعمال شده است", | ||||
|     "xWasUpdatedToY": "{} به {} به روز شد.", | ||||
|     "xWasNotUpdatedToY": "Failed to update {} to {}.", | ||||
|     "xWasNotUpdatedToY": "به روز رسانی {} به {} انجام نشد.", | ||||
|     "errorCheckingUpdates": "خطا در بررسی بهروزرسانیها", | ||||
|     "errorCheckingUpdatesNotifDescription": "اعلانی که وقتی بررسی بهروزرسانی پسزمینه ناموفق است نشان میدهد", | ||||
|     "appsRemoved": "برنامه ها حذف شدند", | ||||
| @@ -189,9 +189,9 @@ | ||||
|     "disableVersionDetection": "غیرفعال کردن تشخیص نسخه", | ||||
|     "noVersionDetectionExplanation": "این گزینه فقط باید برای برنامه هایی استفاده شود که تشخیص نسخه به درستی کار نمی کند.", | ||||
|     "downloadingX": "در حال دانلود {}", | ||||
|     "downloadX": "Download {}", | ||||
|     "downloadedX": "Downloaded {}", | ||||
|     "releaseAsset": "Release Asset", | ||||
|     "downloadX": "دانلود {}", | ||||
|     "downloadedX": "دانلود شده {}", | ||||
|     "releaseAsset": "انتشار دارایی", | ||||
|     "downloadNotifDescription": "کاربر را از پیشرفت دانلود یک برنامه مطلع می کند", | ||||
|     "noAPKFound": "APK پیدا نشد فایل", | ||||
|     "noVersionDetection": "بدون تشخیص نسخه", | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "بهروزرسانیهای پسزمینه را در صورت عدم اتصال به WiFi غیرفعال کنید", | ||||
|     "autoSelectHighestVersionCode": "انتخاب خودکار بالاترین نسخه کد APK", | ||||
|     "versionExtractionRegEx": "نسخه استخراج RegEx", | ||||
|     "trimVersionString": "Trim Version String With RegEx", | ||||
|     "matchGroupToUseForX": "Match Group to Use for \"{}\"", | ||||
|     "matchGroupToUse": "گروه مورد استفاده را مطابقت دهید", | ||||
|     "highlightTouchTargets": "اهداف لمسی کمتر واضح را برجسته کنید", | ||||
|     "pickExportDir": "فهرست برون ریزی را انتخاب کنید", | ||||
| @@ -305,13 +307,14 @@ | ||||
|     "installed": "نصب شده است", | ||||
|     "latest": "آخرین", | ||||
|     "invertRegEx": "معکوس کردن عبارت منظم", | ||||
|     "note": "Note", | ||||
|     "selfHostedNote": "The \"{}\" dropdown can be used to reach self-hosted/custom instances of any source.", | ||||
|     "badDownload": "The APK could not be parsed (incompatible or partial download)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Share new Apps with AppVerifier (if available)", | ||||
|     "appVerifierInstructionToast": "Share to AppVerifier, then return here when ready.", | ||||
|     "wiki": "Help/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", | ||||
|     "note": "یادداشت", | ||||
|     "selfHostedNote": "از منوی کرکره ای \"{}\" می توان برای دسترسی به نمونه های خود میزبانی/سفارشی از هر منبعی استفاده کرد.", | ||||
|     "badDownload": "APK قابل تجزیه نیست (دانلود ناسازگار یا جزئی)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "اشتراکگذاری برنامههای جدید با AppVerifier (در صورت وجود)", | ||||
|     "appVerifierInstructionToast": "در AppVerifier به اشتراک بگذارید، سپس پس از آماده شدن به اینجا برگردید.", | ||||
|     "wiki": "راهنما/ویکی", | ||||
|     "crowdsourcedConfigsLabel": "تنظیمات برنامه Crowdsourced (با مسئولیت خود استفاده کنید)", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "برنامه حذف شود؟", | ||||
|         "other": "برنامه ها حذف شوند؟" | ||||
| @@ -361,8 +364,8 @@ | ||||
|         "other": "{} و {} برنامه دیگر به روز شدند." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Failed to update {} and 1 more app.", | ||||
|         "other": "Failed to update {} and {} more apps." | ||||
|         "one": "{} و 1 برنامه دیگر به روز نشد.", | ||||
|         "other": "{} و {} برنامه دیگر به روز نشد." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} و 1 برنامه دیگر ممکن است به روز شده باشند.", | ||||
|   | ||||
| @@ -1,34 +1,34 @@ | ||||
| { | ||||
|     "invalidURLForSource": "URL d'application {} invalide", | ||||
|     "noReleaseFound": "Impossible de trouver une version adaptée", | ||||
|     "noVersionFound": "Impossible de déterminer la variante de la version", | ||||
|     "invalidURLForSource": "URL de l'application {} invalide", | ||||
|     "noReleaseFound": "Impossible de trouver une publication correspondante", | ||||
|     "noVersionFound": "Impossible de déterminer la version de la publication", | ||||
|     "urlMatchesNoSource": "L'URL ne correspond pas à une source connue", | ||||
|     "cantInstallOlderVersion": "Impossible d'installer une ancienne version d'une application", | ||||
|     "appIdMismatch": "L'ID de paquet téléchargé ne correspond pas à l'ID de l'application existante", | ||||
|     "functionNotImplemented": "Cette classe n'a pas implémenté cette fonction", | ||||
|     "cantInstallOlderVersion": "Impossible d'installer une ancienne version de l'application", | ||||
|     "appIdMismatch": "L'ID du paquet téléchargé ne correspond pas à l'ID de l'application existante", | ||||
|     "functionNotImplemented": "Cette classe n'implémente pas cette fonction", | ||||
|     "placeholder": "Espace réservé", | ||||
|     "someErrors": "Des erreurs se sont produites", | ||||
|     "someErrors": "Des erreurs sont survenues", | ||||
|     "unexpectedError": "Erreur inattendue", | ||||
|     "ok": "D'accord", | ||||
|     "ok": "Ok", | ||||
|     "and": "et", | ||||
|     "githubPATLabel": "Jeton d'Accès Personnel GitHub (Augmente la limite de débit)", | ||||
|     "includePrereleases": "Inclure les avant-premières", | ||||
|     "githubPATLabel": "Jeton d'accès personnel GitHub (augmente la limite de débit)", | ||||
|     "includePrereleases": "Inclure les versions préliminaires", | ||||
|     "fallbackToOlderReleases": "Retour aux anciennes versions", | ||||
|     "filterReleaseTitlesByRegEx": "Filtrer les titres de version par expression régulière", | ||||
|     "invalidRegEx": "Expression régulière invalide", | ||||
|     "noDescription": "Pas de description", | ||||
|     "noDescription": "Aucune description", | ||||
|     "cancel": "Annuler", | ||||
|     "continue": "Continuer", | ||||
|     "requiredInBrackets": "(Requis)", | ||||
|     "dropdownNoOptsError": "ERREUR : LE DÉROULEMENT DOIT AVOIR AU MOINS UNE OPT", | ||||
|     "dropdownNoOptsError": "ERREUR: LA LISTE DÉROULANTE DOIT AVOIR AU MOINS UNE OPTION", | ||||
|     "colour": "Couleur", | ||||
|     "standard": "Standard", | ||||
|     "custom": "Sur mesure", | ||||
|     "useMaterialYou": "Utiliser le matériel que vous", | ||||
|     "custom": "Personnalisé", | ||||
|     "useMaterialYou": "Utiliser Material You", | ||||
|     "githubStarredRepos": "Dépôts étoilés GitHub", | ||||
|     "uname": "Nom d'utilisateur", | ||||
|     "wrongArgNum": "Mauvais nombre d'arguments fournis", | ||||
|     "xIsTrackOnly": "{} est en 'Suivi uniquement'", | ||||
|     "wrongArgNum": "Nombre incorrect des arguments fournis", | ||||
|     "xIsTrackOnly": "{} en Suivi uniquement", | ||||
|     "source": "Source", | ||||
|     "app": "Application", | ||||
|     "appsFromSourceAreTrackOnly": "Les applications de cette source sont en 'Suivi uniquement'.", | ||||
| @@ -36,51 +36,51 @@ | ||||
|     "trackOnlyAppDescription": "L'application sera suivie pour les mises à jour, mais Obtainium ne pourra pas la télécharger ou l'installer.", | ||||
|     "cancelled": "Annulé", | ||||
|     "appAlreadyAdded": "Application déjà ajoutée", | ||||
|     "alreadyUpToDateQuestion": "Application déjà à jour ?", | ||||
|     "addApp": "Ajouter une application", | ||||
|     "appSourceURL": "URL de la source de l'application", | ||||
|     "alreadyUpToDateQuestion": "L'application est à jour?", | ||||
|     "addApp": "Ajouter Appli", | ||||
|     "appSourceURL": "URL source de l'application", | ||||
|     "error": "Erreur", | ||||
|     "add": "Ajouter", | ||||
|     "searchSomeSourcesLabel": "Rechercher (certaines sources uniquement)", | ||||
|     "search": "Rechercher", | ||||
|     "additionalOptsFor": "Options supplémentaires pour {}", | ||||
|     "supportedSources": "Sources prises en charge ", | ||||
|     "supportedSources": "Sources prises en charge", | ||||
|     "trackOnlyInBrackets": "(Suivi uniquement)", | ||||
|     "searchableInBrackets": "(Intérrogeable)", | ||||
|     "searchableInBrackets": "(Interrogeable)", | ||||
|     "appsString": "Applications", | ||||
|     "noApps": "Aucune application", | ||||
|     "noAppsForFilter": "Aucune application pour le filtre", | ||||
|     "noAppsForFilter": "Aucune application à filtrer", | ||||
|     "byX": "Par {}", | ||||
|     "percentProgress": "Progrès: {}%", | ||||
|     "percentProgress": "Progression : {}%", | ||||
|     "pleaseWait": "Veuillez patienter", | ||||
|     "updateAvailable": "Mise à jour disponible", | ||||
|     "notInstalled": "Non installé", | ||||
|     "pseudoVersion": "pseudo-version", | ||||
|     "pseudoVersion": "Version fictive", | ||||
|     "selectAll": "Tout sélectionner", | ||||
|     "deselectX": "Déselectionner {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} sera supprimé d'Obtainium mais restera installé sur l'appareil.", | ||||
|     "removeSelectedAppsQuestion": "Supprimer les applications sélectionnées ?", | ||||
|     "removeSelectedApps": "Supprimer les applications sélectionnées", | ||||
|     "updateX": "Mise à jour {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} sera supprimée d'Obtainium mais restera installée sur l'appareil.", | ||||
|     "removeSelectedAppsQuestion": "Supprimer les applications sélectionnées ?", | ||||
|     "removeSelectedApps": "Les applications sélectionnées ont été supprimées", | ||||
|     "updateX": "Mettre à jour {}", | ||||
|     "installX": "Installer {}", | ||||
|     "markXTrackOnlyAsUpdated": "Marquer {}\n(Suivi uniquement)\n comme mis à jour", | ||||
|     "markXTrackOnlyAsUpdated": "Marquer {}\n(Suivi uniquement)\ncomme étant à jour", | ||||
|     "changeX": "Changer {}", | ||||
|     "installUpdateApps": "Installer/Mettre à jour les applications", | ||||
|     "installUpdateSelectedApps": "Installer/Mettre à jour les applications sélectionnées", | ||||
|     "markXSelectedAppsAsUpdated": "Marquer {} les applications sélectionnées comme étant à jour ?", | ||||
|     "markXSelectedAppsAsUpdated": "Marquer les {} applications sélectionnées comme étant à jour ?", | ||||
|     "no": "Non", | ||||
|     "yes": "Oui", | ||||
|     "markSelectedAppsUpdated": "Marquer les applications sélectionnées comme étant à jour", | ||||
|     "markSelectedAppsUpdated": "Marquer les application sélectionnées comme étant à jour", | ||||
|     "pinToTop": "Épingler en haut", | ||||
|     "unpinFromTop": "Désépingler du haut", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Réinitialiser le statu d'installation des applications sélectionnées ?", | ||||
|     "installStatusOfXWillBeResetExplanation": "Le statu d'installation de toutes les applications sélectionnées sera réinitialisé.\n\nCela peut aider lorsque la version de l'application affichée dans Obtainium est incorrecte en raison d'échecs de mises à jour ou d'autres problèmes.", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Réinitialiser l'état d'installation des applications sélectionnées ?", | ||||
|     "installStatusOfXWillBeResetExplanation": "L'état d'installation de toutes les applications sélectionnées sera réinitialisé.\n\nCela peut être utile lorsque la version de l'application affichée dans Obtainium est incorrecte en raison de l'échec des mises à jour ou d'autres problèmes.", | ||||
|     "customLinkMessage": "Ces liens fonctionnent sur les appareils sur lesquels Obtainium est installé", | ||||
|     "shareAppConfigLinks": "Partager la configuration de l'application sous forme de lien HTML", | ||||
|     "shareSelectedAppURLs": "Partager les URL d'application sélectionnées", | ||||
|     "resetInstallStatus": "Réinitialiser le statu d'installation", | ||||
|     "shareSelectedAppURLs": "Partager les URL des applications sélectionnées", | ||||
|     "resetInstallStatus": "L'état d'installation des applications a été réinitialisé", | ||||
|     "more": "Plus", | ||||
|     "removeOutdatedFilter": "Supprimer le filtre d'application obsolète", | ||||
|     "removeOutdatedFilter": "Supprimer le filtre des applications obsolètes", | ||||
|     "showOutdatedOnly": "Afficher uniquement les applications obsolètes", | ||||
|     "filter": "Filtre", | ||||
|     "filterApps": "Filtrer les applications", | ||||
| @@ -91,42 +91,42 @@ | ||||
|     "importExport": "Importer/Exporter", | ||||
|     "settings": "Paramètres", | ||||
|     "exportedTo": "Exporté vers {}", | ||||
|     "obtainiumExport": "Exporter d'Obtainium", | ||||
|     "obtainiumExport": "Exporter Obtainium", | ||||
|     "invalidInput": "Entrée invalide", | ||||
|     "importedX": "Importé {}", | ||||
|     "obtainiumImport": "Importer d'Obtainium", | ||||
|     "importFromURLList": "Importer à partir de la liste d'URL", | ||||
|     "searchQuery": "Requête", | ||||
|     "appURLList": "Liste d'URL d'application", | ||||
|     "line": "Queue", | ||||
|     "obtainiumImport": "Importer sur Obtainium", | ||||
|     "importFromURLList": "Importer depuis une liste d'URL", | ||||
|     "searchQuery": "Requête de recherche", | ||||
|     "appURLList": "Liste d'URL de l'application", | ||||
|     "line": "Ligne", | ||||
|     "searchX": "Rechercher {}", | ||||
|     "noResults": "Aucun résultat trouvé", | ||||
|     "importX": "Importer {}", | ||||
|     "importedAppsIdDisclaimer": "Les applications importées peuvent s'afficher à tort comme \"Non installées\".\nPour résoudre ce problème, réinstallez-les via Obtainium.\nCela ne devrait pas affecter les données de l'application.\n\nN'affecte que les URL et les méthodes d'importation tierces.", | ||||
|     "importErrors": "Erreurs d'importation", | ||||
|     "importedXOfYApps": "{} sur {} applications importées.", | ||||
|     "followingURLsHadErrors": "Les URL suivantes comportaient des erreurs :", | ||||
|     "selectURL": "Sélectionnez l'URL", | ||||
|     "selectURLs": "Sélectionnez les URLs", | ||||
|     "pick": "Prendre", | ||||
|     "noResults": "Aucun résultat", | ||||
|     "importX": "Importation de {}", | ||||
|     "importedAppsIdDisclaimer": "Les applications importées peuvent s'afficher de manière incorrecte comme étant \"Non installées\".\nPour résoudre ce problème, réinstallez-les via Obtainium.\nCela n'affectera pas les données des applications.\n\nN'affecte que les méthodes d'importation d'URL et par des tiers.", | ||||
|     "importErrors": "Erreurs lors de l'importation", | ||||
|     "importedXOfYApps": "{} applications sur {} ont été importés.", | ||||
|     "followingURLsHadErrors": "Les URL suivants comportent des erreurs :", | ||||
|     "selectURL": "Sélectionner l'URL", | ||||
|     "selectURLs": "Sélectionner les URL", | ||||
|     "pick": "Choisir", | ||||
|     "theme": "Thème", | ||||
|     "dark": "Sombre", | ||||
|     "light": "Clair", | ||||
|     "followSystem": "Suivre le système", | ||||
|     "followSystemThemeExplanation": "Il n'est possible de suivre le thème du système qu'en utilisant des applications tierces.", | ||||
|     "useBlackTheme": "Utilisez le thème noir pur", | ||||
|     "appSortBy": "Applications triées par", | ||||
|     "followSystem": "Correspondre au système", | ||||
|     "followSystemThemeExplanation": "Correspondre au thème du système est possible en utilisant des applications tierces.", | ||||
|     "useBlackTheme": "Utiliser un thème Noir", | ||||
|     "appSortBy": "Trier les applications par", | ||||
|     "authorName": "Auteur/Nom", | ||||
|     "nameAuthor": "Nom/Auteur", | ||||
|     "asAdded": "Comme ajouté", | ||||
|     "asAdded": "Date d'ajout", | ||||
|     "appSortOrder": "Ordre de tri des applications", | ||||
|     "ascending": "Ascendant", | ||||
|     "descending": "Descendant", | ||||
|     "bgUpdateCheckInterval": "Intervalle de vérification des mises à jour en arrière-plan", | ||||
|     "neverManualOnly": "Jamais - Manuel uniquement", | ||||
|     "bgUpdateCheckInterval": "Intervalle de recherche de mises à jour en arrière-plan", | ||||
|     "neverManualOnly": "Jamais - Manuellement uniquement", | ||||
|     "appearance": "Apparence", | ||||
|     "showWebInAppView": "Afficher la page Web source dans la vue de l'application", | ||||
|     "pinUpdates": "Épingler les mises à jour dans la vue Top des applications", | ||||
|     "showWebInAppView": "Afficher la page Web source dans l'onglet 'Applications'", | ||||
|     "pinUpdates": "Épingler les mises à jour en haut de l'onglet 'Applications'", | ||||
|     "updates": "Mises à jour", | ||||
|     "sourceSpecific": "Spécifique à la source", | ||||
|     "appSource": "Source de l'application", | ||||
| @@ -135,32 +135,32 @@ | ||||
|     "close": "Fermer", | ||||
|     "share": "Partager", | ||||
|     "appNotFound": "Application introuvable", | ||||
|     "obtainiumExportHyphenatedLowercase": "exportation d'Obtainium", | ||||
|     "pickAnAPK": "Choisissez un APK", | ||||
|     "appHasMoreThanOnePackage": "{} a plus d'un paquet :", | ||||
|     "obtainiumExportHyphenatedLowercase": "export-obtainium", | ||||
|     "pickAnAPK": "Selectionner un APK", | ||||
|     "appHasMoreThanOnePackage": "{} a plus d'un paquet:", | ||||
|     "deviceSupportsXArch": "Votre appareil prend en charge l'architecture CPU {}.", | ||||
|     "deviceSupportsFollowingArchs": "Votre appareil prend en charge les architectures CPU suivantes :", | ||||
|     "deviceSupportsFollowingArchs": "Votre appareil prend en charge les architectures CPU suivants: ", | ||||
|     "warning": "Avertissement", | ||||
|     "sourceIsXButPackageFromYPrompt": "La source de l'application est '{}' mais la version du paquet provient de '{}'. Continuer?", | ||||
|     "sourceIsXButPackageFromYPrompt": "La source de l'application est '{}' mais le paquet de mise à jour provient de '{}'. Continuer ?", | ||||
|     "updatesAvailable": "Mises à jour disponibles", | ||||
|     "updatesAvailableNotifDescription": "Avertit l'utilisateur que des mises à jour sont disponibles pour une ou plusieurs applications suivies par Obtainium", | ||||
|     "updatesAvailableNotifDescription": "Notifie à l'utilisateur que des mises à jour sont disponibles pour une ou plusieurs applications suivies par Obtainium.", | ||||
|     "noNewUpdates": "Aucune nouvelle mise à jour.", | ||||
|     "xHasAnUpdate": "{} a une mise à jour.", | ||||
|     "appsUpdated": "Applications mises à jour", | ||||
|     "appsNotUpdated": "Échec de la mise à jour des applications", | ||||
|     "appsUpdatedNotifDescription": "Avertit l'utilisateur que les mises à jour d'une ou plusieurs applications ont été appliquées en arrière-plan", | ||||
|     "xWasUpdatedToY": "{} a été mis à jour pour {}.", | ||||
|     "appsUpdatedNotifDescription": "Notifie à l'utilisateur que des mises à jour d'une ou plusieurs applications ont été installées en arrière-plan.", | ||||
|     "xWasUpdatedToY": "{} a été mis à jour en {}.", | ||||
|     "xWasNotUpdatedToY": "Échec de la mise à jour de {} vers {}.", | ||||
|     "errorCheckingUpdates": "Erreur lors de la vérification des mises à jour", | ||||
|     "errorCheckingUpdatesNotifDescription": "Une notification qui s'affiche lorsque la vérification de la mise à jour en arrière-plan échoue", | ||||
|     "errorCheckingUpdates": "Erreur lors de la recherche de mises à jour", | ||||
|     "errorCheckingUpdatesNotifDescription": "Notifie l'utilisateur lorsque la recherche de mises à jour en arrière-plan échoue.", | ||||
|     "appsRemoved": "Applications supprimées", | ||||
|     "appsRemovedNotifDescription": "Avertit l'utilisateur qu'une ou plusieurs applications ont été supprimées en raison d'erreurs lors de leur chargement", | ||||
|     "xWasRemovedDueToErrorY": "{} a été supprimé en raison de cette erreur : {}", | ||||
|     "appsRemovedNotifDescription": "Notifie à l'utilisateur qu'une ou plusieurs applications ont été supprimées en raison d'erreurs lors de leur chargement.", | ||||
|     "xWasRemovedDueToErrorY": "{} a été supprimée en raison de cette erreur : {}", | ||||
|     "completeAppInstallation": "Installation complète de l'application", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium doit être ouvert pour installer des applications", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium doit être ouvert pour installer les applications", | ||||
|     "completeAppInstallationNotifDescription": "Demande à l'utilisateur de retourner sur Obtainium pour terminer l'installation d'une application", | ||||
|     "checkingForUpdates": "Vérification des mises à jour", | ||||
|     "checkingForUpdatesNotifDescription": "Notification transitoire qui apparaît lors de la recherche de mises à jour", | ||||
|     "checkingForUpdates": "Recherche de mises à jour", | ||||
|     "checkingForUpdatesNotifDescription": "Notification temporaire qui apparaît lors de la recherche de mises à jour", | ||||
|     "pleaseAllowInstallPerm": "Veuillez autoriser Obtainium à installer des applications", | ||||
|     "trackOnly": "Suivi uniquement", | ||||
|     "errorWithHttpStatusCode": "Erreur {}", | ||||
| @@ -168,165 +168,168 @@ | ||||
|     "unknown": "Inconnu", | ||||
|     "none": "Aucun", | ||||
|     "never": "Jamais", | ||||
|     "latestVersionX": "Dernière version: {}", | ||||
|     "installedVersionX": "Version installée : {}", | ||||
|     "lastUpdateCheckX": "Vérification de la dernière mise à jour : {}", | ||||
|     "remove": "Retirer", | ||||
|     "yesMarkUpdated": "Oui, marquer comme mis à jour", | ||||
|     "latestVersionX": "Dernière version : {}", | ||||
|     "installedVersionX": "Version installée : {}", | ||||
|     "lastUpdateCheckX": "Dernière recherche de mises à jour : {}", | ||||
|     "remove": "Supprimer", | ||||
|     "yesMarkUpdated": "Oui, marquer comme étant à jour", | ||||
|     "fdroid": "F-Droid Officiel", | ||||
|     "appIdOrName": "ID ou nom de l'application", | ||||
|     "appId": "ID de l'application", | ||||
|     "appWithIdOrNameNotFound": "Aucune application n'a été trouvée avec cet identifiant ou ce nom", | ||||
|     "reposHaveMultipleApps": "Les dépôts peuvent contenir plusieurs applications", | ||||
|     "fdroidThirdPartyRepo": "Dépôt tiers F-Droid", | ||||
|     "steamMobile": "Vapeur Mobile", | ||||
|     "steamChat": "Chat sur Steam", | ||||
|     "steamMobile": "Application mobile Steam", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Installer", | ||||
|     "markInstalled": "Marquer installée", | ||||
|     "markInstalled": "Marquer comme étant installé", | ||||
|     "update": "Mettre à jour", | ||||
|     "markUpdated": "Marquer à jour", | ||||
|     "additionalOptions": "Options additionnelles", | ||||
|     "disableVersionDetection": "Désactiver la détection de version", | ||||
|     "noVersionDetectionExplanation": "Cette option ne doit être utilisée que pour les applications où la détection de version ne fonctionne pas correctement.", | ||||
|     "markUpdated": "Marquer comme étant à jour", | ||||
|     "additionalOptions": "Options supplémentaires", | ||||
|     "disableVersionDetection": "Désactiver la détection de la version", | ||||
|     "noVersionDetectionExplanation": "Cette option ne doit être utilisée que pour les applications où la détection de la version ne fonctionne pas correctement.", | ||||
|     "downloadingX": "Téléchargement {}", | ||||
|     "downloadX": "Télécharger {}", | ||||
|     "downloadedX": "Téléchargé {}", | ||||
|     "releaseAsset": "Actif libéré", | ||||
|     "downloadNotifDescription": "Avertit l'utilisateur de la progression du téléchargement d'une application", | ||||
|     "releaseAsset": "Version active", | ||||
|     "downloadNotifDescription": "Notifie l'utilisateur sur l'avancement du téléchargement d'une application", | ||||
|     "noAPKFound": "Aucun APK trouvé", | ||||
|     "noVersionDetection": "Aucune de détection de version", | ||||
|     "noVersionDetection": "Aucune version trouvée", | ||||
|     "categorize": "Catégoriser", | ||||
|     "categories": "Catégories", | ||||
|     "category": "Catégorie", | ||||
|     "noCategory": "Aucune catégorie", | ||||
|     "noCategories": "Aucune catégorie", | ||||
|     "deleteCategoriesQuestion": "Supprimer les catégories ?", | ||||
|     "categoryDeleteWarning": "Toutes les applications dans les catégories supprimées seront définies sur non catégorisées.", | ||||
|     "noCategories": "Aucune catégories", | ||||
|     "deleteCategoriesQuestion": "Supprimer les catégories?", | ||||
|     "categoryDeleteWarning": "Toutes les applications des catégories supprimées seront définies comme non catégorisées .", | ||||
|     "addCategory": "Ajouter une catégorie", | ||||
|     "label": "Étiquette", | ||||
|     "label": "Nom", | ||||
|     "language": "Langue", | ||||
|     "copiedToClipboard": "Copié dans le presse-papier", | ||||
|     "storagePermissionDenied": "Autorisation de stockage refusée", | ||||
|     "selectedCategorizeWarning": "Cela remplacera toutes les catégorie définies pour les applications sélectionnées.", | ||||
|     "storagePermissionDenied": "Permission de stockage refusée", | ||||
|     "selectedCategorizeWarning": "Cela va remplacer toutes les catégories définies des applications sélectionnées.", | ||||
|     "filterAPKsByRegEx": "Filtrer les APK par expression régulière", | ||||
|     "removeFromObtainium": "Supprimer d'Obtainium", | ||||
|     "uninstallFromDevice": "Désinstaller de l'appareil", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Fonctionne uniquement pour les applications avec la détection de version désactivée.", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Ne fonctionne que pour les applications dont la détection de la version est désactivée.", | ||||
|     "releaseDateAsVersion": "Utiliser la date de sortie comme version", | ||||
|     "releaseDateAsVersionExplanation": "Cette option ne doit être utilisée que pour les applications où la détection de version ne fonctionne pas correctement, mais dont une date de sortie est disponible.", | ||||
|     "changes": "Changements", | ||||
|     "releaseDateAsVersionExplanation": "Cette option ne doit être utilisée que pour les applications pour lesquelles la détection de la version ne fonctionne pas correctement, mais dont une date de sortie est disponible.", | ||||
|     "changes": "Modifications", | ||||
|     "releaseDate": "Date de sortie", | ||||
|     "importFromURLsInFile": "Importer à partir d'URL dans un fichier (comme OPML)", | ||||
|     "versionDetectionExplanation": "Réconcilier la chaîne de version avec la version détectée à partir du système d'exploitation", | ||||
|     "versionDetection": "Détection des versions", | ||||
|     "standardVersionDetection": "Détection de version standard", | ||||
|     "groupByCategory": "Regrouper par catégorie", | ||||
|     "autoApkFilterByArch": "Si possible, essayez de filtrer les APK par architecture CPU", | ||||
|     "importFromURLsInFile": "Importer à partir des URLs d'un fichier (comme OPML)", | ||||
|     "versionDetectionExplanation": "Reporter la chaîne de version selon la version détectée par le système d'exploitation", | ||||
|     "versionDetection": "Détection de la version", | ||||
|     "standardVersionDetection": "Détection de la version standard", | ||||
|     "groupByCategory": "Grouper par catégorie", | ||||
|     "autoApkFilterByArch": "Essayer de filtrer les APKs par architecture CPU si possible", | ||||
|     "overrideSource": "Remplacer la source", | ||||
|     "dontShowAgain": "Ne plus montrer", | ||||
|     "dontShowTrackOnlyWarnings": "Ne pas afficher l'avertissement 'Track-Only'", | ||||
|     "dontShowAPKOriginWarnings": "Ne pas afficher les avertissements sur l'origine de l'APK", | ||||
|     "dontShowAgain": "Ne plus afficher", | ||||
|     "dontShowTrackOnlyWarnings": "Ne plus afficher les erreurs 'Suivi uniquement'", | ||||
|     "dontShowAPKOriginWarnings": "Ne plus afficher les erreurs sur l'origine de l'APK", | ||||
|     "moveNonInstalledAppsToBottom": "Déplacer les applications non installées vers le bas de la vue Applications", | ||||
|     "gitlabPATLabel": "Jeton d'accès personnel GitLab", | ||||
|     "about": "À propos de", | ||||
|     "requiresCredentialsInSettings": "{}: Cela nécessite des identifiants supplémentaires (dans Paramètres)", | ||||
|     "checkOnStart": "Vérifier les mises à jour au démarrage", | ||||
|     "tryInferAppIdFromCode": "Essayez de déduire l'ID de l'application à partir du code source", | ||||
|     "about": "À propos", | ||||
|     "requiresCredentialsInSettings": "{} a besoin d'un complément d'information (dans les Paramètres)", | ||||
|     "checkOnStart": "Rechercher les mises à jour au démarrage", | ||||
|     "tryInferAppIdFromCode": "Essayer de déduire l'identifiant de l'application à partir du code source", | ||||
|     "removeOnExternalUninstall": "Supprimer automatiquement les applications désinstallées en externe", | ||||
|     "pickHighestVersionCode": "Sélectionner automatiquement le code de version de l'APK la plus élevée", | ||||
|     "checkUpdateOnDetailPage": "Vérifier les mises à jour lors de l'ouverture de la page détaillée d'une application", | ||||
|     "pickHighestVersionCode": "Sélectionner automatiquement la version la plus récente du code APK", | ||||
|     "checkUpdateOnDetailPage": "Rechercher les mises à jour lors de l'ouverture de la page détaillée d'une application", | ||||
|     "disablePageTransitions": "Désactiver les animations de transition de page", | ||||
|     "reversePageTransitions": "Inverser les animations de transition de page", | ||||
|     "minStarCount": "Nombre minimum d'étoiles", | ||||
|     "addInfoBelow": "Ajoutez ces informations ci-dessous.", | ||||
|     "addInfoInSettings": "Ajoutez ces informations dans les paramètres.", | ||||
|     "githubSourceNote": "La limite de débit GitHub peut être évitée à l'aide d'une clé API.", | ||||
|     "sortByLastLinkSegment": "Trier uniquement sur le dernier segment du lien", | ||||
|     "addInfoBelow": "Ajoutez cette information ci-dessous.", | ||||
|     "addInfoInSettings": "Ajoutez cette information dans les paramètres.", | ||||
|     "githubSourceNote": "La limitation du débit de GitHub peut être évitée à l'aide d'une clé d'API.", | ||||
|     "sortByLastLinkSegment": "Trier par le dernier segment du lien", | ||||
|     "filterReleaseNotesByRegEx": "Filtrer les notes de version par expression régulière", | ||||
|     "customLinkFilterRegex": "Filtre du lien APK personnalisé par expression régulière (par défaut '.apk$')", | ||||
|     "appsPossiblyUpdated": "Tentative de mise à jour de l'application", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Avertit l'utilisateur que des mises à jour d'une ou plusieurs applications ont été potentiellement appliquées en arrière-plan", | ||||
|     "xWasPossiblyUpdatedToY": "{} a peut-être été mis à jour vers {}.", | ||||
|     "customLinkFilterRegex": "Filtre de lien APK personnalisé par expression régulière (par défaut '.apk$')", | ||||
|     "appsPossiblyUpdated": "Tentative de mise à jour des applications", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Notifie à l'utilisateur que des mises à jour d'une ou plusieurs applications ont potentiellement été appliquées en arrière-plan", | ||||
|     "xWasPossiblyUpdatedToY": "{} peut être mis à jour en {}.", | ||||
|     "enableBackgroundUpdates": "Activer les mises à jour en arrière-plan", | ||||
|     "backgroundUpdateReqsExplanation": "Les mises à jour en arrière-plan peuvent ne pas être possibles pour toutes les applications.", | ||||
|     "backgroundUpdateLimitsExplanation": "Le succès d'une installation en arrière-plan ne peut être déterminé qu'à l'ouverture d'Obtainium.", | ||||
|     "verifyLatestTag": "Vérifiez la balise 'Latest'", | ||||
|     "intermediateLinkRegex": " Filtrer un lien \" intermédiaire \" à visiter ", | ||||
|     "filterByLinkText": "Filtrer les liens par le texte du lien", | ||||
|     "backgroundUpdateLimitsExplanation": "Le résultat d'une installation en arrière-plan ne peut être déterminé qu'à l'ouverture d'Obtainium.", | ||||
|     "verifyLatestTag": "Vérifier la balise 'latest'", | ||||
|     "intermediateLinkRegex": "Filtrer un lien 'intermédiaire' à visiter", | ||||
|     "filterByLinkText": "Filtrer les liens par texte du lien", | ||||
|     "intermediateLinkNotFound": "Lien intermédiaire introuvable", | ||||
|     "intermediateLink": "Lien intermédiaire", | ||||
|     "exemptFromBackgroundUpdates": "Exempté des mises à jour en arrière-plan (si activé)", | ||||
|     "bgUpdatesOnWiFiOnly": "Désactiver les mises à jour en arrière-plan lorsque vous n'êtes pas connecté au WiFi", | ||||
|     "autoSelectHighestVersionCode": "Sélection automatique du code de version de l'APK la plus élevée", | ||||
|     "versionExtractionRegEx": "Expression régulière d'extraction de version", | ||||
|     "matchGroupToUse": "Groupe de correspondance pour l'expression régulière d'extraction de version", | ||||
|     "highlightTouchTargets": "Mettre en évidence les cibles tactiles moins évidentes", | ||||
|     "pickExportDir": "Choisir le répertoire d'exportation", | ||||
|     "autoExportOnChanges": "Exporter automatiquement après modification", | ||||
|     "exemptFromBackgroundUpdates": "Exclure de la mise à jour en arrière-plan (si activé)", | ||||
|     "bgUpdatesOnWiFiOnly": "Désactiver les mises à jour en arrière-plan lorsque vous n'êtes pas en WiFi", | ||||
|     "autoSelectHighestVersionCode": "Sélectionner automatiquement la version la plus récente du code APK", | ||||
|     "versionExtractionRegEx": "Extraire la version par Expression régulière", | ||||
|     "trimVersionString": "Découper la version par Expression régulière", | ||||
|     "matchGroupToUseForX": "Groupe de correspondance à utiliser pour \"{}\"", | ||||
|     "matchGroupToUse": "Groupe de correspondance à utiliser pour l'extraction de la version par Expression régulière", | ||||
|     "highlightTouchTargets": "Mettre en évidence les touches moins évidentes", | ||||
|     "pickExportDir": "Selectionner le dossier d'exportation", | ||||
|     "autoExportOnChanges": "Exporter automatiquement lors de modifications", | ||||
|     "includeSettings": "Inclure les paramètres", | ||||
|     "filterVersionsByRegEx": "Filtrer les versions par expression régulière", | ||||
|     "trySelectingSuggestedVersionCode": "Essayez de sélectionner le code de la version APK suggérée", | ||||
|     "dontSortReleasesList": "Conserver l'ordre des version de l'API", | ||||
|     "trySelectingSuggestedVersionCode": "Essayer de sélectionner la version suggérée du code APK", | ||||
|     "dontSortReleasesList": "Conserver l'ordre de la version de l'API", | ||||
|     "reverseSort": "Tri inversé", | ||||
|     "takeFirstLink": "Prendre le premier lien", | ||||
|     "skipSort": "Sauter le tri", | ||||
|     "debugMenu": "Menu de débogage", | ||||
|     "takeFirstLink": "Utiliser le premier lien", | ||||
|     "skipSort": "Ignorer le tri", | ||||
|     "debugMenu": "Menu de déboggage", | ||||
|     "bgTaskStarted": "Tâche en arrière-plan démarrée - vérifier les journaux.", | ||||
|     "runBgCheckNow": "Exécuter maintenant la vérification de la mise à jour en arrière-plan", | ||||
|     "versionExtractWholePage": "Appliquer l'expression régulière d'extraction de version sur l'ensemble de la page", | ||||
|     "runBgCheckNow": "Exécuter la recherche de mise à jour en arrière-plan maintenant", | ||||
|     "versionExtractWholePage": "Appliquer l'extraction de la version par expression régulière à l'ensemble de la page", | ||||
|     "installing": "Installation", | ||||
|     "skipUpdateNotifications": "Ignorer les notifications de mise à jour", | ||||
|     "updatesAvailableNotifChannel": "Mises à jour disponibles", | ||||
|     "appsUpdatedNotifChannel": "Applications mises à jour", | ||||
|     "appsPossiblyUpdatedNotifChannel": "Tentative de mise à jour de l'application", | ||||
|     "errorCheckingUpdatesNotifChannel": "Erreur lors de la vérification des mises à jour", | ||||
|     "appsPossiblyUpdatedNotifChannel": "Essayer de mettre à jour les applications", | ||||
|     "errorCheckingUpdatesNotifChannel": "Erreur lors de la recherche de mises à jour", | ||||
|     "appsRemovedNotifChannel": "Applications supprimées", | ||||
|     "downloadingXNotifChannel": "Téléchargement {}", | ||||
|     "completeAppInstallationNotifChannel": "Installation complète de l'application", | ||||
|     "checkingForUpdatesNotifChannel": "Vérification des mises à jour", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Vérifiez uniquement les mises à jour des applications installées et 'Track-Only'", | ||||
|     "checkingForUpdatesNotifChannel": "Recherche de mises à jour", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Rechercher uniquement les mises à jour des applications installées et des applications 'Suivi uniquement'", | ||||
|     "supportFixedAPKURL": "Prise en charge des URL APK fixes", | ||||
|     "selectX": "Sélectionner {}", | ||||
|     "parallelDownloads": "Autoriser les téléchargements parallèles", | ||||
|     "parallelDownloads": "Autoriser les téléchargements simultanés", | ||||
|     "useShizuku": "Utiliser Shizuku ou Sui pour l'installation", | ||||
|     "shizukuBinderNotFound": "Service Shizuku compatible non trouvé", | ||||
|     "shizukuOld": "Ancienne version de Shizuku (<11) - la mettre à jour", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku fonctionne sur Android < 8.1 avec ADB - mettre à jour Android ou utiliser Sui à la place", | ||||
|     "shizukuBinderNotFound": "Le service Shizuku est introuvable", | ||||
|     "shizukuOld": "Ancienne version de Shizuku (<11) - veuillez le mettre à jour", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku fonctionne sur Android < 8.1 avec ADB - veuillez mettre à jour Android ou utiliser Sui à la place", | ||||
|     "shizukuPretendToBeGooglePlay": "Définir Google Play comme source d'installation (si Shizuku est utilisé)", | ||||
|     "useSystemFont": "Utiliser la police du système", | ||||
|     "useVersionCodeAsOSVersion": "Utiliser le code de version de l'application comme version détectée par le système d'exploitation", | ||||
|     "requestHeader": "En-tête de demande", | ||||
|     "useLatestAssetDateAsReleaseDate": "Utiliser le dernier élément téléversé comme date de sortie", | ||||
|     "defaultPseudoVersioningMethod": "Méthode de pseudo-version par défaut", | ||||
|     "partialAPKHash": "Hash APK partiel", | ||||
|     "APKLinkHash": "Hash de lien APK", | ||||
|     "directAPKLink": "Lien APK direct", | ||||
|     "pseudoVersionInUse": "Une pseudo-version est utilisée", | ||||
|     "useVersionCodeAsOSVersion": "Utiliser le code de version de l'application détectée par le système d'exploitation", | ||||
|     "requestHeader": "Intitulé de la demande", | ||||
|     "useLatestAssetDateAsReleaseDate": "Utiliser le dernier élément mis en ligne comme date de sortie", | ||||
|     "defaultPseudoVersioningMethod": "Méthode de version fictive par défaut", | ||||
|     "partialAPKHash": "Hash partiel de l'APK", | ||||
|     "APKLinkHash": "Hash du lien APK", | ||||
|     "directAPKLink": "Lien direct de l'APK", | ||||
|     "pseudoVersionInUse": "Version fictive utilisé", | ||||
|     "installed": "Installée", | ||||
|     "latest": "Dernier", | ||||
|     "latest": "Dernière version", | ||||
|     "invertRegEx": "Inverser l'expression régulière", | ||||
|     "note": "Note", | ||||
|     "selfHostedNote": "La liste déroulante \"{}\" peut être utilisée pour accéder aux instances auto-hébergées/personnalisées de n'importe quelle source.", | ||||
|     "selfHostedNote": "La liste déroulante \"{}\" peut être utilisée pour accéder à des instances auto-hébergées/personnalisées de n'importe quelle source.", | ||||
|     "badDownload": "L'APK n'a pas pu être analysé (téléchargement incompatible ou partiel)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Partager les nouvelles applications avec AppVerifier (si disponible)", | ||||
|     "appVerifierInstructionToast": "Partagez avec AppVerifier, puis revenez ici lorsque vous êtes prêt.", | ||||
|     "appVerifierInstructionToast": "Partagez avec AppVerifier, puis revenez ici lorsque tout est prêt.", | ||||
|     "wiki": "Aide/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Configurations d'applications par la foule (utilisation à vos risques et périls)", | ||||
|     "crowdsourcedConfigsLabel": "Configurations d'applications par la communauté (à utiliser à vos risques et périls)", | ||||
|     "allowInsecure": "Autoriser les requêtes HTTP non sécurisées", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Supprimer l'application ?", | ||||
|         "other": "Supprimer les applications ?" | ||||
|         "one": "Supprimer l'application ?", | ||||
|         "other": "Supprimer les applications ?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "Trop de demandes (taux limité) - réessayez dans {} minute", | ||||
|         "other": "Trop de demandes (taux limité) - réessayez dans {} minutes" | ||||
|         "one": "Trop de requêtes (taux limité) - réessayez dans {} minute", | ||||
|         "other": "Trop de requêtes (taux limité) - réessayez dans {} minutes" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "La vérification de la mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative de vérification sera planifié dans {} minute", | ||||
|         "other": "La vérification de la mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative de vérification sera planifié dans {} minute" | ||||
|         "one": "La recherche de mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative programmée dans {} minute", | ||||
|         "other": "La recherche de mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative programmée dans {} minutes" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "La vérification des mises à jour en arrière-plan a trouvée {} mise à jour - l'utilisateur sera notifié si nécessaire", | ||||
|         "other": "La vérification des mises à jour en arrière-plan a trouvé {} mises à jour - l'utilisateur sera notifié si nécessaire" | ||||
|         "one": "La recherche de mises à jour en arrière-plan à trouvée {} mise à jour - l'utilisateur sera notifié si nécessaire", | ||||
|         "other": "La recherche de mises à jour en arrière-plan à trouvée {} mises à jour - l'utilisateur sera notifié si nécessaire" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} Application", | ||||
| @@ -337,7 +340,7 @@ | ||||
|         "other": "{} URL" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} Minutes", | ||||
|         "one": "{} Minute", | ||||
|         "other": "{} Minutes" | ||||
|     }, | ||||
|     "hour": { | ||||
| @@ -349,8 +352,8 @@ | ||||
|         "other": "{} Jours" | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "{n} journal effacé (avant = {before}, après = {after})", | ||||
|         "other": "{n} journaux effacés (avant = {before}, après = {after})" | ||||
|         "one": "Nettoyage du journal {n} (avant = {before}, après = {after})", | ||||
|         "other": "Nettoyage des journaux {n} (avant = {before}, après = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} et 1 autre application ont des mises à jour.", | ||||
| @@ -358,15 +361,15 @@ | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} et 1 autre application ont été mises à jour.", | ||||
|         "other": "{} et {} autres applications ont été mises à jour." | ||||
|         "other": "{} et {} autres applications ont étés mis à jour." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Échec de la mise à jour de {} et d'une autre application.", | ||||
|         "one": "Échec de la mise à jour de {} et 1 autre application.", | ||||
|         "other": "Échec de la mise à jour de {} et {} autres applications." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "une": "{} et 1 application supplémentaire ont peut-être été mises à jour.", | ||||
|         "other": "{} et {} autres applications peuvent avoir été mises à jour." | ||||
|         "one": "{} et 1 autre application ont peut-être été mises à jour.", | ||||
|         "other": "{} et {} autres applications ont peut-être étés mis à jour." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|   | ||||
| @@ -1,198 +1,198 @@ | ||||
| { | ||||
|     "invalidURLForSource": "Érvénytelen a(z) {} app URL-je", | ||||
|     "invalidURLForSource": "Érvénytelen a(z) {} alkalmazás webcíme", | ||||
|     "noReleaseFound": "Nem található megfelelő kiadás", | ||||
|     "noVersionFound": "Nem sikerült meghatározni a kiadás verzióját", | ||||
|     "urlMatchesNoSource": "Az URL nem egyezik ismert forrással", | ||||
|     "cantInstallOlderVersion": "Nem telepíthető egy app régebbi verziója", | ||||
|     "appIdMismatch": "A letöltött csomagazonosító nem egyezik a meglévő app azonosítóval", | ||||
|     "functionNotImplemented": "Ez az osztály nem valósította meg ezt a függvényt", | ||||
|     "placeholder": "Helykitöltő", | ||||
|     "urlMatchesNoSource": "A webcím nem egyezik egyetlen ismert forrással sem", | ||||
|     "cantInstallOlderVersion": "Nem telepíthető egy alkalmazás régebbi verziója", | ||||
|     "appIdMismatch": "A letöltött csomagazonosító nem egyezik a meglévő alkalmazás azonosítójával", | ||||
|     "functionNotImplemented": "Ebben az ágban nincs implementálva ez a funkció", | ||||
|     "placeholder": "Helyőrző", | ||||
|     "someErrors": "Néhány hiba történt", | ||||
|     "unexpectedError": "Váratlan hiba", | ||||
|     "ok": "Oké", | ||||
|     "ok": "Rendben", | ||||
|     "and": "és", | ||||
|     "githubPATLabel": "GitHub Personal Access Token (megnöveli a díjkorlátot)", | ||||
|     "githubPATLabel": "GitHub személyes hozzáférési token (megnöveli a lekérdezés-korlátozást)", | ||||
|     "includePrereleases": "Tartalmazza az előzetes kiadásokat", | ||||
|     "fallbackToOlderReleases": "Visszatérés a régebbi kiadásokhoz", | ||||
|     "filterReleaseTitlesByRegEx": "A kiadás címeinek szűrése reguláris kifejezéssel", | ||||
|     "invalidRegEx": "Érvénytelen reguláris kifejezés", | ||||
|     "noDescription": "Nincs leírás", | ||||
|     "cancel": "Mégse", | ||||
|     "continue": "Tovább", | ||||
|     "requiredInBrackets": "(Kötelező)", | ||||
|     "dropdownNoOptsError": "HIBA: A LEDOBÁST LEGALÁBB EGY OPCIÓHOZ KELL RENDELNI", | ||||
|     "continue": "Folytatás", | ||||
|     "requiredInBrackets": "(Szükséges)", | ||||
|     "dropdownNoOptsError": "HIBA: A LEGÖRDÜLŐ LISTÁNAK LEGALÁBB EGY OPCIÓVAL KELL RENDELKEZNIE", | ||||
|     "colour": "Szín", | ||||
|     "standard": "Standard", | ||||
|     "custom": "Custom", | ||||
|     "useMaterialYou": "Használja az Ön által használt anyagot", | ||||
|     "githubStarredRepos": "GitHub Csillagos Repo-k", | ||||
|     "uname": "Felh.név", | ||||
|     "wrongArgNum": "Rossz számú argumentumot adott meg", | ||||
|     "standard": "Általános", | ||||
|     "custom": "Egyéni", | ||||
|     "useMaterialYou": "Material You használata", | ||||
|     "githubStarredRepos": "Csillagozott GitHub tárolók", | ||||
|     "uname": "Felhasználónév", | ||||
|     "wrongArgNum": "A megadott argumentumok száma nem megfelelő", | ||||
|     "xIsTrackOnly": "A(z) {} csak nyomonkövethető", | ||||
|     "source": "Forrás", | ||||
|     "app": "Alkalmazás", | ||||
|     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások 'Csak nyomon követhetőek'.", | ||||
|     "youPickedTrackOnly": "A 'Csak követés' opciót választotta.", | ||||
|     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások „csak nyomonkövethetőek”.", | ||||
|     "youPickedTrackOnly": "„Csak nyomonkövetés” opciót választotta.", | ||||
|     "trackOnlyAppDescription": "Az alkalmazás frissítéseit nyomon követi, de az Obtainium nem tudja letölteni vagy telepíteni.", | ||||
|     "cancelled": "Törölve", | ||||
|     "appAlreadyAdded": "Az app már hozzáadva", | ||||
|     "alreadyUpToDateQuestion": "Az app már naprakész?", | ||||
|     "addApp": "App hozzáadás", | ||||
|     "appSourceURL": "App forrás URL", | ||||
|     "cancelled": "Visszavonva", | ||||
|     "appAlreadyAdded": "Az alkalmazás már hozzá van adva", | ||||
|     "alreadyUpToDateQuestion": "Az alkalmazás már naprakész?", | ||||
|     "addApp": "Hozzáadás", | ||||
|     "appSourceURL": "Alkalmazás forrásának webcíme", | ||||
|     "error": "Hiba", | ||||
|     "add": "Hozzáadás", | ||||
|     "searchSomeSourcesLabel": "Keresés (csak egyes források)", | ||||
|     "searchSomeSourcesLabel": "Keresés (csak bizonyos források)", | ||||
|     "search": "Keresés", | ||||
|     "additionalOptsFor": "További lehetőségek a következőhöz: {}", | ||||
|     "supportedSources": "Támogatott források", | ||||
|     "trackOnlyInBrackets": "(Csak nyomonkövetés)", | ||||
|     "searchableInBrackets": "(Kereshető)", | ||||
|     "appsString": "Appok", | ||||
|     "noApps": "Nincs App", | ||||
|     "noAppsForFilter": "Nincsenek appok a szűrőhöz", | ||||
|     "appsString": "Alkalmazások", | ||||
|     "noApps": "Nincsenek alkalmazások", | ||||
|     "noAppsForFilter": "Nincsenek alkalmazások a szűrőhöz", | ||||
|     "byX": "Fejlesztő: {}", | ||||
|     "percentProgress": "Folyamat: {}%", | ||||
|     "pleaseWait": "Kis türelmet", | ||||
|     "updateAvailable": "Frissítés érhető el", | ||||
|     "notInstalled": "Nem telepített", | ||||
|     "pseudoVersion": "ál-verzió", | ||||
|     "selectAll": "Mindet kiválaszt", | ||||
|     "deselectX": "Törölje {} kijelölését", | ||||
|     "pseudoVersion": "pszeudo-verzió", | ||||
|     "selectAll": "Összes kiválasztása", | ||||
|     "deselectX": "A(z) {} kiválasztásának elvetése", | ||||
|     "xWillBeRemovedButRemainInstalled": "A(z) {} el lesz távolítva az Obtainiumból, de továbbra is telepítve marad az eszközön.", | ||||
|     "removeSelectedAppsQuestion": "Eltávolítja a kiválasztott appokat?", | ||||
|     "removeSelectedApps": "Távolítsa el a kiválasztott appokat", | ||||
|     "updateX": "Frissítés: {}", | ||||
|     "installX": "Telepítés: {}", | ||||
|     "markXTrackOnlyAsUpdated": "Jelölje meg: {}\n(Csak nyomon követhető)\nmint Frissített", | ||||
|     "changeX": "Változás {}", | ||||
|     "installUpdateApps": "Appok telepítése/frissítése", | ||||
|     "installUpdateSelectedApps": "Telepítse/frissítse a kiválasztott appokat", | ||||
|     "markXSelectedAppsAsUpdated": "Megjelöl {} kiválasztott alkalmazást frissítettként?", | ||||
|     "removeSelectedAppsQuestion": "A kiválasztott alkalmazások eltávolítása?", | ||||
|     "removeSelectedApps": "A kiválasztott alkalmazások eltávolítása", | ||||
|     "updateX": "A(z) {} frissítése", | ||||
|     "installX": "A(z) {} telepítése", | ||||
|     "markXTrackOnlyAsUpdated": "Megjelölés: {}\n(Csak nyomon követhető)\nmint Frissített", | ||||
|     "changeX": "{} változtatás", | ||||
|     "installUpdateApps": "Alkalmazások telepítése/frissítése", | ||||
|     "installUpdateSelectedApps": "A kiválasztott alkalmazások telepítése/frissítése", | ||||
|     "markXSelectedAppsAsUpdated": "A(z) {} kiválasztott alkalmazás megjelölése frissítettként?", | ||||
|     "no": "Nem", | ||||
|     "yes": "Igen", | ||||
|     "markSelectedAppsUpdated": "Jelölje meg a kiválasztott appokat frissítettként", | ||||
|     "pinToTop": "Rögzítés felülre", | ||||
|     "unpinFromTop": "Eltávolít felülről", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Visszaállítja a kiválasztott appok telepítési állapotát?", | ||||
|     "installStatusOfXWillBeResetExplanation": "A kiválasztott appok telepítési állapota visszaáll.\n\nEz akkor segíthet, ha az Obtainiumban megjelenített app verzió hibás, frissítések vagy egyéb problémák miatt.", | ||||
|     "markSelectedAppsUpdated": "A kiválasztott alkalmazások megjelölése frissítettként", | ||||
|     "pinToTop": "Kitűzés felülre", | ||||
|     "unpinFromTop": "Kitűzés megszüntetése", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "Visszaállítja a kiválasztott alkalmazások telepítési állapotát?", | ||||
|     "installStatusOfXWillBeResetExplanation": "A kiválasztott alkalmazások telepítési állapota visszaáll.\n\nEz akkor segíthet, ha az Obtainiumban megjelenített alkalmazás-verzió hibás, a frissítések vagy egyéb problémák miatt.", | ||||
|     "customLinkMessage": "Ezek a hivatkozások a telepített Obtainium-mal rendelkező eszközökön működnek", | ||||
|     "shareAppConfigLinks": "App konfiguráció megosztása HTML linkként", | ||||
|     "shareSelectedAppURLs": "Ossza meg a kiválasztott app URL címeit", | ||||
|     "shareAppConfigLinks": "Alkalmazás-konfiguráció megosztása HTML hivatkozásként", | ||||
|     "shareSelectedAppURLs": "A kiválasztott alkalmazás hivatkozásának megosztása", | ||||
|     "resetInstallStatus": "Telepítési állapot visszaállítása", | ||||
|     "more": "További", | ||||
|     "removeOutdatedFilter": "Távolítsa el az elavult app szűrőt", | ||||
|     "showOutdatedOnly": "Csak az elavult appok megjelenítése", | ||||
|     "removeOutdatedFilter": "Elavult-alkalmazás szűrő eltávolítása", | ||||
|     "showOutdatedOnly": "Csak az elavult alkalmazások megjelenítése", | ||||
|     "filter": "Szűrő", | ||||
|     "filterApps": "Appok szűrése", | ||||
|     "appName": "App név", | ||||
|     "filterApps": "Alkalmazások szűrése", | ||||
|     "appName": "Név", | ||||
|     "author": "Szerző", | ||||
|     "upToDateApps": "Naprakész appok", | ||||
|     "nonInstalledApps": "Nem telepített appok", | ||||
|     "importExport": "Import Export", | ||||
|     "upToDateApps": "Naprakész alkalmazások", | ||||
|     "nonInstalledApps": "Nem telepített alkalmazások", | ||||
|     "importExport": "Import/Export", | ||||
|     "settings": "Beállítások", | ||||
|     "exportedTo": "Exportálva ide {}", | ||||
|     "obtainiumExport": "Obtainium Adat Exportálás", | ||||
|     "exportedTo": "Exportálva ide: {}", | ||||
|     "obtainiumExport": "Obtainium adatainak exportálása", | ||||
|     "invalidInput": "Hibás bemenet", | ||||
|     "importedX": "Importálva innen {}", | ||||
|     "obtainiumImport": "Obtainium Adat Importálás", | ||||
|     "importFromURLList": "Importálás URL listából", | ||||
|     "importedX": "Importálva innen: {}", | ||||
|     "obtainiumImport": "Obtainium adatok importálása", | ||||
|     "importFromURLList": "Importálás webcím-listából", | ||||
|     "searchQuery": "Keresési lekérdezés", | ||||
|     "appURLList": "App URL lista", | ||||
|     "appURLList": "Alkalmazás webcím-lista", | ||||
|     "line": "Sor", | ||||
|     "searchX": "Keresés {}", | ||||
|     "searchX": "{} keresése", | ||||
|     "noResults": "Nincs találat", | ||||
|     "importX": "Importálás: {}", | ||||
|     "importedAppsIdDisclaimer": "Előfordulhat, hogy az importált appok helytelenül \"Nincs telepítve\" jelzéssel jelennek meg.\nA probléma megoldásához telepítse újra őket az Obtainiumon keresztül.\nEz nem érinti az alkalmazásadatokat.\n\nCsak az URL-ekre és a harmadik féltől származó importálási módszerekre vonatkozik..", | ||||
|     "importX": "{} importálása", | ||||
|     "importedAppsIdDisclaimer": "Előfordulhat, hogy az importált alkalmazások helytelenül „Nincs telepítve” jelzéssel jelennek meg.\nA probléma megoldásához telepítse újra őket az Obtainiumon keresztül.\nEz nem érinti az alkalmazásadatokat.\n\nCsak a hivatkozásokra és a harmadik féltől származó importálási módszerekre vonatkozik..", | ||||
|     "importErrors": "Importálási hibák", | ||||
|     "importedXOfYApps": "{}/{} app importálva.", | ||||
|     "followingURLsHadErrors": "A következő URL-ek hibákat tartalmaztak:", | ||||
|     "selectURL": "Válassza ki az URL-t", | ||||
|     "selectURLs": "Kiválasztott URL-ek", | ||||
|     "pick": "Válasszon", | ||||
|     "importedXOfYApps": "{}/{} alkalmazás importálva.", | ||||
|     "followingURLsHadErrors": "A következő webcímek hibákat tartalmaztak:", | ||||
|     "selectURL": "Webcím kiválasztása", | ||||
|     "selectURLs": "Webcímek kiválasztása", | ||||
|     "pick": "Kiválasztás", | ||||
|     "theme": "Téma", | ||||
|     "dark": "Sötét", | ||||
|     "light": "Világos", | ||||
|     "followSystem": "Rendszer szerint", | ||||
|     "followSystem": "Rendszerbeállítás használata", | ||||
|     "followSystemThemeExplanation": "A következő rendszer téma csak harmadik féltől származó alkalmazások használatával lehetséges", | ||||
|     "useBlackTheme": "Használjon teljesen fekete sötét témát", | ||||
|     "appSortBy": "App rendezés...", | ||||
|     "appSortBy": "Elrendezés", | ||||
|     "authorName": "Szerző/Név", | ||||
|     "nameAuthor": "Név/Szerző", | ||||
|     "asAdded": "Mint Hozzáadott", | ||||
|     "appSortOrder": "Appok rendezése", | ||||
|     "asAdded": "Mint hozzáadott", | ||||
|     "appSortOrder": "Rendezési sorrend", | ||||
|     "ascending": "Emelkedő", | ||||
|     "descending": "Csökkenő", | ||||
|     "bgUpdateCheckInterval": "Háttérfrissítés ellenőrzés időköze", | ||||
|     "neverManualOnly": "Soha – csak manuális", | ||||
|     "bgUpdateCheckInterval": "Időtartam a frissítések háttér-ellenőrzése között", | ||||
|     "neverManualOnly": "Soha – csak kézi", | ||||
|     "appearance": "Megjelenés", | ||||
|     "showWebInAppView": "Forrás megjelenítése az Appok nézetben", | ||||
|     "pinUpdates": "Frissítések kitűzése az App nézet tetejére", | ||||
|     "showWebInAppView": "Forrás megjelenítése az alkalmazásnézetben", | ||||
|     "pinUpdates": "Frissítések kitűzése az alkalmazásnézet tetejére", | ||||
|     "updates": "Frissítések", | ||||
|     "sourceSpecific": "Forrás-specifikus", | ||||
|     "appSource": "App forrás", | ||||
|     "sourceSpecific": "Forrásspecifikus", | ||||
|     "appSource": "Alkalmazás forrása", | ||||
|     "noLogs": "Nincsenek naplók", | ||||
|     "appLogs": "App naplók", | ||||
|     "appLogs": "Alkalmazásnaplók", | ||||
|     "close": "Bezárás", | ||||
|     "share": "Megosztás", | ||||
|     "appNotFound": "App nem található", | ||||
|     "appNotFound": "Az alkalmazás nem található", | ||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||
|     "pickAnAPK": "Válasszon egy APK-t", | ||||
|     "appHasMoreThanOnePackage": "A(z) {} egynél több csomaggal rendelkezik:", | ||||
|     "deviceSupportsXArch": "Eszköze támogatja a {} CPU architektúrát.", | ||||
|     "deviceSupportsFollowingArchs": "Az eszköze a következő CPU architektúrákat támogatja:", | ||||
|     "deviceSupportsXArch": "Ez az eszköz támogatja a(z) {} CPU architektúrát.", | ||||
|     "deviceSupportsFollowingArchs": "Ez az eszköz a következő CPU architektúrákat támogatja:", | ||||
|     "warning": "Figyelem", | ||||
|     "sourceIsXButPackageFromYPrompt": "Az alkalmazás forrása „{}”, de a kiadási csomag innen származik: „{}”. Folytatja?", | ||||
|     "updatesAvailable": "Frissítések érhetők el", | ||||
|     "updatesAvailableNotifDescription": "Értesíti a felhasználót, hogy frissítések állnak rendelkezésre egy vagy több, az Obtainium által nyomon követett alkalmazáshoz", | ||||
|     "sourceIsXButPackageFromYPrompt": "Az alkalmazás forrása a(z) „{}” tároló, de a kiadási csomag innen származik: „{}”. Folytatja?", | ||||
|     "updatesAvailable": "Frissítések érhetőek el", | ||||
|     "updatesAvailableNotifDescription": "Értesíti a felhasználót, hogy egy vagy több, az Obtainium által nyomonkövetett alkalmazáshoz frissítések állnak rendelkezésre", | ||||
|     "noNewUpdates": "Nincsenek új frissítések.", | ||||
|     "xHasAnUpdate": "A(z) {} frissítést kapott.", | ||||
|     "appsUpdated": "Alkalmazások frissítve", | ||||
|     "appsNotUpdated": "Nem sikerült frissíteni az alkalmazásokat", | ||||
|     "appsUpdatedNotifDescription": "Értesíti a felhasználót, hogy egy/több app frissítése megtörtént a háttérben", | ||||
|     "xWasUpdatedToY": "{} frissítve a következőre: {}.", | ||||
|     "xWasNotUpdatedToY": "A {} frissítése a {}-ra nem sikerült.", | ||||
|     "appsUpdatedNotifDescription": "Értesíti a felhasználót, hogy egy vagy több alkalmazás frissítése a háttérben történt.", | ||||
|     "xWasUpdatedToY": "A(z) {} frissítve lett a következőre: {}.", | ||||
|     "xWasNotUpdatedToY": "Nem sikerült frissíteni a következőt: {}, erre: {}.", | ||||
|     "errorCheckingUpdates": "Hiba a frissítések keresésekor", | ||||
|     "errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, ha a háttérbeli frissítések ellenőrzése sikertelen", | ||||
|     "errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, ha a háttérfrissítés ellenőrzése nem sikerül", | ||||
|     "appsRemoved": "Alkalmazások eltávolítva", | ||||
|     "appsRemovedNotifDescription": "Értesíti a felhasználót egy vagy több alkalmazás eltávolításáról a betöltésük során fellépő hibák miatt", | ||||
|     "xWasRemovedDueToErrorY": "A(z) {} a következő hiba miatt lett eltávolítva: {}", | ||||
|     "completeAppInstallation": "Teljes app telepítés", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Az Obtainiumnak megnyitva kell lennie az alkalmazások telepítéséhez", | ||||
|     "appsRemovedNotifDescription": "Értesíti a felhasználót, hogy egy vagy több alkalmazás betöltés közbeni hiba miatt eltávolításra került", | ||||
|     "xWasRemovedDueToErrorY": "A(z) {} eltávolításra került a következő hiba miatt: {}", | ||||
|     "completeAppInstallation": "Teljes alkalmazástelepítés", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Az alkalmazások telepítéséhez az Obtainiumnak megnyitva kell lennie", | ||||
|     "completeAppInstallationNotifDescription": "Megkéri a felhasználót, hogy térjen vissza az Obtainiumhoz, hogy befejezze az alkalmazás telepítését", | ||||
|     "checkingForUpdates": "Frissítések keresése", | ||||
|     "checkingForUpdatesNotifDescription": "Átmeneti értesítés, amely a frissítések keresésekor jelenik meg", | ||||
|     "pleaseAllowInstallPerm": "Kérjük, engedélyezze az Obtainiumnak az alkalmazások telepítését", | ||||
|     "trackOnly": "Csak követés", | ||||
|     "pleaseAllowInstallPerm": "Engedélyezze az Obtainiumnak az alkalmazások telepítését", | ||||
|     "trackOnly": "Csak nyomonkövetés", | ||||
|     "errorWithHttpStatusCode": "Hiba {}", | ||||
|     "versionCorrectionDisabled": "Verzió korrekció letiltva (úgy tűnik, a beépülő modul nem működik)", | ||||
|     "versionCorrectionDisabled": "Verziókorrekció letiltva (úgy tűnik, hogy a bővítmény nem működik)", | ||||
|     "unknown": "Ismeretlen", | ||||
|     "none": "Egyik sem", | ||||
|     "none": "Semmi", | ||||
|     "never": "Soha", | ||||
|     "latestVersionX": "Legújabb verzió: {}", | ||||
|     "installedVersionX": "Telepített verzió: {}", | ||||
|     "lastUpdateCheckX": "Frissítés ellenőrizve: {}", | ||||
|     "lastUpdateCheckX": "Frissítések utoljára ellenőrizve: {}", | ||||
|     "remove": "Eltávolítás", | ||||
|     "yesMarkUpdated": "Igen, megjelölés frissítettként", | ||||
|     "fdroid": "F-Droid hivatalos", | ||||
|     "appIdOrName": "App ID vagy név", | ||||
|     "appId": "Alkalmazásazonosító", | ||||
|     "appWithIdOrNameNotFound": "Nem található app ezzel az azonosítóval vagy névvel", | ||||
|     "reposHaveMultipleApps": "A repók több alkalmazást is tartalmazhatnak", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Harmadik-fél Repo", | ||||
|     "steamMobile": "Steam mobil", | ||||
|     "appIdOrName": "Az alkalmazás-azonosító vagy név", | ||||
|     "appId": "Alkalmazás-azonosító", | ||||
|     "appWithIdOrNameNotFound": "Nem található alkalmazás ezzel az azonosítóval vagy névvel", | ||||
|     "reposHaveMultipleApps": "A tárolók több alkalmazást is tartalmazhatnak", | ||||
|     "fdroidThirdPartyRepo": "F-Droid harmadik féltől származó tároló", | ||||
|     "steamMobile": "Steam Mobil", | ||||
|     "steamChat": "Steam Chat", | ||||
|     "install": "Telepít", | ||||
|     "markInstalled": "Telepítettnek jelöl", | ||||
|     "update": "Frissít", | ||||
|     "markUpdated": "Frissítettnek jelöl", | ||||
|     "additionalOptions": "További lehetőségek", | ||||
|     "install": "Telepítés", | ||||
|     "markInstalled": "Telepítettnek jelölés", | ||||
|     "update": "Frissítés", | ||||
|     "markUpdated": "Frissítettnek jelölés", | ||||
|     "additionalOptions": "További beállítások", | ||||
|     "disableVersionDetection": "Verzió érzékelés letiltása", | ||||
|     "noVersionDetectionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzióérzékelés nem működik megfelelően.", | ||||
|     "downloadingX": "{} letöltés", | ||||
|     "downloadX": "Letöltés {}", | ||||
|     "downloadedX": "Letöltés {}", | ||||
|     "releaseAsset": "Kiadási tartalom", | ||||
|     "downloadNotifDescription": "Értesíti a felhasználót az app letöltésének előrehaladásáról", | ||||
|     "downloadingX": "{} letöltése", | ||||
|     "downloadX": "{} letöltése", | ||||
|     "downloadedX": "{} letöltve", | ||||
|     "releaseAsset": "Kiadási csomag", | ||||
|     "downloadNotifDescription": "Értesíti a felhasználót az alkalmazás letöltésének előrehaladásáról", | ||||
|     "noAPKFound": "Nem található APK", | ||||
|     "noVersionDetection": "Nincs verzió érzékelés", | ||||
|     "categorize": "Kategorizálás", | ||||
| @@ -201,8 +201,8 @@ | ||||
|     "noCategory": "Nincs kategória", | ||||
|     "noCategories": "Nincsenek kategóriák", | ||||
|     "deleteCategoriesQuestion": "Törli a kategóriákat?", | ||||
|     "categoryDeleteWarning": "A(z) {} összes app kategorizálatlan állapotba kerül.", | ||||
|     "addCategory": "Új kategória", | ||||
|     "categoryDeleteWarning": "A törölt kategóriákban lévő összes alkalmazás kategorizálatlanná válik.", | ||||
|     "addCategory": "Kategória hozzáadása", | ||||
|     "label": "Címke", | ||||
|     "language": "Nyelv", | ||||
|     "copiedToClipboard": "Másolva a vágólapra", | ||||
| @@ -210,131 +210,134 @@ | ||||
|     "selectedCategorizeWarning": "Ez felváltja a kiválasztott alkalmazások meglévő kategória-beállításait.", | ||||
|     "filterAPKsByRegEx": "Az APK-k szűrése reguláris kifejezéssel", | ||||
|     "removeFromObtainium": "Eltávolítás az Obtainiumból", | ||||
|     "uninstallFromDevice": "Eltávolítás a készülékről", | ||||
|     "uninstallFromDevice": "Eltávolítás az eszközről", | ||||
|     "onlyWorksWithNonVersionDetectApps": "Csak azoknál az alkalmazásoknál működik, amelyeknél a verzióérzékelés le van tiltva.", | ||||
|     "releaseDateAsVersion": "Használja a Kiadás dátumát, mint verziót", | ||||
|     "releaseDateAsVersionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzió érzékelése nem működik megfelelően, de elérhető a kiadás dátuma.", | ||||
|     "changes": "Változtatások", | ||||
|     "releaseDateAsVersion": "Használja a kiadás dátumát, mint verziót", | ||||
|     "releaseDateAsVersionExplanation": "Ezt a beállítást csak olyan alkalmazásoknál szabad használni, ahol a verzióérzékelés nem működik megfelelően, de elérhető a kiadás dátuma.", | ||||
|     "changes": "Változások", | ||||
|     "releaseDate": "Kiadás dátuma", | ||||
|     "importFromURLsInFile": "Importálás fájlban található URL-ből (mint pl. OPML)", | ||||
|     "versionDetectionExplanation": "A verzió karakterlánc egyeztetése az OS által észlelt verzióval", | ||||
|     "versionDetection": "Verzió érzékelés", | ||||
|     "standardVersionDetection": "Alapért. verzió érzékelés", | ||||
|     "groupByCategory": "Csoportosítás Kategória alapján", | ||||
|     "importFromURLsInFile": "Importálás fájlban található webcímből (pl. OPML)", | ||||
|     "versionDetectionExplanation": "A verzió-karakterlánc egyeztetése az OS által észlelt verzióval", | ||||
|     "versionDetection": "Verzióérzékelés", | ||||
|     "standardVersionDetection": "Alapértelmezett verzióérzékelés", | ||||
|     "groupByCategory": "Csoportosítás kategória alapján", | ||||
|     "autoApkFilterByArch": "Ha lehetséges, próbálja CPU architektúra szerint szűrni az APK-kat", | ||||
|     "overrideSource": "Forrás felülbírálása", | ||||
|     "dontShowAgain": "Ne mutassa ezt újra", | ||||
|     "dontShowTrackOnlyWarnings": "Ne jelenítsen meg 'Csak nyomon követés' figyelmeztetést", | ||||
|     "dontShowAPKOriginWarnings": "Ne jelenítsen meg az APK eredetére vonatkozó figyelmeztetéseket", | ||||
|     "moveNonInstalledAppsToBottom": "Helyezze át a nem telepített appokat az App nézet aljára", | ||||
|     "overrideSource": "Forrás felülírása", | ||||
|     "dontShowAgain": "Ne jelenítse meg ezt többé", | ||||
|     "dontShowTrackOnlyWarnings": "Ne jelenítse meg a „Csak nyomonkövetés” figyelmeztetést", | ||||
|     "dontShowAPKOriginWarnings": "Ne jelenítse meg az APK eredetére vonatkozó figyelmeztetéseket", | ||||
|     "moveNonInstalledAppsToBottom": "Helyezze át a nem telepített alkalmazásokat az alkalmazás-nézet aljára", | ||||
|     "gitlabPATLabel": "GitLab személyes hozzáférési token", | ||||
|     "about": "Rólunk", | ||||
|     "requiresCredentialsInSettings": "{}: Ehhez további hitelesítő adatokra van szükség (a Beállításokban)", | ||||
|     "checkOnStart": "Egyszer az alkalmazás indításakor is", | ||||
|     "tryInferAppIdFromCode": "Próbálja kikövetkeztetni az app azonosítót a forráskódból", | ||||
|     "removeOnExternalUninstall": "A külsőleg eltávolított appok auto. eltávolítása", | ||||
|     "pickHighestVersionCode": "A legmagasabb verziószámú APK auto. kiválasztása", | ||||
|     "checkUpdateOnDetailPage": "Frissítések keresése az app részleteit tartalmazó oldal megnyitásakor", | ||||
|     "about": "Névjegy", | ||||
|     "requiresCredentialsInSettings": "A(z) {} alkalmazásnak további hitelesítő adatokra van szüksége (a beállításokban)", | ||||
|     "checkOnStart": "Frissítések keresése indításkor", | ||||
|     "tryInferAppIdFromCode": "Próbálja meg kikövetkeztetni az alkalmazás azonosítóját a forráskódból", | ||||
|     "removeOnExternalUninstall": "A külsőleg eltávolított alkalmazások automatikus eltávolítása", | ||||
|     "pickHighestVersionCode": "A legmagasabb verziószámú APK automatikus kiválasztása", | ||||
|     "checkUpdateOnDetailPage": "Frissítések keresése az alkalmazás részleteit tartalmazó oldal megnyitásakor", | ||||
|     "disablePageTransitions": "Lap áttűnési animációk letiltása", | ||||
|     "reversePageTransitions": "Fordított lap áttűnési animációk", | ||||
|     "minStarCount": "Minimális csillag szám", | ||||
|     "addInfoBelow": "Adja hozzá ezt az infót alább.", | ||||
|     "addInfoInSettings": "Adja hozzá ezt az infót a Beállításokban.", | ||||
|     "githubSourceNote": "A GitHub sebességkorlátozás elkerülhető API-kulcs használatával.", | ||||
|     "sortByLastLinkSegment": "Rendezés csak a link utolsó szegmense szerint", | ||||
|     "minStarCount": "Minimális csillagozási szám", | ||||
|     "addInfoBelow": "Adja hozzá ezt az információt alább.", | ||||
|     "addInfoInSettings": "Adja hozzá ezt az információt a beállításokban.", | ||||
|     "githubSourceNote": "A GitHub/GitLab lekérdezés-korlátozás elkerülhető egy API-kulcs használatával.", | ||||
|     "sortByLastLinkSegment": "Rendezés csak a hivatkozás utolsó szegmense szerint", | ||||
|     "filterReleaseNotesByRegEx": "Kiadási megjegyzések szűrése reguláris kifejezéssel", | ||||
|     "customLinkFilterRegex": "Egyéni APK hivatkozásszűrő reguláris kifejezéssel (Alapérték '.apk$')", | ||||
|     "appsPossiblyUpdated": "App frissítési kísérlet", | ||||
|     "customLinkFilterRegex": "Egyéni APK hivatkozásszűrő reguláris kifejezéssel (Alapértelmezett „.apk$”)", | ||||
|     "appsPossiblyUpdated": "Megkísérelt alkalmazás-frissítések", | ||||
|     "appsPossiblyUpdatedNotifDescription": "Értesíti a felhasználót, hogy egy vagy több alkalmazás frissítése lehetséges a háttérben", | ||||
|     "xWasPossiblyUpdatedToY": "{} frissítve lehet erre {}.", | ||||
|     "enableBackgroundUpdates": "Frissítések a háttérben", | ||||
|     "backgroundUpdateReqsExplanation": "Előfordulhat, hogy nem minden appnál lehetséges a háttérbeli frissítés.", | ||||
|     "xWasPossiblyUpdatedToY": "A(z) {} frissülhetett a következőre: {}.", | ||||
|     "enableBackgroundUpdates": "Háttérfrissítések engedélyezése", | ||||
|     "backgroundUpdateReqsExplanation": "Előfordulhat, hogy nem minden alkalmazásnál lehetséges a háttérbeli frissítés.", | ||||
|     "backgroundUpdateLimitsExplanation": "A háttérben történő telepítés sikeressége csak az Obtainium megnyitásakor állapítható meg.", | ||||
|     "verifyLatestTag": "Ellenőrizze a „legújabb” címkét", | ||||
|     "intermediateLinkRegex": "Szűrés egy 'köztes' látogatási linkre", | ||||
|     "filterByLinkText": "A hivatkozások szűrése linkszöveg alapján", | ||||
|     "intermediateLinkNotFound": "Köztes link nem található", | ||||
|     "intermediateLink": "Köztes link", | ||||
|     "intermediateLinkRegex": "Szűrés egy „köztes” hivatkozás megnyitásához", | ||||
|     "filterByLinkText": "Hivatkozások szűrése egy hivatkozásszöveg alapján", | ||||
|     "intermediateLinkNotFound": "Köztes hivatkozás nem található", | ||||
|     "intermediateLink": "Köztes hivatkozás", | ||||
|     "exemptFromBackgroundUpdates": "Mentes a háttérben történő frissítések alól (ha engedélyezett)", | ||||
|     "bgUpdatesOnWiFiOnly": "Tiltsa le a háttérben frissítéseket, ha nincs Wi-Fi-n", | ||||
|     "autoSelectHighestVersionCode": "A legmagasabb verziószámú APK auto. kiválasztása", | ||||
|     "versionExtractionRegEx": "Verzió kibontása reguláris kifejezéssel", | ||||
|     "matchGroupToUse": "Párosítsa a csoportot a használathoz", | ||||
|     "highlightTouchTargets": "Emelje ki a kevésbé nyilvánvaló érintési célokat", | ||||
|     "pickExportDir": "Válassza az Exportálási könyvtárat", | ||||
|     "autoExportOnChanges": "Auto-exportálás a változások után", | ||||
|     "bgUpdatesOnWiFiOnly": "A háttérben futó frissítések letiltása, ha nincs Wi-Fi", | ||||
|     "autoSelectHighestVersionCode": "A legmagasabb verziószámú APK automatikus kiválasztása", | ||||
|     "versionExtractionRegEx": "Verziókarakterlánc-kivonatolása reguláris kifejezéssel", | ||||
|     "trimVersionString": "Verziókarakterlánc levágása reguláris kifejezéssel", | ||||
|     "matchGroupToUseForX": "A(z) „{}” esetén használandó csoport egyeztetése", | ||||
|     "matchGroupToUse": "Verziókarakterlánc-kivonatoláshoz használandó csoport reguláris kifejezéssel való egyeztetése", | ||||
|     "highlightTouchTargets": "A kevésbé nyilvánvaló érintési pontok kiemelése", | ||||
|     "pickExportDir": "Válassza ki a könyvtárat, ahová exportálni szeretne", | ||||
|     "autoExportOnChanges": "Automatikus exportálás a változások után", | ||||
|     "includeSettings": "Tartalmazza a beállításokat", | ||||
|     "filterVersionsByRegEx": "Verziók szűrése reguláris kifejezéssel", | ||||
|     "trySelectingSuggestedVersionCode": "Próbálja ki a javasolt verziókódú APK-t", | ||||
|     "dontSortReleasesList": "Az API-ból származó kiadási sorrend megőrzése", | ||||
|     "reverseSort": "Fordított rendezés", | ||||
|     "takeFirstLink": "Vegye az első linket", | ||||
|     "skipSort": "A válogatás kihagyása", | ||||
|     "reverseSort": "Fordított elrendezés", | ||||
|     "takeFirstLink": "Vegye az első hivatkozást", | ||||
|     "skipSort": "Rendezés kihagyása", | ||||
|     "debugMenu": "Hibakereső menü", | ||||
|     "bgTaskStarted": "A háttérfeladat elindult – ellenőrizze a naplókat.", | ||||
|     "runBgCheckNow": "Futtassa a Háttérben frissítés ellenőrzését most", | ||||
|     "versionExtractWholePage": "Alkalmazza a Version Extraction Regex-et az egész oldalra", | ||||
|     "runBgCheckNow": "Frissítések keresése a háttérben", | ||||
|     "versionExtractWholePage": "„Verziókarakterlánc-kivonatolása reguláris kifejezéssel” alkalmazása az egész oldalra", | ||||
|     "installing": "Telepítés", | ||||
|     "skipUpdateNotifications": "A frissítési értesítések kihagyása", | ||||
|     "updatesAvailableNotifChannel": "Frissítések érhetők el", | ||||
|     "updatesAvailableNotifChannel": "Frissítések érhetőek el", | ||||
|     "appsUpdatedNotifChannel": "Alkalmazások frissítve", | ||||
|     "appsPossiblyUpdatedNotifChannel": "App frissítési kísérlet", | ||||
|     "appsPossiblyUpdatedNotifChannel": "Megkísérelt alkalmazás-frissítések", | ||||
|     "errorCheckingUpdatesNotifChannel": "Hiba a frissítések keresésekor", | ||||
|     "appsRemovedNotifChannel": "Alkalmazások eltávolítva", | ||||
|     "downloadingXNotifChannel": "{} letöltés", | ||||
|     "completeAppInstallationNotifChannel": "Teljes app telepítés", | ||||
|     "appsRemovedNotifChannel": "Eltávolított alkalmazások", | ||||
|     "downloadingXNotifChannel": "A(z) {} letöltése", | ||||
|     "completeAppInstallationNotifChannel": "Teljes alkalmazás telepítés", | ||||
|     "checkingForUpdatesNotifChannel": "Frissítések keresése", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Csak a telepített és a csak követhető appokat ellenőrizze frissítésekért", | ||||
|     "supportFixedAPKURL": "Támogatja a rögzített APK URL-eket", | ||||
|     "selectX": "Kiválaszt {}", | ||||
|     "parallelDownloads": "Párhuzamos letöltéseket enged", | ||||
|     "useShizuku": "Használja Shizuku vagy Sui telepítéséhez", | ||||
|     "shizukuBinderNotFound": "A Shizuku nem fut", | ||||
|     "shizukuOld": "Régi Shizuku verzió (<11) - frissítsd!", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku fut Android < 8.1 ADB-vel - frissítse az Androidot vagy használja a Sui-t helyette", | ||||
|     "shizukuPretendToBeGooglePlay": "Állítsa be a Google Play-t telepítési forrásként (ha Shizuku-t használ)", | ||||
|     "useSystemFont": "Használja a rendszer betűtípusát", | ||||
|     "useVersionCodeAsOSVersion": "Az app verziókód használata a rendszer által észlelt verzióként", | ||||
|     "requestHeader": "Kérelem fejléc", | ||||
|     "useLatestAssetDateAsReleaseDate": "Használja a legújabb tartalomfeltöltést megjelenési dátumként", | ||||
|     "defaultPseudoVersioningMethod": "Alapértelmezett álverziós módszer", | ||||
|     "partialAPKHash": "Részleges APK Hash", | ||||
|     "APKLinkHash": "APK Link Hash", | ||||
|     "directAPKLink": "Közvetlen APK Link", | ||||
|     "pseudoVersionInUse": "Egy álverzió van használatban", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Csak a telepített és a csak nyomonkövethető alkalmazások frissítéseinek ellenőrzése", | ||||
|     "supportFixedAPKURL": "Támogatja a rögzített APK webcímeket", | ||||
|     "selectX": "{} kiválasztása", | ||||
|     "parallelDownloads": "Párhuzamos letöltések engedélyezése", | ||||
|     "useShizuku": "Shizuku vagy Sui használata a telepítéshez", | ||||
|     "shizukuBinderNotFound": "A Shizuku szolgáltatás nem fut", | ||||
|     "shizukuOld": "Régi Shizuku verzió (<11) - frissítse", | ||||
|     "shizukuOldAndroidWithADB": "A Shizuku csak Android < 8.1 ADB-vel fut - frissítse az Androidot vagy használja a Sui-t helyette", | ||||
|     "shizukuPretendToBeGooglePlay": "Állítsa be a Google Playt telepítési forrásként (ha Shizukut használ)", | ||||
|     "useSystemFont": "A rendszer betűtípusának használata", | ||||
|     "useVersionCodeAsOSVersion": "Az alkalmazás verziókódjának használata a rendszer által észlelt verzióként", | ||||
|     "requestHeader": "Kérelemfejléc", | ||||
|     "useLatestAssetDateAsReleaseDate": "A kiadás dátumaként használja a legutóbbi csomagfeltöltést", | ||||
|     "defaultPseudoVersioningMethod": "Alapértelmezett pszeudo-verziós módszer", | ||||
|     "partialAPKHash": "Részleges APK hasító értéke", | ||||
|     "APKLinkHash": "APK hivatkozás hasító értéke", | ||||
|     "directAPKLink": "Közvetlen APK hivatkozás", | ||||
|     "pseudoVersionInUse": "Egy pszeudo-verzió van használatban", | ||||
|     "installed": "Telepített", | ||||
|     "latest": "Legújabb", | ||||
|     "invertRegEx": "Invertált reguláris kifejezés", | ||||
|     "note": "Megjegyzés:", | ||||
|     "selfHostedNote": "A \"{}\" legördülő menü használható bármely forrás saját üzemeltetésű/egyéni példányainak eléréséhez.", | ||||
|     "invertRegEx": "Reguláris kifejezés invertálása", | ||||
|     "note": "Megjegyzés", | ||||
|     "selfHostedNote": "A(z) \„{}\” legördülő menü segítségével elérhetővé válnak a bármilyen más forrásból származó saját üzemeltetésű- vagy egyéni példányok.", | ||||
|     "badDownload": "Az APK-t nem lehetett elemezni (inkompatibilis vagy részleges letöltés)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Új alkalmazások megosztása az AppVerifierrel (ha elérhető)", | ||||
|     "appVerifierInstructionToast": "Ossza meg az AppVerifierrel, majd térjen vissza ide, ha kész.", | ||||
|     "wiki": "Súgó/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (használat saját felelősségre)", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsource-ből származó alkalmazások beállítása (saját felelősségére használja)", | ||||
|     "allowInsecure": "Bizonytalan HTTP-kérések engedélyezése", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Eltávolítja az alkalmazást?", | ||||
|         "other": "Eltávolítja az alkalmazásokat?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva", | ||||
|         "other": "Túl sok kérés (korlátozott arány) – próbálja újra {} perc múlva" | ||||
|         "one": "Túl sok kérés (lekérdezés-korlátozás) – próbálja újra {} perc múlva,\nvagy adjon meg egy GitHub/GitLab API-kulcsot", | ||||
|         "other": "Túl sok kérés (lekérdezés-korlátozás) – próbálja újra {} perc múlva,\nvagy adjon meg egy GitHub/GitLab API-kulcsot" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "A háttérfrissítések ellenőrzése {}-t észlelt, {} perc múlva ütemezi az újrapróbálkozást", | ||||
|         "other": "A háttérfrissítések ellenőrzése {}-t észlelt, {} perc múlva ütemezi az újrapróbálkozást" | ||||
|         "one": "Frissítések ellenőrzése a háttérben a következőt észlelte: {} - Újrapróbálkozás: {} perc múlva", | ||||
|         "other": "Frissítések ellenőrzése a háttérben a következőt észlelte: {} - Újrapróbálkozás: {} perc múlva" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "A háttérfrissítés ellenőrzése {} frissítést talált – szükség esetén értesíti a felhasználót", | ||||
|         "other": "A háttérfrissítés ellenőrzése {} frissítést talált – szükség esetén értesíti a felhasználót" | ||||
|         "one": "Frissítések ellenőrzése a háttérben {} frissítést talált - szükség esetén értesíti a felhasználót", | ||||
|         "other": "Frissítések ellenőrzése a háttérben {} frissítést talált - szükség esetén értesíti a felhasználót" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} app", | ||||
|         "other": "{} app" | ||||
|         "one": "{} alkalmazás", | ||||
|         "other": "{} alkalmazás" | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URL" | ||||
|         "one": "{} webcím", | ||||
|         "other": "{} webcím" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} perc", | ||||
| @@ -349,27 +352,27 @@ | ||||
|         "other": "{} nap" | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "{n} napló törölve (előtte = {előtte}, utána = {utána})", | ||||
|         "other": "{n} napló törölve (előtte = {előtte}, utána = {utána})" | ||||
|         "one": "{n} napló törölve lett ({after} után és {before} előtt)", | ||||
|         "other": "{n} napló törölve lett ({after} után és {before} előtt)" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "A(z) {} és 1 további alkalmazás frissítéseket kapott.", | ||||
|         "other": "{} és {} további alkalmazás frissítéseket kapott." | ||||
|         "one": "A(z) {} és 1 további alkalmazás frissítést kapott.", | ||||
|         "other": "A(z) {} és {} további alkalmazás frissítést kapott." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "A(z) {} és 1 további alkalmazás frissítve.", | ||||
|         "other": "{} és {} további alkalmazás frissítve." | ||||
|         "other": "A(z) {} és {} további alkalmazás frissítve." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Nem sikerült frissíteni {} és még 1 alkalmazást.", | ||||
|         "other": "Nem sikerült frissíteni {} és {} további alkalmazásokat." | ||||
|         "one": "Nem sikerült frissíteni a következőt: {}, valamint 1 további alkalmazást.", | ||||
|         "other": "Nem sikerült frissíteni a következőt: {}, valamint {} további alkalmazást." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} és 1 további alkalmazás is frissült.", | ||||
|         "other": "{} és {} további alkalmazás is frissült." | ||||
|         "one": "A(z) {}, valamint 1 további alkalmazás sikeresen frissítve.", | ||||
|         "other": "A(z) {}, valamint {} további alkalmazás sikeresen frissítve." | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APK-k" | ||||
|         "other": "{} APK" | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Disattiva aggiornamenti in secondo piano quando non si usa il WiFi", | ||||
|     "autoSelectHighestVersionCode": "Auto-seleziona APK con versionCode più alto", | ||||
|     "versionExtractionRegEx": "RegEx di estrazione versione", | ||||
|     "trimVersionString": "Tagliare la stringa della versione con RegEx", | ||||
|     "matchGroupToUseForX": "Gruppo di corrispondenza da utilizzare per \"{}\"", | ||||
|     "matchGroupToUse": "Gruppo da usare", | ||||
|     "highlightTouchTargets": "Evidenzia elementi toccabili meno ovvi", | ||||
|     "pickExportDir": "Scegli cartella esp.", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "Condividete con AppVerifier, quindi tornate qui quando siete pronti.", | ||||
|     "wiki": "Aiuto/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Configurazioni di app in crowdsourcing (uso a proprio rischio)", | ||||
|     "allowInsecure": "Consentire le richieste HTTP non sicure", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Rimuovere l'app?", | ||||
|         "other": "Rimuovere le app?" | ||||
|   | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "WiFiを使用していない場合、バックグラウンドアップデートを無効にする", | ||||
|     "autoSelectHighestVersionCode": "最も高いバージョンコードのAPKを自動で選択する", | ||||
|     "versionExtractionRegEx": "バージョン抽出の正規表現", | ||||
|     "trimVersionString": "正規表現でバージョン文字列をトリムする", | ||||
|     "matchGroupToUseForX": "\"{}\"に使用するマッチしたグループ", | ||||
|     "matchGroupToUse": "使用するマッチしたグループ", | ||||
|     "highlightTouchTargets": "目立たないタップ可能な対象をハイライトする", | ||||
|     "pickExportDir": "エクスポートディレクトリを選択", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "AppVerifierに共有し、準備ができたらここに戻ってください。", | ||||
|     "wiki": "ヘルプ/ウィキ", | ||||
|     "crowdsourcedConfigsLabel": "クラウドソーシングによるアプリの設定(利用は自己責任で)", | ||||
|     "allowInsecure": "安全でないHTTPリクエストを許可する", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "アプリを削除しますか?", | ||||
|         "other": "アプリを削除しますか?" | ||||
|   | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Achtergrond-updates uitschakelen wanneer niet verbonden met WiFi", | ||||
|     "autoSelectHighestVersionCode": "De APK met de hoogste versiecode automatisch selecteren", | ||||
|     "versionExtractionRegEx": "Reguliere expressie voor versie-extractie", | ||||
|     "trimVersionString": "Versie string trimmen met RegEx", | ||||
|     "matchGroupToUseForX": "Overeenkomende groep te gebruiken voor \"{}\"", | ||||
|     "matchGroupToUse": "Overeenkomende groep om te gebruiken voor de reguliere expressie voor versie-extractie", | ||||
|     "highlightTouchTargets": "Minder voor de hand liggende aanraakdoelen markeren.", | ||||
|     "pickExportDir": "Kies de exportmap", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "Deel het met AppVerifier en keer daarna hier terug.", | ||||
|     "wiki": "Help/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App-configuraties (gebruik op eigen risico)", | ||||
|     "allowInsecure": "Onveilige HTTP-verzoeken toestaan", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "App verwijderen?", | ||||
|         "other": "Apps verwijderen?" | ||||
|   | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Wyłącz aktualizacje w tle, gdy nie ma połączenia z Wi-Fi", | ||||
|     "autoSelectHighestVersionCode": "Automatycznie wybierz najwyższy kod wersji APK", | ||||
|     "versionExtractionRegEx": "Wyrażenie regularne wyodrębniające wersję", | ||||
|     "trimVersionString": "Przycinanie łańcucha wersji za pomocą RegEx", | ||||
|     "matchGroupToUseForX": "Dopasuj grupę do użycia dla \"{}\"", | ||||
|     "matchGroupToUse": "Dopasuj grupę do użycia dla wyrażenia regularnego wyodrębniania wersji", | ||||
|     "highlightTouchTargets": "Wyróżnij mniej oczywiste elementy dotykowe", | ||||
|     "pickExportDir": "Wybierz katalog eksportu", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "Udostępnij w AppVerifier, a następnie wróć tutaj, gdy będziesz gotowy.", | ||||
|     "wiki": "Pomoc/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Konfiguracje aplikacji pochodzące z crowdsourcingu (korzystanie na własne ryzyko)", | ||||
|     "allowInsecure": "Zezwalaj na niezabezpieczone żądania HTTP", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Usunąć aplikację?", | ||||
|         "few": "Usunąć aplikacje?", | ||||
|   | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Desative as atualizações em segundo-plano quando não estiver conectado no Wi-Fi", | ||||
|     "autoSelectHighestVersionCode": "Auto-selecionar a versão mais recente", | ||||
|     "versionExtractionRegEx": "Regex de extração de versão", | ||||
|     "trimVersionString": "Cortar a cadeia de caracteres da versão com RegEx", | ||||
|     "matchGroupToUseForX": "Grupo de correspondência a utilizar para \"{}\"", | ||||
|     "matchGroupToUse": "Grupo correspondente a ser usado no Regex de extração de versão", | ||||
|     "highlightTouchTargets": "Realçar áreas sensíveis ao toque que são menos óbvias", | ||||
|     "pickExportDir": "Escolher diretório para exportação", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "Partilhe com o AppVerifier e, em seguida, regresse aqui quando estiver pronto.", | ||||
|     "wiki": "Ajuda/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Configurações de aplicações de crowdsourcing (utilização por sua conta e risco)", | ||||
|     "allowInsecure": "Permitir pedidos HTTP inseguros", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Remover aplicativo?", | ||||
|         "other": "Remover aplicativos?" | ||||
|   | ||||
| @@ -129,7 +129,7 @@ | ||||
|     "pinUpdates": "Отображать обновления приложений сверху списка", | ||||
|     "updates": "Обновления", | ||||
|     "sourceSpecific": "Настройки источников", | ||||
|     "appSource": "Исходный код", | ||||
|     "appSource": "Источник", | ||||
|     "noLogs": "Нет журналов", | ||||
|     "appLogs": "Логи", | ||||
|     "close": "Закрыть", | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Отключить фоновые обновления, если нет соединения с Wi-Fi", | ||||
|     "autoSelectHighestVersionCode": "Автоматически выбирать APK с актуальной версией кода", | ||||
|     "versionExtractionRegEx": "Регулярное выражение для извлечения версии", | ||||
|     "trimVersionString": "Обрезка строки версии с помощью RegEx", | ||||
|     "matchGroupToUseForX": "Группа соответствия, которую следует использовать для \"{}\"", | ||||
|     "matchGroupToUse": "Выберите группу для использования", | ||||
|     "highlightTouchTargets": "Выделить менее очевидные элементы управления касанием", | ||||
|     "pickExportDir": "Выбрать каталог для экспорта", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "Поделитесь с AppVerifier, а затем вернитесь сюда, когда будете готовы.", | ||||
|     "wiki": "Помощь/Вики", | ||||
|     "crowdsourcedConfigsLabel": "Конфигурации приложений на основе краудсорсинга (используйте на свой страх и риск)", | ||||
|     "allowInsecure": "Разрешить небезопасные HTTP-запросы", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Удалить приложение?", | ||||
|         "other": "Удалить приложения?" | ||||
|   | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Inaktivera Bakgrundsuppdateringar utan WiFi", | ||||
|     "autoSelectHighestVersionCode": "Välj automatiskt högsta versionskod APK", | ||||
|     "versionExtractionRegEx": "Version Extraction RegEx", | ||||
|     "trimVersionString": "Trimma versionssträng med RegEx", | ||||
|     "matchGroupToUseForX": "Matchningsgrupp att använda för \"{}\"", | ||||
|     "matchGroupToUse": "Match Group to Use", | ||||
|     "highlightTouchTargets": "Markera mindre uppenbara beröringsobjekt", | ||||
|     "pickExportDir": "Välj Exportsökväg", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "Dela till AppVerifier och återvänd sedan hit när du är klar.", | ||||
|     "wiki": "Hjälp/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourcade appkonfigurationer (använd på egen risk)", | ||||
|     "allowInsecure": "Tillåt osäkra HTTP-förfrågningar", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Ta Bort App?", | ||||
|         "other": "Ta Bort Appar?" | ||||
|   | ||||
| @@ -3,7 +3,7 @@ | ||||
|     "noReleaseFound": "Uygun bir sürüm bulunamadı", | ||||
|     "noVersionFound": "Sürüm bulunamadı", | ||||
|     "urlMatchesNoSource": "URL, bilinen bir kaynağa uymuyor", | ||||
|     "cantInstallOlderVersion": "Eski bir sürümü yükleyemem", | ||||
|     "cantInstallOlderVersion": "Eski bir sürüm yüklenemez", | ||||
|     "appIdMismatch": "İndirilen paket kimliği mevcut Uygulama kimliği ile eşleşmiyor", | ||||
|     "functionNotImplemented": "Bu sınıf bu işlevi uygulamamıştır", | ||||
|     "placeholder": "Yer Tutucu", | ||||
| @@ -13,22 +13,22 @@ | ||||
|     "and": "ve", | ||||
|     "githubPATLabel": "GitHub Kişisel Erişim Anahtarı (Sınırlamayı Artırır)", | ||||
|     "includePrereleases": "Ön sürümleri dahil et", | ||||
|     "fallbackToOlderReleases": "Daha eski sürümlere geri dön", | ||||
|     "fallbackToOlderReleases": "Daha eski sürümleri alternatif olarak tut", | ||||
|     "filterReleaseTitlesByRegEx": "Düzenli İfadelerle Sürüm Başlıklarını Filtrele", | ||||
|     "invalidRegEx": "Geçersiz düzenli ifade", | ||||
|     "noDescription": "Açıklama yok", | ||||
|     "cancel": "İptal", | ||||
|     "continue": "Devam Et", | ||||
|     "requiredInBrackets": "(Gerekli)", | ||||
|     "dropdownNoOptsError": "HATA: DİPLOMADA EN AZ BİR SEÇENEK OLMALI", | ||||
|     "dropdownNoOptsError": "HATA: AÇILIR MENÜDE EN AZ BİR SEÇENEK OLMALI", | ||||
|     "colour": "Renk", | ||||
|     "standard": "Standart", | ||||
|     "custom": "Özel", | ||||
|     "useMaterialYou": "Sizin Malzemenizi Kullanın", | ||||
|     "githubStarredRepos": "GitHub'a Yıldızlı Depolar", | ||||
|     "useMaterialYou": "MaterialYou Kullanın", | ||||
|     "githubStarredRepos": "GitHub Yıldızlı Depolar", | ||||
|     "uname": "Kullanıcı Adı", | ||||
|     "wrongArgNum": "Hatalı argüman sayısı sağlandı", | ||||
|     "xIsTrackOnly": "{} yalnızca Takip Edilen", | ||||
|     "wrongArgNum": "Hatalı sayıda argüman sağlandı", | ||||
|     "xIsTrackOnly": "{} yalnızca Takip Ediliyor", | ||||
|     "source": "Kaynak", | ||||
|     "app": "Uygulama", | ||||
|     "appsFromSourceAreTrackOnly": "Bu kaynaktan gelen uygulamalar 'Yalnızca Takip Edilen'dir.", | ||||
| @@ -41,9 +41,9 @@ | ||||
|     "appSourceURL": "Uygulama Kaynak URL'si", | ||||
|     "error": "Hata", | ||||
|     "add": "Ekle", | ||||
|     "searchSomeSourcesLabel": "Ara (Bazı Kaynaklar Yalnızca)", | ||||
|     "searchSomeSourcesLabel": "Ara (Yalnızca Bazı Kaynaklar)", | ||||
|     "search": "Ara", | ||||
|     "additionalOptsFor": "{} İçin Ek Seçenekler", | ||||
|     "additionalOptsFor": "{} için Ek Seçenekler", | ||||
|     "supportedSources": "Desteklenen Kaynaklar", | ||||
|     "trackOnlyInBrackets": "(Yalnızca Takip)", | ||||
|     "searchableInBrackets": "(Aranabilir)", | ||||
| @@ -173,13 +173,13 @@ | ||||
|     "lastUpdateCheckX": "Son Güncelleme Kontrolü: {}", | ||||
|     "remove": "Kaldır", | ||||
|     "yesMarkUpdated": "Evet, Güncellendi olarak İşaretle", | ||||
|     "fdroid": "F-Droid Resmi", | ||||
|     "fdroid": "Resmi F-Droid", | ||||
|     "appIdOrName": "Uygulama Kimliği veya Adı", | ||||
|     "appId": "Uygulama Kimliği", | ||||
|     "appWithIdOrNameNotFound": "Bu kimlik veya ada sahip bir uygulama bulunamadı", | ||||
|     "reposHaveMultipleApps": "Depolar birden fazla uygulama içerebilir", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Üçüncü Taraf Depo", | ||||
|     "steamMobile": "Buhar Mobil", | ||||
|     "fdroidThirdPartyRepo": "F-Droid Üçüncü Parti Depo", | ||||
|     "steamMobile": "Steam Mobil", | ||||
|     "steamChat": "Steam Sohbet", | ||||
|     "install": "Yükle", | ||||
|     "markInstalled": "Yüklendi olarak İşaretle", | ||||
| @@ -222,7 +222,7 @@ | ||||
|     "standardVersionDetection": "Standart sürüm tespiti", | ||||
|     "groupByCategory": "Kategoriye Göre Grupla", | ||||
|     "autoApkFilterByArch": "Mümkünse APK'leri CPU mimarisi ile filtreleme girişimi", | ||||
|     "overrideSource": "Kaynağı Geçersiz Kıl", | ||||
|     "overrideSource": "Öncelenecek Kaynak", | ||||
|     "dontShowAgain": "Bunu tekrar gösterme", | ||||
|     "dontShowTrackOnlyWarnings": "'Yalnızca Takip Edilen' uyarılarını gösterme", | ||||
|     "dontShowAPKOriginWarnings": "APK kaynağı uyarılarını gösterme", | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "WiFi olmadığında arka plan güncellemelerini devre dışı bırak", | ||||
|     "autoSelectHighestVersionCode": "Otomatik olarak en yüksek sürüm kodunu seç", | ||||
|     "versionExtractionRegEx": "Sürüm Çıkarma Düzenli İfade", | ||||
|     "trimVersionString": "RegEx ile Sürüm Dizesini Kırpma", | ||||
|     "matchGroupToUseForX": "\"{}\" için Kullanılacak Grubu Eşleştirin", | ||||
|     "matchGroupToUse": "Sürüm Çıkarma Regex için Kullanılacak Eşleşme Grubu", | ||||
|     "highlightTouchTargets": "Daha az belirgin dokunma hedeflerini vurgula", | ||||
|     "pickExportDir": "Dışa Aktarılacak Klasörü Seç", | ||||
| @@ -286,10 +288,10 @@ | ||||
|     "checkingForUpdatesNotifChannel": "Güncellemeler Kontrol Ediliyor", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Yalnızca yüklü ve Yalnızca İzleme Uygulamalarını güncelleme", | ||||
|     "supportFixedAPKURL": "Sabit APK URL'lerini destekleyin", | ||||
|     "selectX": "Seçme {}", | ||||
|     "selectX": "{} Tanesini Seç", | ||||
|     "parallelDownloads": "Paralel indirmelere izin ver", | ||||
|     "useShizuku": "Yüklemek için Shizuku veya Sui'yi kullanın", | ||||
|     "shizukuBinderNotFound": "Shizuku is not running", | ||||
|     "shizukuBinderNotFound": "Shizuku servisi çalışmıyor", | ||||
|     "shizukuOld": "Eski Shizuku sürümü (<11) - güncelleyin", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku ADB ile Android < 8.1 üzerinde çalışıyor - Android'i güncelleyin veya bunun yerine Sui kullanın", | ||||
|     "shizukuPretendToBeGooglePlay": "Google Play'i yükleme kaynağı olarak ayarlayın (Shizuku kullanılıyorsa)", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "AppVerifier ile paylaşın, hazır olduğunuzda buraya dönün.", | ||||
|     "wiki": "Yardım/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Kitle Kaynaklı Uygulama Yapılandırmaları (riski size ait olmak üzere kullanın)", | ||||
|     "allowInsecure": "Güvensiz HTTP isteklerine izin ver", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Uygulamayı Kaldır?", | ||||
|         "other": "Uygulamaları Kaldır?" | ||||
| @@ -334,7 +337,7 @@ | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} URL", | ||||
|         "other": "{} URL'ler" | ||||
|         "other": "{} URL" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} Dakika", | ||||
| @@ -370,6 +373,6 @@ | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} APK", | ||||
|         "other": "{} APK'lar" | ||||
|         "other": "{} APK" | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Вимкнути фонові оновлення поза Wi-Fi", | ||||
|     "autoSelectHighestVersionCode": "Автоматичний вибір APK з найвищим кодом версії", | ||||
|     "versionExtractionRegEx": "Регулярний вираз для вилучення рядка версії", | ||||
|     "trimVersionString": "Обрізати рядок версії за допомогою RegEx", | ||||
|     "matchGroupToUseForX": "Група збігів для \"{}\"", | ||||
|     "matchGroupToUse": "Група співпадінь для використання в регулярному виразі вилучення версії", | ||||
|     "highlightTouchTargets": "Підсвічувати менш очевидні області дотику", | ||||
|     "pickExportDir": "Вибрати каталог експорту", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "Надішліть на AppVerifier, а потім поверніться сюди, коли будете готові.", | ||||
|     "wiki": "Довідка/Вікі", | ||||
|     "crowdsourcedConfigsLabel": "Краудсорсингові конфігурації додатків (використовуйте на свій страх і ризик)", | ||||
|     "allowInsecure": "Дозволити незахищені HTTP-запити", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Видалити застосунок?", | ||||
|         "other": "Видалити застосунки?" | ||||
|   | ||||
| @@ -22,9 +22,9 @@ | ||||
|     "requiredInBrackets": "(Yêu cầu)", | ||||
|     "dropdownNoOptsError": "LỖI: TẢI XUỐNG PHẢI CÓ ÍT NHẤT MỘT LỰA CHỌN", | ||||
|     "colour": "Màu sắc", | ||||
|     "standard": "Standard", | ||||
|     "custom": "Custom", | ||||
|     "useMaterialYou": "Use Material You", | ||||
|     "standard": "Mặc định", | ||||
|     "custom": "Tùy chỉnh", | ||||
|     "useMaterialYou": "Sử dụng Material You", | ||||
|     "githubStarredRepos": "Kho lưu trữ có gắn dấu sao GitHub", | ||||
|     "uname": "Tên người dùng", | ||||
|     "wrongArgNum": "Số lượng đối số được cung cấp sai", | ||||
| @@ -147,10 +147,10 @@ | ||||
|     "noNewUpdates": "Không có bản cập nhật mới.", | ||||
|     "xHasAnUpdate": "{} có bản cập nhật.", | ||||
|     "appsUpdated": "Ứng dụng đã cập nhật ", | ||||
|     "appsNotUpdated": "Failed to update applications", | ||||
|     "appsNotUpdated": "Ứng dụng đã cập nhật không thành công", | ||||
|     "appsUpdatedNotifDescription": "Thông báo cho người dùng rằng các bản cập nhật cho một hoặc nhiều Ứng dụng đã được áp dụng trong nền", | ||||
|     "xWasUpdatedToY": "{} đã được cập nhật thành {}.", | ||||
|     "xWasNotUpdatedToY": "Failed to update {} to {}.", | ||||
|     "xWasNotUpdatedToY": "{} đã cập nhật thành {} không thành công.", | ||||
|     "errorCheckingUpdates": "Lỗi kiểm tra bản cập nhật", | ||||
|     "errorCheckingUpdatesNotifDescription": "Thông báo hiển thị khi kiểm tra cập nhật nền không thành công", | ||||
|     "appsRemoved": "Ứng dụng đã loại bỏ", | ||||
| @@ -189,8 +189,8 @@ | ||||
|     "disableVersionDetection": "Tắt tính năng phát hiện phiên bản", | ||||
|     "noVersionDetectionExplanation": "Chỉ nên sử dụng tùy chọn này cho Ứng dụng mà tính năng phát hiện phiên bản không hoạt động chính xác.", | ||||
|     "downloadingX": "Đang tải xuống {}", | ||||
|     "downloadX": "Download {}", | ||||
|     "downloadedX": "Downloaded {}", | ||||
|     "downloadX": "Tải xuống {}", | ||||
|     "downloadedX": "Đã tải xuống {}", | ||||
|     "releaseAsset": "Release Asset", | ||||
|     "downloadNotifDescription": "Thông báo cho người dùng về tiến trình tải xuống Ứng dụng", | ||||
|     "noAPKFound": "Không tìm thấy APK", | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "Tắt cập nhật nền khi không có WiFi", | ||||
|     "autoSelectHighestVersionCode": "Tự động chọn APK mã phiên bản cao nhất", | ||||
|     "versionExtractionRegEx": "Trích xuất phiên bản RegEx", | ||||
|     "trimVersionString": "Trim Version String With RegEx", | ||||
|     "matchGroupToUseForX": "Match Group to Use for \"{}\"", | ||||
|     "matchGroupToUse": "Nhóm đối sánh để sử dụng cho Regex trích xuất phiên bản", | ||||
|     "highlightTouchTargets": "Đánh dấu các mục tiêu cảm ứng ít rõ ràng hơn", | ||||
|     "pickExportDir": "Chọn thư mục xuất", | ||||
| @@ -288,10 +290,10 @@ | ||||
|     "supportFixedAPKURL": "Hỗ trợ URL APK cố định", | ||||
|     "selectX": "Lựa chọn {}", | ||||
|     "parallelDownloads": "Cho phép tải đa luồng", | ||||
|     "useShizuku": "Use Shizuku or Sui to install", | ||||
|     "useShizuku": "Sử dụng Shizuku hoặc Sui để cài đặt", | ||||
|     "shizukuBinderNotFound": "Shizuku chưa khởi động", | ||||
|     "shizukuOld": "Old Shizuku version (<11) - update it", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku running on Android < 8.1 with ADB - update Android or use Sui instead", | ||||
|     "shizukuOld": "Phiên bản Shizuku lỗi thời (<11) - hãy cập nhật nó", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku chạy trên Android < 8.1 với ADB - hãy cập nhật Android hoặc thay bằng Sui", | ||||
|     "shizukuPretendToBeGooglePlay": "Set Google Play as the installation source (if Shizuku is used)", | ||||
|     "useSystemFont": "Sử dụng phông chữ hệ thống", | ||||
|     "useVersionCodeAsOSVersion": "Sử dụng Mã phiên bản ứng dụng làm phiên bản do hệ điều hành phát hiện", | ||||
| @@ -310,8 +312,9 @@ | ||||
|     "badDownload": "Không thể phân tích cú pháp APK (tải xuống một phần hoặc không tương thích)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "Chia sẻ ứng dụng mới với AppVerifier (nếu có)", | ||||
|     "appVerifierInstructionToast": "Chia sẻ lên AppVerifier, sau đó quay lại đây khi sẵn sàng.", | ||||
|     "wiki": "Help/Wiki", | ||||
|     "wiki": "Trợ giúp/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "Gỡ ứng dụng?", | ||||
|         "other": "Gỡ ứng dụng?" | ||||
| @@ -361,8 +364,8 @@ | ||||
|         "other": "{} và {} ứng dụng khác đã được cập nhật." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "Failed to update {} and 1 more app.", | ||||
|         "other": "Failed to update {} and {} more apps." | ||||
|         "one": "{} và 1 ứng dụng khác đã cập nhật không thành công.", | ||||
|         "other": "{} và {} ứng dụng khác đã cập nhật không thảnh công." | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} và 1 ứng dụng khác có thể đã được cập nhật.", | ||||
|   | ||||
							
								
								
									
										378
									
								
								assets/translations/zh-Hant-TW.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										378
									
								
								assets/translations/zh-Hant-TW.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,378 @@ | ||||
| { | ||||
|     "invalidURLForSource": "不是有效的 {} 應用程式 URL", | ||||
|     "noReleaseFound": "找不到合適的版本", | ||||
|     "noVersionFound": "無法確定版本", | ||||
|     "urlMatchesNoSource": "URL 不符合已知來源", | ||||
|     "cantInstallOlderVersion": "無法安裝舊版本的應用程式", | ||||
|     "appIdMismatch": "下載的套件 ID 與現有的應用程式 ID 不相符", | ||||
|     "functionNotImplemented": "此類別尚未實作此功能", | ||||
|     "placeholder": "佔位字串", | ||||
|     "someErrors": "發生了一些錯誤", | ||||
|     "unexpectedError": "意外錯誤", | ||||
|     "ok": "確定", | ||||
|     "and": "和", | ||||
|     "githubPATLabel": "GitHub 個人存取權杖(放寬速率限制)", | ||||
|     "includePrereleases": "包含預先釋出版本", | ||||
|     "fallbackToOlderReleases": "回退到舊版本", | ||||
|     "filterReleaseTitlesByRegEx": "用正則表達式過濾版本發佈標題", | ||||
|     "invalidRegEx": "無效的正則表達式", | ||||
|     "noDescription": "沒有描述", | ||||
|     "cancel": "取消", | ||||
|     "continue": "繼續", | ||||
|     "requiredInBrackets": "(必填)", | ||||
|     "dropdownNoOptsError": "錯誤:下拉選單必須至少有一個選項", | ||||
|     "colour": "顏色", | ||||
|     "standard": "標準", | ||||
|     "custom": "自訂", | ||||
|     "useMaterialYou": "使用 Material You", | ||||
|     "githubStarredRepos": "GitHub 打星星的專案", | ||||
|     "uname": "使用者名稱", | ||||
|     "wrongArgNum": "提供的參數數量錯誤", | ||||
|     "xIsTrackOnly": "{} 是僅追蹤", | ||||
|     "source": "來源", | ||||
|     "app": "應用程式", | ||||
|     "appsFromSourceAreTrackOnly": "來自此來源的應用程式是「僅追蹤」。", | ||||
|     "youPickedTrackOnly": "您已選擇「僅追蹤」選項。", | ||||
|     "trackOnlyAppDescription": "該應用程式將被追蹤更新,但 Obtainium 將無法下載或安裝它。", | ||||
|     "cancelled": "已取消", | ||||
|     "appAlreadyAdded": "應用程式已新增", | ||||
|     "alreadyUpToDateQuestion": "應用程式已經是最新的?", | ||||
|     "addApp": "新增應用程式", | ||||
|     "appSourceURL": "應用程式來源 URL", | ||||
|     "error": "錯誤", | ||||
|     "add": "新增", | ||||
|     "searchSomeSourcesLabel": "搜尋(僅限部分來源)", | ||||
|     "search": "搜尋", | ||||
|     "additionalOptsFor": "{} 的其他選項", | ||||
|     "supportedSources": "支援的來源", | ||||
|     "trackOnlyInBrackets": "(僅追蹤)", | ||||
|     "searchableInBrackets": "(可搜尋)", | ||||
|     "appsString": "應用程式", | ||||
|     "noApps": "無應用程式", | ||||
|     "noAppsForFilter": "無符合過濾條件的應用程式", | ||||
|     "byX": "由 {}", | ||||
|     "percentProgress": "進度:{}%", | ||||
|     "pleaseWait": "請稍候", | ||||
|     "updateAvailable": "有可用的更新", | ||||
|     "notInstalled": "未安裝", | ||||
|     "pseudoVersion": "偽版本", | ||||
|     "selectAll": "全選", | ||||
|     "deselectX": "取消選取 {}", | ||||
|     "xWillBeRemovedButRemainInstalled": "{} 將從 Obtainium 中移除,但仍然安裝在裝置上。", | ||||
|     "removeSelectedAppsQuestion": "移除選取的應用程式?", | ||||
|     "removeSelectedApps": "移除選取的應用程式", | ||||
|     "updateX": "更新 {}", | ||||
|     "installX": "安裝 {}", | ||||
|     "markXTrackOnlyAsUpdated": "標記 {}\n(僅追蹤)\n為已更新", | ||||
|     "changeX": "更改 {}", | ||||
|     "installUpdateApps": "安裝/更新應用程式", | ||||
|     "installUpdateSelectedApps": "安裝/更新選取的應用程式", | ||||
|     "markXSelectedAppsAsUpdated": "標記 {} 個選取的應用程式為已更新?", | ||||
|     "no": "否", | ||||
|     "yes": "是", | ||||
|     "markSelectedAppsUpdated": "標記選取的應用程式為已更新", | ||||
|     "pinToTop": "釘選到頂端", | ||||
|     "unpinFromTop": "取消釘選", | ||||
|     "resetInstallStatusForSelectedAppsQuestion": "重設選取應用程式的安裝狀態?", | ||||
|     "installStatusOfXWillBeResetExplanation": "任何選取應用程式的安裝狀態將被重設。\n\n這可以在由於更新失敗或其他問題導致 Obtainium 顯示的應用程式版本不正確時有所幫助。", | ||||
|     "customLinkMessage": "這些連結適用於已安裝 Obtainium 的裝置", | ||||
|     "shareAppConfigLinks": "分享應用程式設定為 HTML 連結", | ||||
|     "shareSelectedAppURLs": "分享選取的應用程式 URL", | ||||
|     "resetInstallStatus": "重設安裝狀態", | ||||
|     "more": "更多", | ||||
|     "removeOutdatedFilter": "移除過時應用程式過濾", | ||||
|     "showOutdatedOnly": "僅顯示過時的應用程式", | ||||
|     "filter": "過濾", | ||||
|     "filterApps": "過濾應用程式", | ||||
|     "appName": "應用程式名稱", | ||||
|     "author": "作者", | ||||
|     "upToDateApps": "最新的應用程式", | ||||
|     "nonInstalledApps": "未安裝的應用程式", | ||||
|     "importExport": "匯入/匯出", | ||||
|     "settings": "設定", | ||||
|     "exportedTo": "匯出到 {}", | ||||
|     "obtainiumExport": "Obtainium 匯出", | ||||
|     "invalidInput": "無效的輸入", | ||||
|     "importedX": "已匯入 {}", | ||||
|     "obtainiumImport": "Obtainium 匯入", | ||||
|     "importFromURLList": "從 URL 清單匯入", | ||||
|     "searchQuery": "搜尋查詢", | ||||
|     "appURLList": "應用程式 URL 清單", | ||||
|     "line": "行", | ||||
|     "searchX": "搜尋 {}", | ||||
|     "noResults": "沒有找到結果", | ||||
|     "importX": "匯入 {}", | ||||
|     "importedAppsIdDisclaimer": "匯入的應用程式可能會錯誤地顯示為「未安裝」。\n要修正此問題,請透過 Obtainium 重新安裝它們。\n這不應該影響應用程式資料。\n\n僅影響 URL 和第三方匯入方法。", | ||||
|     "importErrors": "匯入錯誤", | ||||
|     "importedXOfYApps": "已匯入 {} 個中的 {} 個應用程式。", | ||||
|     "followingURLsHadErrors": "以下 URL 有錯誤:", | ||||
|     "selectURL": "選擇 URL", | ||||
|     "selectURLs": "選擇多個 URL", | ||||
|     "pick": "選取", | ||||
|     "theme": "主題", | ||||
|     "dark": "深色", | ||||
|     "light": "淺色", | ||||
|     "followSystem": "跟隨系統", | ||||
|     "followSystemThemeExplanation": "僅使用第三方應用程式時才可跟隨系統主題", | ||||
|     "useBlackTheme": "使用純黑色深色主題", | ||||
|     "appSortBy": "應用程式排序依據", | ||||
|     "authorName": "作者/名稱", | ||||
|     "nameAuthor": "名稱/作者", | ||||
|     "asAdded": "新增順序", | ||||
|     "appSortOrder": "應用程式排序順序", | ||||
|     "ascending": "升序", | ||||
|     "descending": "降序", | ||||
|     "bgUpdateCheckInterval": "背景更新檢查間隔", | ||||
|     "neverManualOnly": "從不 - 僅手動", | ||||
|     "appearance": "外觀", | ||||
|     "showWebInAppView": "在應用程式檢視中顯示來源網頁", | ||||
|     "pinUpdates": "將更新釘選至應用程式檢視的頂端", | ||||
|     "updates": "更新", | ||||
|     "sourceSpecific": "特定來源", | ||||
|     "appSource": "應用程式來源", | ||||
|     "noLogs": "無日誌", | ||||
|     "appLogs": "應用程式日誌", | ||||
|     "close": "關閉", | ||||
|     "share": "分享", | ||||
|     "appNotFound": "沒有找到應用程式", | ||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||
|     "pickAnAPK": "選擇一個 APK", | ||||
|     "appHasMoreThanOnePackage": "{} 有多個套件:", | ||||
|     "deviceSupportsXArch": "您的裝置支援 {} CPU 架構。", | ||||
|     "deviceSupportsFollowingArchs": "您的裝置支援以下 CPU 架構:", | ||||
|     "warning": "警告", | ||||
|     "sourceIsXButPackageFromYPrompt": "應用程式來源是 「{}」,但發佈套件來自 「{}」。要繼續嗎?", | ||||
|     "updatesAvailable": "有可用的更新", | ||||
|     "updatesAvailableNotifDescription": "通知使用者有一個或多個由 Obtainium 追蹤的應用程式有更新", | ||||
|     "noNewUpdates": "沒有新更新。", | ||||
|     "xHasAnUpdate": "{} 有一個更新。", | ||||
|     "appsUpdated": "應用程式已更新", | ||||
|     "appsNotUpdated": "未能更新應用程式", | ||||
|     "appsUpdatedNotifDescription": "通知使用者一個或多個應用程式的更新已在背景中套用", | ||||
|     "xWasUpdatedToY": "{} 已更新到 {}。", | ||||
|     "xWasNotUpdatedToY": "未能將 {} 更新到 {}。", | ||||
|     "errorCheckingUpdates": "檢查更新時出錯", | ||||
|     "errorCheckingUpdatesNotifDescription": "背景檢查更新失敗時顯示的通知", | ||||
|     "appsRemoved": "應用程式已移除", | ||||
|     "appsRemovedNotifDescription": "通知使用者由於載入時出錯,一個或多個應用程式已被移除", | ||||
|     "xWasRemovedDueToErrorY": "{} 已因以下錯誤被移除:{}", | ||||
|     "completeAppInstallation": "完成應用程式安裝", | ||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium 必須開啟才能安裝應用程式", | ||||
|     "completeAppInstallationNotifDescription": "請使用者回到 Obtainium 以完成應用程式安裝", | ||||
|     "checkingForUpdates": "正在檢查更新", | ||||
|     "checkingForUpdatesNotifDescription": "檢查更新時顯示的暫時性通知", | ||||
|     "pleaseAllowInstallPerm": "請允許 Obtainium 安裝應用程式", | ||||
|     "trackOnly": "僅追蹤", | ||||
|     "errorWithHttpStatusCode": "錯誤 {}", | ||||
|     "versionCorrectionDisabled": "版本校正已停用(外掛程式似乎無法正常工作)", | ||||
|     "unknown": "未知", | ||||
|     "none": "無", | ||||
|     "never": "從不", | ||||
|     "latestVersionX": "最新版本:{}", | ||||
|     "installedVersionX": "已安裝版本:{}", | ||||
|     "lastUpdateCheckX": "上次檢查更新時間:{}", | ||||
|     "remove": "移除", | ||||
|     "yesMarkUpdated": "是,標記為已更新", | ||||
|     "fdroid": "F-Droid 官方", | ||||
|     "appIdOrName": "應用程式 ID 或名稱", | ||||
|     "appId": "應用程式 ID", | ||||
|     "appWithIdOrNameNotFound": "找不到具有該 ID 或名稱的應用程式", | ||||
|     "reposHaveMultipleApps": "倉庫可能包含多個應用程式", | ||||
|     "fdroidThirdPartyRepo": "F-Droid 第三方倉庫", | ||||
|     "steamMobile": "Steam 行動版", | ||||
|     "steamChat": "Steam 聊天", | ||||
|     "install": "安裝", | ||||
|     "markInstalled": "標記為已安裝", | ||||
|     "update": "更新", | ||||
|     "markUpdated": "標記為已更新", | ||||
|     "additionalOptions": "額外選項", | ||||
|     "disableVersionDetection": "停用版本偵測", | ||||
|     "noVersionDetectionExplanation": "此選項僅應用於版本偵測無法正確工作的應用程式。", | ||||
|     "downloadingX": "正在下載 {}", | ||||
|     "downloadX": "下載 {}", | ||||
|     "downloadedX": "已下載 {}", | ||||
|     "releaseAsset": "發佈資源", | ||||
|     "downloadNotifDescription": "通知使用者應用程式下載進度", | ||||
|     "noAPKFound": "沒有找到 APK", | ||||
|     "noVersionDetection": "無版本偵測", | ||||
|     "categorize": "分類", | ||||
|     "categories": "類別", | ||||
|     "category": "類別", | ||||
|     "noCategory": "無類別", | ||||
|     "noCategories": "無類別", | ||||
|     "deleteCategoriesQuestion": "刪除類別?", | ||||
|     "categoryDeleteWarning": "所有在已刪除類別中的應用程式將被設定為未分類。", | ||||
|     "addCategory": "新增類別", | ||||
|     "label": "標籤", | ||||
|     "language": "語言", | ||||
|     "copiedToClipboard": "已複製到剪貼簿", | ||||
|     "storagePermissionDenied": "存取權限被拒絕", | ||||
|     "selectedCategorizeWarning": "這將替換選取應用程式的任何現有類別設定。", | ||||
|     "filterAPKsByRegEx": "用正則表達式過濾 APK", | ||||
|     "removeFromObtainium": "從 Obtainium 移除", | ||||
|     "uninstallFromDevice": "從裝置解除安裝", | ||||
|     "onlyWorksWithNonVersionDetectApps": "僅適用於停用版本偵測的應用程式。", | ||||
|     "releaseDateAsVersion": "使用發佈日期作為版本字串", | ||||
|     "releaseDateAsVersionExplanation": "此選項僅應用於版本偵測無法正確工作但有發佈日期的應用程式。", | ||||
|     "changes": "變更", | ||||
|     "releaseDate": "發佈日期", | ||||
|     "importFromURLsInFile": "從檔案中的 URL 匯入(如 OPML)", | ||||
|     "versionDetectionExplanation": "將版本字串與作業系統偵測到的版本對比", | ||||
|     "versionDetection": "版本偵測", | ||||
|     "standardVersionDetection": "標準版本偵測", | ||||
|     "groupByCategory": "按類別分組", | ||||
|     "autoApkFilterByArch": "如果可能,嘗試按 CPU 架構過濾 APK", | ||||
|     "overrideSource": "覆蓋來源", | ||||
|     "dontShowAgain": "不要再顯示", | ||||
|     "dontShowTrackOnlyWarnings": "不要顯示「僅追蹤」警告", | ||||
|     "dontShowAPKOriginWarnings": "不要顯示 APK 來源警告", | ||||
|     "moveNonInstalledAppsToBottom": "將未安裝的應用程式移到應用程式檢視的末端", | ||||
|     "gitlabPATLabel": "GitLab 個人存取權杖", | ||||
|     "about": "關於", | ||||
|     "requiresCredentialsInSettings": "{} 需要額外的憑證(在設定中)", | ||||
|     "checkOnStart": "啟動時檢查更新", | ||||
|     "tryInferAppIdFromCode": "嘗試從原始碼推斷應用程式 ID", | ||||
|     "removeOnExternalUninstall": "自動移除外部解除安裝的應用程式", | ||||
|     "pickHighestVersionCode": "自動選取最高版本號的 APK", | ||||
|     "checkUpdateOnDetailPage": "在開啟應用程式詳細頁面時檢查更新", | ||||
|     "disablePageTransitions": "停用頁面過渡動畫", | ||||
|     "reversePageTransitions": "反轉頁面過渡動畫", | ||||
|     "minStarCount": "最少星星數", | ||||
|     "addInfoBelow": "在下方新增此資訊。", | ||||
|     "addInfoInSettings": "在設定中增加此資訊。", | ||||
|     "githubSourceNote": "使用 API 金鑰可以避免 GitHub 的速率限制。", | ||||
|     "sortByLastLinkSegment": "僅按連結的最後一段排序", | ||||
|     "filterReleaseNotesByRegEx": "用正則表達式過濾發佈說明", | ||||
|     "customLinkFilterRegex": "自定 APK 連結過濾正則表達式(預設為 '.apk$')", | ||||
|     "appsPossiblyUpdated": "嘗試更新應用程式", | ||||
|     "appsPossiblyUpdatedNotifDescription": "通知使用者一個或多個應用程式的更新可能已在背景中套用", | ||||
|     "xWasPossiblyUpdatedToY": "{} 可能已更新到 {}。", | ||||
|     "enableBackgroundUpdates": "啟用背景更新", | ||||
|     "backgroundUpdateReqsExplanation": "並非所有應用程式都能進行背景更新。", | ||||
|     "backgroundUpdateLimitsExplanation": "背景安裝的成功與否只能在開啟 Obtainium 時確定。", | ||||
|     "verifyLatestTag": "驗證「最新」標籤", | ||||
|     "intermediateLinkRegex": "過濾要存取的「中間」連結", | ||||
|     "filterByLinkText": "按連結文字過濾連結", | ||||
|     "intermediateLinkNotFound": "沒有找到中間連結", | ||||
|     "intermediateLink": "中間連結", | ||||
|     "exemptFromBackgroundUpdates": "免除背景更新(若已啟用)", | ||||
|     "bgUpdatesOnWiFiOnly": "停用非 WiFi 的背景更新", | ||||
|     "autoSelectHighestVersionCode": "自動選擇最高 versionCode 的 APK", | ||||
|     "versionExtractionRegEx": "版本字串提取正則表達式", | ||||
|     "trimVersionString": "用正則表達式修剪版本字串", | ||||
|     "matchGroupToUseForX": "用於「{}」的對應群組", | ||||
|     "matchGroupToUse": "要用於版本字串提取的對應群組", | ||||
|     "highlightTouchTargets": "突出顯示不明顯的觸控目標", | ||||
|     "pickExportDir": "選擇匯出目錄", | ||||
|     "autoExportOnChanges": "更改時自動匯出", | ||||
|     "includeSettings": "包含設定", | ||||
|     "filterVersionsByRegEx": "用正則表達式過濾版本", | ||||
|     "trySelectingSuggestedVersionCode": "嘗試選擇建議的 versionCode APK", | ||||
|     "dontSortReleasesList": "保留 API 的發佈順序", | ||||
|     "reverseSort": "反向排序", | ||||
|     "takeFirstLink": "使用第一個連結", | ||||
|     "skipSort": "跳過排序", | ||||
|     "debugMenu": "除錯選單", | ||||
|     "bgTaskStarted": "背景任務已啟動 - 檢查日誌。", | ||||
|     "runBgCheckNow": "立即執行背景更新檢查", | ||||
|     "versionExtractWholePage": "將版本字串提取正則表達式套用於整個頁面", | ||||
|     "installing": "正在安裝", | ||||
|     "skipUpdateNotifications": "跳過更新通知", | ||||
|     "updatesAvailableNotifChannel": "有可用的更新", | ||||
|     "appsUpdatedNotifChannel": "應用程式已更新", | ||||
|     "appsPossiblyUpdatedNotifChannel": "嘗試更新應用程式", | ||||
|     "errorCheckingUpdatesNotifChannel": "檢查更新錯誤", | ||||
|     "appsRemovedNotifChannel": "應用程式已移除", | ||||
|     "downloadingXNotifChannel": "正在下載 {}", | ||||
|     "completeAppInstallationNotifChannel": "完成應用程式安裝", | ||||
|     "checkingForUpdatesNotifChannel": "正在檢查更新", | ||||
|     "onlyCheckInstalledOrTrackOnlyApps": "僅檢查已安裝和僅追蹤的應用程式更新", | ||||
|     "supportFixedAPKURL": "支援固定的 APK 網址", | ||||
|     "selectX": "選擇 {}", | ||||
|     "parallelDownloads": "允許平行下載", | ||||
|     "useShizuku": "使用 Shizuku 或 Sui 來安裝", | ||||
|     "shizukuBinderNotFound": "Shizuku 服務未運作", | ||||
|     "shizukuOld": "舊版 Shizuku (<11) - 請更新", | ||||
|     "shizukuOldAndroidWithADB": "Shizuku 在 Android 8.1 以下版本使用 ADB 執行 - 請更新 Android 或改用 Sui", | ||||
|     "shizukuPretendToBeGooglePlay": "設定 Google Play 為安裝來源(如果使用 Shizuku)", | ||||
|     "useSystemFont": "使用系統字型", | ||||
|     "useVersionCodeAsOSVersion": "使用應用程式 versionCode 作為作業系統偵測的版本", | ||||
|     "requestHeader": "請求標頭", | ||||
|     "useLatestAssetDateAsReleaseDate": "使用最新資源上傳日期作為發佈日期", | ||||
|     "defaultPseudoVersioningMethod": "預設偽版本管理方法", | ||||
|     "partialAPKHash": "部分 APK Hash", | ||||
|     "APKLinkHash": "APK 連結 Hash", | ||||
|     "directAPKLink": "直接 APK 連結", | ||||
|     "pseudoVersionInUse": "正在使用偽版本", | ||||
|     "installed": "已安裝", | ||||
|     "latest": "最新", | ||||
|     "invertRegEx": "反轉正則表達式", | ||||
|     "note": "備註", | ||||
|     "selfHostedNote": "可使用「{}」下拉選單來存取任何來源的自行託管/自訂執行個體。", | ||||
|     "badDownload": "無法解析 APK(不相容或下載不完整)", | ||||
|     "beforeNewInstallsShareToAppVerifier": "將新應用程式分享到 AppVerifier(如果可用)", | ||||
|     "appVerifierInstructionToast": "分享至 AppVerifier,然後準備好時回到此處。", | ||||
|     "wiki": "幫助/維基", | ||||
|     "crowdsourcedConfigsLabel": "群眾外包的應用程式設定(使用風險自負)", | ||||
|     "allowInsecure": "Allow insecure HTTP requests", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "移除應用程式?", | ||||
|         "other": "移除應用程式?" | ||||
|     }, | ||||
|     "tooManyRequestsTryAgainInMinutes": { | ||||
|         "one": "請求過多(速率限制)- {} 分鐘後重試", | ||||
|         "other": "請求過多(速率限制)- {} 分鐘後重試" | ||||
|     }, | ||||
|     "bgUpdateGotErrorRetryInMinutes": { | ||||
|         "one": "背景更新檢查遇到 {},將在 {} 分鐘後重新檢查", | ||||
|         "other": "背景更新檢查遇到 {},將在 {} 分鐘後重新檢查" | ||||
|     }, | ||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||
|         "one": "背景更新檢查發現 {} 個更新 - 如果需要將通知使用者", | ||||
|         "other": "背景更新檢查發現 {} 個更新 - 如果需要將通知使用者" | ||||
|     }, | ||||
|     "apps": { | ||||
|         "one": "{} 個應用程式", | ||||
|         "other": "{} 個應用程式" | ||||
|     }, | ||||
|     "url": { | ||||
|         "one": "{} 個 URL", | ||||
|         "other": "{} 個 URL" | ||||
|     }, | ||||
|     "minute": { | ||||
|         "one": "{} 分鐘", | ||||
|         "other": "{} 分鐘" | ||||
|     }, | ||||
|     "hour": { | ||||
|         "one": "{} 小時", | ||||
|         "other": "{} 小時" | ||||
|     }, | ||||
|     "day": { | ||||
|         "one": "{} 天", | ||||
|         "other": "{} 天" | ||||
|     }, | ||||
|     "clearedNLogsBeforeXAfterY": { | ||||
|         "one": "清除 {n} 個日誌(之前 = {before},之後 = {after})", | ||||
|         "other": "清除 {n} 個日誌(之前 = {before},之後 = {after})" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesAvailable": { | ||||
|         "one": "{} 和另外 1 個應用程式有更新。", | ||||
|         "other": "{} 和另外 {} 個應用程式有更新。" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesInstalled": { | ||||
|         "one": "{} 和另外 1 個應用程式已更新。", | ||||
|         "other": "{} 和另外 {} 個應用程式已更新。" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesFailed": { | ||||
|         "one": "更新 {} 和另外 1 個應用程式失敗。", | ||||
|         "other": "更新 {} 和另外 {} 個應用程式失敗。" | ||||
|     }, | ||||
|     "xAndNMoreUpdatesPossiblyInstalled": { | ||||
|         "one": "{} 和另外 1 個應用程式可能已更新。", | ||||
|         "other": "{} 和另外 {} 個應用程式可能已更新。" | ||||
|     }, | ||||
|     "apk": { | ||||
|         "one": "{} 個 APK", | ||||
|         "other": "{} 個 APK" | ||||
|     } | ||||
| } | ||||
| @@ -259,6 +259,8 @@ | ||||
|     "bgUpdatesOnWiFiOnly": "未连接 Wi-Fi 时禁用后台更新", | ||||
|     "autoSelectHighestVersionCode": "自动选择内部版本号最高的 APK 文件", | ||||
|     "versionExtractionRegEx": "提取版本号的正则表达式", | ||||
|     "trimVersionString": "使用 RegEx 修剪版本字符串", | ||||
|     "matchGroupToUseForX": "用于\"{}\"的匹配组", | ||||
|     "matchGroupToUse": "从上述匹配结果中引用的捕获组", | ||||
|     "highlightTouchTargets": "突出展示不明显的可交互区域", | ||||
|     "pickExportDir": "选择导出文件夹", | ||||
| @@ -312,6 +314,7 @@ | ||||
|     "appVerifierInstructionToast": "分享至 AppVerifier,完成后返回此处。", | ||||
|     "wiki": "帮助/Wiki", | ||||
|     "crowdsourcedConfigsLabel": "众包应用程序配置(使用风险自负)", | ||||
|     "allowInsecure": "允许不安全的 HTTP 请求", | ||||
|     "removeAppQuestion": { | ||||
|         "one": "是否删除应用?", | ||||
|         "other": "是否删除应用?" | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| 		<ul> | ||||
| 			<li>GitHub</li> | ||||
| 			<li>GitLab</li> | ||||
| 			<li>Codeberg</li> | ||||
| 			<li>Forgejo (Codeberg)</li> | ||||
| 			<li>F-Droid</li> | ||||
| 			<li>Third Party F-Droid Repos</li> | ||||
| 			<li>IzzyOnDroid</li> | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| 		<ul> | ||||
| 			<li>GitHub</li> | ||||
| 			<li>GitLab</li> | ||||
| 			<li>Codeberg</li> | ||||
| 			<li>Forgejo (Codeberg)</li> | ||||
| 			<li>F-Droid</li> | ||||
| 			<li>Third Party F-Droid Repos</li> | ||||
| 			<li>IzzyOnDroid</li> | ||||
|   | ||||
| @@ -10,7 +10,7 @@ class APKCombo extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/+[^/]+/+[^/]+', | ||||
|         caseSensitive: false); | ||||
|   | ||||
| @@ -32,7 +32,7 @@ class APKMirror extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/apk/[^/]+/[^/]+', | ||||
|         caseSensitive: false); | ||||
|   | ||||
| @@ -1,5 +1,7 @@ | ||||
| import 'package:device_info_plus/device_info_plus.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:html/parser.dart'; | ||||
| import 'package:obtainium/app_sources/html.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| @@ -27,16 +29,17 @@ class APKPure extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegExB = RegExp( | ||||
|         '^https?://m.${getSourceRegex(hosts)}/+[^/]+/+[^/]+(/+[^/]+)?', | ||||
|         '^https?://m.${getSourceRegex(hosts)}(/+[^/]{2})?/+[^/]+/+[^/]+', | ||||
|         caseSensitive: false); | ||||
|     RegExpMatch? match = standardUrlRegExB.firstMatch(url); | ||||
|     if (match != null) { | ||||
|       url = 'https://${getSourceRegex(hosts)}${Uri.parse(url).path}'; | ||||
|       var uri = Uri.parse(url); | ||||
|       url = 'https://${uri.host.substring(2)}${uri.path}'; | ||||
|     } | ||||
|     RegExp standardUrlRegExA = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/+[^/]+/+[^/]+(/+[^/]+)?', | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}(/+[^/]{2})?/+[^/]+/+[^/]+', | ||||
|         caseSensitive: false); | ||||
|     match = standardUrlRegExA.firstMatch(url); | ||||
|     if (match == null) { | ||||
| @@ -58,40 +61,109 @@ class APKPure extends AppSource { | ||||
|   ) async { | ||||
|     String appId = (await tryInferringAppId(standardUrl))!; | ||||
|     String host = Uri.parse(standardUrl).host; | ||||
|     var res = await sourceRequest('$standardUrl/download', additionalSettings); | ||||
|     var resChangelog = await sourceRequest(standardUrl, additionalSettings); | ||||
|     if (res.statusCode == 200 && resChangelog.statusCode == 200) { | ||||
|       var html = parse(res.body); | ||||
|       var htmlChangelog = parse(resChangelog.body); | ||||
|       String? version = html.querySelector('span.info-sdk span')?.text.trim(); | ||||
|       if (version == null) { | ||||
|         throw NoVersionError(); | ||||
|       } | ||||
|       String? dateString = | ||||
|           html.querySelector('span.info-other span.date')?.text.trim(); | ||||
|       DateTime? releaseDate = parseDateTimeMMMddCommayyyy(dateString); | ||||
|       String type = html.querySelector('a.info-tag')?.text.trim() ?? 'APK'; | ||||
|       List<MapEntry<String, String>> apkUrls = [ | ||||
|         MapEntry('$appId.apk', | ||||
|             'https://d.${hosts.contains(host) ? 'cdnpure.com' : host}/b/$type/$appId?version=latest') | ||||
|       ]; | ||||
|       String author = html | ||||
|               .querySelector('span.info-sdk') | ||||
|               ?.text | ||||
|               .trim() | ||||
|               .substring(version.length + 4) ?? | ||||
|           Uri.parse(standardUrl).pathSegments.reversed.last; | ||||
|       String appName = | ||||
|           html.querySelector('h1.info-title')?.text.trim() ?? appId; | ||||
|       String? changeLog = htmlChangelog | ||||
|           .querySelector("div.whats-new-info p:not(.date)") | ||||
|           ?.innerHtml | ||||
|           .trim() | ||||
|           .replaceAll("<br>", "  \n"); | ||||
|       return APKDetails(version, apkUrls, AppNames(author, appName), | ||||
|           releaseDate: releaseDate, changeLog: changeLog); | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|  | ||||
|     var res0 = await sourceRequest('$standardUrl/versions', additionalSettings); | ||||
|     var versionLinks = await grabLinksCommon(res0, { | ||||
|       'skipSort': true, | ||||
|       'customLinkFilterRegex': '$standardUrl/download/[^/]+\$' | ||||
|     }); | ||||
|  | ||||
|     var supportedArchs = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||
|  | ||||
|     if (additionalSettings['autoApkFilterByArch'] != true) { | ||||
|       // No need to request multiple versions when we're not going to filter them (always pick the top one) | ||||
|       versionLinks = versionLinks.sublist(0, 1); | ||||
|     } | ||||
|     if (versionLinks.isEmpty) { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
|  | ||||
|     for (var i = 0; i < versionLinks.length; i++) { | ||||
|       var link = versionLinks[i]; | ||||
|       var res = await sourceRequest(link.key, additionalSettings); | ||||
|       if (res.statusCode == 200) { | ||||
|         var html = parse(res.body); | ||||
|         var apksDiv = | ||||
|             html.querySelector('#version-list div div.show-more-content'); | ||||
|         DateTime? topReleaseDate; | ||||
|         var apkUrls = apksDiv | ||||
|                 ?.querySelectorAll('div.group-title') | ||||
|                 .map((e) { | ||||
|                   String architectureString = e.text.trim(); | ||||
|                   if (architectureString.toLowerCase() == 'unlimited' || | ||||
|                       architectureString.toLowerCase() == 'universal') { | ||||
|                     architectureString = ''; | ||||
|                   } | ||||
|                   List<String> architectures = architectureString | ||||
|                       .split(',') | ||||
|                       .map((e) => e.trim()) | ||||
|                       .where((e) => e.isNotEmpty) | ||||
|                       .toList(); | ||||
|                   // Only take the first APK for each architecture, ignore others for now, for simplicity | ||||
|                   // Unclear why there can even be multiple APKs for the same version and arch | ||||
|                   var apkInfo = e.nextElementSibling?.querySelector('div.info'); | ||||
|                   String? versionCode = RegExp('[0-9]+') | ||||
|                       .firstMatch(apkInfo | ||||
|                               ?.querySelector('div.info-top span.code') | ||||
|                               ?.text ?? | ||||
|                           '') | ||||
|                       ?.group(0) | ||||
|                       ?.trim(); | ||||
|                   var types = apkInfo | ||||
|                           ?.querySelectorAll('div.info-top span.tag') | ||||
|                           .map((e) => e.text.trim()) | ||||
|                           .map((t) => t == 'APKs' ? 'APK' : t) ?? | ||||
|                       []; | ||||
|                   String type = types.isEmpty | ||||
|                       ? 'APK' | ||||
|                       : types.length == 1 | ||||
|                           ? types.first | ||||
|                           : types.last; | ||||
|                   String? dateString = apkInfo | ||||
|                       ?.querySelector('div.info-bottom span.time') | ||||
|                       ?.text | ||||
|                       .trim(); | ||||
|                   DateTime? releaseDate = | ||||
|                       parseDateTimeMMMddCommayyyy(dateString); | ||||
|                   if (additionalSettings['autoApkFilterByArch'] == true && | ||||
|                       architectures.isNotEmpty && | ||||
|                       architectures | ||||
|                           .where((a) => supportedArchs.contains(a)) | ||||
|                           .isEmpty) { | ||||
|                     return const MapEntry('', ''); | ||||
|                   } | ||||
|                   topReleaseDate ??= | ||||
|                       releaseDate; // Just use the release date of the first APK in the list as the release date for this version | ||||
|                   return MapEntry( | ||||
|                       '$appId-$versionCode-$architectureString.${type.toLowerCase()}', | ||||
|                       'https://d.${hosts.contains(host) ? 'cdnpure.com' : host}/b/$type/$appId?versionCode=$versionCode'); | ||||
|                 }) | ||||
|                 .where((e) => e.key.isNotEmpty) | ||||
|                 .toList() ?? | ||||
|             []; | ||||
|         if (apkUrls.isEmpty) { | ||||
|           continue; | ||||
|         } | ||||
|         String version = Uri.parse(link.key).pathSegments.last; | ||||
|         String author = html | ||||
|                 .querySelector('span.info-sdk') | ||||
|                 ?.text | ||||
|                 .trim() | ||||
|                 .substring(version.length + 4) ?? | ||||
|             Uri.parse(standardUrl).pathSegments.reversed.last; | ||||
|         String appName = | ||||
|             html.querySelector('h1.info-title')?.text.trim() ?? appId; | ||||
|         String? changeLog = html | ||||
|             .querySelector('div.module.change-log') | ||||
|             ?.innerHtml | ||||
|             .trim() | ||||
|             .replaceAll("<br>", "  \n"); | ||||
|         return APKDetails(version, apkUrls, AppNames(author, appName), | ||||
|             releaseDate: topReleaseDate, changeLog: changeLog); | ||||
|       } else { | ||||
|         throw getObtainiumHttpError(res); | ||||
|       } | ||||
|     } | ||||
|     throw NoAPKError(); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -14,7 +14,7 @@ class Aptoide extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://([^\\.]+\\.){2,}${getSourceRegex(hosts)}', | ||||
|         caseSensitive: false); | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import 'package:obtainium/providers/source_provider.dart'; | ||||
| class Codeberg extends AppSource { | ||||
|   GitHub gh = GitHub(); | ||||
|   Codeberg() { | ||||
|     name = 'Forgejo (Codeberg)'; | ||||
|     hosts = ['codeberg.org']; | ||||
|  | ||||
|     additionalSourceAppSpecificSettingFormItems = | ||||
| @@ -15,7 +16,7 @@ class Codeberg extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', | ||||
|         caseSensitive: false); | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:obtainium/app_sources/html.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class DirectAPKLink extends AppSource { | ||||
|   HTML html = HTML(); | ||||
|  | ||||
|   DirectAPKLink() { | ||||
|     neverAutoSelect = true; | ||||
|     name = tr('directAPKLink'); | ||||
|     additionalSourceAppSpecificSettingFormItems = html | ||||
|         .additionalSourceAppSpecificSettingFormItems | ||||
| @@ -24,6 +24,27 @@ class DirectAPKLink extends AppSource { | ||||
|     ]; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     if (!forSelection) { | ||||
|       return url; | ||||
|     } | ||||
|     RegExp standardUrlRegExA = RegExp('.+\\.apk\$', caseSensitive: false); | ||||
|     var match = standardUrlRegExA.firstMatch(url); | ||||
|     if (match == null) { | ||||
|       throw InvalidURLError(name); | ||||
|     } | ||||
|     return match.group(0)!; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<Map<String, String>?> getRequestHeaders( | ||||
|       Map<String, dynamic> additionalSettings, | ||||
|       {bool forAPKDownload = false}) { | ||||
|     return html.getRequestHeaders(additionalSettings, | ||||
|         forAPKDownload: forAPKDownload); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
|   | ||||
| @@ -38,7 +38,7 @@ class FDroid extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegExB = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/+[^/]+/+packages/+[^/]+', | ||||
|         caseSensitive: false); | ||||
|   | ||||
| @@ -9,7 +9,7 @@ class FDroidRepo extends AppSource { | ||||
|   FDroidRepo() { | ||||
|     name = tr('fdroidThirdPartyRepo'); | ||||
|     canSearch = true; | ||||
|     excludeFromMassSearch = true; | ||||
|     includeAdditionalOptsInMainSearch = true; | ||||
|     neverAutoSelect = true; | ||||
|     showReleaseDateAsVersionToggle = true; | ||||
|  | ||||
| @@ -43,7 +43,7 @@ class FDroidRepo extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     var standardUri = Uri.parse(url); | ||||
|     var pathSegments = standardUri.pathSegments; | ||||
|     if (pathSegments.isNotEmpty && pathSegments.last == 'index.xml') { | ||||
| @@ -86,6 +86,27 @@ class FDroidRepo extends AppSource { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void runOnAddAppInputChange(String userInput) { | ||||
|     additionalSourceAppSpecificSettingFormItems = | ||||
|         additionalSourceAppSpecificSettingFormItems.map((row) { | ||||
|       row = row.map((item) { | ||||
|         if (item.key == 'appIdOrName') { | ||||
|           try { | ||||
|             var appId = Uri.parse(userInput).queryParameters['appId']; | ||||
|             if (appId != null && item is GeneratedFormTextField) { | ||||
|               item.required = false; | ||||
|             } | ||||
|           } catch (e) { | ||||
|             // | ||||
|           } | ||||
|         } | ||||
|         return item; | ||||
|       }).toList(); | ||||
|       return row; | ||||
|     }).toList(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   App endOfGetAppChanges(App app) { | ||||
|     var uri = Uri.parse(app.url); | ||||
| @@ -142,6 +163,7 @@ class FDroidRepo extends AppSource { | ||||
|     if (appIdOrName == null) { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
|     additionalSettings['appIdOrName'] = appIdOrName; | ||||
|     var res = | ||||
|         await sourceRequestWithURLVariants(standardUrl, additionalSettings); | ||||
|     if (res.statusCode == 200) { | ||||
|   | ||||
| @@ -154,7 +154,7 @@ class GitHub extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', | ||||
|         caseSensitive: false); | ||||
| @@ -171,7 +171,7 @@ class GitHub extends AppSource { | ||||
|       {bool forAPKDownload = false}) async { | ||||
|     var token = await getTokenIfAny(additionalSettings); | ||||
|     var headers = <String, String>{}; | ||||
|     if (token != null) { | ||||
|     if (token != null && token.isNotEmpty) { | ||||
|       headers[HttpHeaders.authorizationHeader] = 'Token $token'; | ||||
|     } | ||||
|     if (forAPKDownload == true) { | ||||
| @@ -285,7 +285,9 @@ class GitHub extends AppSource { | ||||
|       DateTime? getPublishDateFromRelease(dynamic rel) => | ||||
|           rel?['published_at'] != null | ||||
|               ? DateTime.parse(rel['published_at']) | ||||
|               : null; | ||||
|               : rel?['commit']?['created'] != null | ||||
|                   ? DateTime.parse(rel['commit']['created']) | ||||
|                   : null; | ||||
|       DateTime? getNewestAssetDateFromRelease(dynamic rel) { | ||||
|         var t = (rel['assets'] as List<dynamic>?) | ||||
|             ?.map((e) { | ||||
|   | ||||
| @@ -52,7 +52,7 @@ class GitLab extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', | ||||
|         caseSensitive: false); | ||||
| @@ -111,6 +111,14 @@ class GitLab extends AppSource { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<String> apkUrlPrefetchModifier(String apkUrl, String standardUrl, | ||||
|       Map<String, dynamic> additionalSettings) async { | ||||
|     String? PAT = await getPATIfAny(hostChanged ? additionalSettings : {}); | ||||
|     String optionalAuth = (PAT != null) ? 'private_token=$PAT' : ''; | ||||
|     return '$apkUrl?$optionalAuth'; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
| @@ -123,6 +131,18 @@ class GitLab extends AppSource { | ||||
|  | ||||
|     bool trackOnly = additionalSettings['trackOnly'] == true; | ||||
|  | ||||
|     // Get project ID | ||||
|     Response res0 = await sourceRequest( | ||||
|         'https://${hosts[0]}/api/v4/projects/${names.author}%2F${names.name}?$optionalAuth', | ||||
|         additionalSettings); | ||||
|     if (res0.statusCode != 200) { | ||||
|       throw getObtainiumHttpError(res0); | ||||
|     } | ||||
|     int? projectId = jsonDecode(res0.body)['id']; | ||||
|     if (projectId == null) { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
|  | ||||
|     // Request data from REST API | ||||
|     Response res = await sourceRequest( | ||||
|         'https://${hosts[0]}/api/v4/projects/${names.author}%2F${names.name}/${trackOnly ? 'repository/tags' : 'releases'}?$optionalAuth', | ||||
| @@ -149,11 +169,12 @@ class GitLab extends AppSource { | ||||
|               .join('.apk\n') | ||||
|               .split('\n') | ||||
|               .where((s) => s.startsWith('/uploads/') && s.endsWith('apk')) | ||||
|               .map((s) => '$standardUrl$s') | ||||
|               .map((s) => 'https://${hosts[0]}/-/project/$projectId$s') | ||||
|               .toList(); | ||||
|       var apkUrlsSet = apkUrlsFromAssets.toSet(); | ||||
|       apkUrlsSet.addAll(uploadedAPKsFromDescription); | ||||
|       var releaseDateString = e['released_at'] ?? e['created_at']; | ||||
|       var releaseDateString = | ||||
|           e['released_at'] ?? e['created_at'] ?? e['commit']?['created_at']; | ||||
|       DateTime? releaseDate = | ||||
|           releaseDateString != null ? DateTime.parse(releaseDateString) : null; | ||||
|       return APKDetails( | ||||
| @@ -181,7 +202,7 @@ class GitLab extends AppSource { | ||||
|     } | ||||
|  | ||||
|     finalResult.apkUrls = finalResult.apkUrls.map((apkUrl) { | ||||
|       if (RegExp('^$standardUrl/-/jobs/[0-9]+/artifacts/file/[^/]+\$') | ||||
|       if (RegExp('^$standardUrl/-/jobs/[0-9]+/artifacts/file/[^/]+') | ||||
|           .hasMatch(apkUrl.value)) { | ||||
|         return MapEntry( | ||||
|             apkUrl.key, apkUrl.value.replaceFirst('/file/', '/raw/')); | ||||
|   | ||||
| @@ -92,7 +92,89 @@ bool _isNumeric(String s) { | ||||
|   return s.codeUnitAt(0) >= 48 && s.codeUnitAt(0) <= 57; | ||||
| } | ||||
|  | ||||
| // Given an HTTP response, grab some links according to the common additional settings | ||||
| // (those that apply to intermediate and final steps) | ||||
| Future<List<MapEntry<String, String>>> grabLinksCommon( | ||||
|     Response res, Map<String, dynamic> additionalSettings) async { | ||||
|   if (res.statusCode != 200) { | ||||
|     throw getObtainiumHttpError(res); | ||||
|   } | ||||
|   var html = parse(res.body); | ||||
|   List<MapEntry<String, String>> allLinks = html | ||||
|       .querySelectorAll('a') | ||||
|       .map((element) => MapEntry( | ||||
|           element.attributes['href'] ?? '', | ||||
|           element.text.isNotEmpty | ||||
|               ? element.text | ||||
|               : (element.attributes['href'] ?? '').split('/').last)) | ||||
|       .where((element) => element.key.isNotEmpty) | ||||
|       .map((e) => MapEntry(ensureAbsoluteUrl(e.key, res.request!.url), e.value)) | ||||
|       .toList(); | ||||
|   if (allLinks.isEmpty) { | ||||
|     allLinks = RegExp( | ||||
|             r'(http|ftp|https)://([\w_-]+(?:(?:\.[\w_-]+)+))([\w.,@?^=%&:/~+#-]*[\w@?^=%&/~+#-])?') | ||||
|         .allMatches(res.body) | ||||
|         .map((match) => | ||||
|             MapEntry(match.group(0)!, match.group(0)?.split('/').last ?? '')) | ||||
|         .toList(); | ||||
|   } | ||||
|   List<MapEntry<String, String>> links = []; | ||||
|   bool skipSort = additionalSettings['skipSort'] == true; | ||||
|   bool filterLinkByText = additionalSettings['filterByLinkText'] == true; | ||||
|   if ((additionalSettings['customLinkFilterRegex'] as String?)?.isNotEmpty == | ||||
|       true) { | ||||
|     var reg = RegExp(additionalSettings['customLinkFilterRegex']); | ||||
|     links = allLinks.where((element) { | ||||
|       var link = element.key; | ||||
|       try { | ||||
|         link = Uri.decodeFull(element.key); | ||||
|       } catch (e) { | ||||
|         // Some links may not have valid encoding | ||||
|       } | ||||
|       return reg.hasMatch(filterLinkByText ? element.value : link); | ||||
|     }).toList(); | ||||
|   } else { | ||||
|     links = allLinks.where((element) { | ||||
|       var link = element.key; | ||||
|       try { | ||||
|         link = Uri.decodeFull(element.key); | ||||
|       } catch (e) { | ||||
|         // Some links may not have valid encoding | ||||
|       } | ||||
|       return Uri.parse(filterLinkByText ? element.value : link) | ||||
|           .path | ||||
|           .toLowerCase() | ||||
|           .endsWith('.apk'); | ||||
|     }).toList(); | ||||
|   } | ||||
|   if (!skipSort) { | ||||
|     links.sort((a, b) => additionalSettings['sortByLastLinkSegment'] == true | ||||
|         ? compareAlphaNumeric(a.key.split('/').where((e) => e.isNotEmpty).last, | ||||
|             b.key.split('/').where((e) => e.isNotEmpty).last) | ||||
|         : compareAlphaNumeric(a.key, b.key)); | ||||
|   } | ||||
|   if (additionalSettings['reverseSort'] == true) { | ||||
|     links = links.reversed.toList(); | ||||
|   } | ||||
|   return links; | ||||
| } | ||||
|  | ||||
| class HTML extends AppSource { | ||||
|   @override | ||||
|   List<List<GeneratedFormItem>> get combinedAppSpecificSettingFormItems { | ||||
|     return super.combinedAppSpecificSettingFormItems.map((r) { | ||||
|       return r.map((e) { | ||||
|         if (e.key == 'versionExtractionRegEx') { | ||||
|           e.label = tr('versionExtractionRegEx'); | ||||
|         } | ||||
|         if (e.key == 'matchGroupToUse') { | ||||
|           e.label = tr('matchGroupToUse'); | ||||
|         } | ||||
|         return e; | ||||
|       }).toList(); | ||||
|     }).toList(); | ||||
|   } | ||||
|  | ||||
|   var finalStepFormitems = [ | ||||
|     [ | ||||
|       GeneratedFormTextField('customLinkFilterRegex', | ||||
| @@ -206,79 +288,10 @@ class HTML extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return url; | ||||
|   } | ||||
|  | ||||
|   // Given an HTTP response, grab some links according to the common additional settings | ||||
|   // (those that apply to intermediate and final steps) | ||||
|   Future<List<MapEntry<String, String>>> grabLinksCommon( | ||||
|       Response res, Map<String, dynamic> additionalSettings) async { | ||||
|     if (res.statusCode != 200) { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|     var html = parse(res.body); | ||||
|     List<MapEntry<String, String>> allLinks = html | ||||
|         .querySelectorAll('a') | ||||
|         .map((element) => MapEntry( | ||||
|             element.attributes['href'] ?? '', | ||||
|             element.text.isNotEmpty | ||||
|                 ? element.text | ||||
|                 : (element.attributes['href'] ?? '').split('/').last)) | ||||
|         .where((element) => element.key.isNotEmpty) | ||||
|         .map((e) => | ||||
|             MapEntry(ensureAbsoluteUrl(e.key, res.request!.url), e.value)) | ||||
|         .toList(); | ||||
|     if (allLinks.isEmpty) { | ||||
|       allLinks = RegExp( | ||||
|               r'(http|ftp|https)://([\w_-]+(?:(?:\.[\w_-]+)+))([\w.,@?^=%&:/~+#-]*[\w@?^=%&/~+#-])?') | ||||
|           .allMatches(res.body) | ||||
|           .map((match) => | ||||
|               MapEntry(match.group(0)!, match.group(0)?.split('/').last ?? '')) | ||||
|           .toList(); | ||||
|     } | ||||
|     List<MapEntry<String, String>> links = []; | ||||
|     bool skipSort = additionalSettings['skipSort'] == true; | ||||
|     bool filterLinkByText = additionalSettings['filterByLinkText'] == true; | ||||
|     if ((additionalSettings['customLinkFilterRegex'] as String?)?.isNotEmpty == | ||||
|         true) { | ||||
|       var reg = RegExp(additionalSettings['customLinkFilterRegex']); | ||||
|       links = allLinks.where((element) { | ||||
|         var link = element.key; | ||||
|         try { | ||||
|           link = Uri.decodeFull(element.key); | ||||
|         } catch (e) { | ||||
|           // Some links may not have valid encoding | ||||
|         } | ||||
|         return reg.hasMatch(filterLinkByText ? element.value : link); | ||||
|       }).toList(); | ||||
|     } else { | ||||
|       links = allLinks.where((element) { | ||||
|         var link = element.key; | ||||
|         try { | ||||
|           link = Uri.decodeFull(element.key); | ||||
|         } catch (e) { | ||||
|           // Some links may not have valid encoding | ||||
|         } | ||||
|         return Uri.parse(filterLinkByText ? element.value : link) | ||||
|             .path | ||||
|             .toLowerCase() | ||||
|             .endsWith('.apk'); | ||||
|       }).toList(); | ||||
|     } | ||||
|     if (!skipSort) { | ||||
|       links.sort((a, b) => additionalSettings['sortByLastLinkSegment'] == true | ||||
|           ? compareAlphaNumeric( | ||||
|               a.key.split('/').where((e) => e.isNotEmpty).last, | ||||
|               b.key.split('/').where((e) => e.isNotEmpty).last) | ||||
|           : compareAlphaNumeric(a.key, b.key)); | ||||
|     } | ||||
|     if (additionalSettings['reverseSort'] == true) { | ||||
|       links = links.reversed.toList(); | ||||
|     } | ||||
|     return links; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
| @@ -332,10 +345,14 @@ class HTML extends AppSource { | ||||
|         additionalSettings['versionExtractWholePage'] == true | ||||
|             ? versionExtractionWholePageString | ||||
|             : relDecoded); | ||||
|     version ??= | ||||
|         additionalSettings['defaultPseudoVersioningMethod'] == 'APKLinkHash' | ||||
|             ? rel.hashCode.toString() | ||||
|             : (await checkPartialDownloadHashDynamic(rel)).toString(); | ||||
|     version ??= additionalSettings['defaultPseudoVersioningMethod'] == | ||||
|             'APKLinkHash' | ||||
|         ? rel.hashCode.toString() | ||||
|         : (await checkPartialDownloadHashDynamic(rel, | ||||
|                 headers: await getRequestHeaders(additionalSettings, | ||||
|                     forAPKDownload: true), | ||||
|                 allowInsecure: additionalSettings['allowInsecure'] == true)) | ||||
|             .toString(); | ||||
|     return APKDetails(version, [rel].map((e) => MapEntry(e, e)).toList(), | ||||
|         AppNames(uri.host, tr('app'))); | ||||
|   } | ||||
|   | ||||
| @@ -12,7 +12,7 @@ class HuaweiAppGallery extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}(/#)?/(app|appdl)/[^/]+', | ||||
|         caseSensitive: false); | ||||
| @@ -73,21 +73,23 @@ class HuaweiAppGallery extends AppSource { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
|     String appId = appIdFromRedirectDlUrl(res.headers['location']!); | ||||
|     if (appId.isEmpty) { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
|     var relDateStr = | ||||
|         res.headers['location']?.split('?')[0].split('.').reversed.toList()[1]; | ||||
|     var relDateStrAdj = relDateStr?.split(''); | ||||
|     var tempLen = relDateStrAdj?.length ?? 0; | ||||
|     var i = 2; | ||||
|     while (i < tempLen) { | ||||
|       relDateStrAdj?.insert((i + i ~/ 2 - 1), '-'); | ||||
|       i += 2; | ||||
|     } | ||||
|     var relDate = relDateStrAdj == null | ||||
|         ? null | ||||
|         : DateFormat('yy-MM-dd-HH-mm', 'en_US').parse(relDateStrAdj.join('')); | ||||
|     if (relDateStr == null) { | ||||
|     if (relDateStr == null || relDateStr.length != 10) { | ||||
|       throw NoVersionError(); | ||||
|     } | ||||
|     var relDateStrAdj = relDateStr.split(''); | ||||
|     var tempLen = relDateStrAdj.length; | ||||
|     var i = 2; | ||||
|     while (i < tempLen) { | ||||
|       relDateStrAdj.insert((i + i ~/ 2 - 1), '-'); | ||||
|       i += 2; | ||||
|     } | ||||
|     var relDate = | ||||
|         DateFormat('yy-MM-dd-HH-mm', 'en_US').parse(relDateStrAdj.join('')); | ||||
|     return APKDetails( | ||||
|         relDateStr, [MapEntry('$appId.apk', dlUrl)], AppNames(name, appId), | ||||
|         releaseDate: relDate); | ||||
|   | ||||
| @@ -14,7 +14,7 @@ class IzzyOnDroid extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegExA = RegExp( | ||||
|         '^https?://android.${getSourceRegex(hosts)}/repo/apk/[^/]+', | ||||
|         caseSensitive: false); | ||||
|   | ||||
| @@ -10,7 +10,7 @@ class Mullvad extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}', | ||||
|         caseSensitive: false); | ||||
|   | ||||
| @@ -10,7 +10,7 @@ class NeutronCode extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/downloads/file/[^/]+', | ||||
|         caseSensitive: false); | ||||
|   | ||||
| @@ -9,7 +9,7 @@ class Signal extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return 'https://${hosts[0]}'; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -9,7 +9,7 @@ class SourceForge extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     var sourceRegex = getSourceRegex(hosts); | ||||
|     RegExp standardUrlRegExC = | ||||
|         RegExp('^https?://(www\\.)?$sourceRegex/p/.+', caseSensitive: false); | ||||
|   | ||||
| @@ -20,7 +20,7 @@ class SourceHut extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', | ||||
|         caseSensitive: false); | ||||
|   | ||||
| @@ -20,7 +20,7 @@ class SteamMobile extends AppSource { | ||||
|   final apks = {'steam': tr('steamMobile'), 'steam-chat-app': tr('steamChat')}; | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return 'https://${hosts[0]}'; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -11,7 +11,7 @@ class TelegramApp extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return 'https://${hosts[0]}'; | ||||
|   } | ||||
|  | ||||
|   | ||||
							
								
								
									
										78
									
								
								lib/app_sources/tencent.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								lib/app_sources/tencent.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,78 @@ | ||||
| import 'dart:convert'; | ||||
|  | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
|  | ||||
| class Tencent extends AppSource { | ||||
|   Tencent() { | ||||
|     name = 'Tencent App Store'; | ||||
|     hosts = ['sj.qq.com']; | ||||
|     naiveStandardVersionDetection = true; | ||||
|     showReleaseDateAsVersionToggle = true; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://${getSourceRegex(hosts)}/appdetail/[^/]+', | ||||
|         caseSensitive: false); | ||||
|     var match = standardUrlRegEx.firstMatch(url); | ||||
|     if (match == null) { | ||||
|       throw InvalidURLError(name); | ||||
|     } | ||||
|     return match.group(0)!; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<String?> tryInferringAppId(String standardUrl, | ||||
|       {Map<String, dynamic> additionalSettings = const {}}) async { | ||||
|     return Uri.parse(standardUrl).pathSegments.last; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<APKDetails> getLatestAPKDetails( | ||||
|     String standardUrl, | ||||
|     Map<String, dynamic> additionalSettings, | ||||
|   ) async { | ||||
|     String appId = (await tryInferringAppId(standardUrl))!; | ||||
|     String baseHost = Uri.parse(standardUrl) | ||||
|         .host | ||||
|         .split('.') | ||||
|         .reversed | ||||
|         .toList() | ||||
|         .sublist(0, 2) | ||||
|         .reversed | ||||
|         .join('.'); | ||||
|  | ||||
|     var res = await sourceRequest( | ||||
|         'https://upage.html5.$baseHost/wechat-apkinfo', additionalSettings, | ||||
|         followRedirects: false, postBody: {"packagename": appId}); | ||||
|  | ||||
|     if (res.statusCode == 200) { | ||||
|       var json = jsonDecode(res.body); | ||||
|       if (json['app_detail_records'][appId] == null) { | ||||
|         throw NoReleasesError(); | ||||
|       } | ||||
|       var version = | ||||
|           json['app_detail_records'][appId]['apk_all_data']['version_name']; | ||||
|       var apkUrl = json['app_detail_records'][appId]['apk_all_data']['url']; | ||||
|       if (apkUrl == null) { | ||||
|         throw NoAPKError(); | ||||
|       } | ||||
|       var appName = json['app_detail_records'][appId]['app_info']['name']; | ||||
|       var author = json['app_detail_records'][appId]['app_info']['author']; | ||||
|       var releaseDate = | ||||
|           json['app_detail_records'][appId]['app_info']['update_time']; | ||||
|  | ||||
|       return APKDetails( | ||||
|           version, | ||||
|           [MapEntry(Uri.parse(apkUrl).queryParameters['fsname']!, apkUrl)], | ||||
|           AppNames(author, appName), | ||||
|           releaseDate: releaseDate != null | ||||
|               ? DateTime.fromMillisecondsSinceEpoch(releaseDate * 1000) | ||||
|               : null); | ||||
|     } else { | ||||
|       throw getObtainiumHttpError(res); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -10,10 +10,11 @@ class Uptodown extends AppSource { | ||||
|     allowSubDomains = true; | ||||
|     naiveStandardVersionDetection = true; | ||||
|     showReleaseDateAsVersionToggle = true; | ||||
|     urlsAlwaysHaveExtension = true; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     RegExp standardUrlRegEx = RegExp( | ||||
|         '^https?://([^\\.]+\\.){2,}${getSourceRegex(hosts)}', | ||||
|         caseSensitive: false); | ||||
| @@ -39,20 +40,29 @@ class Uptodown extends AppSource { | ||||
|     } | ||||
|     var html = parse(res.body); | ||||
|     String? version = html.querySelector('div.version')?.innerHtml; | ||||
|     String? apkUrl = | ||||
|         '${standardUrl.split('/').reversed.toList().sublist(1).reversed.join('/')}/post-download'; | ||||
|     String? name = html.querySelector('#detail-app-name')?.innerHtml.trim(); | ||||
|     String? author = html.querySelector('#author-link')?.innerHtml.trim(); | ||||
|     var detailElements = html.querySelectorAll('#technical-information td'); | ||||
|     String? appId = (detailElements.elementAtOrNull(2))?.innerHtml.trim(); | ||||
|     String? dateStr = (detailElements.elementAtOrNull(29))?.innerHtml.trim(); | ||||
|     String? fileId = | ||||
|         html.querySelector('#detail-app-name')?.attributes['data-file-id']; | ||||
|     String? extension = html | ||||
|         .querySelectorAll('td') | ||||
|         .where((e) => e.text.toLowerCase().trim() == 'file type') | ||||
|         .firstOrNull | ||||
|         ?.nextElementSibling | ||||
|         ?.text | ||||
|         .toLowerCase() | ||||
|         .trim(); | ||||
|     return Map.fromEntries([ | ||||
|       MapEntry('version', version), | ||||
|       MapEntry('apkUrl', apkUrl), | ||||
|       MapEntry('appId', appId), | ||||
|       MapEntry('name', name), | ||||
|       MapEntry('author', author), | ||||
|       MapEntry('dateStr', dateStr) | ||||
|       MapEntry('dateStr', dateStr), | ||||
|       MapEntry('fileId', fileId), | ||||
|       MapEntry('extension', extension) | ||||
|     ]); | ||||
|   } | ||||
|  | ||||
| @@ -64,14 +74,16 @@ class Uptodown extends AppSource { | ||||
|     var appDetails = | ||||
|         await getAppDetailsFromPage(standardUrl, additionalSettings); | ||||
|     var version = appDetails['version']; | ||||
|     var apkUrl = appDetails['apkUrl']; | ||||
|     var appId = appDetails['appId']; | ||||
|     var fileId = appDetails['fileId']; | ||||
|     var extension = appDetails['extension']; | ||||
|     if (version == null) { | ||||
|       throw NoVersionError(); | ||||
|     } | ||||
|     if (apkUrl == null) { | ||||
|     if (fileId == null) { | ||||
|       throw NoAPKError(); | ||||
|     } | ||||
|     var apkUrl = '$standardUrl/$fileId-x'; | ||||
|     if (appId == null) { | ||||
|       throw NoReleasesError(); | ||||
|     } | ||||
| @@ -82,8 +94,8 @@ class Uptodown extends AppSource { | ||||
|     if (dateStr != null) { | ||||
|       relDate = parseDateTimeMMMddCommayyyy(dateStr); | ||||
|     } | ||||
|     return APKDetails( | ||||
|         version, getApkUrlsFromUrls([apkUrl]), AppNames(author, appName), | ||||
|     return APKDetails(version, [MapEntry('$appId.$extension', apkUrl)], | ||||
|         AppNames(author, appName), | ||||
|         releaseDate: relDate); | ||||
|   } | ||||
|  | ||||
| @@ -96,7 +108,7 @@ class Uptodown extends AppSource { | ||||
|     } | ||||
|     var html = parse(res.body); | ||||
|     var finalUrlKey = | ||||
|         html.querySelector('.post-download')?.attributes['data-url']; | ||||
|         html.querySelector('#detail-download-button')?.attributes['data-url']; | ||||
|     if (finalUrlKey == null) { | ||||
|       throw NoAPKError(); | ||||
|     } | ||||
|   | ||||
| @@ -21,7 +21,7 @@ class VLC extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return 'https://${hosts[0]}'; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -10,7 +10,7 @@ class WhatsApp extends AppSource { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     return 'https://${hosts[0]}'; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:obtainium/components/generated_form_modal.dart'; | ||||
| import 'package:obtainium/providers/source_provider.dart'; | ||||
| import 'package:flutter_typeahead/flutter_typeahead.dart'; | ||||
|  | ||||
| abstract class GeneratedFormItem { | ||||
|   late String key; | ||||
| @@ -28,6 +29,7 @@ class GeneratedFormTextField extends GeneratedFormItem { | ||||
|   late String? hint; | ||||
|   late bool password; | ||||
|   late TextInputType? textInputType; | ||||
|   late List<String>? autoCompleteOptions; | ||||
|  | ||||
|   GeneratedFormTextField(super.key, | ||||
|       {super.label, | ||||
| @@ -39,7 +41,8 @@ class GeneratedFormTextField extends GeneratedFormItem { | ||||
|       this.max = 1, | ||||
|       this.hint, | ||||
|       this.password = false, | ||||
|       this.textInputType}); | ||||
|       this.textInputType, | ||||
|       this.autoCompleteOptions}); | ||||
|  | ||||
|   @override | ||||
|   String ensureType(val) { | ||||
| @@ -274,38 +277,62 @@ class _GeneratedFormState extends State<GeneratedForm> { | ||||
|         var formItem = e.value; | ||||
|         if (formItem is GeneratedFormTextField) { | ||||
|           final formFieldKey = GlobalKey<FormFieldState>(); | ||||
|           return TextFormField( | ||||
|             keyboardType: formItem.textInputType, | ||||
|             obscureText: formItem.password, | ||||
|             autocorrect: !formItem.password, | ||||
|             enableSuggestions: !formItem.password, | ||||
|             key: formFieldKey, | ||||
|             initialValue: values[formItem.key], | ||||
|             autovalidateMode: AutovalidateMode.onUserInteraction, | ||||
|             onChanged: (value) { | ||||
|           var ctrl = TextEditingController(text: values[formItem.key]); | ||||
|           return TypeAheadField<String>( | ||||
|             controller: ctrl, | ||||
|             builder: (context, controller, focusNode) { | ||||
|               return TextFormField( | ||||
|                 controller: ctrl, | ||||
|                 focusNode: focusNode, | ||||
|                 keyboardType: formItem.textInputType, | ||||
|                 obscureText: formItem.password, | ||||
|                 autocorrect: !formItem.password, | ||||
|                 enableSuggestions: !formItem.password, | ||||
|                 key: formFieldKey, | ||||
|                 autovalidateMode: AutovalidateMode.onUserInteraction, | ||||
|                 onChanged: (value) { | ||||
|                   setState(() { | ||||
|                     values[formItem.key] = value; | ||||
|                     someValueChanged(); | ||||
|                   }); | ||||
|                 }, | ||||
|                 decoration: InputDecoration( | ||||
|                     helperText: | ||||
|                         formItem.label + (formItem.required ? ' *' : ''), | ||||
|                     hintText: formItem.hint), | ||||
|                 minLines: formItem.max <= 1 ? null : formItem.max, | ||||
|                 maxLines: formItem.max <= 1 ? 1 : formItem.max, | ||||
|                 validator: (value) { | ||||
|                   if (formItem.required && | ||||
|                       (value == null || value.trim().isEmpty)) { | ||||
|                     return '${formItem.label} ${tr('requiredInBrackets')}'; | ||||
|                   } | ||||
|                   for (var validator in formItem.additionalValidators) { | ||||
|                     String? result = validator(value); | ||||
|                     if (result != null) { | ||||
|                       return result; | ||||
|                     } | ||||
|                   } | ||||
|                   return null; | ||||
|                 }, | ||||
|               ); | ||||
|             }, | ||||
|             itemBuilder: (context, value) { | ||||
|               return ListTile(title: Text(value)); | ||||
|             }, | ||||
|             onSelected: (value) { | ||||
|               ctrl.text = value; | ||||
|               setState(() { | ||||
|                 values[formItem.key] = value; | ||||
|                 someValueChanged(); | ||||
|               }); | ||||
|             }, | ||||
|             decoration: InputDecoration( | ||||
|                 helperText: formItem.label + (formItem.required ? ' *' : ''), | ||||
|                 hintText: formItem.hint), | ||||
|             minLines: formItem.max <= 1 ? null : formItem.max, | ||||
|             maxLines: formItem.max <= 1 ? 1 : formItem.max, | ||||
|             validator: (value) { | ||||
|               if (formItem.required && | ||||
|                   (value == null || value.trim().isEmpty)) { | ||||
|                 return '${formItem.label} ${tr('requiredInBrackets')}'; | ||||
|               } | ||||
|               for (var validator in formItem.additionalValidators) { | ||||
|                 String? result = validator(value); | ||||
|                 if (result != null) { | ||||
|                   return result; | ||||
|                 } | ||||
|               } | ||||
|               return null; | ||||
|             suggestionsCallback: (search) { | ||||
|               return formItem.autoCompleteOptions | ||||
|                   ?.where((t) => t.toLowerCase().contains(search.toLowerCase())) | ||||
|                   .toList(); | ||||
|             }, | ||||
|             hideOnEmpty: true, | ||||
|           ); | ||||
|         } else if (formItem is GeneratedFormDropdown) { | ||||
|           if (formItem.opts!.isEmpty) { | ||||
|   | ||||
| @@ -23,6 +23,7 @@ import 'package:easy_localization/src/localization.dart'; | ||||
| List<MapEntry<Locale, String>> supportedLocales = const [ | ||||
|   MapEntry(Locale('en'), 'English'), | ||||
|   MapEntry(Locale('zh'), '简体中文'), | ||||
|   MapEntry(Locale('zh_Hant_TW'), '臺灣話'), | ||||
|   MapEntry(Locale('it'), 'Italiano'), | ||||
|   MapEntry(Locale('ja'), '日本語'), | ||||
|   MapEntry(Locale('hu'), 'Magyar'), | ||||
| @@ -40,6 +41,9 @@ List<MapEntry<Locale, String>> supportedLocales = const [ | ||||
|   MapEntry(Locale('vi'), 'Tiếng Việt'), | ||||
|   MapEntry(Locale('tr'), 'Türkçe'), | ||||
|   MapEntry(Locale('uk'), 'Українська'), | ||||
|   MapEntry(Locale('da'), 'Dansk'), | ||||
|   MapEntry(Locale('en', 'EO'), | ||||
|       'Esperanto'), // https://github.com/aissat/easy_localization/issues/220#issuecomment-846035493 | ||||
| ]; | ||||
| const fallbackLocale = Locale('en'); | ||||
| const localeDir = 'assets/translations'; | ||||
| @@ -212,20 +216,23 @@ class _ObtainiumState extends State<Obtainium> { | ||||
|       // Decide on a colour/brightness scheme based on OS and user settings | ||||
|       ColorScheme lightColorScheme; | ||||
|       ColorScheme darkColorScheme; | ||||
|       if (lightDynamic != null && darkDynamic != null && settingsProvider.useMaterialYou) { | ||||
|       if (lightDynamic != null && | ||||
|           darkDynamic != null && | ||||
|           settingsProvider.useMaterialYou) { | ||||
|         lightColorScheme = lightDynamic.harmonized(); | ||||
|         darkColorScheme = darkDynamic.harmonized(); | ||||
|       } else { | ||||
|         lightColorScheme = ColorScheme.fromSeed(seedColor: settingsProvider.themeColor); | ||||
|         lightColorScheme = | ||||
|             ColorScheme.fromSeed(seedColor: settingsProvider.themeColor); | ||||
|         darkColorScheme = ColorScheme.fromSeed( | ||||
|             seedColor: settingsProvider.themeColor, brightness: Brightness.dark); | ||||
|             seedColor: settingsProvider.themeColor, | ||||
|             brightness: Brightness.dark); | ||||
|       } | ||||
|  | ||||
|       // set the background and surface colors to pure black in the amoled theme | ||||
|       if (settingsProvider.useBlackTheme) { | ||||
|         darkColorScheme = darkColorScheme | ||||
|             .copyWith(background: Colors.black, surface: Colors.black) | ||||
|             .harmonized(); | ||||
|         darkColorScheme = | ||||
|             darkColorScheme.copyWith(surface: Colors.black).harmonized(); | ||||
|       } | ||||
|  | ||||
|       if (settingsProvider.useSystemFont) NativeFeatures.loadSystemFont(); | ||||
| @@ -241,15 +248,17 @@ class _ObtainiumState extends State<Obtainium> { | ||||
|               colorScheme: settingsProvider.theme == ThemeSettings.dark | ||||
|                   ? darkColorScheme | ||||
|                   : lightColorScheme, | ||||
|               fontFamily: | ||||
|                   settingsProvider.useSystemFont ? 'SystemFont' : 'Metropolis'), | ||||
|               fontFamily: settingsProvider.useSystemFont | ||||
|                   ? 'SystemFont' | ||||
|                   : 'Wix-Madefor-Display'), | ||||
|           darkTheme: ThemeData( | ||||
|               useMaterial3: true, | ||||
|               colorScheme: settingsProvider.theme == ThemeSettings.light | ||||
|                   ? lightColorScheme | ||||
|                   : darkColorScheme, | ||||
|               fontFamily: | ||||
|                   settingsProvider.useSystemFont ? 'SystemFont' : 'Metropolis'), | ||||
|               fontFamily: settingsProvider.useSystemFont | ||||
|                   ? 'SystemFont' | ||||
|                   : 'Wix-Madefor-Display'), | ||||
|           home: Shortcuts(shortcuts: <LogicalKeySet, Intent>{ | ||||
|             LogicalKeySet(LogicalKeyboardKey.select): const ActivateIntent(), | ||||
|           }, child: const HomePage())); | ||||
|   | ||||
| @@ -51,10 +51,13 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|   } | ||||
|  | ||||
|   changeUserInput(String input, bool valid, bool isBuilding, | ||||
|       {bool updateUrlInput = false}) { | ||||
|       {bool updateUrlInput = false, String? overrideSource}) { | ||||
|     userInput = input; | ||||
|     if (!isBuilding) { | ||||
|       setState(() { | ||||
|         if (overrideSource != null) { | ||||
|           pickedSourceOverride = overrideSource; | ||||
|         } | ||||
|         if (updateUrlInput) { | ||||
|           urlInputKey++; | ||||
|         } | ||||
| @@ -68,6 +71,7 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|         if (pickedSource.runtimeType != source.runtimeType || | ||||
|             (prevHost != null && prevHost != source?.hosts[0])) { | ||||
|           pickedSource = source; | ||||
|           pickedSource?.runOnAddAppInputChange(userInput); | ||||
|           additionalSettings = source != null | ||||
|               ? getDefaultValuesFromFormItems( | ||||
|                   source.combinedAppSpecificSettingFormItems) | ||||
| @@ -259,9 +263,7 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|         searching = true; | ||||
|       }); | ||||
|       var sourceStrings = <String, List<String>>{}; | ||||
|       sourceProvider.sources | ||||
|           .where((e) => e.canSearch && !e.excludeFromMassSearch) | ||||
|           .forEach((s) { | ||||
|       sourceProvider.sources.where((e) => e.canSearch).forEach((s) { | ||||
|         sourceStrings[s.name] = [s.name]; | ||||
|       }); | ||||
|       try { | ||||
| @@ -282,32 +284,78 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|           settingsProvider.searchDeselected = sourceStrings.keys | ||||
|               .where((s) => !searchSources.contains(s)) | ||||
|               .toList(); | ||||
|           var results = await Future.wait(sourceProvider.sources | ||||
|               .where((e) => searchSources.contains(e.name)) | ||||
|               .map((e) async { | ||||
|           List<MapEntry<String, Map<String, List<String>>>?> results = | ||||
|               (await Future.wait(sourceProvider.sources | ||||
|                       .where((e) => searchSources.contains(e.name)) | ||||
|                       .map((e) async { | ||||
|             try { | ||||
|               return await e.search(searchQuery); | ||||
|               Map<String, dynamic>? querySettings = {}; | ||||
|               if (e.includeAdditionalOptsInMainSearch) { | ||||
|                 querySettings = await showDialog<Map<String, dynamic>?>( | ||||
|                     context: context, | ||||
|                     builder: (BuildContext ctx) { | ||||
|                       return GeneratedFormModal( | ||||
|                         title: tr('searchX', args: [e.name]), | ||||
|                         items: [ | ||||
|                           ...e.searchQuerySettingFormItems.map((e) => [e]), | ||||
|                           [ | ||||
|                             GeneratedFormTextField('url', | ||||
|                                 label: e.hosts.isNotEmpty | ||||
|                                     ? tr('overrideSource') | ||||
|                                     : plural('url', 1).substring(2), | ||||
|                                 autoCompleteOptions: [ | ||||
|                                   ...(e.hosts.isNotEmpty ? [e.hosts[0]] : []), | ||||
|                                   ...appsProvider.apps.values | ||||
|                                       .where((a) => | ||||
|                                           sourceProvider | ||||
|                                               .getSource(a.app.url, | ||||
|                                                   overrideSource: | ||||
|                                                       a.app.overrideSource) | ||||
|                                               .runtimeType == | ||||
|                                           e.runtimeType) | ||||
|                                       .map((a) { | ||||
|                                     var uri = Uri.parse(a.app.url); | ||||
|                                     return '${uri.origin}${uri.path}'; | ||||
|                                   }) | ||||
|                                 ], | ||||
|                                 defaultValue: | ||||
|                                     e.hosts.isNotEmpty ? e.hosts[0] : '', | ||||
|                                 required: true) | ||||
|                           ], | ||||
|                         ], | ||||
|                       ); | ||||
|                     }); | ||||
|                 if (querySettings == null) { | ||||
|                   return null; | ||||
|                 } | ||||
|               } | ||||
|               return MapEntry(e.runtimeType.toString(), | ||||
|                   await e.search(searchQuery, querySettings: querySettings)); | ||||
|             } catch (err) { | ||||
|               if (err is! CredsNeededError) { | ||||
|                 rethrow; | ||||
|               } else { | ||||
|                 err.unexpected = true; | ||||
|                 showError(err, context); | ||||
|                 return <String, List<String>>{}; | ||||
|                 return null; | ||||
|               } | ||||
|             } | ||||
|           })); | ||||
|           }))) | ||||
|                   .where((a) => a != null) | ||||
|                   .toList(); | ||||
|  | ||||
|           // Interleave results instead of simple reduce | ||||
|           Map<String, List<String>> res = {}; | ||||
|           Map<String, MapEntry<String, List<String>>> res = {}; | ||||
|           var si = 0; | ||||
|           var done = false; | ||||
|           while (!done) { | ||||
|             done = true; | ||||
|             for (var r in results) { | ||||
|               if (r.length > si) { | ||||
|               var sourceName = r!.key; | ||||
|               if (r.value.length > si) { | ||||
|                 done = false; | ||||
|                 res.addEntries([r.entries.elementAt(si)]); | ||||
|                 var singleRes = r.value.entries.elementAt(si); | ||||
|                 res[singleRes.key] = MapEntry(sourceName, singleRes.value); | ||||
|               } | ||||
|             } | ||||
|             si++; | ||||
| @@ -322,13 +370,15 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|                   context: context, | ||||
|                   builder: (BuildContext ctx) { | ||||
|                     return SelectionModal( | ||||
|                       entries: res, | ||||
|                       entries: res.map((k, v) => MapEntry(k, v.value)), | ||||
|                       selectedByDefault: false, | ||||
|                       onlyOneSelectionAllowed: true, | ||||
|                     ); | ||||
|                   }); | ||||
|           if (selectedUrls != null && selectedUrls.isNotEmpty) { | ||||
|             changeUserInput(selectedUrls[0], true, false, updateUrlInput: true); | ||||
|             var sourceName = res[selectedUrls[0]]?.key; | ||||
|             changeUserInput(selectedUrls[0], true, false, | ||||
|                 updateUrlInput: true, overrideSource: sourceName); | ||||
|           } | ||||
|         } | ||||
|       } catch (e) { | ||||
| @@ -349,7 +399,7 @@ class AddAppPageState extends State<AddAppPage> { | ||||
|                   [ | ||||
|                     GeneratedFormDropdown( | ||||
|                         'overrideSource', | ||||
|                         defaultValue: '', | ||||
|                         defaultValue: pickedSourceOverride ?? '', | ||||
|                         [ | ||||
|                           MapEntry('', tr('none')), | ||||
|                           ...sourceProvider.sources.map( | ||||
|   | ||||
| @@ -161,25 +161,46 @@ class _AppPageState extends State<AppPage> { | ||||
|           if (app?.app.apkUrls.isNotEmpty == true || | ||||
|               app?.app.otherAssetUrls.isNotEmpty == true) | ||||
|             GestureDetector( | ||||
|               onTap: app?.app == null || updating | ||||
|                   ? null | ||||
|                   : () async { | ||||
|                       try { | ||||
|                         await appsProvider | ||||
|                             .downloadAppAssets([app!.app.id], context); | ||||
|                       } catch (e) { | ||||
|                         showError(e, context); | ||||
|                       } | ||||
|                     }, | ||||
|               child: Text( | ||||
|                 tr('downloadX', args: [tr('releaseAsset').toLowerCase()]), | ||||
|                 textAlign: TextAlign.center, | ||||
|                 style: Theme.of(context).textTheme.labelSmall!.copyWith( | ||||
|                       decoration: TextDecoration.underline, | ||||
|                       fontStyle: FontStyle.italic, | ||||
|                     ), | ||||
|               ), | ||||
|             ), | ||||
|                 onTap: app?.app == null || updating | ||||
|                     ? null | ||||
|                     : () async { | ||||
|                         try { | ||||
|                           await appsProvider | ||||
|                               .downloadAppAssets([app!.app.id], context); | ||||
|                         } catch (e) { | ||||
|                           showError(e, context); | ||||
|                         } | ||||
|                       }, | ||||
|                 child: Row( | ||||
|                   mainAxisAlignment: MainAxisAlignment.center, | ||||
|                   children: [ | ||||
|                     Container( | ||||
|                         decoration: BoxDecoration( | ||||
|                             borderRadius: BorderRadius.circular(12), | ||||
|                             color: settingsProvider.highlightTouchTargets | ||||
|                                 ? (Theme.of(context).brightness == | ||||
|                                             Brightness.light | ||||
|                                         ? Theme.of(context).primaryColor | ||||
|                                         : Theme.of(context).primaryColorLight) | ||||
|                                     .withAlpha(20) | ||||
|                                 : null), | ||||
|                         padding: settingsProvider.highlightTouchTargets | ||||
|                             ? const EdgeInsetsDirectional.fromSTEB(12, 6, 12, 6) | ||||
|                             : const EdgeInsetsDirectional.fromSTEB(0, 6, 0, 6), | ||||
|                         margin: | ||||
|                             const EdgeInsetsDirectional.fromSTEB(0, 6, 0, 0), | ||||
|                         child: Text( | ||||
|                           tr('downloadX', | ||||
|                               args: [tr('releaseAsset').toLowerCase()]), | ||||
|                           textAlign: TextAlign.center, | ||||
|                           style: | ||||
|                               Theme.of(context).textTheme.labelSmall!.copyWith( | ||||
|                                     decoration: TextDecoration.underline, | ||||
|                                     fontStyle: FontStyle.italic, | ||||
|                                   ), | ||||
|                         )) | ||||
|                   ], | ||||
|                 )), | ||||
|           const SizedBox( | ||||
|             height: 48, | ||||
|           ), | ||||
| @@ -226,18 +247,27 @@ class _AppPageState extends State<AppPage> { | ||||
|           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|           children: [ | ||||
|             const SizedBox(height: 20), | ||||
|             app?.icon != null | ||||
|                 ? Row(mainAxisAlignment: MainAxisAlignment.center, children: [ | ||||
|                     GestureDetector( | ||||
|                       child: Image.memory( | ||||
|                         app!.icon!, | ||||
|                         height: 150, | ||||
|                         gaplessPlayback: true, | ||||
|                       ), | ||||
|                       onTap: () => pm.openApp(app.app.id), | ||||
|                     ) | ||||
|                   ]) | ||||
|                 : Container(), | ||||
|             FutureBuilder( | ||||
|                 future: | ||||
|                     appsProvider.updateAppIcon(app?.app.id, ignoreCache: true), | ||||
|                 builder: (ctx, val) { | ||||
|                   return app?.icon != null | ||||
|                       ? Row( | ||||
|                           mainAxisAlignment: MainAxisAlignment.center, | ||||
|                           children: [ | ||||
|                               GestureDetector( | ||||
|                                 onTap: app == null | ||||
|                                     ? null | ||||
|                                     : () => pm.openApp(app.app.id), | ||||
|                                 child: Image.memory( | ||||
|                                   app!.icon!, | ||||
|                                   height: 150, | ||||
|                                   gaplessPlayback: true, | ||||
|                                 ), | ||||
|                               ) | ||||
|                             ]) | ||||
|                       : Container(); | ||||
|                 }), | ||||
|             const SizedBox( | ||||
|               height: 25, | ||||
|             ), | ||||
| @@ -286,7 +316,7 @@ class _AppPageState extends State<AppPage> { | ||||
|         ? WebViewWidget( | ||||
|             controller: WebViewController() | ||||
|               ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||
|               ..setBackgroundColor(Theme.of(context).colorScheme.background) | ||||
|               ..setBackgroundColor(Theme.of(context).colorScheme.surface) | ||||
|               ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||
|               ..setNavigationDelegate( | ||||
|                 NavigationDelegate( | ||||
|   | ||||
| @@ -143,11 +143,14 @@ class AppsPageState extends State<AppsPage> { | ||||
|   final GlobalKey<RefreshIndicatorState> _refreshIndicatorKey = | ||||
|       GlobalKey<RefreshIndicatorState>(); | ||||
|  | ||||
|   late final ScrollController scrollController = ScrollController(); | ||||
|  | ||||
|   var sourceProvider = SourceProvider(); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     var appsProvider = context.watch<AppsProvider>(); | ||||
|     var settingsProvider = context.watch<SettingsProvider>(); | ||||
|     var sourceProvider = SourceProvider(); | ||||
|     var listedApps = appsProvider.getAppValues().toList(); | ||||
|  | ||||
|     refresh() { | ||||
| @@ -354,7 +357,11 @@ class AppsPageState extends State<AppsPage> { | ||||
|           SliverFillRemaining( | ||||
|               child: Center( | ||||
|                   child: Text( | ||||
|             appsProvider.apps.isEmpty ? tr('noApps') : tr('noAppsForFilter'), | ||||
|             appsProvider.apps.isEmpty | ||||
|                 ? appsProvider.loadingApps | ||||
|                     ? tr('pleaseWait') | ||||
|                     : tr('noApps') | ||||
|                 : tr('noAppsForFilter'), | ||||
|             style: Theme.of(context).textTheme.headlineMedium, | ||||
|             textAlign: TextAlign.center, | ||||
|           ))), | ||||
| @@ -402,29 +409,38 @@ class AppsPageState extends State<AppsPage> { | ||||
|     } | ||||
|  | ||||
|     getAppIcon(int appIndex) { | ||||
|       return listedApps[appIndex].icon != null | ||||
|           ? Image.memory( | ||||
|               listedApps[appIndex].icon!, | ||||
|               gaplessPlayback: true, | ||||
|             ) | ||||
|           : Row( | ||||
|               mainAxisSize: MainAxisSize.min, | ||||
|               mainAxisAlignment: MainAxisAlignment.center, | ||||
|               children: [ | ||||
|                   Transform( | ||||
|                       alignment: Alignment.center, | ||||
|                       transform: Matrix4.rotationZ(0.31), | ||||
|                       child: Padding( | ||||
|                         padding: const EdgeInsets.all(15), | ||||
|                         child: Image( | ||||
|                           image: const AssetImage( | ||||
|                               'assets/graphics/icon_small.png'), | ||||
|                           color: Colors.white.withOpacity(0.3), | ||||
|                           colorBlendMode: BlendMode.modulate, | ||||
|                           gaplessPlayback: true, | ||||
|                         ), | ||||
|                       )), | ||||
|                 ]); | ||||
|       return FutureBuilder( | ||||
|           future: appsProvider.updateAppIcon(listedApps[appIndex].app.id), | ||||
|           builder: (ctx, val) { | ||||
|             return listedApps[appIndex].icon != null | ||||
|                 ? Image.memory( | ||||
|                     listedApps[appIndex].icon!, | ||||
|                     gaplessPlayback: true, | ||||
|                     opacity: AlwaysStoppedAnimation( | ||||
|                         listedApps[appIndex].installedInfo == null ? 0.6 : 1), | ||||
|                   ) | ||||
|                 : Row( | ||||
|                     mainAxisSize: MainAxisSize.min, | ||||
|                     mainAxisAlignment: MainAxisAlignment.center, | ||||
|                     children: [ | ||||
|                         Transform( | ||||
|                             alignment: Alignment.center, | ||||
|                             transform: Matrix4.rotationZ(0.31), | ||||
|                             child: Padding( | ||||
|                               padding: const EdgeInsets.all(15), | ||||
|                               child: Image( | ||||
|                                 image: const AssetImage( | ||||
|                                     'assets/graphics/icon_small.png'), | ||||
|                                 color: Theme.of(context).brightness == | ||||
|                                         Brightness.dark | ||||
|                                     ? Colors.white.withOpacity(0.4) | ||||
|                                     : Colors.white.withOpacity(0.3), | ||||
|                                 colorBlendMode: BlendMode.modulate, | ||||
|                                 gaplessPlayback: true, | ||||
|                               ), | ||||
|                             )), | ||||
|                       ]); | ||||
|           }); | ||||
|     } | ||||
|  | ||||
|     getVersionText(int appIndex) { | ||||
| @@ -452,7 +468,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|           hasUpdate ? getUpdateButton(index) : const SizedBox.shrink(), | ||||
|           hasUpdate | ||||
|               ? const SizedBox( | ||||
|                   width: 10, | ||||
|                   width: 5, | ||||
|                 ) | ||||
|               : const SizedBox.shrink(), | ||||
|           GestureDetector( | ||||
| @@ -503,7 +519,7 @@ class AppsPageState extends State<AppsPage> { | ||||
|       ); | ||||
|  | ||||
|       var transparent = | ||||
|           Theme.of(context).colorScheme.background.withAlpha(0).value; | ||||
|           Theme.of(context).colorScheme.surface.withAlpha(0).value; | ||||
|       List<double> stops = [ | ||||
|         ...listedApps[index].app.categories.asMap().entries.map( | ||||
|             (e) => ((e.key / (listedApps[index].app.categories.length - 1)))), | ||||
| @@ -903,6 +919,27 @@ class AppsPageState extends State<AppsPage> { | ||||
|                                 }, | ||||
|                           child: Text(tr('shareAppConfigLinks'))), | ||||
|                       const Divider(), | ||||
|                       TextButton( | ||||
|                           onPressed: selectedAppIds.isEmpty | ||||
|                               ? null | ||||
|                               : () { | ||||
|                                   var exportJSON = jsonEncode( | ||||
|                                       appsProvider.generateExportJSON( | ||||
|                                           appIds: selectedApps | ||||
|                                               .map((e) => e.id) | ||||
|                                               .toList(), | ||||
|                                           overrideExportSettings: false)); | ||||
|                                   XFile f = XFile.fromData( | ||||
|                                       Uint8List.fromList( | ||||
|                                           utf8.encode(exportJSON)), | ||||
|                                       mimeType: 'application/json', | ||||
|                                       name: | ||||
|                                           '${tr('obtainiumExportHyphenatedLowercase')}-${selectedApps.length}-${DateTime.now().millisecondsSinceEpoch}'); | ||||
|                                   Share.shareXFiles([f]); | ||||
|                                 }, | ||||
|                           child: Text( | ||||
|                               '${tr('share')} - ${tr('obtainiumExport')}')), | ||||
|                       const Divider(), | ||||
|                       TextButton( | ||||
|                           onPressed: () { | ||||
|                             appsProvider | ||||
| @@ -1042,7 +1079,9 @@ class AppsPageState extends State<AppsPage> { | ||||
|           IconButton( | ||||
|               color: Theme.of(context).colorScheme.primary, | ||||
|               style: const ButtonStyle(visualDensity: VisualDensity.compact), | ||||
|               tooltip: '${tr('filter')}${isFilterOff ? '' : ' *'}', | ||||
|               tooltip: isFilterOff | ||||
|                   ? tr('filterApps') | ||||
|                   : '${tr('filter')} - ${tr('remove')}', | ||||
|               onPressed: isFilterOff | ||||
|                   ? showFilterDialog | ||||
|                   : () { | ||||
| @@ -1051,8 +1090,8 @@ class AppsPageState extends State<AppsPage> { | ||||
|                       }); | ||||
|                     }, | ||||
|               icon: Icon(isFilterOff | ||||
|                   ? Icons.filter_list_rounded | ||||
|                   : Icons.filter_list_off_rounded)), | ||||
|                   ? Icons.search_rounded | ||||
|                   : Icons.search_off_rounded)), | ||||
|           const SizedBox( | ||||
|             width: 10, | ||||
|           ), | ||||
| @@ -1087,11 +1126,17 @@ class AppsPageState extends State<AppsPage> { | ||||
|       body: RefreshIndicator( | ||||
|           key: _refreshIndicatorKey, | ||||
|           onRefresh: refresh, | ||||
|           child: CustomScrollView(slivers: <Widget>[ | ||||
|             CustomAppBar(title: tr('appsString')), | ||||
|             ...getLoadingWidgets(), | ||||
|             getDisplayedList() | ||||
|           ])), | ||||
|           child: Scrollbar( | ||||
|               interactive: true, | ||||
|               controller: scrollController, | ||||
|               child: CustomScrollView( | ||||
|                   physics: const AlwaysScrollableScrollPhysics(), | ||||
|                   controller: scrollController, | ||||
|                   slivers: <Widget>[ | ||||
|                     CustomAppBar(title: tr('appsString')), | ||||
|                     ...getLoadingWidgets(), | ||||
|                     getDisplayedList() | ||||
|                   ]))), | ||||
|       persistentFooterButtons: appsProvider.apps.isEmpty | ||||
|           ? null | ||||
|           : [ | ||||
|   | ||||
| @@ -103,7 +103,7 @@ class _HomePageState extends State<HomePage> { | ||||
|                   }) != | ||||
|               null) { | ||||
|             // ignore: use_build_context_synchronously | ||||
|             var appsProvider = await context.read<AppsProvider>(); | ||||
|             var appsProvider = context.read<AppsProvider>(); | ||||
|             var result = await appsProvider.import(action == 'app' | ||||
|                 ? '{ "apps": [$dataStr] }' | ||||
|                 : '{ "apps": $dataStr }'); | ||||
|   | ||||
| @@ -33,7 +33,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | ||||
|     var settingsProvider = context.watch<SettingsProvider>(); | ||||
|  | ||||
|     var outlineButtonStyle = ButtonStyle( | ||||
|       shape: MaterialStateProperty.all( | ||||
|       shape: WidgetStateProperty.all( | ||||
|         StadiumBorder( | ||||
|           side: BorderSide( | ||||
|             width: 1, | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import 'package:flex_color_picker/flex_color_picker.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:obtainium/components/custom_app_bar.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/components/generated_form_modal.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| import 'package:obtainium/main.dart'; | ||||
| import 'package:obtainium/providers/apps_provider.dart'; | ||||
| @@ -945,6 +946,25 @@ class _LogsDialogState extends State<LogsDialog> { | ||||
|         ], | ||||
|       ), | ||||
|       actions: [ | ||||
|         TextButton( | ||||
|             onPressed: () async { | ||||
|               var cont = (await showDialog<Map<String, dynamic>?>( | ||||
|                       context: context, | ||||
|                       builder: (BuildContext ctx) { | ||||
|                         return GeneratedFormModal( | ||||
|                           title: tr('appLogs'), | ||||
|                           items: const [], | ||||
|                           initValid: true, | ||||
|                           message: tr('removeFromObtainium'), | ||||
|                         ); | ||||
|                       })) != | ||||
|                   null; | ||||
|               if (cont) { | ||||
|                 logsProvider.clear(); | ||||
|                 Navigator.of(context).pop(); | ||||
|               } | ||||
|             }, | ||||
|             child: Text(tr('remove'))), | ||||
|         TextButton( | ||||
|             onPressed: () { | ||||
|               Navigator.of(context).pop(); | ||||
|   | ||||
| @@ -17,6 +17,7 @@ import 'package:device_info_plus/device_info_plus.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:http/io_client.dart'; | ||||
| import 'package:obtainium/components/generated_form.dart'; | ||||
| import 'package:obtainium/components/generated_form_modal.dart'; | ||||
| import 'package:obtainium/custom_errors.dart'; | ||||
| @@ -146,17 +147,23 @@ Future<File> downloadFileWithRetry(String url, String fileName, | ||||
|     bool fileNameHasExt, Function? onProgress, String destDir, | ||||
|     {bool useExisting = true, | ||||
|     Map<String, String>? headers, | ||||
|     int retries = 3}) async { | ||||
|     int retries = 3, | ||||
|     bool allowInsecure = false}) async { | ||||
|   try { | ||||
|     return await downloadFile( | ||||
|         url, fileName, fileNameHasExt, onProgress, destDir, | ||||
|         useExisting: useExisting, headers: headers); | ||||
|         useExisting: useExisting, | ||||
|         headers: headers, | ||||
|         allowInsecure: allowInsecure); | ||||
|   } catch (e) { | ||||
|     if (retries > 0 && e is ClientException) { | ||||
|       await Future.delayed(const Duration(seconds: 5)); | ||||
|       return await downloadFileWithRetry( | ||||
|           url, fileName, fileNameHasExt, onProgress, destDir, | ||||
|           useExisting: useExisting, headers: headers, retries: (retries - 1)); | ||||
|           useExisting: useExisting, | ||||
|           headers: headers, | ||||
|           retries: (retries - 1), | ||||
|           allowInsecure: allowInsecure); | ||||
|     } else { | ||||
|       rethrow; | ||||
|     } | ||||
| @@ -173,11 +180,14 @@ String hashListOfLists(List<List<int>> data) { | ||||
| Future<String> checkPartialDownloadHashDynamic(String url, | ||||
|     {int startingSize = 1024, | ||||
|     int lowerLimit = 128, | ||||
|     Map<String, String>? headers}) async { | ||||
|     Map<String, String>? headers, | ||||
|     bool allowInsecure = false}) async { | ||||
|   for (int i = startingSize; i >= lowerLimit; i -= 256) { | ||||
|     List<String> ab = await Future.wait([ | ||||
|       checkPartialDownloadHash(url, i, headers: headers), | ||||
|       checkPartialDownloadHash(url, i, headers: headers) | ||||
|       checkPartialDownloadHash(url, i, | ||||
|           headers: headers, allowInsecure: allowInsecure), | ||||
|       checkPartialDownloadHash(url, i, | ||||
|           headers: headers, allowInsecure: allowInsecure) | ||||
|     ]); | ||||
|     if (ab[0] == ab[1]) { | ||||
|       return ab[0]; | ||||
| @@ -187,13 +197,13 @@ Future<String> checkPartialDownloadHashDynamic(String url, | ||||
| } | ||||
|  | ||||
| Future<String> checkPartialDownloadHash(String url, int bytesToGrab, | ||||
|     {Map<String, String>? headers}) async { | ||||
|     {Map<String, String>? headers, bool allowInsecure = false}) async { | ||||
|   var req = Request('GET', Uri.parse(url)); | ||||
|   if (headers != null) { | ||||
|     req.headers.addAll(headers); | ||||
|   } | ||||
|   req.headers[HttpHeaders.rangeHeader] = 'bytes=0-$bytesToGrab'; | ||||
|   var client = http.Client(); | ||||
|   var client = IOClient(createHttpClient(allowInsecure)); | ||||
|   var response = await client.send(req); | ||||
|   if (response.statusCode < 200 || response.statusCode > 299) { | ||||
|     throw ObtainiumError(response.reasonPhrase ?? tr('unexpectedError')); | ||||
| @@ -204,12 +214,14 @@ Future<String> checkPartialDownloadHash(String url, int bytesToGrab, | ||||
|  | ||||
| Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|     Function? onProgress, String destDir, | ||||
|     {bool useExisting = true, Map<String, String>? headers}) async { | ||||
|     {bool useExisting = true, | ||||
|     Map<String, String>? headers, | ||||
|     bool allowInsecure = false}) async { | ||||
|   // Send the initial request but cancel it as soon as you have the headers | ||||
|   var reqHeaders = headers ?? {}; | ||||
|   var req = Request('GET', Uri.parse(url)); | ||||
|   req.headers.addAll(reqHeaders); | ||||
|   var client = http.Client(); | ||||
|   var client = IOClient(createHttpClient(allowInsecure)); | ||||
|   StreamedResponse response = await client.send(req); | ||||
|   var resHeaders = response.headers; | ||||
|  | ||||
| @@ -220,10 +232,14 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|   if (ext.endsWith('"') || ext.endsWith("other")) { | ||||
|     ext = ext.substring(0, ext.length - 1); | ||||
|   } | ||||
|   if (url.toLowerCase().endsWith('.apk') && ext != 'apk') { | ||||
|   if (((Uri.tryParse(url)?.path ?? url).toLowerCase().endsWith('.apk') || | ||||
|           ext == 'attachment') && | ||||
|       ext != 'apk') { | ||||
|     ext = 'apk'; | ||||
|   } | ||||
|   fileName = fileName.split('/').last; // Ensure the fileName is a file name | ||||
|   fileName = fileNameHasExt | ||||
|       ? fileName | ||||
|       : fileName.split('/').last; // Ensure the fileName is a file name | ||||
|   File downloadedFile = File('$destDir/$fileName.$ext'); | ||||
|   if (fileNameHasExt) { | ||||
|     // If the user says the filename already has an ext, ignore whatever you inferred from above | ||||
| @@ -271,7 +287,7 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
|   IOSink? sink; | ||||
|   if (rangeFeatureEnabled && fullContentLength != null && rangeStart > 0) { | ||||
|     client.close(); | ||||
|     client = http.Client(); | ||||
|     client = IOClient(createHttpClient(allowInsecure)); | ||||
|     req = Request('GET', Uri.parse(url)); | ||||
|     req.headers.addAll(reqHeaders); | ||||
|     req.headers.addAll({'range': 'bytes=$rangeStart-${fullContentLength - 1}'}); | ||||
| @@ -314,12 +330,12 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||
| } | ||||
|  | ||||
| Future<Map<String, String>> getHeaders(String url, | ||||
|     {Map<String, String>? headers}) async { | ||||
|     {Map<String, String>? headers, bool allowInsecure = false}) async { | ||||
|   var req = http.Request('GET', Uri.parse(url)); | ||||
|   if (headers != null) { | ||||
|     req.headers.addAll(headers); | ||||
|   } | ||||
|   var client = http.Client(); | ||||
|   var client = IOClient(createHttpClient(allowInsecure)); | ||||
|   var response = await client.send(req); | ||||
|   if (response.statusCode < 200 || response.statusCode > 299) { | ||||
|     throw ObtainiumError(response.reasonPhrase ?? tr('unexpectedError')); | ||||
| @@ -329,6 +345,10 @@ Future<Map<String, String>> getHeaders(String url, | ||||
|   return returnHeaders; | ||||
| } | ||||
|  | ||||
| Future<List<PackageInfo>> getAllInstalledInfo() async { | ||||
|   return await pm.getInstalledPackages() ?? []; | ||||
| } | ||||
|  | ||||
| Future<PackageInfo?> getInstalledInfo(String? packageName, | ||||
|     {bool printErr = true}) async { | ||||
|   if (packageName != null) { | ||||
| @@ -355,28 +375,40 @@ class AppsProvider with ChangeNotifier { | ||||
|   late Stream<FGBGType>? foregroundStream; | ||||
|   late StreamSubscription<FGBGType>? foregroundSubscription; | ||||
|   late Directory APKDir; | ||||
|   late Directory iconsCacheDir; | ||||
|   late SettingsProvider settingsProvider = SettingsProvider(); | ||||
|  | ||||
|   Iterable<AppInMemory> getAppValues() => apps.values.map((a) => a.deepCopy()); | ||||
|  | ||||
|   AppsProvider({isBg = false}) { | ||||
|     // Subscribe to changes in the app foreground status | ||||
|     foregroundStream = FGBGEvents.stream.asBroadcastStream(); | ||||
|     foregroundStream = FGBGEvents.instance.stream.asBroadcastStream(); | ||||
|     foregroundSubscription = foregroundStream?.listen((event) async { | ||||
|       isForeground = event == FGBGType.foreground; | ||||
|       if (isForeground) loadApps(); | ||||
|       if (isForeground) { | ||||
|         await loadApps(); | ||||
|       } | ||||
|     }); | ||||
|     () async { | ||||
|       await settingsProvider.initializeSettings(); | ||||
|       var cacheDirs = await getExternalCacheDirectories(); | ||||
|       if (cacheDirs?.isNotEmpty ?? false) { | ||||
|         APKDir = cacheDirs!.first; | ||||
|         iconsCacheDir = Directory('${cacheDirs.first.path}/icons'); | ||||
|         if (!iconsCacheDir.existsSync()) { | ||||
|           iconsCacheDir.createSync(); | ||||
|         } | ||||
|       } else { | ||||
|         APKDir = | ||||
|             Directory('${(await getExternalStorageDirectory())!.path}/apks'); | ||||
|         if (!APKDir.existsSync()) { | ||||
|           APKDir.createSync(); | ||||
|         } | ||||
|         iconsCacheDir = | ||||
|             Directory('${(await getExternalStorageDirectory())!.path}/icons'); | ||||
|         if (!iconsCacheDir.existsSync()) { | ||||
|           iconsCacheDir.createSync(); | ||||
|         } | ||||
|       } | ||||
|       if (!isBg) { | ||||
|         // Load Apps into memory (in background processes, this is done later instead of in the constructor) | ||||
| @@ -421,7 +453,8 @@ class AppsProvider with ChangeNotifier { | ||||
|   } | ||||
|  | ||||
|   Future<Object> downloadApp(App app, BuildContext? context, | ||||
|       {NotificationsProvider? notificationsProvider}) async { | ||||
|       {NotificationsProvider? notificationsProvider, | ||||
|       bool useExisting = true}) async { | ||||
|     var notifId = DownloadNotification(app.finalName, 0).id; | ||||
|     if (apps[app.id] != null) { | ||||
|       apps[app.id]!.downloadProgress = 0; | ||||
| @@ -438,11 +471,15 @@ class AppsProvider with ChangeNotifier { | ||||
|       notificationsProvider?.cancel(notif.id); | ||||
|       int? prevProg; | ||||
|       var fileNameNoExt = '${app.id}-${downloadUrl.hashCode}'; | ||||
|       if (source.urlsAlwaysHaveExtension) { | ||||
|         fileNameNoExt = | ||||
|             '$fileNameNoExt.${app.apkUrls[app.preferredApkIndex].key.split('.').last}'; | ||||
|       } | ||||
|       var headers = await source.getRequestHeaders(app.additionalSettings, | ||||
|           forAPKDownload: true); | ||||
|       var downloadedFile = await downloadFileWithRetry( | ||||
|           downloadUrl, fileNameNoExt, false, headers: headers, | ||||
|           (double? progress) { | ||||
|           downloadUrl, fileNameNoExt, source.urlsAlwaysHaveExtension, | ||||
|           headers: headers, (double? progress) { | ||||
|         int? prog = progress?.ceil(); | ||||
|         if (apps[app.id] != null) { | ||||
|           apps[app.id]!.downloadProgress = progress; | ||||
| @@ -453,7 +490,9 @@ class AppsProvider with ChangeNotifier { | ||||
|           notificationsProvider?.notify(notif); | ||||
|         } | ||||
|         prevProg = prog; | ||||
|       }, APKDir.path); | ||||
|       }, APKDir.path, | ||||
|           useExisting: useExisting, | ||||
|           allowInsecure: app.additionalSettings['allowInsecure'] == true); | ||||
|       // Set to 90 for remaining steps, will make null in 'finally' | ||||
|       if (apps[app.id] != null) { | ||||
|         apps[app.id]!.downloadProgress = -1; | ||||
| @@ -476,8 +515,17 @@ class AppsProvider with ChangeNotifier { | ||||
|             .listSync() | ||||
|             .where((e) => e.path.toLowerCase().endsWith('.apk')) | ||||
|             .toList(); | ||||
|         newInfo = | ||||
|             await pm.getPackageArchiveInfo(archiveFilePath: apks.first.path); | ||||
|         for (var i = 0; i < apks.length; i++) { | ||||
|           try { | ||||
|             newInfo = await pm.getPackageArchiveInfo( | ||||
|                 archiveFilePath: apks.first.path); | ||||
|             break; | ||||
|           } catch (e) { | ||||
|             if (i == apks.length - 1) { | ||||
|               rethrow; | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       if (newInfo == null) { | ||||
|         downloadedFile.delete(); | ||||
| @@ -563,7 +611,7 @@ class AppsProvider with ChangeNotifier { | ||||
|     if (!isForeground) { | ||||
|       await notificationsProvider.notify(completeInstallationNotification, | ||||
|           cancelExisting: true); | ||||
|       while (await FGBGEvents.stream.first != FGBGType.foreground) {} | ||||
|       while (await FGBGEvents.instance.stream.first != FGBGType.foreground) {} | ||||
|       await notificationsProvider.cancel(completeInstallationNotification.id); | ||||
|     } | ||||
|   } | ||||
| @@ -710,7 +758,8 @@ class AppsProvider with ChangeNotifier { | ||||
|   } | ||||
|  | ||||
|   Future<MapEntry<String, String>?> confirmAppFileUrl( | ||||
|       App app, BuildContext? context, bool pickAnyAsset) async { | ||||
|       App app, BuildContext? context, bool pickAnyAsset, | ||||
|       {bool evenIfSingleChoice = false}) async { | ||||
|     var urlsToSelectFrom = app.apkUrls; | ||||
|     if (pickAnyAsset) { | ||||
|       urlsToSelectFrom = [...urlsToSelectFrom, ...app.otherAssetUrls]; | ||||
| @@ -721,7 +770,8 @@ class AppsProvider with ChangeNotifier { | ||||
|     // get device supported architecture | ||||
|     List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||
|  | ||||
|     if (urlsToSelectFrom.length > 1 && context != null) { | ||||
|     if ((urlsToSelectFrom.length > 1 || evenIfSingleChoice) && | ||||
|         context != null) { | ||||
|       appFileUrl = await showDialog( | ||||
|           // ignore: use_build_context_synchronously | ||||
|           context: context, | ||||
| @@ -766,7 +816,8 @@ class AppsProvider with ChangeNotifier { | ||||
|   Future<List<String>> downloadAndInstallLatestApps( | ||||
|       List<String> appIds, BuildContext? context, | ||||
|       {NotificationsProvider? notificationsProvider, | ||||
|       bool forceParallelDownloads = false}) async { | ||||
|       bool forceParallelDownloads = false, | ||||
|       bool useExisting = true}) async { | ||||
|     notificationsProvider = | ||||
|         notificationsProvider ?? context?.read<NotificationsProvider>(); | ||||
|     List<String> appsToInstall = []; | ||||
| @@ -818,21 +869,86 @@ class AppsProvider with ChangeNotifier { | ||||
|     appsToInstall = | ||||
|         moveStrToEnd(appsToInstall, obtainiumId, strB: obtainiumTempId); | ||||
|  | ||||
|     Future<String> updateFn(String id, {bool skipInstalls = false}) async { | ||||
|     Future<void> installFn(String id, bool willBeSilent, | ||||
|         DownloadedApk? downloadedFile, DownloadedXApkDir? downloadedDir) async { | ||||
|       apps[id]?.downloadProgress = -1; | ||||
|       notifyListeners(); | ||||
|       try { | ||||
|         bool sayInstalled = true; | ||||
|         var contextIfNewInstall = | ||||
|             apps[id]?.installedInfo == null ? context : null; | ||||
|         bool needBGWorkaround = | ||||
|             willBeSilent && context == null && !settingsProvider.useShizuku; | ||||
|         if (downloadedFile != null) { | ||||
|           if (needBGWorkaround) { | ||||
|             // ignore: use_build_context_synchronously | ||||
|             installApk(downloadedFile, contextIfNewInstall, | ||||
|                 needsBGWorkaround: true, | ||||
|                 shizukuPretendToBeGooglePlay: apps[id]! | ||||
|                         .app | ||||
|                         .additionalSettings['shizukuPretendToBeGooglePlay'] == | ||||
|                     true); | ||||
|           } else { | ||||
|             // ignore: use_build_context_synchronously | ||||
|             sayInstalled = await installApk(downloadedFile, contextIfNewInstall, | ||||
|                 shizukuPretendToBeGooglePlay: apps[id]! | ||||
|                         .app | ||||
|                         .additionalSettings['shizukuPretendToBeGooglePlay'] == | ||||
|                     true); | ||||
|           } | ||||
|         } else { | ||||
|           if (needBGWorkaround) { | ||||
|             // ignore: use_build_context_synchronously | ||||
|             installXApkDir(downloadedDir!, contextIfNewInstall, | ||||
|                 needsBGWorkaround: true); | ||||
|           } else { | ||||
|             // ignore: use_build_context_synchronously | ||||
|             sayInstalled = await installXApkDir( | ||||
|                 downloadedDir!, contextIfNewInstall, | ||||
|                 shizukuPretendToBeGooglePlay: apps[id]! | ||||
|                         .app | ||||
|                         .additionalSettings['shizukuPretendToBeGooglePlay'] == | ||||
|                     true); | ||||
|           } | ||||
|         } | ||||
|         if (willBeSilent && context == null) { | ||||
|           if (!settingsProvider.useShizuku) { | ||||
|             notificationsProvider?.notify(SilentUpdateAttemptNotification( | ||||
|                 [apps[id]!.app], | ||||
|                 id: id.hashCode)); | ||||
|           } else { | ||||
|             notificationsProvider?.notify(SilentUpdateNotification( | ||||
|                 [apps[id]!.app], sayInstalled, | ||||
|                 id: id.hashCode)); | ||||
|           } | ||||
|         } | ||||
|         if (sayInstalled) { | ||||
|           installedIds.add(id); | ||||
|         } | ||||
|       } finally { | ||||
|         apps[id]?.downloadProgress = null; | ||||
|         notifyListeners(); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     Future<Map<Object?, Object?>> downloadFn(String id, | ||||
|         {bool skipInstalls = false}) async { | ||||
|       bool willBeSilent = false; | ||||
|       DownloadedApk? downloadedFile; | ||||
|       DownloadedXApkDir? downloadedDir; | ||||
|       try { | ||||
|         var downloadedArtifact = | ||||
|             // ignore: use_build_context_synchronously | ||||
|             await downloadApp(apps[id]!.app, context, | ||||
|                 notificationsProvider: notificationsProvider); | ||||
|         DownloadedApk? downloadedFile; | ||||
|         DownloadedXApkDir? downloadedDir; | ||||
|                 notificationsProvider: notificationsProvider, | ||||
|                 useExisting: useExisting); | ||||
|         if (downloadedArtifact is DownloadedApk) { | ||||
|           downloadedFile = downloadedArtifact; | ||||
|         } else { | ||||
|           downloadedDir = downloadedArtifact as DownloadedXApkDir; | ||||
|         } | ||||
|         id = downloadedFile?.appId ?? downloadedDir!.appId; | ||||
|         bool willBeSilent = await canInstallSilently(apps[id]!.app); | ||||
|         willBeSilent = await canInstallSilently(apps[id]!.app); | ||||
|         if (!settingsProvider.useShizuku) { | ||||
|           if (!(await settingsProvider.getInstallPermission(enforce: false))) { | ||||
|             throw ObtainiumError(tr('cancelled')); | ||||
| @@ -853,79 +969,37 @@ class AppsProvider with ChangeNotifier { | ||||
|           // ignore: use_build_context_synchronously | ||||
|           await waitForUserToReturnToForeground(context); | ||||
|         } | ||||
|         apps[id]?.downloadProgress = -1; | ||||
|         notifyListeners(); | ||||
|         try { | ||||
|           if (!skipInstalls) { | ||||
|             bool sayInstalled = true; | ||||
|             var contextIfNewInstall = | ||||
|                 apps[id]?.installedInfo == null ? context : null; | ||||
|             bool needBGWorkaround = | ||||
|                 willBeSilent && context == null && !settingsProvider.useShizuku; | ||||
|             if (downloadedFile != null) { | ||||
|               if (needBGWorkaround) { | ||||
|                 // ignore: use_build_context_synchronously | ||||
|                 installApk(downloadedFile, contextIfNewInstall, | ||||
|                     needsBGWorkaround: true); | ||||
|               } else { | ||||
|                 // ignore: use_build_context_synchronously | ||||
|                 sayInstalled = await installApk( | ||||
|                     downloadedFile, contextIfNewInstall, | ||||
|                     shizukuPretendToBeGooglePlay: | ||||
|                         apps[id]!.app.additionalSettings[ | ||||
|                                 'shizukuPretendToBeGooglePlay'] == | ||||
|                             true); | ||||
|               } | ||||
|             } else { | ||||
|               if (needBGWorkaround) { | ||||
|                 // ignore: use_build_context_synchronously | ||||
|                 installXApkDir(downloadedDir!, contextIfNewInstall, | ||||
|                     needsBGWorkaround: true); | ||||
|               } else { | ||||
|                 // ignore: use_build_context_synchronously | ||||
|                 sayInstalled = await installXApkDir( | ||||
|                     downloadedDir!, contextIfNewInstall, | ||||
|                     shizukuPretendToBeGooglePlay: | ||||
|                         apps[id]!.app.additionalSettings[ | ||||
|                                 'shizukuPretendToBeGooglePlay'] == | ||||
|                             true); | ||||
|               } | ||||
|             } | ||||
|             if (willBeSilent && context == null) { | ||||
|               if (!settingsProvider.useShizuku) { | ||||
|                 notificationsProvider?.notify(SilentUpdateAttemptNotification( | ||||
|                     [apps[id]!.app], | ||||
|                     id: id.hashCode)); | ||||
|               } else { | ||||
|                 notificationsProvider?.notify(SilentUpdateNotification( | ||||
|                     [apps[id]!.app], sayInstalled, | ||||
|                     id: id.hashCode)); | ||||
|               } | ||||
|             } | ||||
|             if (sayInstalled) { | ||||
|               installedIds.add(id); | ||||
|             } | ||||
|           } | ||||
|         } finally { | ||||
|           apps[id]?.downloadProgress = null; | ||||
|           notifyListeners(); | ||||
|         } | ||||
|       } catch (e) { | ||||
|         errors.add(id, e, appName: apps[id]?.name); | ||||
|       } | ||||
|       return id; | ||||
|       return { | ||||
|         'id': id, | ||||
|         'willBeSilent': willBeSilent, | ||||
|         'downloadedFile': downloadedFile, | ||||
|         'downloadedDir': downloadedDir | ||||
|       }; | ||||
|     } | ||||
|  | ||||
|     List<Map<Object?, Object?>> downloadResults = []; | ||||
|     if (forceParallelDownloads || !settingsProvider.parallelDownloads) { | ||||
|       for (var id in appsToInstall) { | ||||
|         await updateFn(id); | ||||
|         downloadResults.add(await downloadFn(id)); | ||||
|       } | ||||
|     } else { | ||||
|       List<String> ids = await Future.wait( | ||||
|           appsToInstall.map((id) => updateFn(id, skipInstalls: true))); | ||||
|       for (var id in ids) { | ||||
|         if (!errors.appIdNames.containsKey(id)) { | ||||
|           await updateFn(id); | ||||
|       downloadResults = await Future.wait( | ||||
|           appsToInstall.map((id) => downloadFn(id, skipInstalls: true))); | ||||
|     } | ||||
|     for (var res in downloadResults) { | ||||
|       if (!errors.appIdNames.containsKey(res['id'])) { | ||||
|         try { | ||||
|           await installFn( | ||||
|               res['id'] as String, | ||||
|               res['willBeSilent'] as bool, | ||||
|               res['downloadedFile'] as DownloadedApk?, | ||||
|               res['downloadedDir'] as DownloadedXApkDir?); | ||||
|         } catch (e) { | ||||
|           var id = res['id'] as String; | ||||
|           errors.add(id, e, appName: apps[id]?.name); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| @@ -951,7 +1025,8 @@ class AppsProvider with ChangeNotifier { | ||||
|       if (apps[id]!.app.apkUrls.isNotEmpty || | ||||
|           apps[id]!.app.otherAssetUrls.isNotEmpty) { | ||||
|         // ignore: use_build_context_synchronously | ||||
|         fileUrl = await confirmAppFileUrl(apps[id]!.app, context, true); | ||||
|         fileUrl = await confirmAppFileUrl(apps[id]!.app, context, true, | ||||
|             evenIfSingleChoice: true); | ||||
|       } | ||||
|       if (fileUrl != null) { | ||||
|         filesToDownload.add(MapEntry(fileUrl, apps[id]!.app)); | ||||
| @@ -985,7 +1060,8 @@ class AppsProvider with ChangeNotifier { | ||||
|                 .getRequestHeaders(app.additionalSettings, | ||||
|                     forAPKDownload: | ||||
|                         fileUrl.key.endsWith('.apk') ? true : false), | ||||
|             useExisting: false); | ||||
|             useExisting: false, | ||||
|             allowInsecure: app.additionalSettings['allowInsecure'] == true); | ||||
|         notificationsProvider | ||||
|             .notify(DownloadedNotification(fileUrl.key, fileUrl.value)); | ||||
|       } catch (e) { | ||||
| @@ -1144,17 +1220,6 @@ class AppsProvider with ChangeNotifier { | ||||
|         : false; | ||||
|   } | ||||
|  | ||||
|   Future<void> updateInstallStatusInMemory(AppInMemory app) async { | ||||
|     apps[app.app.id]?.installedInfo = await getInstalledInfo(app.app.id); | ||||
|     apps[app.app.id]?.icon = | ||||
|         await apps[app.app.id]?.installedInfo?.applicationInfo?.getAppIcon(); | ||||
|     apps[app.app.id]?.app.name = await (apps[app.app.id] | ||||
|             ?.installedInfo | ||||
|             ?.applicationInfo | ||||
|             ?.getAppLabel()) ?? | ||||
|         app.name; | ||||
|   } | ||||
|  | ||||
|   Future<void> loadApps({String? singleId}) async { | ||||
|     while (loadingApps) { | ||||
|       await Future.delayed(const Duration(microseconds: 1)); | ||||
| @@ -1163,67 +1228,75 @@ class AppsProvider with ChangeNotifier { | ||||
|     notifyListeners(); | ||||
|     var sp = SourceProvider(); | ||||
|     List<List<String>> errors = []; | ||||
|     List<App?> newApps = (await getAppsDir()) // Parse Apps from JSON | ||||
|     var installedAppsData = await getAllInstalledInfo(); | ||||
|     List<String> removedAppIds = []; | ||||
|     await Future.wait((await getAppsDir()) // Parse Apps from JSON | ||||
|         .listSync() | ||||
|         .where((item) => item.path.toLowerCase().endsWith('.json')) | ||||
|         .where((item) => | ||||
|             singleId == null || | ||||
|             item.path.split('/').last.toLowerCase() == | ||||
|                 '${singleId.toLowerCase()}.json') | ||||
|         .map((e) { | ||||
|       try { | ||||
|         return App.fromJson(jsonDecode(File(e.path).readAsStringSync())); | ||||
|       } catch (err) { | ||||
|         if (err is FormatException) { | ||||
|           logs.add('Corrupt JSON when loading App (will be ignored): $e'); | ||||
|           e.renameSync('${e.path}.corrupt'); | ||||
|         } else { | ||||
|           rethrow; | ||||
|         .map((item) async { | ||||
|       App? app; | ||||
|       if (item.path.toLowerCase().endsWith('.json') && | ||||
|           (singleId == null || | ||||
|               item.path.split('/').last.toLowerCase() == | ||||
|                   '${singleId.toLowerCase()}.json')) { | ||||
|         try { | ||||
|           app = App.fromJson(jsonDecode(File(item.path).readAsStringSync())); | ||||
|         } catch (err) { | ||||
|           if (err is FormatException) { | ||||
|             logs.add('Corrupt JSON when loading App (will be ignored): $e'); | ||||
|             item.renameSync('${item.path}.corrupt'); | ||||
|           } else { | ||||
|             rethrow; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     }).toList(); | ||||
|     for (var app in newApps) { | ||||
|       // Put Apps into memory to list them (fast) | ||||
|       if (app != null) { | ||||
|         // Save the app to the in-memory list without grabbing any OS info first | ||||
|         apps.update( | ||||
|             app.id, | ||||
|             (value) => AppInMemory( | ||||
|                 app!, value.downloadProgress, value.installedInfo, value.icon), | ||||
|             ifAbsent: () => AppInMemory(app!, null, null, null)); | ||||
|         notifyListeners(); | ||||
|         try { | ||||
|           // Try getting the app's source to ensure no invalid apps get loaded | ||||
|           sp.getSource(app.url, overrideSource: app.overrideSource); | ||||
|           // If the app is installed, grab its OS data and reconcile install statuses | ||||
|           PackageInfo? installedInfo; | ||||
|           try { | ||||
|             installedInfo = | ||||
|                 installedAppsData.firstWhere((i) => i.packageName == app!.id); | ||||
|           } catch (e) { | ||||
|             // If the app isn't installed the above throws an error | ||||
|           } | ||||
|           // Reconcile differences between the installed and recorded install info | ||||
|           var moddedApp = | ||||
|               getCorrectedInstallStatusAppIfPossible(app, installedInfo); | ||||
|           if (moddedApp != null) { | ||||
|             app = moddedApp; | ||||
|             // Note the app ID if it was uninstalled externally | ||||
|             if (moddedApp.installedVersion == null) { | ||||
|               removedAppIds.add(moddedApp.id); | ||||
|             } | ||||
|           } | ||||
|           // Update the app in memory with install info and corrections | ||||
|           apps.update( | ||||
|               app.id, | ||||
|               (value) => AppInMemory( | ||||
|                   app, value.downloadProgress, value.installedInfo, value.icon), | ||||
|               ifAbsent: () => AppInMemory(app, null, null, null)); | ||||
|                   app!, value.downloadProgress, installedInfo, value.icon), | ||||
|               ifAbsent: () => AppInMemory(app!, null, installedInfo, null)); | ||||
|           notifyListeners(); | ||||
|         } catch (e) { | ||||
|           errors.add([app.id, app.finalName, e.toString()]); | ||||
|           errors.add([app!.id, app.finalName, e.toString()]); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     notifyListeners(); | ||||
|     })); | ||||
|     if (errors.isNotEmpty) { | ||||
|       removeApps(errors.map((e) => e[0]).toList()); | ||||
|       NotificationsProvider().notify( | ||||
|           AppsRemovedNotification(errors.map((e) => [e[1], e[2]]).toList())); | ||||
|     } | ||||
|     // Get install status and other OS info for each App (slow) | ||||
|     await Future.wait(apps.values.map((app) { | ||||
|       return updateInstallStatusInMemory(app); | ||||
|     })); | ||||
|     notifyListeners(); | ||||
|     // Reconcile version differences | ||||
|     List<App> modifiedApps = []; | ||||
|     for (var app in apps.values) { | ||||
|       var moddedApp = | ||||
|           getCorrectedInstallStatusAppIfPossible(app.app, app.installedInfo); | ||||
|       if (moddedApp != null) { | ||||
|         modifiedApps.add(moddedApp); | ||||
|       } | ||||
|     } | ||||
|     if (modifiedApps.isNotEmpty) { | ||||
|       await saveApps(modifiedApps, attemptToCorrectInstallStatus: false); | ||||
|       var removedAppIds = modifiedApps | ||||
|           .where((a) => a.installedVersion == null) | ||||
|           .map((e) => e.id) | ||||
|           .toList(); | ||||
|       // After reconciliation, delete externally uninstalled Apps if needed | ||||
|     // Delete externally uninstalled Apps if needed | ||||
|     if (removedAppIds.isNotEmpty) { | ||||
|       if (removedAppIds.isNotEmpty) { | ||||
|         if (settingsProvider.removeOnExternalUninstall) { | ||||
|           await removeApps(removedAppIds); | ||||
| @@ -1234,11 +1307,33 @@ class AppsProvider with ChangeNotifier { | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<void> updateAppIcon(String? appId, {bool ignoreCache = false}) async { | ||||
|     if (apps[appId]?.icon == null) { | ||||
|       var cachedIcon = File('${iconsCacheDir.path}/$appId.png'); | ||||
|       var alreadyCached = cachedIcon.existsSync() && !ignoreCache; | ||||
|       var icon = alreadyCached | ||||
|           ? (await cachedIcon.readAsBytes()) | ||||
|           : (await apps[appId]?.installedInfo?.applicationInfo?.getAppIcon()); | ||||
|       if (icon != null && !alreadyCached) { | ||||
|         cachedIcon.writeAsBytes(icon.toList()); | ||||
|       } | ||||
|       if (icon != null) { | ||||
|         apps.update( | ||||
|             apps[appId]!.app.id, | ||||
|             (value) => AppInMemory(apps[appId]!.app, value.downloadProgress, | ||||
|                 value.installedInfo, icon), | ||||
|             ifAbsent: () => AppInMemory( | ||||
|                 apps[appId]!.app, null, apps[appId]?.installedInfo, icon)); | ||||
|         notifyListeners(); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> saveApps(List<App> apps, | ||||
|       {bool attemptToCorrectInstallStatus = true, | ||||
|       bool onlyIfExists = true}) async { | ||||
|     attemptToCorrectInstallStatus = attemptToCorrectInstallStatus; | ||||
|     for (var a in apps) { | ||||
|     await Future.wait(apps.map((a) async { | ||||
|       var app = a.deepCopy(); | ||||
|       PackageInfo? info = await getInstalledInfo(app.id); | ||||
|       var icon = await info?.applicationInfo?.getAppIcon(); | ||||
| @@ -1260,14 +1355,14 @@ class AppsProvider with ChangeNotifier { | ||||
|           rethrow; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     })); | ||||
|     notifyListeners(); | ||||
|     export(isAuto: true); | ||||
|   } | ||||
|  | ||||
|   Future<void> removeApps(List<String> appIds) async { | ||||
|     var apkFiles = APKDir.listSync(); | ||||
|     for (var appId in appIds) { | ||||
|     await Future.wait(appIds.map((appId) async { | ||||
|       File file = File('${(await getAppsDir()).path}/$appId.json'); | ||||
|       if (file.existsSync()) { | ||||
|         file.deleteSync(recursive: true); | ||||
| @@ -1281,7 +1376,7 @@ class AppsProvider with ChangeNotifier { | ||||
|       if (apps.containsKey(appId)) { | ||||
|         apps.remove(appId); | ||||
|       } | ||||
|     } | ||||
|     })); | ||||
|     if (appIds.isNotEmpty) { | ||||
|       notifyListeners(); | ||||
|       export(isAuto: true); | ||||
| @@ -1461,6 +1556,34 @@ class AppsProvider with ChangeNotifier { | ||||
|     return updateAppIds; | ||||
|   } | ||||
|  | ||||
|   Map<String, dynamic> generateExportJSON( | ||||
|       {List<String>? appIds, bool? overrideExportSettings}) { | ||||
|     Map<String, dynamic> finalExport = {}; | ||||
|     finalExport['apps'] = apps.values | ||||
|         .where((e) { | ||||
|           if (appIds == null) { | ||||
|             return true; | ||||
|           } else { | ||||
|             return appIds.contains(e.app.id); | ||||
|           } | ||||
|         }) | ||||
|         .map((e) => e.app.toJson()) | ||||
|         .toList(); | ||||
|     bool shouldExportSettings = settingsProvider.exportSettings; | ||||
|     if (overrideExportSettings != null) { | ||||
|       shouldExportSettings = overrideExportSettings; | ||||
|     } | ||||
|     if (shouldExportSettings) { | ||||
|       finalExport['settings'] = Map<String, Object?>.fromEntries( | ||||
|           (settingsProvider.prefs | ||||
|                   ?.getKeys() | ||||
|                   .map((key) => MapEntry(key, settingsProvider.prefs?.get(key))) | ||||
|                   .toList()) ?? | ||||
|               []); | ||||
|     } | ||||
|     return finalExport; | ||||
|   } | ||||
|  | ||||
|   Future<String?> export( | ||||
|       {bool pickOnly = false, isAuto = false, SettingsProvider? sp}) async { | ||||
|     SettingsProvider settingsProvider = sp ?? this.settingsProvider; | ||||
| @@ -1491,22 +1614,13 @@ class AppsProvider with ChangeNotifier { | ||||
|     } | ||||
|     String? returnPath; | ||||
|     if (!pickOnly) { | ||||
|       Map<String, dynamic> finalExport = {}; | ||||
|       finalExport['apps'] = apps.values.map((e) => e.app.toJson()).toList(); | ||||
|       if (settingsProvider.exportSettings) { | ||||
|         finalExport['settings'] = Map<String, Object?>.fromEntries( | ||||
|             (settingsProvider.prefs | ||||
|                     ?.getKeys() | ||||
|                     .map((key) => | ||||
|                         MapEntry(key, settingsProvider.prefs?.get(key))) | ||||
|                     .toList()) ?? | ||||
|                 []); | ||||
|       } | ||||
|       var encoder = const JsonEncoder.withIndent("    "); | ||||
|       Map<String, dynamic> finalExport = generateExportJSON(); | ||||
|       var result = await saf.createFile(exportDir, | ||||
|           displayName: | ||||
|               '${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().toIso8601String().replaceAll(':', '-')}${isAuto ? '-auto' : ''}.json', | ||||
|           mimeType: 'application/json', | ||||
|           bytes: Uint8List.fromList(utf8.encode(jsonEncode(finalExport)))); | ||||
|           bytes: Uint8List.fromList(utf8.encode(encoder.convert(finalExport)))); | ||||
|       if (result == null) { | ||||
|         throw ObtainiumError(tr('unexpectedError')); | ||||
|       } | ||||
| @@ -1540,6 +1654,8 @@ class AppsProvider with ChangeNotifier { | ||||
|       settingsMap.forEach((key, value) { | ||||
|         if (value is int) { | ||||
|           settingsProvider.prefs?.setInt(key, value); | ||||
|         } else if (value is double) { | ||||
|           settingsProvider.prefs?.setDouble(key, value); | ||||
|         } else if (value is bool) { | ||||
|           settingsProvider.prefs?.setBool(key, value); | ||||
|         } else if (value is List) { | ||||
| @@ -1613,7 +1729,9 @@ class _AppFilePickerState extends State<AppFilePicker> { | ||||
|           ? tr('selectX', args: [tr('releaseAsset').toLowerCase()]) | ||||
|           : tr('pickAnAPK')), | ||||
|       content: Column(children: [ | ||||
|         Text(tr('appHasMoreThanOnePackage', args: [widget.app.finalName])), | ||||
|         urlsToSelectFrom.length > 1 | ||||
|             ? Text(tr('appHasMoreThanOnePackage', args: [widget.app.finalName])) | ||||
|             : const SizedBox.shrink(), | ||||
|         const SizedBox(height: 16), | ||||
|         ...urlsToSelectFrom.map( | ||||
|           (u) => RadioListTile<String>( | ||||
| @@ -1726,7 +1844,9 @@ Future<void> bgUpdateCheck(String taskId, Map<String, dynamic>? params) async { | ||||
|   int maxRetryWaitSeconds = 5; | ||||
|  | ||||
|   var netResult = await (Connectivity().checkConnectivity()); | ||||
|   if (netResult.contains(ConnectivityResult.none)) { | ||||
|   if (netResult.contains(ConnectivityResult.none) || | ||||
|       netResult.isEmpty || | ||||
|       (netResult.contains(ConnectivityResult.vpn) && netResult.length == 1)) { | ||||
|     logs.add('BG update task: No network.'); | ||||
|     return; | ||||
|   } | ||||
|   | ||||
| @@ -2,11 +2,13 @@ | ||||
| // AppSource is an abstract class with a concrete implementation for each source | ||||
|  | ||||
| import 'dart:convert'; | ||||
| import 'dart:io'; | ||||
|  | ||||
| import 'package:device_info_plus/device_info_plus.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:html/dom.dart'; | ||||
| import 'package:http/http.dart'; | ||||
| import 'package:http/io_client.dart'; | ||||
| import 'package:obtainium/app_sources/apkmirror.dart'; | ||||
| import 'package:obtainium/app_sources/apkpure.dart'; | ||||
| import 'package:obtainium/app_sources/aptoide.dart'; | ||||
| @@ -26,6 +28,7 @@ import 'package:obtainium/app_sources/sourceforge.dart'; | ||||
| import 'package:obtainium/app_sources/sourcehut.dart'; | ||||
| import 'package:obtainium/app_sources/steammobile.dart'; | ||||
| import 'package:obtainium/app_sources/telegramapp.dart'; | ||||
| import 'package:obtainium/app_sources/tencent.dart'; | ||||
| import 'package:obtainium/app_sources/uptodown.dart'; | ||||
| import 'package:obtainium/app_sources/vlc.dart'; | ||||
| import 'package:obtainium/app_sources/whatsapp.dart'; | ||||
| @@ -354,11 +357,17 @@ preStandardizeUrl(String url) { | ||||
|       url.toLowerCase().indexOf('https://') != 0) { | ||||
|     url = 'https://$url'; | ||||
|   } | ||||
|   var uri = Uri.tryParse(url); | ||||
|   var trailingSlash = ((uri?.path.endsWith('/') ?? false) || | ||||
|           ((uri?.path.isEmpty ?? false) && url.endsWith('/'))) && | ||||
|       (uri?.queryParameters.isEmpty ?? false); | ||||
|  | ||||
|   url = url | ||||
|       .split('/') | ||||
|       .where((e) => e.isNotEmpty) | ||||
|       .join('/') | ||||
|       .replaceFirst(':/', '://'); | ||||
|           .split('/') | ||||
|           .where((e) => e.isNotEmpty) | ||||
|           .join('/') | ||||
|           .replaceFirst(':/', '://') + | ||||
|       (trailingSlash ? '/' : ''); | ||||
|   return url; | ||||
| } | ||||
|  | ||||
| @@ -393,6 +402,15 @@ getSourceRegex(List<String> hosts) { | ||||
|   return '(${hosts.join('|').replaceAll('.', '\\.')})'; | ||||
| } | ||||
|  | ||||
| HttpClient createHttpClient(bool insecure) { | ||||
|   final client = HttpClient(); | ||||
|   if (insecure) { | ||||
|     client.badCertificateCallback = | ||||
|         (X509Certificate cert, String host, int port) => true; | ||||
|   } | ||||
|   return client; | ||||
| } | ||||
|  | ||||
| abstract class AppSource { | ||||
|   List<String> hosts = []; | ||||
|   bool hostChanged = false; | ||||
| @@ -406,6 +424,7 @@ abstract class AppSource { | ||||
|   bool showReleaseDateAsVersionToggle = false; | ||||
|   bool versionDetectionDisallowed = false; | ||||
|   List<String> excludeCommonSettingKeys = []; | ||||
|   bool urlsAlwaysHaveExtension = false; | ||||
|  | ||||
|   AppSource() { | ||||
|     name = runtimeType.toString(); | ||||
| @@ -447,21 +466,33 @@ abstract class AppSource { | ||||
|  | ||||
|   Future<Response> sourceRequest( | ||||
|       String url, Map<String, dynamic> additionalSettings, | ||||
|       {bool followRedirects = true}) async { | ||||
|       {bool followRedirects = true, Object? postBody}) async { | ||||
|     var requestHeaders = await getRequestHeaders(additionalSettings); | ||||
|     if (requestHeaders != null || followRedirects == false) { | ||||
|       var req = Request('GET', Uri.parse(url)); | ||||
|       var req = Request(postBody == null ? 'GET' : 'POST', Uri.parse(url)); | ||||
|       req.followRedirects = followRedirects; | ||||
|       if (requestHeaders != null) { | ||||
|         req.headers.addAll(requestHeaders); | ||||
|       } | ||||
|       return Response.fromStream(await Client().send(req)); | ||||
|       if (postBody != null) { | ||||
|         req.headers[HttpHeaders.contentTypeHeader] = 'application/json'; | ||||
|         req.body = jsonEncode(postBody); | ||||
|       } | ||||
|       return Response.fromStream(await IOClient( | ||||
|               createHttpClient(additionalSettings['allowInsecure'] == true)) | ||||
|           .send(req)); | ||||
|     } else { | ||||
|       return get(Uri.parse(url)); | ||||
|       return postBody == null | ||||
|           ? get(Uri.parse(url)) | ||||
|           : post(Uri.parse(url), body: jsonEncode(postBody)); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   String sourceSpecificStandardizeURL(String url) { | ||||
|   void runOnAddAppInputChange(String inputUrl) { | ||||
|     // | ||||
|   } | ||||
|  | ||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||
|     throw NotImplementedError(); | ||||
|   } | ||||
|  | ||||
| @@ -485,13 +516,15 @@ abstract class AppSource { | ||||
|     ], | ||||
|     [ | ||||
|       GeneratedFormTextField('versionExtractionRegEx', | ||||
|           label: tr('versionExtractionRegEx'), | ||||
|           label: tr('trimVersionString'), | ||||
|           required: false, | ||||
|           additionalValidators: [(value) => regExValidator(value)]), | ||||
|     ], | ||||
|     [ | ||||
|       GeneratedFormTextField('matchGroupToUse', | ||||
|           label: tr('matchGroupToUse'), required: false, hint: '\$0') | ||||
|           label: tr('matchGroupToUseForX', args: [tr('trimVersionString')]), | ||||
|           required: false, | ||||
|           hint: '\$0') | ||||
|     ], | ||||
|     [ | ||||
|       GeneratedFormSwitch('versionDetection', | ||||
| @@ -523,8 +556,11 @@ abstract class AppSource { | ||||
|     [GeneratedFormTextField('appName', label: tr('appName'), required: false)], | ||||
|     [ | ||||
|       GeneratedFormSwitch('shizukuPretendToBeGooglePlay', | ||||
|           label: tr('shizukuPretendToBeGooglePlay'), | ||||
|           defaultValue: false) | ||||
|           label: tr('shizukuPretendToBeGooglePlay'), defaultValue: false) | ||||
|     ], | ||||
|     [ | ||||
|       GeneratedFormSwitch('allowInsecure', | ||||
|           label: tr('allowInsecure'), defaultValue: false) | ||||
|     ], | ||||
|     [ | ||||
|       GeneratedFormSwitch('exemptFromBackgroundUpdates', | ||||
| @@ -616,7 +652,7 @@ abstract class AppSource { | ||||
|   } | ||||
|  | ||||
|   bool canSearch = false; | ||||
|   bool excludeFromMassSearch = false; | ||||
|   bool includeAdditionalOptsInMainSearch = false; | ||||
|   List<GeneratedFormItem> searchQuerySettingFormItems = []; | ||||
|   Future<Map<String, List<String>>> search(String query, | ||||
|       {Map<String, dynamic> querySettings = const {}}) { | ||||
| @@ -752,9 +788,10 @@ class SourceProvider { | ||||
|         APKPure(), | ||||
|         Aptoide(), | ||||
|         Uptodown(), | ||||
|         APKMirror(), | ||||
|         HuaweiAppGallery(), | ||||
|         Tencent(), | ||||
|         Jenkins(), | ||||
|         APKMirror(), | ||||
|         Signal(), | ||||
|         VLC(), | ||||
|         WhatsApp(), | ||||
| @@ -797,7 +834,7 @@ class SourceProvider { | ||||
|       for (var s in sources.where( | ||||
|           (element) => element.hosts.isEmpty && !element.neverAutoSelect)) { | ||||
|         try { | ||||
|           s.sourceSpecificStandardizeURL(url); | ||||
|           s.sourceSpecificStandardizeURL(url, forSelection: true); | ||||
|           source = s; | ||||
|           break; | ||||
|         } catch (e) { | ||||
|   | ||||
							
								
								
									
										402
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										402
									
								
								pubspec.lock
									
									
									
									
									
								
							| @@ -5,10 +5,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: android_intent_plus | ||||
|       sha256: "2bfdbee8d65e7c26f88b66f0a91f2863da4d3596d8a658b4162c8de5cf04b074" | ||||
|       sha256: "007703c1b2cac7ca98add3336b98cffa4baa11d5133cc463293dba9daa39cdf6" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.0.2" | ||||
|     version: "5.1.0" | ||||
|   android_package_installer: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -47,18 +47,42 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: app_links | ||||
|       sha256: "1c2b9e9c56d80d17610bcbd111b37187875c5d0ded8654caa1bda14ea753d001" | ||||
|       sha256: ad1a6d598e7e39b46a34f746f9a8b011ee147e4c275d407fa457e7a62f84dd99 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.0.1" | ||||
|     version: "6.3.2" | ||||
|   app_links_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: app_links_linux | ||||
|       sha256: f5f7173a78609f3dfd4c2ff2c95bd559ab43c80a87dc6a095921d96c05688c81 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.0.3" | ||||
|   app_links_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: app_links_platform_interface | ||||
|       sha256: "05f5379577c513b534a29ddea68176a4d4802c46180ee8e2e966257158772a3f" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.2" | ||||
|   app_links_web: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: app_links_web | ||||
|       sha256: af060ed76183f9e2b87510a9480e56a5352b6c249778d07bd2c95fc35632a555 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.0.4" | ||||
|   archive: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: archive | ||||
|       sha256: ecf4273855368121b1caed0d10d4513c7241dfc813f7d3c8933b36622ae9b265 | ||||
|       sha256: cb6a278ef2dbb298455e1a713bda08524a175630ec643a242c399c932a0a1f7d | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.5.1" | ||||
|     version: "3.6.1" | ||||
|   args: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -79,10 +103,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: background_fetch | ||||
|       sha256: "2fe367c9be0e256dadb75b8b637b0b58a2a2d2317b7c8420bb1ae8b41e23fde3" | ||||
|       sha256: e9f26ae54d88310b7ac2a68f2f9fcee0081a4d5f11100f233a70702021e7ac4f | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.3.4" | ||||
|     version: "1.3.7" | ||||
|   boolean_selector: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -135,34 +159,34 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: connectivity_plus | ||||
|       sha256: db7a4e143dc72cc3cb2044ef9b052a7ebfe729513e6a82943bc3526f784365b8 | ||||
|       sha256: "2056db5241f96cdc0126bd94459fc4cdc13876753768fc7a31c425e50a7177d0" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.0.3" | ||||
|     version: "6.0.5" | ||||
|   connectivity_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: connectivity_plus_platform_interface | ||||
|       sha256: b6a56efe1e6675be240de39107281d4034b64ac23438026355b4234042a35adb | ||||
|       sha256: "42657c1715d48b167930d5f34d00222ac100475f73d10162ddf43e714932f204" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.0" | ||||
|     version: "2.0.1" | ||||
|   cross_file: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: cross_file | ||||
|       sha256: "55d7b444feb71301ef6b8838dbc1ae02e63dd48c8773f3810ff53bb1e2945b32" | ||||
|       sha256: "7caf6a750a0c04effbb52a676dce9a4a592e10ad35c34d6d2d0e4811160d5670" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.3.4+1" | ||||
|     version: "0.3.4+2" | ||||
|   crypto: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: crypto | ||||
|       sha256: ff625774173754681d66daaf4a448684fb04b78f902da9cb3d308c19cc5e8bab | ||||
|       sha256: ec30d999af904f33454ba22ed9a86162b35e52b44ac4807d1d93c288041d7d27 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.0.3" | ||||
|     version: "3.0.5" | ||||
|   csslib: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -191,18 +215,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: device_info_plus | ||||
|       sha256: eead12d1a1ed83d8283ab4c2f3fca23ac4082f29f25f29dff0f758f57d06ec91 | ||||
|       sha256: a7fd703482b391a87d60b6061d04dfdeab07826b96f9abd8f5ed98068acc0074 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "10.1.0" | ||||
|     version: "10.1.2" | ||||
|   device_info_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: device_info_plus_platform_interface | ||||
|       sha256: d3b01d5868b50ae571cd1dc6e502fc94d956b665756180f7b16ead09e836fd64 | ||||
|       sha256: "282d3cf731045a2feb66abfe61bbc40870ae50a3ed10a4d3d217556c35c8c2ba" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "7.0.0" | ||||
|     version: "7.0.1" | ||||
|   dynamic_color: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -247,10 +271,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: ffi | ||||
|       sha256: "493f37e7df1804778ff3a53bd691d8692ddf69702cf4c1c1096a2e41b4779e21" | ||||
|       sha256: "16ed7b077ef01ad6170a3d0c57caa4a112a38d7a2ed5602e0aca9ca6f3d98da6" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.2" | ||||
|     version: "2.1.3" | ||||
|   file: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -263,10 +287,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: file_picker | ||||
|       sha256: "29c90806ac5f5fb896547720b73b17ee9aed9bba540dc5d91fe29f8c5745b10a" | ||||
|       sha256: "167bb619cdddaa10ef2907609feb8a79c16dfa479d3afaf960f8e223f754bf12" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "8.0.3" | ||||
|     version: "8.1.2" | ||||
|   fixnum: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -279,18 +303,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: flex_color_picker | ||||
|       sha256: "5c846437069fb7afdd7ade6bf37e628a71d2ab0787095ddcb1253bf9345d5f3a" | ||||
|       sha256: "12dc855ae8ef5491f529b1fc52c655f06dcdf4114f1f7fdecafa41eec2ec8d79" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.4.1" | ||||
|     version: "3.6.0" | ||||
|   flex_seed_scheme: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flex_seed_scheme | ||||
|       sha256: "4cee2f1d07259f77e8b36f4ec5f35499d19e74e17c7dce5b819554914082bc01" | ||||
|       sha256: "7639d2c86268eff84a909026eb169f008064af0fb3696a651b24b0fa24a40334" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.5.0" | ||||
|     version: "3.4.1" | ||||
|   flutter: | ||||
|     dependency: "direct main" | ||||
|     description: flutter | ||||
| @@ -308,50 +332,98 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: flutter_fgbg | ||||
|       sha256: "08c4d2fd229e3df26083d5aecc3dea9ff4f2d188f8cd57aaf2b3f047bd08a047" | ||||
|       sha256: e02ad0738ba5fc7f331b62acb0d74aa540626a6441ae18fad685faa5ac4ad7a5 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.3.0" | ||||
|     version: "0.6.0" | ||||
|   flutter_keyboard_visibility: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_keyboard_visibility | ||||
|       sha256: "98664be7be0e3ffca00de50f7f6a287ab62c763fc8c762e0a21584584a3ff4f8" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.0.0" | ||||
|   flutter_keyboard_visibility_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_keyboard_visibility_linux | ||||
|       sha256: "6fba7cd9bb033b6ddd8c2beb4c99ad02d728f1e6e6d9b9446667398b2ac39f08" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.0.0" | ||||
|   flutter_keyboard_visibility_macos: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_keyboard_visibility_macos | ||||
|       sha256: c5c49b16fff453dfdafdc16f26bdd8fb8d55812a1d50b0ce25fc8d9f2e53d086 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.0.0" | ||||
|   flutter_keyboard_visibility_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_keyboard_visibility_platform_interface | ||||
|       sha256: e43a89845873f7be10cb3884345ceb9aebf00a659f479d1c8f4293fcb37022a4 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.0" | ||||
|   flutter_keyboard_visibility_web: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_keyboard_visibility_web | ||||
|       sha256: d3771a2e752880c79203f8d80658401d0c998e4183edca05a149f5098ce6e3d1 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.0" | ||||
|   flutter_keyboard_visibility_windows: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_keyboard_visibility_windows | ||||
|       sha256: fc4b0f0b6be9b93ae527f3d527fb56ee2d918cd88bbca438c478af7bcfd0ef73 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.0.0" | ||||
|   flutter_launcher_icons: | ||||
|     dependency: "direct dev" | ||||
|     description: | ||||
|       name: flutter_launcher_icons | ||||
|       sha256: "526faf84284b86a4cb36d20a5e45147747b7563d921373d4ee0559c54fcdbcea" | ||||
|       sha256: "619817c4b65b322b5104b6bb6dfe6cda62d9729bd7ad4303ecc8b4e690a67a77" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.13.1" | ||||
|     version: "0.14.1" | ||||
|   flutter_lints: | ||||
|     dependency: "direct dev" | ||||
|     description: | ||||
|       name: flutter_lints | ||||
|       sha256: "3f41d009ba7172d5ff9be5f6e6e6abb4300e263aab8866d2a0842ed2a70f8f0c" | ||||
|       sha256: "5398f14efa795ffb7a33e9b6a08798b26a180edac4ad7db3f231e40f82ce11e1" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.0.0" | ||||
|     version: "5.0.0" | ||||
|   flutter_local_notifications: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: flutter_local_notifications | ||||
|       sha256: "40e6fbd2da7dcc7ed78432c5cdab1559674b4af035fddbfb2f9a8f9c2112fcef" | ||||
|       sha256: "49eeef364fddb71515bc78d5a8c51435a68bccd6e4d68e25a942c5e47761ae71" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "17.1.2" | ||||
|     version: "17.2.3" | ||||
|   flutter_local_notifications_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_local_notifications_linux | ||||
|       sha256: "33f741ef47b5f63cc7f78fe75eeeac7e19f171ff3c3df054d84c1e38bedb6a03" | ||||
|       sha256: c49bd06165cad9beeb79090b18cd1eb0296f4bf4b23b84426e37dd7c027fc3af | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.0.0+1" | ||||
|     version: "4.0.1" | ||||
|   flutter_local_notifications_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_local_notifications_platform_interface | ||||
|       sha256: "340abf67df238f7f0ef58f4a26d2a83e1ab74c77ab03cd2b2d5018ac64db30b7" | ||||
|       sha256: "85f8d07fe708c1bdcf45037f2c0109753b26ae077e9d9e899d55971711a4ea66" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "7.1.0" | ||||
|     version: "7.2.0" | ||||
|   flutter_localizations: | ||||
|     dependency: transitive | ||||
|     description: flutter | ||||
| @@ -361,23 +433,31 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: flutter_markdown | ||||
|       sha256: "9921f9deda326f8a885e202b1e35237eadfc1345239a0f6f0f1ff287e047547f" | ||||
|       sha256: e17575ca576a34b46c58c91f9948891117a1bd97815d2e661813c7f90c647a78 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.7.1" | ||||
|     version: "0.7.3+2" | ||||
|   flutter_plugin_android_lifecycle: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: flutter_plugin_android_lifecycle | ||||
|       sha256: "8cf40eebf5dec866a6d1956ad7b4f7016e6c0cc69847ab946833b7d43743809f" | ||||
|       sha256: "9ee02950848f61c4129af3d6ec84a1cfc0e47931abc746b03e7a3bc3e8ff6eda" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.19" | ||||
|     version: "2.0.22" | ||||
|   flutter_test: | ||||
|     dependency: "direct dev" | ||||
|     description: flutter | ||||
|     source: sdk | ||||
|     version: "0.0.0" | ||||
|   flutter_typeahead: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: flutter_typeahead | ||||
|       sha256: d64712c65db240b1057559b952398ebb6e498077baeebf9b0731dade62438a6d | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.2.0" | ||||
|   flutter_web_plugins: | ||||
|     dependency: transitive | ||||
|     description: flutter | ||||
| @@ -387,18 +467,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: fluttertoast | ||||
|       sha256: "81b68579e23fcbcada2db3d50302813d2371664afe6165bc78148050ab94bf66" | ||||
|       sha256: "95f349437aeebe524ef7d6c9bde3e6b4772717cf46a0eb6a3ceaddc740b297cc" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "8.2.5" | ||||
|     version: "8.2.8" | ||||
|   fraction: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: fraction | ||||
|       sha256: "09e9504c9177bbd77df56e5d147abfbb3b43360e64bf61510059c14d6a82d524" | ||||
|       sha256: ac0d9904bb8211eb28606bdf623ff9f222c53240d8e9b927a07c149d356eddc2 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.0.2" | ||||
|     version: "5.0.3" | ||||
|   gtk: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -427,10 +507,10 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: http | ||||
|       sha256: "761a297c042deedc1ffbb156d6e2af13886bb305c2a343a4d972504cd67dd938" | ||||
|       sha256: b9c29a161230ee03d3ccf545097fccd9b87a5264228c5d348202e0f0c28f9010 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.2.1" | ||||
|     version: "1.2.2" | ||||
|   http_parser: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -443,18 +523,18 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: image | ||||
|       sha256: "4c68bfd5ae83e700b5204c1e74451e7bf3cf750e6843c6e158289cf56bda018e" | ||||
|       sha256: "2237616a36c0d69aef7549ab439b833fb7f9fb9fc861af2cc9ac3eedddd69ca8" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.1.7" | ||||
|     version: "4.2.0" | ||||
|   intl: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: intl | ||||
|       sha256: "3bc132a9dbce73a7e4a21a17d06e1878839ffbf975568bc875c60537824b0c4d" | ||||
|       sha256: d6f56758b7d3014a48af9701c085700aac781a92a87a62b1333b46d8879661cf | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.18.1" | ||||
|     version: "0.19.0" | ||||
|   json_annotation: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -467,36 +547,36 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: leak_tracker | ||||
|       sha256: "78eb209deea09858f5269f5a5b02be4049535f568c07b275096836f01ea323fa" | ||||
|       sha256: "3f87a60e8c63aecc975dda1ceedbc8f24de75f09e4856ea27daf8958f2f0ce05" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "10.0.0" | ||||
|     version: "10.0.5" | ||||
|   leak_tracker_flutter_testing: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: leak_tracker_flutter_testing | ||||
|       sha256: b46c5e37c19120a8a01918cfaf293547f47269f7cb4b0058f21531c2465d6ef0 | ||||
|       sha256: "932549fb305594d82d7183ecd9fa93463e9914e1b67cacc34bc40906594a1806" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.1" | ||||
|     version: "3.0.5" | ||||
|   leak_tracker_testing: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: leak_tracker_testing | ||||
|       sha256: a597f72a664dbd293f3bfc51f9ba69816f84dcd403cdac7066cb3f6003f3ab47 | ||||
|       sha256: "6ba465d5d76e67ddf503e1161d1f4a6bc42306f9d66ca1e8f079a47290fb06d3" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.0.1" | ||||
|     version: "3.0.1" | ||||
|   lints: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: lints | ||||
|       sha256: "976c774dd944a42e83e2467f4cc670daef7eed6295b10b36ae8c85bcbf828235" | ||||
|       sha256: "3315600f3fb3b135be672bf4a178c55f274bebe368325ae18462c89ac1e3b413" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.0.0" | ||||
|     version: "5.0.0" | ||||
|   markdown: | ||||
|     dependency: transitive | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: markdown | ||||
|       sha256: ef2a1298144e3f985cc736b22e0ccdaf188b5b3970648f2d9dc13efd1d9df051 | ||||
| @@ -515,26 +595,26 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: material_color_utilities | ||||
|       sha256: "0e0a020085b65b6083975e499759762399b4475f766c21668c4ecca34ea74e5a" | ||||
|       sha256: f7142bb1154231d7ea5f96bc7bde4bda2a0945d2806bb11670e30b850d56bdec | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.8.0" | ||||
|     version: "0.11.1" | ||||
|   meta: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: meta | ||||
|       sha256: d584fa6707a52763a52446f02cc621b077888fb63b93bbcb1143a7be5a0c0c04 | ||||
|       sha256: bdb68674043280c3428e9ec998512fb681678676b3c54e773629ffe74419f8c7 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.11.0" | ||||
|     version: "1.15.0" | ||||
|   mime: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: mime | ||||
|       sha256: "2e123074287cc9fd6c09de8336dae606d1ddb88d9ac47358826db698c176a1f2" | ||||
|       sha256: "801fd0b26f14a4a58ccb09d5892c3fbdeff209594300a542492cf13fba9d247a" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.0.5" | ||||
|     version: "1.0.6" | ||||
|   nested: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -563,18 +643,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: path_provider | ||||
|       sha256: c9e7d3a4cd1410877472158bee69963a4579f78b68c65a2b7d40d1a7a88bb161 | ||||
|       sha256: fec0d61223fba3154d87759e3cc27fe2c8dc498f6386c6d6fc80d1afdd1bf378 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.3" | ||||
|     version: "2.1.4" | ||||
|   path_provider_android: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_android | ||||
|       sha256: a248d8146ee5983446bf03ed5ea8f6533129a12b11f12057ad1b4a67a2b3b41d | ||||
|       sha256: "6f01f8e37ec30b07bc424b4deabac37cacb1bc7e2e515ad74486039918a37eb7" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.2.4" | ||||
|     version: "2.2.10" | ||||
|   path_provider_foundation: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -603,10 +683,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: path_provider_windows | ||||
|       sha256: "8bc9f22eee8690981c22aa7fc602f5c85b497a6fb2ceb35ee5a5e5ed85ad8170" | ||||
|       sha256: bd6f00dbd873bfb70d0761682da2b3a2c2fccc2b9e84c495821639601d81afe7 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.2.1" | ||||
|     version: "2.3.0" | ||||
|   permission_handler: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -619,34 +699,34 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: permission_handler_android | ||||
|       sha256: "8bb852cd759488893805c3161d0b2b5db55db52f773dbb014420b304055ba2c5" | ||||
|       sha256: "76e4ab092c1b240d31177bb64d2b0bea43f43d0e23541ec866151b9f7b2490fa" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "12.0.6" | ||||
|     version: "12.0.12" | ||||
|   permission_handler_apple: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: permission_handler_apple | ||||
|       sha256: e9ad66020b89ff1b63908f247c2c6f931c6e62699b756ef8b3c4569350cd8662 | ||||
|       sha256: e6f6d73b12438ef13e648c4ae56bd106ec60d17e90a59c4545db6781229082a0 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "9.4.4" | ||||
|     version: "9.4.5" | ||||
|   permission_handler_html: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: permission_handler_html | ||||
|       sha256: "54bf176b90f6eddd4ece307e2c06cf977fb3973719c35a93b85cc7093eb6070d" | ||||
|       sha256: af26edbbb1f2674af65a8f4b56e1a6f526156bc273d0e65dd8075fab51c78851 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.1.1" | ||||
|     version: "0.1.3+2" | ||||
|   permission_handler_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: permission_handler_platform_interface | ||||
|       sha256: "48d4fcf201a1dad93ee869ab0d4101d084f49136ec82a8a06ed9cfeacab9fd20" | ||||
|       sha256: e9c8eadee926c4532d0305dff94b85bf961f16759c3af791486613152af4b4f9 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.2.1" | ||||
|     version: "4.2.3" | ||||
|   permission_handler_windows: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -667,10 +747,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: platform | ||||
|       sha256: "12220bb4b65720483f8fa9450b4332347737cf8213dd2840d8b2c823e47243ec" | ||||
|       sha256: "9b71283fc13df574056616011fb138fd3b793ea47cc509c189a6c3fa5f8a1a65" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.1.4" | ||||
|     version: "3.1.5" | ||||
|   plugin_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -679,6 +759,38 @@ packages: | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.1.8" | ||||
|   pointer_interceptor: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: pointer_interceptor | ||||
|       sha256: "57210410680379aea8b1b7ed6ae0c3ad349bfd56fe845b8ea934a53344b9d523" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.10.1+2" | ||||
|   pointer_interceptor_ios: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: pointer_interceptor_ios | ||||
|       sha256: a6906772b3205b42c44614fcea28f818b1e5fdad73a4ca742a7bd49818d9c917 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.10.1" | ||||
|   pointer_interceptor_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: pointer_interceptor_platform_interface | ||||
|       sha256: "0597b0560e14354baeb23f8375cd612e8bd4841bf8306ecb71fcd0bb78552506" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.10.0+1" | ||||
|   pointer_interceptor_web: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: pointer_interceptor_web | ||||
|       sha256: "7a7087782110f8c1827170660b09f8aa893e0e9a61431dbbe2ac3fc482e8c044" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.10.2+1" | ||||
|   provider: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -691,74 +803,74 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: share_plus | ||||
|       sha256: ef3489a969683c4f3d0239010cc8b7a2a46543a8d139e111c06c558875083544 | ||||
|       sha256: "468c43f285207c84bcabf5737f33b914ceb8eb38398b91e5e3ad1698d1b72a52" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "9.0.0" | ||||
|     version: "10.0.2" | ||||
|   share_plus_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: share_plus_platform_interface | ||||
|       sha256: "0f9e4418835d1b2c3ae78fdb918251959106cefdbc4dd43526e182f80e82f6d4" | ||||
|       sha256: "6ababf341050edff57da8b6990f11f4e99eaba837865e2e6defe16d039619db5" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.0.0" | ||||
|     version: "5.0.0" | ||||
|   shared_preferences: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: shared_preferences | ||||
|       sha256: d3bbe5553a986e83980916ded2f0b435ef2e1893dfaa29d5a7a790d0eca12180 | ||||
|       sha256: "746e5369a43170c25816cc472ee016d3a66bc13fcf430c0bc41ad7b4b2922051" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.2.3" | ||||
|     version: "2.3.2" | ||||
|   shared_preferences_android: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: shared_preferences_android | ||||
|       sha256: "1ee8bf911094a1b592de7ab29add6f826a7331fb854273d55918693d5364a1f2" | ||||
|       sha256: "480ba4345773f56acda9abf5f50bd966f581dac5d514e5fc4a18c62976bbba7e" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.2.2" | ||||
|     version: "2.3.2" | ||||
|   shared_preferences_foundation: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: shared_preferences_foundation | ||||
|       sha256: "0a8a893bf4fd1152f93fec03a415d11c27c74454d96e2318a7ac38dd18683ab7" | ||||
|       sha256: c4b35f6cb8f63c147312c054ce7c2254c8066745125264f0c88739c417fc9d9f | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.4.0" | ||||
|     version: "2.5.2" | ||||
|   shared_preferences_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: shared_preferences_linux | ||||
|       sha256: "9f2cbcf46d4270ea8be39fa156d86379077c8a5228d9dfdb1164ae0bb93f1faa" | ||||
|       sha256: "580abfd40f415611503cae30adf626e6656dfb2f0cee8f465ece7b6defb40f2f" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.3.2" | ||||
|     version: "2.4.1" | ||||
|   shared_preferences_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: shared_preferences_platform_interface | ||||
|       sha256: "22e2ecac9419b4246d7c22bfbbda589e3acf5c0351137d87dd2939d984d37c3b" | ||||
|       sha256: "57cbf196c486bc2cf1f02b85784932c6094376284b3ad5779d1b1c6c6a816b80" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.3.2" | ||||
|     version: "2.4.1" | ||||
|   shared_preferences_web: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: shared_preferences_web | ||||
|       sha256: "9aee1089b36bd2aafe06582b7d7817fd317ef05fc30e6ba14bff247d0933042a" | ||||
|       sha256: d2ca4132d3946fec2184261726b355836a82c33d7d5b67af32692aff18a4684e | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.3.0" | ||||
|     version: "2.4.2" | ||||
|   shared_preferences_windows: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: shared_preferences_windows | ||||
|       sha256: "841ad54f3c8381c480d0c9b508b89a34036f512482c407e6df7a9c4aa2ef8f59" | ||||
|       sha256: "94ef0f72b2d71bc3e700e025db3710911bd51a71cefb65cc609dd0d9a982e3c1" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.3.2" | ||||
|     version: "2.4.1" | ||||
|   shared_storage: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
| @@ -801,18 +913,18 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: sqflite | ||||
|       sha256: a43e5a27235518c03ca238e7b4732cf35eabe863a369ceba6cbefa537a66f16d | ||||
|       sha256: ff5a2436ef8ebdfda748fbfe957f9981524cb5ff11e7bafa8c42771840e8a788 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.3.3+1" | ||||
|     version: "2.3.3+2" | ||||
|   sqflite_common: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: sqflite_common | ||||
|       sha256: "3da423ce7baf868be70e2c0976c28a1bb2f73644268b7ffa7d2e08eab71f16a4" | ||||
|       sha256: "2d8e607db72e9cb7748c9c6e739e2c9618320a5517de693d5a24609c4671b1a4" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.5.4" | ||||
|     version: "2.5.4+4" | ||||
|   stack_trace: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -841,10 +953,10 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: synchronized | ||||
|       sha256: "539ef412b170d65ecdafd780f924e5be3f60032a1128df156adad6c5b373d558" | ||||
|       sha256: "69fe30f3a8b04a0be0c15ae6490fc859a78ef4c43ae2dd5e8a623d45bfcf9225" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.1.0+1" | ||||
|     version: "3.3.0+3" | ||||
|   term_glyph: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -857,18 +969,18 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: test_api | ||||
|       sha256: "5c2f730018264d276c20e4f1503fd1308dfbbae39ec8ee63c5236311ac06954b" | ||||
|       sha256: "5b8a98dafc4d5c4c9c72d8b31ab2b23fc13422348d2997120294d3bac86b4ddb" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.6.1" | ||||
|     version: "0.7.2" | ||||
|   timezone: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: timezone | ||||
|       sha256: a6ccda4a69a442098b602c44e61a1e2b4bf6f5516e875bbf0f427d5df14745d5 | ||||
|       sha256: "2236ec079a174ce07434e89fcd3fcda430025eb7692244139a9cf54fdcf1fc7d" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.9.3" | ||||
|     version: "0.9.4" | ||||
|   typed_data: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -881,42 +993,42 @@ packages: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: url_launcher | ||||
|       sha256: "6ce1e04375be4eed30548f10a315826fd933c1e493206eab82eed01f438c8d2e" | ||||
|       sha256: "21b704ce5fa560ea9f3b525b43601c678728ba46725bab9b01187b4831377ed3" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.2.6" | ||||
|     version: "6.3.0" | ||||
|   url_launcher_android: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_android | ||||
|       sha256: "360a6ed2027f18b73c8d98e159dda67a61b7f2e0f6ec26e86c3ada33b0621775" | ||||
|       sha256: e35a698ac302dd68e41f73250bd9517fe3ab5fa4f18fe4647a0872db61bacbab | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.3.1" | ||||
|     version: "6.3.10" | ||||
|   url_launcher_ios: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_ios | ||||
|       sha256: "7068716403343f6ba4969b4173cbf3b84fc768042124bc2c011e5d782b24fe89" | ||||
|       sha256: e43b677296fadce447e987a2f519dcf5f6d1e527dc35d01ffab4fff5b8a7063e | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "6.3.0" | ||||
|     version: "6.3.1" | ||||
|   url_launcher_linux: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_linux | ||||
|       sha256: ab360eb661f8879369acac07b6bb3ff09d9471155357da8443fd5d3cf7363811 | ||||
|       sha256: e2b9622b4007f97f504cd64c0128309dfb978ae66adbe944125ed9e1750f06af | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.1.1" | ||||
|     version: "3.2.0" | ||||
|   url_launcher_macos: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_macos | ||||
|       sha256: "9a1a42d5d2d95400c795b2914c36fdcb525870c752569438e4ebb09a2b5d90de" | ||||
|       sha256: "769549c999acdb42b8bcfa7c43d72bf79a382ca7441ab18a808e101149daf672" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.2.0" | ||||
|     version: "3.2.1" | ||||
|   url_launcher_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -929,26 +1041,26 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_web | ||||
|       sha256: "8d9e750d8c9338601e709cd0885f95825086bd8b642547f26bda435aade95d8a" | ||||
|       sha256: "772638d3b34c779ede05ba3d38af34657a05ac55b06279ea6edd409e323dca8e" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "2.3.1" | ||||
|     version: "2.3.3" | ||||
|   url_launcher_windows: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: url_launcher_windows | ||||
|       sha256: ecf9725510600aa2bb6d7ddabe16357691b6d2805f66216a97d1b881e21beff7 | ||||
|       sha256: "49c10f879746271804767cb45551ec5592cdab00ee105c06dddde1a98f73b185" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.1.1" | ||||
|     version: "3.1.2" | ||||
|   uuid: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: uuid | ||||
|       sha256: "814e9e88f21a176ae1359149021870e87f7cddaf633ab678a5d2b0bff7fd1ba8" | ||||
|       sha256: a5be9ef6618a7ac1e964353ef476418026db906c4facdedaa299b7a2e71690ff | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.4.0" | ||||
|     version: "4.5.1" | ||||
|   vector_math: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -961,34 +1073,34 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: vm_service | ||||
|       sha256: b3d56ff4341b8f182b96aceb2fa20e3dcb336b9f867bc0eafc0de10f1048e957 | ||||
|       sha256: "5c5f338a667b4c644744b661f309fb8080bb94b18a7e91ef1dbd343bed00ed6d" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "13.0.0" | ||||
|     version: "14.2.5" | ||||
|   web: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: web | ||||
|       sha256: "97da13628db363c635202ad97068d47c5b8aa555808e7a9411963c533b449b27" | ||||
|       sha256: cd3543bd5798f6ad290ea73d210f423502e71900302dde696f8bff84bf89a1cb | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "0.5.1" | ||||
|     version: "1.1.0" | ||||
|   webview_flutter: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: webview_flutter | ||||
|       sha256: "25e1b6e839e8cbfbd708abc6f85ed09d1727e24e08e08c6b8590d7c65c9a8932" | ||||
|       sha256: ec81f57aa1611f8ebecf1d2259da4ef052281cb5ad624131c93546c79ccc7736 | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "4.7.0" | ||||
|     version: "4.9.0" | ||||
|   webview_flutter_android: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: webview_flutter_android | ||||
|       sha256: dad3313c9ead95517bb1cae5e1c9d20ba83729d5a59e5e83c0a2d66203f27f91 | ||||
|       sha256: "6e64fcb1c19d92024da8f33503aaeeda35825d77142c01d0ea2aa32edc79fdc8" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.16.1" | ||||
|     version: "3.16.7" | ||||
|   webview_flutter_platform_interface: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -1001,26 +1113,26 @@ packages: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: webview_flutter_wkwebview | ||||
|       sha256: f12f8d8a99784b863e8b85e4a9a5e3cf1839d6803d2c0c3e0533a8f3c5a992a7 | ||||
|       sha256: "1942a12224ab31e9508cf00c0c6347b931b023b8a4f0811e5dec3b06f94f117d" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "3.13.0" | ||||
|     version: "3.15.0" | ||||
|   win32: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: win32 | ||||
|       sha256: "0eaf06e3446824099858367950a813472af675116bf63f008a4c2a75ae13e9cb" | ||||
|       sha256: "68d1e89a91ed61ad9c370f9f8b6effed9ae5e0ede22a270bdfa6daf79fc2290a" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "5.5.0" | ||||
|     version: "5.5.4" | ||||
|   win32_registry: | ||||
|     dependency: transitive | ||||
|     description: | ||||
|       name: win32_registry | ||||
|       sha256: "10589e0d7f4e053f2c61023a31c9ce01146656a70b7b7f0828c0b46d7da2a9bb" | ||||
|       sha256: "21ec76dfc731550fd3e2ce7a33a9ea90b828fdf19a5c3bcf556fa992cfa99852" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.1.3" | ||||
|     version: "1.1.5" | ||||
|   xdg_directories: | ||||
|     dependency: transitive | ||||
|     description: | ||||
| @@ -1046,5 +1158,5 @@ packages: | ||||
|     source: hosted | ||||
|     version: "3.1.2" | ||||
| sdks: | ||||
|   dart: ">=3.3.3 <4.0.0" | ||||
|   flutter: ">=3.19.0" | ||||
|   dart: ">=3.5.0 <4.0.0" | ||||
|   flutter: ">=3.24.0" | ||||
|   | ||||
							
								
								
									
										18
									
								
								pubspec.yaml
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								pubspec.yaml
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | ||||
| name: obtainium | ||||
| description: Get Android App Updates Directly From the Source. | ||||
| description: Get Android app updates straight from the source. | ||||
|  | ||||
| # The following line prevents the package from being accidentally published to | ||||
| # pub.dev using `flutter pub publish`. This is preferred for private packages. | ||||
| @@ -17,7 +17,7 @@ publish_to: 'none' # Remove this line if you wish to publish to pub.dev | ||||
| # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html | ||||
| # In Windows, build-name is used as the major, minor, and patch parts | ||||
| # of the product and file versions while build-number is used as the build suffix. | ||||
| version: 1.1.8+2265 | ||||
| version: 1.1.22+2279 | ||||
|  | ||||
| environment: | ||||
|   sdk: '>=3.0.0 <4.0.0' | ||||
| @@ -37,7 +37,7 @@ dependencies: | ||||
|   # Use with the CupertinoIcons class for iOS style icons. | ||||
|   cupertino_icons: ^1.0.5 | ||||
|   path_provider: ^2.0.11 | ||||
|   flutter_fgbg: ^0.3.0 # Try removing reliance on this | ||||
|   flutter_fgbg: ^0.6.0 | ||||
|   flutter_local_notifications: ^17.0.0 | ||||
|   provider: ^6.0.3 | ||||
|   http: ^1.0.0 | ||||
| @@ -56,7 +56,7 @@ dependencies: | ||||
|       url: https://github.com/ImranR98/android_package_installer | ||||
|       ref: main | ||||
|   android_package_manager: ^0.7.0 | ||||
|   share_plus: ^9.0.0 | ||||
|   share_plus: ^10.0.0 | ||||
|   sqflite: ^2.2.0+3 | ||||
|   easy_localization: ^3.0.1 | ||||
|   android_intent_plus: ^5.0.1 | ||||
| @@ -79,17 +79,19 @@ dependencies: | ||||
|       url: https://github.com/re7gog/shizuku_apk_installer | ||||
|       ref: master | ||||
|  | ||||
|   markdown: any | ||||
|   flutter_typeahead: ^5.2.0 | ||||
| dev_dependencies: | ||||
|   flutter_test: | ||||
|     sdk: flutter | ||||
|   flutter_launcher_icons: ^0.13.1 | ||||
|   flutter_launcher_icons: ^0.14.1 | ||||
|  | ||||
|   # The "flutter_lints" package below contains a set of recommended lints to | ||||
|   # encourage good coding practices. The lint set provided by the package is | ||||
|   # activated in the `analysis_options.yaml` file located at the root of your | ||||
|   # package. See that file for information about deactivating specific lint | ||||
|   # rules and activating additional ones. | ||||
|   flutter_lints: ^4.0.0 | ||||
|   flutter_lints: ^5.0.0 | ||||
|  | ||||
| flutter_launcher_icons: | ||||
|   android: "ic_launcher" | ||||
| @@ -143,6 +145,6 @@ flutter: | ||||
|   # see https://flutter.dev/custom-fonts/#from-packages | ||||
|  | ||||
|   fonts: | ||||
|       - family: Metropolis | ||||
|       - family: Wix-Madefor-Display | ||||
|         fonts: | ||||
|           - asset: assets/fonts/Metropolis-Regular.otf | ||||
|           - asset: assets/fonts/WixMadeforDisplay-Regular.otf | ||||
		Reference in New Issue
	
	Block a user