mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-10-31 13:33:28 +01:00 
			
		
		
		
	Compare commits
	
		
			263 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 | ||
|  | dbcb4b3c09 | ||
|  | b231c756e6 | ||
|  | 3cb3f7fdd4 | ||
|  | 9837e8e325 | ||
|  | 73d4814f18 | ||
|  | 0a9219c314 | ||
|  | 56c5a73d9a | ||
|  | a30e063246 | ||
|  | bd26b6514a | ||
|  | 3ea8c7e888 | ||
|  | 5f2ec5ce6f | ||
|  | 783ce9d555 | ||
|  | a719b2475b | ||
|  | 84df499ea6 | ||
|  | 0d25b74050 | ||
|  | 0effbc3841 | ||
|  | 7478a7af22 | ||
|  | 0838a6d30b | ||
|  | 8cee268d13 | ||
|  | a3fddc5400 | ||
|  | c0a2e372e5 | ||
|  | c633963203 | ||
|  | 299f457938 | ||
|  | 37e62c922b | ||
|  | 95722ce47b | ||
|  | 8b806b3ef1 | ||
|  | 09221b3526 | ||
|  | 31b6250082 | ||
|  | 07372da91b | ||
|  | 5c36bcfb4b | ||
|  | e5012b1fcb | ||
|  | 6f951175a4 | ||
|  | be52ec372f | ||
|  | eb7126afc3 | ||
|  | 26fc63a02a | ||
|  | d33ca0948f | ||
|  | f76637a2e1 | ||
|  | b7de627c7b | ||
|  | 27fc60d437 | ||
|  | ec240f946e | ||
|  | ecd80fc371 | ||
|  | 68fa660e6d | ||
|  | 70f9e33d17 | ||
|  | 2e2dffd8e2 | ||
|  | 0b1d5bf514 | ||
|  | 2570c8e289 | ||
|  | 43a8ba4de1 | ||
|  | 3f2fb1c1ed | ||
|  | 40d303fb57 | ||
|  | 2c1687c33d | ||
|  | b688e7f160 | ||
|  | 0c80d88583 | ||
|  | b293b1e9ff | ||
|  | 61038a8969 | ||
|  | b92d1541bf | ||
|  | 13f0ccb10d | ||
|  | ed732cfad2 | ||
|  | fb206aee84 | ||
|  | 285530784d | ||
|  | 0ddb5b5e81 | ||
|  | c16cda1962 | ||
|  | 49022726d3 | ||
|  | a311894b9f | ||
|  | 29e13efd66 | ||
|  | 8c2a97e092 | ||
|  | 34d571f586 | ||
|  | a20b87889b | ||
|  | c7319cd958 | ||
|  | d61fd800ef | ||
|  | 12dda8bfa9 | ||
|  | 0657f832e1 | ||
|  | 6431357b15 | ||
|  | dcc42bdfe5 | ||
|  | 85718dc3a3 | ||
|  | 9dff352796 | ||
|  | c5477767a0 | ||
|  | 3131ef8c4e | ||
|  | 4d9f05aa87 | ||
|  | 7b882d9bd8 | ||
|  | d1a2831922 | ||
|  | b042050ea3 | ||
|  | 2d43dfe0a7 | ||
|  | b1a740223c | ||
|  | 41c98d97b1 | ||
|  | fb06babb96 | ||
|  | ccc0e7696b | ||
|  | 6d41ed8011 | ||
|  | c1476a7d58 | ||
|  | 2289e58dda | ||
|  | 049bcfbaf5 | ||
|  | 5b5f922b54 | ||
|  | 6545498c21 | ||
|  | 9717db0ca4 | 
							
								
								
									
										2
									
								
								.flutter
									
									
									
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										2
									
								
								.flutter
									
									
									
									
									
								
							 Submodule .flutter updated: 300451adae...2663184aa7
									
								
							
							
								
								
									
										14
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,10 @@ on: | |||||||
|       beta: |       beta: | ||||||
|         type: boolean |         type: boolean | ||||||
|         description: Is beta? |         description: Is beta? | ||||||
|      |       draft: | ||||||
|  |         type: boolean | ||||||
|  |         description: Is draft? | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build: |   build: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
| @@ -35,13 +38,7 @@ jobs: | |||||||
|       - name: Check submodule |       - name: Check submodule | ||||||
|         id: check_submodule |         id: check_submodule | ||||||
|         run: | |         run: | | ||||||
|           SUBMODULE_COMMIT_LONG="$(git submodule status | head -1 | tail -c +2 | awk '{print $1}')" |           git checkout ${{ inputs.checkout }} | ||||||
|           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 |  | ||||||
|  |  | ||||||
|       - name: Extract Version |       - name: Extract Version | ||||||
|         id: extract_version       |         id: extract_version       | ||||||
| @@ -92,5 +89,6 @@ jobs: | |||||||
|           token: ${{ secrets.GH_ACCESS_TOKEN }} |           token: ${{ secrets.GH_ACCESS_TOKEN }} | ||||||
|           tag: "${{ steps.extract_version.outputs.tag }}" |           tag: "${{ steps.extract_version.outputs.tag }}" | ||||||
|           prerelease: "${{ steps.extract_version.outputs.beta }}" |           prerelease: "${{ steps.extract_version.outputs.beta }}" | ||||||
|  |           draft: "${{ inputs.draft }}" | ||||||
|           artifacts: ./build/app/outputs/flutter-apk/*-release*.apk* |           artifacts: ./build/app/outputs/flutter-apk/*-release*.apk* | ||||||
|           generateReleaseNotes: true |           generateReleaseNotes: true | ||||||
|   | |||||||
							
								
								
									
										27
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								README.md
									
									
									
									
									
								
							| @@ -2,19 +2,21 @@ | |||||||
|  |  | ||||||
| [](https://techforpalestine.org/learn-more) | [](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. | Obtainium allows you to install and update apps directly from their releases pages, and receive notifications when new releases are made available. | ||||||
|  |  | ||||||
| Motivation: [Side Of Burritos - You should use this instead of F-Droid | How to use app RSS feed](https://youtu.be/FFz57zNR_M0) | More info: | ||||||
|  | - [Obtainium/wiki](https://github.com/ImranR98/Obtainium/wiki) | ||||||
| Read the Wiki: [https://github.com/ImranR98/Obtainium/wiki](https://github.com/ImranR98/Obtainium/wiki) | - [AppVerifier](https://github.com/soupslurpr/AppVerifier) - App verification tool (recommended, integrates with Obtainium) | ||||||
|  | - [apps.obtainium.imranr.dev](https://apps.obtainium.imranr.dev/) - Crowdsourced app configurations | ||||||
|  | - [Side Of Burritos - You should use this instead of F-Droid | How to use app RSS feed](https://youtu.be/FFz57zNR_M0) - Original motivation for this app | ||||||
|  |  | ||||||
| Currently supported App sources: | Currently supported App sources: | ||||||
| - Open Source - General: | - Open Source - General: | ||||||
|   - [GitHub](https://github.com/) |   - [GitHub](https://github.com/) | ||||||
|   - [GitLab](https://gitlab.com/) |   - [GitLab](https://gitlab.com/) | ||||||
|   - [Codeberg](https://codeberg.org/) |   - [Forgejo](https://forgejo.org/) ([Codeberg](https://codeberg.org/)) | ||||||
|   - [F-Droid](https://f-droid.org/) |   - [F-Droid](https://f-droid.org/) | ||||||
|   - Third Party F-Droid Repos |   - Third Party F-Droid Repos | ||||||
|   - [IzzyOnDroid](https://android.izzysoft.de/) |   - [IzzyOnDroid](https://android.izzysoft.de/) | ||||||
| @@ -23,19 +25,28 @@ Currently supported App sources: | |||||||
|   - [APKPure](https://apkpure.net/) |   - [APKPure](https://apkpure.net/) | ||||||
|   - [Aptoide](https://aptoide.com/) |   - [Aptoide](https://aptoide.com/) | ||||||
|   - [Uptodown](https://uptodown.com/) |   - [Uptodown](https://uptodown.com/) | ||||||
|   - [APKMirror](https://apkmirror.com/) (Track-Only) |  | ||||||
|   - [Huawei AppGallery](https://appgallery.huawei.com/) |   - [Huawei AppGallery](https://appgallery.huawei.com/) | ||||||
|  |   - [Tencent App Store](https://sj.qq.com/) | ||||||
|   - Jenkins Jobs |   - Jenkins Jobs | ||||||
|  |   - [APKMirror](https://apkmirror.com/) (Track-Only) | ||||||
| - Open Source - App-Specific: | - Open Source - App-Specific: | ||||||
|   - [Mullvad](https://mullvad.net/en/) |  | ||||||
|   - [Signal](https://signal.org/) |   - [Signal](https://signal.org/) | ||||||
|   - [VLC](https://videolan.org/) |   - [VLC](https://videolan.org/) | ||||||
| - Other - App-Specific: | - Other - App-Specific: | ||||||
|   - [WhatsApp](https://whatsapp.com) |   - [WhatsApp](https://whatsapp.com) | ||||||
|   - [Telegram App](https://telegram.org) |   - [Telegram App](https://telegram.org) | ||||||
|   - [Neutron Code](https://neutroncode.com) |   - [Neutron Code](https://neutroncode.com) | ||||||
|  | - Direct APK Link | ||||||
| - "HTML" (Fallback): Any other URL that returns an HTML page with links to APK files | - "HTML" (Fallback): Any other URL that returns an HTML page with links to APK files | ||||||
|  |  | ||||||
|  | ## Finding App Configurations | ||||||
|  |  | ||||||
|  | You can find crowdsourced app configurations at [apps.obtainium.imranr.dev](https://apps.obtainium.imranr.dev). | ||||||
|  |  | ||||||
|  | If you can't find the configuration for an app you want, feel free to leave a request on the [discussions page](https://github.com/ImranR98/apps.obtainium.imranr.dev/discussions/new?category=app-requests). | ||||||
|  |  | ||||||
|  | Or, contribute some configurations to the website by creating a PR at [this repo](https://github.com/ImranR98/apps.obtainium.imranr.dev). | ||||||
|  |  | ||||||
| ## Installation | ## Installation | ||||||
|  |  | ||||||
| [<img src="https://github.com/machiav3lli/oandbackupx/blob/034b226cea5c1b30eb4f6a6f313e4dadcbb0ece4/badge_github.png" | [<img src="https://github.com/machiav3lli/oandbackupx/blob/034b226cea5c1b30eb4f6a6f313e4dadcbb0ece4/badge_github.png" | ||||||
|   | |||||||
| @@ -92,20 +92,6 @@ repositories { | |||||||
|     maven { url 'https://jitpack.io' } |     maven { url 'https://jitpack.io' } | ||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { |  | ||||||
|     def shizuku_version = '13.1.5' |  | ||||||
|     implementation "dev.rikka.shizuku:api:$shizuku_version" |  | ||||||
|     implementation "dev.rikka.shizuku:provider:$shizuku_version" |  | ||||||
|  |  | ||||||
|     def hidden_api_version = '4.3.1' |  | ||||||
|     implementation "dev.rikka.tools.refine:runtime:$hidden_api_version" |  | ||||||
|     implementation "dev.rikka.hidden:compat:$hidden_api_version" |  | ||||||
|     compileOnly "dev.rikka.hidden:stub:$hidden_api_version" |  | ||||||
|     implementation "org.lsposed.hiddenapibypass:hiddenapibypass:4.3" |  | ||||||
|  |  | ||||||
|     implementation "com.github.topjohnwu.libsu:core:5.2.2" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| ext.abiCodes = ["x86_64": 1, "armeabi-v7a": 2, "arm64-v8a": 3] | ext.abiCodes = ["x86_64": 1, "armeabi-v7a": 2, "arm64-v8a": 3] | ||||||
| import com.android.build.OutputFile | import com.android.build.OutputFile | ||||||
| android.applicationVariants.all { variant -> | android.applicationVariants.all { variant -> | ||||||
|   | |||||||
| @@ -6,7 +6,8 @@ | |||||||
|         android:name="${applicationName}" |         android:name="${applicationName}" | ||||||
|         android:icon="@mipmap/ic_launcher" |         android:icon="@mipmap/ic_launcher" | ||||||
|         android:requestLegacyExternalStorage="true" |         android:requestLegacyExternalStorage="true" | ||||||
|         android:usesCleartextTraffic="true"> |         android:usesCleartextTraffic="true" | ||||||
|  |         android:localeConfig="@xml/locales_config"> | ||||||
|         <activity |         <activity | ||||||
|             android:name=".MainActivity" |             android:name=".MainActivity" | ||||||
|             android:exported="true" |             android:exported="true" | ||||||
| @@ -47,7 +48,7 @@ | |||||||
|             android:value="2" /> |             android:value="2" /> | ||||||
|         <provider |         <provider | ||||||
|             android:name="androidx.core.content.FileProvider" |             android:name="androidx.core.content.FileProvider" | ||||||
|             android:authorities="dev.imranr.obtainium" |             android:authorities="${applicationId}" | ||||||
|             android:grantUriPermissions="true"> |             android:grantUriPermissions="true"> | ||||||
|             <meta-data |             <meta-data | ||||||
|                 android:name="android.support.FILE_PROVIDER_PATHS" |                 android:name="android.support.FILE_PROVIDER_PATHS" | ||||||
| @@ -72,4 +73,4 @@ | |||||||
|         android:name="android.permission.WRITE_EXTERNAL_STORAGE" |         android:name="android.permission.WRITE_EXTERNAL_STORAGE" | ||||||
|         android:maxSdkVersion="29" />\ |         android:maxSdkVersion="29" />\ | ||||||
|     <uses-permission android:name="android.permission.QUERY_ALL_PACKAGES" /> |     <uses-permission android:name="android.permission.QUERY_ALL_PACKAGES" /> | ||||||
| </manifest> | </manifest> | ||||||
|   | |||||||
| @@ -1,44 +0,0 @@ | |||||||
| package dev.imranr.obtainium |  | ||||||
|  |  | ||||||
| import android.util.Xml |  | ||||||
| import org.xmlpull.v1.XmlPullParser |  | ||||||
| import java.io.File |  | ||||||
| import java.io.FileInputStream |  | ||||||
|  |  | ||||||
| class DefaultSystemFont { |  | ||||||
|     fun get(): String { |  | ||||||
|         return try { |  | ||||||
|             val file = File("/system/etc/fonts.xml") |  | ||||||
|             val fileStream = FileInputStream(file) |  | ||||||
|             parseFontsFileStream(fileStream) |  | ||||||
|         } catch (e: Exception) { |  | ||||||
|             e.message ?: "Unknown fonts.xml parsing exception" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private fun parseFontsFileStream(fileStream: FileInputStream): String { |  | ||||||
|         fileStream.use { stream -> |  | ||||||
|             val parser = Xml.newPullParser() |  | ||||||
|             parser.setFeature(XmlPullParser.FEATURE_PROCESS_NAMESPACES, false) |  | ||||||
|             parser.setInput(stream, null) |  | ||||||
|             parser.nextTag() |  | ||||||
|             return parseFonts(parser) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private fun parseFonts(parser: XmlPullParser): String { |  | ||||||
|         while (!((parser.next() == XmlPullParser.END_TAG) && (parser.name == "family"))) { |  | ||||||
|             if ((parser.eventType == XmlPullParser.START_TAG) && (parser.name == "font") |  | ||||||
|                 && (parser.getAttributeValue(null, "style") == "normal") |  | ||||||
|                 && (parser.getAttributeValue(null, "weight") == "400")) { |  | ||||||
|                 break |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         parser.next() |  | ||||||
|         val fontFile = parser.text.trim() |  | ||||||
|         if (fontFile == "") { |  | ||||||
|             throw NoSuchFieldException("The font filename couldn't be found in fonts.xml") |  | ||||||
|         } |  | ||||||
|         return "/system/fonts/$fontFile" |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,179 +1,5 @@ | |||||||
| package dev.imranr.obtainium | package dev.imranr.obtainium | ||||||
|  |  | ||||||
| import android.content.Intent |  | ||||||
| import android.content.IntentSender |  | ||||||
| import android.content.pm.IPackageInstaller |  | ||||||
| import android.content.pm.IPackageInstallerSession |  | ||||||
| import android.content.pm.PackageInstaller |  | ||||||
| import android.content.pm.PackageManager |  | ||||||
| import android.net.Uri |  | ||||||
| import android.os.Build |  | ||||||
| import android.os.Bundle |  | ||||||
| import android.os.Process |  | ||||||
| import androidx.annotation.NonNull |  | ||||||
| import com.topjohnwu.superuser.Shell |  | ||||||
| import dev.imranr.obtainium.util.IIntentSenderAdaptor |  | ||||||
| import dev.imranr.obtainium.util.IntentSenderUtils |  | ||||||
| import dev.imranr.obtainium.util.PackageInstallerUtils |  | ||||||
| import dev.imranr.obtainium.util.ShizukuSystemServerApi |  | ||||||
| import io.flutter.embedding.android.FlutterActivity | import io.flutter.embedding.android.FlutterActivity | ||||||
| import io.flutter.embedding.engine.FlutterEngine |  | ||||||
| import io.flutter.plugin.common.MethodChannel |  | ||||||
| import io.flutter.plugin.common.MethodChannel.Result |  | ||||||
| import java.io.IOException |  | ||||||
| import java.util.concurrent.CountDownLatch |  | ||||||
| import org.lsposed.hiddenapibypass.HiddenApiBypass |  | ||||||
| import rikka.shizuku.Shizuku |  | ||||||
| import rikka.shizuku.Shizuku.OnRequestPermissionResultListener |  | ||||||
| import rikka.shizuku.ShizukuBinderWrapper |  | ||||||
|  |  | ||||||
| class MainActivity: FlutterActivity() { | class MainActivity: FlutterActivity() | ||||||
|     private var nativeChannel: MethodChannel? = null |  | ||||||
|     private val SHIZUKU_PERMISSION_REQUEST_CODE = (10..200).random() |  | ||||||
|  |  | ||||||
|     private fun shizukuCheckPermission(result: Result) { |  | ||||||
|         try { |  | ||||||
|             if (Shizuku.isPreV11()) {  // Unsupported |  | ||||||
|                 result.success(-1) |  | ||||||
|             } else if (Shizuku.checkSelfPermission() == PackageManager.PERMISSION_GRANTED) { |  | ||||||
|                 result.success(1) |  | ||||||
|             } else if (Shizuku.shouldShowRequestPermissionRationale()) {  // Deny and don't ask again |  | ||||||
|                 result.success(0) |  | ||||||
|             } else { |  | ||||||
|                 Shizuku.requestPermission(SHIZUKU_PERMISSION_REQUEST_CODE) |  | ||||||
|                 result.success(-2) |  | ||||||
|             } |  | ||||||
|         } catch (_: Exception) {  // If shizuku not running |  | ||||||
|             result.success(-1) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private val shizukuRequestPermissionResultListener = OnRequestPermissionResultListener { |  | ||||||
|             requestCode: Int, grantResult: Int -> |  | ||||||
|         if (requestCode == SHIZUKU_PERMISSION_REQUEST_CODE) { |  | ||||||
|             val res = if (grantResult == PackageManager.PERMISSION_GRANTED) 1 else 0 |  | ||||||
|             nativeChannel!!.invokeMethod("resPermShizuku", mapOf("res" to res)) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private fun shizukuInstallApk(apkFileUri: String, result: Result) { |  | ||||||
|         val uri = Uri.parse(apkFileUri) |  | ||||||
|         var res = false |  | ||||||
|         var session: PackageInstaller.Session? = null |  | ||||||
|         try { |  | ||||||
|             val iPackageInstaller: IPackageInstaller = |  | ||||||
|                 ShizukuSystemServerApi.PackageManager_getPackageInstaller() |  | ||||||
|             val isRoot = Shizuku.getUid() == 0 |  | ||||||
|             // The reason for use "com.android.shell" as installer package under adb |  | ||||||
|             // is that getMySessions will check installer package's owner |  | ||||||
|             val installerPackageName = if (isRoot) packageName else "com.android.shell" |  | ||||||
|             var installerAttributionTag: String? = null |  | ||||||
|             if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) { |  | ||||||
|                 installerAttributionTag = attributionTag |  | ||||||
|             } |  | ||||||
|             val userId = if (isRoot) Process.myUserHandle().hashCode() else 0 |  | ||||||
|             val packageInstaller = PackageInstallerUtils.createPackageInstaller( |  | ||||||
|                 iPackageInstaller, installerPackageName, installerAttributionTag, userId) |  | ||||||
|             val params = |  | ||||||
|                 PackageInstaller.SessionParams(PackageInstaller.SessionParams.MODE_FULL_INSTALL) |  | ||||||
|             var installFlags: Int = PackageInstallerUtils.getInstallFlags(params) |  | ||||||
|             installFlags = installFlags or (0x00000002/*PackageManager.INSTALL_REPLACE_EXISTING*/ |  | ||||||
|                     or 0x00000004 /*PackageManager.INSTALL_ALLOW_TEST*/) |  | ||||||
|             PackageInstallerUtils.setInstallFlags(params, installFlags) |  | ||||||
|             val sessionId = packageInstaller.createSession(params) |  | ||||||
|             val iSession = IPackageInstallerSession.Stub.asInterface( |  | ||||||
|                 ShizukuBinderWrapper(iPackageInstaller.openSession(sessionId).asBinder())) |  | ||||||
|             session = PackageInstallerUtils.createSession(iSession) |  | ||||||
|             val inputStream = contentResolver.openInputStream(uri) |  | ||||||
|             val openedSession = session.openWrite("apk.apk", 0, -1) |  | ||||||
|             val buffer = ByteArray(8192) |  | ||||||
|             var length: Int |  | ||||||
|             try { |  | ||||||
|                 while (inputStream!!.read(buffer).also { length = it } > 0) { |  | ||||||
|                     openedSession.write(buffer, 0, length) |  | ||||||
|                     openedSession.flush() |  | ||||||
|                     session.fsync(openedSession) |  | ||||||
|                 } |  | ||||||
|             } finally { |  | ||||||
|                 try { |  | ||||||
|                     inputStream!!.close() |  | ||||||
|                     openedSession.close() |  | ||||||
|                 } catch (e: IOException) { |  | ||||||
|                     e.printStackTrace() |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             val results = arrayOf<Intent?>(null) |  | ||||||
|             val countDownLatch = CountDownLatch(1) |  | ||||||
|             val intentSender: IntentSender = |  | ||||||
|                 IntentSenderUtils.newInstance(object : IIntentSenderAdaptor() { |  | ||||||
|                     override fun send(intent: Intent?) { |  | ||||||
|                         results[0] = intent |  | ||||||
|                         countDownLatch.countDown() |  | ||||||
|                     } |  | ||||||
|                 }) |  | ||||||
|             session.commit(intentSender) |  | ||||||
|             countDownLatch.await() |  | ||||||
|             res = results[0]!!.getIntExtra( |  | ||||||
|                 PackageInstaller.EXTRA_STATUS, PackageInstaller.STATUS_FAILURE) == 0 |  | ||||||
|         } catch (_: Exception) { |  | ||||||
|             res = false |  | ||||||
|         } finally { |  | ||||||
|             if (session != null) { |  | ||||||
|                 try { |  | ||||||
|                     session.close() |  | ||||||
|                 } catch (_: Exception) { |  | ||||||
|                     res = false |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         result.success(res) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private fun rootCheckPermission(result: Result) { |  | ||||||
|         Shell.getShell(Shell.GetShellCallback( |  | ||||||
|             fun(shell: Shell) { |  | ||||||
|                 result.success(shell.isRoot) |  | ||||||
|             } |  | ||||||
|         )) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private fun rootInstallApk(apkFilePath: String, result: Result) { |  | ||||||
|         Shell.sh("pm install -r -t " + apkFilePath).submit { out -> |  | ||||||
|             val builder = StringBuilder() |  | ||||||
|             for (data in out.getOut()) { builder.append(data) } |  | ||||||
|             result.success(builder.toString().endsWith("Success")) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun configureFlutterEngine(@NonNull flutterEngine: FlutterEngine) { |  | ||||||
|         super.configureFlutterEngine(flutterEngine) |  | ||||||
|         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) { |  | ||||||
|             HiddenApiBypass.addHiddenApiExemptions("") |  | ||||||
|         } |  | ||||||
|         Shizuku.addRequestPermissionResultListener(shizukuRequestPermissionResultListener) |  | ||||||
|         nativeChannel = MethodChannel( |  | ||||||
|             flutterEngine.dartExecutor.binaryMessenger, "native") |  | ||||||
|         nativeChannel!!.setMethodCallHandler { |  | ||||||
|             call, result -> |  | ||||||
|             if (call.method == "getSystemFont") { |  | ||||||
|                 val res = DefaultSystemFont().get() |  | ||||||
|                 result.success(res) |  | ||||||
|             } else if (call.method == "checkPermissionShizuku") { |  | ||||||
|                 shizukuCheckPermission(result) |  | ||||||
|             } else if (call.method == "checkPermissionRoot") { |  | ||||||
|                 rootCheckPermission(result) |  | ||||||
|             } else if (call.method == "installWithShizuku") { |  | ||||||
|                 val apkFileUri: String? = call.argument("apkFileUri") |  | ||||||
|                 shizukuInstallApk(apkFileUri!!, result) |  | ||||||
|             } else if (call.method == "installWithRoot") { |  | ||||||
|                 val apkFilePath: String? = call.argument("apkFilePath") |  | ||||||
|                 rootInstallApk(apkFilePath!!, result) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun onDestroy() { |  | ||||||
|         super.onDestroy() |  | ||||||
|         Shizuku.removeRequestPermissionResultListener(shizukuRequestPermissionResultListener) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -1,37 +0,0 @@ | |||||||
| package dev.imranr.obtainium.util; |  | ||||||
|  |  | ||||||
| import android.annotation.SuppressLint; |  | ||||||
| import android.app.Application; |  | ||||||
| import android.os.Build; |  | ||||||
|  |  | ||||||
| import java.lang.reflect.InvocationTargetException; |  | ||||||
| import java.lang.reflect.Method; |  | ||||||
|  |  | ||||||
| public class ApplicationUtils { |  | ||||||
|  |  | ||||||
|     private static Application application; |  | ||||||
|  |  | ||||||
|     public static Application getApplication() { |  | ||||||
|         return application; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static void setApplication(Application application) { |  | ||||||
|         ApplicationUtils.application = application; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static String getProcessName() { |  | ||||||
|         if (Build.VERSION.SDK_INT >= 28) |  | ||||||
|             return Application.getProcessName(); |  | ||||||
|         else { |  | ||||||
|             try { |  | ||||||
|                 @SuppressLint("PrivateApi") |  | ||||||
|                 Class<?> activityThread = Class.forName("android.app.ActivityThread"); |  | ||||||
|                 @SuppressLint("DiscouragedPrivateApi") |  | ||||||
|                 Method method = activityThread.getDeclaredMethod("currentProcessName"); |  | ||||||
|                 return (String) method.invoke(null); |  | ||||||
|             } catch (ClassNotFoundException | NoSuchMethodException | IllegalAccessException | InvocationTargetException e) { |  | ||||||
|                 throw new RuntimeException(e); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,23 +0,0 @@ | |||||||
| package dev.imranr.obtainium.util; |  | ||||||
|  |  | ||||||
| import android.content.IIntentReceiver; |  | ||||||
| import android.content.IIntentSender; |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.os.Bundle; |  | ||||||
| import android.os.IBinder; |  | ||||||
|  |  | ||||||
| public abstract class IIntentSenderAdaptor extends IIntentSender.Stub { |  | ||||||
|  |  | ||||||
|     public abstract void send(Intent intent); |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public int send(int code, Intent intent, String resolvedType, IIntentReceiver finishedReceiver, String requiredPermission, Bundle options) { |  | ||||||
|         send(intent); |  | ||||||
|         return 0; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void send(int code, Intent intent, String resolvedType, IBinder whitelistToken, IIntentReceiver finishedReceiver, String requiredPermission, Bundle options) { |  | ||||||
|         send(intent); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| package dev.imranr.obtainium.util; |  | ||||||
|  |  | ||||||
| import android.content.IIntentSender; |  | ||||||
| import android.content.IntentSender; |  | ||||||
|  |  | ||||||
| import java.lang.reflect.InvocationTargetException; |  | ||||||
|  |  | ||||||
| public class IntentSenderUtils { |  | ||||||
|  |  | ||||||
|     public static IntentSender newInstance(IIntentSender binder) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException { |  | ||||||
|         //noinspection JavaReflectionMemberAccess |  | ||||||
|         return IntentSender.class.getConstructor(IIntentSender.class).newInstance(binder); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,41 +0,0 @@ | |||||||
| package dev.imranr.obtainium.util; |  | ||||||
|  |  | ||||||
| import android.content.Context; |  | ||||||
| import android.content.pm.IPackageInstaller; |  | ||||||
| import android.content.pm.IPackageInstallerSession; |  | ||||||
| import android.content.pm.PackageInstaller; |  | ||||||
| import android.content.pm.PackageManager; |  | ||||||
| import android.os.Build; |  | ||||||
|  |  | ||||||
| import java.lang.reflect.InvocationTargetException; |  | ||||||
|  |  | ||||||
| @SuppressWarnings({"JavaReflectionMemberAccess"}) |  | ||||||
| public class PackageInstallerUtils { |  | ||||||
|  |  | ||||||
|     public static PackageInstaller createPackageInstaller(IPackageInstaller installer, String installerPackageName, String installerAttributionTag, int userId) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException { |  | ||||||
|         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) { |  | ||||||
|             return PackageInstaller.class.getConstructor(IPackageInstaller.class, String.class, String.class, int.class) |  | ||||||
|                     .newInstance(installer, installerPackageName, installerAttributionTag, userId); |  | ||||||
|         } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { |  | ||||||
|             return PackageInstaller.class.getConstructor(IPackageInstaller.class, String.class, int.class) |  | ||||||
|                     .newInstance(installer, installerPackageName, userId); |  | ||||||
|         } else { |  | ||||||
|             return PackageInstaller.class.getConstructor(Context.class, PackageManager.class, IPackageInstaller.class, String.class, int.class) |  | ||||||
|                     .newInstance(ApplicationUtils.getApplication(), ApplicationUtils.getApplication().getPackageManager(), installer, installerPackageName, userId); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static PackageInstaller.Session createSession(IPackageInstallerSession session) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException { |  | ||||||
|         return PackageInstaller.Session.class.getConstructor(IPackageInstallerSession.class) |  | ||||||
|                 .newInstance(session); |  | ||||||
|  |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static int getInstallFlags(PackageInstaller.SessionParams params) throws NoSuchFieldException, IllegalAccessException { |  | ||||||
|         return (int) PackageInstaller.SessionParams.class.getDeclaredField("installFlags").get(params); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static void setInstallFlags(PackageInstaller.SessionParams params, int newValue) throws NoSuchFieldException, IllegalAccessException { |  | ||||||
|         PackageInstaller.SessionParams.class.getDeclaredField("installFlags").set(params, newValue); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,68 +0,0 @@ | |||||||
| package dev.imranr.obtainium.util; |  | ||||||
|  |  | ||||||
| import android.content.Context; |  | ||||||
| import android.content.pm.IPackageInstaller; |  | ||||||
| import android.content.pm.IPackageManager; |  | ||||||
| import android.content.pm.UserInfo; |  | ||||||
| import android.os.Build; |  | ||||||
| import android.os.IUserManager; |  | ||||||
| import android.os.RemoteException; |  | ||||||
|  |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| import rikka.shizuku.ShizukuBinderWrapper; |  | ||||||
| import rikka.shizuku.SystemServiceHelper; |  | ||||||
|  |  | ||||||
| public class ShizukuSystemServerApi { |  | ||||||
|  |  | ||||||
|     private static final Singleton<IPackageManager> PACKAGE_MANAGER = new Singleton<IPackageManager>() { |  | ||||||
|         @Override |  | ||||||
|         protected IPackageManager create() { |  | ||||||
|             return IPackageManager.Stub.asInterface(new ShizukuBinderWrapper(SystemServiceHelper.getSystemService("package"))); |  | ||||||
|         } |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     private static final Singleton<IUserManager> USER_MANAGER = new Singleton<IUserManager>() { |  | ||||||
|         @Override |  | ||||||
|         protected IUserManager create() { |  | ||||||
|             return IUserManager.Stub.asInterface(new ShizukuBinderWrapper(SystemServiceHelper.getSystemService(Context.USER_SERVICE))); |  | ||||||
|         } |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     public static IPackageInstaller PackageManager_getPackageInstaller() throws RemoteException { |  | ||||||
|         IPackageInstaller packageInstaller = PACKAGE_MANAGER.get().getPackageInstaller(); |  | ||||||
|         return IPackageInstaller.Stub.asInterface(new ShizukuBinderWrapper(packageInstaller.asBinder())); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static List<UserInfo> UserManager_getUsers(boolean excludePartial, boolean excludeDying, boolean excludePreCreated) throws RemoteException { |  | ||||||
|         if (Build.VERSION.SDK_INT >= 30) { |  | ||||||
|             return USER_MANAGER.get().getUsers(excludePartial, excludeDying, excludePreCreated); |  | ||||||
|         } else { |  | ||||||
|             try { |  | ||||||
|                 return USER_MANAGER.get().getUsers(excludeDying); |  | ||||||
|             } catch (NoSuchFieldError e) { |  | ||||||
|                 return USER_MANAGER.get().getUsers(excludePartial, excludeDying, excludePreCreated); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // method 2: use transactRemote directly |  | ||||||
|     /*public static List<UserInfo> UserManager_getUsers(boolean excludeDying) { |  | ||||||
|         Parcel data = SystemServiceHelper.obtainParcel(Context.USER_SERVICE, "android.os.IUserManager", "getUsers"); |  | ||||||
|         Parcel reply = Parcel.obtain(); |  | ||||||
|         data.writeInt(excludeDying ? 1 : 0); |  | ||||||
|  |  | ||||||
|         List<UserInfo> res = null; |  | ||||||
|         try { |  | ||||||
|             ShizukuService.transactRemote(data, reply, 0); |  | ||||||
|             reply.readException(); |  | ||||||
|             res = reply.createTypedArrayList(UserInfo.CREATOR); |  | ||||||
|         } catch (RemoteException e) { |  | ||||||
|             Log.e("ShizukuSample", "UserManager#getUsers", e); |  | ||||||
|         } finally { |  | ||||||
|             data.recycle(); |  | ||||||
|             reply.recycle(); |  | ||||||
|         } |  | ||||||
|         return res; |  | ||||||
|     }*/ |  | ||||||
| } |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| package dev.imranr.obtainium.util; |  | ||||||
|  |  | ||||||
| public abstract class Singleton<T> { |  | ||||||
|  |  | ||||||
|     private T mInstance; |  | ||||||
|  |  | ||||||
|     protected abstract T create(); |  | ||||||
|  |  | ||||||
|     public final T get() { |  | ||||||
|         synchronized (this) { |  | ||||||
|             if (mInstance == null) { |  | ||||||
|                 mInstance = create(); |  | ||||||
|             } |  | ||||||
|             return mInstance; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										22
									
								
								android/app/src/main/res/xml/locales_config.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								android/app/src/main/res/xml/locales_config.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <locale-config xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|  |     <locale android:name="bs"/> | ||||||
|  |     <locale android:name="cs"/> | ||||||
|  |     <locale android:name="de"/> | ||||||
|  |     <locale android:name="en"/> | ||||||
|  |     <locale android:name="es"/> | ||||||
|  |     <locale android:name="fa"/> | ||||||
|  |     <locale android:name="fr"/> | ||||||
|  |     <locale android:name="hu"/> | ||||||
|  |     <locale android:name="it"/> | ||||||
|  |     <locale android:name="ja"/> | ||||||
|  |     <locale android:name="nl"/> | ||||||
|  |     <locale android:name="pl"/> | ||||||
|  |     <locale android:name="pt"/> | ||||||
|  |     <locale android:name="ru"/> | ||||||
|  |     <locale android:name="sv"/> | ||||||
|  |     <locale android:name="tr"/> | ||||||
|  |     <locale android:name="uk"/> | ||||||
|  |     <locale android:name="vi"/> | ||||||
|  |     <locale android:name="zh"/> | ||||||
|  | </locale-config> | ||||||
| @@ -1,3 +1,3 @@ | |||||||
| org.gradle.jvmargs=-Xmx1536M | org.gradle.jvmargs=-Xmx2048M | ||||||
| android.useAndroidX=true | android.useAndroidX=true | ||||||
| android.enableJetifier=true | android.enableJetifier=true | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								assets/fonts/WixMadeforDisplay-Regular.otf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								assets/fonts/WixMadeforDisplay-Regular.otf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(obavezno)", |     "requiredInBrackets": "(obavezno)", | ||||||
|     "dropdownNoOptsError": "GREŠKA: PADAJUĆI MENI MORA IMATI NAJMANJE JEDNU OPCIJU", |     "dropdownNoOptsError": "GREŠKA: PADAJUĆI MENI MORA IMATI NAJMANJE JEDNU OPCIJU", | ||||||
|     "colour": "Boja", |     "colour": "Boja", | ||||||
|  |     "standard": "Standard", | ||||||
|  |     "custom": "Custom", | ||||||
|  |     "useMaterialYou": "Koristi Material You temu", | ||||||
|     "githubStarredRepos": "GitHub repo-i sa zvjezdicom", |     "githubStarredRepos": "GitHub repo-i sa zvjezdicom", | ||||||
|     "uname": "Korisničko ime", |     "uname": "Korisničko ime", | ||||||
|     "wrongArgNum": "Naveden je pogrešan broj argumenata", |     "wrongArgNum": "Naveden je pogrešan broj argumenata", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Tamna", |     "dark": "Tamna", | ||||||
|     "light": "Svijetla", |     "light": "Svijetla", | ||||||
|     "followSystem": "Pratite sistem", |     "followSystem": "Pratite sistem", | ||||||
|  |     "followSystemThemeExplanation": "Praćenje sistemske teme je moguće jedino koristeći aplikacije treće strane", | ||||||
|     "useBlackTheme": "Koristite čisto crnu tamnu temu", |     "useBlackTheme": "Koristite čisto crnu tamnu temu", | ||||||
|     "appSortBy": "Aplikacije sortirane po", |     "appSortBy": "Aplikacije sortirane po", | ||||||
|     "authorName": "Autor/Ime", |     "authorName": "Autor/Ime", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Nema novih ažuriranja.", |     "noNewUpdates": "Nema novih ažuriranja.", | ||||||
|     "xHasAnUpdate": "{} ima ažuriranje.", |     "xHasAnUpdate": "{} ima ažuriranje.", | ||||||
|     "appsUpdated": "Aplikacije su ažurirane", |     "appsUpdated": "Aplikacije su ažurirane", | ||||||
|  |     "appsNotUpdated": "Neuspješno ažuriranje aplikacija", | ||||||
|     "appsUpdatedNotifDescription": "Obavještava korisnika da su u pozadini primijenjena ažuriranja na jednu ili više aplikacija", |     "appsUpdatedNotifDescription": "Obavještava korisnika da su u pozadini primijenjena ažuriranja na jednu ili više aplikacija", | ||||||
|     "xWasUpdatedToY": "{} je ažuriran na {}.", |     "xWasUpdatedToY": "{} je ažuriran na {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Neuspješno ažuriranje {} na {}.", | ||||||
|     "errorCheckingUpdates": "Greška pri provjeri ažuriranja", |     "errorCheckingUpdates": "Greška pri provjeri ažuriranja", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Obavijest koja se prikazuje kada provjera sigurnosnog ažuriranja ne uspije", |     "errorCheckingUpdatesNotifDescription": "Obavijest koja se prikazuje kada provjera sigurnosnog ažuriranja ne uspije", | ||||||
|     "appsRemoved": "Aplikacije su uklonjene", |     "appsRemoved": "Aplikacije su uklonjene", | ||||||
| @@ -185,7 +191,7 @@ | |||||||
|     "downloadingX": "Preuzimanje {}", |     "downloadingX": "Preuzimanje {}", | ||||||
|     "downloadX": "Download {}", |     "downloadX": "Download {}", | ||||||
|     "downloadedX": "Downloaded {}", |     "downloadedX": "Downloaded {}", | ||||||
|     "releaseAsset": "Release Asset", |     "releaseAsset": "Fajlovi verzije", | ||||||
|     "downloadNotifDescription": "Obavještava korisnika o napretku u preuzimanju aplikacije", |     "downloadNotifDescription": "Obavještava korisnika o napretku u preuzimanju aplikacije", | ||||||
|     "noAPKFound": "APK nije pronađen", |     "noAPKFound": "APK nije pronađen", | ||||||
|     "noVersionDetection": "Nema detekcije verzije", |     "noVersionDetection": "Nema detekcije verzije", | ||||||
| @@ -247,12 +253,14 @@ | |||||||
|     "verifyLatestTag": "Provjerite 'posljednu' ('latest') oznaku", |     "verifyLatestTag": "Provjerite 'posljednu' ('latest') oznaku", | ||||||
|     "intermediateLinkRegex": "Filter za 'srednju' vezu za posjetu", |     "intermediateLinkRegex": "Filter za 'srednju' vezu za posjetu", | ||||||
|     "filterByLinkText": "Filtriraj linkove prema tekstu linka", |     "filterByLinkText": "Filtriraj linkove prema tekstu linka", | ||||||
|     "intermediateLinkNotFound": "Intermediate link nije nađen", |     "intermediateLinkNotFound": "Intermediate veza nije nađena", | ||||||
|     "intermediateLink": "srednja karika", |     "intermediateLink": "Intermediate veza", | ||||||
|     "exemptFromBackgroundUpdates": "Izuzmi iz ažuriranja u pozadini (ako su uključeni)", |     "exemptFromBackgroundUpdates": "Izuzmi iz ažuriranja u pozadini (ako su uključeni)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Isključite ažuriranje u pozadini kada niste na WiFi-ju", |     "bgUpdatesOnWiFiOnly": "Isključite ažuriranje u pozadini kada niste na WiFi-ju", | ||||||
|     "autoSelectHighestVersionCode": "Automatski izaberite najveću (verziju) versionCode APK-a", |     "autoSelectHighestVersionCode": "Automatski izaberite najveću (verziju) versionCode APK-a", | ||||||
|     "versionExtractionRegEx": "RegEx ekstrakcija verzije", |     "versionExtractionRegEx": "RegEx ekstrakcija verzije", | ||||||
|  |     "trimVersionString": "Trim Version String With RegEx", | ||||||
|  |     "matchGroupToUseForX": "Match Group to Use for \"{}\"", | ||||||
|     "matchGroupToUse": "Podjesite grupu za upotebu", |     "matchGroupToUse": "Podjesite grupu za upotebu", | ||||||
|     "highlightTouchTargets": "Istaknite manje vidljive touch mete", |     "highlightTouchTargets": "Istaknite manje vidljive touch mete", | ||||||
|     "pickExportDir": "Izaberite datoteku za izvoz", |     "pickExportDir": "Izaberite datoteku za izvoz", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Podržite fiksne APK URL-ove", |     "supportFixedAPKURL": "Podržite fiksne APK URL-ove", | ||||||
|     "selectX": "Izaberite {}", |     "selectX": "Izaberite {}", | ||||||
|     "parallelDownloads": "Dozvoli paralelna preuzimanja", |     "parallelDownloads": "Dozvoli paralelna preuzimanja", | ||||||
|     "installMethod": "Način instalacije", |     "useShizuku": "Koristi Shizuku ili Sui za instaliranje", | ||||||
|     "normal": "normalno", |     "shizukuBinderNotFound": "Shizuku nije pokrenut", | ||||||
|     "root": "korijen", |     "shizukuOld": "Stara Shizuku verzija (<11) - ažurirajte je", | ||||||
|     "shizukuBinderNotFound": "Shizuku is not running", |     "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", |     "useSystemFont": "Koristite sistemski font", | ||||||
|     "systemFontError": "Greška pri učitavanju sistemskog fonta: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Koristite kod verzije aplikacije kao verziju koju je otkrio OS", |     "useVersionCodeAsOSVersion": "Koristite kod verzije aplikacije kao verziju koju je otkrio OS", | ||||||
|     "requestHeader": "Zaglavlje zahtjeva", |     "requestHeader": "Zaglavlje zahtjeva", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Koristite najnovije otpremanje materijala kao datum izdavanja", |     "useLatestAssetDateAsReleaseDate": "Koristite najnovije otpremanje materijala kao datum izdavanja", | ||||||
| @@ -299,11 +307,14 @@ | |||||||
|     "installed": "Instalirano", |     "installed": "Instalirano", | ||||||
|     "latest": "Najnoviji", |     "latest": "Najnoviji", | ||||||
|     "invertRegEx": "Obrni regularni izraz", |     "invertRegEx": "Obrni regularni izraz", | ||||||
|     "note": "Note", |     "note": "Zabelješke", | ||||||
|     "selfHostedNote": "The \"{}\" dropdown can be used to reach self-hosted/custom instances of any source.", |     "selfHostedNote": "\"{}\" padajući meni se može koristiti da dosegnete vlastite/prilagođene instance bilo kojeg izvora.", | ||||||
|     "badDownload": "The APK could not be parsed (incompatible or partial download)", |     "badDownload": "APK ne može biti raščlanjen (nekomaptibilno ili delimično preuzimanje)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Share new Apps with AppVerifier (if available)", |     "beforeNewInstallsShareToAppVerifier": "Dijeli nove aplikacije sa AppVerifier-om (ako je dostupno)", | ||||||
|     "appVerifierInstructionToast": "Share to AppVerifier, then return here when ready.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "Želite li ukloniti aplikaciju?", |         "one": "Želite li ukloniti aplikaciju?", | ||||||
|         "other": "Želite li ukloniti aplikacije?" |         "other": "Želite li ukloniti aplikacije?" | ||||||
| @@ -352,12 +363,16 @@ | |||||||
|         "one": "{} i još 1 aplikacija je ažurirana.", |         "one": "{} i još 1 aplikacija je ažurirana.", | ||||||
|         "other": "{} i još {} aplikacija je ažurirano." |         "other": "{} i još {} aplikacija je ažurirano." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Failed to update {} and 1 more app.", | ||||||
|  |         "other": "Failed to update {} and {} more apps." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} i još jedna aplikacija je vjerovatno ažurirana.", |         "one": "{} i još jedna aplikacija je vjerovatno ažurirana.", | ||||||
|         "other": "{} i još {} aplikacija su vjerovatno ažurirane." |         "other": "{} i još {} aplikacija su vjerovatno ažurirane." | ||||||
|     }, |     }, | ||||||
|     "apk": { |     "apk": { | ||||||
|         "one": "{} APK", |         "one": "{} APK", | ||||||
|         "other": "{} APKs" |         "other": "{} APK-a" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(Požadované)", |     "requiredInBrackets": "(Požadované)", | ||||||
|     "dropdownNoOptsError": "ERROR: DROPDOWN MUSÍ MÍT AŽ JEDNU MOŽNOST", |     "dropdownNoOptsError": "ERROR: DROPDOWN MUSÍ MÍT AŽ JEDNU MOŽNOST", | ||||||
|     "colour": "Barva", |     "colour": "Barva", | ||||||
|  |     "standard": "Standardní", | ||||||
|  |     "custom": "Vlastní", | ||||||
|  |     "useMaterialYou": "Použijte materiál, který jste", | ||||||
|     "githubStarredRepos": "GitHub označená hvězdičkou", |     "githubStarredRepos": "GitHub označená hvězdičkou", | ||||||
|     "uname": "Uživatelské jméno", |     "uname": "Uživatelské jméno", | ||||||
|     "wrongArgNum": "Nesprávný počet zadaných argumentů", |     "wrongArgNum": "Nesprávný počet zadaných argumentů", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Tmavé", |     "dark": "Tmavé", | ||||||
|     "light": "Světlé", |     "light": "Světlé", | ||||||
|     "followSystem": "Jako systém", |     "followSystem": "Jako systém", | ||||||
|  |     "followSystemThemeExplanation": "Sledování motivu systému je možné pouze pomocí aplikací třetích stran.", | ||||||
|     "useBlackTheme": "Použít čistě černé tmavé téma", |     "useBlackTheme": "Použít čistě černé tmavé téma", | ||||||
|     "appSortBy": "Seřadit podle", |     "appSortBy": "Seřadit podle", | ||||||
|     "authorName": "Autor/Jméno", |     "authorName": "Autor/Jméno", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Žádné nové aktualizace.", |     "noNewUpdates": "Žádné nové aktualizace.", | ||||||
|     "xHasAnUpdate": "{} má aktualizaci.", |     "xHasAnUpdate": "{} má aktualizaci.", | ||||||
|     "appsUpdated": "Aplikace aktualizovány", |     "appsUpdated": "Aplikace aktualizovány", | ||||||
|  |     "appsNotUpdated": "Nepodařilo se aktualizovat aplikace", | ||||||
|     "appsUpdatedNotifDescription": "Upozornit, že byly provedeny aktualizace jedné nebo více aplikací na pozadí", |     "appsUpdatedNotifDescription": "Upozornit, že byly provedeny aktualizace jedné nebo více aplikací na pozadí", | ||||||
|     "xWasUpdatedToY": "{} byla aktualizována na {}", |     "xWasUpdatedToY": "{} byla aktualizována na {}", | ||||||
|  |     "xWasNotUpdatedToY": "Nepodařilo se aktualizovat {} na {}.", | ||||||
|     "errorCheckingUpdates": "Chyba kontroly aktualizací", |     "errorCheckingUpdates": "Chyba kontroly aktualizací", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Zobrazit oznámení při neúspěšné kontrole aktualizací na pozadí", |     "errorCheckingUpdatesNotifDescription": "Zobrazit oznámení při neúspěšné kontrole aktualizací na pozadí", | ||||||
|     "appsRemoved": "Odstraněné aplikace", |     "appsRemoved": "Odstraněné aplikace", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "Deaktivovat aktualizace na pozadí, pokud není k dispozici Wi-Fi", |     "bgUpdatesOnWiFiOnly": "Deaktivovat aktualizace na pozadí, pokud není k dispozici Wi-Fi", | ||||||
|     "autoSelectHighestVersionCode": "Automaticky vybrat nejvyšší verzi APK", |     "autoSelectHighestVersionCode": "Automaticky vybrat nejvyšší verzi APK", | ||||||
|     "versionExtractionRegEx": "Extrakce verze pomocí RegEx", |     "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ě", |     "matchGroupToUse": "Odpovídá použité skupině", | ||||||
|     "highlightTouchTargets": "Zvýraznit méně zjevné cíle dotyku", |     "highlightTouchTargets": "Zvýraznit méně zjevné cíle dotyku", | ||||||
|     "pickExportDir": "Vybrat adresář pro export", |     "pickExportDir": "Vybrat adresář pro export", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Odhadnout novější verzi na základě prvních třiceti číslic kontrolního součtu adresy URL APK, pokud není podporována jinak", |     "supportFixedAPKURL": "Odhadnout novější verzi na základě prvních třiceti číslic kontrolního součtu adresy URL APK, pokud není podporována jinak", | ||||||
|     "selectX": "Vybrat {}", |     "selectX": "Vybrat {}", | ||||||
|     "parallelDownloads": "Povolit souběžné stahování", |     "parallelDownloads": "Povolit souběžné stahování", | ||||||
|     "installMethod": "Metoda instalace", |     "useShizuku": "K instalaci použijte Shizuku nebo Sui", | ||||||
|     "normal": "Normální", |  | ||||||
|     "root": "Správce", |  | ||||||
|     "shizukuBinderNotFound": "Shizuku neběží", |     "shizukuBinderNotFound": "Shizuku neběží", | ||||||
|  |     "shizukuOld": "Stará verze Shizuku (<11) - aktualizujte ji", | ||||||
|  |     "shizukuOldAndroidWithADB": "Shizuku běží na Androidu < 8.1 s ADB - aktualizujte Android nebo místo toho použijte Sui", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Nastavení Google Play jako zdroje instalace (pokud se používá Shizuku)", | ||||||
|     "useSystemFont": "Použít systémové písmo", |     "useSystemFont": "Použít systémové písmo", | ||||||
|     "systemFontError": "Chyba při načítání systémového písma: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Použít kód verze aplikace jako verzi zjištěnou OS", |     "useVersionCodeAsOSVersion": "Použít kód verze aplikace jako verzi zjištěnou OS", | ||||||
|     "requestHeader": "Hlavička požadavku", |     "requestHeader": "Hlavička požadavku", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Použít poslední nahrané dílo jako datum vydání", |     "useLatestAssetDateAsReleaseDate": "Použít poslední nahrané dílo jako datum vydání", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "APK nelze analyzovat (nekompatibilní nebo částečné stažení)", |     "badDownload": "APK nelze analyzovat (nekompatibilní nebo částečné stažení)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Sdílení nových aplikací s aplikací AppVerifier (pokud je k dispozici)", |     "beforeNewInstallsShareToAppVerifier": "Sdílení nových aplikací s aplikací AppVerifier (pokud je k dispozici)", | ||||||
|     "appVerifierInstructionToast": "Sdílejte do aplikace AppVerifier a po dokončení se sem vraťte.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "Odstranit Apku?", |         "one": "Odstranit Apku?", | ||||||
|         "other": "Odstranit Apky?" |         "other": "Odstranit Apky?" | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} a 1 další aplikace mají aktualizace.", |         "one": "{} a 1 další aplikace mají aktualizace.", | ||||||
|         "other": "{} a {} další aplikace byly aktualizovány." |         "other": "{} a {} další aplikace byly aktualizovány." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Nepodařilo se aktualizovat {} a 1 další aplikaci.", | ||||||
|  |         "other": "Nepodařilo se aktualizovat {} a {} další aplikace." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} a 1 další aplikace možno aktualizovat", |         "one": "{} a 1 další aplikace možno aktualizovat", | ||||||
|         "other": "{} a {} další aplikace mohou být aktualizovány." |         "other": "{} a {} další aplikace mohou být aktualizovány." | ||||||
|   | |||||||
							
								
								
									
										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", |     "unexpectedError": "Unerwarteter Fehler", | ||||||
|     "ok": "OK", |     "ok": "OK", | ||||||
|     "and": "und", |     "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", |     "includePrereleases": "Vorabversionen einbeziehen", | ||||||
|     "fallbackToOlderReleases": "Fallback auf ältere Versionen", |     "fallbackToOlderReleases": "Fallback auf ältere Versionen", | ||||||
|     "filterReleaseTitlesByRegEx": "Release-Titel nach regulärem Ausdruck\nfiltern", |     "filterReleaseTitlesByRegEx": "Release-Titel nach regulärem Ausdruck\nfiltern", | ||||||
| @@ -19,17 +19,20 @@ | |||||||
|     "noDescription": "Keine Beschreibung", |     "noDescription": "Keine Beschreibung", | ||||||
|     "cancel": "Abbrechen", |     "cancel": "Abbrechen", | ||||||
|     "continue": "Weiter", |     "continue": "Weiter", | ||||||
|     "requiredInBrackets": "(Benötigt)", |     "requiredInBrackets": "(wird benötigt)", | ||||||
|     "dropdownNoOptsError": "FEHLER: DROPDOWN MUSS MINDESTENS EINE OPTION HABEN", |     "dropdownNoOptsError": "FEHLER: DROPDOWN MUSS MINDESTENS EINE OPTION HABEN", | ||||||
|     "colour": "Farbe", |     "colour": "Farbe", | ||||||
|  |     "standard": "Standard", | ||||||
|  |     "custom": "Benutzerdefiniert", | ||||||
|  |     "useMaterialYou": "Material You verwenden", | ||||||
|     "githubStarredRepos": "GitHub Starred Repos", |     "githubStarredRepos": "GitHub Starred Repos", | ||||||
|     "uname": "Benutzername", |     "uname": "Benutzername", | ||||||
|     "wrongArgNum": "Falsche Anzahl von Argumenten (Parametern) übermittelt", |     "wrongArgNum": "Falsche Anzahl von Argumenten (Parametern) übermittelt", | ||||||
|     "xIsTrackOnly": "{} ist nur zur Nachverfolgung", |     "xIsTrackOnly": "{} ist nur zur Nachverfolgung", | ||||||
|     "source": "Quelle", |     "source": "Quelle", | ||||||
|     "app": "App", |     "app": "App", | ||||||
|     "appsFromSourceAreTrackOnly": "Apps aus dieser Quelle sind nur zum Nachverfolgen.", |     "appsFromSourceAreTrackOnly": "Apps aus dieser Quelle sind nur zur Nachverfolgung.", | ||||||
|     "youPickedTrackOnly": "Sie haben die Option „Nur Nachverfolgen“ gewählt.", |     "youPickedTrackOnly": "Sie haben die Option „Nur nachverfolgen“ gewählt.", | ||||||
|     "trackOnlyAppDescription": "Die App wird auf Aktualisierungen überwacht, aber Obtainium wird sie nicht herunterladen oder installieren.", |     "trackOnlyAppDescription": "Die App wird auf Aktualisierungen überwacht, aber Obtainium wird sie nicht herunterladen oder installieren.", | ||||||
|     "cancelled": "Abgebrochen", |     "cancelled": "Abgebrochen", | ||||||
|     "appAlreadyAdded": "App bereits hinzugefügt", |     "appAlreadyAdded": "App bereits hinzugefügt", | ||||||
| @@ -42,7 +45,7 @@ | |||||||
|     "search": "Suchen", |     "search": "Suchen", | ||||||
|     "additionalOptsFor": "Zusatzoptionen für {}", |     "additionalOptsFor": "Zusatzoptionen für {}", | ||||||
|     "supportedSources": "Unterstützte Quellen", |     "supportedSources": "Unterstützte Quellen", | ||||||
|     "trackOnlyInBrackets": "(Nur Nachverfolgen)", |     "trackOnlyInBrackets": "(Nur nachverfolgen)", | ||||||
|     "searchableInBrackets": "(Durchsuchbar)", |     "searchableInBrackets": "(Durchsuchbar)", | ||||||
|     "appsString": "Apps", |     "appsString": "Apps", | ||||||
|     "noApps": "Keine Apps", |     "noApps": "Keine Apps", | ||||||
| @@ -58,22 +61,22 @@ | |||||||
|     "xWillBeRemovedButRemainInstalled": "{} wird aus Obtainium entfernt, bleibt aber auf dem Gerät installiert.", |     "xWillBeRemovedButRemainInstalled": "{} wird aus Obtainium entfernt, bleibt aber auf dem Gerät installiert.", | ||||||
|     "removeSelectedAppsQuestion": "Ausgewählte Apps entfernen?", |     "removeSelectedAppsQuestion": "Ausgewählte Apps entfernen?", | ||||||
|     "removeSelectedApps": "Ausgewählte Apps entfernen", |     "removeSelectedApps": "Ausgewählte Apps entfernen", | ||||||
|     "updateX": "Aktualisiere {}", |     "updateX": "{} aktualisieren", | ||||||
|     "installX": "Installiere {}", |     "installX": "{} installieren", | ||||||
|     "markXTrackOnlyAsUpdated": "Markiere {}\n(Nur Nachverfolgen)\nals aktualisiert", |     "markXTrackOnlyAsUpdated": "Markiere {}\n(Nur nachverfolgen)\nals aktualisiert", | ||||||
|     "changeX": "Ändere {}", |     "changeX": "{} ändern", | ||||||
|     "installUpdateApps": "Apps installieren/aktualisieren", |     "installUpdateApps": "Apps installieren/aktualisieren", | ||||||
|     "installUpdateSelectedApps": "Ausgewählte 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", |     "no": "Nein", | ||||||
|     "yes": "Ja", |     "yes": "Ja", | ||||||
|     "markSelectedAppsUpdated": "Markiere ausgewählte Apps als aktuell", |     "markSelectedAppsUpdated": "Ausgewählte Apps als aktuell markieren", | ||||||
|     "pinToTop": "Oben anheften", |     "pinToTop": "Oben anheften", | ||||||
|     "unpinFromTop": "„Oben anheften“ aufheben", |     "unpinFromTop": "„Oben anheften“ aufheben", | ||||||
|     "resetInstallStatusForSelectedAppsQuestion": "Installationsstatus für ausgewählte Apps zurücksetzen?", |     "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.", |     "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", |     "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", |     "shareSelectedAppURLs": "Ausgewählte App-URLs teilen", | ||||||
|     "resetInstallStatus": "Installationsstatus zurücksetzen", |     "resetInstallStatus": "Installationsstatus zurücksetzen", | ||||||
|     "more": "Mehr", |     "more": "Mehr", | ||||||
| @@ -81,25 +84,25 @@ | |||||||
|     "showOutdatedOnly": "Nur nicht aktuelle Apps anzeigen", |     "showOutdatedOnly": "Nur nicht aktuelle Apps anzeigen", | ||||||
|     "filter": "Filter", |     "filter": "Filter", | ||||||
|     "filterApps": "Apps filtern", |     "filterApps": "Apps filtern", | ||||||
|     "appName": "App Name", |     "appName": "App-Name", | ||||||
|     "author": "Autor", |     "author": "Autor", | ||||||
|     "upToDateApps": "Apps mit aktueller Version", |     "upToDateApps": "Apps mit aktueller Version", | ||||||
|     "nonInstalledApps": "Nicht installierte Apps", |     "nonInstalledApps": "Nicht installierte Apps", | ||||||
|     "importExport": "Import/Export", |     "importExport": "Import/Export", | ||||||
|     "settings": "Einstellungen", |     "settings": "Einstellungen", | ||||||
|     "exportedTo": "Exportiert zu {}", |     "exportedTo": "Exportiert nach {}", | ||||||
|     "obtainiumExport": "Obtainium-Export", |     "obtainiumExport": "Obtainium-Export", | ||||||
|     "invalidInput": "Ungültige Eingabe", |     "invalidInput": "Ungültige Eingabe", | ||||||
|     "importedX": "Importiert {}", |     "importedX": "{} importiert", | ||||||
|     "obtainiumImport": "Obtainium-Import", |     "obtainiumImport": "Obtainium-Import", | ||||||
|     "importFromURLList": "Importieren aus URL-Liste", |     "importFromURLList": "Aus URL-Liste importieren", | ||||||
|     "searchQuery": "Suchanfrage", |     "searchQuery": "Suchanfrage", | ||||||
|     "appURLList": "App URL-Liste", |     "appURLList": "App-URL-Liste", | ||||||
|     "line": "Linie", |     "line": "Linie", | ||||||
|     "searchX": "Suche {}", |     "searchX": "{} suchen", | ||||||
|     "noResults": "Keine Ergebnisse gefunden", |     "noResults": "Keine Ergebnisse gefunden", | ||||||
|     "importX": "Importieren {}", |     "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.", |     "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", |     "importErrors": "Importfehler", | ||||||
|     "importedXOfYApps": "{} von {} Apps importiert.", |     "importedXOfYApps": "{} von {} Apps importiert.", | ||||||
|     "followingURLsHadErrors": "Bei folgenden URLs traten Fehler auf:", |     "followingURLsHadErrors": "Bei folgenden URLs traten Fehler auf:", | ||||||
| @@ -110,7 +113,8 @@ | |||||||
|     "dark": "Dunkel", |     "dark": "Dunkel", | ||||||
|     "light": "Hell", |     "light": "Hell", | ||||||
|     "followSystem": "System folgen", |     "followSystem": "System folgen", | ||||||
|     "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", |     "appSortBy": "App sortieren nach", | ||||||
|     "authorName": "Autor/Name", |     "authorName": "Autor/Name", | ||||||
|     "nameAuthor": "Name/Autor", |     "nameAuthor": "Name/Autor", | ||||||
| @@ -142,21 +146,23 @@ | |||||||
|     "updatesAvailableNotifDescription": "Benachrichtigt den Nutzer, dass Aktualisierungen für eine oder mehrere von Obtainium verfolgte Apps verfügbar sind", |     "updatesAvailableNotifDescription": "Benachrichtigt den Nutzer, dass Aktualisierungen für eine oder mehrere von Obtainium verfolgte Apps verfügbar sind", | ||||||
|     "noNewUpdates": "Keine neuen Aktualisierungen.", |     "noNewUpdates": "Keine neuen Aktualisierungen.", | ||||||
|     "xHasAnUpdate": "{} hat eine Aktualisierung.", |     "xHasAnUpdate": "{} hat eine Aktualisierung.", | ||||||
|     "appsUpdated": "Apps aktualisiert", |     "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", |     "appsUpdatedNotifDescription": "Benachrichtigt den Benutzer, dass Aktualisierungen für eine oder mehrere Apps im Hintergrund durchgeführt wurden", | ||||||
|     "xWasUpdatedToY": "{} wurde auf {} aktualisiert.", |     "xWasUpdatedToY": "{} wurde auf {} aktualisiert.", | ||||||
|  |     "xWasNotUpdatedToY": "Die Aktualisierung von {} auf {} ist fehlgeschlagen.", | ||||||
|     "errorCheckingUpdates": "Fehler beim Prüfen auf Aktualisierungen", |     "errorCheckingUpdates": "Fehler beim Prüfen auf Aktualisierungen", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Eine Benachrichtigung, die angezeigt wird, wenn die Prüfung der Hintergrundaktualisierung fehlschlägt", |     "errorCheckingUpdatesNotifDescription": "Eine Benachrichtigung, die angezeigt wird, wenn die Prüfung der Hintergrundaktualisierung fehlschlägt", | ||||||
|     "appsRemoved": "Apps entfernt", |     "appsRemoved": "Apps entfernt", | ||||||
|     "appsRemovedNotifDescription": "Benachrichtigt den Benutzer, dass eine oder mehrere Apps aufgrund von Fehlern beim Laden entfernt wurden", |     "appsRemovedNotifDescription": "Benachrichtigt den Benutzer, dass eine oder mehrere Apps aufgrund von Fehlern beim Laden entfernt wurden", | ||||||
|     "xWasRemovedDueToErrorY": "{} wurde aufgrund des folgenden Fehlers entfernt: {}", |     "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", |     "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", |     "completeAppInstallationNotifDescription": "Aufforderung an den Benutzer, zu Obtainium zurückzukehren, um die Installation einer App abzuschließen", | ||||||
|     "checkingForUpdates": "Nach Aktualisierungen suchen", |     "checkingForUpdates": "Nach Aktualisierungen suchen", | ||||||
|     "checkingForUpdatesNotifDescription": "Vorübergehende Benachrichtigung, die bei der Suche nach Aktualisierungen angezeigt wird", |     "checkingForUpdatesNotifDescription": "Vorübergehende Benachrichtigung, die bei der Suche nach Aktualisierungen angezeigt wird", | ||||||
|     "pleaseAllowInstallPerm": "Bitte erlauben Sie Obtainium die Installation von Apps", |     "pleaseAllowInstallPerm": "Bitte erlauben Sie Obtainium die Installation von Apps", | ||||||
|     "trackOnly": "Nur Nachverfolgen", |     "trackOnly": "Nur nachverfolgen", | ||||||
|     "errorWithHttpStatusCode": "Fehler {}", |     "errorWithHttpStatusCode": "Fehler {}", | ||||||
|     "versionCorrectionDisabled": "Versionskorrektur deaktiviert (Plugin scheint nicht zu funktionieren)", |     "versionCorrectionDisabled": "Versionskorrektur deaktiviert (Plugin scheint nicht zu funktionieren)", | ||||||
|     "unknown": "Unbekannt", |     "unknown": "Unbekannt", | ||||||
| @@ -167,25 +173,25 @@ | |||||||
|     "lastUpdateCheckX": "Letzte Aktualisierungsprüfung: {}", |     "lastUpdateCheckX": "Letzte Aktualisierungsprüfung: {}", | ||||||
|     "remove": "Entfernen", |     "remove": "Entfernen", | ||||||
|     "yesMarkUpdated": "Ja, als aktualisiert markieren", |     "yesMarkUpdated": "Ja, als aktualisiert markieren", | ||||||
|     "fdroid": "offizielles F-Droid-Repo", |     "fdroid": "Offizielles F-Droid-Repo", | ||||||
|     "appIdOrName": "App ID oder Name", |     "appIdOrName": "App-ID oder Name", | ||||||
|     "appId": "App ID", |     "appId": "App-ID", | ||||||
|     "appWithIdOrNameNotFound": "Es wurde keine App mit dieser ID oder diesem Namen gefunden", |     "appWithIdOrNameNotFound": "Es wurde keine App mit dieser ID oder diesem Namen gefunden", | ||||||
|     "reposHaveMultipleApps": "Repos können mehrere Apps enthalten", |     "reposHaveMultipleApps": "Repos können mehrere Apps enthalten", | ||||||
|     "fdroidThirdPartyRepo": "F-Droid Drittparteienrepo", |     "fdroidThirdPartyRepo": "F-Droid-Drittanbieter-Repo", | ||||||
|     "steamMobile": "Steam Mobile", |     "steamMobile": "Steam Mobile", | ||||||
|     "steamChat": "Steam-Chat", |     "steamChat": "Steam-Chat", | ||||||
|     "install": "Installieren", |     "install": "Installieren", | ||||||
|     "markInstalled": "Als Installiert markieren", |     "markInstalled": "Als installiert markieren", | ||||||
|     "update": "Aktualisieren", |     "update": "Aktualisieren", | ||||||
|     "markUpdated": "Als Aktuell markieren", |     "markUpdated": "Als aktuell markieren", | ||||||
|     "additionalOptions": "Zusätzliche Optionen", |     "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.", |     "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 {}", |     "downloadX": "{} herunterladen", | ||||||
|     "downloadedX": "Heruntergeladen {}", |     "downloadedX": "{} heruntergeladen", | ||||||
|     "releaseAsset": "Asset freigeben", |     "releaseAsset": "Release-Asset", | ||||||
|     "downloadNotifDescription": "Benachrichtigt den Nutzer über den Fortschritt beim Herunterladen einer App", |     "downloadNotifDescription": "Benachrichtigt den Nutzer über den Fortschritt beim Herunterladen einer App", | ||||||
|     "noAPKFound": "Keine APK gefunden", |     "noAPKFound": "Keine APK gefunden", | ||||||
|     "noVersionDetection": "Keine Versionserkennung", |     "noVersionDetection": "Keine Versionserkennung", | ||||||
| @@ -210,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.", |     "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", |     "changes": "Änderungen", | ||||||
|     "releaseDate": "Veröffentlichungsdatum", |     "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", |     "versionDetectionExplanation": "Abgleich der Versionsnummer mit der vom Betriebssystem erkannten Version", | ||||||
|     "versionDetection": "Versionserkennung", |     "versionDetection": "Versionserkennung", | ||||||
|     "standardVersionDetection": "Standardversionserkennung", |     "standardVersionDetection": "Standardversionserkennung", | ||||||
|     "groupByCategory": "Nach Kategorie gruppieren", |     "groupByCategory": "Nach Kategorie gruppieren", | ||||||
|     "autoApkFilterByArch": "Nach Möglichkeit versuchen, APKs nach CPU-Architektur zu filtern", |     "autoApkFilterByArch": "Nach Möglichkeit versuchen, APKs nach CPU-Architektur zu filtern", | ||||||
|     "overrideSource": "Quelle überschreiben", |     "overrideSource": "Quelle überschreiben", | ||||||
|     "dontShowAgain": "Nicht noch einmal zeigen", |     "dontShowAgain": "Nicht noch einmal anzeigen", | ||||||
|     "dontShowTrackOnlyWarnings": "Warnung für 'Nur Nachverfolgen' nicht anzeigen", |     "dontShowTrackOnlyWarnings": "Warnung für 'Nur nachverfolgen' nicht anzeigen", | ||||||
|     "dontShowAPKOriginWarnings": "Warnung für APK-Herkunft nicht anzeigen", |     "dontShowAPKOriginWarnings": "Warnung für APK-Herkunft nicht anzeigen", | ||||||
|     "moveNonInstalledAppsToBottom": "Nicht installierte Apps ans Ende der Apps Ansicht verschieben", |     "moveNonInstalledAppsToBottom": "Nicht installierte Apps ans Ende der App-Ansicht verschieben", | ||||||
|     "gitlabPATLabel": "GitLab Personal Access Token", |     "gitlabPATLabel": "Persönlicher Zugangstoken für GitLab", | ||||||
|     "about": "Über", |     "about": "Über", | ||||||
|     "requiresCredentialsInSettings": "{}: Benötigt zusätzliche Anmeldedaten (in den Einstellungen)", |     "requiresCredentialsInSettings": "{}: Benötigt zusätzliche Anmeldedaten (in den Einstellungen)", | ||||||
|     "checkOnStart": "Überprüfe einmalig beim Start", |     "checkOnStart": "Einmalig beim Start überprüfen", | ||||||
|     "tryInferAppIdFromCode": "Versuche, die App-ID aus dem Quellcode zu ermitteln", |     "tryInferAppIdFromCode": "Versuchen, die App-ID aus dem Quellcode zu ermitteln", | ||||||
|     "removeOnExternalUninstall": "Automatisches Entfernen von extern deinstallierten Apps", |     "removeOnExternalUninstall": "Automatisches Entfernen von extern deinstallierten Apps", | ||||||
|     "pickHighestVersionCode": "Automatische Auswahl des APK mit höchstem Versionscode", |     "pickHighestVersionCode": "Automatische Auswahl des APK mit höchstem Versionscode", | ||||||
|     "checkUpdateOnDetailPage": "Nach Aktualisierungen suchen, wenn eine App-Detailseite geöffnet wird", |     "checkUpdateOnDetailPage": "Nach Aktualisierungen suchen, wenn eine App-Detailseite geöffnet wird", | ||||||
|     "disablePageTransitions": "Animationen für Seitenübergänge deaktivieren", |     "disablePageTransitions": "Animationen für Seitenübergänge deaktivieren", | ||||||
|     "reversePageTransitions": "Umgekehrte Animationen für Seitenübergänge", |     "reversePageTransitions": "Umgekehrte Animationen für Seitenübergänge", | ||||||
|     "minStarCount": "Minimale Anzahl von Sternen", |     "minStarCount": "Minimale Anzahl von Sternen", | ||||||
|     "addInfoBelow": "Fügen Sie diese Informationen unten hinzu.", |     "addInfoBelow": "Diese Information unten hinzufügen.", | ||||||
|     "addInfoInSettings": "Fügen Sie diese Info in den Einstellungen hinzu.", |     "addInfoInSettings": "Diese Information in den Einstellungen hinzufügen.", | ||||||
|     "githubSourceNote": "Die GitHub-Ratenbegrenzung kann mit einem API-Schlüssel umgangen werden.", |     "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", |     "filterReleaseNotesByRegEx": "Versionshinweise nach regulärem Ausdruck filtern", | ||||||
|     "customLinkFilterRegex": "Benutzerdefinierter APK Link Filter nach Regulärem Ausdruck (Standard '.apk$')", |     "customLinkFilterRegex": "Benutzerdefinierter APK-Linkfilter durch regulären Ausdruck (Standard '.apk$')", | ||||||
|     "appsPossiblyUpdated": "App Aktualisierungen wurden versucht", |     "appsPossiblyUpdated": "App-Aktualisierungen wurden versucht", | ||||||
|     "appsPossiblyUpdatedNotifDescription": "Benachrichtigt den Benutzer, dass Aktualisierungen für eine oder mehrere Apps möglicherweise im Hintergrund durchgeführt wurden", |     "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 {}.", |     "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.", |     "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.", |     "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", |     "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", |     "intermediateLinkNotFound": "„Zwischen“-Link nicht gefunden", | ||||||
|     "intermediateLink": "„Zwischen“-Link", |     "intermediateLink": "„Zwischen“-Link", | ||||||
|     "exemptFromBackgroundUpdates": "Ausschluss von Hintergrundaktualisierungen (falls aktiviert)", |     "exemptFromBackgroundUpdates": "Ausschluss von Hintergrundaktualisierungen (falls aktiviert)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Hintergrundaktualisierungen deaktivieren, wenn kein WLAN vorhanden ist", |     "bgUpdatesOnWiFiOnly": "Hintergrundaktualisierungen deaktivieren, wenn kein WLAN vorhanden ist", | ||||||
|     "autoSelectHighestVersionCode": "Automatisch höchste APK-Version auswählen", |     "autoSelectHighestVersionCode": "Automatisch höchste APK-Version auswählen", | ||||||
|     "versionExtractionRegEx": "Versions-Extraktion per RegEx", |     "versionExtractionRegEx": "Versionsextraktion per RegEx", | ||||||
|     "matchGroupToUse": "zu verwendende Gruppe abgleichen", |     "trimVersionString": "Versionszeichenfolge mit RegEx kürzen", | ||||||
|  |     "matchGroupToUseForX": "Zu verwendende Abgleichsgruppe für „{}“", | ||||||
|  |     "matchGroupToUse": "Zu verwendende Gruppe abgleichen", | ||||||
|     "highlightTouchTargets": "Weniger offensichtliche Touch-Ziele hervorheben", |     "highlightTouchTargets": "Weniger offensichtliche Touch-Ziele hervorheben", | ||||||
|     "pickExportDir": "Export-Verzeichnis wählen", |     "pickExportDir": "Export-Verzeichnis wählen", | ||||||
|     "autoExportOnChanges": "Automatischer Export bei Änderung(en)", |     "autoExportOnChanges": "Automatischer Export bei Änderung(en)", | ||||||
| @@ -262,55 +270,58 @@ | |||||||
|     "trySelectingSuggestedVersionCode": "Versuchen, den vorgeschlagenen APK-Versionscode auszuwählen", |     "trySelectingSuggestedVersionCode": "Versuchen, den vorgeschlagenen APK-Versionscode auszuwählen", | ||||||
|     "dontSortReleasesList": "Freigaberelease von der API ordern", |     "dontSortReleasesList": "Freigaberelease von der API ordern", | ||||||
|     "reverseSort": "Umgekehrtes Sortieren", |     "reverseSort": "Umgekehrtes Sortieren", | ||||||
|     "takeFirstLink": "Verwende den ersten Link", |     "takeFirstLink": "Ersten Link verwenden", | ||||||
|     "skipSort": "Überspringe Sortieren", |     "skipSort": "Sortieren überspringen", | ||||||
|     "debugMenu": "Debug-Menü", |     "debugMenu": "Debug-Menü", | ||||||
|     "bgTaskStarted": "Hintergrundaufgabe gestartet – Logs prüfen.", |     "bgTaskStarted": "Hintergrundaufgabe gestartet – Logs prüfen.", | ||||||
|     "runBgCheckNow": "Hintergrundaktualisierungsprüfung jetzt durchführen", |     "runBgCheckNow": "Hintergrundaktualisierungsprüfung jetzt durchführen", | ||||||
|     "versionExtractWholePage": "Versions-Extraktion per RegEx auf die gesamte Seite anwenden", |     "versionExtractWholePage": "Versionsextraktion per RegEx auf die gesamte Seite anwenden", | ||||||
|     "installing": "Installiere", |     "installing": "Installieren", | ||||||
|     "skipUpdateNotifications": "Keine Benachrichtigung zu App-Aktualisierungen geben", |     "skipUpdateNotifications": "Update-Benachrichtigungen überspringen", | ||||||
|     "updatesAvailableNotifChannel": "Aktualisierungen verfügbar", |     "updatesAvailableNotifChannel": "Aktualisierungen verfügbar", | ||||||
|     "appsUpdatedNotifChannel": "Apps aktualisiert", |     "appsUpdatedNotifChannel": "Apps aktualisiert", | ||||||
|     "appsPossiblyUpdatedNotifChannel": "App Aktualisierungen wurden versucht", |     "appsPossiblyUpdatedNotifChannel": "App-Aktualisierungen wurden versucht", | ||||||
|     "errorCheckingUpdatesNotifChannel": "Fehler beim Prüfen auf Aktualisierungen", |     "errorCheckingUpdatesNotifChannel": "Fehler beim Prüfen auf Aktualisierungen", | ||||||
|     "appsRemovedNotifChannel": "Apps entfernt", |     "appsRemovedNotifChannel": "Apps entfernt", | ||||||
|     "downloadingXNotifChannel": "Lade {} herunter", |     "downloadingXNotifChannel": "{} herunterladen", | ||||||
|     "completeAppInstallationNotifChannel": "App Installation abschließen", |     "completeAppInstallationNotifChannel": "App-Installation abschließen", | ||||||
|     "checkingForUpdatesNotifChannel": "Nach Aktualisierungen suchen", |     "checkingForUpdatesNotifChannel": "Nach Aktualisierungen suchen", | ||||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Überprüfe nur installierte und mit „nur Nachverfolgen“ markierte Apps auf Aktualisierungen", |     "onlyCheckInstalledOrTrackOnlyApps": "Nur installierte und mit „Nur nachverfolgen“ markierte Apps auf Aktualisierungen prüfen", | ||||||
|     "supportFixedAPKURL": "neuere Version anhand der ersten dreißig Zahlen der Checksumme der APK URL erraten, wenn anderweitig nicht unterstützt", |     "supportFixedAPKURL": "Unterstützung von festen APK-URLs", | ||||||
|     "selectX": "Wähle {}", |     "selectX": "{} wählen", | ||||||
|     "parallelDownloads": "Erlaube parallele Downloads", |     "parallelDownloads": "Parallele Downloads erlauben", | ||||||
|     "installMethod": "Installationsmethode", |     "useShizuku": "Shizuku oder Sui zur Installation verwenden", | ||||||
|     "normal": "Normal", |     "shizukuBinderNotFound": "Kompatibler Shizuku-Dienst wurde nicht gefunden", | ||||||
|     "root": "Root", |     "shizukuOld": "Alte Shizuku-Version (< 11) - aktualisieren Sie sie", | ||||||
|     "shizukuBinderNotFound": "Kompatibler Shizukudienst wurde nicht gefunden", |     "shizukuOldAndroidWithADB": "Shizuku läuft auf Android < 8.1 mit ADB - aktualisieren Sie Android oder verwenden Sie stattdessen Sui", | ||||||
|     "useSystemFont": "Verwende die Systemschriftart", |     "shizukuPretendToBeGooglePlay": "Google Play als Installationsquelle festlegen (wenn Shizuku verwendet wird)", | ||||||
|     "systemFontError": "Fehler beim Laden der Systemschriftart: {}", |     "useSystemFont": "Systemschriftart verwenden", | ||||||
|     "useVersionCodeAsOSVersion": "Verwende die Appversion als erkannte Version vom Betriebssystem", |     "useVersionCodeAsOSVersion": "App-Version als erkannte Version vom Betriebssystem verwenden", | ||||||
|     "requestHeader": "Request Header", |     "requestHeader": "Kopfzeile anfordern", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Den letzten Asset-Upload als Veröffentlichungsdatum verwenden", |     "useLatestAssetDateAsReleaseDate": "Letzten Asset-Upload als Veröffentlichungsdatum verwenden", | ||||||
|     "defaultPseudoVersioningMethod": "Standardmäßiges Verfahren zur Pseudo-Versionierung", |     "defaultPseudoVersioningMethod": "Standardmäßiges Verfahren zur Pseudo-Versionierung", | ||||||
|     "partialAPKHash": "partieller APK-Hash", |     "partialAPKHash": "Partieller APK-Hash", | ||||||
|     "APKLinkHash": "APK-Link-Hash", |     "APKLinkHash": "APK-Link-Hash", | ||||||
|     "directAPKLink": "Direkter APK-Link", |     "directAPKLink": "Direkter APK-Link", | ||||||
|     "pseudoVersionInUse": "Es werden Pseudoversionen verwendet", |     "pseudoVersionInUse": "Es werden Pseudoversionen verwendet", | ||||||
|     "installed": "Installiert", |     "installed": "Installiert", | ||||||
|     "latest": "Neueste Version", |     "latest": "Neueste Version", | ||||||
|     "invertRegEx": "Regulären Ausdruck  invertieren", |     "invertRegEx": "Regulären Ausdruck invertieren", | ||||||
|     "note": "Hinweis", |     "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)", |     "badDownload": "Die APK konnte nicht geparst werden (inkompatibler oder teilweiser Download)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Neue Apps mit AppVerifier teilen (falls verfügbar)", |     "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.", |     "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-Konfigurationen (Verwendung auf eigene Gefahr)", | ||||||
|  |     "allowInsecure": "Unsichere HTTP-Anfragen zulassen", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "App entfernen?", |         "one": "App entfernen?", | ||||||
|         "other": "Apps entfernen?" |         "other": "Apps entfernen?" | ||||||
|     }, |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Zu viele Anfragen (Rate begrenzt) – versuche es in {} Minute erneut", |         "one": "Zu viele Anfragen (Rate begrenzt) – versuchen Sie es in {} Minute erneut", | ||||||
|         "other": "Zu viele Anfragen (Rate begrenzt) – versuche es in {} Minuten erneut" |         "other": "Zu viele Anfragen (Rate begrenzt) – versuchen Sie es in {} Minuten erneut" | ||||||
|     }, |     }, | ||||||
|     "bgUpdateGotErrorRetryInMinutes": { |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|         "one": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minute geplant", |         "one": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minute geplant", | ||||||
| @@ -321,12 +332,12 @@ | |||||||
|         "other": "Die Hintergrundaktualisierungsprüfung fand {} Aktualisierungen – benachrichtigt den Benutzer, falls erforderlich" |         "other": "Die Hintergrundaktualisierungsprüfung fand {} Aktualisierungen – benachrichtigt den Benutzer, falls erforderlich" | ||||||
|     }, |     }, | ||||||
|     "apps": { |     "apps": { | ||||||
|         "eine": "{} App", |         "one": "{} App", | ||||||
|         "andere": "{} Apps" |         "other": "{} Apps" | ||||||
|     }, |     }, | ||||||
|     "url": { |     "url": { | ||||||
|         "eine": "{} URL", |         "one": "{} URL", | ||||||
|         "andere": "{} URLs" |         "other": "{} URLs" | ||||||
|     }, |     }, | ||||||
|     "minute": { |     "minute": { | ||||||
|         "one": "{} Minute", |         "one": "{} Minute", | ||||||
| @@ -349,12 +360,16 @@ | |||||||
|         "other": "{} und {} weitere Apps haben Aktualisierungen." |         "other": "{} und {} weitere Apps haben Aktualisierungen." | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesInstalled": { |     "xAndNMoreUpdatesInstalled": { | ||||||
|         "one": "{} und 1 weitere Anwendung wurden aktualisiert.", |         "one": "{} und 1 weitere App wurden aktualisiert.", | ||||||
|         "other": "{} und {} weitere Anwendungen wurden aktualisiert." |         "other": "{} und {} weitere Apps wurden aktualisiert." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Die Aktualisierung von {} und 1 weiteren App ist fehlgeschlagen.", | ||||||
|  |         "other": "Die Aktualisierung von {} und {} weiteren Apps ist fehlgeschlagen." | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} und 1 weitere Anwendung wurden möglicherweise aktualisiert.", |         "one": "{} und 1 weitere App wurden möglicherweise aktualisiert.", | ||||||
|         "other": "{} und {} weitere Anwendungen wurden möglicherweise aktualisiert." |         "other": "{} und {} weitere Apps wurden möglicherweise aktualisiert." | ||||||
|     }, |     }, | ||||||
|     "apk": { |     "apk": { | ||||||
|         "one": "{} 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" | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(Required)", |     "requiredInBrackets": "(Required)", | ||||||
|     "dropdownNoOptsError": "ERROR: DROPDOWN MUST HAVE AT LEAST ONE OPT", |     "dropdownNoOptsError": "ERROR: DROPDOWN MUST HAVE AT LEAST ONE OPT", | ||||||
|     "colour": "Colour", |     "colour": "Colour", | ||||||
|  |     "standard": "Standard", | ||||||
|  |     "custom": "Custom", | ||||||
|  |     "useMaterialYou": "Use Material You", | ||||||
|     "githubStarredRepos": "GitHub Starred Repos", |     "githubStarredRepos": "GitHub Starred Repos", | ||||||
|     "uname": "Username", |     "uname": "Username", | ||||||
|     "wrongArgNum": "Wrong number of arguments provided", |     "wrongArgNum": "Wrong number of arguments provided", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Dark", |     "dark": "Dark", | ||||||
|     "light": "Light", |     "light": "Light", | ||||||
|     "followSystem": "Follow System", |     "followSystem": "Follow System", | ||||||
|  |     "followSystemThemeExplanation": "Following system theme is possible only by using third-party applications", | ||||||
|     "useBlackTheme": "Use pure black dark theme", |     "useBlackTheme": "Use pure black dark theme", | ||||||
|     "appSortBy": "App Sort By", |     "appSortBy": "App Sort By", | ||||||
|     "authorName": "Author/Name", |     "authorName": "Author/Name", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "No new updates.", |     "noNewUpdates": "No new updates.", | ||||||
|     "xHasAnUpdate": "{} has an update.", |     "xHasAnUpdate": "{} has an update.", | ||||||
|     "appsUpdated": "Apps Updated", |     "appsUpdated": "Apps Updated", | ||||||
|  |     "appsNotUpdated": "Failed to update applications", | ||||||
|     "appsUpdatedNotifDescription": "Notifies the user that updates to one or more Apps were applied in the background", |     "appsUpdatedNotifDescription": "Notifies the user that updates to one or more Apps were applied in the background", | ||||||
|     "xWasUpdatedToY": "{} was updated to {}.", |     "xWasUpdatedToY": "{} was updated to {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Failed to update {} to {}.", | ||||||
|     "errorCheckingUpdates": "Error Checking for Updates", |     "errorCheckingUpdates": "Error Checking for Updates", | ||||||
|     "errorCheckingUpdatesNotifDescription": "A notification that shows when background update checking fails", |     "errorCheckingUpdatesNotifDescription": "A notification that shows when background update checking fails", | ||||||
|     "appsRemoved": "Apps Removed", |     "appsRemoved": "Apps Removed", | ||||||
| @@ -253,7 +259,9 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "Disable background updates when not on WiFi", |     "bgUpdatesOnWiFiOnly": "Disable background updates when not on WiFi", | ||||||
|     "autoSelectHighestVersionCode": "Auto-select highest versionCode APK", |     "autoSelectHighestVersionCode": "Auto-select highest versionCode APK", | ||||||
|     "versionExtractionRegEx": "Version String Extraction RegEx", |     "versionExtractionRegEx": "Version String Extraction RegEx", | ||||||
|     "matchGroupToUse": "Match Group to Use for 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", |     "highlightTouchTargets": "Highlight less obvious touch targets", | ||||||
|     "pickExportDir": "Pick Export Directory", |     "pickExportDir": "Pick Export Directory", | ||||||
|     "autoExportOnChanges": "Auto-export on changes", |     "autoExportOnChanges": "Auto-export on changes", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Support fixed APK URLs", |     "supportFixedAPKURL": "Support fixed APK URLs", | ||||||
|     "selectX": "Select {}", |     "selectX": "Select {}", | ||||||
|     "parallelDownloads": "Allow parallel downloads", |     "parallelDownloads": "Allow parallel downloads", | ||||||
|     "installMethod": "Installation method", |     "useShizuku": "Use Shizuku or Sui to install", | ||||||
|     "normal": "Normal", |     "shizukuBinderNotFound": "Shizuku service not running", | ||||||
|     "root": "Root", |     "shizukuOld": "Old Shizuku version (<11) - update it", | ||||||
|     "shizukuBinderNotFound": "Сompatible Shizuku service wasn't found", |     "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)", | ||||||
|     "useSystemFont": "Use the system font", |     "useSystemFont": "Use the system font", | ||||||
|     "systemFontError": "Error loading the system font: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Use app versionCode as OS-detected version", |     "useVersionCodeAsOSVersion": "Use app versionCode as OS-detected version", | ||||||
|     "requestHeader": "Request header", |     "requestHeader": "Request header", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Use latest asset upload as release date", |     "useLatestAssetDateAsReleaseDate": "Use latest asset upload as release date", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "The APK could not be parsed (incompatible or partial download)", |     "badDownload": "The APK could not be parsed (incompatible or partial download)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Share new Apps with AppVerifier (if available)", |     "beforeNewInstallsShareToAppVerifier": "Share new Apps with AppVerifier (if available)", | ||||||
|     "appVerifierInstructionToast": "Share to AppVerifier, then return here when ready.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "Remove App?", |         "one": "Remove App?", | ||||||
|         "other": "Remove Apps?" |         "other": "Remove Apps?" | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} and 1 more app was updated.", |         "one": "{} and 1 more app was updated.", | ||||||
|         "other": "{} and {} more apps were updated." |         "other": "{} and {} more apps were updated." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Failed to update {} and 1 more app.", | ||||||
|  |         "other": "Failed to update {} and {} more apps." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} and 1 more app may have been updated.", |         "one": "{} and 1 more app may have been updated.", | ||||||
|         "other": "{} and {} more apps may have been updated." |         "other": "{} and {} more apps may have been updated." | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| { | { | ||||||
|     "invalidURLForSource": "El URL de la aplicación {} no es válido", |     "invalidURLForSource": "El URL de la aplicación {} no es válido", | ||||||
|     "noReleaseFound": "No se ha podido encontrar una versión válida", |     "noReleaseFound": "No se ha encontrado una versión válida", | ||||||
|     "noVersionFound": "No se ha podido determinar la versión", |     "noVersionFound": "No se ha podido determinar la versión", | ||||||
|     "urlMatchesNoSource": "El URL no coincide con ninguna fuente conocida", |     "urlMatchesNoSource": "El URL no coincide con ninguna fuente conocida", | ||||||
|     "cantInstallOlderVersion": "No se puede instalar una versión previa de la aplicación", |     "cantInstallOlderVersion": "No se puede instalar una versión previa de la aplicación", | ||||||
|     "appIdMismatch": "El id. del paquete descargado no coincide con la ID de la aplicación instalada", |     "appIdMismatch": "El ID del paquete descargado no coincide con el ID de la aplicación instalada", | ||||||
|     "functionNotImplemented": "Esta clase no ha implementado esta función", |     "functionNotImplemented": "Esta clase no ha implementado esta función", | ||||||
|     "placeholder": "Espacio reservado", |     "placeholder": "Espacio reservado", | ||||||
|     "someErrors": "Han ocurrido algunos errores", |     "someErrors": "Han ocurrido algunos errores", | ||||||
| @@ -21,15 +21,18 @@ | |||||||
|     "continue": "Continuar", |     "continue": "Continuar", | ||||||
|     "requiredInBrackets": "(Requerido)", |     "requiredInBrackets": "(Requerido)", | ||||||
|     "dropdownNoOptsError": "ERROR: EL DESPLEGABLE DEBE TENER AL MENOS UNA OPCIÓN", |     "dropdownNoOptsError": "ERROR: EL DESPLEGABLE DEBE TENER AL MENOS UNA OPCIÓN", | ||||||
|     "colour": "Color", |     "colour": "color", | ||||||
|     "githubStarredRepos": "Repositorios favoritos en GitHub", |     "standard": "Estándar", | ||||||
|  |     "custom": "A medida", | ||||||
|  |     "useMaterialYou": "Aplicar 'Material You'", | ||||||
|  |     "githubStarredRepos": "repositorios favoritos en GitHub", | ||||||
|     "uname": "Nombre de usuario", |     "uname": "Nombre de usuario", | ||||||
|     "wrongArgNum": "Número de argumentos provistos inválido", |     "wrongArgNum": "Número de argumentos provistos inválido", | ||||||
|     "xIsTrackOnly": "{} es de 'sólo seguimiento'", |     "xIsTrackOnly": "{} es de 'sólo seguimiento'", | ||||||
|     "source": "Origen", |     "source": "origen", | ||||||
|     "app": "Aplicación", |     "app": "Aplicación", | ||||||
|     "appsFromSourceAreTrackOnly": "Las aplicaciones de este origen son de 'solo seguimiento'.", |     "appsFromSourceAreTrackOnly": "Las aplicaciones de este origen son solo para seguimiento.", | ||||||
|     "youPickedTrackOnly": "Debe seleccionar la opción de 'solo seguimiento'.", |     "youPickedTrackOnly": "Debe seleccionar la opción de 'solo para seguimiento'.", | ||||||
|     "trackOnlyAppDescription": "Se hará el seguimiento de actualizaciones para la aplicación, pero Obtainium no será capaz de descargar o actualizarla.", |     "trackOnlyAppDescription": "Se hará el seguimiento de actualizaciones para la aplicación, pero Obtainium no será capaz de descargar o actualizarla.", | ||||||
|     "cancelled": "Cancelado", |     "cancelled": "Cancelado", | ||||||
|     "appAlreadyAdded": "Aplicación añadida anteriormente", |     "appAlreadyAdded": "Aplicación añadida anteriormente", | ||||||
| @@ -42,14 +45,14 @@ | |||||||
|     "search": "Buscar", |     "search": "Buscar", | ||||||
|     "additionalOptsFor": "Opciones adicionales para {}", |     "additionalOptsFor": "Opciones adicionales para {}", | ||||||
|     "supportedSources": "Fuentes admitidas", |     "supportedSources": "Fuentes admitidas", | ||||||
|     "trackOnlyInBrackets": "(Solo seguimiento)", |     "trackOnlyInBrackets": "(Solo para seguimiento)", | ||||||
|     "searchableInBrackets": "(permite búsqueda)", |     "searchableInBrackets": "(permite búsqueda)", | ||||||
|     "appsString": "Aplicaciones", |     "appsString": "Aplicaciones", | ||||||
|     "noApps": "Sin Aplicaciones", |     "noApps": "Sin Aplicaciones", | ||||||
|     "noAppsForFilter": "Sin aplicaciones para filtrar", |     "noAppsForFilter": "Sin aplicaciones para filtrar", | ||||||
|     "byX": "por: {}", |     "byX": "por: {}", | ||||||
|     "percentProgress": "Progreso: {} %", |     "percentProgress": "Progreso: {} %", | ||||||
|     "pleaseWait": "Espere un momento", |     "pleaseWait": "Espere...", | ||||||
|     "updateAvailable": "Actualización disponible", |     "updateAvailable": "Actualización disponible", | ||||||
|     "notInstalled": "No instalado", |     "notInstalled": "No instalado", | ||||||
|     "pseudoVersion": "pseudoversión", |     "pseudoVersion": "pseudoversión", | ||||||
| @@ -60,7 +63,7 @@ | |||||||
|     "removeSelectedApps": "Eliminar aplicaciones seleccionadas", |     "removeSelectedApps": "Eliminar aplicaciones seleccionadas", | ||||||
|     "updateX": "Actualizar {}", |     "updateX": "Actualizar {}", | ||||||
|     "installX": "Instalar {}", |     "installX": "Instalar {}", | ||||||
|     "markXTrackOnlyAsUpdated": "Marcar {}\n(Solo seguimiento)\ncomo actualizada", |     "markXTrackOnlyAsUpdated": "Marcar {}\n(solo para seguimiento)\ncomo actualizada", | ||||||
|     "changeX": "Cambiar {}", |     "changeX": "Cambiar {}", | ||||||
|     "installUpdateApps": "Instalar/actualizar aplicaciones", |     "installUpdateApps": "Instalar/actualizar aplicaciones", | ||||||
|     "installUpdateSelectedApps": "Instalar/actualizar aplicaciones seleccionadas", |     "installUpdateSelectedApps": "Instalar/actualizar aplicaciones seleccionadas", | ||||||
| @@ -97,7 +100,7 @@ | |||||||
|     "appURLList": "Lista de URL de aplicaciones", |     "appURLList": "Lista de URL de aplicaciones", | ||||||
|     "line": "Línea", |     "line": "Línea", | ||||||
|     "searchX": "Buscar {}", |     "searchX": "Buscar {}", | ||||||
|     "noResults": "No se encontró ningún resultado", |     "noResults": "No se ha encontrado ningún resultado", | ||||||
|     "importX": "Importar desde {}", |     "importX": "Importar desde {}", | ||||||
|     "importedAppsIdDisclaimer": "Las aplicaciones importadas podrían mostrarse incorrectamente como «No instalada».\nPara solucionarlo, reinstálelas a través de Obtainium.\nEsto no debería afectar a los datos de las aplicaciones.\n\nSolo afecta a los URL y a los métodos de importación mediante terceros.", |     "importedAppsIdDisclaimer": "Las aplicaciones importadas podrían mostrarse incorrectamente como «No instalada».\nPara solucionarlo, reinstálelas a través de Obtainium.\nEsto no debería afectar a los datos de las aplicaciones.\n\nSolo afecta a los URL y a los métodos de importación mediante terceros.", | ||||||
|     "importErrors": "Errores de Importación", |     "importErrors": "Errores de Importación", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Oscuro", |     "dark": "Oscuro", | ||||||
|     "light": "Claro", |     "light": "Claro", | ||||||
|     "followSystem": "Seguir al sistema", |     "followSystem": "Seguir al sistema", | ||||||
|  |     "followSystemThemeExplanation": "Seguir el tema del sistema sólo es posible utilizando aplicaciones de terceros", | ||||||
|     "useBlackTheme": "Negro puro en tema oscuro", |     "useBlackTheme": "Negro puro en tema oscuro", | ||||||
|     "appSortBy": "Ordenar aplicaciones por", |     "appSortBy": "Ordenar aplicaciones por", | ||||||
|     "authorName": "Autor/nombre", |     "authorName": "Autor/nombre", | ||||||
| @@ -118,14 +122,14 @@ | |||||||
|     "appSortOrder": "Orden de Clasificación", |     "appSortOrder": "Orden de Clasificación", | ||||||
|     "ascending": "Ascendente", |     "ascending": "Ascendente", | ||||||
|     "descending": "Descendente", |     "descending": "Descendente", | ||||||
|     "bgUpdateCheckInterval": "Comprobación actualizaciones en segundo plano", |     "bgUpdateCheckInterval": "Comprobar actualizaciones en segundo plano", | ||||||
|     "neverManualOnly": "Nunca, solo manual", |     "neverManualOnly": "Nunca, solo manual", | ||||||
|     "appearance": "Apariencia", |     "appearance": "Apariencia", | ||||||
|     "showWebInAppView": "Mostrar vista de la web de origen", |     "showWebInAppView": "Mostrar vista de la web de origen", | ||||||
|     "pinUpdates": "Anclar actualizaciones al principio", |     "pinUpdates": "Anclar actualizaciones al principio", | ||||||
|     "updates": "Actualizaciones", |     "updates": "Actualizaciones", | ||||||
|     "sourceSpecific": "Fuente específica", |     "sourceSpecific": "Fuente específica", | ||||||
|     "appSource": "Obtainium en GitHub", |     "appSource": "Filtrar por fuente", | ||||||
|     "noLogs": "Ningún registro", |     "noLogs": "Ningún registro", | ||||||
|     "appLogs": "Registros", |     "appLogs": "Registros", | ||||||
|     "close": "Cerrar", |     "close": "Cerrar", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "No hay nuevas actualizaciones.", |     "noNewUpdates": "No hay nuevas actualizaciones.", | ||||||
|     "xHasAnUpdate": "{} tiene una actualización.", |     "xHasAnUpdate": "{} tiene una actualización.", | ||||||
|     "appsUpdated": "Aplicaciones actualizadas", |     "appsUpdated": "Aplicaciones actualizadas", | ||||||
|  |     "appsNotUpdated": "Error al actualizar las aplicaciones", | ||||||
|     "appsUpdatedNotifDescription": "Notifica al usuario de que una o más aplicaciones han sido actualizadas en segundo plano", |     "appsUpdatedNotifDescription": "Notifica al usuario de que una o más aplicaciones han sido actualizadas en segundo plano", | ||||||
|     "xWasUpdatedToY": "{} ha sido actualizada a {}.", |     "xWasUpdatedToY": "{} ha sido actualizada a {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Error al actualizar {} a {}.", | ||||||
|     "errorCheckingUpdates": "Error al buscar actualizaciones", |     "errorCheckingUpdates": "Error al buscar actualizaciones", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Una notificación que muestra cuándo la comprobación de actualizaciones en segundo plano falla", |     "errorCheckingUpdatesNotifDescription": "Una notificación que muestra cuándo la comprobación de actualizaciones en segundo plano falla", | ||||||
|     "appsRemoved": "Aplicaciones eliminadas", |     "appsRemoved": "Aplicaciones eliminadas", | ||||||
| @@ -170,9 +176,9 @@ | |||||||
|     "fdroid": "Repositorio oficial F-Droid", |     "fdroid": "Repositorio oficial F-Droid", | ||||||
|     "appIdOrName": "ID o Nombre de la Aplicación", |     "appIdOrName": "ID o Nombre de la Aplicación", | ||||||
|     "appId": "ID de la Aplicación", |     "appId": "ID de la Aplicación", | ||||||
|     "appWithIdOrNameNotFound": "No se han encontrado aplicaciones con esa ID o nombre", |     "appWithIdOrNameNotFound": "No se han encontrado aplicaciones con ese ID o nombre", | ||||||
|     "reposHaveMultipleApps": "Los repositorios pueden contener varias aplicaciones", |     "reposHaveMultipleApps": "Los repositorios pueden contener varias aplicaciones", | ||||||
|     "fdroidThirdPartyRepo": "Repositorio de tercera parte F-Droid", |     "fdroidThirdPartyRepo": "Repositorio de terceros F-Droid", | ||||||
|     "steamMobile": "Steam para móviles", |     "steamMobile": "Steam para móviles", | ||||||
|     "steamChat": "Chat de Steam", |     "steamChat": "Chat de Steam", | ||||||
|     "install": "Instalar", |     "install": "Instalar", | ||||||
| @@ -185,9 +191,9 @@ | |||||||
|     "downloadingX": "Descargando {}", |     "downloadingX": "Descargando {}", | ||||||
|     "downloadX": "Descargar {}", |     "downloadX": "Descargar {}", | ||||||
|     "downloadedX": "Descargado {}", |     "downloadedX": "Descargado {}", | ||||||
|     "releaseAsset": "Liberar activos", |     "releaseAsset": "Recurso publicado", | ||||||
|     "downloadNotifDescription": "Notifica al usuario del progreso de descarga de una aplicación", |     "downloadNotifDescription": "Notifica al usuario del progreso de descarga de una aplicación", | ||||||
|     "noAPKFound": "No se encontró el paquete de instalación APK", |     "noAPKFound": "No se ha encontrado el paquete de instalación APK", | ||||||
|     "noVersionDetection": "Sin detección de versiones", |     "noVersionDetection": "Sin detección de versiones", | ||||||
|     "categorize": "Catogorizar", |     "categorize": "Catogorizar", | ||||||
|     "categories": "Categorías", |     "categories": "Categorías", | ||||||
| @@ -199,7 +205,7 @@ | |||||||
|     "addCategory": "Añadir categoría", |     "addCategory": "Añadir categoría", | ||||||
|     "label": "Nombre", |     "label": "Nombre", | ||||||
|     "language": "Idioma", |     "language": "Idioma", | ||||||
|     "copiedToClipboard": "Se copió en el portapapeles", |     "copiedToClipboard": "Copiado en el portapapeles", | ||||||
|     "storagePermissionDenied": "Permiso de almacenamiento rechazado", |     "storagePermissionDenied": "Permiso de almacenamiento rechazado", | ||||||
|     "selectedCategorizeWarning": "Esto reemplazará cualquier ajuste de categoría para las aplicaciones seleccionadas.", |     "selectedCategorizeWarning": "Esto reemplazará cualquier ajuste de categoría para las aplicaciones seleccionadas.", | ||||||
|     "filterAPKsByRegEx": "Filtrar por APK", |     "filterAPKsByRegEx": "Filtrar por APK", | ||||||
| @@ -214,18 +220,18 @@ | |||||||
|     "versionDetectionExplanation": "Conciliar la cadena de versión con la versión detectada desde el sistema operativo", |     "versionDetectionExplanation": "Conciliar la cadena de versión con la versión detectada desde el sistema operativo", | ||||||
|     "versionDetection": "Detección de versiones", |     "versionDetection": "Detección de versiones", | ||||||
|     "standardVersionDetection": "Por versión", |     "standardVersionDetection": "Por versión", | ||||||
|     "groupByCategory": "Agrupar por categoría", |     "groupByCategory": "Agrupar por categorías", | ||||||
|     "autoApkFilterByArch": "Filtrar APK por arquitectura del procesador (si es posible)", |     "autoApkFilterByArch": "Filtrar APK por arquitectura del procesador (si es posible)", | ||||||
|     "overrideSource": "Anular fuente", |     "overrideSource": "Forzar desde la fuente", | ||||||
|     "dontShowAgain": "No mostrar de nuevo", |     "dontShowAgain": "No mostrar de nuevo", | ||||||
|     "dontShowTrackOnlyWarnings": "No mostrar avisos sobre apps en 'solo seguimiento'", |     "dontShowTrackOnlyWarnings": "No mostrar avisos sobre apps 'solo para seguimiento'", | ||||||
|     "dontShowAPKOriginWarnings": "No mostrar avisos sobre las fuentes de las APKs", |     "dontShowAPKOriginWarnings": "No mostrar avisos sobre las fuentes de las APKs", | ||||||
|     "moveNonInstalledAppsToBottom": "Mover Apps no instaladas al final", |     "moveNonInstalledAppsToBottom": "Mover apps no instaladas al final", | ||||||
|     "gitlabPATLabel": "Token de acceso personal a GitLab", |     "gitlabPATLabel": "Token de acceso personal a GitLab", | ||||||
|     "about": "Acerca", |     "about": "Acerca", | ||||||
|     "requiresCredentialsInSettings": "{}: Esto requiere credenciales adicionales (en ajustes)", |     "requiresCredentialsInSettings": "{}: Esto requiere credenciales adicionales (en ajustes)", | ||||||
|     "checkOnStart": "Comprobar actualizaciones al inicio", |     "checkOnStart": "Comprobar actualizaciones al inicio", | ||||||
|     "tryInferAppIdFromCode": "Intentar deducir la ID de la app por el código fuente", |     "tryInferAppIdFromCode": "Intentar deducir el ID de la app por el código fuente", | ||||||
|     "removeOnExternalUninstall": "Auto eliminar apps desinstaladas externamente", |     "removeOnExternalUninstall": "Auto eliminar apps desinstaladas externamente", | ||||||
|     "pickHighestVersionCode": "Auto selección de versión superior del paquete APK", |     "pickHighestVersionCode": "Auto selección de versión superior del paquete APK", | ||||||
|     "checkUpdateOnDetailPage": "Comprobar actualizaciones al abrir detalles de la app", |     "checkUpdateOnDetailPage": "Comprobar actualizaciones al abrir detalles de la app", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "Deshabilitar las actualizaciones en segundo plano sin WiFi", |     "bgUpdatesOnWiFiOnly": "Deshabilitar las actualizaciones en segundo plano sin WiFi", | ||||||
|     "autoSelectHighestVersionCode": "Auto selección del paquete APK con versión más reciente", |     "autoSelectHighestVersionCode": "Auto selección del paquete APK con versión más reciente", | ||||||
|     "versionExtractionRegEx": "Versión de extracción regex", |     "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", |     "matchGroupToUse": "Grupo a usar para versión de extracción regex", | ||||||
|     "highlightTouchTargets": "Resaltar objetivos menos obvios", |     "highlightTouchTargets": "Resaltar objetivos menos obvios", | ||||||
|     "pickExportDir": "Directorio para exportar", |     "pickExportDir": "Directorio para exportar", | ||||||
| @@ -280,30 +288,33 @@ | |||||||
|     "checkingForUpdatesNotifChannel": "Buscando actualizaciones", |     "checkingForUpdatesNotifChannel": "Buscando actualizaciones", | ||||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Comprobar actualizaciones solo para apps instaladas o en seguimiento", |     "onlyCheckInstalledOrTrackOnlyApps": "Comprobar actualizaciones solo para apps instaladas o en seguimiento", | ||||||
|     "supportFixedAPKURL": "Soporte para URLs fijas de APK", |     "supportFixedAPKURL": "Soporte para URLs fijas de APK", | ||||||
|     "selectX": "Selecciona {}", |     "selectX": "Elija {}", | ||||||
|     "parallelDownloads": "Permitir descargas paralelas", |     "parallelDownloads": "Permitir descargas paralelas", | ||||||
|     "installMethod": "Método de instalación", |     "useShizuku": "Usar Shizuku o Sui para instalar", | ||||||
|     "normal": "Normal", |  | ||||||
|     "root": "Raíz", |  | ||||||
|     "shizukuBinderNotFound": "Shizuku no funciona", |     "shizukuBinderNotFound": "Shizuku no funciona", | ||||||
|     "useSystemFont": "Usar la fuente de impresión del sistema", |     "shizukuOld": "Versión antigua de Shizuku (<11) - actualícela", | ||||||
|     "systemFontError": "Error al cargar la fuente de impresión del sistema: {}", |     "shizukuOldAndroidWithADB": "Shizuku corriendo en Android < 8.1 con ADB - actualiza Android o usa Sui en su lugar", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Establecer Google Play como fuente de instalación (si se usa Shizuku)", | ||||||
|  |     "useSystemFont": "Usar fuente del sistema", | ||||||
|     "useVersionCodeAsOSVersion": "Usar la versión de la aplicación como versión detectada por el sistema operativo", |     "useVersionCodeAsOSVersion": "Usar la versión de la aplicación como versión detectada por el sistema operativo", | ||||||
|     "requestHeader": "Encabezado de solicitud", |     "requestHeader": "Encabezado de solicitud", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Usar la última carga de recursos como fecha de lanzamiento", |     "useLatestAssetDateAsReleaseDate": "Usar la última carga del recurso como fecha de lanzamiento", | ||||||
|     "defaultPseudoVersioningMethod": "Método de pseudoversionado predeterminado", |     "defaultPseudoVersioningMethod": "Método de pseudoversionado predeterminado", | ||||||
|     "partialAPKHash": "Hash de APK parcial", |     "partialAPKHash": "Hash de APK parcial", | ||||||
|     "APKLinkHash": "Hash de enlace APK", |     "APKLinkHash": "Hash de enlace APK", | ||||||
|     "directAPKLink": "Enlace APK directo", |     "directAPKLink": "Enlace APK directo", | ||||||
|     "pseudoVersionInUse": "Se está utilizando una pseudoversión", |     "pseudoVersionInUse": "Se está usando una pseudoversión", | ||||||
|     "installed": "Instalado", |     "installed": "Instalado", | ||||||
|     "latest": "Versión más reciente", |     "latest": "Versión más reciente", | ||||||
|     "invertRegEx": "Invertir expresión regular", |     "invertRegEx": "Invertir expresión regular", | ||||||
|     "note": "Nota", |     "note": "Nota", | ||||||
|     "selfHostedNote": "El desplegable «{}» puede utilizarse para acceder a instancias autoalojadas/personalizadas de cualquier fuente.", |     "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)", |     "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.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "¿Eliminar aplicación?", |         "one": "¿Eliminar aplicación?", | ||||||
|         "other": "¿Eliminar aplicaciones?" |         "other": "¿Eliminar aplicaciones?" | ||||||
| @@ -325,8 +336,8 @@ | |||||||
|         "other": "{} Aplicaciones" |         "other": "{} Aplicaciones" | ||||||
|     }, |     }, | ||||||
|     "url": { |     "url": { | ||||||
|         "uno": "{} URL", |         "one": "{} URL", | ||||||
|         "otro": "{} URL" |         "other": "{} URL" | ||||||
|     }, |     }, | ||||||
|     "minute": { |     "minute": { | ||||||
|         "one": "{} minuto", |         "one": "{} minuto", | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} y 1 aplicación más se han actualizado.", |         "one": "{} y 1 aplicación más se han actualizado.", | ||||||
|         "other": "{} y {} aplicaciones más se han actualizado." |         "other": "{} y {} aplicaciones más se han actualizado." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Error al actualizar {} y 1 aplicación más.", | ||||||
|  |         "other": "No se han podido actualizar {} y {} aplicaciones más." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} y 1 aplicación más podría haber sido actualizada.", |         "one": "{} y 1 aplicación más podría haber sido actualizada.", | ||||||
|         "other": "{} y {} aplicaciones más podrían haber sido actualizadas." |         "other": "{} y {} aplicaciones más podrían haber sido actualizadas." | ||||||
|   | |||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(ضروری)", |     "requiredInBrackets": "(ضروری)", | ||||||
|     "dropdownNoOptsError": "خطا: کشویی باید حداقل یک گزینه داشته باشد", |     "dropdownNoOptsError": "خطا: کشویی باید حداقل یک گزینه داشته باشد", | ||||||
|     "colour": "رنگ", |     "colour": "رنگ", | ||||||
|  |     "standard": "Standard", | ||||||
|  |     "custom": "Custom", | ||||||
|  |     "useMaterialYou": "Use Material You", | ||||||
|     "githubStarredRepos": "مخازن ستاره دار گیتهاب", |     "githubStarredRepos": "مخازن ستاره دار گیتهاب", | ||||||
|     "uname": "نام کاربری", |     "uname": "نام کاربری", | ||||||
|     "wrongArgNum": "تعداد آرگومان های ارائه شده اشتباه است", |     "wrongArgNum": "تعداد آرگومان های ارائه شده اشتباه است", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "تاریک", |     "dark": "تاریک", | ||||||
|     "light": "روشن", |     "light": "روشن", | ||||||
|     "followSystem": "هماهنگ با سیستم", |     "followSystem": "هماهنگ با سیستم", | ||||||
|  |     "followSystemThemeExplanation": "دنبال کردن تم سیستم فقط با استفاده از برنامه های شخص ثالث امکان پذیر است", | ||||||
|     "useBlackTheme": "استفاده از تم تیره سیاه خالص", |     "useBlackTheme": "استفاده از تم تیره سیاه خالص", | ||||||
|     "appSortBy": "مرتب سازی برنامه بر اساس", |     "appSortBy": "مرتب سازی برنامه بر اساس", | ||||||
|     "authorName": "سازنده/اسم", |     "authorName": "سازنده/اسم", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "به روز رسانی جدیدی وجود ندارد.", |     "noNewUpdates": "به روز رسانی جدیدی وجود ندارد.", | ||||||
|     "xHasAnUpdate": "{} یک به روز رسانی دارد.", |     "xHasAnUpdate": "{} یک به روز رسانی دارد.", | ||||||
|     "appsUpdated": "برنامه ها به روز شدند", |     "appsUpdated": "برنامه ها به روز شدند", | ||||||
|  |     "appsNotUpdated": "به روز رسانی برنامه ها ناموفق بود", | ||||||
|     "appsUpdatedNotifDescription": "به کاربر اطلاع می دهد که به روز رسانی یک یا چند برنامه در پس زمینه اعمال شده است", |     "appsUpdatedNotifDescription": "به کاربر اطلاع می دهد که به روز رسانی یک یا چند برنامه در پس زمینه اعمال شده است", | ||||||
|     "xWasUpdatedToY": "{} به {} به روز شد.", |     "xWasUpdatedToY": "{} به {} به روز شد.", | ||||||
|  |     "xWasNotUpdatedToY": "به روز رسانی {} به {} انجام نشد.", | ||||||
|     "errorCheckingUpdates": "خطا در بررسی بهروزرسانیها", |     "errorCheckingUpdates": "خطا در بررسی بهروزرسانیها", | ||||||
|     "errorCheckingUpdatesNotifDescription": "اعلانی که وقتی بررسی بهروزرسانی پسزمینه ناموفق است نشان میدهد", |     "errorCheckingUpdatesNotifDescription": "اعلانی که وقتی بررسی بهروزرسانی پسزمینه ناموفق است نشان میدهد", | ||||||
|     "appsRemoved": "برنامه ها حذف شدند", |     "appsRemoved": "برنامه ها حذف شدند", | ||||||
| @@ -183,9 +189,9 @@ | |||||||
|     "disableVersionDetection": "غیرفعال کردن تشخیص نسخه", |     "disableVersionDetection": "غیرفعال کردن تشخیص نسخه", | ||||||
|     "noVersionDetectionExplanation": "این گزینه فقط باید برای برنامه هایی استفاده شود که تشخیص نسخه به درستی کار نمی کند.", |     "noVersionDetectionExplanation": "این گزینه فقط باید برای برنامه هایی استفاده شود که تشخیص نسخه به درستی کار نمی کند.", | ||||||
|     "downloadingX": "در حال دانلود {}", |     "downloadingX": "در حال دانلود {}", | ||||||
|     "downloadX": "Download {}", |     "downloadX": "دانلود {}", | ||||||
|     "downloadedX": "Downloaded {}", |     "downloadedX": "دانلود شده {}", | ||||||
|     "releaseAsset": "Release Asset", |     "releaseAsset": "انتشار دارایی", | ||||||
|     "downloadNotifDescription": "کاربر را از پیشرفت دانلود یک برنامه مطلع می کند", |     "downloadNotifDescription": "کاربر را از پیشرفت دانلود یک برنامه مطلع می کند", | ||||||
|     "noAPKFound": "APK پیدا نشد فایل", |     "noAPKFound": "APK پیدا نشد فایل", | ||||||
|     "noVersionDetection": "بدون تشخیص نسخه", |     "noVersionDetection": "بدون تشخیص نسخه", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "بهروزرسانیهای پسزمینه را در صورت عدم اتصال به WiFi غیرفعال کنید", |     "bgUpdatesOnWiFiOnly": "بهروزرسانیهای پسزمینه را در صورت عدم اتصال به WiFi غیرفعال کنید", | ||||||
|     "autoSelectHighestVersionCode": "انتخاب خودکار بالاترین نسخه کد APK", |     "autoSelectHighestVersionCode": "انتخاب خودکار بالاترین نسخه کد APK", | ||||||
|     "versionExtractionRegEx": "نسخه استخراج RegEx", |     "versionExtractionRegEx": "نسخه استخراج RegEx", | ||||||
|  |     "trimVersionString": "Trim Version String With RegEx", | ||||||
|  |     "matchGroupToUseForX": "Match Group to Use for \"{}\"", | ||||||
|     "matchGroupToUse": "گروه مورد استفاده را مطابقت دهید", |     "matchGroupToUse": "گروه مورد استفاده را مطابقت دهید", | ||||||
|     "highlightTouchTargets": "اهداف لمسی کمتر واضح را برجسته کنید", |     "highlightTouchTargets": "اهداف لمسی کمتر واضح را برجسته کنید", | ||||||
|     "pickExportDir": "فهرست برون ریزی را انتخاب کنید", |     "pickExportDir": "فهرست برون ریزی را انتخاب کنید", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "پشتیبانی از URL های APK ثابت", |     "supportFixedAPKURL": "پشتیبانی از URL های APK ثابت", | ||||||
|     "selectX": "انتخاب کنید {}", |     "selectX": "انتخاب کنید {}", | ||||||
|     "parallelDownloads": "اجازه دانلود موازی", |     "parallelDownloads": "اجازه دانلود موازی", | ||||||
|     "installMethod": "روش نصب", |     "useShizuku": "Use Shizuku or Sui to install", | ||||||
|     "normal": "طبیعی", |  | ||||||
|     "root": "ریشه", |  | ||||||
|     "shizukuBinderNotFound": "Shizuku در حال اجرا نیست", |     "shizukuBinderNotFound": "Shizuku در حال اجرا نیست", | ||||||
|  |     "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)", | ||||||
|     "useSystemFont": "استفاده از فونت سیستم", |     "useSystemFont": "استفاده از فونت سیستم", | ||||||
|     "systemFontError": "خطا در بارگیری فونت سیستم: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "استفاده کد نسخه برنامه به جای نسخه شناسایی شده توسط سیستم عامل استفاده کنید", |     "useVersionCodeAsOSVersion": "استفاده کد نسخه برنامه به جای نسخه شناسایی شده توسط سیستم عامل استفاده کنید", | ||||||
|     "requestHeader": "درخواست سطر بالایی", |     "requestHeader": "درخواست سطر بالایی", | ||||||
|     "useLatestAssetDateAsReleaseDate": "استفاده از آخرین بارگذاری دارایی به عنوان تاریخ انتشار", |     "useLatestAssetDateAsReleaseDate": "استفاده از آخرین بارگذاری دارایی به عنوان تاریخ انتشار", | ||||||
| @@ -299,11 +307,14 @@ | |||||||
|     "installed": "نصب شده است", |     "installed": "نصب شده است", | ||||||
|     "latest": "آخرین", |     "latest": "آخرین", | ||||||
|     "invertRegEx": "معکوس کردن عبارت منظم", |     "invertRegEx": "معکوس کردن عبارت منظم", | ||||||
|     "note": "Note", |     "note": "یادداشت", | ||||||
|     "selfHostedNote": "The \"{}\" dropdown can be used to reach self-hosted/custom instances of any source.", |     "selfHostedNote": "از منوی کرکره ای \"{}\" می توان برای دسترسی به نمونه های خود میزبانی/سفارشی از هر منبعی استفاده کرد.", | ||||||
|     "badDownload": "The APK could not be parsed (incompatible or partial download)", |     "badDownload": "APK قابل تجزیه نیست (دانلود ناسازگار یا جزئی)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Share new Apps with AppVerifier (if available)", |     "beforeNewInstallsShareToAppVerifier": "اشتراکگذاری برنامههای جدید با AppVerifier (در صورت وجود)", | ||||||
|     "appVerifierInstructionToast": "Share to AppVerifier, then return here when ready.", |     "appVerifierInstructionToast": "در AppVerifier به اشتراک بگذارید، سپس پس از آماده شدن به اینجا برگردید.", | ||||||
|  |     "wiki": "راهنما/ویکی", | ||||||
|  |     "crowdsourcedConfigsLabel": "تنظیمات برنامه Crowdsourced (با مسئولیت خود استفاده کنید)", | ||||||
|  |     "allowInsecure": "Allow insecure HTTP requests", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "برنامه حذف شود؟", |         "one": "برنامه حذف شود؟", | ||||||
|         "other": "برنامه ها حذف شوند؟" |         "other": "برنامه ها حذف شوند؟" | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} و 1 برنامه دیگر به روز شدند.", |         "one": "{} و 1 برنامه دیگر به روز شدند.", | ||||||
|         "other": "{} و {} برنامه دیگر به روز شدند." |         "other": "{} و {} برنامه دیگر به روز شدند." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "{} و 1 برنامه دیگر به روز نشد.", | ||||||
|  |         "other": "{} و {} برنامه دیگر به روز نشد." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} و 1 برنامه دیگر ممکن است به روز شده باشند.", |         "one": "{} و 1 برنامه دیگر ممکن است به روز شده باشند.", | ||||||
|         "other": "ممکن است {} و {} برنامه های دیگر به روز شده باشند." |         "other": "ممکن است {} و {} برنامه های دیگر به روز شده باشند." | ||||||
|   | |||||||
| @@ -1,31 +1,34 @@ | |||||||
| { | { | ||||||
|     "invalidURLForSource": "URL d'application {} invalide", |     "invalidURLForSource": "URL de l'application {} invalide", | ||||||
|     "noReleaseFound": "Impossible de trouver une version appropriée", |     "noReleaseFound": "Impossible de trouver une publication correspondante", | ||||||
|     "noVersionFound": "Impossible de déterminer la version de la version", |     "noVersionFound": "Impossible de déterminer la version de la publication", | ||||||
|     "urlMatchesNoSource": "L'URL ne correspond pas à une source connue", |     "urlMatchesNoSource": "L'URL ne correspond pas à une source connue", | ||||||
|     "cantInstallOlderVersion": "Impossible d'installer une ancienne version d'une application", |     "cantInstallOlderVersion": "Impossible d'installer une ancienne version de l'application", | ||||||
|     "appIdMismatch": "L'ID de paquet téléchargé ne correspond pas à l'ID de l'application existante", |     "appIdMismatch": "L'ID du paquet téléchargé ne correspond pas à l'ID de l'application existante", | ||||||
|     "functionNotImplemented": "Cette classe n'a pas implémenté cette fonction", |     "functionNotImplemented": "Cette classe n'implémente pas cette fonction", | ||||||
|     "placeholder": "Espace réservé", |     "placeholder": "Espace réservé", | ||||||
|     "someErrors": "Des erreurs se sont produites", |     "someErrors": "Des erreurs sont survenues", | ||||||
|     "unexpectedError": "Erreur inattendue", |     "unexpectedError": "Erreur inattendue", | ||||||
|     "ok": "D'accord", |     "ok": "Ok", | ||||||
|     "and": "et", |     "and": "et", | ||||||
|     "githubPATLabel": "Jeton d'Accès Personnel GitHub (Augmente la limite de débit)", |     "githubPATLabel": "Jeton d'accès personnel GitHub (augmente la limite de débit)", | ||||||
|     "includePrereleases": "Inclure les avant-premières", |     "includePrereleases": "Inclure les versions préliminaires", | ||||||
|     "fallbackToOlderReleases": "Retour aux anciennes versions", |     "fallbackToOlderReleases": "Retour aux anciennes versions", | ||||||
|     "filterReleaseTitlesByRegEx": "Filtrer les titres de version par expression régulière", |     "filterReleaseTitlesByRegEx": "Filtrer les titres de version par expression régulière", | ||||||
|     "invalidRegEx": "Expression régulière invalide", |     "invalidRegEx": "Expression régulière invalide", | ||||||
|     "noDescription": "Pas de description", |     "noDescription": "Aucune description", | ||||||
|     "cancel": "Annuler", |     "cancel": "Annuler", | ||||||
|     "continue": "Continuer", |     "continue": "Continuer", | ||||||
|     "requiredInBrackets": "(Requis)", |     "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", |     "colour": "Couleur", | ||||||
|  |     "standard": "Standard", | ||||||
|  |     "custom": "Personnalisé", | ||||||
|  |     "useMaterialYou": "Utiliser Material You", | ||||||
|     "githubStarredRepos": "Dépôts étoilés GitHub", |     "githubStarredRepos": "Dépôts étoilés GitHub", | ||||||
|     "uname": "Nom d'utilisateur", |     "uname": "Nom d'utilisateur", | ||||||
|     "wrongArgNum": "Mauvais nombre d'arguments fournis", |     "wrongArgNum": "Nombre incorrect des arguments fournis", | ||||||
|     "xIsTrackOnly": "{} est en 'Suivi uniquement'", |     "xIsTrackOnly": "{} en Suivi uniquement", | ||||||
|     "source": "Source", |     "source": "Source", | ||||||
|     "app": "Application", |     "app": "Application", | ||||||
|     "appsFromSourceAreTrackOnly": "Les applications de cette source sont en 'Suivi uniquement'.", |     "appsFromSourceAreTrackOnly": "Les applications de cette source sont en 'Suivi uniquement'.", | ||||||
| @@ -33,51 +36,51 @@ | |||||||
|     "trackOnlyAppDescription": "L'application sera suivie pour les mises à jour, mais Obtainium ne pourra pas la télécharger ou l'installer.", |     "trackOnlyAppDescription": "L'application sera suivie pour les mises à jour, mais Obtainium ne pourra pas la télécharger ou l'installer.", | ||||||
|     "cancelled": "Annulé", |     "cancelled": "Annulé", | ||||||
|     "appAlreadyAdded": "Application déjà ajoutée", |     "appAlreadyAdded": "Application déjà ajoutée", | ||||||
|     "alreadyUpToDateQuestion": "Application déjà à jour ?", |     "alreadyUpToDateQuestion": "L'application est à jour?", | ||||||
|     "addApp": "Ajouter une application", |     "addApp": "Ajouter Appli", | ||||||
|     "appSourceURL": "URL de la source de l'application", |     "appSourceURL": "URL source de l'application", | ||||||
|     "error": "Erreur", |     "error": "Erreur", | ||||||
|     "add": "Ajouter", |     "add": "Ajouter", | ||||||
|     "searchSomeSourcesLabel": "Rechercher (certaines sources uniquement)", |     "searchSomeSourcesLabel": "Rechercher (certaines sources uniquement)", | ||||||
|     "search": "Rechercher", |     "search": "Rechercher", | ||||||
|     "additionalOptsFor": "Options supplémentaires pour {}", |     "additionalOptsFor": "Options supplémentaires pour {}", | ||||||
|     "supportedSources": "Sources prises en charge ", |     "supportedSources": "Sources prises en charge", | ||||||
|     "trackOnlyInBrackets": "(Suivi uniquement)", |     "trackOnlyInBrackets": "(Suivi uniquement)", | ||||||
|     "searchableInBrackets": "(Recherchable)", |     "searchableInBrackets": "(Interrogeable)", | ||||||
|     "appsString": "Applications", |     "appsString": "Applications", | ||||||
|     "noApps": "Aucune application", |     "noApps": "Aucune application", | ||||||
|     "noAppsForFilter": "Aucune application pour le filtre", |     "noAppsForFilter": "Aucune application à filtrer", | ||||||
|     "byX": "Par {}", |     "byX": "Par {}", | ||||||
|     "percentProgress": "Progrès: {}%", |     "percentProgress": "Progression : {}%", | ||||||
|     "pleaseWait": "Veuillez patienter", |     "pleaseWait": "Veuillez patienter", | ||||||
|     "updateAvailable": "Mise à jour disponible", |     "updateAvailable": "Mise à jour disponible", | ||||||
|     "notInstalled": "Pas installé", |     "notInstalled": "Non installé", | ||||||
|     "pseudoVersion": "pseudo-version", |     "pseudoVersion": "Version fictive", | ||||||
|     "selectAll": "Tout sélectionner", |     "selectAll": "Tout sélectionner", | ||||||
|     "deselectX": "Déselectionner {}", |     "deselectX": "Déselectionner {}", | ||||||
|     "xWillBeRemovedButRemainInstalled": "{} sera supprimé d'Obtainium mais restera installé sur l'appareil.", |     "xWillBeRemovedButRemainInstalled": "{} sera supprimée d'Obtainium mais restera installée sur l'appareil.", | ||||||
|     "removeSelectedAppsQuestion": "Supprimer les applications sélectionnées ?", |     "removeSelectedAppsQuestion": "Supprimer les applications sélectionnées ?", | ||||||
|     "removeSelectedApps": "Supprimer les applications sélectionnées", |     "removeSelectedApps": "Les applications sélectionnées ont été supprimées", | ||||||
|     "updateX": "Mise à jour {}", |     "updateX": "Mettre à jour {}", | ||||||
|     "installX": "Installer {}", |     "installX": "Installer {}", | ||||||
|     "markXTrackOnlyAsUpdated": "Marquer {}\n(Suivi uniquement)\nas mis à jour", |     "markXTrackOnlyAsUpdated": "Marquer {}\n(Suivi uniquement)\ncomme étant à jour", | ||||||
|     "changeX": "Changer {}", |     "changeX": "Changer {}", | ||||||
|     "installUpdateApps": "Installer/Mettre à jour les applications", |     "installUpdateApps": "Installer/Mettre à jour les applications", | ||||||
|     "installUpdateSelectedApps": "Installer/Mettre à jour les applications sélectionnées", |     "installUpdateSelectedApps": "Installer/Mettre à jour les applications sélectionnées", | ||||||
|     "markXSelectedAppsAsUpdated": "Marquer {} les applications sélectionnées comme mises à jour ?", |     "markXSelectedAppsAsUpdated": "Marquer les {} applications sélectionnées comme étant à jour ?", | ||||||
|     "no": "Non", |     "no": "Non", | ||||||
|     "yes": "Oui", |     "yes": "Oui", | ||||||
|     "markSelectedAppsUpdated": "Marquer les applications sélectionnées comme mises à jour", |     "markSelectedAppsUpdated": "Marquer les application sélectionnées comme étant à jour", | ||||||
|     "pinToTop": "Épingler en haut", |     "pinToTop": "Épingler en haut", | ||||||
|     "unpinFromTop": "Détacher du haut", |     "unpinFromTop": "Désépingler du haut", | ||||||
|     "resetInstallStatusForSelectedAppsQuestion": "Réinitialiser l'état d'installation des applications sélectionnées ?", |     "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 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.", |     "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 Obtenirium est installé", |     "customLinkMessage": "Ces liens fonctionnent sur les appareils sur lesquels Obtainium est installé", | ||||||
|     "shareAppConfigLinks": "Partager la configuration de l'application sous forme de lien HTML", |     "shareAppConfigLinks": "Partager la configuration de l'application sous forme de lien HTML", | ||||||
|     "shareSelectedAppURLs": "Partager les URL d'application sélectionnées", |     "shareSelectedAppURLs": "Partager les URL des applications sélectionnées", | ||||||
|     "resetInstallStatus": "Réinitialiser le statut d'installation", |     "resetInstallStatus": "L'état d'installation des applications a été réinitialisé", | ||||||
|     "more": "Plus", |     "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", |     "showOutdatedOnly": "Afficher uniquement les applications obsolètes", | ||||||
|     "filter": "Filtre", |     "filter": "Filtre", | ||||||
|     "filterApps": "Filtrer les applications", |     "filterApps": "Filtrer les applications", | ||||||
| @@ -88,41 +91,42 @@ | |||||||
|     "importExport": "Importer/Exporter", |     "importExport": "Importer/Exporter", | ||||||
|     "settings": "Paramètres", |     "settings": "Paramètres", | ||||||
|     "exportedTo": "Exporté vers {}", |     "exportedTo": "Exporté vers {}", | ||||||
|     "obtainiumExport": "Exportation d'Obtainium", |     "obtainiumExport": "Exporter Obtainium", | ||||||
|     "invalidInput": "Entrée invalide", |     "invalidInput": "Entrée invalide", | ||||||
|     "importedX": "Importé {}", |     "importedX": "Importé {}", | ||||||
|     "obtainiumImport": "Importation d'Obtainium", |     "obtainiumImport": "Importer sur Obtainium", | ||||||
|     "importFromURLList": "Importer à partir de la liste d'URL", |     "importFromURLList": "Importer depuis une liste d'URL", | ||||||
|     "searchQuery": "Requête de recherche", |     "searchQuery": "Requête de recherche", | ||||||
|     "appURLList": "Liste d'URL d'application", |     "appURLList": "Liste d'URL de l'application", | ||||||
|     "line": "Queue", |     "line": "Ligne", | ||||||
|     "searchX": "Rechercher {}", |     "searchX": "Rechercher {}", | ||||||
|     "noResults": "Aucun résultat trouvé", |     "noResults": "Aucun résultat", | ||||||
|     "importX": "Importer {}", |     "importX": "Importation de {}", | ||||||
|     "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.", |     "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 d'importation", |     "importErrors": "Erreurs lors de l'importation", | ||||||
|     "importedXOfYApps": "{} sur {} applications importées.", |     "importedXOfYApps": "{} applications sur {} ont été importés.", | ||||||
|     "followingURLsHadErrors": "Les URL suivantes comportaient des erreurs :", |     "followingURLsHadErrors": "Les URL suivants comportent des erreurs :", | ||||||
|     "selectURL": "Sélectionnez l'URL", |     "selectURL": "Sélectionner l'URL", | ||||||
|     "selectURLs": "Sélectionnez les URLs", |     "selectURLs": "Sélectionner les URL", | ||||||
|     "pick": "Prendre", |     "pick": "Choisir", | ||||||
|     "theme": "Thème", |     "theme": "Thème", | ||||||
|     "dark": "Sombre", |     "dark": "Sombre", | ||||||
|     "light": "Clair", |     "light": "Clair", | ||||||
|     "followSystem": "Suivre le système", |     "followSystem": "Correspondre au système", | ||||||
|     "useBlackTheme": "Utilisez le thème noir pur et sombre", |     "followSystemThemeExplanation": "Correspondre au thème du système est possible en utilisant des applications tierces.", | ||||||
|     "appSortBy": "Applications triées par", |     "useBlackTheme": "Utiliser un thème Noir", | ||||||
|  |     "appSortBy": "Trier les applications par", | ||||||
|     "authorName": "Auteur/Nom", |     "authorName": "Auteur/Nom", | ||||||
|     "nameAuthor": "Nom/Auteur", |     "nameAuthor": "Nom/Auteur", | ||||||
|     "asAdded": "Comme ajouté", |     "asAdded": "Date d'ajout", | ||||||
|     "appSortOrder": "Ordre de tri des applications", |     "appSortOrder": "Ordre de tri des applications", | ||||||
|     "ascending": "Ascendant", |     "ascending": "Ascendant", | ||||||
|     "descending": "Descendanr", |     "descending": "Descendant", | ||||||
|     "bgUpdateCheckInterval": "Intervalle de vérification des mises à jour en arrière-plan", |     "bgUpdateCheckInterval": "Intervalle de recherche de mises à jour en arrière-plan", | ||||||
|     "neverManualOnly": "Jamais - Manuel uniquement", |     "neverManualOnly": "Jamais - Manuellement uniquement", | ||||||
|     "appearance": "Apparence", |     "appearance": "Apparence", | ||||||
|     "showWebInAppView": "Afficher la page Web source dans la vue de l'application", |     "showWebInAppView": "Afficher la page Web source dans l'onglet 'Applications'", | ||||||
|     "pinUpdates": "Épingler les mises à jour dans la vue Top des applications", |     "pinUpdates": "Épingler les mises à jour en haut de l'onglet 'Applications'", | ||||||
|     "updates": "Mises à jour", |     "updates": "Mises à jour", | ||||||
|     "sourceSpecific": "Spécifique à la source", |     "sourceSpecific": "Spécifique à la source", | ||||||
|     "appSource": "Source de l'application", |     "appSource": "Source de l'application", | ||||||
| @@ -131,30 +135,32 @@ | |||||||
|     "close": "Fermer", |     "close": "Fermer", | ||||||
|     "share": "Partager", |     "share": "Partager", | ||||||
|     "appNotFound": "Application introuvable", |     "appNotFound": "Application introuvable", | ||||||
|     "obtainiumExportHyphenatedLowercase": "exportation d'obtainium", |     "obtainiumExportHyphenatedLowercase": "export-obtainium", | ||||||
|     "pickAnAPK": "Choisissez un APK", |     "pickAnAPK": "Selectionner un APK", | ||||||
|     "appHasMoreThanOnePackage": "{} a plus d'un paquet :", |     "appHasMoreThanOnePackage": "{} a plus d'un paquet:", | ||||||
|     "deviceSupportsXArch": "Votre appareil prend en charge l'architecture de processeur {}.", |     "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", |     "warning": "Avertissement", | ||||||
|     "sourceIsXButPackageFromYPrompt": "La source de l'application est '{}' mais le paquet de version provient de '{}'. Continuer?", |     "sourceIsXButPackageFromYPrompt": "La source de l'application est '{}' mais le paquet de mise à jour provient de '{}'. Continuer ?", | ||||||
|     "updatesAvailable": "Mises à jour disponibles", |     "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.", |     "noNewUpdates": "Aucune nouvelle mise à jour.", | ||||||
|     "xHasAnUpdate": "{} a une mise à jour.", |     "xHasAnUpdate": "{} a une mise à jour.", | ||||||
|     "appsUpdated": "Applications mises à jour", |     "appsUpdated": "Applications mises à jour", | ||||||
|     "appsUpdatedNotifDescription": "Avertit l'utilisateur que les mises à jour d'une ou plusieurs applications ont été appliquées en arrière-plan", |     "appsNotUpdated": "Échec de la mise à jour des applications", | ||||||
|     "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.", | ||||||
|     "errorCheckingUpdates": "Erreur lors de la vérification des mises à jour", |     "xWasUpdatedToY": "{} a été mis à jour en {}.", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Une notification qui s'affiche lorsque la vérification de la mise à jour en arrière-plan échoue", |     "xWasNotUpdatedToY": "Échec de la mise à jour de {} vers {}.", | ||||||
|  |     "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", |     "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", |     "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é en raison de cette erreur : {}", |     "xWasRemovedDueToErrorY": "{} a été supprimée en raison de cette erreur : {}", | ||||||
|     "completeAppInstallation": "Installation complète de l'application", |     "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", |     "completeAppInstallationNotifDescription": "Demande à l'utilisateur de retourner sur Obtainium pour terminer l'installation d'une application", | ||||||
|     "checkingForUpdates": "Vérification des mises à jour", |     "checkingForUpdates": "Recherche de mises à jour", | ||||||
|     "checkingForUpdatesNotifDescription": "Notification transitoire qui apparaît lors de la recherche de mises à jour", |     "checkingForUpdatesNotifDescription": "Notification temporaire qui apparaît lors de la recherche de mises à jour", | ||||||
|     "pleaseAllowInstallPerm": "Veuillez autoriser Obtainium à installer des applications", |     "pleaseAllowInstallPerm": "Veuillez autoriser Obtainium à installer des applications", | ||||||
|     "trackOnly": "Suivi uniquement", |     "trackOnly": "Suivi uniquement", | ||||||
|     "errorWithHttpStatusCode": "Erreur {}", |     "errorWithHttpStatusCode": "Erreur {}", | ||||||
| @@ -162,163 +168,168 @@ | |||||||
|     "unknown": "Inconnu", |     "unknown": "Inconnu", | ||||||
|     "none": "Aucun", |     "none": "Aucun", | ||||||
|     "never": "Jamais", |     "never": "Jamais", | ||||||
|     "latestVersionX": "Dernière version: {}", |     "latestVersionX": "Dernière version : {}", | ||||||
|     "installedVersionX": "Version installée : {}", |     "installedVersionX": "Version installée : {}", | ||||||
|     "lastUpdateCheckX": "Vérification de la dernière mise à jour : {}", |     "lastUpdateCheckX": "Dernière recherche de mises à jour : {}", | ||||||
|     "remove": "Retirer", |     "remove": "Supprimer", | ||||||
|     "yesMarkUpdated": "Oui, marquer comme mis à jour", |     "yesMarkUpdated": "Oui, marquer comme étant à jour", | ||||||
|     "fdroid": "F-Droid Officiel", |     "fdroid": "F-Droid Officiel", | ||||||
|     "appIdOrName": "ID ou nom de l'application", |     "appIdOrName": "ID ou nom de l'application", | ||||||
|     "appId": "ID de l'application", |     "appId": "ID de l'application", | ||||||
|     "appWithIdOrNameNotFound": "Aucune application n'a été trouvée avec cet identifiant ou ce nom", |     "appWithIdOrNameNotFound": "Aucune application n'a été trouvée avec cet identifiant ou ce nom", | ||||||
|     "reposHaveMultipleApps": "Les dépôts peuvent contenir plusieurs applications", |     "reposHaveMultipleApps": "Les dépôts peuvent contenir plusieurs applications", | ||||||
|     "fdroidThirdPartyRepo": "Dépôt tiers F-Droid", |     "fdroidThirdPartyRepo": "Dépôt tiers F-Droid", | ||||||
|     "steamMobile": "Vapeur Mobile", |     "steamMobile": "Application mobile Steam", | ||||||
|     "steamChat": "Chat sur Steam", |     "steamChat": "Steam Chat", | ||||||
|     "install": "Installer", |     "install": "Installer", | ||||||
|     "markInstalled": "Marquer installée", |     "markInstalled": "Marquer comme étant installé", | ||||||
|     "update": "Mettre à jour", |     "update": "Mettre à jour", | ||||||
|     "markUpdated": "Marquer à jour", |     "markUpdated": "Marquer comme étant à jour", | ||||||
|     "additionalOptions": "Options additionelles", |     "additionalOptions": "Options supplémentaires", | ||||||
|     "disableVersionDetection": "Désactiver la détection de version", |     "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 version ne fonctionne pas correctement.", |     "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 {}", |     "downloadingX": "Téléchargement {}", | ||||||
|     "downloadX": "Télécharger {}", |     "downloadX": "Télécharger {}", | ||||||
|     "downloadedX": "Téléchargé {}", |     "downloadedX": "Téléchargé {}", | ||||||
|     "releaseAsset": "Actif libéré", |     "releaseAsset": "Version active", | ||||||
|     "downloadNotifDescription": "Avertit l'utilisateur de la progression du téléchargement d'une application", |     "downloadNotifDescription": "Notifie l'utilisateur sur l'avancement du téléchargement d'une application", | ||||||
|     "noAPKFound": "Aucun APK trouvé", |     "noAPKFound": "Aucun APK trouvé", | ||||||
|     "noVersionDetection": "Pas de détection de version", |     "noVersionDetection": "Aucune version trouvée", | ||||||
|     "categorize": "Catégoriser", |     "categorize": "Catégoriser", | ||||||
|     "categories": "Catégories", |     "categories": "Catégories", | ||||||
|     "category": "Catégorie", |     "category": "Catégorie", | ||||||
|     "noCategory": "Aucune catégorie", |     "noCategory": "Aucune catégorie", | ||||||
|     "noCategories": "Aucune catégorie", |     "noCategories": "Aucune catégories", | ||||||
|     "deleteCategoriesQuestion": "Supprimer les catégories ?", |     "deleteCategoriesQuestion": "Supprimer les catégories?", | ||||||
|     "categoryDeleteWarning": "Toutes les applications dans les catégories supprimées seront définies sur non catégorisées.", |     "categoryDeleteWarning": "Toutes les applications des catégories supprimées seront définies comme non catégorisées .", | ||||||
|     "addCategory": "Ajouter une catégorie", |     "addCategory": "Ajouter une catégorie", | ||||||
|     "label": "Étiquette", |     "label": "Nom", | ||||||
|     "language": "Langue", |     "language": "Langue", | ||||||
|     "copiedToClipboard": "Copié dans le presse-papier", |     "copiedToClipboard": "Copié dans le presse-papier", | ||||||
|     "storagePermissionDenied": "Autorisation de stockage refusée", |     "storagePermissionDenied": "Permission de stockage refusée", | ||||||
|     "selectedCategorizeWarning": "Cela remplacera tous les paramètres de catégorie existants pour les applications sélectionnées.", |     "selectedCategorizeWarning": "Cela va remplacer toutes les catégories définies des applications sélectionnées.", | ||||||
|     "filterAPKsByRegEx": "Filtrer les APK par expression régulière", |     "filterAPKsByRegEx": "Filtrer les APK par expression régulière", | ||||||
|     "removeFromObtainium": "Supprimer d'Obtainium", |     "removeFromObtainium": "Supprimer d'Obtainium", | ||||||
|     "uninstallFromDevice": "Désinstaller de l'appareil", |     "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", |     "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 une date de sortie est disponible.", |     "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": "Changements", |     "changes": "Modifications", | ||||||
|     "releaseDate": "Date de sortie", |     "releaseDate": "Date de sortie", | ||||||
|     "importFromURLsInFile": "Importer à partir d'URL dans un fichier (comme OPML)", |     "importFromURLsInFile": "Importer à partir des URLs d'un fichier (comme OPML)", | ||||||
|     "versionDetectionExplanation": "Réconcilier la chaîne de version avec la version détectée à partir du système d'exploitation", |     "versionDetectionExplanation": "Reporter la chaîne de version selon la version détectée par le système d'exploitation", | ||||||
|     "versionDetection": "Détection des versions", |     "versionDetection": "Détection de la version", | ||||||
|     "standardVersionDetection": "Détection de version standard", |     "standardVersionDetection": "Détection de la version standard", | ||||||
|     "groupByCategory": "Regrouper par catégorie", |     "groupByCategory": "Grouper par catégorie", | ||||||
|     "autoApkFilterByArch": "Essayez de filtrer les APK par architecture CPU si possible", |     "autoApkFilterByArch": "Essayer de filtrer les APKs par architecture CPU si possible", | ||||||
|     "overrideSource": "Remplacer la source", |     "overrideSource": "Remplacer la source", | ||||||
|     "dontShowAgain": "Ne montre plus ça", |     "dontShowAgain": "Ne plus afficher", | ||||||
|     "dontShowTrackOnlyWarnings": "Don't Show the 'Track-Only' Warning", |     "dontShowTrackOnlyWarnings": "Ne plus afficher les erreurs 'Suivi uniquement'", | ||||||
|     "dontShowAPKOriginWarnings": "Ne pas afficher les avertissements sur l'origine de l'APK", |     "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", |     "moveNonInstalledAppsToBottom": "Déplacer les applications non installées vers le bas de la vue Applications", | ||||||
|     "gitlabPATLabel": "Jeton d'accès personnel GitLab", |     "gitlabPATLabel": "Jeton d'accès personnel GitLab", | ||||||
|     "about": "À propos de", |     "about": "À propos", | ||||||
|     "requiresCredentialsInSettings": "{}: This needs additional credentials (in Settings)", |     "requiresCredentialsInSettings": "{} a besoin d'un complément d'information (dans les Paramètres)", | ||||||
|     "checkOnStart": "Vérifier les mises à jour au démarrage", |     "checkOnStart": "Rechercher les mises à jour au démarrage", | ||||||
|     "tryInferAppIdFromCode": "Essayez de déduire l'ID de l'application à partir du code source", |     "tryInferAppIdFromCode": "Essayer de déduire l'identifiant de l'application à partir du code source", | ||||||
|     "removeOnExternalUninstall": "Supprimer automatiquement les applications désinstallées en externe", |     "removeOnExternalUninstall": "Supprimer automatiquement les applications désinstallées en externe", | ||||||
|     "pickHighestVersionCode": "Sélectionner automatiquement le code APK de la version la plus élevée", |     "pickHighestVersionCode": "Sélectionner automatiquement la version la plus récente du code APK", | ||||||
|     "checkUpdateOnDetailPage": "Vérifier les mises à jour lors de l'ouverture d'une page de détails d'application", |     "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", |     "disablePageTransitions": "Désactiver les animations de transition de page", | ||||||
|     "reversePageTransitions": "Animations de transition de page inversée", |     "reversePageTransitions": "Inverser les animations de transition de page", | ||||||
|     "minStarCount": "Nombre minimum d'étoiles", |     "minStarCount": "Nombre minimum d'étoiles", | ||||||
|     "addInfoBelow": "Ajoutez ces informations ci-dessous.", |     "addInfoBelow": "Ajoutez cette information ci-dessous.", | ||||||
|     "addInfoInSettings": "Ajoutez ces informations dans les paramètres.", |     "addInfoInSettings": "Ajoutez cette information dans les paramètres.", | ||||||
|     "githubSourceNote": "La limitation du débit GitHub peut être évitée à l'aide d'une clé API.", |     "githubSourceNote": "La limitation du débit de GitHub peut être évitée à l'aide d'une clé d'API.", | ||||||
|     "sortByLastLinkSegment": "Trier uniquement sur le dernier segment du lien", |     "sortByLastLinkSegment": "Trier par le dernier segment du lien", | ||||||
|     "filterReleaseNotesByRegEx": "Filtrer les notes de version par expression régulière", |     "filterReleaseNotesByRegEx": "Filtrer les notes de version par expression régulière", | ||||||
|     "customLinkFilterRegex": "Filtre de lien APK personnalisé par expression régulière (par défaut '.apk$')", |     "customLinkFilterRegex": "Filtre de lien APK personnalisé par expression régulière (par défaut '.apk$')", | ||||||
|     "appsPossiblyUpdated": "Tentative de mise à jour de l'application", |     "appsPossiblyUpdated": "Tentative de mise à jour des applications", | ||||||
|     "appsPossiblyUpdatedNotifDescription": "Avertit l'utilisateur que des mises à jour d'une ou plusieurs applications ont été potentiellement appliquées en arrière-plan", |     "appsPossiblyUpdatedNotifDescription": "Notifie à l'utilisateur que des mises à jour d'une ou plusieurs applications ont potentiellement été appliquées en arrière-plan", | ||||||
|     "xWasPossiblyUpdatedToY": "{} a peut-être été mis à jour vers {}.", |     "xWasPossiblyUpdatedToY": "{} peut être mis à jour en {}.", | ||||||
|     "enableBackgroundUpdates": "Activer les mises à jour en arrière-plan", |     "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.", |     "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'Obetium.", |     "backgroundUpdateLimitsExplanation": "Le résultat d'une installation en arrière-plan ne peut être déterminé qu'à l'ouverture d'Obtainium.", | ||||||
|     "verifyLatestTag": "Vérifiez la balise 'dernière'", |     "verifyLatestTag": "Vérifier la balise 'latest'", | ||||||
|     "intermediateLinkRegex": " Filtrer un lien \" intermédiaire \" à visiter ", |     "intermediateLinkRegex": "Filtrer un lien 'intermédiaire' à visiter", | ||||||
|     "filterByLinkText": "Filtrer les liens par texte de lien", |     "filterByLinkText": "Filtrer les liens par texte du lien", | ||||||
|     "intermediateLinkNotFound": "Lien intermédiaire introuvable", |     "intermediateLinkNotFound": "Lien intermédiaire introuvable", | ||||||
|     "intermediateLink": "Lien intermédiaire", |     "intermediateLink": "Lien intermédiaire", | ||||||
|     "exemptFromBackgroundUpdates": "Exempt des mises à jour en arrière-plan (si activé)", |     "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 connecté au WiFi", |     "bgUpdatesOnWiFiOnly": "Désactiver les mises à jour en arrière-plan lorsque vous n'êtes pas en WiFi", | ||||||
|     "autoSelectHighestVersionCode": "Sélection automatique de la version la plus élevéeCode APK", |     "autoSelectHighestVersionCode": "Sélectionner automatiquement la version la plus récente du code APK", | ||||||
|     "versionExtractionRegEx": "Version Extraction RegEx", |     "versionExtractionRegEx": "Extraire la version par Expression régulière", | ||||||
|     "matchGroupToUse": "Match Group to Use", |     "trimVersionString": "Découper la version par Expression régulière", | ||||||
|     "highlightTouchTargets": "Mettez en évidence les cibles tactiles moins évidentes", |     "matchGroupToUseForX": "Groupe de correspondance à utiliser pour \"{}\"", | ||||||
|     "pickExportDir": "Choisir le répertoire d'exportation", |     "matchGroupToUse": "Groupe de correspondance à utiliser pour l'extraction de la version par Expression régulière", | ||||||
|     "autoExportOnChanges": "Exportation automatique sur modifications", |     "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", |     "includeSettings": "Inclure les paramètres", | ||||||
|     "filterVersionsByRegEx": "Filtrer les versions par expression régulière", |     "filterVersionsByRegEx": "Filtrer les versions par expression régulière", | ||||||
|     "trySelectingSuggestedVersionCode": "Essayez de sélectionner la version suggéréeCode APK", |     "trySelectingSuggestedVersionCode": "Essayer de sélectionner la version suggérée du code APK", | ||||||
|     "dontSortReleasesList": "Conserver la commande de version de l'API", |     "dontSortReleasesList": "Conserver l'ordre de la version de l'API", | ||||||
|     "reverseSort": "Tri inversé", |     "reverseSort": "Tri inversé", | ||||||
|     "takeFirstLink": "Prendre le premier lien", |     "takeFirstLink": "Utiliser le premier lien", | ||||||
|     "skipSort": "Sauter le tri", |     "skipSort": "Ignorer le tri", | ||||||
|     "debugMenu": "Menu de débogage", |     "debugMenu": "Menu de déboggage", | ||||||
|     "bgTaskStarted": "Tâche en arrière-plan démarrée - vérifier les journaux.", |     "bgTaskStarted": "Tâche en arrière-plan démarrée - vérifier les journaux.", | ||||||
|     "runBgCheckNow": "Exécuter la vérification de la mise à jour en arrière-plan maintenant", |     "runBgCheckNow": "Exécuter la recherche de mise à jour en arrière-plan maintenant", | ||||||
|     "versionExtractWholePage": "Apply Version Extraction Regex to Entire Page", |     "versionExtractWholePage": "Appliquer l'extraction de la version par expression régulière à l'ensemble de la page", | ||||||
|     "installing": "Installation", |     "installing": "Installation", | ||||||
|     "skipUpdateNotifications": "Ignorer les notifications de mise à jour", |     "skipUpdateNotifications": "Ignorer les notifications de mise à jour", | ||||||
|     "updatesAvailableNotifChannel": "Mises à jour disponibles", |     "updatesAvailableNotifChannel": "Mises à jour disponibles", | ||||||
|     "appsUpdatedNotifChannel": "Applications mises à jour", |     "appsUpdatedNotifChannel": "Applications mises à jour", | ||||||
|     "appsPossiblyUpdatedNotifChannel": "Tentative de mise à jour de l'application", |     "appsPossiblyUpdatedNotifChannel": "Essayer de mettre à jour les applications", | ||||||
|     "errorCheckingUpdatesNotifChannel": "Erreur lors de la vérification des mises à jour", |     "errorCheckingUpdatesNotifChannel": "Erreur lors de la recherche de mises à jour", | ||||||
|     "appsRemovedNotifChannel": "Applications supprimées", |     "appsRemovedNotifChannel": "Applications supprimées", | ||||||
|     "downloadingXNotifChannel": "Téléchargement {}", |     "downloadingXNotifChannel": "Téléchargement {}", | ||||||
|     "completeAppInstallationNotifChannel": "Installation complète de l'application", |     "completeAppInstallationNotifChannel": "Installation complète de l'application", | ||||||
|     "checkingForUpdatesNotifChannel": "Vérification des mises à jour", |     "checkingForUpdatesNotifChannel": "Recherche de mises à jour", | ||||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Vérifiez uniquement les mises à jour des applications installées et de suivi uniquement", |     "onlyCheckInstalledOrTrackOnlyApps": "Rechercher uniquement les mises à jour des applications installées et des applications 'Suivi uniquement'", | ||||||
|     "supportFixedAPKURL": "Prise en charge des URL APK fixes", |     "supportFixedAPKURL": "Prise en charge des URL APK fixes", | ||||||
|     "selectX": "Sélectionner {}", |     "selectX": "Sélectionner {}", | ||||||
|     "parallelDownloads": "Autoriser les téléchargements parallèles", |     "parallelDownloads": "Autoriser les téléchargements simultanés", | ||||||
|     "installMethod": "Méthode d'installation", |     "useShizuku": "Utiliser Shizuku ou Sui pour l'installation", | ||||||
|     "normal": "Normale", |     "shizukuBinderNotFound": "Le service Shizuku est introuvable", | ||||||
|     "root": "Racine", |     "shizukuOld": "Ancienne version de Shizuku (<11) - veuillez le mettre à jour", | ||||||
|     "shizukuBinderNotFound": "Shizuku is not running", |     "shizukuOldAndroidWithADB": "Shizuku fonctionne sur Android < 8.1 avec ADB - veuillez mettre à jour Android ou utiliser Sui à la place", | ||||||
|     "useSystemFont": "Utiliser la police système", |     "shizukuPretendToBeGooglePlay": "Définir Google Play comme source d'installation (si Shizuku est utilisé)", | ||||||
|     "systemFontError": "Erreur de chargement de la police système : {}", |     "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", |     "useVersionCodeAsOSVersion": "Utiliser le code de version de l'application détectée par le système d'exploitation", | ||||||
|     "requestHeader": "En-tête de demande", |     "requestHeader": "Intitulé de la demande", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Utiliser le dernier téléchargement d'élément comme date de sortie", |     "useLatestAssetDateAsReleaseDate": "Utiliser le dernier élément mis en ligne comme date de sortie", | ||||||
|     "defaultPseudoVersioningMethod": "Méthode de pseudo-version par défaut", |     "defaultPseudoVersioningMethod": "Méthode de version fictive par défaut", | ||||||
|     "partialAPKHash": "Hash APK partiel", |     "partialAPKHash": "Hash partiel de l'APK", | ||||||
|     "APKLinkHash": "Hash de lien APK", |     "APKLinkHash": "Hash du lien APK", | ||||||
|     "directAPKLink": "Lien APK direct", |     "directAPKLink": "Lien direct de l'APK", | ||||||
|     "pseudoVersionInUse": "Une pseudo-version est utilisée", |     "pseudoVersionInUse": "Version fictive utilisé", | ||||||
|     "installed": "Installée", |     "installed": "Installée", | ||||||
|     "latest": "Dernier", |     "latest": "Dernière version", | ||||||
|     "invertRegEx": "Inverser l'expression régulière", |     "invertRegEx": "Inverser l'expression régulière", | ||||||
|     "note": "Note", |     "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)", |     "badDownload": "L'APK n'a pas pu être analysé (téléchargement incompatible ou partiel)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Partager les nouvelles applications avec AppVerifier (si disponible)", |     "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 communauté (à utiliser à vos risques et périls)", | ||||||
|  |     "allowInsecure": "Autoriser les requêtes HTTP non sécurisées", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Supprimer l'application ?", |         "one": "Supprimer l'application ?", | ||||||
|         "other": "Supprimer les applications ?" |         "other": "Supprimer les applications ?" | ||||||
|     }, |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Trop de demandes (taux limité) - réessayez dans {} minute", |         "one": "Trop de requêtes (taux limité) - réessayez dans {} minute", | ||||||
|         "other": "Trop de demandes (taux limité) - réessayez dans {} minutes" |         "other": "Trop de requêtes (taux limité) - réessayez dans {} minutes" | ||||||
|     }, |     }, | ||||||
|     "bgUpdateGotErrorRetryInMinutes": { |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|         "one": "La vérification de la mise à jour en arrière-plan a rencontré un {}, planifiera une nouvelle tentative de vérification dans {} minute", |         "one": "La recherche de mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative programmée dans {} minute", | ||||||
|         "other": "La vérification de la mise à jour en arrière-plan a rencontré un {}, planifiera une nouvelle tentative de vérification dans {} minutes" |         "other": "La recherche de mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative programmée dans {} minutes" | ||||||
|     }, |     }, | ||||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|         "one": "La vérification des mises à jour en arrière-plan trouvée {} mise à jour - avertira l'utilisateur 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 vérification des mises à jour en arrière-plan a trouvé {} mises à jour - avertira l'utilisateur 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": { |     "apps": { | ||||||
|         "one": "{} Application", |         "one": "{} Application", | ||||||
| @@ -329,7 +340,7 @@ | |||||||
|         "other": "{} URL" |         "other": "{} URL" | ||||||
|     }, |     }, | ||||||
|     "minute": { |     "minute": { | ||||||
|         "one": "{} Minutes", |         "one": "{} Minute", | ||||||
|         "other": "{} Minutes" |         "other": "{} Minutes" | ||||||
|     }, |     }, | ||||||
|     "hour": { |     "hour": { | ||||||
| @@ -341,8 +352,8 @@ | |||||||
|         "other": "{} Jours" |         "other": "{} Jours" | ||||||
|     }, |     }, | ||||||
|     "clearedNLogsBeforeXAfterY": { |     "clearedNLogsBeforeXAfterY": { | ||||||
|         "one": "{n} journal effacé (avant = {before}, après = {after})", |         "one": "Nettoyage du journal {n} (avant = {before}, après = {after})", | ||||||
|         "other": "{n} journaux effacés (avant = {before}, après = {after})" |         "other": "Nettoyage des journaux {n} (avant = {before}, après = {after})" | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesAvailable": { |     "xAndNMoreUpdatesAvailable": { | ||||||
|         "one": "{} et 1 autre application ont des mises à jour.", |         "one": "{} et 1 autre application ont des mises à jour.", | ||||||
| @@ -350,11 +361,15 @@ | |||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesInstalled": { |     "xAndNMoreUpdatesInstalled": { | ||||||
|         "one": "{} et 1 autre application ont été mises à jour.", |         "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 1 autre application.", | ||||||
|  |         "other": "Échec de la mise à jour de {} et {} autres applications." | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "une": "{} et 1 application supplémentaire ont peut-être été mises à jour.", |         "one": "{} et 1 autre application ont peut-être été mises à jour.", | ||||||
|         "other": "{} et {} autres applications peuvent avoir été mises à jour." |         "other": "{} et {} autres applications ont peut-être étés mis à jour." | ||||||
|     }, |     }, | ||||||
|     "apk": { |     "apk": { | ||||||
|         "one": "{} APK", |         "one": "{} APK", | ||||||
|   | |||||||
| @@ -1,192 +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", |     "noReleaseFound": "Nem található megfelelő kiadás", | ||||||
|     "noVersionFound": "Nem sikerült meghatározni a kiadás verzióját", |     "noVersionFound": "Nem sikerült meghatározni a kiadás verzióját", | ||||||
|     "urlMatchesNoSource": "Az URL nem egyezik ismert forrással", |     "urlMatchesNoSource": "A webcím nem egyezik egyetlen ismert forrással sem", | ||||||
|     "cantInstallOlderVersion": "Nem telepíthető egy app régebbi verziója", |     "cantInstallOlderVersion": "Nem telepíthető egy alkalmazás régebbi verziója", | ||||||
|     "appIdMismatch": "A letöltött csomagazonosító nem egyezik a meglévő app azonosítóval", |     "appIdMismatch": "A letöltött csomagazonosító nem egyezik a meglévő alkalmazás azonosítójával", | ||||||
|     "functionNotImplemented": "Ez az osztály nem valósította meg ezt a függvényt", |     "functionNotImplemented": "Ebben az ágban nincs implementálva ez a funkció", | ||||||
|     "placeholder": "Helykitöltő", |     "placeholder": "Helyőrző", | ||||||
|     "someErrors": "Néhány hiba történt", |     "someErrors": "Néhány hiba történt", | ||||||
|     "unexpectedError": "Váratlan hiba", |     "unexpectedError": "Váratlan hiba", | ||||||
|     "ok": "Oké", |     "ok": "Rendben", | ||||||
|     "and": "és", |     "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", |     "includePrereleases": "Tartalmazza az előzetes kiadásokat", | ||||||
|     "fallbackToOlderReleases": "Visszatérés a régebbi kiadásokhoz", |     "fallbackToOlderReleases": "Visszatérés a régebbi kiadásokhoz", | ||||||
|     "filterReleaseTitlesByRegEx": "A kiadás címeinek szűrése reguláris kifejezéssel", |     "filterReleaseTitlesByRegEx": "A kiadás címeinek szűrése reguláris kifejezéssel", | ||||||
|     "invalidRegEx": "Érvénytelen reguláris kifejezés", |     "invalidRegEx": "Érvénytelen reguláris kifejezés", | ||||||
|     "noDescription": "Nincs leírás", |     "noDescription": "Nincs leírás", | ||||||
|     "cancel": "Mégse", |     "cancel": "Mégse", | ||||||
|     "continue": "Tovább", |     "continue": "Folytatás", | ||||||
|     "requiredInBrackets": "(Kötelező)", |     "requiredInBrackets": "(Szükséges)", | ||||||
|     "dropdownNoOptsError": "HIBA: A LEDOBÁST LEGALÁBB EGY OPCIÓHOZ KELL RENDELNI", |     "dropdownNoOptsError": "HIBA: A LEGÖRDÜLŐ LISTÁNAK LEGALÁBB EGY OPCIÓVAL KELL RENDELKEZNIE", | ||||||
|     "colour": "Szín", |     "colour": "Szín", | ||||||
|     "githubStarredRepos": "GitHub Csillagos Repo-k", |     "standard": "Általános", | ||||||
|     "uname": "Felh.név", |     "custom": "Egyéni", | ||||||
|     "wrongArgNum": "Rossz számú argumentumot adott meg", |     "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ő", |     "xIsTrackOnly": "A(z) {} csak nyomonkövethető", | ||||||
|     "source": "Forrás", |     "source": "Forrás", | ||||||
|     "app": "Alkalmazás", |     "app": "Alkalmazás", | ||||||
|     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások 'Csak nyomon követhetőek'.", |     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások „csak nyomonkövethetőek”.", | ||||||
|     "youPickedTrackOnly": "A 'Csak követés' opciót választotta.", |     "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.", |     "trackOnlyAppDescription": "Az alkalmazás frissítéseit nyomon követi, de az Obtainium nem tudja letölteni vagy telepíteni.", | ||||||
|     "cancelled": "Törölve", |     "cancelled": "Visszavonva", | ||||||
|     "appAlreadyAdded": "Az app már hozzáadva", |     "appAlreadyAdded": "Az alkalmazás már hozzá van adva", | ||||||
|     "alreadyUpToDateQuestion": "Az app már naprakész?", |     "alreadyUpToDateQuestion": "Az alkalmazás már naprakész?", | ||||||
|     "addApp": "App hozzáadás", |     "addApp": "Hozzáadás", | ||||||
|     "appSourceURL": "App forrás URL", |     "appSourceURL": "Alkalmazás forrásának webcíme", | ||||||
|     "error": "Hiba", |     "error": "Hiba", | ||||||
|     "add": "Hozzáadás", |     "add": "Hozzáadás", | ||||||
|     "searchSomeSourcesLabel": "Keresés (csak egyes források)", |     "searchSomeSourcesLabel": "Keresés (csak bizonyos források)", | ||||||
|     "search": "Keresés", |     "search": "Keresés", | ||||||
|     "additionalOptsFor": "További lehetőségek a következőhöz: {}", |     "additionalOptsFor": "További lehetőségek a következőhöz: {}", | ||||||
|     "supportedSources": "Támogatott források", |     "supportedSources": "Támogatott források", | ||||||
|     "trackOnlyInBrackets": "(Csak nyomonkövetés)", |     "trackOnlyInBrackets": "(Csak nyomonkövetés)", | ||||||
|     "searchableInBrackets": "(Kereshető)", |     "searchableInBrackets": "(Kereshető)", | ||||||
|     "appsString": "Appok", |     "appsString": "Alkalmazások", | ||||||
|     "noApps": "Nincs App", |     "noApps": "Nincsenek alkalmazások", | ||||||
|     "noAppsForFilter": "Nincsenek appok a szűrőhöz", |     "noAppsForFilter": "Nincsenek alkalmazások a szűrőhöz", | ||||||
|     "byX": "Fejlesztő: {}", |     "byX": "Fejlesztő: {}", | ||||||
|     "percentProgress": "Folyamat: {}%", |     "percentProgress": "Folyamat: {}%", | ||||||
|     "pleaseWait": "Kis türelmet", |     "pleaseWait": "Kis türelmet", | ||||||
|     "updateAvailable": "Frissítés érhető el", |     "updateAvailable": "Frissítés érhető el", | ||||||
|     "notInstalled": "Nem telepített", |     "notInstalled": "Nem telepített", | ||||||
|     "pseudoVersion": "ál-verzió", |     "pseudoVersion": "pszeudo-verzió", | ||||||
|     "selectAll": "Mindet kiválaszt", |     "selectAll": "Összes kiválasztása", | ||||||
|     "deselectX": "Törölje {} kijelölését", |     "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.", |     "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?", |     "removeSelectedAppsQuestion": "A kiválasztott alkalmazások eltávolítása?", | ||||||
|     "removeSelectedApps": "Távolítsa el a kiválasztott appokat", |     "removeSelectedApps": "A kiválasztott alkalmazások eltávolítása", | ||||||
|     "updateX": "Frissítés: {}", |     "updateX": "A(z) {} frissítése", | ||||||
|     "installX": "Telepítés: {}", |     "installX": "A(z) {} telepítése", | ||||||
|     "markXTrackOnlyAsUpdated": "Jelölje meg: {}\n(Csak nyomon követhető)\nmint Frissített", |     "markXTrackOnlyAsUpdated": "Megjelölés: {}\n(Csak nyomon követhető)\nmint Frissített", | ||||||
|     "changeX": "Változás {}", |     "changeX": "{} változtatás", | ||||||
|     "installUpdateApps": "Appok telepítése/frissítése", |     "installUpdateApps": "Alkalmazások telepítése/frissítése", | ||||||
|     "installUpdateSelectedApps": "Telepítse/frissítse a kiválasztott appokat", |     "installUpdateSelectedApps": "A kiválasztott alkalmazások telepítése/frissítése", | ||||||
|     "markXSelectedAppsAsUpdated": "Megjelöl {} kiválasztott alkalmazást frissítettként?", |     "markXSelectedAppsAsUpdated": "A(z) {} kiválasztott alkalmazás megjelölése frissítettként?", | ||||||
|     "no": "Nem", |     "no": "Nem", | ||||||
|     "yes": "Igen", |     "yes": "Igen", | ||||||
|     "markSelectedAppsUpdated": "Jelölje meg a kiválasztott appokat frissítettként", |     "markSelectedAppsUpdated": "A kiválasztott alkalmazások megjelölése frissítettként", | ||||||
|     "pinToTop": "Rögzítés felülre", |     "pinToTop": "Kitűzés felülre", | ||||||
|     "unpinFromTop": "Eltávolít felülről", |     "unpinFromTop": "Kitűzés megszüntetése", | ||||||
|     "resetInstallStatusForSelectedAppsQuestion": "Visszaállítja a kiválasztott appok telepítési állapotát?", |     "resetInstallStatusForSelectedAppsQuestion": "Visszaállítja a kiválasztott alkalmazások 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.", |     "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", |     "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", |     "shareAppConfigLinks": "Alkalmazás-konfiguráció megosztása HTML hivatkozásként", | ||||||
|     "shareSelectedAppURLs": "Ossza meg a kiválasztott app URL címeit", |     "shareSelectedAppURLs": "A kiválasztott alkalmazás hivatkozásának megosztása", | ||||||
|     "resetInstallStatus": "Telepítési állapot visszaállítása", |     "resetInstallStatus": "Telepítési állapot visszaállítása", | ||||||
|     "more": "További", |     "more": "További", | ||||||
|     "removeOutdatedFilter": "Távolítsa el az elavult app szűrőt", |     "removeOutdatedFilter": "Elavult-alkalmazás szűrő eltávolítása", | ||||||
|     "showOutdatedOnly": "Csak az elavult appok megjelenítése", |     "showOutdatedOnly": "Csak az elavult alkalmazások megjelenítése", | ||||||
|     "filter": "Szűrő", |     "filter": "Szűrő", | ||||||
|     "filterApps": "Appok szűrése", |     "filterApps": "Alkalmazások szűrése", | ||||||
|     "appName": "App név", |     "appName": "Név", | ||||||
|     "author": "Szerző", |     "author": "Szerző", | ||||||
|     "upToDateApps": "Naprakész appok", |     "upToDateApps": "Naprakész alkalmazások", | ||||||
|     "nonInstalledApps": "Nem telepített appok", |     "nonInstalledApps": "Nem telepített alkalmazások", | ||||||
|     "importExport": "Import Export", |     "importExport": "Import/Export", | ||||||
|     "settings": "Beállítások", |     "settings": "Beállítások", | ||||||
|     "exportedTo": "Exportálva ide {}", |     "exportedTo": "Exportálva ide: {}", | ||||||
|     "obtainiumExport": "Obtainium Adat Exportálás", |     "obtainiumExport": "Obtainium adatainak exportálása", | ||||||
|     "invalidInput": "Hibás bemenet", |     "invalidInput": "Hibás bemenet", | ||||||
|     "importedX": "Importálva innen {}", |     "importedX": "Importálva innen: {}", | ||||||
|     "obtainiumImport": "Obtainium Adat Importálás", |     "obtainiumImport": "Obtainium adatok importálása", | ||||||
|     "importFromURLList": "Importálás URL listából", |     "importFromURLList": "Importálás webcím-listából", | ||||||
|     "searchQuery": "Keresési lekérdezés", |     "searchQuery": "Keresési lekérdezés", | ||||||
|     "appURLList": "App URL lista", |     "appURLList": "Alkalmazás webcím-lista", | ||||||
|     "line": "Sor", |     "line": "Sor", | ||||||
|     "searchX": "Keresés {}", |     "searchX": "{} keresése", | ||||||
|     "noResults": "Nincs találat", |     "noResults": "Nincs találat", | ||||||
|     "importX": "Importálás: {}", |     "importX": "{} importálása", | ||||||
|     "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..", |     "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", |     "importErrors": "Importálási hibák", | ||||||
|     "importedXOfYApps": "{}/{} app importálva.", |     "importedXOfYApps": "{}/{} alkalmazás importálva.", | ||||||
|     "followingURLsHadErrors": "A következő URL-ek hibákat tartalmaztak:", |     "followingURLsHadErrors": "A következő webcímek hibákat tartalmaztak:", | ||||||
|     "selectURL": "Válassza ki az URL-t", |     "selectURL": "Webcím kiválasztása", | ||||||
|     "selectURLs": "Kiválasztott URL-ek", |     "selectURLs": "Webcímek kiválasztása", | ||||||
|     "pick": "Válasszon", |     "pick": "Kiválasztás", | ||||||
|     "theme": "Téma", |     "theme": "Téma", | ||||||
|     "dark": "Sötét", |     "dark": "Sötét", | ||||||
|     "light": "Világos", |     "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", |     "useBlackTheme": "Használjon teljesen fekete sötét témát", | ||||||
|     "appSortBy": "App rendezés...", |     "appSortBy": "Elrendezés", | ||||||
|     "authorName": "Szerző/Név", |     "authorName": "Szerző/Név", | ||||||
|     "nameAuthor": "Név/Szerző", |     "nameAuthor": "Név/Szerző", | ||||||
|     "asAdded": "Mint Hozzáadott", |     "asAdded": "Mint hozzáadott", | ||||||
|     "appSortOrder": "Appok rendezése", |     "appSortOrder": "Rendezési sorrend", | ||||||
|     "ascending": "Emelkedő", |     "ascending": "Emelkedő", | ||||||
|     "descending": "Csökkenő", |     "descending": "Csökkenő", | ||||||
|     "bgUpdateCheckInterval": "Háttérfrissítés ellenőrzés időköze", |     "bgUpdateCheckInterval": "Időtartam a frissítések háttér-ellenőrzése között", | ||||||
|     "neverManualOnly": "Soha – csak manuális", |     "neverManualOnly": "Soha – csak kézi", | ||||||
|     "appearance": "Megjelenés", |     "appearance": "Megjelenés", | ||||||
|     "showWebInAppView": "Forrás megjelenítése az Appok nézetben", |     "showWebInAppView": "Forrás megjelenítése az alkalmazásnézetben", | ||||||
|     "pinUpdates": "Frissítések kitűzése az App nézet tetejére", |     "pinUpdates": "Frissítések kitűzése az alkalmazásnézet tetejére", | ||||||
|     "updates": "Frissítések", |     "updates": "Frissítések", | ||||||
|     "sourceSpecific": "Forrás-specifikus", |     "sourceSpecific": "Forrásspecifikus", | ||||||
|     "appSource": "App forrás", |     "appSource": "Alkalmazás forrása", | ||||||
|     "noLogs": "Nincsenek naplók", |     "noLogs": "Nincsenek naplók", | ||||||
|     "appLogs": "App naplók", |     "appLogs": "Alkalmazásnaplók", | ||||||
|     "close": "Bezárás", |     "close": "Bezárás", | ||||||
|     "share": "Megosztás", |     "share": "Megosztás", | ||||||
|     "appNotFound": "App nem található", |     "appNotFound": "Az alkalmazás nem található", | ||||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-export", |     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||||
|     "pickAnAPK": "Válasszon egy APK-t", |     "pickAnAPK": "Válasszon egy APK-t", | ||||||
|     "appHasMoreThanOnePackage": "A(z) {} egynél több csomaggal rendelkezik:", |     "appHasMoreThanOnePackage": "A(z) {} egynél több csomaggal rendelkezik:", | ||||||
|     "deviceSupportsXArch": "Eszköze támogatja a {} CPU architektúrát.", |     "deviceSupportsXArch": "Ez az eszköz támogatja a(z) {} CPU architektúrát.", | ||||||
|     "deviceSupportsFollowingArchs": "Az eszköze a következő CPU architektúrákat támogatja:", |     "deviceSupportsFollowingArchs": "Ez az eszköz a következő CPU architektúrákat támogatja:", | ||||||
|     "warning": "Figyelem", |     "warning": "Figyelem", | ||||||
|     "sourceIsXButPackageFromYPrompt": "Az alkalmazás forrása „{}”, de a kiadási csomag innen származik: „{}”. Folytatja?", |     "sourceIsXButPackageFromYPrompt": "Az alkalmazás forrása a(z) „{}” tároló, de a kiadási csomag innen származik: „{}”. Folytatja?", | ||||||
|     "updatesAvailable": "Frissítések érhetők el", |     "updatesAvailable": "Frissítések érhetőek 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", |     "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.", |     "noNewUpdates": "Nincsenek új frissítések.", | ||||||
|     "xHasAnUpdate": "A(z) {} frissítést kapott.", |     "xHasAnUpdate": "A(z) {} frissítést kapott.", | ||||||
|     "appsUpdated": "Alkalmazások frissítve", |     "appsUpdated": "Alkalmazások frissítve", | ||||||
|     "appsUpdatedNotifDescription": "Értesíti a felhasználót, hogy egy/több app frissítése megtörtént a háttérben", |     "appsNotUpdated": "Nem sikerült frissíteni az alkalmazásokat", | ||||||
|     "xWasUpdatedToY": "{} frissítve a következőre: {}.", |     "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", |     "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", |     "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", |     "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) {} a következő hiba miatt lett eltávolítva: {}", |     "xWasRemovedDueToErrorY": "A(z) {} eltávolításra került a következő hiba miatt: {}", | ||||||
|     "completeAppInstallation": "Teljes app telepítés", |     "completeAppInstallation": "Teljes alkalmazástelepítés", | ||||||
|     "obtainiumMustBeOpenToInstallApps": "Az Obtainiumnak megnyitva kell lennie az alkalmazások telepítéséhez", |     "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", |     "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", |     "checkingForUpdates": "Frissítések keresése", | ||||||
|     "checkingForUpdatesNotifDescription": "Átmeneti értesítés, amely a frissítések keresésekor jelenik meg", |     "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", |     "pleaseAllowInstallPerm": "Engedélyezze az Obtainiumnak az alkalmazások telepítését", | ||||||
|     "trackOnly": "Csak követés", |     "trackOnly": "Csak nyomonkövetés", | ||||||
|     "errorWithHttpStatusCode": "Hiba {}", |     "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", |     "unknown": "Ismeretlen", | ||||||
|     "none": "Egyik sem", |     "none": "Semmi", | ||||||
|     "never": "Soha", |     "never": "Soha", | ||||||
|     "latestVersionX": "Legújabb verzió: {}", |     "latestVersionX": "Legújabb verzió: {}", | ||||||
|     "installedVersionX": "Telepített verzió: {}", |     "installedVersionX": "Telepített verzió: {}", | ||||||
|     "lastUpdateCheckX": "Frissítés ellenőrizve: {}", |     "lastUpdateCheckX": "Frissítések utoljára ellenőrizve: {}", | ||||||
|     "remove": "Eltávolítás", |     "remove": "Eltávolítás", | ||||||
|     "yesMarkUpdated": "Igen, megjelölés frissítettként", |     "yesMarkUpdated": "Igen, megjelölés frissítettként", | ||||||
|     "fdroid": "F-Droid hivatalos", |     "fdroid": "F-Droid hivatalos", | ||||||
|     "appIdOrName": "App ID vagy név", |     "appIdOrName": "Az alkalmazás-azonosító vagy név", | ||||||
|     "appId": "Alkalmazásazonosító", |     "appId": "Alkalmazás-azonosító", | ||||||
|     "appWithIdOrNameNotFound": "Nem található app ezzel az azonosítóval vagy névvel", |     "appWithIdOrNameNotFound": "Nem található alkalmazás ezzel az azonosítóval vagy névvel", | ||||||
|     "reposHaveMultipleApps": "A repók több alkalmazást is tartalmazhatnak", |     "reposHaveMultipleApps": "A tárolók több alkalmazást is tartalmazhatnak", | ||||||
|     "fdroidThirdPartyRepo": "F-Droid Harmadik-fél Repo", |     "fdroidThirdPartyRepo": "F-Droid harmadik féltől származó tároló", | ||||||
|     "steamMobile": "Steam mobil", |     "steamMobile": "Steam Mobil", | ||||||
|     "steamChat": "Steam Chat", |     "steamChat": "Steam Chat", | ||||||
|     "install": "Telepít", |     "install": "Telepítés", | ||||||
|     "markInstalled": "Telepítettnek jelöl", |     "markInstalled": "Telepítettnek jelölés", | ||||||
|     "update": "Frissít", |     "update": "Frissítés", | ||||||
|     "markUpdated": "Frissítettnek jelöl", |     "markUpdated": "Frissítettnek jelölés", | ||||||
|     "additionalOptions": "További lehetőségek", |     "additionalOptions": "További beállítások", | ||||||
|     "disableVersionDetection": "Verzió érzékelés letiltása", |     "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.", |     "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", |     "downloadingX": "{} letöltése", | ||||||
|     "downloadX": "Letöltés {}", |     "downloadX": "{} letöltése", | ||||||
|     "downloadedX": "Letöltés {}", |     "downloadedX": "{} letöltve", | ||||||
|     "releaseAsset": "Release Asset", |     "releaseAsset": "Kiadási csomag", | ||||||
|     "downloadNotifDescription": "Értesíti a felhasználót az app letöltésének előrehaladásáról", |     "downloadNotifDescription": "Értesíti a felhasználót az alkalmazás letöltésének előrehaladásáról", | ||||||
|     "noAPKFound": "Nem található APK", |     "noAPKFound": "Nem található APK", | ||||||
|     "noVersionDetection": "Nincs verzió érzékelés", |     "noVersionDetection": "Nincs verzió érzékelés", | ||||||
|     "categorize": "Kategorizálás", |     "categorize": "Kategorizálás", | ||||||
| @@ -195,8 +201,8 @@ | |||||||
|     "noCategory": "Nincs kategória", |     "noCategory": "Nincs kategória", | ||||||
|     "noCategories": "Nincsenek kategóriák", |     "noCategories": "Nincsenek kategóriák", | ||||||
|     "deleteCategoriesQuestion": "Törli a kategóriákat?", |     "deleteCategoriesQuestion": "Törli a kategóriákat?", | ||||||
|     "categoryDeleteWarning": "A(z) {} összes app kategorizálatlan állapotba kerül.", |     "categoryDeleteWarning": "A törölt kategóriákban lévő összes alkalmazás kategorizálatlanná válik.", | ||||||
|     "addCategory": "Új kategória", |     "addCategory": "Kategória hozzáadása", | ||||||
|     "label": "Címke", |     "label": "Címke", | ||||||
|     "language": "Nyelv", |     "language": "Nyelv", | ||||||
|     "copiedToClipboard": "Másolva a vágólapra", |     "copiedToClipboard": "Másolva a vágólapra", | ||||||
| @@ -204,129 +210,134 @@ | |||||||
|     "selectedCategorizeWarning": "Ez felváltja a kiválasztott alkalmazások meglévő kategória-beállításait.", |     "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", |     "filterAPKsByRegEx": "Az APK-k szűrése reguláris kifejezéssel", | ||||||
|     "removeFromObtainium": "Eltávolítás az Obtainiumból", |     "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.", |     "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", |     "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.", |     "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áltoztatások", |     "changes": "Változások", | ||||||
|     "releaseDate": "Kiadás dátuma", |     "releaseDate": "Kiadás dátuma", | ||||||
|     "importFromURLsInFile": "Importálás fájlban található URL-ből (mint pl. OPML)", |     "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", |     "versionDetectionExplanation": "A verzió-karakterlánc egyeztetése az OS által észlelt verzióval", | ||||||
|     "versionDetection": "Verzió érzékelés", |     "versionDetection": "Verzióérzékelés", | ||||||
|     "standardVersionDetection": "Alapért. verzió érzékelés", |     "standardVersionDetection": "Alapértelmezett verzióérzékelés", | ||||||
|     "groupByCategory": "Csoportosítás Kategória alapján", |     "groupByCategory": "Csoportosítás kategória alapján", | ||||||
|     "autoApkFilterByArch": "Ha lehetséges, próbálja CPU architektúra szerint szűrni az APK-kat", |     "autoApkFilterByArch": "Ha lehetséges, próbálja CPU architektúra szerint szűrni az APK-kat", | ||||||
|     "overrideSource": "Forrás felülbírálása", |     "overrideSource": "Forrás felülírása", | ||||||
|     "dontShowAgain": "Ne mutassa ezt újra", |     "dontShowAgain": "Ne jelenítse meg ezt többé", | ||||||
|     "dontShowTrackOnlyWarnings": "Ne jelenítsen meg 'Csak nyomon követés' figyelmeztetést", |     "dontShowTrackOnlyWarnings": "Ne jelenítse meg a „Csak nyomonkövetés” figyelmeztetést", | ||||||
|     "dontShowAPKOriginWarnings": "Ne jelenítsen meg az APK eredetére vonatkozó figyelmeztetéseket", |     "dontShowAPKOriginWarnings": "Ne jelenítse meg az APK eredetére vonatkozó figyelmeztetéseket", | ||||||
|     "moveNonInstalledAppsToBottom": "Helyezze át a nem telepített appokat az App nézet aljára", |     "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", |     "gitlabPATLabel": "GitLab személyes hozzáférési token", | ||||||
|     "about": "Rólunk", |     "about": "Névjegy", | ||||||
|     "requiresCredentialsInSettings": "{}: Ehhez további hitelesítő adatokra van szükség (a Beállításokban)", |     "requiresCredentialsInSettings": "A(z) {} alkalmazásnak további hitelesítő adatokra van szüksége (a beállításokban)", | ||||||
|     "checkOnStart": "Egyszer az alkalmazás indításakor is", |     "checkOnStart": "Frissítések keresése indításkor", | ||||||
|     "tryInferAppIdFromCode": "Próbálja kikövetkeztetni az app azonosítót a forráskódból", |     "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 appok auto. eltávolítása", |     "removeOnExternalUninstall": "A külsőleg eltávolított alkalmazások automatikus eltávolítása", | ||||||
|     "pickHighestVersionCode": "A legmagasabb verziószámú APK auto. kiválasztása", |     "pickHighestVersionCode": "A legmagasabb verziószámú APK automatikus kiválasztása", | ||||||
|     "checkUpdateOnDetailPage": "Frissítések keresése az app részleteit tartalmazó oldal megnyitásakor", |     "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", |     "disablePageTransitions": "Lap áttűnési animációk letiltása", | ||||||
|     "reversePageTransitions": "Fordított lap áttűnési animációk", |     "reversePageTransitions": "Fordított lap áttűnési animációk", | ||||||
|     "minStarCount": "Minimális csillag szám", |     "minStarCount": "Minimális csillagozási szám", | ||||||
|     "addInfoBelow": "Adja hozzá ezt az infót alább.", |     "addInfoBelow": "Adja hozzá ezt az információt alább.", | ||||||
|     "addInfoInSettings": "Adja hozzá ezt az infót a Beállításokban.", |     "addInfoInSettings": "Adja hozzá ezt az információt a beállításokban.", | ||||||
|     "githubSourceNote": "A GitHub sebességkorlátozás elkerülhető API-kulcs használatával.", |     "githubSourceNote": "A GitHub/GitLab lekérdezés-korlátozás elkerülhető egy API-kulcs használatával.", | ||||||
|     "sortByLastLinkSegment": "Rendezés csak a link utolsó szegmense szerint", |     "sortByLastLinkSegment": "Rendezés csak a hivatkozás utolsó szegmense szerint", | ||||||
|     "filterReleaseNotesByRegEx": "Kiadási megjegyzések szűrése reguláris kifejezéssel", |     "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$')", |     "customLinkFilterRegex": "Egyéni APK hivatkozásszűrő reguláris kifejezéssel (Alapértelmezett „.apk$”)", | ||||||
|     "appsPossiblyUpdated": "App frissítési kísérlet", |     "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", |     "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 {}.", |     "xWasPossiblyUpdatedToY": "A(z) {} frissülhetett a következőre: {}.", | ||||||
|     "enableBackgroundUpdates": "Frissítések a háttérben", |     "enableBackgroundUpdates": "Háttérfrissítések engedélyezése", | ||||||
|     "backgroundUpdateReqsExplanation": "Előfordulhat, hogy nem minden appnál lehetséges a háttérbeli frissítés.", |     "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.", |     "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", |     "verifyLatestTag": "Ellenőrizze a „legújabb” címkét", | ||||||
|     "intermediateLinkRegex": "Szűrés egy 'köztes' látogatási linkre", |     "intermediateLinkRegex": "Szűrés egy „köztes” hivatkozás megnyitásához", | ||||||
|     "filterByLinkText": "A hivatkozások szűrése linkszöveg alapján", |     "filterByLinkText": "Hivatkozások szűrése egy hivatkozásszöveg alapján", | ||||||
|     "intermediateLinkNotFound": "Köztes link nem található", |     "intermediateLinkNotFound": "Köztes hivatkozás nem található", | ||||||
|     "intermediateLink": "Köztes link", |     "intermediateLink": "Köztes hivatkozás", | ||||||
|     "exemptFromBackgroundUpdates": "Mentes a háttérben történő frissítések alól (ha engedélyezett)", |     "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", |     "bgUpdatesOnWiFiOnly": "A háttérben futó frissítések letiltása, ha nincs Wi-Fi", | ||||||
|     "autoSelectHighestVersionCode": "A legmagasabb verziószámú APK auto. kiválasztása", |     "autoSelectHighestVersionCode": "A legmagasabb verziószámú APK automatikus kiválasztása", | ||||||
|     "versionExtractionRegEx": "Verzió kibontása reguláris kifejezéssel", |     "versionExtractionRegEx": "Verziókarakterlánc-kivonatolása reguláris kifejezéssel", | ||||||
|     "matchGroupToUse": "Párosítsa a csoportot a használathoz", |     "trimVersionString": "Verziókarakterlánc levágása reguláris kifejezéssel", | ||||||
|     "highlightTouchTargets": "Emelje ki a kevésbé nyilvánvaló érintési célokat", |     "matchGroupToUseForX": "A(z) „{}” esetén használandó csoport egyeztetése", | ||||||
|     "pickExportDir": "Válassza az Exportálási könyvtárat", |     "matchGroupToUse": "Verziókarakterlánc-kivonatoláshoz használandó csoport reguláris kifejezéssel való egyeztetése", | ||||||
|     "autoExportOnChanges": "Auto-exportálás a változások után", |     "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", |     "includeSettings": "Tartalmazza a beállításokat", | ||||||
|     "filterVersionsByRegEx": "Verziók szűrése reguláris kifejezéssel", |     "filterVersionsByRegEx": "Verziók szűrése reguláris kifejezéssel", | ||||||
|     "trySelectingSuggestedVersionCode": "Próbálja ki a javasolt verziókódú APK-t", |     "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", |     "dontSortReleasesList": "Az API-ból származó kiadási sorrend megőrzése", | ||||||
|     "reverseSort": "Fordított rendezés", |     "reverseSort": "Fordított elrendezés", | ||||||
|     "takeFirstLink": "Vegye az első linket", |     "takeFirstLink": "Vegye az első hivatkozást", | ||||||
|     "skipSort": "A válogatás kihagyása", |     "skipSort": "Rendezés kihagyása", | ||||||
|     "debugMenu": "Hibakereső menü", |     "debugMenu": "Hibakereső menü", | ||||||
|     "bgTaskStarted": "A háttérfeladat elindult – ellenőrizze a naplókat.", |     "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", |     "runBgCheckNow": "Frissítések keresése a háttérben", | ||||||
|     "versionExtractWholePage": "Alkalmazza a Version Extraction Regex-et az egész oldalra", |     "versionExtractWholePage": "„Verziókarakterlánc-kivonatolása reguláris kifejezéssel” alkalmazása az egész oldalra", | ||||||
|     "installing": "Telepítés", |     "installing": "Telepítés", | ||||||
|     "skipUpdateNotifications": "A frissítési értesítések kihagyása", |     "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", |     "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", |     "errorCheckingUpdatesNotifChannel": "Hiba a frissítések keresésekor", | ||||||
|     "appsRemovedNotifChannel": "Alkalmazások eltávolítva", |     "appsRemovedNotifChannel": "Eltávolított alkalmazások", | ||||||
|     "downloadingXNotifChannel": "{} letöltés", |     "downloadingXNotifChannel": "A(z) {} letöltése", | ||||||
|     "completeAppInstallationNotifChannel": "Teljes app telepítés", |     "completeAppInstallationNotifChannel": "Teljes alkalmazás telepítés", | ||||||
|     "checkingForUpdatesNotifChannel": "Frissítések keresése", |     "checkingForUpdatesNotifChannel": "Frissítések keresése", | ||||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Csak a telepített és a csak követhető appokat ellenőrizze frissítésekért", |     "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 URL-eket", |     "supportFixedAPKURL": "Támogatja a rögzített APK webcímeket", | ||||||
|     "selectX": "Kiválaszt {}", |     "selectX": "{} kiválasztása", | ||||||
|     "parallelDownloads": "Párhuzamos letöltéseket enged", |     "parallelDownloads": "Párhuzamos letöltések engedélyezése", | ||||||
|     "installMethod": "Telepítési mód", |     "useShizuku": "Shizuku vagy Sui használata a telepítéshez", | ||||||
|     "normal": "Normál", |     "shizukuBinderNotFound": "A Shizuku szolgáltatás nem fut", | ||||||
|     "root": "Root", |     "shizukuOld": "Régi Shizuku verzió (<11) - frissítse", | ||||||
|     "shizukuBinderNotFound": "A Shizuku nem fut", |     "shizukuOldAndroidWithADB": "A Shizuku csak Android < 8.1 ADB-vel fut - frissítse az Androidot vagy használja a Sui-t helyette", | ||||||
|     "useSystemFont": "Használja a rendszer betűtípusát", |     "shizukuPretendToBeGooglePlay": "Állítsa be a Google Playt telepítési forrásként (ha Shizukut használ)", | ||||||
|     "systemFontError": "Hiba a rendszer betűtípusának betöltésekor: {}", |     "useSystemFont": "A rendszer betűtípusának használata", | ||||||
|     "useVersionCodeAsOSVersion": "Az app verziókód használata a rendszer által észlelt verzióként", |     "useVersionCodeAsOSVersion": "Az alkalmazás verziókódjának használata a rendszer által észlelt verzióként", | ||||||
|     "requestHeader": "Kérelem fejléc", |     "requestHeader": "Kérelemfejléc", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Használja a legújabb tartalomfeltöltést megjelenési dátumként", |     "useLatestAssetDateAsReleaseDate": "A kiadás dátumaként használja a legutóbbi csomagfeltöltést", | ||||||
|     "defaultPseudoVersioningMethod": "Alapértelmezett álverziós módszer", |     "defaultPseudoVersioningMethod": "Alapértelmezett pszeudo-verziós módszer", | ||||||
|     "partialAPKHash": "Részleges APK Hash", |     "partialAPKHash": "Részleges APK hasító értéke", | ||||||
|     "APKLinkHash": "APK Link Hash", |     "APKLinkHash": "APK hivatkozás hasító értéke", | ||||||
|     "directAPKLink": "Közvetlen APK Link", |     "directAPKLink": "Közvetlen APK hivatkozás", | ||||||
|     "pseudoVersionInUse": "Egy álverzió van használatban", |     "pseudoVersionInUse": "Egy pszeudo-verzió van használatban", | ||||||
|     "installed": "Telepített", |     "installed": "Telepített", | ||||||
|     "latest": "Legújabb", |     "latest": "Legújabb", | ||||||
|     "invertRegEx": "Invertált reguláris kifejezés", |     "invertRegEx": "Reguláris kifejezés invertálása", | ||||||
|     "note": "Megjegyzé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.", |     "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)", |     "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ő)", |     "beforeNewInstallsShareToAppVerifier": "Új alkalmazások megosztása az AppVerifierrel (ha elérhető)", | ||||||
|     "appVerifierInstructionToast": "Ossza meg az AppVerifierrel, majd térjen vissza ide, ha kész.", |     "appVerifierInstructionToast": "Ossza meg az AppVerifierrel, majd térjen vissza ide, ha kész.", | ||||||
|  |     "wiki": "Súgó/Wiki", | ||||||
|  |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "Eltávolítja az alkalmazást?", |         "one": "Eltávolítja az alkalmazást?", | ||||||
|         "other": "Eltávolítja az alkalmazásokat?" |         "other": "Eltávolítja az alkalmazásokat?" | ||||||
|     }, |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "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 (korlátozott arány) – próbálja újra {} perc múlva" |         "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": { |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|         "one": "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": "A háttérfrissítések ellenőrzése {}-t észlelt, {} perc múlva ütemezi az újrapróbálkozást" |         "other": "Frissítések ellenőrzése a háttérben a következőt észlelte: {} - Újrapróbálkozás: {} perc múlva" | ||||||
|     }, |     }, | ||||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { |     "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", |         "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": "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": "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": { |     "apps": { | ||||||
|         "one": "{} app", |         "one": "{} alkalmazás", | ||||||
|         "other": "{} app" |         "other": "{} alkalmazás" | ||||||
|     }, |     }, | ||||||
|     "url": { |     "url": { | ||||||
|         "one": "{} URL", |         "one": "{} webcím", | ||||||
|         "other": "{} URL" |         "other": "{} webcím" | ||||||
|     }, |     }, | ||||||
|     "minute": { |     "minute": { | ||||||
|         "one": "{} perc", |         "one": "{} perc", | ||||||
| @@ -341,23 +352,27 @@ | |||||||
|         "other": "{} nap" |         "other": "{} nap" | ||||||
|     }, |     }, | ||||||
|     "clearedNLogsBeforeXAfterY": { |     "clearedNLogsBeforeXAfterY": { | ||||||
|         "one": "{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 (előtte = {előtte}, utána = {utána})" |         "other": "{n} napló törölve lett ({after} után és {before} előtt)" | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesAvailable": { |     "xAndNMoreUpdatesAvailable": { | ||||||
|         "one": "A(z) {} és 1 további alkalmazás frissítéseket kapott.", |         "one": "A(z) {} és 1 további alkalmazás frissítést kapott.", | ||||||
|         "other": "{} és {} további alkalmazás frissítéseket kapott." |         "other": "A(z) {} és {} további alkalmazás frissítést kapott." | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesInstalled": { |     "xAndNMoreUpdatesInstalled": { | ||||||
|         "one": "A(z) {} és 1 további alkalmazás frissítve.", |         "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 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": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} és 1 további alkalmazás is frissült.", |         "one": "A(z) {}, valamint 1 további alkalmazás sikeresen frissítve.", | ||||||
|         "other": "{} és {} további alkalmazás is frissült." |         "other": "A(z) {}, valamint {} további alkalmazás sikeresen frissítve." | ||||||
|     }, |     }, | ||||||
|     "apk": { |     "apk": { | ||||||
|         "one": "{} APK", |         "one": "{} APK", | ||||||
|         "other": "{} APK-k" |         "other": "{} APK" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(richiesto)", |     "requiredInBrackets": "(richiesto)", | ||||||
|     "dropdownNoOptsError": "ERRORE: LA TENDINA DEVE AVERE ALMENO UN'OPZIONE", |     "dropdownNoOptsError": "ERRORE: LA TENDINA DEVE AVERE ALMENO UN'OPZIONE", | ||||||
|     "colour": "Colore", |     "colour": "Colore", | ||||||
|  |     "standard": "Standard", | ||||||
|  |     "custom": "Personalizzato", | ||||||
|  |     "useMaterialYou": "Utilizzate il materiale che avete a disposizione", | ||||||
|     "githubStarredRepos": "repository stellati da GitHub", |     "githubStarredRepos": "repository stellati da GitHub", | ||||||
|     "uname": "Nome utente", |     "uname": "Nome utente", | ||||||
|     "wrongArgNum": "Numero di argomenti forniti errato", |     "wrongArgNum": "Numero di argomenti forniti errato", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Scuro", |     "dark": "Scuro", | ||||||
|     "light": "Chiaro", |     "light": "Chiaro", | ||||||
|     "followSystem": "Segui il sistema", |     "followSystem": "Segui il sistema", | ||||||
|  |     "followSystemThemeExplanation": "È possibile seguire il tema di sistema solo utilizzando applicazioni di terze parti.", | ||||||
|     "useBlackTheme": "Usa il tema nero puro", |     "useBlackTheme": "Usa il tema nero puro", | ||||||
|     "appSortBy": "App ordinate per", |     "appSortBy": "App ordinate per", | ||||||
|     "authorName": "Autore/Nome", |     "authorName": "Autore/Nome", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Nessun nuovo aggiornamento.", |     "noNewUpdates": "Nessun nuovo aggiornamento.", | ||||||
|     "xHasAnUpdate": "Aggiornamento disponibile per {}", |     "xHasAnUpdate": "Aggiornamento disponibile per {}", | ||||||
|     "appsUpdated": "App aggiornate", |     "appsUpdated": "App aggiornate", | ||||||
|  |     "appsNotUpdated": "Impossibile aggiornare le applicazioni", | ||||||
|     "appsUpdatedNotifDescription": "Notifica all'utente che una o più app sono state aggiornate in secondo piano", |     "appsUpdatedNotifDescription": "Notifica all'utente che una o più app sono state aggiornate in secondo piano", | ||||||
|     "xWasUpdatedToY": "{} è stato aggiornato alla {}.", |     "xWasUpdatedToY": "{} è stato aggiornato alla {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Impossibile aggiornare {} a {}.", | ||||||
|     "errorCheckingUpdates": "Controllo degli errori per gli aggiornamenti", |     "errorCheckingUpdates": "Controllo degli errori per gli aggiornamenti", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Una notifica che mostra quando il controllo degli aggiornamenti in secondo piano fallisce", |     "errorCheckingUpdatesNotifDescription": "Una notifica che mostra quando il controllo degli aggiornamenti in secondo piano fallisce", | ||||||
|     "appsRemoved": "App rimosse", |     "appsRemoved": "App rimosse", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "Disattiva aggiornamenti in secondo piano quando non si usa il WiFi", |     "bgUpdatesOnWiFiOnly": "Disattiva aggiornamenti in secondo piano quando non si usa il WiFi", | ||||||
|     "autoSelectHighestVersionCode": "Auto-seleziona APK con versionCode più alto", |     "autoSelectHighestVersionCode": "Auto-seleziona APK con versionCode più alto", | ||||||
|     "versionExtractionRegEx": "RegEx di estrazione versione", |     "versionExtractionRegEx": "RegEx di estrazione versione", | ||||||
|  |     "trimVersionString": "Tagliare la stringa della versione con RegEx", | ||||||
|  |     "matchGroupToUseForX": "Gruppo di corrispondenza da utilizzare per \"{}\"", | ||||||
|     "matchGroupToUse": "Gruppo da usare", |     "matchGroupToUse": "Gruppo da usare", | ||||||
|     "highlightTouchTargets": "Evidenzia elementi toccabili meno ovvi", |     "highlightTouchTargets": "Evidenzia elementi toccabili meno ovvi", | ||||||
|     "pickExportDir": "Scegli cartella esp.", |     "pickExportDir": "Scegli cartella esp.", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Supporta URL fissi di APK", |     "supportFixedAPKURL": "Supporta URL fissi di APK", | ||||||
|     "selectX": "Seleziona {}", |     "selectX": "Seleziona {}", | ||||||
|     "parallelDownloads": "Permetti download paralleli", |     "parallelDownloads": "Permetti download paralleli", | ||||||
|     "installMethod": "Metodo d'installazione", |     "useShizuku": "Utilizzare Shizuku o Sui per installare", | ||||||
|     "normal": "Normale", |  | ||||||
|     "root": "Root", |  | ||||||
|     "shizukuBinderNotFound": "Shizuku non è in esecuzione", |     "shizukuBinderNotFound": "Shizuku non è in esecuzione", | ||||||
|  |     "shizukuOld": "Vecchia versione di Shizuku (<11) - aggiornarla", | ||||||
|  |     "shizukuOldAndroidWithADB": "Shizuku funziona su Android < 8.1 con ADB - aggiornare Android o utilizzare Sui al suo posto", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Impostare Google Play come fonte di installazione (se si usa Shizuku)", | ||||||
|     "useSystemFont": "Usa i caratteri di sistema", |     "useSystemFont": "Usa i caratteri di sistema", | ||||||
|     "systemFontError": "Errore durante il caricamento dei caratteri di sistema: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Usa il codice versione dell'app come versione rilevata dal sistema operativo", |     "useVersionCodeAsOSVersion": "Usa il codice versione dell'app come versione rilevata dal sistema operativo", | ||||||
|     "requestHeader": "Intestazione della richiesta", |     "requestHeader": "Intestazione della richiesta", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Usa l'ultimo caricamento della risorsa come data di rilascio", |     "useLatestAssetDateAsReleaseDate": "Usa l'ultimo caricamento della risorsa come data di rilascio", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "Non è stato possibile analizzare l'APK (download incompatibile o parziale).", |     "badDownload": "Non è stato possibile analizzare l'APK (download incompatibile o parziale).", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Condividere le nuove applicazioni con AppVerifier (se disponibile)", |     "beforeNewInstallsShareToAppVerifier": "Condividere le nuove applicazioni con AppVerifier (se disponibile)", | ||||||
|     "appVerifierInstructionToast": "Condividete con AppVerifier, quindi tornate qui quando siete pronti.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "Rimuovere l'app?", |         "one": "Rimuovere l'app?", | ||||||
|         "other": "Rimuovere le app?" |         "other": "Rimuovere le app?" | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} e un'altra app sono state aggiornate.", |         "one": "{} e un'altra app sono state aggiornate.", | ||||||
|         "other": "{} e altre {} app sono state aggiornate." |         "other": "{} e altre {} app sono state aggiornate." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Non è riuscito ad aggiornare {} e altre 1 app.", | ||||||
|  |         "other": "Non è riuscito ad aggiornare {} e {} altre applicazioni." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} e un'altra app potrebbero essere state aggiornate.", |         "one": "{} e un'altra app potrebbero essere state aggiornate.", | ||||||
|         "other": "{} e altre {} app potrebbero essere state aggiornate." |         "other": "{} e altre {} app potrebbero essere state aggiornate." | ||||||
|   | |||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(必須)", |     "requiredInBrackets": "(必須)", | ||||||
|     "dropdownNoOptsError": "エラー: ドロップダウンには、少なくとも1つのオプションが必要です", |     "dropdownNoOptsError": "エラー: ドロップダウンには、少なくとも1つのオプションが必要です", | ||||||
|     "colour": "カラー", |     "colour": "カラー", | ||||||
|  |     "standard": "スタンダード", | ||||||
|  |     "custom": "カスタム", | ||||||
|  |     "useMaterialYou": "Material Youを使用する", | ||||||
|     "githubStarredRepos": "Githubでスターしたリポジトリ", |     "githubStarredRepos": "Githubでスターしたリポジトリ", | ||||||
|     "uname": "ユーザー名", |     "uname": "ユーザー名", | ||||||
|     "wrongArgNum": "提供する引数の数が間違っています", |     "wrongArgNum": "提供する引数の数が間違っています", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "ダーク", |     "dark": "ダーク", | ||||||
|     "light": "ライト", |     "light": "ライト", | ||||||
|     "followSystem": "システムに従う", |     "followSystem": "システムに従う", | ||||||
|  |     "followSystemThemeExplanation": "以下のシステムテーマは、サードパーティのアプリケーションを使用することによってのみ可能です。", | ||||||
|     "useBlackTheme": "ピュアブラックダークテーマを使用する", |     "useBlackTheme": "ピュアブラックダークテーマを使用する", | ||||||
|     "appSortBy": "アプリの並び方", |     "appSortBy": "アプリの並び方", | ||||||
|     "authorName": "作者名/アプリ名", |     "authorName": "作者名/アプリ名", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "新しいアップデートはありません", |     "noNewUpdates": "新しいアップデートはありません", | ||||||
|     "xHasAnUpdate": "{} のアップデートが利用可能です。", |     "xHasAnUpdate": "{} のアップデートが利用可能です。", | ||||||
|     "appsUpdated": "アプリをアップデートしました", |     "appsUpdated": "アプリをアップデートしました", | ||||||
|  |     "appsNotUpdated": "アプリケーションの更新に失敗", | ||||||
|     "appsUpdatedNotifDescription": "1つまたは複数のAppのアップデートがバックグラウンドで適用されたことをユーザーに通知する", |     "appsUpdatedNotifDescription": "1つまたは複数のAppのアップデートがバックグラウンドで適用されたことをユーザーに通知する", | ||||||
|     "xWasUpdatedToY": "{} が {} にアップデートされました", |     "xWasUpdatedToY": "{} が {} にアップデートされました", | ||||||
|  |     "xWasNotUpdatedToY": "への更新に失敗しました。", | ||||||
|     "errorCheckingUpdates": "アップデート確認中のエラー", |     "errorCheckingUpdates": "アップデート確認中のエラー", | ||||||
|     "errorCheckingUpdatesNotifDescription": "バックグラウンドでのアップデート確認に失敗した際に表示される通知", |     "errorCheckingUpdatesNotifDescription": "バックグラウンドでのアップデート確認に失敗した際に表示される通知", | ||||||
|     "appsRemoved": "削除されたアプリ", |     "appsRemoved": "削除されたアプリ", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "WiFiを使用していない場合、バックグラウンドアップデートを無効にする", |     "bgUpdatesOnWiFiOnly": "WiFiを使用していない場合、バックグラウンドアップデートを無効にする", | ||||||
|     "autoSelectHighestVersionCode": "最も高いバージョンコードのAPKを自動で選択する", |     "autoSelectHighestVersionCode": "最も高いバージョンコードのAPKを自動で選択する", | ||||||
|     "versionExtractionRegEx": "バージョン抽出の正規表現", |     "versionExtractionRegEx": "バージョン抽出の正規表現", | ||||||
|  |     "trimVersionString": "正規表現でバージョン文字列をトリムする", | ||||||
|  |     "matchGroupToUseForX": "\"{}\"に使用するマッチしたグループ", | ||||||
|     "matchGroupToUse": "使用するマッチしたグループ", |     "matchGroupToUse": "使用するマッチしたグループ", | ||||||
|     "highlightTouchTargets": "目立たないタップ可能な対象をハイライトする", |     "highlightTouchTargets": "目立たないタップ可能な対象をハイライトする", | ||||||
|     "pickExportDir": "エクスポートディレクトリを選択", |     "pickExportDir": "エクスポートディレクトリを選択", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "固定されたAPKのURLをサポートする", |     "supportFixedAPKURL": "固定されたAPKのURLをサポートする", | ||||||
|     "selectX": "{} 選択", |     "selectX": "{} 選択", | ||||||
|     "parallelDownloads": "並行ダウンロードを許可する", |     "parallelDownloads": "並行ダウンロードを許可する", | ||||||
|     "installMethod": "インストール方法", |     "useShizuku": "ShizukuまたはSuiを使用してインストールする", | ||||||
|     "normal": "通常", |  | ||||||
|     "root": "Root", |  | ||||||
|     "shizukuBinderNotFound": "Shizukuが起動していません", |     "shizukuBinderNotFound": "Shizukuが起動していません", | ||||||
|  |     "shizukuOld": "古いShizukuのバージョン (<11) - アップデートしてください", | ||||||
|  |     "shizukuOldAndroidWithADB": "ShizukuがAndroid 8.1未満でADBを使用して動作しています - Androidをアップデートするか、代わりにSuiを使用してください", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Google Playをインストール元として設定する(Shizukuを使用する場合)", | ||||||
|     "useSystemFont": "システムフォントを使用する", |     "useSystemFont": "システムフォントを使用する", | ||||||
|     "systemFontError": "システムフォントの読み込みエラー: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "アプリのバージョンコードをOSで検出されたバージョンとして使用する", |     "useVersionCodeAsOSVersion": "アプリのバージョンコードをOSで検出されたバージョンとして使用する", | ||||||
|     "requestHeader": "リクエストヘッダー", |     "requestHeader": "リクエストヘッダー", | ||||||
|     "useLatestAssetDateAsReleaseDate": "最新のアセットアップロードをリリース日として使用する", |     "useLatestAssetDateAsReleaseDate": "最新のアセットアップロードをリリース日として使用する", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "APK を解析できませんでした(互換性がないか、部分的にダウンロードされています)。", |     "badDownload": "APK を解析できませんでした(互換性がないか、部分的にダウンロードされています)。", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "AppVerifierで新しいアプリを共有する(利用可能な場合)", |     "beforeNewInstallsShareToAppVerifier": "AppVerifierで新しいアプリを共有する(利用可能な場合)", | ||||||
|     "appVerifierInstructionToast": "AppVerifierに共有し、準備ができたらここに戻ってください。", |     "appVerifierInstructionToast": "AppVerifierに共有し、準備ができたらここに戻ってください。", | ||||||
|  |     "wiki": "ヘルプ/ウィキ", | ||||||
|  |     "crowdsourcedConfigsLabel": "クラウドソーシングによるアプリの設定(利用は自己責任で)", | ||||||
|  |     "allowInsecure": "安全でないHTTPリクエストを許可する", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "アプリを削除しますか?", |         "one": "アプリを削除しますか?", | ||||||
|         "other": "アプリを削除しますか?" |         "other": "アプリを削除しますか?" | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} とさらに {} 個のアプリがアップデートされました。", |         "one": "{} とさらに {} 個のアプリがアップデートされました。", | ||||||
|         "other": "{} とさらに {} 個のアプリがアップデートされました。" |         "other": "{} とさらに {} 個のアプリがアップデートされました。" | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "更新に失敗しました。", | ||||||
|  |         "other": "アプリのアップデートに失敗しました。" | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} とさらに 1 個のアプリがアップデートされた可能性があります。", |         "one": "{} とさらに 1 個のアプリがアップデートされた可能性があります。", | ||||||
|         "other": "{} とさらに {} 個のアプリがアップデートされた可能性があります。" |         "other": "{} とさらに {} 個のアプリがアップデートされた可能性があります。" | ||||||
|   | |||||||
| @@ -1,131 +1,135 @@ | |||||||
| { | { | ||||||
|     "invalidURLForSource": "Geen valide {} app URL", |     "invalidURLForSource": "Ongeldige app-URL voor {}", | ||||||
|     "noReleaseFound": "Kan geen geschikte release vinden", |     "noReleaseFound": "Geen geschikte release gevonden", | ||||||
|     "noVersionFound": "Kan de versie niet bepalen", |     "noVersionFound": "Geen versie kunnen bepalen", | ||||||
|     "urlMatchesNoSource": "URL komt niet overeen met bekende bron", |     "urlMatchesNoSource": "URL komt niet overeen met bekende bron", | ||||||
|     "cantInstallOlderVersion": "Kan geen oudere versie van de app installeren", |     "cantInstallOlderVersion": "Kan geen oudere versie van de app installeren", | ||||||
|     "appIdMismatch": "Gedownloade pakket-ID komt niet overeen met de bestaande app-ID", |     "appIdMismatch": "Gedownload pakket-ID komt niet overeen met de bestaande app-ID", | ||||||
|     "functionNotImplemented": "Deze class heeft deze functie niet geïmplementeerd.", |     "functionNotImplemented": "Deze klasse heeft deze functie niet geïmplementeerd.", | ||||||
|     "placeholder": "Plaatshouder", |     "placeholder": "Dummy", | ||||||
|     "someErrors": "Er zijn enkele fouten opgetreden", |     "someErrors": "Er zijn enkele fouten opgetreden", | ||||||
|     "unexpectedError": "Onverwachte fout", |     "unexpectedError": "Onverwachte fout", | ||||||
|     "ok": "Ok", |     "ok": "Oké", | ||||||
|     "and": "en", |     "and": "en", | ||||||
|     "githubPATLabel": "GitHub Personal Access Token\n(Verhoogt limiet aantal verzoeken)", |     "githubPATLabel": "GitHub Personal Access Token\n(Verhoogt limiet aantal verzoeken)", | ||||||
|     "includePrereleases": "Bevat prereleases", |     "includePrereleases": "Inclusief pre-releases", | ||||||
|     "fallbackToOlderReleases": "Terugvallen op oudere releases", |     "fallbackToOlderReleases": "Terugvallen op oudere releases", | ||||||
|     "filterReleaseTitlesByRegEx": "Filter release-titels met reguliere expressies.", |     "filterReleaseTitlesByRegEx": "Release-titels filteren met reguliere expressies.", | ||||||
|     "invalidRegEx": "Ongeldige reguliere expressie", |     "invalidRegEx": "Ongeldige reguliere expressie", | ||||||
|     "noDescription": "Geen omschrijving", |     "noDescription": "Geen omschrijving", | ||||||
|     "cancel": "Annuleer", |     "cancel": "Annuleren", | ||||||
|     "continue": "Ga verder", |     "continue": "Doorgaan", | ||||||
|     "requiredInBrackets": "(Verplicht)", |     "requiredInBrackets": "(Verplicht)", | ||||||
|     "dropdownNoOptsError": "FOUTMELDING: DROPDOWN MOET TENMINSTE ÉÉN OPT HEBBEN", |     "dropdownNoOptsError": "FOUTMELDING: UITKLAPMENU MOET TENMINSTE EEN OPT HEBBEN", | ||||||
|     "colour": "Kleur", |     "colour": "Kleur", | ||||||
|  |     "standard": "Standaard", | ||||||
|  |     "custom": "Aangepast", | ||||||
|  |     "useMaterialYou": "Material You gebruiken", | ||||||
|     "githubStarredRepos": "GitHub-repo's met ster", |     "githubStarredRepos": "GitHub-repo's met ster", | ||||||
|     "uname": "Gebruikersnaam", |     "uname": "Gebruikersnaam", | ||||||
|     "wrongArgNum": "Onjuist aantal argumenten verstrekt.", |     "wrongArgNum": "Incorrect aantal argumenten.", | ||||||
|     "xIsTrackOnly": "{} is alleen tracken", |     "xIsTrackOnly": "{} is 'Alleen volgen'", | ||||||
|     "source": "Bron", |     "source": "Bron", | ||||||
|     "app": "App", |     "app": "App", | ||||||
|     "appsFromSourceAreTrackOnly": "Apps van deze bron zijn 'Track-Only'.", |     "appsFromSourceAreTrackOnly": "Apps van deze bron zijn 'Alleen volgen'.", | ||||||
|     "youPickedTrackOnly": "Je hebt de 'Track-Only' optie geselecteerd.", |     "youPickedTrackOnly": "De optie 'Alleen volgen' is geselecteerd.", | ||||||
|     "trackOnlyAppDescription": "De app zal worden gevolgd voor updates, maar Obtainium zal niet in staat zijn om deze te downloaden of te installeren.", |     "trackOnlyAppDescription": "De app zal worden gevolgd voor updates, maar Obtainium zal niet in staat zijn om deze te downloaden of te installeren.", | ||||||
|     "cancelled": "Geannuleerd", |     "cancelled": "Geannuleerd", | ||||||
|     "appAlreadyAdded": "App al toegevoegd", |     "appAlreadyAdded": "App reeds toegevoegd", | ||||||
|     "alreadyUpToDateQuestion": "Is de app al up-to-date?", |     "alreadyUpToDateQuestion": "App al bijgewerkt?", | ||||||
|     "addApp": "App toevoegen", |     "addApp": "App toevoegen", | ||||||
|     "appSourceURL": "App bron URL", |     "appSourceURL": "App-bron URL", | ||||||
|     "error": "Foutmelding", |     "error": "Foutmelding", | ||||||
|     "add": "Toevoegen", |     "add": "Toevoegen", | ||||||
|     "searchSomeSourcesLabel": "Zoeken (Alleen sommige bronnen)", |     "searchSomeSourcesLabel": "Zoeken (sommige bronnen)", | ||||||
|     "search": "Zoeken", |     "search": "Zoeken", | ||||||
|     "additionalOptsFor": "Aanvullende opties voor {}", |     "additionalOptsFor": "Aanvullende opties voor {}", | ||||||
|     "supportedSources": "Ondersteunde bronnen", |     "supportedSources": "Ondersteunde bronnen", | ||||||
|     "trackOnlyInBrackets": "(Alleen track)", |     "trackOnlyInBrackets": "(Alleen volgen)", | ||||||
|     "searchableInBrackets": "(Doorzoekbaar)", |     "searchableInBrackets": "(Doorzoekbaar)", | ||||||
|     "appsString": "Apps", |     "appsString": "Apps", | ||||||
|     "noApps": "Geen Apps", |     "noApps": "Geen Apps", | ||||||
|     "noAppsForFilter": "Geen Apps voor filter", |     "noAppsForFilter": "Geen Apps voor filter", | ||||||
|     "byX": "Door {}", |     "byX": "Door {}", | ||||||
|     "percentProgress": "Vooruitgang: {}%", |     "percentProgress": "Voortgang: {}%", | ||||||
|     "pleaseWait": "Even geduld", |     "pleaseWait": "Even geduld", | ||||||
|     "updateAvailable": "Update beschikbaar", |     "updateAvailable": "Update beschikbaar", | ||||||
|     "notInstalled": "Niet geinstalleerd", |     "notInstalled": "Niet geinstalleerd", | ||||||
|     "pseudoVersion": "pseudo-versie", |     "pseudoVersion": "pseudo-versie", | ||||||
|     "selectAll": "Selecteer alles", |     "selectAll": "Alles selecteren", | ||||||
|     "deselectX": "Deselecteer {}", |     "deselectX": "Selectie opheffen {}", | ||||||
|     "xWillBeRemovedButRemainInstalled": "{} zal worden verwijderd uit Obtainium, maar blijft geïnstalleerd op het apparaat.", |     "xWillBeRemovedButRemainInstalled": "{} zal worden gewist uit Obtainium, maar blijft geïnstalleerd op het apparaat.", | ||||||
|     "removeSelectedAppsQuestion": "Geselecteerde apps verwijderen??", |     "removeSelectedAppsQuestion": "Geselecteerde apps verwijderen?", | ||||||
|     "removeSelectedApps": "Geselecteerde apps verwijderen", |     "removeSelectedApps": "Geselecteerde apps verwijderen", | ||||||
|     "updateX": "Update {}", |     "updateX": "{} bijwerken", | ||||||
|     "installX": "Installeer {}", |     "installX": "{} installeren", | ||||||
|     "markXTrackOnlyAsUpdated": "Markeer {}\n(Track-Only)\nals up-to-date", |     "markXTrackOnlyAsUpdated": "{}\n(Alleen volgen)\nmarkeren als bijgewerkt", | ||||||
|     "changeX": "Verander {}", |     "changeX": "{} wijzigen", | ||||||
|     "installUpdateApps": "Installeer/Update apps", |     "installUpdateApps": "Apps installeren/bijwerken", | ||||||
|     "installUpdateSelectedApps": "Installeer/Update geselecteerde apps", |     "installUpdateSelectedApps": "Geselecteerde apps installeren/bijwerken", | ||||||
|     "markXSelectedAppsAsUpdated": "{} geselecteerde apps markeren als up-to-date?", |     "markXSelectedAppsAsUpdated": "{} geselecteerde apps markeren als bijgewerkt?", | ||||||
|     "no": "Nee", |     "no": "Nee", | ||||||
|     "yes": "Ja", |     "yes": "Ja", | ||||||
|     "markSelectedAppsUpdated": "Markeer geselecteerde aps als up-to-date", |     "markSelectedAppsUpdated": "Geselecteerde apps markeren als bijgewerkt", | ||||||
|     "pinToTop": "Vastzetten aan de bovenkant", |     "pinToTop": "Bovenaan plaatsen", | ||||||
|     "unpinFromTop": "Losmaken van de bovenkant", |     "unpinFromTop": "Bovenaan wegnemen", | ||||||
|     "resetInstallStatusForSelectedAppsQuestion": "Installatiestatus resetten voor geselecteerde apps?", |     "resetInstallStatusForSelectedAppsQuestion": "Installatiestatus herstellen voor geselecteerde apps?", | ||||||
|     "installStatusOfXWillBeResetExplanation": "De installatiestatus van alle geselecteerde apps zal worden gereset.\n\nDit kan helpen wanneer de versie van de app die in Obtainium wordt weergegeven onjuist is vanwege mislukte updates of andere problemen.", |     "installStatusOfXWillBeResetExplanation": "De installatiestatus van alle geselecteerde apps zal worden hersteld.\n\nDit kan helpen wanneer de versie van de app die in Obtainium wordt weergegeven onjuist is vanwege mislukte updates of andere problemen.", | ||||||
|     "customLinkMessage": "Deze links werken op apparaten waarop Obtainium is geïnstalleerd", |     "customLinkMessage": "Deze koppelingen werken op apparaten waarop Obtainium is geïnstalleerd", | ||||||
|     "shareAppConfigLinks": "App-configuratie delen als HTML-link", |     "shareAppConfigLinks": "App-configuratie delen als HTML-link", | ||||||
|     "shareSelectedAppURLs": "Deel geselecteerde app URL's", |     "shareSelectedAppURLs": "Geselecteerde app-URL's delen", | ||||||
|     "resetInstallStatus": "Reset installatiestatus", |     "resetInstallStatus": "Installatiestatus herstellen", | ||||||
|     "more": "Meer", |     "more": "Meer", | ||||||
|     "removeOutdatedFilter": "Verwijder out-of-date app filter", |     "removeOutdatedFilter": "Verouderde apps-filter verwijderen", | ||||||
|     "showOutdatedOnly": "Toon alleen out-of-date apps", |     "showOutdatedOnly": "Alleen verouderde apps weergeven", | ||||||
|     "filter": "Filter", |     "filter": "Filteren", | ||||||
|     "filterApps": "Filter apps", |     "filterApps": "Apps filteren", | ||||||
|     "appName": "App naam", |     "appName": "App-naam", | ||||||
|     "author": "Auteur", |     "author": "Auteur", | ||||||
|     "upToDateApps": "Up-to-date apps", |     "upToDateApps": "Bijgewerkte apps", | ||||||
|     "nonInstalledApps": "Niet-geïnstalleerde apps", |     "nonInstalledApps": "Niet-geïnstalleerde apps", | ||||||
|     "importExport": "Importeren/Exporteren", |     "importExport": "Importeren/exporteren", | ||||||
|     "settings": "Instellingen", |     "settings": "Instellingen", | ||||||
|     "exportedTo": "Geëxporteerd naar {}", |     "exportedTo": "Geëxporteerd naar {}", | ||||||
|     "obtainiumExport": "Obtainium export", |     "obtainiumExport": "Obtainium export", | ||||||
|     "invalidInput": "Ongeldige invoer", |     "invalidInput": "Ongeldige invoer", | ||||||
|     "importedX": "Geïmporteerd {}", |     "importedX": "{} geïmporteerd", | ||||||
|     "obtainiumImport": "Obtainium import", |     "obtainiumImport": "Obtainium import", | ||||||
|     "importFromURLList": "Importeer van URL-lijsten", |     "importFromURLList": "Importeren van URL-lijsten", | ||||||
|     "searchQuery": "Zoekopdracht", |     "searchQuery": "Zoekopdracht", | ||||||
|     "appURLList": "App URL-lijst", |     "appURLList": "App URL-lijst", | ||||||
|     "line": "Lijn", |     "line": "Regel", | ||||||
|     "searchX": "Zoek {}", |     "searchX": "{} zoeken", | ||||||
|     "noResults": "Geen resultaten gevonden", |     "noResults": "Geen resultaten gevonden", | ||||||
|     "importX": "Importeer {}", |     "importX": "{} importeren", | ||||||
|     "importedAppsIdDisclaimer": "Geïmporteerde apps kunnen mogelijk onjuist worden weergegeven als \"Niet geïnstalleerd\".\nOm dit op te lossen, herinstalleer ze via Obtainium.\nDit zou geen invloed moeten hebben op app-gegevens.\n\nDit heeft alleen invloed op URL- en importmethoden van derden.", |     "importedAppsIdDisclaimer": "Geïmporteerde apps kunnen mogelijk onjuist worden weergegeven als \"Niet geïnstalleerd\".\nOm dit op te lossen, installeer deze opnieuw via Obtainium.\nDit zou geen invloed moeten hebben op app-gegevens.\n\nDit heeft alleen invloed op URL- en importmethoden van derden.", | ||||||
|     "importErrors": "Import foutmeldingen", |     "importErrors": "Fouten bij het importeren", | ||||||
|     "importedXOfYApps": "{} van {} apps geïmporteerd.", |     "importedXOfYApps": "{} van {} apps geïmporteerd.", | ||||||
|     "followingURLsHadErrors": "De volgende URL's bevatten fouten:", |     "followingURLsHadErrors": "De volgende URL's bevatten fouten:", | ||||||
|     "selectURL": "Selecteer URL", |     "selectURL": "URL selecteren", | ||||||
|     "selectURLs": "Selecteer URL's", |     "selectURLs": "URL's selecteren", | ||||||
|     "pick": "Kies", |     "pick": "Kiezen", | ||||||
|     "theme": "Thema", |     "theme": "Thema", | ||||||
|     "dark": "Donker", |     "dark": "Donker", | ||||||
|     "light": "Licht", |     "light": "Licht", | ||||||
|     "followSystem": "Volg systeem", |     "followSystem": "Systeem volgen", | ||||||
|     "useBlackTheme": "Gebruik zwart thema", |     "followSystemThemeExplanation": "Het volgen van het systeemthema is alleen mogelijk met applicaties van derden", | ||||||
|     "appSortBy": "App sorteren op", |     "useBlackTheme": "Zwart thema gebruiken", | ||||||
|  |     "appSortBy": "Sortering", | ||||||
|     "authorName": "Auteur/Naam", |     "authorName": "Auteur/Naam", | ||||||
|     "nameAuthor": "Naam/Auteur", |     "nameAuthor": "Naam/Auteur", | ||||||
|     "asAdded": "Zoals toegevoegd", |     "asAdded": "Datum toegevoegd", | ||||||
|     "appSortOrder": "App sorteervolgorde", |     "appSortOrder": "Volgorde", | ||||||
|     "ascending": "Oplopend", |     "ascending": "Oplopend", | ||||||
|     "descending": "Aflopend", |     "descending": "Aflopend", | ||||||
|     "bgUpdateCheckInterval": "Frequentie voor achtergrondupdatecontrole", |     "bgUpdateCheckInterval": "Frequentie voor achtergrond-updatecontrole", | ||||||
|     "neverManualOnly": "Nooit - Alleen handmatig", |     "neverManualOnly": "Nooit - Alleen handmatig", | ||||||
|     "appearance": "Weergave", |     "appearance": "Weergave", | ||||||
|     "showWebInAppView": "Toon de bronwebpagina in app-weergave", |     "showWebInAppView": "Bron-webpagina weergeven in app-weergave", | ||||||
|     "pinUpdates": "Updates bovenaan in de apps-weergave vastpinnen", |     "pinUpdates": "Updates bovenaan plaatsen in de apps-weergave", | ||||||
|     "updates": "Updates", |     "updates": "Updates", | ||||||
|     "sourceSpecific": "Bron-specifiek", |     "sourceSpecific": "Bron-specifiek", | ||||||
|     "appSource": "App bron", |     "appSource": "App-bron", | ||||||
|     "noLogs": "Geen logs", |     "noLogs": "Geen logs", | ||||||
|     "appLogs": "App logs", |     "appLogs": "App logs", | ||||||
|     "close": "Sluiten", |     "close": "Sluiten", | ||||||
| @@ -133,30 +137,32 @@ | |||||||
|     "appNotFound": "App niet gevonden", |     "appNotFound": "App niet gevonden", | ||||||
|     "obtainiumExportHyphenatedLowercase": "obtainium-export", |     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||||
|     "pickAnAPK": "Kies een APK", |     "pickAnAPK": "Kies een APK", | ||||||
|     "appHasMoreThanOnePackage": "{} heeft meer dan één package:", |     "appHasMoreThanOnePackage": "{} biedt verschillende pakketten:", | ||||||
|     "deviceSupportsXArch": "Jouw apparaat support de {} CPU-architectuur.", |     "deviceSupportsXArch": "Dit apparaat ondersteunt de {} CPU-architectuur.", | ||||||
|     "deviceSupportsFollowingArchs": "Je apparaat ondersteunt de volgende CPU-architecturen:", |     "deviceSupportsFollowingArchs": "Dit apparaat ondersteunt de volgende CPU-architecturen:", | ||||||
|     "warning": "Waarschuwing", |     "warning": "Waarschuwing", | ||||||
|     "sourceIsXButPackageFromYPrompt": "De appbron is '{}' maar de release package komt van '{}'. Doorgaan?", |     "sourceIsXButPackageFromYPrompt": "De app-bron is '{}' maar het release-pakket komt van '{}'. Doorgaan?", | ||||||
|     "updatesAvailable": "Updates beschikbaar", |     "updatesAvailable": "Updates beschikbaar", | ||||||
|     "updatesAvailableNotifDescription": "Stelt de gebruiker op de hoogte dat er updates beschikbaar zijn voor één of meer apps die worden bijgehouden door Obtainium.", |     "updatesAvailableNotifDescription": "Stelt de gebruiker op de hoogte dat er updates beschikbaar zijn voor een of meer apps die worden bijgehouden door Obtainium.", | ||||||
|     "noNewUpdates": "Geen nieuwe updates.", |     "noNewUpdates": "Geen nieuwe updates.", | ||||||
|     "xHasAnUpdate": "{} heeft een update.", |     "xHasAnUpdate": "{} heeft een update.", | ||||||
|     "appsUpdated": "Apps bijgewerkt", |     "appsUpdated": "Apps bijgewerkt", | ||||||
|  |     "appsNotUpdated": "Applicaties konden niet worden bijgewerkt", | ||||||
|     "appsUpdatedNotifDescription": "Stelt de gebruiker op de hoogte dat updates voor één of meer apps in de achtergrond zijn toegepast.", |     "appsUpdatedNotifDescription": "Stelt de gebruiker op de hoogte dat updates voor één of meer apps in de achtergrond zijn toegepast.", | ||||||
|     "xWasUpdatedToY": "{} is bijgewerkt naar {}.", |     "xWasUpdatedToY": "{} is bijgewerkt naar {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Het bijwerken van {} naar {} is mislukt.", | ||||||
|     "errorCheckingUpdates": "Fout bij het controleren op updates", |     "errorCheckingUpdates": "Fout bij het controleren op updates", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Een melding die verschijnt wanneer het controleren op updates in de achtergrond mislukt", |     "errorCheckingUpdatesNotifDescription": "Een melding die verschijnt wanneer de achtergrondcontrole op updates mislukt", | ||||||
|     "appsRemoved": "Apps verwijderd", |     "appsRemoved": "Apps verwijderd", | ||||||
|     "appsRemovedNotifDescription": "Stelt de gebruiker op de hoogte dat één of meer apps zijn verwijderd vanwege fouten tijdens het laden ervan", |     "appsRemovedNotifDescription": "Stelt de gebruiker op de hoogte dat een of meer apps zijn verwijderd vanwege fouten tijdens het laden", | ||||||
|     "xWasRemovedDueToErrorY": "{} is verwijderd vanwege deze foutmelding: {}", |     "xWasRemovedDueToErrorY": "{} is verwijderd vanwege de fout: {}", | ||||||
|     "completeAppInstallation": "Complete app installatie", |     "completeAppInstallation": "App-installatie voltooien", | ||||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium moet geopend zijn om apps te installeren", |     "obtainiumMustBeOpenToInstallApps": "Obtainium moet geopend zijn om apps te installeren", | ||||||
|     "completeAppInstallationNotifDescription": "Vraagt de gebruiker om terug te keren naar Obtainium om de installatie van een app af te ronden", |     "completeAppInstallationNotifDescription": "Vraagt de gebruiker om terug te keren naar Obtainium om de installatie van een app af te ronden", | ||||||
|     "checkingForUpdates": "Controleren op updates", |     "checkingForUpdates": "Controleren op updates", | ||||||
|     "checkingForUpdatesNotifDescription": "Tijdelijke melding die verschijnt tijdens het controleren op updates", |     "checkingForUpdatesNotifDescription": "Tijdelijke melding die verschijnt tijdens het controleren op updates", | ||||||
|     "pleaseAllowInstallPerm": "Sta Obtainium toe om apps te installeren", |     "pleaseAllowInstallPerm": "Toestaan dat Obtainium apps installeert", | ||||||
|     "trackOnly": "Alleen track", |     "trackOnly": "'Alleen volgen'", | ||||||
|     "errorWithHttpStatusCode": "Foutmelding {}", |     "errorWithHttpStatusCode": "Foutmelding {}", | ||||||
|     "versionCorrectionDisabled": "Versiecorrectie uitgeschakeld (de plug-in lijkt niet te werken)", |     "versionCorrectionDisabled": "Versiecorrectie uitgeschakeld (de plug-in lijkt niet te werken)", | ||||||
|     "unknown": "Onbekend", |     "unknown": "Onbekend", | ||||||
| @@ -166,25 +172,25 @@ | |||||||
|     "installedVersionX": "Geïnstalleerde versie: {}", |     "installedVersionX": "Geïnstalleerde versie: {}", | ||||||
|     "lastUpdateCheckX": "Laatste updatecontrole: {}", |     "lastUpdateCheckX": "Laatste updatecontrole: {}", | ||||||
|     "remove": "Verwijderen", |     "remove": "Verwijderen", | ||||||
|     "yesMarkUpdated": "Ja, markeer als bijgewerkt", |     "yesMarkUpdated": "Ja, markeren als bijgewerkt", | ||||||
|     "fdroid": "F-Droid-ambtenaar", |     "fdroid": "F-Droid (Officieel)", | ||||||
|     "appIdOrName": "App ID of naam", |     "appIdOrName": "App-ID of naam", | ||||||
|     "appId": "App-ID", |     "appId": "App-ID", | ||||||
|     "appWithIdOrNameNotFound": "Er werd geen app gevonden met dat ID of die naam", |     "appWithIdOrNameNotFound": "Er is geen app gevonden met dat ID of die naam", | ||||||
|     "reposHaveMultipleApps": "Repositories kunnen meerdere apps bevatten", |     "reposHaveMultipleApps": "Repositories kunnen meerdere apps bevatten", | ||||||
|     "fdroidThirdPartyRepo": "F-Droid Repository van derden", |     "fdroidThirdPartyRepo": "F-Droid Repository voor derden", | ||||||
|     "steamMobile": "Stoommobiel", |     "steamMobile": "Steam Mobile", | ||||||
|     "steamChat": "Steamchat", |     "steamChat": "Steam Chat", | ||||||
|     "install": "Installeren", |     "install": "Installeren", | ||||||
|     "markInstalled": "Als geïnstalleerd markere", |     "markInstalled": "Als geïnstalleerd markeren", | ||||||
|     "update": "Update", |     "update": "Bijwerken", | ||||||
|     "markUpdated": "Markeren als bijgewerkt", |     "markUpdated": "Als bijgewerkt markeren", | ||||||
|     "additionalOptions": "Aanvullende opties", |     "additionalOptions": "Aanvullende opties", | ||||||
|     "disableVersionDetection": "Versieherkenning uitschakelen", |     "disableVersionDetection": "Versieherkenning uitschakelen", | ||||||
|     "noVersionDetectionExplanation": "Deze optie moet alleen worden gebruikt voor apps waar versieherkenning niet correct werkt.", |     "noVersionDetectionExplanation": "Deze optie moet alleen worden gebruikt voor apps waar versieherkenning niet correct werkt.", | ||||||
|     "downloadingX": "Downloaden {}", |     "downloadingX": "{} downloaden", | ||||||
|     "downloadX": "Downloaden", |     "downloadX": "Downloaden", | ||||||
|     "downloadedX": "Gedownload {}", |     "downloadedX": "{} gedownload", | ||||||
|     "releaseAsset": "Release Activa", |     "releaseAsset": "Release Activa", | ||||||
|     "downloadNotifDescription": "Stelt de gebruiker op de hoogte van de voortgang bij het downloaden van een app", |     "downloadNotifDescription": "Stelt de gebruiker op de hoogte van de voortgang bij het downloaden van een app", | ||||||
|     "noAPKFound": "Geen APK gevonden", |     "noAPKFound": "Geen APK gevonden", | ||||||
| @@ -197,100 +203,102 @@ | |||||||
|     "deleteCategoriesQuestion": "Categorieën verwijderen?", |     "deleteCategoriesQuestion": "Categorieën verwijderen?", | ||||||
|     "categoryDeleteWarning": "Alle apps in verwijderde categorieën worden teruggezet naar 'ongecategoriseerd'.", |     "categoryDeleteWarning": "Alle apps in verwijderde categorieën worden teruggezet naar 'ongecategoriseerd'.", | ||||||
|     "addCategory": "Categorie toevoegen", |     "addCategory": "Categorie toevoegen", | ||||||
|     "label": "Etiket", |     "label": "Label", | ||||||
|     "language": "Taal", |     "language": "Taal", | ||||||
|     "copiedToClipboard": "Gekopieerd naar klembord", |     "copiedToClipboard": "Gekopieerd naar klembord", | ||||||
|     "storagePermissionDenied": "Toegang tot opslag geweigerd", |     "storagePermissionDenied": "Toegang tot opslag geweigerd", | ||||||
|     "selectedCategorizeWarning": "Dit zal eventuele bestaande categorie-instellingen voor de geselecteerde apps vervangen.", |     "selectedCategorizeWarning": "Dit zal eventuele bestaande categorie-instellingen voor de geselecteerde apps vervangen.", | ||||||
|     "filterAPKsByRegEx": "Filter APK's op reguliere expressie", |     "filterAPKsByRegEx": "APK's flteren met reguliere expressie", | ||||||
|     "removeFromObtainium": "Verwijder van Obtainium", |     "removeFromObtainium": "Uit Obtainium verwijderen", | ||||||
|     "uninstallFromDevice": "Verwijder van apparaat", |     "uninstallFromDevice": "Van apparaat verwijderen", | ||||||
|     "onlyWorksWithNonVersionDetectApps": "Werkt alleen voor apps waarbij versieherkenning is uitgeschakeld.", |     "onlyWorksWithNonVersionDetectApps": "Werkt alleen voor apps waarbij versieherkenning is uitgeschakeld.", | ||||||
|     "releaseDateAsVersion": "Gebruik de releasedatum als versie", |     "releaseDateAsVersion": "Releasedatum als versie gebruiken", | ||||||
|     "releaseDateAsVersionExplanation": "Deze optie moet alleen worden gebruikt voor apps waar versieherkenning niet correct werkt, maar waar wel een releasedatum beschikbaar is.", |     "releaseDateAsVersionExplanation": "Deze optie moet alleen worden gebruikt voor apps waar versieherkenning niet correct werkt, maar waar wel een releasedatum beschikbaar is.", | ||||||
|     "changes": "Veranderingen", |     "changes": "Aanpassingen", | ||||||
|     "releaseDate": "Releasedatum", |     "releaseDate": "Releasedatum", | ||||||
|     "importFromURLsInFile": "Importeren vanaf URL's in een bestand (zoals OPML)", |     "importFromURLsInFile": "Importeren vanaf URL's in een bestand (zoals OPML)", | ||||||
|     "versionDetectionExplanation": "Versiereeks afstemmen met versie gedetecteerd door besturingssysteem", |     "versionDetectionExplanation": "Versiereeks afstemmen met versie gedetecteerd door besturingssysteem", | ||||||
|     "versionDetection": "Versieherkenning", |     "versionDetection": "Versieherkenning", | ||||||
|     "standardVersionDetection": "Standaard versieherkenning", |     "standardVersionDetection": "Standaard versieherkenning", | ||||||
|     "groupByCategory": "Groepeer op categorie", |     "groupByCategory": "Groeperen op categorie", | ||||||
|     "autoApkFilterByArch": "Poging om APK's te filteren op CPU-architectuur indien mogelijk", |     "autoApkFilterByArch": "Probeer APK's te filteren op CPU-architectuur, indien mogelijk", | ||||||
|     "overrideSource": "Bron overschrijven", |     "overrideSource": "Bron overschrijven", | ||||||
|     "dontShowAgain": "Laat dit niet meer zien", |     "dontShowAgain": "Laat dit niet meer zien", | ||||||
|     "dontShowTrackOnlyWarnings": "Geen waarschuwingen voor 'Track-Only' weergeven", |     "dontShowTrackOnlyWarnings": "Geen waarschuwingen weergeven voor 'Alleen volgen'", | ||||||
|     "dontShowAPKOriginWarnings": "APK-herkomstwaarschuwingen niet weergeven", |     "dontShowAPKOriginWarnings": "Geen waarschuwingen weergeven voor APK-herkomst", | ||||||
|     "moveNonInstalledAppsToBottom": "Verplaats niet-geïnstalleerde apps naar de onderkant van de apps-weergave", |     "moveNonInstalledAppsToBottom": "Niet-geïnstalleerde apps onderaan de apps-lijst plaatsen", | ||||||
|     "gitlabPATLabel": "GitLab persoonlijk toegangskenmerk", |     "gitlabPATLabel": "GitLab persoonlijk toegangskenmerk", | ||||||
|     "about": "Over", |     "about": "Over", | ||||||
|     "requiresCredentialsInSettings": "{}: Dit vereist aanvullende referenties (in Instellingen)", |     "requiresCredentialsInSettings": "{} vereist aanvullende referenties (in Instellingen)", | ||||||
|     "checkOnStart": "Controleren op updates bij opstarten", |     "checkOnStart": "Bij opstarten op updates controleren", | ||||||
|     "tryInferAppIdFromCode": "Probeer de app-ID af te leiden uit de broncode", |     "tryInferAppIdFromCode": "Probeer de app-ID af te leiden uit de broncode", | ||||||
|     "removeOnExternalUninstall": "Automatisch extern verwijderde apps verwijderen", |     "removeOnExternalUninstall": "Extern verwijderde apps automatisch verwijderen", | ||||||
|     "pickHighestVersionCode": "Automatisch de APK met de hoogste versiecode selecteren", |     "pickHighestVersionCode": "De APK met de hoogste versiecode automatisch selecteren", | ||||||
|     "checkUpdateOnDetailPage": "Controleren op updates bij het openen van een app-detailpagina", |     "checkUpdateOnDetailPage": "Controleren op updates bij het openen van een app-detailpagina", | ||||||
|     "disablePageTransitions": "Schakel overgangsanimaties tussen pagina's uit", |     "disablePageTransitions": "Overgangsanimaties tussen pagina's uitschakelen", | ||||||
|     "reversePageTransitions": "Omgekeerde overgangsanimaties tussen pagina's", |     "reversePageTransitions": "Overgangsanimaties tussen pagina's herstellen", | ||||||
|     "minStarCount": "Minimale Github Stars", |     "minStarCount": "Minimum Github Stars", | ||||||
|     "addInfoBelow": "Voeg deze informatie hieronder toe.", |     "addInfoBelow": "Deze informatie hieronder toevoegen.", | ||||||
|     "addInfoInSettings": "Voeg deze informatie toe in de instellingen.", |     "addInfoInSettings": "Deze informatie toevoegen in de instellingen.", | ||||||
|     "githubSourceNote": "Beperkingen van GitHub kunnen worden vermeden door het gebruik van een API-sleutel.", |     "githubSourceNote": "Beperkingen van GitHub kunnen worden vermeden door het gebruik van een API-sleutel.", | ||||||
|     "sortByLastLinkSegment": "Sorteren op alleen het laatste segment van de link", |     "sortByLastLinkSegment": "Alleen sorteren op het laatste segment van de link", | ||||||
|     "filterReleaseNotesByRegEx": "Filter release-opmerkingen met een reguliere expressie.", |     "filterReleaseNotesByRegEx": "Release-opmerkingen fiteren met een reguliere expressie.", | ||||||
|     "customLinkFilterRegex": "Aangepaste APK-linkfilter met een reguliere expressie (Standaard '.apk$').", |     "customLinkFilterRegex": "Aangepaste APK-links filteren met een reguliere expressie (Standaard '.apk$').", | ||||||
|     "appsPossiblyUpdated": "Poging tot app-updates", |     "appsPossiblyUpdated": "Pogingen tot app-updates", | ||||||
|     "appsPossiblyUpdatedNotifDescription": "Stelt de gebruiker op de hoogte dat updates voor één of meer apps mogelijk in de achtergrond zijn toegepast", |     "appsPossiblyUpdatedNotifDescription": "Stelt de gebruiker op de hoogte dat updates voor één of meer apps mogelijk in de achtergrond zijn toegepast", | ||||||
|     "xWasPossiblyUpdatedToY": "{} mogelijk bijgewerkt naar {}.", |     "xWasPossiblyUpdatedToY": "{} kan bijgewerkt zijn naar {}.", | ||||||
|     "enableBackgroundUpdates": "Achtergrondupdates inschakelen", |     "enableBackgroundUpdates": "Achtergrond-updates inschakelen", | ||||||
|     "backgroundUpdateReqsExplanation": "Achtergrondupdates zijn mogelijk niet voor alle apps mogelijk.", |     "backgroundUpdateReqsExplanation": "Achtergrond-updates zijn niet voor alle apps mogelijk.", | ||||||
|     "backgroundUpdateLimitsExplanation": "Het succes van een installatie in de achtergrond kan alleen worden bepaald wanneer Obtainium is geopend.", |     "backgroundUpdateLimitsExplanation": "Het succes van een installatie in de achtergrond kan alleen worden bepaald wanneer Obtainium is geopend.", | ||||||
|     "verifyLatestTag": "Verifieer de 'Laatste'-tag", |     "verifyLatestTag": "Het label 'Laatste' verifiëren", | ||||||
|     "intermediateLinkRegex": "Filter voor een 'Intermediaire' link om te bezoeken", |     "intermediateLinkRegex": "Filteren op een 'Intermediaire' link om te bezoeken", | ||||||
|     "filterByLinkText": "Links filteren op linktekst", |     "filterByLinkText": "Links filteren op linktekst", | ||||||
|     "intermediateLinkNotFound": "Tussenliggende link niet gevonden", |     "intermediateLinkNotFound": "Intermediaire link niet gevonden", | ||||||
|     "intermediateLink": "Intermediaire link", |     "intermediateLink": "Intermediaire link", | ||||||
|     "exemptFromBackgroundUpdates": "Vrijgesteld van achtergrondupdates (indien ingeschakeld)", |     "exemptFromBackgroundUpdates": "Vrijgesteld van achtergrond-updates (indien ingeschakeld)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Achtergrondupdates uitschakelen wanneer niet verbonden met WiFi", |     "bgUpdatesOnWiFiOnly": "Achtergrond-updates uitschakelen wanneer niet verbonden met WiFi", | ||||||
|     "autoSelectHighestVersionCode": "Automatisch de APK met de hoogste versiecode selecteren", |     "autoSelectHighestVersionCode": "De APK met de hoogste versiecode automatisch selecteren", | ||||||
|     "versionExtractionRegEx": "Reguliere expressie voor versie-extractie", |     "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", |     "matchGroupToUse": "Overeenkomende groep om te gebruiken voor de reguliere expressie voor versie-extractie", | ||||||
|     "highlightTouchTargets": "Markeer minder voor de hand liggende aanraakdoelen.", |     "highlightTouchTargets": "Minder voor de hand liggende aanraakdoelen markeren.", | ||||||
|     "pickExportDir": "Kies de exportmap", |     "pickExportDir": "Kies de exportmap", | ||||||
|     "autoExportOnChanges": "Automatisch exporteren bij wijzigingen", |     "autoExportOnChanges": "Bij wijzigingen automatisch exporteren", | ||||||
|     "includeSettings": "Instellingen opnemen", |     "includeSettings": "Instellingen opnemen", | ||||||
|     "filterVersionsByRegEx": "Filter versies met een reguliere expressie", |     "filterVersionsByRegEx": "Versies met een reguliere expressie filteren", | ||||||
|     "trySelectingSuggestedVersionCode": "Probeer de voorgestelde versiecode APK te selecteren", |     "trySelectingSuggestedVersionCode": "Probeer de voorgestelde versiecode APK te selecteren", | ||||||
|     "dontSortReleasesList": "Volgorde van releases behouden vanuit de API", |     "dontSortReleasesList": "Volgorde van releases behouden vanuit de API", | ||||||
|     "reverseSort": "Sortering omkeren", |     "reverseSort": "Omgekeerde sortering", | ||||||
|     "takeFirstLink": "Neem de eerste link", |     "takeFirstLink": "Neem de eerste link", | ||||||
|     "skipSort": "Sorteren overslaan", |     "skipSort": "Sortering overslaan", | ||||||
|     "debugMenu": "Debug menu", |     "debugMenu": "Debug-menu", | ||||||
|     "bgTaskStarted": "Achtergrondtaak gestart - controleer de logs.", |     "bgTaskStarted": "Achtergrondtaak gestart - controleer de logs.", | ||||||
|     "runBgCheckNow": "Voer nu een achtergrondupdatecontrole uit", |     "runBgCheckNow": "Nu een achtergrond-updatecontrole uitvoeren", | ||||||
|     "versionExtractWholePage": "De reguliere expressie voor versie-extractie toepassen op de hele pagina", |     "versionExtractWholePage": "De reguliere expressie voor versie-extractie toepassen op de hele pagina", | ||||||
|     "installing": "Installeren", |     "installing": "Installeren", | ||||||
|     "skipUpdateNotifications": "Updatemeldingen overslaan", |     "skipUpdateNotifications": "Updatemeldingen overslaan", | ||||||
|     "updatesAvailableNotifChannel": "Updates beschikbaar", |     "updatesAvailableNotifChannel": "Updates beschikbaar", | ||||||
|     "appsUpdatedNotifChannel": "Apps bijgewerkt", |     "appsUpdatedNotifChannel": "Apps bijgewerkt", | ||||||
|     "appsPossiblyUpdatedNotifChannel": "Poging tot app-updates", |     "appsPossiblyUpdatedNotifChannel": "Pogingen tot app-updates", | ||||||
|     "errorCheckingUpdatesNotifChannel": "Foutcontrole bij het zoeken naar updates", |     "errorCheckingUpdatesNotifChannel": "Foutcontrole bij het zoeken naar updates", | ||||||
|     "appsRemovedNotifChannel": "Apps verwijderd", |     "appsRemovedNotifChannel": "Apps verwijderd", | ||||||
|     "downloadingXNotifChannel": "{} downloaden", |     "downloadingXNotifChannel": "{} downloaden", | ||||||
|     "completeAppInstallationNotifChannel": "Voltooien van de app-installatie", |     "completeAppInstallationNotifChannel": "App-installatie voltooien", | ||||||
|     "checkingForUpdatesNotifChannel": "Controleren op updates", |     "checkingForUpdatesNotifChannel": "Controleren op updates", | ||||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Alleen geïnstalleerde en Track-Only apps controleren op updates", |     "onlyCheckInstalledOrTrackOnlyApps": "Alleen geïnstalleerde apps en 'Alleen volgen' controleren op updates", | ||||||
|     "supportFixedAPKURL": "Ondersteuning vaste APK URL's", |     "supportFixedAPKURL": "Vaste APK-URL's ondersteunen", | ||||||
|     "selectX": "Selecteer {}", |     "selectX": "{} selecteren", | ||||||
|     "parallelDownloads": "Parallelle downloads toestaan", |     "parallelDownloads": "Parallelle downloads toestaan", | ||||||
|     "installMethod": "Installatiemethode", |     "useShizuku": "Shizuku of Sui gebruiken om te installeren", | ||||||
|     "normal": "Normaal", |     "shizukuBinderNotFound": "Shizuku is niet actief", | ||||||
|     "root": "Wortel", |     "shizukuOld": "Verouderde Shizuku-versie (<11) - bijwerken", | ||||||
|     "shizukuBinderNotFound": "Shizuku draait niet", |     "shizukuOldAndroidWithADB": "Shizuku draait op Android < 8.1 met ADB - update Android of gebruik in plaats daarvan Sui", | ||||||
|     "useSystemFont": "Gebruik het systeemlettertype", |     "shizukuPretendToBeGooglePlay": "Google Play instellen als installatiebron (bij Shizuku)", | ||||||
|     "systemFontError": "Fout bij het laden van het systeemlettertype: {}", |     "useSystemFont": "Systeemlettertype gebruiken", | ||||||
|     "useVersionCodeAsOSVersion": "Gebruik app versieCode als door OS gedetecteerde versie", |     "useVersionCodeAsOSVersion": "App versiecode gebruiken als door OS gedetecteerde versie", | ||||||
|     "requestHeader": "Verzoekkoptekst", |     "requestHeader": "Verzoekkoptekst", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Gebruik laatste upload als releasedatum", |     "useLatestAssetDateAsReleaseDate": "Laatste upload als releasedatum gebruiken", | ||||||
|     "defaultPseudoVersioningMethod": "Standaard pseudo-versiebeheermethode", |     "defaultPseudoVersioningMethod": "Standaard pseudo-versiebeheermethode", | ||||||
|     "partialAPKHash": "Gedeeltelijke APK-hash", |     "partialAPKHash": "Gedeeltelijke APK-hash", | ||||||
|     "APKLinkHash": "APK-link-hash", |     "APKLinkHash": "APK-link-hash", | ||||||
| @@ -303,22 +311,25 @@ | |||||||
|     "selfHostedNote": "De \"{}\" dropdown kan gebruikt worden om zelf gehoste/aangepaste instanties van elke bron te bereiken.", |     "selfHostedNote": "De \"{}\" dropdown kan gebruikt worden om zelf gehoste/aangepaste instanties van elke bron te bereiken.", | ||||||
|     "badDownload": "De APK kon niet worden verwerkt (incompatibele of gedeeltelijke download)", |     "badDownload": "De APK kon niet worden verwerkt (incompatibele of gedeeltelijke download)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Nieuwe Apps delen met AppVerifier (indien beschikbaar)", |     "beforeNewInstallsShareToAppVerifier": "Nieuwe Apps delen met AppVerifier (indien beschikbaar)", | ||||||
|     "appVerifierInstructionToast": "Deel naar AppVerifier en keer hier terug als je klaar bent.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "App verwijderen?", |         "one": "App verwijderen?", | ||||||
|         "other": "Apps verwijderen?" |         "other": "Apps verwijderen?" | ||||||
|     }, |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Te veel verzoeken (aantal beperkt) - probeer het opnieuw in {} minuut", |         "one": "Te veel verzoeken (aantal beperkt) - opnieuw proberen over {} minuut", | ||||||
|         "other": "Te veel verzoeken (aantal beperkt) - probeer het opnieuw in {} minuten" |         "other": "Te veel verzoeken (aantal beperkt) - opnieuw proberen over {} minuten" | ||||||
|     }, |     }, | ||||||
|     "bgUpdateGotErrorRetryInMinutes": { |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|         "one": "Achtergrondupdatecontrole heeft een {}, zal een hercontrole plannen over {} minuut", |         "one": "Achtergrond-updatecontrole heeft een {}, zal een nieuwe controle plannen over {} minuut", | ||||||
|         "other": "Achtergrondupdatecontrole heeft een {}, zal een hercontrole plannen over {} minuten" |         "other": "Achtergrond-updatecontrole heeft een {}, zal een nieuwe controle plannen over {} minuten" | ||||||
|     }, |     }, | ||||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|         "one": "Achtergrondupdatecontrole heeft {} update gevonden - zal de gebruiker op de hoogte stellen indien nodig", |         "one": "Achtergrond-updatecontrole heeft {} update gevonden - zal de gebruiker op de hoogte stellen indien nodig", | ||||||
|         "other": "Achtergrondupdatecontrole heeft {} updates gevonden - zal de gebruiker op de hoogte stellen indien nodig" |         "other": "Achtergrond-updatecontrole heeft {} updates gevonden - zal de gebruiker op de hoogte stellen indien nodig" | ||||||
|     }, |     }, | ||||||
|     "apps": { |     "apps": { | ||||||
|         "one": "{} app", |         "one": "{} app", | ||||||
| @@ -346,15 +357,19 @@ | |||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesAvailable": { |     "xAndNMoreUpdatesAvailable": { | ||||||
|         "one": "{} en nog 1 app hebben updates.", |         "one": "{} en nog 1 app hebben updates.", | ||||||
|         "other": "{} en {} meer apps hebben updates." |         "other": "{} en nog {} apps hebben updates." | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesInstalled": { |     "xAndNMoreUpdatesInstalled": { | ||||||
|         "one": "{} en nog 1 app is bijgewerkt.", |         "one": "{} en nog 1 app is bijgewerkt.", | ||||||
|         "other": "{} en {} meer apps zijn bijgewerkt." |         "other": "{} en nog {} apps zijn bijgewerkt." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Bijwerken mislukt voor {} en nog 1 app.", | ||||||
|  |         "other": "Bijwerken mislukt voor {} en nog {} apps." | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} en nog 1 app zijn mogelijk bijgewerkt.", |         "one": "{} en nog 1 app zijn mogelijk bijgewerkt.", | ||||||
|         "other": "{} en {} meer apps zijn mogelijk bijgwerkt." |         "other": "{} en nog {} apps zijn mogelijk bijgwerkt." | ||||||
|     }, |     }, | ||||||
|     "apk": { |     "apk": { | ||||||
|         "one": "{} APK", |         "one": "{} APK", | ||||||
|   | |||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(Wymagane)", |     "requiredInBrackets": "(Wymagane)", | ||||||
|     "dropdownNoOptsError": "BŁĄD: LISTA ROZWIJANA MUSI MIEĆ CO NAJMNIEJ JEDNĄ OPCJĘ", |     "dropdownNoOptsError": "BŁĄD: LISTA ROZWIJANA MUSI MIEĆ CO NAJMNIEJ JEDNĄ OPCJĘ", | ||||||
|     "colour": "Kolor", |     "colour": "Kolor", | ||||||
|  |     "standard": "Standard", | ||||||
|  |     "custom": "Niestandardowe", | ||||||
|  |     "useMaterialYou": "Używaj materiałów", | ||||||
|     "githubStarredRepos": "Repozytoria GitHub oznaczone gwiazdką", |     "githubStarredRepos": "Repozytoria GitHub oznaczone gwiazdką", | ||||||
|     "uname": "Nazwa użytkownika", |     "uname": "Nazwa użytkownika", | ||||||
|     "wrongArgNum": "Nieprawidłowa liczba podanych argumentów", |     "wrongArgNum": "Nieprawidłowa liczba podanych argumentów", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Ciemny", |     "dark": "Ciemny", | ||||||
|     "light": "Jasny", |     "light": "Jasny", | ||||||
|     "followSystem": "Zgodny z systemem", |     "followSystem": "Zgodny z systemem", | ||||||
|  |     "followSystemThemeExplanation": "Podążanie za motywem systemowym jest możliwe tylko przy użyciu aplikacji firm trzecich", | ||||||
|     "useBlackTheme": "Użyj czarnego motywu", |     "useBlackTheme": "Użyj czarnego motywu", | ||||||
|     "appSortBy": "Sortuj aplikacje według", |     "appSortBy": "Sortuj aplikacje według", | ||||||
|     "authorName": "Autor/Nazwa", |     "authorName": "Autor/Nazwa", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Brak nowych aktualizacji.", |     "noNewUpdates": "Brak nowych aktualizacji.", | ||||||
|     "xHasAnUpdate": "{} ma aktualizację.", |     "xHasAnUpdate": "{} ma aktualizację.", | ||||||
|     "appsUpdated": "Zaktualizowano aplikacje", |     "appsUpdated": "Zaktualizowano aplikacje", | ||||||
|  |     "appsNotUpdated": "Nie udało się zaktualizować aplikacji", | ||||||
|     "appsUpdatedNotifDescription": "Informuje, gdy co najmniej jedna aplikacja została zaktualizowana w tle", |     "appsUpdatedNotifDescription": "Informuje, gdy co najmniej jedna aplikacja została zaktualizowana w tle", | ||||||
|     "xWasUpdatedToY": "{} zaktualizowano do {}.", |     "xWasUpdatedToY": "{} zaktualizowano do {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Nie udało się zaktualizować {} do {}.", | ||||||
|     "errorCheckingUpdates": "Błąd sprawdzania aktualizacji", |     "errorCheckingUpdates": "Błąd sprawdzania aktualizacji", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Jest wyświetlane, gdy sprawdzanie aktualizacji w tle nie powiedzie się", |     "errorCheckingUpdatesNotifDescription": "Jest wyświetlane, gdy sprawdzanie aktualizacji w tle nie powiedzie się", | ||||||
|     "appsRemoved": "Usunięte aplikacje", |     "appsRemoved": "Usunięte aplikacje", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "Wyłącz aktualizacje w tle, gdy nie ma połączenia z Wi-Fi", |     "bgUpdatesOnWiFiOnly": "Wyłącz aktualizacje w tle, gdy nie ma połączenia z Wi-Fi", | ||||||
|     "autoSelectHighestVersionCode": "Automatycznie wybierz najwyższy kod wersji APK", |     "autoSelectHighestVersionCode": "Automatycznie wybierz najwyższy kod wersji APK", | ||||||
|     "versionExtractionRegEx": "Wyrażenie regularne wyodrębniające wersję", |     "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", |     "matchGroupToUse": "Dopasuj grupę do użycia dla wyrażenia regularnego wyodrębniania wersji", | ||||||
|     "highlightTouchTargets": "Wyróżnij mniej oczywiste elementy dotykowe", |     "highlightTouchTargets": "Wyróżnij mniej oczywiste elementy dotykowe", | ||||||
|     "pickExportDir": "Wybierz katalog eksportu", |     "pickExportDir": "Wybierz katalog eksportu", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Obsługuj stałe adresy URL APK", |     "supportFixedAPKURL": "Obsługuj stałe adresy URL APK", | ||||||
|     "selectX": "Wybierz {}", |     "selectX": "Wybierz {}", | ||||||
|     "parallelDownloads": "Zezwól na pobieranie równoległe", |     "parallelDownloads": "Zezwól na pobieranie równoległe", | ||||||
|     "installMethod": "Metoda instalacji", |     "useShizuku": "Użyj Shizuku lub Sui, aby zainstalować", | ||||||
|     "normal": "Normalna", |  | ||||||
|     "root": "Źródło", |  | ||||||
|     "shizukuBinderNotFound": "Shizuku is not running", |     "shizukuBinderNotFound": "Shizuku is not running", | ||||||
|  |     "shizukuOld": "Stara wersja Shizuku (<11) - zaktualizuj ją", | ||||||
|  |     "shizukuOldAndroidWithADB": "Shizuku działa na Androidzie < 8.1 z ADB - zaktualizuj Androida lub użyj zamiast tego Sui", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Ustaw Google Play jako źródło instalacji (jeśli używana jest aplikacja Shizuku).", | ||||||
|     "useSystemFont": "Użyj czcionki systemowej", |     "useSystemFont": "Użyj czcionki systemowej", | ||||||
|     "systemFontError": "Błąd podczas ładowania czcionki systemowej: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Użyj kodu wersji aplikacji jako wersji wykrytej przez system operacyjny", |     "useVersionCodeAsOSVersion": "Użyj kodu wersji aplikacji jako wersji wykrytej przez system operacyjny", | ||||||
|     "requestHeader": "Nagłówek żądania", |     "requestHeader": "Nagłówek żądania", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Użyj najnowszego przesłanego zasobu jako daty wydania", |     "useLatestAssetDateAsReleaseDate": "Użyj najnowszego przesłanego zasobu jako daty wydania", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "Nie można przeanalizować pliku APK (niekompatybilny lub częściowo pobrany).", |     "badDownload": "Nie można przeanalizować pliku APK (niekompatybilny lub częściowo pobrany).", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Udostępnianie nowych aplikacji za pomocą AppVerifier (jeśli dostępne)", |     "beforeNewInstallsShareToAppVerifier": "Udostępnianie nowych aplikacji za pomocą AppVerifier (jeśli dostępne)", | ||||||
|     "appVerifierInstructionToast": "Udostępnij w AppVerifier, a następnie wróć tutaj, gdy będziesz gotowy.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "Usunąć aplikację?", |         "one": "Usunąć aplikację?", | ||||||
|         "few": "Usunąć aplikacje?", |         "few": "Usunąć aplikacje?", | ||||||
| @@ -376,6 +387,10 @@ | |||||||
|         "many": "{} i {} innych apek zostało zaktualizowanych.", |         "many": "{} i {} innych apek zostało zaktualizowanych.", | ||||||
|         "other": "{} i {} inne apki zostały zaktualizowane." |         "other": "{} i {} inne apki zostały zaktualizowane." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Nie udało się zaktualizować {} i 1 innej aplikacji.", | ||||||
|  |         "other": "Nie udało się zaktualizować {} i {} więcej aplikacji." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} i 1 inna apka mogły zostać zaktualizowane.", |         "one": "{} i 1 inna apka mogły zostać zaktualizowane.", | ||||||
|         "few": "{} i {} inne apki mogły zostać zaktualizowane.", |         "few": "{} i {} inne apki mogły zostać zaktualizowane.", | ||||||
|   | |||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(Necessário)", |     "requiredInBrackets": "(Necessário)", | ||||||
|     "dropdownNoOptsError": "ERRO: O DROPDOWN DEVE TER PELO MENOS UMA OPÇÃO", |     "dropdownNoOptsError": "ERRO: O DROPDOWN DEVE TER PELO MENOS UMA OPÇÃO", | ||||||
|     "colour": "Cor", |     "colour": "Cor", | ||||||
|  |     "standard": "Padrão", | ||||||
|  |     "custom": "Personalizado", | ||||||
|  |     "useMaterialYou": "Utilizar o material que", | ||||||
|     "githubStarredRepos": "repositórios favoritos no GitHub", |     "githubStarredRepos": "repositórios favoritos no GitHub", | ||||||
|     "uname": "Nome de usuário", |     "uname": "Nome de usuário", | ||||||
|     "wrongArgNum": "Número de argumentos errado", |     "wrongArgNum": "Número de argumentos errado", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Escuro", |     "dark": "Escuro", | ||||||
|     "light": "Claro", |     "light": "Claro", | ||||||
|     "followSystem": "Padrão do sistema", |     "followSystem": "Padrão do sistema", | ||||||
|  |     "followSystemThemeExplanation": "O tema do sistema seguinte só é possível através da utilização de aplicações de terceiros", | ||||||
|     "useBlackTheme": "Usar tema preto AMOLED", |     "useBlackTheme": "Usar tema preto AMOLED", | ||||||
|     "appSortBy": "Classificar aplicativo por", |     "appSortBy": "Classificar aplicativo por", | ||||||
|     "authorName": "Autor/Nome", |     "authorName": "Autor/Nome", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Sem novas atualizações.", |     "noNewUpdates": "Sem novas atualizações.", | ||||||
|     "xHasAnUpdate": "{} tem uma atualização.", |     "xHasAnUpdate": "{} tem uma atualização.", | ||||||
|     "appsUpdated": "Aplicativos atualizados", |     "appsUpdated": "Aplicativos atualizados", | ||||||
|  |     "appsNotUpdated": "Falha na atualização das aplicações", | ||||||
|     "appsUpdatedNotifDescription": "Notifica o usuário quando atualizações foram aplicadas em segundo-plano para um ou mais aplicativos ", |     "appsUpdatedNotifDescription": "Notifica o usuário quando atualizações foram aplicadas em segundo-plano para um ou mais aplicativos ", | ||||||
|     "xWasUpdatedToY": "{} foi atualizado para {}.", |     "xWasUpdatedToY": "{} foi atualizado para {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Falha ao atualizar {} para {}.", | ||||||
|     "errorCheckingUpdates": "Erro ao procurar por atualizações", |     "errorCheckingUpdates": "Erro ao procurar por atualizações", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Uma notificação que mostra quando a checagem por atualizações em segundo-plano falha", |     "errorCheckingUpdatesNotifDescription": "Uma notificação que mostra quando a checagem por atualizações em segundo-plano falha", | ||||||
|     "appsRemoved": "Aplicativos removidos", |     "appsRemoved": "Aplicativos removidos", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "Desative as atualizações em segundo-plano quando não estiver conectado no Wi-Fi", |     "bgUpdatesOnWiFiOnly": "Desative as atualizações em segundo-plano quando não estiver conectado no Wi-Fi", | ||||||
|     "autoSelectHighestVersionCode": "Auto-selecionar a versão mais recente", |     "autoSelectHighestVersionCode": "Auto-selecionar a versão mais recente", | ||||||
|     "versionExtractionRegEx": "Regex de extração de versão", |     "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", |     "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", |     "highlightTouchTargets": "Realçar áreas sensíveis ao toque que são menos óbvias", | ||||||
|     "pickExportDir": "Escolher diretório para exportação", |     "pickExportDir": "Escolher diretório para exportação", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Suporte a APK com URLs fixas", |     "supportFixedAPKURL": "Suporte a APK com URLs fixas", | ||||||
|     "selectX": "Selecionar {}", |     "selectX": "Selecionar {}", | ||||||
|     "parallelDownloads": "Permitir downloads paralelos", |     "parallelDownloads": "Permitir downloads paralelos", | ||||||
|     "installMethod": "Método de instalação", |     "useShizuku": "Utilizar Shizuku ou Sui para instalar", | ||||||
|     "normal": "Normal", |  | ||||||
|     "root": "Root", |  | ||||||
|     "shizukuBinderNotFound": "O Shizuku não está rodando", |     "shizukuBinderNotFound": "O Shizuku não está rodando", | ||||||
|  |     "shizukuOld": "Versão antiga do Shizuku (<11) - atualizar", | ||||||
|  |     "shizukuOldAndroidWithADB": "Shizuku a funcionar no Android < 8.1 com ADB - atualizar o Android ou utilizar o Sui", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Definir o Google Play como fonte de instalação (se for utilizado o Shizuku)", | ||||||
|     "useSystemFont": "Usar fonte padrão do sistema", |     "useSystemFont": "Usar fonte padrão do sistema", | ||||||
|     "systemFontError": "Erro ao carregar a fonte do sistema: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Usar versionCode do aplicativo como versão detectada pelo sistema operacional", |     "useVersionCodeAsOSVersion": "Usar versionCode do aplicativo como versão detectada pelo sistema operacional", | ||||||
|     "requestHeader": "Requisitar cabeçalho", |     "requestHeader": "Requisitar cabeçalho", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Use o último upload de recursos como data de lançamento", |     "useLatestAssetDateAsReleaseDate": "Use o último upload de recursos como data de lançamento", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "Não foi possível analisar o APK (transferência incompatível ou parcial)", |     "badDownload": "Não foi possível analisar o APK (transferência incompatível ou parcial)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Partilhar novas aplicações com o AppVerifier (se disponível)", |     "beforeNewInstallsShareToAppVerifier": "Partilhar novas aplicações com o AppVerifier (se disponível)", | ||||||
|     "appVerifierInstructionToast": "Partilhe com o AppVerifier e, em seguida, regresse aqui quando estiver pronto.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "Remover aplicativo?", |         "one": "Remover aplicativo?", | ||||||
|         "other": "Remover aplicativos?" |         "other": "Remover aplicativos?" | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} e um outro aplicativo foram atualizado.", |         "one": "{} e um outro aplicativo foram atualizado.", | ||||||
|         "other": "{} e {} outros aplicativos foram atualizados." |         "other": "{} e {} outros aplicativos foram atualizados." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Falha ao atualizar {} e mais 1 aplicação.", | ||||||
|  |         "other": "Falha ao atualizar {} e {} mais aplicações." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} e um outro aplicativo podem ter sido atualizados.", |         "one": "{} e um outro aplicativo podem ter sido atualizados.", | ||||||
|         "other": "{} e {} outros aplicativos podem ter sido atualizados." |         "other": "{} e {} outros aplicativos podem ter sido atualizados." | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ | |||||||
|     "placeholder": "Заполнитель", |     "placeholder": "Заполнитель", | ||||||
|     "someErrors": "Возникли некоторые ошибки", |     "someErrors": "Возникли некоторые ошибки", | ||||||
|     "unexpectedError": "Неожиданная ошибка", |     "unexpectedError": "Неожиданная ошибка", | ||||||
|     "ok": "Ok", |     "ok": "Ок", | ||||||
|     "and": "и", |     "and": "и", | ||||||
|     "githubPATLabel": "Персональный токен доступа GitHub\n(увеличивает лимит запросов)", |     "githubPATLabel": "Персональный токен доступа GitHub\n(увеличивает лимит запросов)", | ||||||
|     "includePrereleases": "Включить предварительные релизы", |     "includePrereleases": "Включить предварительные релизы", | ||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(обязательно)", |     "requiredInBrackets": "(обязательно)", | ||||||
|     "dropdownNoOptsError": "Ошибка: в выпадающем списке должна быть выбрана хотя бы одна настройка", |     "dropdownNoOptsError": "Ошибка: в выпадающем списке должна быть выбрана хотя бы одна настройка", | ||||||
|     "colour": "Цвет", |     "colour": "Цвет", | ||||||
|  |     "standard": "Стандартный", | ||||||
|  |     "custom": "Индивидуальный", | ||||||
|  |     "useMaterialYou": "Использовать Material You", | ||||||
|     "githubStarredRepos": "Избранные репозитории GitHub", |     "githubStarredRepos": "Избранные репозитории GitHub", | ||||||
|     "uname": "Имя пользователя", |     "uname": "Имя пользователя", | ||||||
|     "wrongArgNum": "Неправильное количество предоставленных аргументов", |     "wrongArgNum": "Неправильное количество предоставленных аргументов", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Тёмная", |     "dark": "Тёмная", | ||||||
|     "light": "Светлая", |     "light": "Светлая", | ||||||
|     "followSystem": "Системная", |     "followSystem": "Системная", | ||||||
|  |     "followSystemThemeExplanation": "Следование системной теме возможно только при использовании сторонних приложений", | ||||||
|     "useBlackTheme": "Использовать чёрную тему", |     "useBlackTheme": "Использовать чёрную тему", | ||||||
|     "appSortBy": "Сортировка приложений", |     "appSortBy": "Сортировка приложений", | ||||||
|     "authorName": "Автор/Название", |     "authorName": "Автор/Название", | ||||||
| @@ -125,13 +129,13 @@ | |||||||
|     "pinUpdates": "Отображать обновления приложений сверху списка", |     "pinUpdates": "Отображать обновления приложений сверху списка", | ||||||
|     "updates": "Обновления", |     "updates": "Обновления", | ||||||
|     "sourceSpecific": "Настройки источников", |     "sourceSpecific": "Настройки источников", | ||||||
|     "appSource": "Исходный код", |     "appSource": "Источник", | ||||||
|     "noLogs": "Нет журналов", |     "noLogs": "Нет журналов", | ||||||
|     "appLogs": "Логи", |     "appLogs": "Логи", | ||||||
|     "close": "Закрыть", |     "close": "Закрыть", | ||||||
|     "share": "Поделиться", |     "share": "Поделиться", | ||||||
|     "appNotFound": "Приложение не найдено", |     "appNotFound": "Приложение не найдено", | ||||||
|     "obtainiumExportHyphenatedLowercase": "получение-экспорт", |     "obtainiumExportHyphenatedLowercase": "экспорт-obtainium", | ||||||
|     "pickAnAPK": "Выберите APK-файл", |     "pickAnAPK": "Выберите APK-файл", | ||||||
|     "appHasMoreThanOnePackage": "{} имеет более одного пакета:", |     "appHasMoreThanOnePackage": "{} имеет более одного пакета:", | ||||||
|     "deviceSupportsXArch": "Ваше устройство поддерживает архитектуру процессора {}", |     "deviceSupportsXArch": "Ваше устройство поддерживает архитектуру процессора {}", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Нет новых обновлений", |     "noNewUpdates": "Нет новых обновлений", | ||||||
|     "xHasAnUpdate": "{} есть обновление", |     "xHasAnUpdate": "{} есть обновление", | ||||||
|     "appsUpdated": "Приложения обновлены", |     "appsUpdated": "Приложения обновлены", | ||||||
|  |     "appsNotUpdated": "Не удалось обновить приложения", | ||||||
|     "appsUpdatedNotifDescription": "Уведомляет об обновлении одного или нескольких приложений в фоновом режиме", |     "appsUpdatedNotifDescription": "Уведомляет об обновлении одного или нескольких приложений в фоновом режиме", | ||||||
|     "xWasUpdatedToY": "{} была обновлена до версии {}", |     "xWasUpdatedToY": "{} была обновлена до версии {}", | ||||||
|  |     "xWasNotUpdatedToY": "Не удалось обновить {} до версии {}", | ||||||
|     "errorCheckingUpdates": "Ошибка при проверке обновлений", |     "errorCheckingUpdates": "Ошибка при проверке обновлений", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Уведомление о завершении проверки обновлений в фоновом режиме с ошибкой", |     "errorCheckingUpdatesNotifDescription": "Уведомление о завершении проверки обновлений в фоновом режиме с ошибкой", | ||||||
|     "appsRemoved": "Приложение удалено", |     "appsRemoved": "Приложение удалено", | ||||||
| @@ -173,8 +179,8 @@ | |||||||
|     "appWithIdOrNameNotFound": "Приложение с таким ID или названием не было найдено", |     "appWithIdOrNameNotFound": "Приложение с таким ID или названием не было найдено", | ||||||
|     "reposHaveMultipleApps": "В хранилище несколько приложений", |     "reposHaveMultipleApps": "В хранилище несколько приложений", | ||||||
|     "fdroidThirdPartyRepo": "Сторонние репозитории F-Droid", |     "fdroidThirdPartyRepo": "Сторонние репозитории F-Droid", | ||||||
|     "steamMobile": "Стим Мобайл", |     "steamMobile": "Приложение Steam", | ||||||
|     "steamChat": "Стим-чат", |     "steamChat": "Steam Chat", | ||||||
|     "install": "Установить", |     "install": "Установить", | ||||||
|     "markInstalled": "Пометить как установленное", |     "markInstalled": "Пометить как установленное", | ||||||
|     "update": "Обновить", |     "update": "Обновить", | ||||||
| @@ -185,7 +191,7 @@ | |||||||
|     "downloadingX": "Загрузка {}", |     "downloadingX": "Загрузка {}", | ||||||
|     "downloadX": "Скачать {}", |     "downloadX": "Скачать {}", | ||||||
|     "downloadedX": "Загружено {}", |     "downloadedX": "Загружено {}", | ||||||
|     "releaseAsset": "Освобождение актива", |     "releaseAsset": "Релизный объект", | ||||||
|     "downloadNotifDescription": "Уведомляет пользователя о прогрессе загрузки приложения", |     "downloadNotifDescription": "Уведомляет пользователя о прогрессе загрузки приложения", | ||||||
|     "noAPKFound": "APK не найден", |     "noAPKFound": "APK не найден", | ||||||
|     "noVersionDetection": "Обнаружение версий отключено", |     "noVersionDetection": "Обнаружение версий отключено", | ||||||
| @@ -248,11 +254,13 @@ | |||||||
|     "intermediateLinkRegex": "Фильтр для \"промежуточной\" ссылки для посещения", |     "intermediateLinkRegex": "Фильтр для \"промежуточной\" ссылки для посещения", | ||||||
|     "filterByLinkText": "Фильтрация ссылок по тексту ссылки", |     "filterByLinkText": "Фильтрация ссылок по тексту ссылки", | ||||||
|     "intermediateLinkNotFound": "Промежуточная ссылка не найдена", |     "intermediateLinkNotFound": "Промежуточная ссылка не найдена", | ||||||
|     "intermediateLink": "Промежуточное звено", |     "intermediateLink": "Промежуточная ссылка", | ||||||
|     "exemptFromBackgroundUpdates": "Исключить из фоновых обновлений (если включено)", |     "exemptFromBackgroundUpdates": "Исключить из фоновых обновлений (если включено)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Отключить фоновые обновления, если нет соединения с Wi-Fi", |     "bgUpdatesOnWiFiOnly": "Отключить фоновые обновления, если нет соединения с Wi-Fi", | ||||||
|     "autoSelectHighestVersionCode": "Автоматически выбирать APK с актуальной версией кода", |     "autoSelectHighestVersionCode": "Автоматически выбирать APK с актуальной версией кода", | ||||||
|     "versionExtractionRegEx": "Регулярное выражение для извлечения версии", |     "versionExtractionRegEx": "Регулярное выражение для извлечения версии", | ||||||
|  |     "trimVersionString": "Обрезка строки версии с помощью RegEx", | ||||||
|  |     "matchGroupToUseForX": "Группа соответствия, которую следует использовать для \"{}\"", | ||||||
|     "matchGroupToUse": "Выберите группу для использования", |     "matchGroupToUse": "Выберите группу для использования", | ||||||
|     "highlightTouchTargets": "Выделить менее очевидные элементы управления касанием", |     "highlightTouchTargets": "Выделить менее очевидные элементы управления касанием", | ||||||
|     "pickExportDir": "Выбрать каталог для экспорта", |     "pickExportDir": "Выбрать каталог для экспорта", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Поддержка фиксированных URL-адресов APK", |     "supportFixedAPKURL": "Поддержка фиксированных URL-адресов APK", | ||||||
|     "selectX": "Выбрать {}", |     "selectX": "Выбрать {}", | ||||||
|     "parallelDownloads": "Разрешить параллельные загрузки", |     "parallelDownloads": "Разрешить параллельные загрузки", | ||||||
|     "installMethod": "Метод установки", |     "useShizuku": "Использовать Shizuku или Sui для установки", | ||||||
|     "normal": "Нормальный", |     "shizukuBinderNotFound": "Совместимый сервис Shizuku не найден, возможно он не запущен", | ||||||
|     "root": "Суперпользователь", |     "shizukuOld": "Устаревшая версия Shizuku (<11), обновите", | ||||||
|     "shizukuBinderNotFound": "Совместимый сервис Shizuku не найден", |     "shizukuOldAndroidWithADB": "Shizuku работает на Android < 8.1 с ADB, обновите Android или используйте Sui", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Указать Google Play как источник установки (если используется Shizuku)", | ||||||
|     "useSystemFont": "Использовать системный шрифт", |     "useSystemFont": "Использовать системный шрифт", | ||||||
|     "systemFontError": "Ошибка загрузки системного шрифта: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Использовать код версии приложения как версию, обнаруженную ОС", |     "useVersionCodeAsOSVersion": "Использовать код версии приложения как версию, обнаруженную ОС", | ||||||
|     "requestHeader": "Заголовок запроса", |     "requestHeader": "Заголовок запроса", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Использовать последнюю загрузку ресурса в качестве даты выпуска", |     "useLatestAssetDateAsReleaseDate": "Использовать последнюю загрузку ресурса в качестве даты выпуска", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "APK не удалось разобрать (несовместимая или неполная загрузка)", |     "badDownload": "APK не удалось разобрать (несовместимая или неполная загрузка)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Поделитесь новыми приложениями с AppVerifier (если доступно)", |     "beforeNewInstallsShareToAppVerifier": "Поделитесь новыми приложениями с AppVerifier (если доступно)", | ||||||
|     "appVerifierInstructionToast": "Поделитесь с AppVerifier, а затем вернитесь сюда, когда будете готовы.", |     "appVerifierInstructionToast": "Поделитесь с AppVerifier, а затем вернитесь сюда, когда будете готовы.", | ||||||
|  |     "wiki": "Помощь/Вики", | ||||||
|  |     "crowdsourcedConfigsLabel": "Конфигурации приложений на основе краудсорсинга (используйте на свой страх и риск)", | ||||||
|  |     "allowInsecure": "Разрешить небезопасные HTTP-запросы", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Удалить приложение?", |         "one": "Удалить приложение?", | ||||||
|         "other": "Удалить приложения?" |         "other": "Удалить приложения?" | ||||||
| @@ -352,12 +363,16 @@ | |||||||
|         "one": "{} и ещё 1 приложение были обновлены", |         "one": "{} и ещё 1 приложение были обновлены", | ||||||
|         "other": "{} и ещё {} приложений были обновлены" |         "other": "{} и ещё {} приложений были обновлены" | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Не удалось обновить {} и ещё 1 приложение", | ||||||
|  |         "other": "Не удалось обновить {} и ещё {} приложений" | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} и ещё 1 приложение могли быть обновлены", |         "one": "{} и ещё 1 приложение могли быть обновлены", | ||||||
|         "other": "{} и ещё {} приложений могли быть обновлены" |         "other": "{} и ещё {} приложений могли быть обновлены" | ||||||
|     }, |     }, | ||||||
|     "apk": { |     "apk": { | ||||||
|         "one": "{} APK", |         "one": "{} APK", | ||||||
|         "other": "{} APKs" |         "other": "{} APKи" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(Kräver)", |     "requiredInBrackets": "(Kräver)", | ||||||
|     "dropdownNoOptsError": "FEL: DROPDOWN MÅSTE HA MINST ETT OPT", |     "dropdownNoOptsError": "FEL: DROPDOWN MÅSTE HA MINST ETT OPT", | ||||||
|     "colour": "Färg", |     "colour": "Färg", | ||||||
|  |     "standard": "Standard", | ||||||
|  |     "custom": "Anpassad", | ||||||
|  |     "useMaterialYou": "Använd material Du", | ||||||
|     "githubStarredRepos": "GitHub Stjärnmärkta Förråd", |     "githubStarredRepos": "GitHub Stjärnmärkta Förråd", | ||||||
|     "uname": "Användarnamn", |     "uname": "Användarnamn", | ||||||
|     "wrongArgNum": "Fel antal argument har angetts", |     "wrongArgNum": "Fel antal argument har angetts", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Mörkt", |     "dark": "Mörkt", | ||||||
|     "light": "Ljust", |     "light": "Ljust", | ||||||
|     "followSystem": "Följ System", |     "followSystem": "Följ System", | ||||||
|  |     "followSystemThemeExplanation": "Följande systemtema är endast möjligt med hjälp av tredjepartsapplikationer", | ||||||
|     "useBlackTheme": "Använd svart tema", |     "useBlackTheme": "Använd svart tema", | ||||||
|     "appSortBy": "Sortera Appar via", |     "appSortBy": "Sortera Appar via", | ||||||
|     "authorName": "Utvecklare/Namn", |     "authorName": "Utvecklare/Namn", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Inga nya uppdateringar.", |     "noNewUpdates": "Inga nya uppdateringar.", | ||||||
|     "xHasAnUpdate": "{} har en uppdatering.", |     "xHasAnUpdate": "{} har en uppdatering.", | ||||||
|     "appsUpdated": "Appar Uppdaterade", |     "appsUpdated": "Appar Uppdaterade", | ||||||
|  |     "appsNotUpdated": "Misslyckades med att uppdatera applikationer", | ||||||
|     "appsUpdatedNotifDescription": "Meddelar användaren att uppdateringar av en eller flera appar har tillämpats i bakgrunden", |     "appsUpdatedNotifDescription": "Meddelar användaren att uppdateringar av en eller flera appar har tillämpats i bakgrunden", | ||||||
|     "xWasUpdatedToY": "{} uppdaterades till {}.", |     "xWasUpdatedToY": "{} uppdaterades till {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Det gick inte att uppdatera {} till {}.", | ||||||
|     "errorCheckingUpdates": "Fel vid uppdateringskoll", |     "errorCheckingUpdates": "Fel vid uppdateringskoll", | ||||||
|     "errorCheckingUpdatesNotifDescription": "En aviserings som visar när bakgrundsuppdateringarkollar misslyckas", |     "errorCheckingUpdatesNotifDescription": "En aviserings som visar när bakgrundsuppdateringarkollar misslyckas", | ||||||
|     "appsRemoved": "Appar borttagna", |     "appsRemoved": "Appar borttagna", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "Inaktivera Bakgrundsuppdateringar utan WiFi", |     "bgUpdatesOnWiFiOnly": "Inaktivera Bakgrundsuppdateringar utan WiFi", | ||||||
|     "autoSelectHighestVersionCode": "Välj automatiskt högsta versionskod APK", |     "autoSelectHighestVersionCode": "Välj automatiskt högsta versionskod APK", | ||||||
|     "versionExtractionRegEx": "Version Extraction RegEx", |     "versionExtractionRegEx": "Version Extraction RegEx", | ||||||
|  |     "trimVersionString": "Trimma versionssträng med RegEx", | ||||||
|  |     "matchGroupToUseForX": "Matchningsgrupp att använda för \"{}\"", | ||||||
|     "matchGroupToUse": "Match Group to Use", |     "matchGroupToUse": "Match Group to Use", | ||||||
|     "highlightTouchTargets": "Markera mindre uppenbara beröringsobjekt", |     "highlightTouchTargets": "Markera mindre uppenbara beröringsobjekt", | ||||||
|     "pickExportDir": "Välj Exportsökväg", |     "pickExportDir": "Välj Exportsökväg", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Stöd fasta APK-webbadresser", |     "supportFixedAPKURL": "Stöd fasta APK-webbadresser", | ||||||
|     "selectX": "Välj {}", |     "selectX": "Välj {}", | ||||||
|     "parallelDownloads": "Tillåt parallella nedladdningar", |     "parallelDownloads": "Tillåt parallella nedladdningar", | ||||||
|     "installMethod": "Installationsmetod", |     "useShizuku": "Använd Shizuku eller Sui för att installera", | ||||||
|     "normal": "Vanligt", |  | ||||||
|     "root": "Rot", |  | ||||||
|     "shizukuBinderNotFound": "Shizuku is not running", |     "shizukuBinderNotFound": "Shizuku is not running", | ||||||
|  |     "shizukuOld": "Gammal Shizuku-version (<11) - uppdatera den", | ||||||
|  |     "shizukuOldAndroidWithADB": "Shizuku körs på Android < 8.1 med ADB - uppdatera Android eller använd Sui istället", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Ange Google Play som installationskälla (om Shizuku används)", | ||||||
|     "useSystemFont": "Använd systemteckensnittet", |     "useSystemFont": "Använd systemteckensnittet", | ||||||
|     "systemFontError": "Fel vid laddning av systemteckensnittet: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Använd appversionskoden som OS-upptäckt version", |     "useVersionCodeAsOSVersion": "Använd appversionskoden som OS-upptäckt version", | ||||||
|     "requestHeader": "Rubrik för begäran", |     "requestHeader": "Rubrik för begäran", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Använd senaste tillgångsuppladdning som releasedatum", |     "useLatestAssetDateAsReleaseDate": "Använd senaste tillgångsuppladdning som releasedatum", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "APK kunde inte analyseras (inkompatibel eller partiell nedladdning)", |     "badDownload": "APK kunde inte analyseras (inkompatibel eller partiell nedladdning)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Dela nya appar med AppVerifier (om tillgängligt)", |     "beforeNewInstallsShareToAppVerifier": "Dela nya appar med AppVerifier (om tillgängligt)", | ||||||
|     "appVerifierInstructionToast": "Dela till AppVerifier och återvänd sedan hit när du är klar.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "Ta Bort App?", |         "one": "Ta Bort App?", | ||||||
|         "other": "Ta Bort Appar?" |         "other": "Ta Bort Appar?" | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} och 1 till app uppdaterades.", |         "one": "{} och 1 till app uppdaterades.", | ||||||
|         "other": "{} och {} appar till uppdaterades." |         "other": "{} och {} appar till uppdaterades." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Misslyckades med att uppdatera {} och ytterligare 1 app.", | ||||||
|  |         "other": "Det gick inte att uppdatera {} och {} fler appar." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} och 1 till app kan ha uppdaterats.", |         "one": "{} och 1 till app kan ha uppdaterats.", | ||||||
|         "other": "{} och {} appar till kan ha uppdaterats." |         "other": "{} och {} appar till kan ha uppdaterats." | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ | |||||||
|     "noReleaseFound": "Uygun bir sürüm bulunamadı", |     "noReleaseFound": "Uygun bir sürüm bulunamadı", | ||||||
|     "noVersionFound": "Sürüm bulunamadı", |     "noVersionFound": "Sürüm bulunamadı", | ||||||
|     "urlMatchesNoSource": "URL, bilinen bir kaynağa uymuyor", |     "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", |     "appIdMismatch": "İndirilen paket kimliği mevcut Uygulama kimliği ile eşleşmiyor", | ||||||
|     "functionNotImplemented": "Bu sınıf bu işlevi uygulamamıştır", |     "functionNotImplemented": "Bu sınıf bu işlevi uygulamamıştır", | ||||||
|     "placeholder": "Yer Tutucu", |     "placeholder": "Yer Tutucu", | ||||||
| @@ -13,19 +13,22 @@ | |||||||
|     "and": "ve", |     "and": "ve", | ||||||
|     "githubPATLabel": "GitHub Kişisel Erişim Anahtarı (Sınırlamayı Artırır)", |     "githubPATLabel": "GitHub Kişisel Erişim Anahtarı (Sınırlamayı Artırır)", | ||||||
|     "includePrereleases": "Ön sürümleri dahil et", |     "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", |     "filterReleaseTitlesByRegEx": "Düzenli İfadelerle Sürüm Başlıklarını Filtrele", | ||||||
|     "invalidRegEx": "Geçersiz düzenli ifade", |     "invalidRegEx": "Geçersiz düzenli ifade", | ||||||
|     "noDescription": "Açıklama yok", |     "noDescription": "Açıklama yok", | ||||||
|     "cancel": "İptal", |     "cancel": "İptal", | ||||||
|     "continue": "Devam Et", |     "continue": "Devam Et", | ||||||
|     "requiredInBrackets": "(Gerekli)", |     "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", |     "colour": "Renk", | ||||||
|     "githubStarredRepos": "GitHub'a Yıldızlı Depolar", |     "standard": "Standart", | ||||||
|  |     "custom": "Özel", | ||||||
|  |     "useMaterialYou": "MaterialYou Kullanın", | ||||||
|  |     "githubStarredRepos": "GitHub Yıldızlı Depolar", | ||||||
|     "uname": "Kullanıcı Adı", |     "uname": "Kullanıcı Adı", | ||||||
|     "wrongArgNum": "Hatalı argüman sayısı sağlandı", |     "wrongArgNum": "Hatalı sayıda argüman sağlandı", | ||||||
|     "xIsTrackOnly": "{} yalnızca Takip Edilen", |     "xIsTrackOnly": "{} yalnızca Takip Ediliyor", | ||||||
|     "source": "Kaynak", |     "source": "Kaynak", | ||||||
|     "app": "Uygulama", |     "app": "Uygulama", | ||||||
|     "appsFromSourceAreTrackOnly": "Bu kaynaktan gelen uygulamalar 'Yalnızca Takip Edilen'dir.", |     "appsFromSourceAreTrackOnly": "Bu kaynaktan gelen uygulamalar 'Yalnızca Takip Edilen'dir.", | ||||||
| @@ -38,9 +41,9 @@ | |||||||
|     "appSourceURL": "Uygulama Kaynak URL'si", |     "appSourceURL": "Uygulama Kaynak URL'si", | ||||||
|     "error": "Hata", |     "error": "Hata", | ||||||
|     "add": "Ekle", |     "add": "Ekle", | ||||||
|     "searchSomeSourcesLabel": "Ara (Bazı Kaynaklar Yalnızca)", |     "searchSomeSourcesLabel": "Ara (Yalnızca Bazı Kaynaklar)", | ||||||
|     "search": "Ara", |     "search": "Ara", | ||||||
|     "additionalOptsFor": "{} İçin Ek Seçenekler", |     "additionalOptsFor": "{} için Ek Seçenekler", | ||||||
|     "supportedSources": "Desteklenen Kaynaklar", |     "supportedSources": "Desteklenen Kaynaklar", | ||||||
|     "trackOnlyInBrackets": "(Yalnızca Takip)", |     "trackOnlyInBrackets": "(Yalnızca Takip)", | ||||||
|     "searchableInBrackets": "(Aranabilir)", |     "searchableInBrackets": "(Aranabilir)", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Koyu", |     "dark": "Koyu", | ||||||
|     "light": "Aydınlık", |     "light": "Aydınlık", | ||||||
|     "followSystem": "Sistemi Takip Et", |     "followSystem": "Sistemi Takip Et", | ||||||
|  |     "followSystemThemeExplanation": "Sistem temasını takip etmek yalnızca üçüncü taraf uygulamaları kullanarak mümkündür", | ||||||
|     "useBlackTheme": "Saf siyah koyu temasını kullan", |     "useBlackTheme": "Saf siyah koyu temasını kullan", | ||||||
|     "appSortBy": "Uygulama Sıralama Ölçütü", |     "appSortBy": "Uygulama Sıralama Ölçütü", | ||||||
|     "authorName": "Yazar/Ad", |     "authorName": "Yazar/Ad", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Yeni güncelleme yok.", |     "noNewUpdates": "Yeni güncelleme yok.", | ||||||
|     "xHasAnUpdate": "{} güncelleme alıyor.", |     "xHasAnUpdate": "{} güncelleme alıyor.", | ||||||
|     "appsUpdated": "Uygulamalar Güncellendi", |     "appsUpdated": "Uygulamalar Güncellendi", | ||||||
|  |     "appsNotUpdated": "Uygulamalar güncellenemedi", | ||||||
|     "appsUpdatedNotifDescription": "Kullanıcıya bir veya daha fazla uygulamanın arka planda güncellendiğine dair bilgi verir", |     "appsUpdatedNotifDescription": "Kullanıcıya bir veya daha fazla uygulamanın arka planda güncellendiğine dair bilgi verir", | ||||||
|     "xWasUpdatedToY": "{} şu sürüme güncellendi: {}.", |     "xWasUpdatedToY": "{} şu sürüme güncellendi: {}.", | ||||||
|  |     "xWasNotUpdatedToY": "{} öğesi {} olarak güncellenemedi.", | ||||||
|     "errorCheckingUpdates": "Güncellemeler Kontrol Edilirken Hata Oluştu", |     "errorCheckingUpdates": "Güncellemeler Kontrol Edilirken Hata Oluştu", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Arka planda güncelleme kontrolü sırasında hata oluştuğunda görünen bir bildirim", |     "errorCheckingUpdatesNotifDescription": "Arka planda güncelleme kontrolü sırasında hata oluştuğunda görünen bir bildirim", | ||||||
|     "appsRemoved": "Uygulamalar Kaldırıldı", |     "appsRemoved": "Uygulamalar Kaldırıldı", | ||||||
| @@ -167,13 +173,13 @@ | |||||||
|     "lastUpdateCheckX": "Son Güncelleme Kontrolü: {}", |     "lastUpdateCheckX": "Son Güncelleme Kontrolü: {}", | ||||||
|     "remove": "Kaldır", |     "remove": "Kaldır", | ||||||
|     "yesMarkUpdated": "Evet, Güncellendi olarak İşaretle", |     "yesMarkUpdated": "Evet, Güncellendi olarak İşaretle", | ||||||
|     "fdroid": "F-Droid Resmi", |     "fdroid": "Resmi F-Droid", | ||||||
|     "appIdOrName": "Uygulama Kimliği veya Adı", |     "appIdOrName": "Uygulama Kimliği veya Adı", | ||||||
|     "appId": "Uygulama Kimliği", |     "appId": "Uygulama Kimliği", | ||||||
|     "appWithIdOrNameNotFound": "Bu kimlik veya ada sahip bir uygulama bulunamadı", |     "appWithIdOrNameNotFound": "Bu kimlik veya ada sahip bir uygulama bulunamadı", | ||||||
|     "reposHaveMultipleApps": "Depolar birden fazla uygulama içerebilir", |     "reposHaveMultipleApps": "Depolar birden fazla uygulama içerebilir", | ||||||
|     "fdroidThirdPartyRepo": "F-Droid Üçüncü Taraf Depo", |     "fdroidThirdPartyRepo": "F-Droid Üçüncü Parti Depo", | ||||||
|     "steamMobile": "Buhar Mobil", |     "steamMobile": "Steam Mobil", | ||||||
|     "steamChat": "Steam Sohbet", |     "steamChat": "Steam Sohbet", | ||||||
|     "install": "Yükle", |     "install": "Yükle", | ||||||
|     "markInstalled": "Yüklendi olarak İşaretle", |     "markInstalled": "Yüklendi olarak İşaretle", | ||||||
| @@ -216,7 +222,7 @@ | |||||||
|     "standardVersionDetection": "Standart sürüm tespiti", |     "standardVersionDetection": "Standart sürüm tespiti", | ||||||
|     "groupByCategory": "Kategoriye Göre Grupla", |     "groupByCategory": "Kategoriye Göre Grupla", | ||||||
|     "autoApkFilterByArch": "Mümkünse APK'leri CPU mimarisi ile filtreleme girişimi", |     "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", |     "dontShowAgain": "Bunu tekrar gösterme", | ||||||
|     "dontShowTrackOnlyWarnings": "'Yalnızca Takip Edilen' uyarılarını gösterme", |     "dontShowTrackOnlyWarnings": "'Yalnızca Takip Edilen' uyarılarını gösterme", | ||||||
|     "dontShowAPKOriginWarnings": "APK kaynağı uyarılarını gösterme", |     "dontShowAPKOriginWarnings": "APK kaynağı uyarılarını gösterme", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "WiFi olmadığında arka plan güncellemelerini devre dışı bırak", |     "bgUpdatesOnWiFiOnly": "WiFi olmadığında arka plan güncellemelerini devre dışı bırak", | ||||||
|     "autoSelectHighestVersionCode": "Otomatik olarak en yüksek sürüm kodunu seç", |     "autoSelectHighestVersionCode": "Otomatik olarak en yüksek sürüm kodunu seç", | ||||||
|     "versionExtractionRegEx": "Sürüm Çıkarma Düzenli İfade", |     "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", |     "matchGroupToUse": "Sürüm Çıkarma Regex için Kullanılacak Eşleşme Grubu", | ||||||
|     "highlightTouchTargets": "Daha az belirgin dokunma hedeflerini vurgula", |     "highlightTouchTargets": "Daha az belirgin dokunma hedeflerini vurgula", | ||||||
|     "pickExportDir": "Dışa Aktarılacak Klasörü Seç", |     "pickExportDir": "Dışa Aktarılacak Klasörü Seç", | ||||||
| @@ -280,14 +288,14 @@ | |||||||
|     "checkingForUpdatesNotifChannel": "Güncellemeler Kontrol Ediliyor", |     "checkingForUpdatesNotifChannel": "Güncellemeler Kontrol Ediliyor", | ||||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Yalnızca yüklü ve Yalnızca İzleme Uygulamalarını güncelleme", |     "onlyCheckInstalledOrTrackOnlyApps": "Yalnızca yüklü ve Yalnızca İzleme Uygulamalarını güncelleme", | ||||||
|     "supportFixedAPKURL": "Sabit APK URL'lerini destekleyin", |     "supportFixedAPKURL": "Sabit APK URL'lerini destekleyin", | ||||||
|     "selectX": "Seçme {}", |     "selectX": "{} Tanesini Seç", | ||||||
|     "parallelDownloads": "Paralel indirmelere izin ver", |     "parallelDownloads": "Paralel indirmelere izin ver", | ||||||
|     "installMethod": "Kurulum yöntemi", |     "useShizuku": "Yüklemek için Shizuku veya Sui'yi kullanın", | ||||||
|     "normal": "Normal", |     "shizukuBinderNotFound": "Shizuku servisi çalışmıyor", | ||||||
|     "root": "Kök", |     "shizukuOld": "Eski Shizuku sürümü (<11) - güncelleyin", | ||||||
|     "shizukuBinderNotFound": "Shizuku is not running", |     "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)", | ||||||
|     "useSystemFont": "Sistem yazı tipini kullan", |     "useSystemFont": "Sistem yazı tipini kullan", | ||||||
|     "systemFontError": "Sistem yazı tipi yüklenirken hata oluştu: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Uygulama versionCode'unu işletim sistemi tarafından algılanan sürüm olarak kullan", |     "useVersionCodeAsOSVersion": "Uygulama versionCode'unu işletim sistemi tarafından algılanan sürüm olarak kullan", | ||||||
|     "requestHeader": "Başlık talep et", |     "requestHeader": "Başlık talep et", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Yayın tarihi olarak en son öğe yüklemesini kullan", |     "useLatestAssetDateAsReleaseDate": "Yayın tarihi olarak en son öğe yüklemesini kullan", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "APK ayrıştırılamadı (uyumsuz veya kısmi indirme)", |     "badDownload": "APK ayrıştırılamadı (uyumsuz veya kısmi indirme)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Yeni Uygulamaları AppVerifier ile paylaşın (varsa)", |     "beforeNewInstallsShareToAppVerifier": "Yeni Uygulamaları AppVerifier ile paylaşın (varsa)", | ||||||
|     "appVerifierInstructionToast": "AppVerifier ile paylaşın, hazır olduğunuzda buraya dönün.", |     "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": { |     "removeAppQuestion": { | ||||||
|         "one": "Uygulamayı Kaldır?", |         "one": "Uygulamayı Kaldır?", | ||||||
|         "other": "Uygulamaları Kaldır?" |         "other": "Uygulamaları Kaldır?" | ||||||
| @@ -326,7 +337,7 @@ | |||||||
|     }, |     }, | ||||||
|     "url": { |     "url": { | ||||||
|         "one": "{} URL", |         "one": "{} URL", | ||||||
|         "other": "{} URL'ler" |         "other": "{} URL" | ||||||
|     }, |     }, | ||||||
|     "minute": { |     "minute": { | ||||||
|         "one": "{} Dakika", |         "one": "{} Dakika", | ||||||
| @@ -352,12 +363,16 @@ | |||||||
|         "one": "{} ve 1 diğer uygulama güncellendi.", |         "one": "{} ve 1 diğer uygulama güncellendi.", | ||||||
|         "other": "{} ve {} daha fazla uygulama güncellendi." |         "other": "{} ve {} daha fazla uygulama güncellendi." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "{} ve 1 uygulama daha güncellenemedi.", | ||||||
|  |         "other": "{} ve {} daha fazla uygulama güncellenemedi." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} ve 1 diğer uygulama muhtemelen güncellendi.", |         "one": "{} ve 1 diğer uygulama muhtemelen güncellendi.", | ||||||
|         "other": "{} ve {} daha fazla uygulama muhtemelen güncellendi." |         "other": "{} ve {} daha fazla uygulama muhtemelen güncellendi." | ||||||
|     }, |     }, | ||||||
|     "apk": { |     "apk": { | ||||||
|         "one": "{} APK", |         "one": "{} APK", | ||||||
|         "other": "{} APK'lar" |         "other": "{} APK" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(Обов'язково)", |     "requiredInBrackets": "(Обов'язково)", | ||||||
|     "dropdownNoOptsError": "ПОМИЛКА: В ВИПАДАЮЧОМУ СПИСКУ МАЄ БУТИ ХОЧА Б ОДИН ЕЛЕМЕНТ", |     "dropdownNoOptsError": "ПОМИЛКА: В ВИПАДАЮЧОМУ СПИСКУ МАЄ БУТИ ХОЧА Б ОДИН ЕЛЕМЕНТ", | ||||||
|     "colour": "Колір", |     "colour": "Колір", | ||||||
|  |     "standard": "Стандартний", | ||||||
|  |     "custom": "Нестандартний", | ||||||
|  |     "useMaterialYou": "Використовуйте матеріал, який ви", | ||||||
|     "githubStarredRepos": "Відзначені репозиторії GitHub", |     "githubStarredRepos": "Відзначені репозиторії GitHub", | ||||||
|     "uname": "Ім'я користувача", |     "uname": "Ім'я користувача", | ||||||
|     "wrongArgNum": "Надано неправильну кількість аргументів", |     "wrongArgNum": "Надано неправильну кількість аргументів", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Темна", |     "dark": "Темна", | ||||||
|     "light": "Світла", |     "light": "Світла", | ||||||
|     "followSystem": "Дотримуватися системи", |     "followSystem": "Дотримуватися системи", | ||||||
|  |     "followSystemThemeExplanation": "Зміна теми системи можлива лише за допомогою сторонніх додатків", | ||||||
|     "useBlackTheme": "Використовувати чорну тему (Amoled)", |     "useBlackTheme": "Використовувати чорну тему (Amoled)", | ||||||
|     "appSortBy": "Сортувати застосунки за", |     "appSortBy": "Сортувати застосунки за", | ||||||
|     "authorName": "Автор/Назва", |     "authorName": "Автор/Назва", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Немає нових оновлень.", |     "noNewUpdates": "Немає нових оновлень.", | ||||||
|     "xHasAnUpdate": "{} має оновлення.", |     "xHasAnUpdate": "{} має оновлення.", | ||||||
|     "appsUpdated": "Застосунки оновлено", |     "appsUpdated": "Застосунки оновлено", | ||||||
|  |     "appsNotUpdated": "Не вдалося оновити програми", | ||||||
|     "appsUpdatedNotifDescription": "Повідомляє користувача, що оновлення одного чи декількох застосунків було застосовано в фоновому режимі", |     "appsUpdatedNotifDescription": "Повідомляє користувача, що оновлення одного чи декількох застосунків було застосовано в фоновому режимі", | ||||||
|     "xWasUpdatedToY": "{} було оновлено до {}.", |     "xWasUpdatedToY": "{} було оновлено до {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Не вдалося оновити {} на {}.", | ||||||
|     "errorCheckingUpdates": "Помилка перевірки оновлень", |     "errorCheckingUpdates": "Помилка перевірки оновлень", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Повідомлення, яке з'являється, коли перевірка оновлень в фоновому режимі завершується невдачею", |     "errorCheckingUpdatesNotifDescription": "Повідомлення, яке з'являється, коли перевірка оновлень в фоновому режимі завершується невдачею", | ||||||
|     "appsRemoved": "Застосунки видалено", |     "appsRemoved": "Застосунки видалено", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "Вимкнути фонові оновлення поза Wi-Fi", |     "bgUpdatesOnWiFiOnly": "Вимкнути фонові оновлення поза Wi-Fi", | ||||||
|     "autoSelectHighestVersionCode": "Автоматичний вибір APK з найвищим кодом версії", |     "autoSelectHighestVersionCode": "Автоматичний вибір APK з найвищим кодом версії", | ||||||
|     "versionExtractionRegEx": "Регулярний вираз для вилучення рядка версії", |     "versionExtractionRegEx": "Регулярний вираз для вилучення рядка версії", | ||||||
|  |     "trimVersionString": "Обрізати рядок версії за допомогою RegEx", | ||||||
|  |     "matchGroupToUseForX": "Група збігів для \"{}\"", | ||||||
|     "matchGroupToUse": "Група співпадінь для використання в регулярному виразі вилучення версії", |     "matchGroupToUse": "Група співпадінь для використання в регулярному виразі вилучення версії", | ||||||
|     "highlightTouchTargets": "Підсвічувати менш очевидні області дотику", |     "highlightTouchTargets": "Підсвічувати менш очевидні області дотику", | ||||||
|     "pickExportDir": "Вибрати каталог експорту", |     "pickExportDir": "Вибрати каталог експорту", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Підтримка фіксованих посилань на APK", |     "supportFixedAPKURL": "Підтримка фіксованих посилань на APK", | ||||||
|     "selectX": "Вибрати {}", |     "selectX": "Вибрати {}", | ||||||
|     "parallelDownloads": "Дозволити паралельні завантаження", |     "parallelDownloads": "Дозволити паралельні завантаження", | ||||||
|     "installMethod": "Метод встановлення", |     "useShizuku": "Використовуйте Shizuku або Sui для встановлення", | ||||||
|     "normal": "Звичайний", |  | ||||||
|     "root": "Root", |  | ||||||
|     "shizukuBinderNotFound": "Сумісний сервіс Shizuku не було знайдено", |     "shizukuBinderNotFound": "Сумісний сервіс Shizuku не було знайдено", | ||||||
|  |     "shizukuOld": "Стара версія Shizuku (<11) - оновіть її", | ||||||
|  |     "shizukuOldAndroidWithADB": "Shizuku працює на Android < 8.1 з ADB - оновіть Android або використовуйте Sui замість нього", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Виберіть Google Play як джерело встановлення (якщо використовується Shizuku)", | ||||||
|     "useSystemFont": "Використовувати системний шрифт", |     "useSystemFont": "Використовувати системний шрифт", | ||||||
|     "systemFontError": "Помилка завантаження системного шрифту: {}", |  | ||||||
|     "useVersionCodeAsOSVersion": "Використовувати код версії застосунку як версію, визначену операційною системою", |     "useVersionCodeAsOSVersion": "Використовувати код версії застосунку як версію, визначену операційною системою", | ||||||
|     "requestHeader": "Заголовок запиту", |     "requestHeader": "Заголовок запиту", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Використовувати останню дату завантаження ресурсу як дату випуску", |     "useLatestAssetDateAsReleaseDate": "Використовувати останню дату завантаження ресурсу як дату випуску", | ||||||
| @@ -304,6 +312,9 @@ | |||||||
|     "badDownload": "APK не вдалося розпарсити (несумісний або часткове завантаження)", |     "badDownload": "APK не вдалося розпарсити (несумісний або часткове завантаження)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "Діліться новими додатками з AppVerifier (якщо доступно)", |     "beforeNewInstallsShareToAppVerifier": "Діліться новими додатками з AppVerifier (якщо доступно)", | ||||||
|     "appVerifierInstructionToast": "Надішліть на AppVerifier, а потім поверніться сюди, коли будете готові.", |     "appVerifierInstructionToast": "Надішліть на AppVerifier, а потім поверніться сюди, коли будете готові.", | ||||||
|  |     "wiki": "Довідка/Вікі", | ||||||
|  |     "crowdsourcedConfigsLabel": "Краудсорсингові конфігурації додатків (використовуйте на свій страх і ризик)", | ||||||
|  |     "allowInsecure": "Дозволити незахищені HTTP-запити", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Видалити застосунок?", |         "one": "Видалити застосунок?", | ||||||
|         "other": "Видалити застосунки?" |         "other": "Видалити застосунки?" | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} та ще 1 застосунок було оновлено.", |         "one": "{} та ще 1 застосунок було оновлено.", | ||||||
|         "other": "{} та ще {} застосунків було оновлено." |         "other": "{} та ще {} застосунків було оновлено." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Не вдалося оновити {} та ще 1 програму.", | ||||||
|  |         "other": "Не вдалося оновити {} і {} та інші програми." | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} та ще 1 застосунок можливо було оновлено.", |         "one": "{} та ще 1 застосунок можливо було оновлено.", | ||||||
|         "other": "{} та ще {} застосунків можливо було оновлено." |         "other": "{} та ще {} застосунків можливо було оновлено." | ||||||
|   | |||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(Yêu cầu)", |     "requiredInBrackets": "(Yêu cầu)", | ||||||
|     "dropdownNoOptsError": "LỖI: TẢI XUỐNG PHẢI CÓ ÍT NHẤT MỘT LỰA CHỌN", |     "dropdownNoOptsError": "LỖI: TẢI XUỐNG PHẢI CÓ ÍT NHẤT MỘT LỰA CHỌN", | ||||||
|     "colour": "Màu sắc", |     "colour": "Màu sắc", | ||||||
|  |     "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", |     "githubStarredRepos": "Kho lưu trữ có gắn dấu sao GitHub", | ||||||
|     "uname": "Tên người dùng", |     "uname": "Tên người dùng", | ||||||
|     "wrongArgNum": "Số lượng đối số được cung cấp sai", |     "wrongArgNum": "Số lượng đối số được cung cấp sai", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "Tối", |     "dark": "Tối", | ||||||
|     "light": "Sáng", |     "light": "Sáng", | ||||||
|     "followSystem": "Theo hệ thống", |     "followSystem": "Theo hệ thống", | ||||||
|  |     "followSystemThemeExplanation": "Following system theme is possible only by using third-party applications", | ||||||
|     "useBlackTheme": "Nền đen", |     "useBlackTheme": "Nền đen", | ||||||
|     "appSortBy": "Sắp xếp ứng dụng", |     "appSortBy": "Sắp xếp ứng dụng", | ||||||
|     "authorName": "Tác giả", |     "authorName": "Tác giả", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "Không có bản cập nhật mới.", |     "noNewUpdates": "Không có bản cập nhật mới.", | ||||||
|     "xHasAnUpdate": "{} có bản cập nhật.", |     "xHasAnUpdate": "{} có bản cập nhật.", | ||||||
|     "appsUpdated": "Ứng dụng đã cập nhật ", |     "appsUpdated": "Ứng dụng đã cập nhật ", | ||||||
|  |     "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", |     "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 {}.", |     "xWasUpdatedToY": "{} đã được cập nhật thành {}.", | ||||||
|  |     "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", |     "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", |     "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ỏ", |     "appsRemoved": "Ứng dụng đã loại bỏ", | ||||||
| @@ -183,8 +189,8 @@ | |||||||
|     "disableVersionDetection": "Tắt tính năng phát hiện phiên bản", |     "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.", |     "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 {}", |     "downloadingX": "Đang tải xuống {}", | ||||||
|     "downloadX": "Download {}", |     "downloadX": "Tải xuống {}", | ||||||
|     "downloadedX": "Downloaded {}", |     "downloadedX": "Đã tải xuống {}", | ||||||
|     "releaseAsset": "Release Asset", |     "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", |     "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", |     "noAPKFound": "Không tìm thấy APK", | ||||||
| @@ -253,6 +259,8 @@ | |||||||
|     "bgUpdatesOnWiFiOnly": "Tắt cập nhật nền khi không có WiFi", |     "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", |     "autoSelectHighestVersionCode": "Tự động chọn APK mã phiên bản cao nhất", | ||||||
|     "versionExtractionRegEx": "Trích xuất phiên bản RegEx", |     "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", |     "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", |     "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", |     "pickExportDir": "Chọn thư mục xuất", | ||||||
| @@ -282,12 +290,12 @@ | |||||||
|     "supportFixedAPKURL": "Hỗ trợ URL APK cố định", |     "supportFixedAPKURL": "Hỗ trợ URL APK cố định", | ||||||
|     "selectX": "Lựa chọn {}", |     "selectX": "Lựa chọn {}", | ||||||
|     "parallelDownloads": "Cho phép tải đa luồng", |     "parallelDownloads": "Cho phép tải đa luồng", | ||||||
|     "installMethod": "Phương thức cài đặt", |     "useShizuku": "Sử dụng Shizuku hoặc Sui để cài đặt", | ||||||
|     "normal": "Mặc định", |  | ||||||
|     "root": "Root", |  | ||||||
|     "shizukuBinderNotFound": "Shizuku chưa khởi động", |     "shizukuBinderNotFound": "Shizuku chưa khởi động", | ||||||
|  |     "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", |     "useSystemFont": "Sử dụng phông chữ hệ thống", | ||||||
|     "systemFontError": "Lỗi tải 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", |     "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", | ||||||
|     "requestHeader": "Tiêu đề yêu cầu", |     "requestHeader": "Tiêu đề yêu cầu", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Sử dụng nội dung tải lên mới nhất làm ngày phát hành", |     "useLatestAssetDateAsReleaseDate": "Sử dụng nội dung tải lên mới nhất làm ngày phát hành", | ||||||
| @@ -304,6 +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)", |     "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ó)", |     "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.", |     "appVerifierInstructionToast": "Chia sẻ lên AppVerifier, sau đó quay lại đây khi sẵn sàng.", | ||||||
|  |     "wiki": "Trợ giúp/Wiki", | ||||||
|  |     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", | ||||||
|  |     "allowInsecure": "Allow insecure HTTP requests", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Gỡ ứng dụng?", |         "one": "Gỡ ứng dụng?", | ||||||
|         "other": "Gỡ ứng dụng?" |         "other": "Gỡ ứng dụng?" | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} và 1 ứng dụng khác đã được cập nhật.", |         "one": "{} và 1 ứng dụng khác đã được cập nhật.", | ||||||
|         "other": "{} và {} ứng dụng khác đã được cập nhật." |         "other": "{} và {} ứng dụng khác đã được cập nhật." | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "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": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} và 1 ứng dụng khác có thể đã được cập nhật.", |         "one": "{} và 1 ứng dụng khác có thể đã được cập nhật.", | ||||||
|         "other": "{} và {} ứng dụng khác có thể đã được cập nhật." |         "other": "{} và {} ứ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" | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -13,8 +13,8 @@ | |||||||
|     "and": "和", |     "and": "和", | ||||||
|     "githubPATLabel": "GitHub 个人访问令牌(提升 API 请求限额)", |     "githubPATLabel": "GitHub 个人访问令牌(提升 API 请求限额)", | ||||||
|     "includePrereleases": "包含预发行版", |     "includePrereleases": "包含预发行版", | ||||||
|     "fallbackToOlderReleases": "将旧发行版作为备选", |     "fallbackToOlderReleases": "将过往的发行版作为备选", | ||||||
|     "filterReleaseTitlesByRegEx": "筛选发行标题(正则表达式)", |     "filterReleaseTitlesByRegEx": "筛选发行标题的正则表达式", | ||||||
|     "invalidRegEx": "无效的正则表达式", |     "invalidRegEx": "无效的正则表达式", | ||||||
|     "noDescription": "无描述", |     "noDescription": "无描述", | ||||||
|     "cancel": "取消", |     "cancel": "取消", | ||||||
| @@ -22,6 +22,9 @@ | |||||||
|     "requiredInBrackets": "(必填)", |     "requiredInBrackets": "(必填)", | ||||||
|     "dropdownNoOptsError": "错误:下拉菜单必须包含至少一个选项", |     "dropdownNoOptsError": "错误:下拉菜单必须包含至少一个选项", | ||||||
|     "colour": "配色", |     "colour": "配色", | ||||||
|  |     "standard": "标准", | ||||||
|  |     "custom": "定制", | ||||||
|  |     "useMaterialYou": "使用 Material You", | ||||||
|     "githubStarredRepos": "已星标的 GitHub 仓库", |     "githubStarredRepos": "已星标的 GitHub 仓库", | ||||||
|     "uname": "用户名", |     "uname": "用户名", | ||||||
|     "wrongArgNum": "参数数量错误", |     "wrongArgNum": "参数数量错误", | ||||||
| @@ -42,8 +45,8 @@ | |||||||
|     "search": "搜索", |     "search": "搜索", | ||||||
|     "additionalOptsFor": "{} 的更多选项", |     "additionalOptsFor": "{} 的更多选项", | ||||||
|     "supportedSources": "支持的来源", |     "supportedSources": "支持的来源", | ||||||
|     "trackOnlyInBrackets": "(仅追踪)", |     "trackOnlyInBrackets": "(仅追踪)", | ||||||
|     "searchableInBrackets": "(可搜索)", |     "searchableInBrackets": "(可搜索)", | ||||||
|     "appsString": "应用列表", |     "appsString": "应用列表", | ||||||
|     "noApps": "无应用", |     "noApps": "无应用", | ||||||
|     "noAppsForFilter": "没有符合条件的应用", |     "noAppsForFilter": "没有符合条件的应用", | ||||||
| @@ -72,8 +75,8 @@ | |||||||
|     "unpinFromTop": "取消置顶", |     "unpinFromTop": "取消置顶", | ||||||
|     "resetInstallStatusForSelectedAppsQuestion": "是否重置选中应用的安装状态?", |     "resetInstallStatusForSelectedAppsQuestion": "是否重置选中应用的安装状态?", | ||||||
|     "installStatusOfXWillBeResetExplanation": "选中应用的安装状态将会被重置。\n\n当更新安装失败或其他问题导致 Obtainium 中的应用版本显示错误时,可以尝试通过此方法解决。", |     "installStatusOfXWillBeResetExplanation": "选中应用的安装状态将会被重置。\n\n当更新安装失败或其他问题导致 Obtainium 中的应用版本显示错误时,可以尝试通过此方法解决。", | ||||||
|     "customLinkMessage": "这些链接适用于安装了 Gettingium 的设备", |     "customLinkMessage": "分享链接仅适用于已安装 Obtainium 的设备", | ||||||
|     "shareAppConfigLinks": "将应用程序配置共享为 HTML 链接", |     "shareAppConfigLinks": "通过链接分享应用配置", | ||||||
|     "shareSelectedAppURLs": "分享选中应用的 URL", |     "shareSelectedAppURLs": "分享选中应用的 URL", | ||||||
|     "resetInstallStatus": "重置安装状态", |     "resetInstallStatus": "重置安装状态", | ||||||
|     "more": "更多", |     "more": "更多", | ||||||
| @@ -110,6 +113,7 @@ | |||||||
|     "dark": "深色", |     "dark": "深色", | ||||||
|     "light": "浅色", |     "light": "浅色", | ||||||
|     "followSystem": "跟随系统", |     "followSystem": "跟随系统", | ||||||
|  |     "followSystemThemeExplanation": "跟随系统主题仅在使用第三方应用时有效", | ||||||
|     "useBlackTheme": "使用纯黑深色主题", |     "useBlackTheme": "使用纯黑深色主题", | ||||||
|     "appSortBy": "排序依据", |     "appSortBy": "排序依据", | ||||||
|     "authorName": "作者 / 应用名称", |     "authorName": "作者 / 应用名称", | ||||||
| @@ -121,7 +125,7 @@ | |||||||
|     "bgUpdateCheckInterval": "后台更新检查间隔", |     "bgUpdateCheckInterval": "后台更新检查间隔", | ||||||
|     "neverManualOnly": "手动", |     "neverManualOnly": "手动", | ||||||
|     "appearance": "外观", |     "appearance": "外观", | ||||||
|     "showWebInAppView": "应用详情页显示来源网页", |     "showWebInAppView": "应用详情页显示来源网站内容", | ||||||
|     "pinUpdates": "将待更新应用置顶", |     "pinUpdates": "将待更新应用置顶", | ||||||
|     "updates": "更新", |     "updates": "更新", | ||||||
|     "sourceSpecific": "来源", |     "sourceSpecific": "来源", | ||||||
| @@ -131,7 +135,7 @@ | |||||||
|     "close": "关闭", |     "close": "关闭", | ||||||
|     "share": "分享", |     "share": "分享", | ||||||
|     "appNotFound": "未找到应用", |     "appNotFound": "未找到应用", | ||||||
|     "obtainiumExportHyphenatedLowercase": "获取出口", |     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||||
|     "pickAnAPK": "选择一个 APK 文件", |     "pickAnAPK": "选择一个 APK 文件", | ||||||
|     "appHasMoreThanOnePackage": "“{}”有多个架构可用:", |     "appHasMoreThanOnePackage": "“{}”有多个架构可用:", | ||||||
|     "deviceSupportsXArch": "您的设备支持 {} 架构。", |     "deviceSupportsXArch": "您的设备支持 {} 架构。", | ||||||
| @@ -143,8 +147,10 @@ | |||||||
|     "noNewUpdates": "全部应用已是最新。", |     "noNewUpdates": "全部应用已是最新。", | ||||||
|     "xHasAnUpdate": "“{}”可以更新了。", |     "xHasAnUpdate": "“{}”可以更新了。", | ||||||
|     "appsUpdated": "应用已更新", |     "appsUpdated": "应用已更新", | ||||||
|  |     "appsNotUpdated": "更新应用失败", | ||||||
|     "appsUpdatedNotifDescription": "当应用在后台安装更新时发送通知", |     "appsUpdatedNotifDescription": "当应用在后台安装更新时发送通知", | ||||||
|     "xWasUpdatedToY": "“{}”已更新至 {}。", |     "xWasUpdatedToY": "“{}”已更新至 {}。", | ||||||
|  |     "xWasNotUpdatedToY": "未能将 {} 更新为 {}。", | ||||||
|     "errorCheckingUpdates": "检查更新出错", |     "errorCheckingUpdates": "检查更新出错", | ||||||
|     "errorCheckingUpdatesNotifDescription": "当后台检查更新失败时显示的通知", |     "errorCheckingUpdatesNotifDescription": "当后台检查更新失败时显示的通知", | ||||||
|     "appsRemoved": "应用已删除", |     "appsRemoved": "应用已删除", | ||||||
| @@ -173,8 +179,8 @@ | |||||||
|     "appWithIdOrNameNotFound": "未找到符合此 ID 或名称的应用", |     "appWithIdOrNameNotFound": "未找到符合此 ID 或名称的应用", | ||||||
|     "reposHaveMultipleApps": "存储库中可能包含多个应用", |     "reposHaveMultipleApps": "存储库中可能包含多个应用", | ||||||
|     "fdroidThirdPartyRepo": "F-Droid 第三方存储库", |     "fdroidThirdPartyRepo": "F-Droid 第三方存储库", | ||||||
|     "steamMobile": "蒸汽手机", |     "steamMobile": "Steam Mobile", | ||||||
|     "steamChat": "蒸汽聊天", |     "steamChat": "Steam Chat", | ||||||
|     "install": "安装", |     "install": "安装", | ||||||
|     "markInstalled": "标记为已安装", |     "markInstalled": "标记为已安装", | ||||||
|     "update": "更新", |     "update": "更新", | ||||||
| @@ -185,7 +191,7 @@ | |||||||
|     "downloadingX": "正在下载“{}”", |     "downloadingX": "正在下载“{}”", | ||||||
|     "downloadX": "下载 {}", |     "downloadX": "下载 {}", | ||||||
|     "downloadedX": "下载 {}", |     "downloadedX": "下载 {}", | ||||||
|     "releaseAsset": "释放资产", |     "releaseAsset": "发行版附件", | ||||||
|     "downloadNotifDescription": "提示应用的下载进度", |     "downloadNotifDescription": "提示应用的下载进度", | ||||||
|     "noAPKFound": "未找到 APK 文件", |     "noAPKFound": "未找到 APK 文件", | ||||||
|     "noVersionDetection": "禁用版本检测", |     "noVersionDetection": "禁用版本检测", | ||||||
| @@ -195,14 +201,14 @@ | |||||||
|     "noCategory": "无类别", |     "noCategory": "无类别", | ||||||
|     "noCategories": "无类别", |     "noCategories": "无类别", | ||||||
|     "deleteCategoriesQuestion": "是否删除选中的类别?", |     "deleteCategoriesQuestion": "是否删除选中的类别?", | ||||||
|     "categoryDeleteWarning": "被删除类别下的应用将恢复为未分类状态。", |     "categoryDeleteWarning": "被删除类别的应用将恢复为未分类状态。", | ||||||
|     "addCategory": "添加类别", |     "addCategory": "添加类别", | ||||||
|     "label": "标签", |     "label": "标签", | ||||||
|     "language": "语言", |     "language": "语言", | ||||||
|     "copiedToClipboard": "已复制至剪贴板", |     "copiedToClipboard": "已复制至剪贴板", | ||||||
|     "storagePermissionDenied": "已拒绝授予存储权限", |     "storagePermissionDenied": "已拒绝授予存储权限", | ||||||
|     "selectedCategorizeWarning": "这将覆盖选中应用当前的类别设置。", |     "selectedCategorizeWarning": "这将覆盖选中应用当前的类别设置。", | ||||||
|     "filterAPKsByRegEx": "筛选 APK 文件(正则表达式)", |     "filterAPKsByRegEx": "筛选 APK 文件的正则表达式", | ||||||
|     "removeFromObtainium": "从 Obtainium 中删除", |     "removeFromObtainium": "从 Obtainium 中删除", | ||||||
|     "uninstallFromDevice": "从设备中卸载", |     "uninstallFromDevice": "从设备中卸载", | ||||||
|     "onlyWorksWithNonVersionDetectApps": "仅适用于禁用版本检测的应用。", |     "onlyWorksWithNonVersionDetectApps": "仅适用于禁用版本检测的应用。", | ||||||
| @@ -236,29 +242,31 @@ | |||||||
|     "addInfoInSettings": "在“设置”中添加此凭据。", |     "addInfoInSettings": "在“设置”中添加此凭据。", | ||||||
|     "githubSourceNote": "使用访问令牌可避免触发 GitHub 的 API 请求限制。", |     "githubSourceNote": "使用访问令牌可避免触发 GitHub 的 API 请求限制。", | ||||||
|     "sortByLastLinkSegment": "仅根据链接的末尾部分进行筛选", |     "sortByLastLinkSegment": "仅根据链接的末尾部分进行筛选", | ||||||
|     "filterReleaseNotesByRegEx": "筛选发行说明(正则表达式)", |     "filterReleaseNotesByRegEx": "筛选发行说明的正则表达式", | ||||||
|     "customLinkFilterRegex": "筛选自定义来源的 APK 文件链接\n(正则表达式,默认匹配模式为“.apk$”)", |     "customLinkFilterRegex": "筛选自定义来源的 APK 文件链接\n(正则表达式,默认匹配模式为“.apk$”)", | ||||||
|     "appsPossiblyUpdated": "已尝试更新应用", |     "appsPossiblyUpdated": "已尝试更新应用", | ||||||
|     "appsPossiblyUpdatedNotifDescription": "当应用已尝试在后台更新时发送通知", |     "appsPossiblyUpdatedNotifDescription": "当应用已尝试在后台更新时发送通知", | ||||||
|     "xWasPossiblyUpdatedToY": "已尝试将“{}”更新至 {}。", |     "xWasPossiblyUpdatedToY": "已尝试将“{}”更新至 {}。", | ||||||
|     "enableBackgroundUpdates": "启用后台更新", |     "enableBackgroundUpdates": "启用全局后台更新", | ||||||
|     "backgroundUpdateReqsExplanation": "后台更新未必适用于所有的应用。", |     "backgroundUpdateReqsExplanation": "后台更新未必适用于所有的应用。", | ||||||
|     "backgroundUpdateLimitsExplanation": "只有在启动 Obtainium 时才能确认安装是否成功。", |     "backgroundUpdateLimitsExplanation": "只有在启动 Obtainium 时才能确认安装是否成功。", | ||||||
|     "verifyLatestTag": "验证“Latest”标签", |     "verifyLatestTag": "验证“Latest”标签", | ||||||
|     "intermediateLinkRegex": "筛选中转链接(正则表达式)", |     "intermediateLinkRegex": "筛选中转链接的正则表达式", | ||||||
|     "filterByLinkText": "根据链接文本进行筛选", |     "filterByLinkText": "根据链接文本进行筛选", | ||||||
|     "intermediateLinkNotFound": "未找到中转链接", |     "intermediateLinkNotFound": "未找到中转链接", | ||||||
|     "intermediateLink": "中转链接", |     "intermediateLink": "中转链接", | ||||||
|     "exemptFromBackgroundUpdates": "禁用后台更新(如果已经全局启用)", |     "exemptFromBackgroundUpdates": "禁用后台更新(仅此应用生效,即使已启用全局后台更新)", | ||||||
|     "bgUpdatesOnWiFiOnly": "未连接 Wi-Fi 时禁用后台更新", |     "bgUpdatesOnWiFiOnly": "未连接 Wi-Fi 时禁用后台更新", | ||||||
|     "autoSelectHighestVersionCode": "自动选择内部版本号最高的 APK 文件", |     "autoSelectHighestVersionCode": "自动选择内部版本号最高的 APK 文件", | ||||||
|     "versionExtractionRegEx": "版本号提取规则(正则表达式)", |     "versionExtractionRegEx": "提取版本号的正则表达式", | ||||||
|     "matchGroupToUse": "引用的捕获组", |     "trimVersionString": "使用 RegEx 修剪版本字符串", | ||||||
|     "highlightTouchTargets": "突出展示不明显的触摸区域", |     "matchGroupToUseForX": "用于\"{}\"的匹配组", | ||||||
|  |     "matchGroupToUse": "从上述匹配结果中引用的捕获组", | ||||||
|  |     "highlightTouchTargets": "突出展示不明显的可交互区域", | ||||||
|     "pickExportDir": "选择导出文件夹", |     "pickExportDir": "选择导出文件夹", | ||||||
|     "autoExportOnChanges": "数据变更时自动导出", |     "autoExportOnChanges": "数据变更时自动导出", | ||||||
|     "includeSettings": "同时导出应用设置", |     "includeSettings": "同时导出应用设置", | ||||||
|     "filterVersionsByRegEx": "筛选版本号(正则表达式)", |     "filterVersionsByRegEx": "筛选版本号的正则表达式", | ||||||
|     "trySelectingSuggestedVersionCode": "尝试选择推荐版本的 APK 文件", |     "trySelectingSuggestedVersionCode": "尝试选择推荐版本的 APK 文件", | ||||||
|     "dontSortReleasesList": "保持来自 API 的发行顺序", |     "dontSortReleasesList": "保持来自 API 的发行顺序", | ||||||
|     "reverseSort": "反转排序", |     "reverseSort": "反转排序", | ||||||
| @@ -282,15 +290,15 @@ | |||||||
|     "supportFixedAPKURL": "支持固定的 APK 文件链接", |     "supportFixedAPKURL": "支持固定的 APK 文件链接", | ||||||
|     "selectX": "选择{}", |     "selectX": "选择{}", | ||||||
|     "parallelDownloads": "启用并行下载", |     "parallelDownloads": "启用并行下载", | ||||||
|     "installMethod": "安装方式", |     "useShizuku": "使用 Shizuku 或 Sui 安装", | ||||||
|     "normal": "常规", |  | ||||||
|     "root": "root", |  | ||||||
|     "shizukuBinderNotFound": "未发现兼容的 Shizuku 服务", |     "shizukuBinderNotFound": "未发现兼容的 Shizuku 服务", | ||||||
|  |     "shizukuOld": "Shizuku 版本过低(<11)- 请更新", | ||||||
|  |     "shizukuOldAndroidWithADB": "正在低版本 Android(<8.1)系统中以 ADB 模式运行 Shizuku - 请更新 Android 系统版本或使用 Sui 代替", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "将安装来源伪装为 Google Play(需要使用 Shizuku)", | ||||||
|     "useSystemFont": "使用系统字体", |     "useSystemFont": "使用系统字体", | ||||||
|     "systemFontError": "加载系统字体出错:{}", |  | ||||||
|     "useVersionCodeAsOSVersion": "使用内部版本号代替应用定义的版本号", |     "useVersionCodeAsOSVersion": "使用内部版本号代替应用定义的版本号", | ||||||
|     "requestHeader": "请求标头", |     "requestHeader": "请求标头", | ||||||
|     "useLatestAssetDateAsReleaseDate": "使用最近文件上传时间作为发行日期", |     "useLatestAssetDateAsReleaseDate": "使用最新文件上传时间作为发行日期", | ||||||
|     "defaultPseudoVersioningMethod": "默认虚拟版本方案", |     "defaultPseudoVersioningMethod": "默认虚拟版本方案", | ||||||
|     "partialAPKHash": "APK 文件散列值片段", |     "partialAPKHash": "APK 文件散列值片段", | ||||||
|     "APKLinkHash": "APK 文件链接散列值", |     "APKLinkHash": "APK 文件链接散列值", | ||||||
| @@ -302,19 +310,22 @@ | |||||||
|     "note": "备注", |     "note": "备注", | ||||||
|     "selfHostedNote": "可以通过“{}”下拉菜单来指向任意来源的自托管/自定义实例。", |     "selfHostedNote": "可以通过“{}”下拉菜单来指向任意来源的自托管/自定义实例。", | ||||||
|     "badDownload": "无法解析 APK 文件(不兼容或文件不完整)", |     "badDownload": "无法解析 APK 文件(不兼容或文件不完整)", | ||||||
|     "beforeNewInstallsShareToAppVerifier": "与 AppVerifier 共享新应用程序(如有)", |     "beforeNewInstallsShareToAppVerifier": "通过 AppVerifier 校验新应用(如果可用)", | ||||||
|     "appVerifierInstructionToast": "分享到 AppVerifier,准备就绪后返回此处。", |     "appVerifierInstructionToast": "分享至 AppVerifier,完成后返回此处。", | ||||||
|  |     "wiki": "帮助/Wiki", | ||||||
|  |     "crowdsourcedConfigsLabel": "众包应用程序配置(使用风险自负)", | ||||||
|  |     "allowInsecure": "允许不安全的 HTTP 请求", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "是否删除应用?", |         "one": "是否删除应用?", | ||||||
|         "other": "是否删除应用?" |         "other": "是否删除应用?" | ||||||
|     }, |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "API 请求过于频繁(速率限制)- 在 {} 分钟后重试", |         "one": "API 请求过于频繁(速率限制)- 请在 {} 分钟后重试", | ||||||
|         "other": "API 请求过于频繁(速率限制)- 在 {} 分钟后重试" |         "other": "API 请求过于频繁(速率限制)- 请在 {} 分钟后重试" | ||||||
|     }, |     }, | ||||||
|     "bgUpdateGotErrorRetryInMinutes": { |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|         "one": "后台更新检查遇到了“{}”问题,预定于 {} 分钟后重试", |         "one": "后台更新检查遇到了“{}”问题,将于 {} 分钟后重试", | ||||||
|         "other": "后台更新检查遇到了“{}”问题,预定于 {} 分钟后重试" |         "other": "后台更新检查遇到了“{}”问题,将于 {} 分钟后重试" | ||||||
|     }, |     }, | ||||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|         "one": "后台检查发现 {} 个应用更新 - 如有需要将发送通知", |         "one": "后台检查发现 {} 个应用更新 - 如有需要将发送通知", | ||||||
| @@ -352,6 +363,10 @@ | |||||||
|         "one": "{} 和另外 1 个应用已更新。", |         "one": "{} 和另外 1 个应用已更新。", | ||||||
|         "other": "“{}”和另外 {} 个应用已更新。" |         "other": "“{}”和另外 {} 个应用已更新。" | ||||||
|     }, |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "{} 和另外 1 个应用更新失败。", | ||||||
|  |         "other": "{} 和另外 {} 个应用更新失败。" | ||||||
|  |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} 和另外 1 个应用已尝试更新。", |         "one": "{} 和另外 1 个应用已尝试更新。", | ||||||
|         "other": "“{}”和另外 {} 个应用已尝试更新。" |         "other": "“{}”和另外 {} 个应用已尝试更新。" | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ | |||||||
| 		<ul> | 		<ul> | ||||||
| 			<li>GitHub</li> | 			<li>GitHub</li> | ||||||
| 			<li>GitLab</li> | 			<li>GitLab</li> | ||||||
| 			<li>Codeberg</li> | 			<li>Forgejo (Codeberg)</li> | ||||||
| 			<li>F-Droid</li> | 			<li>F-Droid</li> | ||||||
| 			<li>Third Party F-Droid Repos</li> | 			<li>Third Party F-Droid Repos</li> | ||||||
| 			<li>IzzyOnDroid</li> | 			<li>IzzyOnDroid</li> | ||||||
|   | |||||||
							
								
								
									
										54
									
								
								fastlane/metadata/android/ru/full_description.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								fastlane/metadata/android/ru/full_description.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | |||||||
|  | <p>Obtainium позволяет вам устанавливать и обновлять приложения прямо с их объявлений о выпусках и получать уведомления о новых выпусках.</p> | ||||||
|  | <p>Для деталей читайте <a href="https://github.com/ImranR98/Obtainium/wiki">Вики</a></p> | ||||||
|  | <p> | ||||||
|  | 	<b>Поддерживаемые источники приложений:</b> | ||||||
|  | </p> | ||||||
|  | <ul> | ||||||
|  | 	<li> | ||||||
|  | 		<p>Свободное ПО - Общие:</p> | ||||||
|  | 		<ul> | ||||||
|  | 			<li>GitHub</li> | ||||||
|  | 			<li>GitLab</li> | ||||||
|  | 			<li>Forgejo (Codeberg)</li> | ||||||
|  | 			<li>F-Droid</li> | ||||||
|  | 			<li>Third Party F-Droid Repos</li> | ||||||
|  | 			<li>IzzyOnDroid</li> | ||||||
|  | 			<li>SourceForge</li> | ||||||
|  | 			<li>SourceHut</li> | ||||||
|  | 		</ul> | ||||||
|  | 	</li> | ||||||
|  | 	<li> | ||||||
|  | 		<p>Другие - Общие:</p> | ||||||
|  | 		<ul> | ||||||
|  | 			<li>APKPure</li> | ||||||
|  | 			<li>Aptoide</li> | ||||||
|  | 			<li>Uptodowng</li> | ||||||
|  | 			<li>APKMirror (Track-Only)</li> | ||||||
|  | 			<li>Huawei AppGallery</li> | ||||||
|  | 			<li>Jenkins Jobs</li> | ||||||
|  | 		</ul> | ||||||
|  | 	</li> | ||||||
|  | 	<li> | ||||||
|  | 		<p>Свободное ПО - Для отдельных приложений:</p> | ||||||
|  | 		<ul> | ||||||
|  | 			<li>Mullvad</li> | ||||||
|  | 			<li>Signal</li> | ||||||
|  | 			<li>VLC</li> | ||||||
|  | 		</ul> | ||||||
|  | 	</li> | ||||||
|  | 	<li> | ||||||
|  | 		<p>Другие - Для отдельных приложений:</p> | ||||||
|  | 		<ul> | ||||||
|  | 			<li>WhatsApp</li> | ||||||
|  | 			<li>Telegram App</li> | ||||||
|  | 			<li>Neutron Code</li> | ||||||
|  | 		</ul> | ||||||
|  | 	</li> | ||||||
|  | 	<li><p>"HTML" (Подстраховка): Любой другой URL-адрес, который возвращает HTML-страницу со ссылками на APK-файлы.</p></li> | ||||||
|  | </ul> | ||||||
|  | <p> | ||||||
|  | 	<b>Ограничения:</b> | ||||||
|  | </p> | ||||||
|  | <p> | ||||||
|  | 	Для некоторых источников данные собираются с помощью веб-скрапинга и могут легко сломаться из-за изменений в дизайне веб-сайта. В таких случаях более надежные методы могут быть недоступны. | ||||||
|  | </p> | ||||||
							
								
								
									
										1
									
								
								fastlane/metadata/android/ru/short_description.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								fastlane/metadata/android/ru/short_description.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | Получайте обновления приложений прямо из источника | ||||||
| @@ -10,7 +10,7 @@ class APKCombo extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/+[^/]+/+[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}/+[^/]+/+[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|   | |||||||
| @@ -32,7 +32,7 @@ class APKMirror extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/apk/[^/]+/[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}/apk/[^/]+/[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
|  | import 'package:device_info_plus/device_info_plus.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:html/parser.dart'; | import 'package:html/parser.dart'; | ||||||
|  | import 'package:obtainium/app_sources/html.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
| @@ -27,16 +29,17 @@ class APKPure extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegExB = RegExp( |     RegExp standardUrlRegExB = RegExp( | ||||||
|         '^https?://m.${getSourceRegex(hosts)}/+[^/]+/+[^/]+(/+[^/]+)?', |         '^https?://m.${getSourceRegex(hosts)}(/+[^/]{2})?/+[^/]+/+[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|     RegExpMatch? match = standardUrlRegExB.firstMatch(url); |     RegExpMatch? match = standardUrlRegExB.firstMatch(url); | ||||||
|     if (match != null) { |     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( |     RegExp standardUrlRegExA = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/+[^/]+/+[^/]+(/+[^/]+)?', |         '^https?://(www\\.)?${getSourceRegex(hosts)}(/+[^/]{2})?/+[^/]+/+[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|     match = standardUrlRegExA.firstMatch(url); |     match = standardUrlRegExA.firstMatch(url); | ||||||
|     if (match == null) { |     if (match == null) { | ||||||
| @@ -58,40 +61,109 @@ class APKPure extends AppSource { | |||||||
|   ) async { |   ) async { | ||||||
|     String appId = (await tryInferringAppId(standardUrl))!; |     String appId = (await tryInferringAppId(standardUrl))!; | ||||||
|     String host = Uri.parse(standardUrl).host; |     String host = Uri.parse(standardUrl).host; | ||||||
|     var res = await sourceRequest('$standardUrl/download', additionalSettings); |  | ||||||
|     var resChangelog = await sourceRequest(standardUrl, additionalSettings); |     var res0 = await sourceRequest('$standardUrl/versions', additionalSettings); | ||||||
|     if (res.statusCode == 200 && resChangelog.statusCode == 200) { |     var versionLinks = await grabLinksCommon(res0, { | ||||||
|       var html = parse(res.body); |       'skipSort': true, | ||||||
|       var htmlChangelog = parse(resChangelog.body); |       'customLinkFilterRegex': '$standardUrl/download/[^/]+\$' | ||||||
|       String? version = html.querySelector('span.info-sdk span')?.text.trim(); |     }); | ||||||
|       if (version == null) { |  | ||||||
|         throw NoVersionError(); |     var supportedArchs = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||||
|       } |  | ||||||
|       String? dateString = |     if (additionalSettings['autoApkFilterByArch'] != true) { | ||||||
|           html.querySelector('span.info-other span.date')?.text.trim(); |       // No need to request multiple versions when we're not going to filter them (always pick the top one) | ||||||
|       DateTime? releaseDate = parseDateTimeMMMddCommayyyy(dateString); |       versionLinks = versionLinks.sublist(0, 1); | ||||||
|       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); |  | ||||||
|     } |     } | ||||||
|  |     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 |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://([^\\.]+\\.){2,}${getSourceRegex(hosts)}', |         '^https?://([^\\.]+\\.){2,}${getSourceRegex(hosts)}', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import 'package:obtainium/providers/source_provider.dart'; | |||||||
| class Codeberg extends AppSource { | class Codeberg extends AppSource { | ||||||
|   GitHub gh = GitHub(); |   GitHub gh = GitHub(); | ||||||
|   Codeberg() { |   Codeberg() { | ||||||
|  |     name = 'Forgejo (Codeberg)'; | ||||||
|     hosts = ['codeberg.org']; |     hosts = ['codeberg.org']; | ||||||
|  |  | ||||||
|     additionalSourceAppSpecificSettingFormItems = |     additionalSourceAppSpecificSettingFormItems = | ||||||
| @@ -15,7 +16,7 @@ class Codeberg extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|   | |||||||
| @@ -1,12 +1,12 @@ | |||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:obtainium/app_sources/html.dart'; | import 'package:obtainium/app_sources/html.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
| class DirectAPKLink extends AppSource { | class DirectAPKLink extends AppSource { | ||||||
|   HTML html = HTML(); |   HTML html = HTML(); | ||||||
|  |  | ||||||
|   DirectAPKLink() { |   DirectAPKLink() { | ||||||
|     neverAutoSelect = true; |  | ||||||
|     name = tr('directAPKLink'); |     name = tr('directAPKLink'); | ||||||
|     additionalSourceAppSpecificSettingFormItems = html |     additionalSourceAppSpecificSettingFormItems = html | ||||||
|         .additionalSourceAppSpecificSettingFormItems |         .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 |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
|   | |||||||
| @@ -38,7 +38,7 @@ class FDroid extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegExB = RegExp( |     RegExp standardUrlRegExB = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/+[^/]+/+packages/+[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}/+[^/]+/+packages/+[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ class FDroidRepo extends AppSource { | |||||||
|   FDroidRepo() { |   FDroidRepo() { | ||||||
|     name = tr('fdroidThirdPartyRepo'); |     name = tr('fdroidThirdPartyRepo'); | ||||||
|     canSearch = true; |     canSearch = true; | ||||||
|     excludeFromMassSearch = true; |     includeAdditionalOptsInMainSearch = true; | ||||||
|     neverAutoSelect = true; |     neverAutoSelect = true; | ||||||
|     showReleaseDateAsVersionToggle = true; |     showReleaseDateAsVersionToggle = true; | ||||||
|  |  | ||||||
| @@ -43,7 +43,7 @@ class FDroidRepo extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     var standardUri = Uri.parse(url); |     var standardUri = Uri.parse(url); | ||||||
|     var pathSegments = standardUri.pathSegments; |     var pathSegments = standardUri.pathSegments; | ||||||
|     if (pathSegments.isNotEmpty && pathSegments.last == 'index.xml') { |     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 |   @override | ||||||
|   App endOfGetAppChanges(App app) { |   App endOfGetAppChanges(App app) { | ||||||
|     var uri = Uri.parse(app.url); |     var uri = Uri.parse(app.url); | ||||||
| @@ -142,6 +163,7 @@ class FDroidRepo extends AppSource { | |||||||
|     if (appIdOrName == null) { |     if (appIdOrName == null) { | ||||||
|       throw NoReleasesError(); |       throw NoReleasesError(); | ||||||
|     } |     } | ||||||
|  |     additionalSettings['appIdOrName'] = appIdOrName; | ||||||
|     var res = |     var res = | ||||||
|         await sourceRequestWithURLVariants(standardUrl, additionalSettings); |         await sourceRequestWithURLVariants(standardUrl, additionalSettings); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|   | |||||||
| @@ -154,7 +154,7 @@ class GitHub extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
| @@ -171,7 +171,7 @@ class GitHub extends AppSource { | |||||||
|       {bool forAPKDownload = false}) async { |       {bool forAPKDownload = false}) async { | ||||||
|     var token = await getTokenIfAny(additionalSettings); |     var token = await getTokenIfAny(additionalSettings); | ||||||
|     var headers = <String, String>{}; |     var headers = <String, String>{}; | ||||||
|     if (token != null) { |     if (token != null && token.isNotEmpty) { | ||||||
|       headers[HttpHeaders.authorizationHeader] = 'Token $token'; |       headers[HttpHeaders.authorizationHeader] = 'Token $token'; | ||||||
|     } |     } | ||||||
|     if (forAPKDownload == true) { |     if (forAPKDownload == true) { | ||||||
| @@ -273,10 +273,11 @@ class GitHub extends AppSource { | |||||||
|  |  | ||||||
|       List<MapEntry<String, String>> getReleaseAssetUrls(dynamic release) => |       List<MapEntry<String, String>> getReleaseAssetUrls(dynamic release) => | ||||||
|           (release['assets'] as List<dynamic>?)?.map((e) { |           (release['assets'] as List<dynamic>?)?.map((e) { | ||||||
|             return (e['name'] != null) && |             var url = !e['name'].toString().toLowerCase().endsWith('.apk') | ||||||
|                     ((e['url'] ?? e['browser_download_url']) != null) |                 ? (e['browser_download_url'] ?? e['url']) | ||||||
|                 ? MapEntry(e['name'] as String, |                 : (e['url'] ?? e['browser_download_url']); | ||||||
|                     (e['url'] ?? e['browser_download_url']) as String) |             return (e['name'] != null) && (url != null) | ||||||
|  |                 ? MapEntry(e['name'] as String, url as String) | ||||||
|                 : const MapEntry('', ''); |                 : const MapEntry('', ''); | ||||||
|           }).toList() ?? |           }).toList() ?? | ||||||
|           []; |           []; | ||||||
| @@ -284,7 +285,9 @@ class GitHub extends AppSource { | |||||||
|       DateTime? getPublishDateFromRelease(dynamic rel) => |       DateTime? getPublishDateFromRelease(dynamic rel) => | ||||||
|           rel?['published_at'] != null |           rel?['published_at'] != null | ||||||
|               ? DateTime.parse(rel['published_at']) |               ? DateTime.parse(rel['published_at']) | ||||||
|               : null; |               : rel?['commit']?['created'] != null | ||||||
|  |                   ? DateTime.parse(rel['commit']['created']) | ||||||
|  |                   : null; | ||||||
|       DateTime? getNewestAssetDateFromRelease(dynamic rel) { |       DateTime? getNewestAssetDateFromRelease(dynamic rel) { | ||||||
|         var t = (rel['assets'] as List<dynamic>?) |         var t = (rel['assets'] as List<dynamic>?) | ||||||
|             ?.map((e) { |             ?.map((e) { | ||||||
|   | |||||||
| @@ -52,7 +52,7 @@ class GitLab extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', | ||||||
|         caseSensitive: false); |         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 |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
| @@ -123,6 +131,18 @@ class GitLab extends AppSource { | |||||||
|  |  | ||||||
|     bool trackOnly = additionalSettings['trackOnly'] == true; |     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 |     // Request data from REST API | ||||||
|     Response res = await sourceRequest( |     Response res = await sourceRequest( | ||||||
|         'https://${hosts[0]}/api/v4/projects/${names.author}%2F${names.name}/${trackOnly ? 'repository/tags' : 'releases'}?$optionalAuth', |         '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') |               .join('.apk\n') | ||||||
|               .split('\n') |               .split('\n') | ||||||
|               .where((s) => s.startsWith('/uploads/') && s.endsWith('apk')) |               .where((s) => s.startsWith('/uploads/') && s.endsWith('apk')) | ||||||
|               .map((s) => '$standardUrl$s') |               .map((s) => 'https://${hosts[0]}/-/project/$projectId$s') | ||||||
|               .toList(); |               .toList(); | ||||||
|       var apkUrlsSet = apkUrlsFromAssets.toSet(); |       var apkUrlsSet = apkUrlsFromAssets.toSet(); | ||||||
|       apkUrlsSet.addAll(uploadedAPKsFromDescription); |       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 = |       DateTime? releaseDate = | ||||||
|           releaseDateString != null ? DateTime.parse(releaseDateString) : null; |           releaseDateString != null ? DateTime.parse(releaseDateString) : null; | ||||||
|       return APKDetails( |       return APKDetails( | ||||||
| @@ -181,7 +202,7 @@ class GitLab extends AppSource { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     finalResult.apkUrls = finalResult.apkUrls.map((apkUrl) { |     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)) { |           .hasMatch(apkUrl.value)) { | ||||||
|         return MapEntry( |         return MapEntry( | ||||||
|             apkUrl.key, apkUrl.value.replaceFirst('/file/', '/raw/')); |             apkUrl.key, apkUrl.value.replaceFirst('/file/', '/raw/')); | ||||||
|   | |||||||
| @@ -92,7 +92,89 @@ bool _isNumeric(String s) { | |||||||
|   return s.codeUnitAt(0) >= 48 && s.codeUnitAt(0) <= 57; |   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 { | 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 = [ |   var finalStepFormitems = [ | ||||||
|     [ |     [ | ||||||
|       GeneratedFormTextField('customLinkFilterRegex', |       GeneratedFormTextField('customLinkFilterRegex', | ||||||
| @@ -206,69 +288,10 @@ class HTML extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     return url; |     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) => |  | ||||||
|               reg.hasMatch(filterLinkByText ? element.value : element.key)) |  | ||||||
|           .toList(); |  | ||||||
|     } else { |  | ||||||
|       links = allLinks |  | ||||||
|           .where((element) => |  | ||||||
|               Uri.parse(filterLinkByText ? element.value : element.key) |  | ||||||
|                   .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 |   @override | ||||||
|   Future<APKDetails> getLatestAPKDetails( |   Future<APKDetails> getLatestAPKDetails( | ||||||
|     String standardUrl, |     String standardUrl, | ||||||
| @@ -309,17 +332,27 @@ class HTML extends AppSource { | |||||||
|       links = [MapEntry(currentUrl, currentUrl)]; |       links = [MapEntry(currentUrl, currentUrl)]; | ||||||
|     } |     } | ||||||
|     var rel = links.last.key; |     var rel = links.last.key; | ||||||
|  |     var relDecoded = rel; | ||||||
|  |     try { | ||||||
|  |       relDecoded = Uri.decodeFull(rel); | ||||||
|  |     } catch (e) { | ||||||
|  |       // Some links may not have valid encoding | ||||||
|  |     } | ||||||
|     String? version; |     String? version; | ||||||
|     version = extractVersion( |     version = extractVersion( | ||||||
|         additionalSettings['versionExtractionRegEx'] as String?, |         additionalSettings['versionExtractionRegEx'] as String?, | ||||||
|         additionalSettings['matchGroupToUse'] as String?, |         additionalSettings['matchGroupToUse'] as String?, | ||||||
|         additionalSettings['versionExtractWholePage'] == true |         additionalSettings['versionExtractWholePage'] == true | ||||||
|             ? versionExtractionWholePageString |             ? versionExtractionWholePageString | ||||||
|             : rel); |             : relDecoded); | ||||||
|     version ??= |     version ??= additionalSettings['defaultPseudoVersioningMethod'] == | ||||||
|         additionalSettings['defaultPseudoVersioningMethod'] == 'APKLinkHash' |             'APKLinkHash' | ||||||
|             ? rel.hashCode.toString() |         ? rel.hashCode.toString() | ||||||
|             : (await checkPartialDownloadHashDynamic(rel)).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(), |     return APKDetails(version, [rel].map((e) => MapEntry(e, e)).toList(), | ||||||
|         AppNames(uri.host, tr('app'))); |         AppNames(uri.host, tr('app'))); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -6,15 +6,15 @@ import 'package:obtainium/providers/source_provider.dart'; | |||||||
| class HuaweiAppGallery extends AppSource { | class HuaweiAppGallery extends AppSource { | ||||||
|   HuaweiAppGallery() { |   HuaweiAppGallery() { | ||||||
|     name = 'Huawei AppGallery'; |     name = 'Huawei AppGallery'; | ||||||
|     hosts = ['appgallery.huawei.com']; |     hosts = ['appgallery.huawei.com', 'appgallery.cloud.huawei.com']; | ||||||
|     versionDetectionDisallowed = true; |     versionDetectionDisallowed = true; | ||||||
|     showReleaseDateAsVersionToggle = true; |     showReleaseDateAsVersionToggle = true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/app/[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}(/#)?/(app|appdl)/[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|     RegExpMatch? match = standardUrlRegEx.firstMatch(url); |     RegExpMatch? match = standardUrlRegEx.firstMatch(url); | ||||||
|     if (match == null) { |     if (match == null) { | ||||||
| @@ -24,7 +24,7 @@ class HuaweiAppGallery extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   getDlUrl(String standardUrl) => |   getDlUrl(String standardUrl) => | ||||||
|       'https://${hosts[0].replaceAll('appgallery.', 'appgallery.cloud.')}/appdl/${standardUrl.split('/').last}'; |       'https://${hosts[0].replaceAll('appgallery.huawei', 'appgallery.cloud.huawei')}/appdl/${standardUrl.split('/').last}'; | ||||||
|  |  | ||||||
|   requestAppdlRedirect( |   requestAppdlRedirect( | ||||||
|       String dlUrl, Map<String, dynamic> additionalSettings) async { |       String dlUrl, Map<String, dynamic> additionalSettings) async { | ||||||
| @@ -73,21 +73,23 @@ class HuaweiAppGallery extends AppSource { | |||||||
|       throw NoReleasesError(); |       throw NoReleasesError(); | ||||||
|     } |     } | ||||||
|     String appId = appIdFromRedirectDlUrl(res.headers['location']!); |     String appId = appIdFromRedirectDlUrl(res.headers['location']!); | ||||||
|  |     if (appId.isEmpty) { | ||||||
|  |       throw NoReleasesError(); | ||||||
|  |     } | ||||||
|     var relDateStr = |     var relDateStr = | ||||||
|         res.headers['location']?.split('?')[0].split('.').reversed.toList()[1]; |         res.headers['location']?.split('?')[0].split('.').reversed.toList()[1]; | ||||||
|     var relDateStrAdj = relDateStr?.split(''); |     if (relDateStr == null || relDateStr.length != 10) { | ||||||
|     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').parse(relDateStrAdj.join('')); |  | ||||||
|     if (relDateStr == null) { |  | ||||||
|       throw NoVersionError(); |       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( |     return APKDetails( | ||||||
|         relDateStr, [MapEntry('$appId.apk', dlUrl)], AppNames(name, appId), |         relDateStr, [MapEntry('$appId.apk', dlUrl)], AppNames(name, appId), | ||||||
|         releaseDate: relDate); |         releaseDate: relDate); | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ class IzzyOnDroid extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegExA = RegExp( |     RegExp standardUrlRegExA = RegExp( | ||||||
|         '^https?://android.${getSourceRegex(hosts)}/repo/apk/[^/]+', |         '^https?://android.${getSourceRegex(hosts)}/repo/apk/[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ class Mullvad extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}', |         '^https?://(www\\.)?${getSourceRegex(hosts)}', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ class NeutronCode extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/downloads/file/[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}/downloads/file/[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ class Signal extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     return 'https://${hosts[0]}'; |     return 'https://${hosts[0]}'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ class SourceForge extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     var sourceRegex = getSourceRegex(hosts); |     var sourceRegex = getSourceRegex(hosts); | ||||||
|     RegExp standardUrlRegExC = |     RegExp standardUrlRegExC = | ||||||
|         RegExp('^https?://(www\\.)?$sourceRegex/p/.+', caseSensitive: false); |         RegExp('^https?://(www\\.)?$sourceRegex/p/.+', caseSensitive: false); | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ class SourceHut extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ class SteamMobile extends AppSource { | |||||||
|   final apks = {'steam': tr('steamMobile'), 'steam-chat-app': tr('steamChat')}; |   final apks = {'steam': tr('steamMobile'), 'steam-chat-app': tr('steamChat')}; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     return 'https://${hosts[0]}'; |     return 'https://${hosts[0]}'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ class TelegramApp extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     return 'https://${hosts[0]}'; |     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; |     allowSubDomains = true; | ||||||
|     naiveStandardVersionDetection = true; |     naiveStandardVersionDetection = true; | ||||||
|     showReleaseDateAsVersionToggle = true; |     showReleaseDateAsVersionToggle = true; | ||||||
|  |     urlsAlwaysHaveExtension = true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://([^\\.]+\\.){2,}${getSourceRegex(hosts)}', |         '^https?://([^\\.]+\\.){2,}${getSourceRegex(hosts)}', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
| @@ -39,20 +40,29 @@ class Uptodown extends AppSource { | |||||||
|     } |     } | ||||||
|     var html = parse(res.body); |     var html = parse(res.body); | ||||||
|     String? version = html.querySelector('div.version')?.innerHtml; |     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? name = html.querySelector('#detail-app-name')?.innerHtml.trim(); | ||||||
|     String? author = html.querySelector('#author-link')?.innerHtml.trim(); |     String? author = html.querySelector('#author-link')?.innerHtml.trim(); | ||||||
|     var detailElements = html.querySelectorAll('#technical-information td'); |     var detailElements = html.querySelectorAll('#technical-information td'); | ||||||
|     String? appId = (detailElements.elementAtOrNull(2))?.innerHtml.trim(); |     String? appId = (detailElements.elementAtOrNull(2))?.innerHtml.trim(); | ||||||
|     String? dateStr = (detailElements.elementAtOrNull(29))?.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([ |     return Map.fromEntries([ | ||||||
|       MapEntry('version', version), |       MapEntry('version', version), | ||||||
|       MapEntry('apkUrl', apkUrl), |  | ||||||
|       MapEntry('appId', appId), |       MapEntry('appId', appId), | ||||||
|       MapEntry('name', name), |       MapEntry('name', name), | ||||||
|       MapEntry('author', author), |       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 = |     var appDetails = | ||||||
|         await getAppDetailsFromPage(standardUrl, additionalSettings); |         await getAppDetailsFromPage(standardUrl, additionalSettings); | ||||||
|     var version = appDetails['version']; |     var version = appDetails['version']; | ||||||
|     var apkUrl = appDetails['apkUrl']; |  | ||||||
|     var appId = appDetails['appId']; |     var appId = appDetails['appId']; | ||||||
|  |     var fileId = appDetails['fileId']; | ||||||
|  |     var extension = appDetails['extension']; | ||||||
|     if (version == null) { |     if (version == null) { | ||||||
|       throw NoVersionError(); |       throw NoVersionError(); | ||||||
|     } |     } | ||||||
|     if (apkUrl == null) { |     if (fileId == null) { | ||||||
|       throw NoAPKError(); |       throw NoAPKError(); | ||||||
|     } |     } | ||||||
|  |     var apkUrl = '$standardUrl/$fileId-x'; | ||||||
|     if (appId == null) { |     if (appId == null) { | ||||||
|       throw NoReleasesError(); |       throw NoReleasesError(); | ||||||
|     } |     } | ||||||
| @@ -82,8 +94,8 @@ class Uptodown extends AppSource { | |||||||
|     if (dateStr != null) { |     if (dateStr != null) { | ||||||
|       relDate = parseDateTimeMMMddCommayyyy(dateStr); |       relDate = parseDateTimeMMMddCommayyyy(dateStr); | ||||||
|     } |     } | ||||||
|     return APKDetails( |     return APKDetails(version, [MapEntry('$appId.$extension', apkUrl)], | ||||||
|         version, getApkUrlsFromUrls([apkUrl]), AppNames(author, appName), |         AppNames(author, appName), | ||||||
|         releaseDate: relDate); |         releaseDate: relDate); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -96,7 +108,7 @@ class Uptodown extends AppSource { | |||||||
|     } |     } | ||||||
|     var html = parse(res.body); |     var html = parse(res.body); | ||||||
|     var finalUrlKey = |     var finalUrlKey = | ||||||
|         html.querySelector('.post-download')?.attributes['data-url']; |         html.querySelector('#detail-download-button')?.attributes['data-url']; | ||||||
|     if (finalUrlKey == null) { |     if (finalUrlKey == null) { | ||||||
|       throw NoAPKError(); |       throw NoAPKError(); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ class VLC extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     return 'https://${hosts[0]}'; |     return 'https://${hosts[0]}'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ class WhatsApp extends AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|     return 'https://${hosts[0]}'; |     return 'https://${hosts[0]}'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import 'package:easy_localization/easy_localization.dart'; | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:obtainium/components/generated_form_modal.dart'; | import 'package:obtainium/components/generated_form_modal.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  | import 'package:flutter_typeahead/flutter_typeahead.dart'; | ||||||
|  |  | ||||||
| abstract class GeneratedFormItem { | abstract class GeneratedFormItem { | ||||||
|   late String key; |   late String key; | ||||||
| @@ -28,6 +29,7 @@ class GeneratedFormTextField extends GeneratedFormItem { | |||||||
|   late String? hint; |   late String? hint; | ||||||
|   late bool password; |   late bool password; | ||||||
|   late TextInputType? textInputType; |   late TextInputType? textInputType; | ||||||
|  |   late List<String>? autoCompleteOptions; | ||||||
|  |  | ||||||
|   GeneratedFormTextField(super.key, |   GeneratedFormTextField(super.key, | ||||||
|       {super.label, |       {super.label, | ||||||
| @@ -39,7 +41,8 @@ class GeneratedFormTextField extends GeneratedFormItem { | |||||||
|       this.max = 1, |       this.max = 1, | ||||||
|       this.hint, |       this.hint, | ||||||
|       this.password = false, |       this.password = false, | ||||||
|       this.textInputType}); |       this.textInputType, | ||||||
|  |       this.autoCompleteOptions}); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String ensureType(val) { |   String ensureType(val) { | ||||||
| @@ -245,8 +248,8 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|   void someValueChanged({bool isBuilding = false, bool forceInvalid = false}) { |   void someValueChanged({bool isBuilding = false, bool forceInvalid = false}) { | ||||||
|     Map<String, dynamic> returnValues = values; |     Map<String, dynamic> returnValues = values; | ||||||
|     var valid = true; |     var valid = true; | ||||||
|     for (int r = 0; r < widget.items.length; r++) { |     for (int r = 0; r < formInputs.length; r++) { | ||||||
|       for (int i = 0; i < widget.items[r].length; i++) { |       for (int i = 0; i < formInputs[r].length; i++) { | ||||||
|         if (formInputs[r][i] is TextFormField) { |         if (formInputs[r][i] is TextFormField) { | ||||||
|           valid = valid && validateTextField(formInputs[r][i] as TextFormField); |           valid = valid && validateTextField(formInputs[r][i] as TextFormField); | ||||||
|         } |         } | ||||||
| @@ -274,38 +277,62 @@ class _GeneratedFormState extends State<GeneratedForm> { | |||||||
|         var formItem = e.value; |         var formItem = e.value; | ||||||
|         if (formItem is GeneratedFormTextField) { |         if (formItem is GeneratedFormTextField) { | ||||||
|           final formFieldKey = GlobalKey<FormFieldState>(); |           final formFieldKey = GlobalKey<FormFieldState>(); | ||||||
|           return TextFormField( |           var ctrl = TextEditingController(text: values[formItem.key]); | ||||||
|             keyboardType: formItem.textInputType, |           return TypeAheadField<String>( | ||||||
|             obscureText: formItem.password, |             controller: ctrl, | ||||||
|             autocorrect: !formItem.password, |             builder: (context, controller, focusNode) { | ||||||
|             enableSuggestions: !formItem.password, |               return TextFormField( | ||||||
|             key: formFieldKey, |                 controller: ctrl, | ||||||
|             initialValue: values[formItem.key], |                 focusNode: focusNode, | ||||||
|             autovalidateMode: AutovalidateMode.onUserInteraction, |                 keyboardType: formItem.textInputType, | ||||||
|             onChanged: (value) { |                 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(() { |               setState(() { | ||||||
|                 values[formItem.key] = value; |                 values[formItem.key] = value; | ||||||
|                 someValueChanged(); |                 someValueChanged(); | ||||||
|               }); |               }); | ||||||
|             }, |             }, | ||||||
|             decoration: InputDecoration( |             suggestionsCallback: (search) { | ||||||
|                 helperText: formItem.label + (formItem.required ? ' *' : ''), |               return formItem.autoCompleteOptions | ||||||
|                 hintText: formItem.hint), |                   ?.where((t) => t.toLowerCase().contains(search.toLowerCase())) | ||||||
|             minLines: formItem.max <= 1 ? null : formItem.max, |                   .toList(); | ||||||
|             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; |  | ||||||
|             }, |             }, | ||||||
|  |             hideOnEmpty: true, | ||||||
|           ); |           ); | ||||||
|         } else if (formItem is GeneratedFormDropdown) { |         } else if (formItem is GeneratedFormDropdown) { | ||||||
|           if (formItem.opts!.isEmpty) { |           if (formItem.opts!.isEmpty) { | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import 'package:flutter/services.dart'; | |||||||
| import 'package:obtainium/pages/home.dart'; | import 'package:obtainium/pages/home.dart'; | ||||||
| import 'package:obtainium/providers/apps_provider.dart'; | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
| import 'package:obtainium/providers/logs_provider.dart'; | import 'package:obtainium/providers/logs_provider.dart'; | ||||||
|  | import 'package:obtainium/providers/native_provider.dart'; | ||||||
| import 'package:obtainium/providers/notifications_provider.dart'; | import 'package:obtainium/providers/notifications_provider.dart'; | ||||||
| import 'package:obtainium/providers/settings_provider.dart'; | import 'package:obtainium/providers/settings_provider.dart'; | ||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| @@ -22,6 +23,7 @@ import 'package:easy_localization/src/localization.dart'; | |||||||
| List<MapEntry<Locale, String>> supportedLocales = const [ | List<MapEntry<Locale, String>> supportedLocales = const [ | ||||||
|   MapEntry(Locale('en'), 'English'), |   MapEntry(Locale('en'), 'English'), | ||||||
|   MapEntry(Locale('zh'), '简体中文'), |   MapEntry(Locale('zh'), '简体中文'), | ||||||
|  |   MapEntry(Locale('zh_Hant_TW'), '臺灣話'), | ||||||
|   MapEntry(Locale('it'), 'Italiano'), |   MapEntry(Locale('it'), 'Italiano'), | ||||||
|   MapEntry(Locale('ja'), '日本語'), |   MapEntry(Locale('ja'), '日本語'), | ||||||
|   MapEntry(Locale('hu'), 'Magyar'), |   MapEntry(Locale('hu'), 'Magyar'), | ||||||
| @@ -39,6 +41,9 @@ List<MapEntry<Locale, String>> supportedLocales = const [ | |||||||
|   MapEntry(Locale('vi'), 'Tiếng Việt'), |   MapEntry(Locale('vi'), 'Tiếng Việt'), | ||||||
|   MapEntry(Locale('tr'), 'Türkçe'), |   MapEntry(Locale('tr'), 'Türkçe'), | ||||||
|   MapEntry(Locale('uk'), 'Українська'), |   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 fallbackLocale = Locale('en'); | ||||||
| const localeDir = 'assets/translations'; | const localeDir = 'assets/translations'; | ||||||
| @@ -118,8 +123,6 @@ void main() async { | |||||||
|   BackgroundFetch.registerHeadlessTask(backgroundFetchHeadlessTask); |   BackgroundFetch.registerHeadlessTask(backgroundFetchHeadlessTask); | ||||||
| } | } | ||||||
|  |  | ||||||
| var defaultThemeColour = Colors.deepPurple; |  | ||||||
|  |  | ||||||
| class Obtainium extends StatefulWidget { | class Obtainium extends StatefulWidget { | ||||||
|   const Obtainium({super.key}); |   const Obtainium({super.key}); | ||||||
|  |  | ||||||
| @@ -215,22 +218,25 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|       ColorScheme darkColorScheme; |       ColorScheme darkColorScheme; | ||||||
|       if (lightDynamic != null && |       if (lightDynamic != null && | ||||||
|           darkDynamic != null && |           darkDynamic != null && | ||||||
|           settingsProvider.colour == ColourSettings.materialYou) { |           settingsProvider.useMaterialYou) { | ||||||
|         lightColorScheme = lightDynamic.harmonized(); |         lightColorScheme = lightDynamic.harmonized(); | ||||||
|         darkColorScheme = darkDynamic.harmonized(); |         darkColorScheme = darkDynamic.harmonized(); | ||||||
|       } else { |       } else { | ||||||
|         lightColorScheme = ColorScheme.fromSeed(seedColor: defaultThemeColour); |         lightColorScheme = | ||||||
|  |             ColorScheme.fromSeed(seedColor: settingsProvider.themeColor); | ||||||
|         darkColorScheme = ColorScheme.fromSeed( |         darkColorScheme = ColorScheme.fromSeed( | ||||||
|             seedColor: defaultThemeColour, brightness: Brightness.dark); |             seedColor: settingsProvider.themeColor, | ||||||
|  |             brightness: Brightness.dark); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       // set the background and surface colors to pure black in the amoled theme |       // set the background and surface colors to pure black in the amoled theme | ||||||
|       if (settingsProvider.useBlackTheme) { |       if (settingsProvider.useBlackTheme) { | ||||||
|         darkColorScheme = darkColorScheme |         darkColorScheme = | ||||||
|             .copyWith(background: Colors.black, surface: Colors.black) |             darkColorScheme.copyWith(surface: Colors.black).harmonized(); | ||||||
|             .harmonized(); |  | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       if (settingsProvider.useSystemFont) NativeFeatures.loadSystemFont(); | ||||||
|  |  | ||||||
|       return MaterialApp( |       return MaterialApp( | ||||||
|           title: 'Obtainium', |           title: 'Obtainium', | ||||||
|           localizationsDelegates: context.localizationDelegates, |           localizationsDelegates: context.localizationDelegates, | ||||||
| @@ -242,15 +248,17 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|               colorScheme: settingsProvider.theme == ThemeSettings.dark |               colorScheme: settingsProvider.theme == ThemeSettings.dark | ||||||
|                   ? darkColorScheme |                   ? darkColorScheme | ||||||
|                   : lightColorScheme, |                   : lightColorScheme, | ||||||
|               fontFamily: |               fontFamily: settingsProvider.useSystemFont | ||||||
|                   settingsProvider.useSystemFont ? 'SystemFont' : 'Metropolis'), |                   ? 'SystemFont' | ||||||
|  |                   : 'Wix-Madefor-Display'), | ||||||
|           darkTheme: ThemeData( |           darkTheme: ThemeData( | ||||||
|               useMaterial3: true, |               useMaterial3: true, | ||||||
|               colorScheme: settingsProvider.theme == ThemeSettings.light |               colorScheme: settingsProvider.theme == ThemeSettings.light | ||||||
|                   ? lightColorScheme |                   ? lightColorScheme | ||||||
|                   : darkColorScheme, |                   : darkColorScheme, | ||||||
|               fontFamily: |               fontFamily: settingsProvider.useSystemFont | ||||||
|                   settingsProvider.useSystemFont ? 'SystemFont' : 'Metropolis'), |                   ? 'SystemFont' | ||||||
|  |                   : 'Wix-Madefor-Display'), | ||||||
|           home: Shortcuts(shortcuts: <LogicalKeySet, Intent>{ |           home: Shortcuts(shortcuts: <LogicalKeySet, Intent>{ | ||||||
|             LogicalKeySet(LogicalKeyboardKey.select): const ActivateIntent(), |             LogicalKeySet(LogicalKeyboardKey.select): const ActivateIntent(), | ||||||
|           }, child: const HomePage())); |           }, child: const HomePage())); | ||||||
|   | |||||||
| @@ -51,10 +51,13 @@ class AddAppPageState extends State<AddAppPage> { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   changeUserInput(String input, bool valid, bool isBuilding, |   changeUserInput(String input, bool valid, bool isBuilding, | ||||||
|       {bool updateUrlInput = false}) { |       {bool updateUrlInput = false, String? overrideSource}) { | ||||||
|     userInput = input; |     userInput = input; | ||||||
|     if (!isBuilding) { |     if (!isBuilding) { | ||||||
|       setState(() { |       setState(() { | ||||||
|  |         if (overrideSource != null) { | ||||||
|  |           pickedSourceOverride = overrideSource; | ||||||
|  |         } | ||||||
|         if (updateUrlInput) { |         if (updateUrlInput) { | ||||||
|           urlInputKey++; |           urlInputKey++; | ||||||
|         } |         } | ||||||
| @@ -68,6 +71,7 @@ class AddAppPageState extends State<AddAppPage> { | |||||||
|         if (pickedSource.runtimeType != source.runtimeType || |         if (pickedSource.runtimeType != source.runtimeType || | ||||||
|             (prevHost != null && prevHost != source?.hosts[0])) { |             (prevHost != null && prevHost != source?.hosts[0])) { | ||||||
|           pickedSource = source; |           pickedSource = source; | ||||||
|  |           pickedSource?.runOnAddAppInputChange(userInput); | ||||||
|           additionalSettings = source != null |           additionalSettings = source != null | ||||||
|               ? getDefaultValuesFromFormItems( |               ? getDefaultValuesFromFormItems( | ||||||
|                   source.combinedAppSpecificSettingFormItems) |                   source.combinedAppSpecificSettingFormItems) | ||||||
| @@ -259,9 +263,7 @@ class AddAppPageState extends State<AddAppPage> { | |||||||
|         searching = true; |         searching = true; | ||||||
|       }); |       }); | ||||||
|       var sourceStrings = <String, List<String>>{}; |       var sourceStrings = <String, List<String>>{}; | ||||||
|       sourceProvider.sources |       sourceProvider.sources.where((e) => e.canSearch).forEach((s) { | ||||||
|           .where((e) => e.canSearch && !e.excludeFromMassSearch) |  | ||||||
|           .forEach((s) { |  | ||||||
|         sourceStrings[s.name] = [s.name]; |         sourceStrings[s.name] = [s.name]; | ||||||
|       }); |       }); | ||||||
|       try { |       try { | ||||||
| @@ -282,32 +284,78 @@ class AddAppPageState extends State<AddAppPage> { | |||||||
|           settingsProvider.searchDeselected = sourceStrings.keys |           settingsProvider.searchDeselected = sourceStrings.keys | ||||||
|               .where((s) => !searchSources.contains(s)) |               .where((s) => !searchSources.contains(s)) | ||||||
|               .toList(); |               .toList(); | ||||||
|           var results = await Future.wait(sourceProvider.sources |           List<MapEntry<String, Map<String, List<String>>>?> results = | ||||||
|               .where((e) => searchSources.contains(e.name)) |               (await Future.wait(sourceProvider.sources | ||||||
|               .map((e) async { |                       .where((e) => searchSources.contains(e.name)) | ||||||
|  |                       .map((e) async { | ||||||
|             try { |             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) { |             } catch (err) { | ||||||
|               if (err is! CredsNeededError) { |               if (err is! CredsNeededError) { | ||||||
|                 rethrow; |                 rethrow; | ||||||
|               } else { |               } else { | ||||||
|                 err.unexpected = true; |                 err.unexpected = true; | ||||||
|                 showError(err, context); |                 showError(err, context); | ||||||
|                 return <String, List<String>>{}; |                 return null; | ||||||
|               } |               } | ||||||
|             } |             } | ||||||
|           })); |           }))) | ||||||
|  |                   .where((a) => a != null) | ||||||
|  |                   .toList(); | ||||||
|  |  | ||||||
|           // Interleave results instead of simple reduce |           // Interleave results instead of simple reduce | ||||||
|           Map<String, List<String>> res = {}; |           Map<String, MapEntry<String, List<String>>> res = {}; | ||||||
|           var si = 0; |           var si = 0; | ||||||
|           var done = false; |           var done = false; | ||||||
|           while (!done) { |           while (!done) { | ||||||
|             done = true; |             done = true; | ||||||
|             for (var r in results) { |             for (var r in results) { | ||||||
|               if (r.length > si) { |               var sourceName = r!.key; | ||||||
|  |               if (r.value.length > si) { | ||||||
|                 done = false; |                 done = false; | ||||||
|                 res.addEntries([r.entries.elementAt(si)]); |                 var singleRes = r.value.entries.elementAt(si); | ||||||
|  |                 res[singleRes.key] = MapEntry(sourceName, singleRes.value); | ||||||
|               } |               } | ||||||
|             } |             } | ||||||
|             si++; |             si++; | ||||||
| @@ -322,13 +370,15 @@ class AddAppPageState extends State<AddAppPage> { | |||||||
|                   context: context, |                   context: context, | ||||||
|                   builder: (BuildContext ctx) { |                   builder: (BuildContext ctx) { | ||||||
|                     return SelectionModal( |                     return SelectionModal( | ||||||
|                       entries: res, |                       entries: res.map((k, v) => MapEntry(k, v.value)), | ||||||
|                       selectedByDefault: false, |                       selectedByDefault: false, | ||||||
|                       onlyOneSelectionAllowed: true, |                       onlyOneSelectionAllowed: true, | ||||||
|                     ); |                     ); | ||||||
|                   }); |                   }); | ||||||
|           if (selectedUrls != null && selectedUrls.isNotEmpty) { |           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) { |       } catch (e) { | ||||||
| @@ -349,7 +399,7 @@ class AddAppPageState extends State<AddAppPage> { | |||||||
|                   [ |                   [ | ||||||
|                     GeneratedFormDropdown( |                     GeneratedFormDropdown( | ||||||
|                         'overrideSource', |                         'overrideSource', | ||||||
|                         defaultValue: '', |                         defaultValue: pickedSourceOverride ?? '', | ||||||
|                         [ |                         [ | ||||||
|                           MapEntry('', tr('none')), |                           MapEntry('', tr('none')), | ||||||
|                           ...sourceProvider.sources.map( |                           ...sourceProvider.sources.map( | ||||||
|   | |||||||
| @@ -133,7 +133,7 @@ class _AppPageState extends State<AppPage> { | |||||||
|                         child: Text( |                         child: Text( | ||||||
|                           app?.app.releaseDate == null |                           app?.app.releaseDate == null | ||||||
|                               ? tr('changes') |                               ? tr('changes') | ||||||
|                               : app!.app.releaseDate.toString(), |                               : app!.app.releaseDate!.toLocal().toString(), | ||||||
|                           textAlign: TextAlign.center, |                           textAlign: TextAlign.center, | ||||||
|                           style: |                           style: | ||||||
|                               Theme.of(context).textTheme.labelSmall!.copyWith( |                               Theme.of(context).textTheme.labelSmall!.copyWith( | ||||||
| @@ -161,26 +161,46 @@ class _AppPageState extends State<AppPage> { | |||||||
|           if (app?.app.apkUrls.isNotEmpty == true || |           if (app?.app.apkUrls.isNotEmpty == true || | ||||||
|               app?.app.otherAssetUrls.isNotEmpty == true) |               app?.app.otherAssetUrls.isNotEmpty == true) | ||||||
|             GestureDetector( |             GestureDetector( | ||||||
|               onTap: app?.app == null || updating |                 onTap: app?.app == null || updating | ||||||
|                   ? null |                     ? null | ||||||
|                   : () async { |                     : () async { | ||||||
|                       try { |                         try { | ||||||
|                         await appsProvider |                           await appsProvider | ||||||
|                             .downloadAppAssets([app!.app.id], context); |                               .downloadAppAssets([app!.app.id], context); | ||||||
|                       } catch (e) { |                         } catch (e) { | ||||||
|                         showError(e, context); |                           showError(e, context); | ||||||
|                       } |                         } | ||||||
|                     }, |                       }, | ||||||
|               child: Text( |                 child: Row( | ||||||
|                 tr('downloadX', args: [tr('releaseAsset').toLowerCase()]), |                   mainAxisAlignment: MainAxisAlignment.center, | ||||||
|                 textAlign: TextAlign.center, |                   children: [ | ||||||
|                 style: Theme.of(context).textTheme.labelSmall!.copyWith( |                     Container( | ||||||
|                       decoration: |                         decoration: BoxDecoration( | ||||||
|                           changeLogFn != null ? TextDecoration.underline : null, |                             borderRadius: BorderRadius.circular(12), | ||||||
|                       fontStyle: changeLogFn != null ? FontStyle.italic : null, |                             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( |           const SizedBox( | ||||||
|             height: 48, |             height: 48, | ||||||
|           ), |           ), | ||||||
| @@ -227,18 +247,27 @@ class _AppPageState extends State<AppPage> { | |||||||
|           crossAxisAlignment: CrossAxisAlignment.stretch, |           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|           children: [ |           children: [ | ||||||
|             const SizedBox(height: 20), |             const SizedBox(height: 20), | ||||||
|             app?.icon != null |             FutureBuilder( | ||||||
|                 ? Row(mainAxisAlignment: MainAxisAlignment.center, children: [ |                 future: | ||||||
|                     GestureDetector( |                     appsProvider.updateAppIcon(app?.app.id, ignoreCache: true), | ||||||
|                       child: Image.memory( |                 builder: (ctx, val) { | ||||||
|                         app!.icon!, |                   return app?.icon != null | ||||||
|                         height: 150, |                       ? Row( | ||||||
|                         gaplessPlayback: true, |                           mainAxisAlignment: MainAxisAlignment.center, | ||||||
|                       ), |                           children: [ | ||||||
|                       onTap: () => pm.openApp(app.app.id), |                               GestureDetector( | ||||||
|                     ) |                                 onTap: app == null | ||||||
|                   ]) |                                     ? null | ||||||
|                 : Container(), |                                     : () => pm.openApp(app.app.id), | ||||||
|  |                                 child: Image.memory( | ||||||
|  |                                   app!.icon!, | ||||||
|  |                                   height: 150, | ||||||
|  |                                   gaplessPlayback: true, | ||||||
|  |                                 ), | ||||||
|  |                               ) | ||||||
|  |                             ]) | ||||||
|  |                       : Container(); | ||||||
|  |                 }), | ||||||
|             const SizedBox( |             const SizedBox( | ||||||
|               height: 25, |               height: 25, | ||||||
|             ), |             ), | ||||||
| @@ -287,7 +316,7 @@ class _AppPageState extends State<AppPage> { | |||||||
|         ? WebViewWidget( |         ? WebViewWidget( | ||||||
|             controller: WebViewController() |             controller: WebViewController() | ||||||
|               ..setJavaScriptMode(JavaScriptMode.unrestricted) |               ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||||
|               ..setBackgroundColor(Theme.of(context).colorScheme.background) |               ..setBackgroundColor(Theme.of(context).colorScheme.surface) | ||||||
|               ..setJavaScriptMode(JavaScriptMode.unrestricted) |               ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||||
|               ..setNavigationDelegate( |               ..setNavigationDelegate( | ||||||
|                 NavigationDelegate( |                 NavigationDelegate( | ||||||
|   | |||||||
| @@ -143,11 +143,14 @@ class AppsPageState extends State<AppsPage> { | |||||||
|   final GlobalKey<RefreshIndicatorState> _refreshIndicatorKey = |   final GlobalKey<RefreshIndicatorState> _refreshIndicatorKey = | ||||||
|       GlobalKey<RefreshIndicatorState>(); |       GlobalKey<RefreshIndicatorState>(); | ||||||
|  |  | ||||||
|  |   late final ScrollController scrollController = ScrollController(); | ||||||
|  |  | ||||||
|  |   var sourceProvider = SourceProvider(); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     var appsProvider = context.watch<AppsProvider>(); |     var appsProvider = context.watch<AppsProvider>(); | ||||||
|     var settingsProvider = context.watch<SettingsProvider>(); |     var settingsProvider = context.watch<SettingsProvider>(); | ||||||
|     var sourceProvider = SourceProvider(); |  | ||||||
|     var listedApps = appsProvider.getAppValues().toList(); |     var listedApps = appsProvider.getAppValues().toList(); | ||||||
|  |  | ||||||
|     refresh() { |     refresh() { | ||||||
| @@ -354,7 +357,11 @@ class AppsPageState extends State<AppsPage> { | |||||||
|           SliverFillRemaining( |           SliverFillRemaining( | ||||||
|               child: Center( |               child: Center( | ||||||
|                   child: Text( |                   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, |             style: Theme.of(context).textTheme.headlineMedium, | ||||||
|             textAlign: TextAlign.center, |             textAlign: TextAlign.center, | ||||||
|           ))), |           ))), | ||||||
| @@ -402,29 +409,38 @@ class AppsPageState extends State<AppsPage> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     getAppIcon(int appIndex) { |     getAppIcon(int appIndex) { | ||||||
|       return listedApps[appIndex].icon != null |       return FutureBuilder( | ||||||
|           ? Image.memory( |           future: appsProvider.updateAppIcon(listedApps[appIndex].app.id), | ||||||
|               listedApps[appIndex].icon!, |           builder: (ctx, val) { | ||||||
|               gaplessPlayback: true, |             return listedApps[appIndex].icon != null | ||||||
|             ) |                 ? Image.memory( | ||||||
|           : Row( |                     listedApps[appIndex].icon!, | ||||||
|               mainAxisSize: MainAxisSize.min, |                     gaplessPlayback: true, | ||||||
|               mainAxisAlignment: MainAxisAlignment.center, |                     opacity: AlwaysStoppedAnimation( | ||||||
|               children: [ |                         listedApps[appIndex].installedInfo == null ? 0.6 : 1), | ||||||
|                   Transform( |                   ) | ||||||
|                       alignment: Alignment.center, |                 : Row( | ||||||
|                       transform: Matrix4.rotationZ(0.31), |                     mainAxisSize: MainAxisSize.min, | ||||||
|                       child: Padding( |                     mainAxisAlignment: MainAxisAlignment.center, | ||||||
|                         padding: const EdgeInsets.all(15), |                     children: [ | ||||||
|                         child: Image( |                         Transform( | ||||||
|                           image: const AssetImage( |                             alignment: Alignment.center, | ||||||
|                               'assets/graphics/icon_small.png'), |                             transform: Matrix4.rotationZ(0.31), | ||||||
|                           color: Colors.white.withOpacity(0.3), |                             child: Padding( | ||||||
|                           colorBlendMode: BlendMode.modulate, |                               padding: const EdgeInsets.all(15), | ||||||
|                           gaplessPlayback: true, |                               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) { |     getVersionText(int appIndex) { | ||||||
| @@ -437,7 +453,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|               ? tr('changes') |               ? tr('changes') | ||||||
|               : '' |               : '' | ||||||
|           : DateFormat('yyyy-MM-dd') |           : DateFormat('yyyy-MM-dd') | ||||||
|               .format(listedApps[appIndex].app.releaseDate!); |               .format(listedApps[appIndex].app.releaseDate!.toLocal()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     getSingleAppHorizTile(int index) { |     getSingleAppHorizTile(int index) { | ||||||
| @@ -452,7 +468,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|           hasUpdate ? getUpdateButton(index) : const SizedBox.shrink(), |           hasUpdate ? getUpdateButton(index) : const SizedBox.shrink(), | ||||||
|           hasUpdate |           hasUpdate | ||||||
|               ? const SizedBox( |               ? const SizedBox( | ||||||
|                   width: 10, |                   width: 5, | ||||||
|                 ) |                 ) | ||||||
|               : const SizedBox.shrink(), |               : const SizedBox.shrink(), | ||||||
|           GestureDetector( |           GestureDetector( | ||||||
| @@ -503,7 +519,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|       ); |       ); | ||||||
|  |  | ||||||
|       var transparent = |       var transparent = | ||||||
|           Theme.of(context).colorScheme.background.withAlpha(0).value; |           Theme.of(context).colorScheme.surface.withAlpha(0).value; | ||||||
|       List<double> stops = [ |       List<double> stops = [ | ||||||
|         ...listedApps[index].app.categories.asMap().entries.map( |         ...listedApps[index].app.categories.asMap().entries.map( | ||||||
|             (e) => ((e.key / (listedApps[index].app.categories.length - 1)))), |             (e) => ((e.key / (listedApps[index].app.categories.length - 1)))), | ||||||
| @@ -882,11 +898,10 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                           onPressed: selectedAppIds.isEmpty |                           onPressed: selectedAppIds.isEmpty | ||||||
|                               ? null |                               ? null | ||||||
|                               : () { |                               : () { | ||||||
|                                   String urls = |                                   String urls = ''; | ||||||
|                                       '<p>${tr('customLinkMessage')}:</p>\n\n<ul>\n'; |  | ||||||
|                                   for (var a in selectedApps) { |                                   for (var a in selectedApps) { | ||||||
|                                     urls += |                                     urls += | ||||||
|                                         '    <li><a href="obtainium://app/${Uri.encodeComponent(jsonEncode({ |                                         'https://apps.obtainium.imranr.dev/redirect?r=obtainium://app/${Uri.encodeComponent(jsonEncode({ | ||||||
|                                           'id': a.id, |                                           'id': a.id, | ||||||
|                                           'url': a.url, |                                           'url': a.url, | ||||||
|                                           'author': a.author, |                                           'author': a.author, | ||||||
| @@ -894,17 +909,37 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                                           'preferredApkIndex': |                                           'preferredApkIndex': | ||||||
|                                               a.preferredApkIndex, |                                               a.preferredApkIndex, | ||||||
|                                           'additionalSettings': |                                           'additionalSettings': | ||||||
|                                               jsonEncode(a.additionalSettings) |                                               jsonEncode(a.additionalSettings), | ||||||
|                                         }))}">${a.name}</a></li>\n'; |                                           'overrideSource': a.overrideSource | ||||||
|  |                                         }))}\n\n'; | ||||||
|                                   } |                                   } | ||||||
|                                   urls += |  | ||||||
|                                       '</ul>\n\n<p><a href="$obtainiumUrl">${tr('about')}</a></p>'; |  | ||||||
|                                   Share.share(urls, |                                   Share.share(urls, | ||||||
|                                       subject: |                                       subject: | ||||||
|                                           'Obtainium - ${tr('appsString')}'); |                                           'Obtainium - ${tr('appsString')}'); | ||||||
|                                 }, |                                 }, | ||||||
|                           child: Text(tr('shareAppConfigLinks'))), |                           child: Text(tr('shareAppConfigLinks'))), | ||||||
|                       const Divider(), |                       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( |                       TextButton( | ||||||
|                           onPressed: () { |                           onPressed: () { | ||||||
|                             appsProvider |                             appsProvider | ||||||
| @@ -1044,7 +1079,9 @@ class AppsPageState extends State<AppsPage> { | |||||||
|           IconButton( |           IconButton( | ||||||
|               color: Theme.of(context).colorScheme.primary, |               color: Theme.of(context).colorScheme.primary, | ||||||
|               style: const ButtonStyle(visualDensity: VisualDensity.compact), |               style: const ButtonStyle(visualDensity: VisualDensity.compact), | ||||||
|               tooltip: '${tr('filter')}${isFilterOff ? '' : ' *'}', |               tooltip: isFilterOff | ||||||
|  |                   ? tr('filterApps') | ||||||
|  |                   : '${tr('filter')} - ${tr('remove')}', | ||||||
|               onPressed: isFilterOff |               onPressed: isFilterOff | ||||||
|                   ? showFilterDialog |                   ? showFilterDialog | ||||||
|                   : () { |                   : () { | ||||||
| @@ -1053,8 +1090,8 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                       }); |                       }); | ||||||
|                     }, |                     }, | ||||||
|               icon: Icon(isFilterOff |               icon: Icon(isFilterOff | ||||||
|                   ? Icons.filter_list_rounded |                   ? Icons.search_rounded | ||||||
|                   : Icons.filter_list_off_rounded)), |                   : Icons.search_off_rounded)), | ||||||
|           const SizedBox( |           const SizedBox( | ||||||
|             width: 10, |             width: 10, | ||||||
|           ), |           ), | ||||||
| @@ -1089,11 +1126,17 @@ class AppsPageState extends State<AppsPage> { | |||||||
|       body: RefreshIndicator( |       body: RefreshIndicator( | ||||||
|           key: _refreshIndicatorKey, |           key: _refreshIndicatorKey, | ||||||
|           onRefresh: refresh, |           onRefresh: refresh, | ||||||
|           child: CustomScrollView(slivers: <Widget>[ |           child: Scrollbar( | ||||||
|             CustomAppBar(title: tr('appsString')), |               interactive: true, | ||||||
|             ...getLoadingWidgets(), |               controller: scrollController, | ||||||
|             getDisplayedList() |               child: CustomScrollView( | ||||||
|           ])), |                   physics: const AlwaysScrollableScrollPhysics(), | ||||||
|  |                   controller: scrollController, | ||||||
|  |                   slivers: <Widget>[ | ||||||
|  |                     CustomAppBar(title: tr('appsString')), | ||||||
|  |                     ...getLoadingWidgets(), | ||||||
|  |                     getDisplayedList() | ||||||
|  |                   ]))), | ||||||
|       persistentFooterButtons: appsProvider.apps.isEmpty |       persistentFooterButtons: appsProvider.apps.isEmpty | ||||||
|           ? null |           ? null | ||||||
|           : [ |           : [ | ||||||
|   | |||||||
| @@ -13,6 +13,7 @@ import 'package:obtainium/pages/import_export.dart'; | |||||||
| import 'package:obtainium/pages/settings.dart'; | import 'package:obtainium/pages/settings.dart'; | ||||||
| import 'package:obtainium/providers/apps_provider.dart'; | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
| import 'package:obtainium/providers/settings_provider.dart'; | import 'package:obtainium/providers/settings_provider.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| import 'package:provider/provider.dart'; | import 'package:provider/provider.dart'; | ||||||
|  |  | ||||||
| class HomePage extends StatefulWidget { | class HomePage extends StatefulWidget { | ||||||
| @@ -102,13 +103,22 @@ class _HomePageState extends State<HomePage> { | |||||||
|                   }) != |                   }) != | ||||||
|               null) { |               null) { | ||||||
|             // ignore: use_build_context_synchronously |             // ignore: use_build_context_synchronously | ||||||
|             var result = await context.read<AppsProvider>().import( |             var appsProvider = context.read<AppsProvider>(); | ||||||
|                 action == 'app' |             var result = await appsProvider.import(action == 'app' | ||||||
|                     ? '{ "apps": [$dataStr] }' |                 ? '{ "apps": [$dataStr] }' | ||||||
|                     : '{ "apps": $dataStr }'); |                 : '{ "apps": $dataStr }'); | ||||||
|             // ignore: use_build_context_synchronously |             // ignore: use_build_context_synchronously | ||||||
|             showMessage( |             showMessage( | ||||||
|                 tr('importedX', args: [plural('apps', result.key)]), context); |                 tr('importedX', args: [plural('apps', result.key.length)]), | ||||||
|  |                 context); | ||||||
|  |             await appsProvider | ||||||
|  |                 .checkUpdates(specificIds: result.key.map((e) => e.id).toList()) | ||||||
|  |                 .catchError((e) { | ||||||
|  |               if (e is Map && e['errors'] is MultiAppMultiError) { | ||||||
|  |                 showError(e['errors'].toString(), context); | ||||||
|  |               } | ||||||
|  |               return <App>[]; | ||||||
|  |             }); | ||||||
|           } |           } | ||||||
|         } else { |         } else { | ||||||
|           throw ObtainiumError(tr('unknown')); |           throw ObtainiumError(tr('unknown')); | ||||||
| @@ -119,7 +129,7 @@ class _HomePageState extends State<HomePage> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Check initial link if app was in cold state (terminated) |     // Check initial link if app was in cold state (terminated) | ||||||
|     final appLink = await _appLinks.getInitialAppLink(); |     final appLink = await _appLinks.getInitialLink(); | ||||||
|     if (appLink != null) { |     if (appLink != null) { | ||||||
|       await interpretLink(appLink); |       await interpretLink(appLink); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -33,7 +33,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|     var settingsProvider = context.watch<SettingsProvider>(); |     var settingsProvider = context.watch<SettingsProvider>(); | ||||||
|  |  | ||||||
|     var outlineButtonStyle = ButtonStyle( |     var outlineButtonStyle = ButtonStyle( | ||||||
|       shape: MaterialStateProperty.all( |       shape: WidgetStateProperty.all( | ||||||
|         StadiumBorder( |         StadiumBorder( | ||||||
|           side: BorderSide( |           side: BorderSide( | ||||||
|             width: 1, |             width: 1, | ||||||
| @@ -144,7 +144,7 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|             appsProvider.addMissingCategories(settingsProvider); |             appsProvider.addMissingCategories(settingsProvider); | ||||||
|             showMessage( |             showMessage( | ||||||
|                 '${tr('importedX', args: [ |                 '${tr('importedX', args: [ | ||||||
|                       plural('apps', value.key) |                       plural('apps', value.key.length) | ||||||
|                     ])}${value.value ? ' + ${tr('settings')}' : ''}', |                     ])}${value.value ? ' + ${tr('settings')}' : ''}', | ||||||
|                 context); |                 context); | ||||||
|           }); |           }); | ||||||
|   | |||||||
| @@ -1,8 +1,11 @@ | |||||||
| import 'package:device_info_plus/device_info_plus.dart'; | import 'package:device_info_plus/device_info_plus.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:equations/equations.dart'; | ||||||
|  | import 'package:flex_color_picker/flex_color_picker.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:obtainium/components/custom_app_bar.dart'; | import 'package:obtainium/components/custom_app_bar.dart'; | ||||||
| import 'package:obtainium/components/generated_form.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/custom_errors.dart'; | ||||||
| import 'package:obtainium/main.dart'; | import 'package:obtainium/main.dart'; | ||||||
| import 'package:obtainium/providers/apps_provider.dart'; | import 'package:obtainium/providers/apps_provider.dart'; | ||||||
| @@ -12,6 +15,7 @@ import 'package:obtainium/providers/settings_provider.dart'; | |||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
| import 'package:provider/provider.dart'; | import 'package:provider/provider.dart'; | ||||||
| import 'package:share_plus/share_plus.dart'; | import 'package:share_plus/share_plus.dart'; | ||||||
|  | import 'package:shizuku_apk_installer/shizuku_apk_installer.dart'; | ||||||
| import 'package:url_launcher/url_launcher_string.dart'; | import 'package:url_launcher/url_launcher_string.dart'; | ||||||
|  |  | ||||||
| class SettingsPage extends StatefulWidget { | class SettingsPage extends StatefulWidget { | ||||||
| @@ -22,78 +26,184 @@ class SettingsPage extends StatefulWidget { | |||||||
| } | } | ||||||
|  |  | ||||||
| class _SettingsPageState extends State<SettingsPage> { | class _SettingsPageState extends State<SettingsPage> { | ||||||
|  |   List<int> updateIntervalNodes = [ | ||||||
|  |     15, | ||||||
|  |     30, | ||||||
|  |     60, | ||||||
|  |     120, | ||||||
|  |     180, | ||||||
|  |     360, | ||||||
|  |     720, | ||||||
|  |     1440, | ||||||
|  |     4320, | ||||||
|  |     10080, | ||||||
|  |     20160, | ||||||
|  |     43200 | ||||||
|  |   ]; | ||||||
|  |   int updateInterval = 0; | ||||||
|  |   late SplineInterpolation updateIntervalInterpolator; // 🤓 | ||||||
|  |   String updateIntervalLabel = tr('neverManualOnly'); | ||||||
|  |   bool showIntervalLabel = true; | ||||||
|  |   final Map<ColorSwatch<Object>, String> colorsNameMap = | ||||||
|  |       <ColorSwatch<Object>, String>{ | ||||||
|  |     ColorTools.createPrimarySwatch(obtainiumThemeColor): 'Obtainium' | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   void initUpdateIntervalInterpolator() { | ||||||
|  |     List<InterpolationNode> nodes = []; | ||||||
|  |     for (final (index, element) in updateIntervalNodes.indexed) { | ||||||
|  |       nodes.add( | ||||||
|  |           InterpolationNode(x: index.toDouble() + 1, y: element.toDouble())); | ||||||
|  |     } | ||||||
|  |     updateIntervalInterpolator = SplineInterpolation(nodes: nodes); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void processIntervalSliderValue(double val) { | ||||||
|  |     if (val < 0.5) { | ||||||
|  |       updateInterval = 0; | ||||||
|  |       updateIntervalLabel = tr('neverManualOnly'); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     int valInterpolated = 0; | ||||||
|  |     if (val < 1) { | ||||||
|  |       valInterpolated = 15; | ||||||
|  |     } else { | ||||||
|  |       valInterpolated = updateIntervalInterpolator.compute(val).round(); | ||||||
|  |     } | ||||||
|  |     if (valInterpolated < 60) { | ||||||
|  |       updateInterval = valInterpolated; | ||||||
|  |       updateIntervalLabel = plural('minute', valInterpolated); | ||||||
|  |     } else if (valInterpolated < 8 * 60) { | ||||||
|  |       int valRounded = (valInterpolated / 15).floor() * 15; | ||||||
|  |       updateInterval = valRounded; | ||||||
|  |       updateIntervalLabel = plural('hour', valRounded ~/ 60); | ||||||
|  |       int mins = valRounded % 60; | ||||||
|  |       if (mins != 0) updateIntervalLabel += " ${plural('minute', mins)}"; | ||||||
|  |     } else if (valInterpolated < 24 * 60) { | ||||||
|  |       int valRounded = (valInterpolated / 30).floor() * 30; | ||||||
|  |       updateInterval = valRounded; | ||||||
|  |       updateIntervalLabel = plural('hour', valRounded / 60); | ||||||
|  |     } else if (valInterpolated < 7 * 24 * 60) { | ||||||
|  |       int valRounded = (valInterpolated / (12 * 60)).floor() * 12 * 60; | ||||||
|  |       updateInterval = valRounded; | ||||||
|  |       updateIntervalLabel = plural('day', valRounded / (24 * 60)); | ||||||
|  |     } else { | ||||||
|  |       int valRounded = (valInterpolated / (24 * 60)).floor() * 24 * 60; | ||||||
|  |       updateInterval = valRounded; | ||||||
|  |       updateIntervalLabel = plural('day', valRounded ~/ (24 * 60)); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     SettingsProvider settingsProvider = context.watch<SettingsProvider>(); |     SettingsProvider settingsProvider = context.watch<SettingsProvider>(); | ||||||
|     SourceProvider sourceProvider = SourceProvider(); |     SourceProvider sourceProvider = SourceProvider(); | ||||||
|     if (settingsProvider.prefs == null) { |     if (settingsProvider.prefs == null) settingsProvider.initializeSettings(); | ||||||
|       settingsProvider.initializeSettings(); |     initUpdateIntervalInterpolator(); | ||||||
|  |     processIntervalSliderValue(settingsProvider.updateIntervalSliderVal); | ||||||
|  |  | ||||||
|  |     var followSystemThemeExplanation = FutureBuilder( | ||||||
|  |         builder: (ctx, val) { | ||||||
|  |           return ((val.data?.version.sdkInt ?? 30) < 29) | ||||||
|  |               ? Text(tr('followSystemThemeExplanation'), | ||||||
|  |                   style: Theme.of(context).textTheme.labelSmall) | ||||||
|  |               : const SizedBox.shrink(); | ||||||
|  |         }, | ||||||
|  |         future: DeviceInfoPlugin().androidInfo); | ||||||
|  |  | ||||||
|  |     Future<bool> colorPickerDialog() async { | ||||||
|  |       return ColorPicker( | ||||||
|  |         color: settingsProvider.themeColor, | ||||||
|  |         onColorChanged: (Color color) => | ||||||
|  |             setState(() => settingsProvider.themeColor = color), | ||||||
|  |         actionButtons: const ColorPickerActionButtons( | ||||||
|  |           okButton: true, | ||||||
|  |           closeButton: true, | ||||||
|  |           dialogActionButtons: false, | ||||||
|  |         ), | ||||||
|  |         pickersEnabled: const <ColorPickerType, bool>{ | ||||||
|  |           ColorPickerType.both: false, | ||||||
|  |           ColorPickerType.primary: false, | ||||||
|  |           ColorPickerType.accent: false, | ||||||
|  |           ColorPickerType.bw: false, | ||||||
|  |           ColorPickerType.custom: true, | ||||||
|  |           ColorPickerType.wheel: true, | ||||||
|  |         }, | ||||||
|  |         pickerTypeLabels: <ColorPickerType, String>{ | ||||||
|  |           ColorPickerType.custom: tr('standard'), | ||||||
|  |           ColorPickerType.wheel: tr('custom') | ||||||
|  |         }, | ||||||
|  |         title: Text(tr('selectX', args: [tr('colour')]), | ||||||
|  |             style: Theme.of(context).textTheme.titleLarge), | ||||||
|  |         wheelDiameter: 192, | ||||||
|  |         wheelSquareBorderRadius: 32, | ||||||
|  |         width: 48, | ||||||
|  |         height: 48, | ||||||
|  |         borderRadius: 24, | ||||||
|  |         spacing: 8, | ||||||
|  |         runSpacing: 8, | ||||||
|  |         enableShadesSelection: false, | ||||||
|  |         customColorSwatchesAndNames: colorsNameMap, | ||||||
|  |         showMaterialName: true, | ||||||
|  |         showColorName: true, | ||||||
|  |         materialNameTextStyle: Theme.of(context).textTheme.bodySmall, | ||||||
|  |         colorNameTextStyle: Theme.of(context).textTheme.bodySmall, | ||||||
|  |         copyPasteBehavior: | ||||||
|  |             const ColorPickerCopyPasteBehavior(longPressMenu: true), | ||||||
|  |       ).showPickerDialog( | ||||||
|  |         context, | ||||||
|  |         transitionBuilder: (BuildContext context, Animation<double> a1, | ||||||
|  |             Animation<double> a2, Widget widget) { | ||||||
|  |           final double curvedValue = Curves.easeInCubic.transform(a1.value); | ||||||
|  |           return Transform( | ||||||
|  |             alignment: Alignment.center, | ||||||
|  |             transform: Matrix4.diagonal3Values(curvedValue, curvedValue, 1), | ||||||
|  |             child: Opacity(opacity: curvedValue, child: widget), | ||||||
|  |           ); | ||||||
|  |         }, | ||||||
|  |         transitionDuration: const Duration(milliseconds: 250), | ||||||
|  |       ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     var installMethodDropdown = DropdownButtonFormField( |     var colorPicker = ListTile( | ||||||
|         decoration: InputDecoration(labelText: tr('installMethod')), |         dense: true, | ||||||
|         value: settingsProvider.installMethod, |         contentPadding: EdgeInsets.zero, | ||||||
|         items: [ |         title: Text(tr('selectX', args: [tr('colour')])), | ||||||
|           DropdownMenuItem( |         subtitle: Text( | ||||||
|             value: InstallMethodSettings.normal, |             "${ColorTools.nameThatColor(settingsProvider.themeColor)} " | ||||||
|             child: Text(tr('normal')), |             "(${ColorTools.materialNameAndCode(settingsProvider.themeColor, colorSwatchNameMap: colorsNameMap)})"), | ||||||
|           ), |         trailing: ColorIndicator( | ||||||
|           const DropdownMenuItem( |             width: 40, | ||||||
|             value: InstallMethodSettings.shizuku, |             height: 40, | ||||||
|             child: Text('Shizuku'), |             borderRadius: 20, | ||||||
|           ), |             color: settingsProvider.themeColor, | ||||||
|           DropdownMenuItem( |             onSelectFocus: false, | ||||||
|             value: InstallMethodSettings.root, |             onSelect: () async { | ||||||
|             child: Text(tr('root')), |               final Color colorBeforeDialog = settingsProvider.themeColor; | ||||||
|           ) |               if (!(await colorPickerDialog())) { | ||||||
|         ], |                 setState(() { | ||||||
|         onChanged: (value) { |                   settingsProvider.themeColor = colorBeforeDialog; | ||||||
|           if (value != null) { |                 }); | ||||||
|             settingsProvider.installMethod = value; |               } | ||||||
|           } |             })); | ||||||
|         }); |  | ||||||
|  |  | ||||||
|     var themeDropdown = DropdownButtonFormField( |     var useMaterialThemeSwitch = FutureBuilder( | ||||||
|         decoration: InputDecoration(labelText: tr('theme')), |         builder: (ctx, val) { | ||||||
|         value: settingsProvider.theme, |           return ((val.data?.version.sdkInt ?? 0) >= 31) | ||||||
|         items: [ |               ? Row( | ||||||
|           DropdownMenuItem( |                   mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|             value: ThemeSettings.dark, |                   children: [ | ||||||
|             child: Text(tr('dark')), |                     Flexible(child: Text(tr('useMaterialYou'))), | ||||||
|           ), |                     Switch( | ||||||
|           DropdownMenuItem( |                         value: settingsProvider.useMaterialYou, | ||||||
|             value: ThemeSettings.light, |                         onChanged: (value) { | ||||||
|             child: Text(tr('light')), |                           settingsProvider.useMaterialYou = value; | ||||||
|           ), |                         }) | ||||||
|           DropdownMenuItem( |                   ], | ||||||
|             value: ThemeSettings.system, |                 ) | ||||||
|             child: Text(tr('followSystem')), |               : const SizedBox.shrink(); | ||||||
|           ) |         }, | ||||||
|         ], |         future: DeviceInfoPlugin().androidInfo); | ||||||
|         onChanged: (value) { |  | ||||||
|           if (value != null) { |  | ||||||
|             settingsProvider.theme = value; |  | ||||||
|           } |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|     var colourDropdown = DropdownButtonFormField( |  | ||||||
|         decoration: InputDecoration(labelText: tr('colour')), |  | ||||||
|         value: settingsProvider.colour, |  | ||||||
|         items: const [ |  | ||||||
|           DropdownMenuItem( |  | ||||||
|             value: ColourSettings.basic, |  | ||||||
|             child: Text('Obtainium'), |  | ||||||
|           ), |  | ||||||
|           DropdownMenuItem( |  | ||||||
|             value: ColourSettings.materialYou, |  | ||||||
|             child: Text('Material You'), |  | ||||||
|           ) |  | ||||||
|         ], |  | ||||||
|         onChanged: (value) { |  | ||||||
|           if (value != null) { |  | ||||||
|             settingsProvider.colour = value; |  | ||||||
|           } |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|     var sortDropdown = DropdownButtonFormField( |     var sortDropdown = DropdownButtonFormField( | ||||||
|         isExpanded: true, |         isExpanded: true, | ||||||
| @@ -165,30 +275,29 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|           } |           } | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|     var intervalDropdown = DropdownButtonFormField( |     var intervalSlider = Slider( | ||||||
|         decoration: InputDecoration(labelText: tr('bgUpdateCheckInterval')), |       value: settingsProvider.updateIntervalSliderVal, | ||||||
|         value: settingsProvider.updateInterval, |       max: updateIntervalNodes.length.toDouble(), | ||||||
|         items: updateIntervals.map((e) { |       divisions: updateIntervalNodes.length * 20, | ||||||
|           int displayNum = (e < 60 |       label: updateIntervalLabel, | ||||||
|                   ? e |       onChanged: (double value) { | ||||||
|                   : e < 1440 |         setState(() { | ||||||
|                       ? e / 60 |           settingsProvider.updateIntervalSliderVal = value; | ||||||
|                       : e / 1440) |           processIntervalSliderValue(value); | ||||||
|               .round(); |  | ||||||
|           String display = e == 0 |  | ||||||
|               ? tr('neverManualOnly') |  | ||||||
|               : (e < 60 |  | ||||||
|                   ? plural('minute', displayNum) |  | ||||||
|                   : e < 1440 |  | ||||||
|                       ? plural('hour', displayNum) |  | ||||||
|                       : plural('day', displayNum)); |  | ||||||
|           return DropdownMenuItem(value: e, child: Text(display)); |  | ||||||
|         }).toList(), |  | ||||||
|         onChanged: (value) { |  | ||||||
|           if (value != null) { |  | ||||||
|             settingsProvider.updateInterval = value; |  | ||||||
|           } |  | ||||||
|         }); |         }); | ||||||
|  |       }, | ||||||
|  |       onChangeStart: (double value) { | ||||||
|  |         setState(() { | ||||||
|  |           showIntervalLabel = false; | ||||||
|  |         }); | ||||||
|  |       }, | ||||||
|  |       onChangeEnd: (double value) { | ||||||
|  |         setState(() { | ||||||
|  |           showIntervalLabel = true; | ||||||
|  |           settingsProvider.updateInterval = updateInterval; | ||||||
|  |         }); | ||||||
|  |       }, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|     var sourceSpecificFields = sourceProvider.sources.map((e) { |     var sourceSpecificFields = sourceProvider.sources.map((e) { | ||||||
|       if (e.sourceConfigSettingFormItems.isNotEmpty) { |       if (e.sourceConfigSettingFormItems.isNotEmpty) { | ||||||
| @@ -239,15 +348,26 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                                   fontWeight: FontWeight.bold, |                                   fontWeight: FontWeight.bold, | ||||||
|                                   color: Theme.of(context).colorScheme.primary), |                                   color: Theme.of(context).colorScheme.primary), | ||||||
|                             ), |                             ), | ||||||
|                             intervalDropdown, |                             //intervalDropdown, | ||||||
|  |                             height16, | ||||||
|  |                             if (showIntervalLabel) | ||||||
|  |                               SizedBox( | ||||||
|  |                                   child: Text( | ||||||
|  |                                       "${tr('bgUpdateCheckInterval')}: $updateIntervalLabel")) | ||||||
|  |                             else | ||||||
|  |                               const SizedBox(height: 16), | ||||||
|  |                             intervalSlider, | ||||||
|                             FutureBuilder( |                             FutureBuilder( | ||||||
|                                 builder: (ctx, val) { |                                 builder: (ctx, val) { | ||||||
|                                   return (val.data?.version.sdkInt ?? 0) >= 30 |                                   return (settingsProvider.updateInterval > | ||||||
|  |                                               0) && | ||||||
|  |                                           (((val.data?.version.sdkInt ?? 0) >= | ||||||
|  |                                                   30) || | ||||||
|  |                                               settingsProvider.useShizuku) | ||||||
|                                       ? Column( |                                       ? Column( | ||||||
|                                           crossAxisAlignment: |                                           crossAxisAlignment: | ||||||
|                                               CrossAxisAlignment.start, |                                               CrossAxisAlignment.start, | ||||||
|                                           children: [ |                                           children: [ | ||||||
|                                             height16, |  | ||||||
|                                             Row( |                                             Row( | ||||||
|                                               mainAxisAlignment: |                                               mainAxisAlignment: | ||||||
|                                                   MainAxisAlignment |                                                   MainAxisAlignment | ||||||
| @@ -416,7 +536,48 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                                     }) |                                     }) | ||||||
|                               ], |                               ], | ||||||
|                             ), |                             ), | ||||||
|                             installMethodDropdown, |                             height16, | ||||||
|  |                             Row( | ||||||
|  |                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|  |                               children: [ | ||||||
|  |                                 Flexible(child: Text(tr('useShizuku'))), | ||||||
|  |                                 Switch( | ||||||
|  |                                     value: settingsProvider.useShizuku, | ||||||
|  |                                     onChanged: (useShizuku) { | ||||||
|  |                                       if (useShizuku) { | ||||||
|  |                                         ShizukuApkInstaller.checkPermission() | ||||||
|  |                                             .then((resCode) { | ||||||
|  |                                           settingsProvider.useShizuku = | ||||||
|  |                                               resCode!.startsWith('granted'); | ||||||
|  |                                           switch (resCode) { | ||||||
|  |                                             case 'binder_not_found': | ||||||
|  |                                               showError( | ||||||
|  |                                                   ObtainiumError(tr( | ||||||
|  |                                                       'shizukuBinderNotFound')), | ||||||
|  |                                                   context); | ||||||
|  |                                             case 'old_shizuku': | ||||||
|  |                                               showError( | ||||||
|  |                                                   ObtainiumError( | ||||||
|  |                                                       tr('shizukuOld')), | ||||||
|  |                                                   context); | ||||||
|  |                                             case 'old_android_with_adb': | ||||||
|  |                                               showError( | ||||||
|  |                                                   ObtainiumError(tr( | ||||||
|  |                                                       'shizukuOldAndroidWithADB')), | ||||||
|  |                                                   context); | ||||||
|  |                                             case 'denied': | ||||||
|  |                                               showError( | ||||||
|  |                                                   ObtainiumError( | ||||||
|  |                                                       tr('cancelled')), | ||||||
|  |                                                   context); | ||||||
|  |                                           } | ||||||
|  |                                         }); | ||||||
|  |                                       } else { | ||||||
|  |                                         settingsProvider.useShizuku = false; | ||||||
|  |                                       } | ||||||
|  |                                     }) | ||||||
|  |                               ], | ||||||
|  |                             ), | ||||||
|                             height32, |                             height32, | ||||||
|                             Text( |                             Text( | ||||||
|                               tr('sourceSpecific'), |                               tr('sourceSpecific'), | ||||||
| @@ -432,21 +593,49 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                                   fontWeight: FontWeight.bold, |                                   fontWeight: FontWeight.bold, | ||||||
|                                   color: Theme.of(context).colorScheme.primary), |                                   color: Theme.of(context).colorScheme.primary), | ||||||
|                             ), |                             ), | ||||||
|                             themeDropdown, |                             DropdownButtonFormField( | ||||||
|                             height16, |                                 decoration: | ||||||
|                             Row( |                                     InputDecoration(labelText: tr('theme')), | ||||||
|                               mainAxisAlignment: MainAxisAlignment.spaceBetween, |                                 value: settingsProvider.theme, | ||||||
|                               children: [ |                                 items: [ | ||||||
|                                 Flexible(child: Text(tr('useBlackTheme'))), |                                   DropdownMenuItem( | ||||||
|                                 Switch( |                                     value: ThemeSettings.system, | ||||||
|                                     value: settingsProvider.useBlackTheme, |                                     child: Text(tr('followSystem')), | ||||||
|                                     onChanged: (value) { |                                   ), | ||||||
|                                       settingsProvider.useBlackTheme = value; |                                   DropdownMenuItem( | ||||||
|                                     }) |                                     value: ThemeSettings.light, | ||||||
|                               ], |                                     child: Text(tr('light')), | ||||||
|                             ), |                                   ), | ||||||
|                             colourDropdown, |                                   DropdownMenuItem( | ||||||
|  |                                     value: ThemeSettings.dark, | ||||||
|  |                                     child: Text(tr('dark')), | ||||||
|  |                                   ) | ||||||
|  |                                 ], | ||||||
|  |                                 onChanged: (value) { | ||||||
|  |                                   if (value != null) { | ||||||
|  |                                     settingsProvider.theme = value; | ||||||
|  |                                   } | ||||||
|  |                                 }), | ||||||
|  |                             height8, | ||||||
|  |                             if (settingsProvider.theme == ThemeSettings.system) | ||||||
|  |                               followSystemThemeExplanation, | ||||||
|                             height16, |                             height16, | ||||||
|  |                             if (settingsProvider.theme != ThemeSettings.light) | ||||||
|  |                               Row( | ||||||
|  |                                   mainAxisAlignment: | ||||||
|  |                                       MainAxisAlignment.spaceBetween, | ||||||
|  |                                   children: [ | ||||||
|  |                                     Flexible(child: Text(tr('useBlackTheme'))), | ||||||
|  |                                     Switch( | ||||||
|  |                                         value: settingsProvider.useBlackTheme, | ||||||
|  |                                         onChanged: (value) { | ||||||
|  |                                           settingsProvider.useBlackTheme = | ||||||
|  |                                               value; | ||||||
|  |                                         }) | ||||||
|  |                                   ]), | ||||||
|  |                             height8, | ||||||
|  |                             useMaterialThemeSwitch, | ||||||
|  |                             if (!settingsProvider.useMaterialYou) colorPicker, | ||||||
|                             Row( |                             Row( | ||||||
|                               mainAxisAlignment: MainAxisAlignment.start, |                               mainAxisAlignment: MainAxisAlignment.start, | ||||||
|                               crossAxisAlignment: CrossAxisAlignment.start, |                               crossAxisAlignment: CrossAxisAlignment.start, | ||||||
| @@ -460,34 +649,46 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                             ), |                             ), | ||||||
|                             height16, |                             height16, | ||||||
|                             localeDropdown, |                             localeDropdown, | ||||||
|                             height16, |                             FutureBuilder( | ||||||
|                             Row( |                                 builder: (ctx, val) { | ||||||
|                               mainAxisAlignment: MainAxisAlignment.spaceBetween, |                                   return (val.data?.version.sdkInt ?? 0) >= 34 | ||||||
|                               children: [ |                                       ? Column( | ||||||
|                                 Flexible(child: Text(tr('useSystemFont'))), |                                           crossAxisAlignment: | ||||||
|                                 Switch( |                                               CrossAxisAlignment.start, | ||||||
|                                     value: settingsProvider.useSystemFont, |                                           children: [ | ||||||
|                                     onChanged: (useSystemFont) { |                                               height16, | ||||||
|                                       if (useSystemFont) { |                                               Row( | ||||||
|                                         NativeFeatures.loadSystemFont() |                                                   mainAxisAlignment: | ||||||
|                                             .then((fontLoadRes) { |                                                       MainAxisAlignment | ||||||
|                                           if (fontLoadRes == 'ok') { |                                                           .spaceBetween, | ||||||
|                                             settingsProvider.useSystemFont = |                                                   children: [ | ||||||
|                                                 true; |                                                     Flexible( | ||||||
|                                           } else { |                                                         child: Text(tr( | ||||||
|                                             showError( |                                                             'useSystemFont'))), | ||||||
|                                                 ObtainiumError(tr( |                                                     Switch( | ||||||
|                                                     'systemFontError', |                                                         value: settingsProvider | ||||||
|                                                     args: [fontLoadRes])), |                                                             .useSystemFont, | ||||||
|                                                 context); |                                                         onChanged: | ||||||
|                                           } |                                                             (useSystemFont) { | ||||||
|                                         }); |                                                           if (useSystemFont) { | ||||||
|                                       } else { |                                                             NativeFeatures | ||||||
|                                         settingsProvider.useSystemFont = false; |                                                                     .loadSystemFont() | ||||||
|                                       } |                                                                 .then((val) { | ||||||
|                                     }) |                                                               settingsProvider | ||||||
|                               ], |                                                                       .useSystemFont = | ||||||
|                             ), |                                                                   true; | ||||||
|  |                                                             }); | ||||||
|  |                                                           } else { | ||||||
|  |                                                             settingsProvider | ||||||
|  |                                                                     .useSystemFont = | ||||||
|  |                                                                 false; | ||||||
|  |                                                           } | ||||||
|  |                                                         }) | ||||||
|  |                                                   ]) | ||||||
|  |                                             ]) | ||||||
|  |                                       : const SizedBox.shrink(); | ||||||
|  |                                 }, | ||||||
|  |                                 future: DeviceInfoPlugin().androidInfo), | ||||||
|                             height16, |                             height16, | ||||||
|                             Row( |                             Row( | ||||||
|                               mainAxisAlignment: MainAxisAlignment.spaceBetween, |                               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
| @@ -640,17 +841,31 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                 Row( |                 Row( | ||||||
|                   mainAxisAlignment: MainAxisAlignment.spaceAround, |                   mainAxisAlignment: MainAxisAlignment.spaceAround, | ||||||
|                   children: [ |                   children: [ | ||||||
|                     TextButton.icon( |                     IconButton( | ||||||
|                       onPressed: () { |                       onPressed: () { | ||||||
|                         launchUrlString(settingsProvider.sourceUrl, |                         launchUrlString(settingsProvider.sourceUrl, | ||||||
|                             mode: LaunchMode.externalApplication); |                             mode: LaunchMode.externalApplication); | ||||||
|                       }, |                       }, | ||||||
|                       icon: const Icon(Icons.code), |                       icon: const Icon(Icons.code), | ||||||
|                       label: Text( |                       tooltip: tr('appSource'), | ||||||
|                         tr('appSource'), |  | ||||||
|                       ), |  | ||||||
|                     ), |                     ), | ||||||
|                     TextButton.icon( |                     IconButton( | ||||||
|  |                       onPressed: () { | ||||||
|  |                         launchUrlString('${settingsProvider.sourceUrl}/wiki', | ||||||
|  |                             mode: LaunchMode.externalApplication); | ||||||
|  |                       }, | ||||||
|  |                       icon: const Icon(Icons.help_outline_rounded), | ||||||
|  |                       tooltip: tr('wiki'), | ||||||
|  |                     ), | ||||||
|  |                     IconButton( | ||||||
|  |                       onPressed: () { | ||||||
|  |                         launchUrlString('https://apps.obtainium.imranr.dev/', | ||||||
|  |                             mode: LaunchMode.externalApplication); | ||||||
|  |                       }, | ||||||
|  |                       icon: const Icon(Icons.apps_rounded), | ||||||
|  |                       tooltip: tr('crowdsourcedConfigsLabel'), | ||||||
|  |                     ), | ||||||
|  |                     IconButton( | ||||||
|                         onPressed: () { |                         onPressed: () { | ||||||
|                           context.read<LogsProvider>().get().then((logs) { |                           context.read<LogsProvider>().get().then((logs) { | ||||||
|                             if (logs.isEmpty) { |                             if (logs.isEmpty) { | ||||||
| @@ -666,7 +881,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                           }); |                           }); | ||||||
|                         }, |                         }, | ||||||
|                         icon: const Icon(Icons.bug_report_outlined), |                         icon: const Icon(Icons.bug_report_outlined), | ||||||
|                         label: Text(tr('appLogs'))), |                         tooltip: tr('appLogs')) | ||||||
|                   ], |                   ], | ||||||
|                 ), |                 ), | ||||||
|                 const SizedBox( |                 const SizedBox( | ||||||
| @@ -731,6 +946,25 @@ class _LogsDialogState extends State<LogsDialog> { | |||||||
|         ], |         ], | ||||||
|       ), |       ), | ||||||
|       actions: [ |       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( |         TextButton( | ||||||
|             onPressed: () { |             onPressed: () { | ||||||
|               Navigator.of(context).pop(); |               Navigator.of(context).pop(); | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,75 +1,22 @@ | |||||||
| import 'dart:async'; | import 'dart:async'; | ||||||
| import 'dart:io'; | import 'dart:io'; | ||||||
|  | import 'package:android_system_font/android_system_font.dart'; | ||||||
| import 'package:flutter/services.dart'; | import 'package:flutter/services.dart'; | ||||||
|  |  | ||||||
| class NativeFeatures { | class NativeFeatures { | ||||||
|   static const MethodChannel _channel = MethodChannel('native'); |  | ||||||
|   static bool _systemFontLoaded = false; |   static bool _systemFontLoaded = false; | ||||||
|   static bool _callbacksApplied = false; |  | ||||||
|   static int _resPermShizuku = -2;  // not set |  | ||||||
|  |  | ||||||
|   static Future<ByteData> _readFileBytes(String path) async { |   static Future<ByteData> _readFileBytes(String path) async { | ||||||
|     var file = File(path); |     var bytes = await File(path).readAsBytes(); | ||||||
|     var bytes = await file.readAsBytes(); |  | ||||||
|     return ByteData.view(bytes.buffer); |     return ByteData.view(bytes.buffer); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static Future _handleCalls(MethodCall call) async { |   static Future loadSystemFont() async { | ||||||
|     if (call.method == 'resPermShizuku') { |     if (_systemFontLoaded) return; | ||||||
|       _resPermShizuku = call.arguments['res']; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static Future _waitWhile(bool Function() test, |  | ||||||
|       [Duration pollInterval = const Duration(milliseconds: 250)]) { |  | ||||||
|     var completer = Completer(); |  | ||||||
|     check() { |  | ||||||
|       if (test()) { |  | ||||||
|         Timer(pollInterval, check); |  | ||||||
|       } else { |  | ||||||
|         completer.complete(); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     check(); |  | ||||||
|     return completer.future; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static Future<String> loadSystemFont() async { |  | ||||||
|     if (_systemFontLoaded) { return "ok"; } |  | ||||||
|     var getFontRes = await _channel.invokeMethod('getSystemFont'); |  | ||||||
|     if (getFontRes[0] != '/') { return getFontRes; }  // Error |  | ||||||
|     var fontLoader = FontLoader('SystemFont'); |     var fontLoader = FontLoader('SystemFont'); | ||||||
|     fontLoader.addFont(_readFileBytes(getFontRes)); |     var fontFilePath = await AndroidSystemFont().getFilePath(); | ||||||
|     await fontLoader.load(); |     fontLoader.addFont(_readFileBytes(fontFilePath!)); | ||||||
|  |     fontLoader.load(); | ||||||
|     _systemFontLoaded = true; |     _systemFontLoaded = true; | ||||||
|     return "ok"; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static Future<int> checkPermissionShizuku() async { |  | ||||||
|     if (!_callbacksApplied) { |  | ||||||
|       _channel.setMethodCallHandler(_handleCalls); |  | ||||||
|       _callbacksApplied = true; |  | ||||||
|     } |  | ||||||
|     int res = await _channel.invokeMethod('checkPermissionShizuku'); |  | ||||||
|     if (res == -2) { |  | ||||||
|       await _waitWhile(() => _resPermShizuku == -2); |  | ||||||
|       res = _resPermShizuku; |  | ||||||
|       _resPermShizuku = -2; |  | ||||||
|     } |  | ||||||
|     return res; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static Future<bool> checkPermissionRoot() async { |  | ||||||
|     return await _channel.invokeMethod('checkPermissionRoot'); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static Future<bool> installWithShizuku({required String apkFileUri}) async { |  | ||||||
|     return await _channel.invokeMethod( |  | ||||||
|         'installWithShizuku', {'apkFileUri': apkFileUri}); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static Future<bool> installWithRoot({required String apkFilePath}) async { |  | ||||||
|     return await _channel.invokeMethod( |  | ||||||
|         'installWithRoot', {'apkFilePath': apkFilePath}); |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -41,20 +41,26 @@ class UpdateNotification extends ObtainiumNotification { | |||||||
| } | } | ||||||
|  |  | ||||||
| class SilentUpdateNotification extends ObtainiumNotification { | class SilentUpdateNotification extends ObtainiumNotification { | ||||||
|   SilentUpdateNotification(List<App> updates, {int? id}) |   SilentUpdateNotification(List<App> updates, bool succeeded, {int? id}) | ||||||
|       : super( |       : super( | ||||||
|             id ?? 3, |             id ?? 3, | ||||||
|             tr('appsUpdated'), |             succeeded | ||||||
|  |                 ? tr('appsUpdated') | ||||||
|  |                 : tr('appsNotUpdated'), | ||||||
|             '', |             '', | ||||||
|             'APPS_UPDATED', |             'APPS_UPDATED', | ||||||
|             tr('appsUpdatedNotifChannel'), |             tr('appsUpdatedNotifChannel'), | ||||||
|             tr('appsUpdatedNotifDescription'), |             tr('appsUpdatedNotifDescription'), | ||||||
|             Importance.defaultImportance) { |             Importance.defaultImportance) { | ||||||
|     message = updates.length == 1 |     message = updates.length == 1 | ||||||
|         ? tr('xWasUpdatedToY', |         ? tr(succeeded | ||||||
|             args: [updates[0].finalName, updates[0].latestVersion]) |             ? 'xWasUpdatedToY' | ||||||
|         : plural('xAndNMoreUpdatesInstalled', updates.length - 1, |             : 'xWasNotUpdatedToY', | ||||||
|             args: [updates[0].finalName, (updates.length - 1).toString()]); |                 args: [updates[0].finalName, updates[0].latestVersion]) | ||||||
|  |         : plural(succeeded | ||||||
|  |             ? 'xAndNMoreUpdatesInstalled' | ||||||
|  |             : "xAndNMoreUpdatesFailed", | ||||||
|  |                 updates.length - 1, args: [updates[0].finalName, (updates.length - 1).toString()]); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -17,41 +17,14 @@ import 'package:shared_storage/shared_storage.dart' as saf; | |||||||
| String obtainiumTempId = 'imranr98_obtainium_${GitHub().hosts[0]}'; | String obtainiumTempId = 'imranr98_obtainium_${GitHub().hosts[0]}'; | ||||||
| String obtainiumId = 'dev.imranr.obtainium'; | String obtainiumId = 'dev.imranr.obtainium'; | ||||||
| String obtainiumUrl = 'https://github.com/ImranR98/Obtainium'; | String obtainiumUrl = 'https://github.com/ImranR98/Obtainium'; | ||||||
|  | Color obtainiumThemeColor = const Color(0xFF6438B5); | ||||||
| enum InstallMethodSettings { normal, shizuku, root } |  | ||||||
|  |  | ||||||
| enum ThemeSettings { system, light, dark } | enum ThemeSettings { system, light, dark } | ||||||
|  |  | ||||||
| enum ColourSettings { basic, materialYou } |  | ||||||
|  |  | ||||||
| enum SortColumnSettings { added, nameAuthor, authorName, releaseDate } | enum SortColumnSettings { added, nameAuthor, authorName, releaseDate } | ||||||
|  |  | ||||||
| enum SortOrderSettings { ascending, descending } | enum SortOrderSettings { ascending, descending } | ||||||
|  |  | ||||||
| const maxAPIRateLimitMinutes = 30; |  | ||||||
| const minUpdateIntervalMinutes = maxAPIRateLimitMinutes + 30; |  | ||||||
| const maxUpdateIntervalMinutes = 43200; |  | ||||||
| List<int> updateIntervals = [ |  | ||||||
|   15, |  | ||||||
|   30, |  | ||||||
|   60, |  | ||||||
|   120, |  | ||||||
|   180, |  | ||||||
|   360, |  | ||||||
|   720, |  | ||||||
|   1440, |  | ||||||
|   4320, |  | ||||||
|   10080, |  | ||||||
|   20160, |  | ||||||
|   43200, |  | ||||||
|   0 |  | ||||||
| ] |  | ||||||
|     .where((element) => |  | ||||||
|         (element >= minUpdateIntervalMinutes && |  | ||||||
|             element <= maxUpdateIntervalMinutes) || |  | ||||||
|         element == 0) |  | ||||||
|     .toList(); |  | ||||||
|  |  | ||||||
| class SettingsProvider with ChangeNotifier { | class SettingsProvider with ChangeNotifier { | ||||||
|   SharedPreferences? prefs; |   SharedPreferences? prefs; | ||||||
|   String? defaultAppDir; |   String? defaultAppDir; | ||||||
| @@ -75,13 +48,12 @@ class SettingsProvider with ChangeNotifier { | |||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   InstallMethodSettings get installMethod { |   bool get useShizuku{ | ||||||
|     return InstallMethodSettings.values[ |     return prefs?.getBool('useShizuku') ?? false; | ||||||
|         prefs?.getInt('installMethod') ?? InstallMethodSettings.normal.index]; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   set installMethod(InstallMethodSettings t) { |   set useShizuku(bool useShizuku) { | ||||||
|     prefs?.setInt('installMethod', t.index); |     prefs?.setBool('useShizuku', useShizuku); | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -95,13 +67,23 @@ class SettingsProvider with ChangeNotifier { | |||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   ColourSettings get colour { |   Color get themeColor { | ||||||
|     return ColourSettings |     int? colorCode = prefs?.getInt('themeColor'); | ||||||
|         .values[prefs?.getInt('colour') ?? ColourSettings.basic.index]; |     return (colorCode != null) ? | ||||||
|  |         Color(colorCode) : obtainiumThemeColor; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   set colour(ColourSettings t) { |   set themeColor(Color themeColor) { | ||||||
|     prefs?.setInt('colour', t.index); |     prefs?.setInt('themeColor', themeColor.value); | ||||||
|  |     notifyListeners(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   bool get useMaterialYou { | ||||||
|  |     return prefs?.getBool('useMaterialYou') ?? false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   set useMaterialYou(bool useMaterialYou) { | ||||||
|  |     prefs?.setBool('useMaterialYou', useMaterialYou); | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -115,21 +97,20 @@ class SettingsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   int get updateInterval { |   int get updateInterval { | ||||||
|     var min = prefs?.getInt('updateInterval') ?? 360; |     return prefs?.getInt('updateInterval') ?? 360; | ||||||
|     if (!updateIntervals.contains(min)) { |  | ||||||
|       var temp = updateIntervals[0]; |  | ||||||
|       for (var i in updateIntervals) { |  | ||||||
|         if (min > i && i != 0) { |  | ||||||
|           temp = i; |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|       min = temp; |  | ||||||
|     } |  | ||||||
|     return min; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   set updateInterval(int min) { |   set updateInterval(int min) { | ||||||
|     prefs?.setInt('updateInterval', (min < 15 && min != 0) ? 15 : min); |     prefs?.setInt('updateInterval', min); | ||||||
|  |     notifyListeners(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   double get updateIntervalSliderVal { | ||||||
|  |     return prefs?.getDouble('updateIntervalSliderVal') ?? 6.0; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   set updateIntervalSliderVal(double val) { | ||||||
|  |     prefs?.setDouble('updateIntervalSliderVal', val); | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,11 +2,13 @@ | |||||||
| // AppSource is an abstract class with a concrete implementation for each source | // AppSource is an abstract class with a concrete implementation for each source | ||||||
|  |  | ||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
|  | import 'dart:io'; | ||||||
|  |  | ||||||
| import 'package:device_info_plus/device_info_plus.dart'; | import 'package:device_info_plus/device_info_plus.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:html/dom.dart'; | import 'package:html/dom.dart'; | ||||||
| import 'package:http/http.dart'; | import 'package:http/http.dart'; | ||||||
|  | import 'package:http/io_client.dart'; | ||||||
| import 'package:obtainium/app_sources/apkmirror.dart'; | import 'package:obtainium/app_sources/apkmirror.dart'; | ||||||
| import 'package:obtainium/app_sources/apkpure.dart'; | import 'package:obtainium/app_sources/apkpure.dart'; | ||||||
| import 'package:obtainium/app_sources/aptoide.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/sourcehut.dart'; | ||||||
| import 'package:obtainium/app_sources/steammobile.dart'; | import 'package:obtainium/app_sources/steammobile.dart'; | ||||||
| import 'package:obtainium/app_sources/telegramapp.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/uptodown.dart'; | ||||||
| import 'package:obtainium/app_sources/vlc.dart'; | import 'package:obtainium/app_sources/vlc.dart'; | ||||||
| import 'package:obtainium/app_sources/whatsapp.dart'; | import 'package:obtainium/app_sources/whatsapp.dart'; | ||||||
| @@ -354,11 +357,17 @@ preStandardizeUrl(String url) { | |||||||
|       url.toLowerCase().indexOf('https://') != 0) { |       url.toLowerCase().indexOf('https://') != 0) { | ||||||
|     url = 'https://$url'; |     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 |   url = url | ||||||
|       .split('/') |           .split('/') | ||||||
|       .where((e) => e.isNotEmpty) |           .where((e) => e.isNotEmpty) | ||||||
|       .join('/') |           .join('/') | ||||||
|       .replaceFirst(':/', '://'); |           .replaceFirst(':/', '://') + | ||||||
|  |       (trailingSlash ? '/' : ''); | ||||||
|   return url; |   return url; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -393,6 +402,15 @@ getSourceRegex(List<String> hosts) { | |||||||
|   return '(${hosts.join('|').replaceAll('.', '\\.')})'; |   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 { | abstract class AppSource { | ||||||
|   List<String> hosts = []; |   List<String> hosts = []; | ||||||
|   bool hostChanged = false; |   bool hostChanged = false; | ||||||
| @@ -406,6 +424,7 @@ abstract class AppSource { | |||||||
|   bool showReleaseDateAsVersionToggle = false; |   bool showReleaseDateAsVersionToggle = false; | ||||||
|   bool versionDetectionDisallowed = false; |   bool versionDetectionDisallowed = false; | ||||||
|   List<String> excludeCommonSettingKeys = []; |   List<String> excludeCommonSettingKeys = []; | ||||||
|  |   bool urlsAlwaysHaveExtension = false; | ||||||
|  |  | ||||||
|   AppSource() { |   AppSource() { | ||||||
|     name = runtimeType.toString(); |     name = runtimeType.toString(); | ||||||
| @@ -447,21 +466,33 @@ abstract class AppSource { | |||||||
|  |  | ||||||
|   Future<Response> sourceRequest( |   Future<Response> sourceRequest( | ||||||
|       String url, Map<String, dynamic> additionalSettings, |       String url, Map<String, dynamic> additionalSettings, | ||||||
|       {bool followRedirects = true}) async { |       {bool followRedirects = true, Object? postBody}) async { | ||||||
|     var requestHeaders = await getRequestHeaders(additionalSettings); |     var requestHeaders = await getRequestHeaders(additionalSettings); | ||||||
|     if (requestHeaders != null || followRedirects == false) { |     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; |       req.followRedirects = followRedirects; | ||||||
|       if (requestHeaders != null) { |       if (requestHeaders != null) { | ||||||
|         req.headers.addAll(requestHeaders); |         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 { |     } 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(); |     throw NotImplementedError(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -485,13 +516,15 @@ abstract class AppSource { | |||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       GeneratedFormTextField('versionExtractionRegEx', |       GeneratedFormTextField('versionExtractionRegEx', | ||||||
|           label: tr('versionExtractionRegEx'), |           label: tr('trimVersionString'), | ||||||
|           required: false, |           required: false, | ||||||
|           additionalValidators: [(value) => regExValidator(value)]), |           additionalValidators: [(value) => regExValidator(value)]), | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       GeneratedFormTextField('matchGroupToUse', |       GeneratedFormTextField('matchGroupToUse', | ||||||
|           label: tr('matchGroupToUse'), required: false, hint: '\$0') |           label: tr('matchGroupToUseForX', args: [tr('trimVersionString')]), | ||||||
|  |           required: false, | ||||||
|  |           hint: '\$0') | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       GeneratedFormSwitch('versionDetection', |       GeneratedFormSwitch('versionDetection', | ||||||
| @@ -521,6 +554,14 @@ abstract class AppSource { | |||||||
|           label: tr('autoApkFilterByArch'), defaultValue: true) |           label: tr('autoApkFilterByArch'), defaultValue: true) | ||||||
|     ], |     ], | ||||||
|     [GeneratedFormTextField('appName', label: tr('appName'), required: false)], |     [GeneratedFormTextField('appName', label: tr('appName'), required: false)], | ||||||
|  |     [ | ||||||
|  |       GeneratedFormSwitch('shizukuPretendToBeGooglePlay', | ||||||
|  |           label: tr('shizukuPretendToBeGooglePlay'), defaultValue: false) | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       GeneratedFormSwitch('allowInsecure', | ||||||
|  |           label: tr('allowInsecure'), defaultValue: false) | ||||||
|  |     ], | ||||||
|     [ |     [ | ||||||
|       GeneratedFormSwitch('exemptFromBackgroundUpdates', |       GeneratedFormSwitch('exemptFromBackgroundUpdates', | ||||||
|           label: tr('exemptFromBackgroundUpdates')) |           label: tr('exemptFromBackgroundUpdates')) | ||||||
| @@ -611,7 +652,7 @@ abstract class AppSource { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   bool canSearch = false; |   bool canSearch = false; | ||||||
|   bool excludeFromMassSearch = false; |   bool includeAdditionalOptsInMainSearch = false; | ||||||
|   List<GeneratedFormItem> searchQuerySettingFormItems = []; |   List<GeneratedFormItem> searchQuerySettingFormItems = []; | ||||||
|   Future<Map<String, List<String>>> search(String query, |   Future<Map<String, List<String>>> search(String query, | ||||||
|       {Map<String, dynamic> querySettings = const {}}) { |       {Map<String, dynamic> querySettings = const {}}) { | ||||||
| @@ -747,9 +788,10 @@ class SourceProvider { | |||||||
|         APKPure(), |         APKPure(), | ||||||
|         Aptoide(), |         Aptoide(), | ||||||
|         Uptodown(), |         Uptodown(), | ||||||
|         APKMirror(), |  | ||||||
|         HuaweiAppGallery(), |         HuaweiAppGallery(), | ||||||
|  |         Tencent(), | ||||||
|         Jenkins(), |         Jenkins(), | ||||||
|  |         APKMirror(), | ||||||
|         Signal(), |         Signal(), | ||||||
|         VLC(), |         VLC(), | ||||||
|         WhatsApp(), |         WhatsApp(), | ||||||
| @@ -792,7 +834,7 @@ class SourceProvider { | |||||||
|       for (var s in sources.where( |       for (var s in sources.where( | ||||||
|           (element) => element.hosts.isEmpty && !element.neverAutoSelect)) { |           (element) => element.hosts.isEmpty && !element.neverAutoSelect)) { | ||||||
|         try { |         try { | ||||||
|           s.sourceSpecificStandardizeURL(url); |           s.sourceSpecificStandardizeURL(url, forSelection: true); | ||||||
|           source = s; |           source = s; | ||||||
|           break; |           break; | ||||||
|         } catch (e) { |         } catch (e) { | ||||||
|   | |||||||
							
								
								
									
										488
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										488
									
								
								pubspec.lock
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										32
									
								
								pubspec.yaml
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								pubspec.yaml
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | |||||||
| name: obtainium | 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 | # The following line prevents the package from being accidentally published to | ||||||
| # pub.dev using `flutter pub publish`. This is preferred for private packages. | # 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 | # 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 | # 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. | # of the product and file versions while build-number is used as the build suffix. | ||||||
| version: 1.1.2+2259 | version: 1.1.22+2279 | ||||||
|  |  | ||||||
| environment: | environment: | ||||||
|   sdk: '>=3.0.0 <4.0.0' |   sdk: '>=3.0.0 <4.0.0' | ||||||
| @@ -37,7 +37,7 @@ dependencies: | |||||||
|   # Use with the CupertinoIcons class for iOS style icons. |   # Use with the CupertinoIcons class for iOS style icons. | ||||||
|   cupertino_icons: ^1.0.5 |   cupertino_icons: ^1.0.5 | ||||||
|   path_provider: ^2.0.11 |   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 |   flutter_local_notifications: ^17.0.0 | ||||||
|   provider: ^6.0.3 |   provider: ^6.0.3 | ||||||
|   http: ^1.0.0 |   http: ^1.0.0 | ||||||
| @@ -56,30 +56,42 @@ dependencies: | |||||||
|       url: https://github.com/ImranR98/android_package_installer |       url: https://github.com/ImranR98/android_package_installer | ||||||
|       ref: main |       ref: main | ||||||
|   android_package_manager: ^0.7.0 |   android_package_manager: ^0.7.0 | ||||||
|   share_plus: ^8.0.2 |   share_plus: ^10.0.0 | ||||||
|   sqflite: ^2.2.0+3 |   sqflite: ^2.2.0+3 | ||||||
|   easy_localization: ^3.0.1 |   easy_localization: ^3.0.1 | ||||||
|   android_intent_plus: ^5.0.1 |   android_intent_plus: ^5.0.1 | ||||||
|   flutter_markdown: ^0.6.14 |   flutter_markdown: ^0.7.1 | ||||||
|   flutter_archive: ^6.0.0 |   flutter_archive: ^6.0.0 | ||||||
|   hsluv: ^1.1.3 |   hsluv: ^1.1.3 | ||||||
|   connectivity_plus: ^6.0.1 |   connectivity_plus: ^6.0.1 | ||||||
|   shared_storage: ^0.8.0 |   shared_storage: ^0.8.0 | ||||||
|   crypto: ^3.0.3 |   crypto: ^3.0.3 | ||||||
|   app_links: ^4.0.0 |   app_links: ^6.0.1 | ||||||
|   background_fetch: ^1.2.1 |   background_fetch: ^1.2.1 | ||||||
|  |   equations: ^5.0.2 | ||||||
|  |   flex_color_picker: ^3.4.1 | ||||||
|  |   android_system_font: | ||||||
|  |     git: | ||||||
|  |       url: https://github.com/re7gog/android_system_font | ||||||
|  |       ref: master | ||||||
|  |   shizuku_apk_installer: | ||||||
|  |     git: | ||||||
|  |       url: https://github.com/re7gog/shizuku_apk_installer | ||||||
|  |       ref: master | ||||||
|  |  | ||||||
|  |   markdown: any | ||||||
|  |   flutter_typeahead: ^5.2.0 | ||||||
| dev_dependencies: | dev_dependencies: | ||||||
|   flutter_test: |   flutter_test: | ||||||
|     sdk: flutter |     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 |   # The "flutter_lints" package below contains a set of recommended lints to | ||||||
|   # encourage good coding practices. The lint set provided by the package is |   # 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 |   # activated in the `analysis_options.yaml` file located at the root of your | ||||||
|   # package. See that file for information about deactivating specific lint |   # package. See that file for information about deactivating specific lint | ||||||
|   # rules and activating additional ones. |   # rules and activating additional ones. | ||||||
|   flutter_lints: ^3.0.0 |   flutter_lints: ^5.0.0 | ||||||
|  |  | ||||||
| flutter_launcher_icons: | flutter_launcher_icons: | ||||||
|   android: "ic_launcher" |   android: "ic_launcher" | ||||||
| @@ -133,6 +145,6 @@ flutter: | |||||||
|   # see https://flutter.dev/custom-fonts/#from-packages |   # see https://flutter.dev/custom-fonts/#from-packages | ||||||
|  |  | ||||||
|   fonts: |   fonts: | ||||||
|       - family: Metropolis |       - family: Wix-Madefor-Display | ||||||
|         fonts: |         fonts: | ||||||
|           - asset: assets/fonts/Metropolis-Regular.otf |           - asset: assets/fonts/WixMadeforDisplay-Regular.otf | ||||||
		Reference in New Issue
	
	Block a user