Compare commits
	
		
			206 Commits
		
	
	
		
			v1.1.38
			...
			f6587ae8da
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | f6587ae8da | ||
|  | 0a30bf6d8e | ||
|  | 8bec3cf053 | ||
|  | 43efc044d7 | ||
|  | 63e71624f9 | ||
|  | 3edaa143e4 | ||
|  | 798bddd17f | ||
|  | 4f27e25a23 | ||
|  | a2a1f48310 | ||
|  | c246548436 | ||
|  | f67cfeb231 | ||
|  | 9758f8b391 | ||
|  | c45aaff610 | ||
|  | bd32a10de7 | ||
|  | ef83be07cd | ||
|  | 9fe33737eb | ||
|  | 51c7acf466 | ||
|  | d8490c289d | ||
|  | 1d73d4dcbb | ||
|  | 6d28d15d4f | ||
|  | 3890785fc7 | ||
|  | 38d34c810b | ||
|  | b2d6752b80 | ||
|  | 47a6e0dc7c | ||
|  | 5b6299496f | ||
|  | 369127806f | ||
|  | ee9b0e710c | ||
|  | 6e0819b0a7 | ||
|  | d8c9cd6579 | ||
|  | ae2dad01ff | ||
|  | 3345b26fa9 | ||
|  | 99e3b8ac4f | ||
|  | 7d01141db5 | ||
|  | 195c0d1f11 | ||
|  | 6b8bb095d4 | ||
|  | fd9bf721d4 | ||
|  | 1a058a1959 | ||
|  | f78beff5d8 | ||
|  | 39757a3416 | ||
|  | e7ff0bb2cb | ||
|  | b1bd488f22 | ||
|  | 8201348995 | ||
|  | c1ddc57126 | ||
|  | b41b9d3109 | ||
|  | add2574c25 | ||
|  | c702dbcefe | ||
|  | 19710cf7ad | ||
|  | 104a19bc32 | ||
|  | b02de232ee | ||
|  | 971b5dbf11 | ||
|  | eb805bd30a | ||
|  | 21ec90c11c | ||
|  | 51ed211b3d | ||
|  | 34848794e4 | ||
|  | ab3489f60d | ||
|  | 12c878eaba | ||
|  | 6c3a496be1 | ||
|  | b2742dd904 | ||
|  | 1d69e42949 | ||
|  | 9ce74beda5 | ||
|  | 26bca6df64 | ||
|  | 7a7a6f78e9 | ||
|  | 8d062f0296 | ||
|  | 8d54824a7d | ||
|  | 20d3c4141e | ||
|  | 2f54de6d7c | ||
|  | 849a565177 | ||
|  | 972d7d6634 | ||
|  | 2a3251aacb | ||
|  | fa6ed32ad5 | ||
|  | 53caa88d6f | ||
|  | 5dab58cc89 | ||
|  | e061e99451 | ||
|  | a282080fea | ||
|  | 0b812b508a | ||
|  | e639758b15 | ||
|  | f159c0bd44 | ||
|  | 950bf28289 | ||
|  | ecf4326b47 | ||
|  | 98182d9873 | ||
|  | c7c6731732 | ||
|  | b62b60d9df | ||
|  | 3e41913153 | ||
|  | 6b4943349a | ||
|  | 2f60835801 | ||
|  | 3de2121ed8 | ||
|  | e1c80229ab | ||
|  | e9feaf0d8b | ||
|  | 3175597a2a | ||
|  | 6af1748a78 | ||
|  | c9aed8dfc4 | ||
|  | 9c3bdafa47 | ||
|  | d4e857f7f4 | ||
|  | 74d6ffcfb3 | ||
|  | 10634e8ed2 | ||
|  | 2317c5e0d3 | ||
|  | e9dfb494f7 | ||
|  | 71ca60244b | ||
|  | 94ab83ff75 | ||
|  | 2aa91ed535 | ||
|  | 23a72f1c83 | ||
|  | 12a25d5cbc | ||
|  | eb57d150b4 | ||
|  | 5dc8461341 | ||
|  | 5cf751caa8 | ||
|  | 526ebdfc32 | ||
|  | 48f2164fe0 | ||
|  | 0d4bc44aa4 | ||
|  | e78961f0ab | ||
|  | 6784c6c0c6 | ||
|  | b6ef153bfa | ||
|  | 6a3805723d | ||
|  | ebc6671d62 | ||
|  | c87fc6f242 | ||
|  | b23db43620 | ||
|  | b2b10739e3 | ||
|  | 32cc97a18a | ||
|  | 4815e5180e | ||
|  | 14325eb710 | ||
|  | 7607b747c1 | ||
|  | 774bc07663 | ||
|  | 4e43001276 | ||
|  | 46835d0876 | ||
|  | 48d8eb214d | ||
|  | 988f9a6f9f | ||
|  | 51f0d745c9 | ||
|  | ae72302f4c | ||
|  | e157bda0eb | ||
|  | adf05abfac | ||
|  | 6a8cf0096d | ||
|  | 5e6e3f457c | ||
|  | 044bd7f8a0 | ||
|  | 902f29fdcf | ||
|  | 6c6f256976 | ||
|  | fbf2330c0f | ||
|  | 86ac573edd | ||
|  | 6a07138389 | ||
|  | 23c2664ead | ||
|  | 4200e1d954 | ||
|  | 3eb3cf25bf | ||
|  | 1b039fb5a5 | ||
|  | 6a4a15ab4d | ||
|  | a70c6ef1ed | ||
|  | 85e5dddd34 | ||
|  | d9c29c6751 | ||
|  | 0fd496e660 | ||
|  | b3af899ba1 | ||
|  | 9a58643088 | ||
|  | b223522801 | ||
|  | 375caa4511 | ||
|  | 92ebbd9138 | ||
|  | a3c2761aba | ||
|  | 68e38259bd | ||
|  | 4aed749b44 | ||
|  | e135476d86 | ||
|  | 7490942fad | ||
|  | 794be438b0 | ||
|  | f2f055ad83 | ||
|  | 4c94f278e2 | ||
|  | 22acb6a2dd | ||
|  | d32befb832 | ||
|  | 94d8295992 | ||
|  | dcf9f5732a | ||
|  | c89790d58f | ||
|  | 3633c58bea | ||
|  | 9770501aec | ||
|  | 167d0ccced | ||
|  | 430d1f2690 | ||
|  | eb21ba3f6e | ||
|  | 3ab14e2311 | ||
|  | fff3b22e74 | ||
|  | 46354e648a | ||
|  | 4f67ba3f3b | ||
|  | e5db702a67 | ||
|  | 53451fd883 | ||
|  | 491d3cb723 | ||
|  | 6c7644c9b3 | ||
|  | 8539581fe9 | ||
|  | f301f6cedb | ||
|  | 93c8bca038 | ||
|  | 0961e044a7 | ||
|  | 7da631e429 | ||
|  | 66c08f0bfd | ||
|  | 79a2484d68 | ||
|  | c3c7c844dc | ||
|  | 075d0a0854 | ||
|  | cc0416e344 | ||
|  | af7088b754 | ||
|  | ba21137da7 | ||
|  | a7a5749d4f | ||
|  | fcf3c8b635 | ||
|  | 69f578652c | ||
|  | 7593af25b9 | ||
|  | 1e8fd33469 | ||
|  | 03f50e501e | ||
|  | 21dacb6171 | ||
|  | 16ecc88fcd | ||
|  | 1f829289ed | ||
|  | be739b7639 | ||
|  | 4b66aefb33 | ||
|  | 3f948ae73c | ||
|  | de2e881e82 | ||
|  | 9af3aaa82a | ||
|  | 3a5157ced0 | ||
|  | e16320f995 | ||
|  | aac01885f1 | 
							
								
								
									
										15
									
								
								.dockerignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | |||||||
|  | LICENSE.md | ||||||
|  | README.md | ||||||
|  | analysis_options.yaml | ||||||
|  | android | ||||||
|  | assets | ||||||
|  | build | ||||||
|  | build.sh | ||||||
|  | data | ||||||
|  | docker | ||||||
|  | fastlane | ||||||
|  | lib | ||||||
|  | pubspec.lock | ||||||
|  | pubspec.yaml | ||||||
|  | sign.sh | ||||||
|  | test | ||||||
							
								
								
									
										2
									
								
								.flutter
									
									
									
									
									
								
							
							
								
								
								
								
								
							
						
						
							
								
								
									
										16
									
								
								.github/workflows/fastlane.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,16 @@ | |||||||
|  | name: Validate Fastlane metadata | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   workflow_dispatch: | ||||||
|  |   push: | ||||||
|  |       branches: [ "main" ] | ||||||
|  |   pull_request: | ||||||
|  |       branches: [ "main" ] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   go: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |       - name: Validate Fastlane Supply Metadata | ||||||
|  |         uses: ashutoshgngwr/validate-fastlane-supply-metadata@v2 | ||||||
							
								
								
									
										4
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -20,7 +20,7 @@ jobs: | |||||||
|       - uses: actions/setup-java@v4 |       - uses: actions/setup-java@v4 | ||||||
|         with: |         with: | ||||||
|           distribution: 'temurin' # See 'Supported distributions' for available options |           distribution: 'temurin' # See 'Supported distributions' for available options | ||||||
|           java-version: '17' |           java-version: '21' | ||||||
|  |  | ||||||
|       - name: Flutter Doctor |       - name: Flutter Doctor | ||||||
|         id: flutter_doctor |         id: flutter_doctor | ||||||
| @@ -44,7 +44,7 @@ jobs: | |||||||
|  |  | ||||||
|       - name: Build APKs |       - name: Build APKs | ||||||
|         run: | |         run: | | ||||||
|           sed -i 's/signingConfig signingConfigs.release//g' android/app/build.gradle |           sed -i 's/signingConfig = signingConfigs.getByName("release")//g' android/app/build.gradle.kts | ||||||
|           flutter build apk --flavor normal && flutter build apk --split-per-abi --flavor normal |           flutter build apk --flavor normal && flutter build apk --split-per-abi --flavor normal | ||||||
|           for file in build/app/outputs/flutter-apk/app-*normal*.apk*; do mv "$file" "${file//-normal/}"; done |           for file in build/app/outputs/flutter-apk/app-*normal*.apk*; do mv "$file" "${file//-normal/}"; done | ||||||
|           flutter build apk --flavor fdroid -t lib/main_fdroid.dart && flutter build apk --split-per-abi --flavor fdroid -t lib/main_fdroid.dart |           flutter build apk --flavor fdroid -t lib/main_fdroid.dart && flutter build apk --split-per-abi --flavor fdroid -t lib/main_fdroid.dart | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -45,6 +45,8 @@ app.*.map.json | |||||||
| /android/app/debug | /android/app/debug | ||||||
| /android/app/profile | /android/app/profile | ||||||
| /android/app/release | /android/app/release | ||||||
|  | /android/app/.cxx | ||||||
|  |  | ||||||
| # Custom | # Custom | ||||||
| TODO.txt | TODO.txt | ||||||
|  | data | ||||||
| @@ -8,6 +8,7 @@ Obtainium allows you to install and update apps directly from their releases pag | |||||||
|  |  | ||||||
| More info: | More info: | ||||||
| - [Obtainium Wiki](https://wiki.obtainium.imranr.dev/) ([repository](https://github.com/ImranR98/Obtainium-Wiki)) | - [Obtainium Wiki](https://wiki.obtainium.imranr.dev/) ([repository](https://github.com/ImranR98/Obtainium-Wiki)) | ||||||
|  | - [Obtainium 101](https://www.youtube.com/watch?v=0MF_v2OBncw) - Tutorial video | ||||||
| - [AppVerifier](https://github.com/soupslurpr/AppVerifier) - App verification tool (recommended, integrates with Obtainium) | - [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 ([repository](https://github.com/ImranR98/apps.obtainium.imranr.dev)) | - [apps.obtainium.imranr.dev](https://apps.obtainium.imranr.dev/) - Crowdsourced app configurations ([repository](https://github.com/ImranR98/apps.obtainium.imranr.dev)) | ||||||
| - [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 | - [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 | ||||||
| @@ -29,11 +30,14 @@ Currently supported App sources: | |||||||
|   - [Uptodown](https://uptodown.com/) |   - [Uptodown](https://uptodown.com/) | ||||||
|   - [Huawei AppGallery](https://appgallery.huawei.com/) |   - [Huawei AppGallery](https://appgallery.huawei.com/) | ||||||
|   - [Tencent App Store](https://sj.qq.com/) |   - [Tencent App Store](https://sj.qq.com/) | ||||||
|  |   - [CoolApk](https://coolapk.com/) | ||||||
|  |   - [vivo App Store (CN)](https://h5.appstore.vivo.com.cn/) | ||||||
|  |   - [RuStore](https://rustore.ru/) | ||||||
|   - Jenkins Jobs |   - Jenkins Jobs | ||||||
|   - [APKMirror](https://apkmirror.com/) (Track-Only) |   - [APKMirror](https://apkmirror.com/) (Track-Only) | ||||||
| - Other - App-Specific: | - Other - App-Specific: | ||||||
|   - [Telegram App](https://telegram.org) |   - [Telegram App](https://telegram.org/) | ||||||
|   - [Neutron Code](https://neutroncode.com) |   - [Neutron Code](https://neutroncode.com/) | ||||||
| - Direct APK Link | - 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 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,105 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id "com.android.application" |  | ||||||
|     id "kotlin-android" |  | ||||||
|     // The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins. |  | ||||||
|     id "dev.flutter.flutter-gradle-plugin" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def localProperties = new Properties() |  | ||||||
| def localPropertiesFile = rootProject.file('local.properties') |  | ||||||
| if (localPropertiesFile.exists()) { |  | ||||||
|     localPropertiesFile.withReader('UTF-8') { reader -> |  | ||||||
|         localProperties.load(reader) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def flutterVersionCode = localProperties.getProperty('flutter.versionCode') |  | ||||||
| if (flutterVersionCode == null) { |  | ||||||
|     flutterVersionCode = '1' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def flutterVersionName = localProperties.getProperty('flutter.versionName') |  | ||||||
| if (flutterVersionName == null) { |  | ||||||
|     flutterVersionName = '1.0' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def keystoreProperties = new Properties() |  | ||||||
| def keystorePropertiesFile = rootProject.file('key.properties') |  | ||||||
| if (keystorePropertiesFile.exists()) { |  | ||||||
|     keystoreProperties.load(new FileInputStream(keystorePropertiesFile)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| android { |  | ||||||
|     namespace = "dev.imranr.obtainium" |  | ||||||
|     compileSdk = flutter.compileSdkVersion |  | ||||||
|     ndkVersion = flutter.ndkVersion |  | ||||||
|  |  | ||||||
|     compileOptions { |  | ||||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 |  | ||||||
|         targetCompatibility = JavaVersion.VERSION_1_8 |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     kotlinOptions { |  | ||||||
|         jvmTarget = JavaVersion.VERSION_1_8 |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     defaultConfig { |  | ||||||
|         // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). |  | ||||||
|         applicationId = "dev.imranr.obtainium" |  | ||||||
|         // You can update the following values to match your application needs. |  | ||||||
|         // For more information, see: https://flutter.dev/to/review-gradle-config. |  | ||||||
|         minSdk = 24 |  | ||||||
|         targetSdk = flutter.targetSdkVersion |  | ||||||
|         versionCode = flutter.versionCode |  | ||||||
|         versionName = flutter.versionName |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     flavorDimensions "flavor" |  | ||||||
|  |  | ||||||
|     productFlavors { |  | ||||||
|         normal { |  | ||||||
|             dimension "flavor" |  | ||||||
|             applicationIdSuffix "" |  | ||||||
|         } |  | ||||||
|         fdroid { |  | ||||||
|             dimension "flavor" |  | ||||||
|             applicationIdSuffix ".fdroid" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|      |  | ||||||
|     signingConfigs { |  | ||||||
|         release { |  | ||||||
|             keyAlias keystoreProperties['keyAlias'] |  | ||||||
|             keyPassword keystoreProperties['keyPassword'] |  | ||||||
|             storeFile keystoreProperties['storeFile'] ? file(keystoreProperties['storeFile']) : null |  | ||||||
|             storePassword keystoreProperties['storePassword'] |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     buildTypes { |  | ||||||
|         release { |  | ||||||
|             signingConfig signingConfigs.release |  | ||||||
|         } |  | ||||||
|         debug { |  | ||||||
|             applicationIdSuffix = ".debug" |  | ||||||
|             versionNameSuffix = "-debug" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| flutter { |  | ||||||
|     source = "../.." |  | ||||||
| } |  | ||||||
|  |  | ||||||
| ext.abiCodes = ["x86_64": 1, "armeabi-v7a": 2, "arm64-v8a": 3] |  | ||||||
| import com.android.build.OutputFile |  | ||||||
| android.applicationVariants.all { variant -> |  | ||||||
|     variant.outputs.each { output -> |  | ||||||
|         def abiVersionCode = project.ext.abiCodes.get(output.getFilter(OutputFile.ABI)) |  | ||||||
|         if (abiVersionCode != null) { |  | ||||||
|             output.versionCodeOverride = variant.versionCode * 10 + abiVersionCode |  | ||||||
|         } else { |  | ||||||
|             output.versionCodeOverride = variant.versionCode * 10 |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										107
									
								
								android/app/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,107 @@ | |||||||
|  | import java.io.FileInputStream | ||||||
|  | import java.util.Properties | ||||||
|  | import com.android.build.api.variant.FilterConfiguration.FilterType.* | ||||||
|  |  | ||||||
|  | plugins { | ||||||
|  |     id("com.android.application") | ||||||
|  |     id("kotlin-android") | ||||||
|  |     // The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins. | ||||||
|  |     id("dev.flutter.flutter-gradle-plugin") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | val localProperties = Properties() | ||||||
|  | val localPropertiesFile = rootProject.file("local.properties") | ||||||
|  | if (localPropertiesFile.exists()) { | ||||||
|  |     localPropertiesFile.reader(Charsets.UTF_8).use { reader -> | ||||||
|  |         localProperties.load(reader) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var flutterVersionCode = localProperties.getProperty("flutter.versionCode") ?: "1" | ||||||
|  | var flutterVersionName = localProperties.getProperty("flutter.versionName") ?: "1.0" | ||||||
|  |  | ||||||
|  | val keystoreProperties = Properties() | ||||||
|  | val keystorePropertiesFile = rootProject.file("key.properties") | ||||||
|  | if (keystorePropertiesFile.exists()) { | ||||||
|  |     keystoreProperties.load(FileInputStream(keystorePropertiesFile)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | android { | ||||||
|  |     namespace = "dev.imranr.obtainium" | ||||||
|  |     compileSdk = flutter.compileSdkVersion | ||||||
|  |     ndkVersion = "27.0.12077973" // 'flutter.ndkVersion' produces warnings (TODO can/should we switch back?) | ||||||
|  |  | ||||||
|  |     compileOptions { | ||||||
|  |         isCoreLibraryDesugaringEnabled = true | ||||||
|  |         sourceCompatibility = JavaVersion.VERSION_11 | ||||||
|  |         targetCompatibility = JavaVersion.VERSION_11 | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     kotlinOptions { | ||||||
|  |         jvmTarget = JavaVersion.VERSION_11.toString() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     defaultConfig { | ||||||
|  |         applicationId = "dev.imranr.obtainium" | ||||||
|  |         // You can update the following values to match your application needs. | ||||||
|  |         // For more information, see: https://flutter.dev/to/review-gradle-config. | ||||||
|  |         minSdk = 24 | ||||||
|  |         targetSdk = flutter.targetSdkVersion | ||||||
|  |         versionCode = flutterVersionCode.toInt() | ||||||
|  |         versionName = flutterVersionName | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     flavorDimensions("flavor") | ||||||
|  |  | ||||||
|  |     productFlavors { | ||||||
|  |         create("normal") { | ||||||
|  |             dimension = "flavor" | ||||||
|  |             applicationIdSuffix = "" | ||||||
|  |         } | ||||||
|  |         create("fdroid") { | ||||||
|  |             dimension = "flavor" | ||||||
|  |             applicationIdSuffix = ".fdroid" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     signingConfigs { | ||||||
|  |         create("release") { | ||||||
|  |             keyAlias = keystoreProperties["keyAlias"].toString() | ||||||
|  |             keyPassword = keystoreProperties["keyPassword"].toString() | ||||||
|  |             storeFile = keystoreProperties["storeFile"]?.let { file(it) } | ||||||
|  |             storePassword = keystoreProperties["storePassword"].toString() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     buildTypes { | ||||||
|  |         getByName("release") { | ||||||
|  |             signingConfig = signingConfigs.getByName("release") | ||||||
|  |         } | ||||||
|  |         getByName("debug") { | ||||||
|  |             applicationIdSuffix = ".debug" | ||||||
|  |             versionNameSuffix = "-debug" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | val abiCodes = mapOf("x86_64" to 1, "armeabi-v7a" to 2, "arm64-v8a" to 3) | ||||||
|  |  | ||||||
|  | androidComponents { | ||||||
|  |     onVariants { variant -> | ||||||
|  |         variant.outputs.forEach { output -> | ||||||
|  |             val name = output.filters.find { it.filterType == ABI }?.identifier | ||||||
|  |             val baseAbiCode = abiCodes[name] | ||||||
|  |             if (baseAbiCode != null) { | ||||||
|  |                 output.versionCode.set(baseAbiCode + ((output.versionCode.get() ?: 0) * 10)) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | dependencies { | ||||||
|  |     coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:2.1.5") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | flutter { | ||||||
|  |     source = "../.." | ||||||
|  | } | ||||||
| @@ -69,6 +69,7 @@ | |||||||
|     <uses-permission android:name="android.permission.WAKE_LOCK" /> |     <uses-permission android:name="android.permission.WAKE_LOCK" /> | ||||||
|     <uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM" /> |     <uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM" /> | ||||||
|     <uses-permission android:name="android.permission.REQUEST_DELETE_PACKAGES" /> |     <uses-permission android:name="android.permission.REQUEST_DELETE_PACKAGES" /> | ||||||
|  |     <uses-permission android:name="android.permission.ENFORCE_UPDATE_OWNERSHIP"/> | ||||||
|     <uses-permission |     <uses-permission | ||||||
|         android:name="android.permission.WRITE_EXTERNAL_STORAGE" |         android:name="android.permission.WRITE_EXTERNAL_STORAGE" | ||||||
|         android:maxSdkVersion="29" /> |         android:maxSdkVersion="29" /> | ||||||
|   | |||||||
| @@ -2,4 +2,4 @@ package dev.imranr.obtainium | |||||||
|  |  | ||||||
| import io.flutter.embedding.android.FlutterActivity | import io.flutter.embedding.android.FlutterActivity | ||||||
|  |  | ||||||
| class MainActivity: FlutterActivity() | class MainActivity : FlutterActivity() | ||||||
|   | |||||||
| @@ -1,22 +0,0 @@ | |||||||
| allprojects { |  | ||||||
|     repositories { |  | ||||||
|         google() |  | ||||||
|         mavenCentral() |  | ||||||
|         maven { |  | ||||||
|             // [required] background_fetch |  | ||||||
|             url "${project(':background_fetch').projectDir}/libs" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| rootProject.buildDir = "../build" |  | ||||||
| subprojects { |  | ||||||
|     project.buildDir = "${rootProject.buildDir}/${project.name}" |  | ||||||
| } |  | ||||||
| subprojects { |  | ||||||
|     project.evaluationDependsOn(":app") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| tasks.register("clean", Delete) { |  | ||||||
|     delete rootProject.buildDir |  | ||||||
| } |  | ||||||
							
								
								
									
										25
									
								
								android/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | |||||||
|  | allprojects { | ||||||
|  |     repositories { | ||||||
|  |         google() | ||||||
|  |         mavenCentral() | ||||||
|  |         maven { | ||||||
|  |             // [required] background_fetch | ||||||
|  |             url = uri("${project(":background_fetch").projectDir}/libs") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | val newBuildDir: Directory = rootProject.layout.buildDirectory.dir("../../build").get() | ||||||
|  | rootProject.layout.buildDirectory.value(newBuildDir) | ||||||
|  |  | ||||||
|  | subprojects { | ||||||
|  |     val newSubprojectBuildDir: Directory = newBuildDir.dir(project.name) | ||||||
|  |     project.layout.buildDirectory.value(newSubprojectBuildDir) | ||||||
|  | } | ||||||
|  | subprojects { | ||||||
|  |     project.evaluationDependsOn(":app") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | tasks.register<Delete>("clean") { | ||||||
|  |     delete(rootProject.layout.buildDirectory) | ||||||
|  | } | ||||||
| @@ -1,3 +1,3 @@ | |||||||
| org.gradle.jvmargs=-Xmx4G -XX:MaxMetaspaceSize=2G -XX:+HeapDumpOnOutOfMemoryError | org.gradle.jvmargs=-Xmx8G -XX:MaxMetaspaceSize=4G -XX:ReservedCodeCacheSize=512m -XX:+HeapDumpOnOutOfMemoryError | ||||||
| android.useAndroidX=true | android.useAndroidX=true | ||||||
| android.enableJetifier=true | android.enableJetifier=true | ||||||
|   | |||||||
| @@ -2,4 +2,4 @@ distributionBase=GRADLE_USER_HOME | |||||||
| distributionPath=wrapper/dists | distributionPath=wrapper/dists | ||||||
| zipStoreBase=GRADLE_USER_HOME | zipStoreBase=GRADLE_USER_HOME | ||||||
| zipStorePath=wrapper/dists | zipStorePath=wrapper/dists | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.3-all.zip | distributionUrl=https\://services.gradle.org/distributions/gradle-8.12-all.zip | ||||||
|   | |||||||
| @@ -1,25 +0,0 @@ | |||||||
| pluginManagement { |  | ||||||
|     def flutterSdkPath = { |  | ||||||
|         def properties = new Properties() |  | ||||||
|         file("local.properties").withInputStream { properties.load(it) } |  | ||||||
|         def flutterSdkPath = properties.getProperty("flutter.sdk") |  | ||||||
|         assert flutterSdkPath != null, "flutter.sdk not set in local.properties" |  | ||||||
|         return flutterSdkPath |  | ||||||
|     }() |  | ||||||
|  |  | ||||||
|     includeBuild("$flutterSdkPath/packages/flutter_tools/gradle") |  | ||||||
|  |  | ||||||
|     repositories { |  | ||||||
|         google() |  | ||||||
|         mavenCentral() |  | ||||||
|         gradlePluginPortal() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| plugins { |  | ||||||
|     id "dev.flutter.flutter-plugin-loader" version "1.0.0" |  | ||||||
|     id "com.android.application" version "8.1.0" apply false |  | ||||||
|     id "org.jetbrains.kotlin.android" version "1.8.22" apply false |  | ||||||
| } |  | ||||||
|  |  | ||||||
| include ":app" |  | ||||||
							
								
								
									
										25
									
								
								android/settings.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | |||||||
|  | pluginManagement { | ||||||
|  |     val flutterSdkPath = run { | ||||||
|  |         val properties = java.util.Properties() | ||||||
|  |         file("local.properties").inputStream().use { properties.load(it) } | ||||||
|  |         val flutterSdkPath = properties.getProperty("flutter.sdk") | ||||||
|  |         require(flutterSdkPath != null) { "flutter.sdk not set in local.properties" } | ||||||
|  |         flutterSdkPath | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     includeBuild("$flutterSdkPath/packages/flutter_tools/gradle") | ||||||
|  |  | ||||||
|  |     repositories { | ||||||
|  |         google() | ||||||
|  |         mavenCentral() | ||||||
|  |         gradlePluginPortal() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | plugins { | ||||||
|  |     id("dev.flutter.flutter-plugin-loader") version "1.0.0" | ||||||
|  |     id("com.android.application") version "8.7.3" apply false | ||||||
|  |     id("org.jetbrains.kotlin.android") version "2.1.0" apply false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | include(":app") | ||||||
							
								
								
									
										
											BIN
										
									
								
								assets/fonts/Montserrat-Regular.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								assets/graphics/icon-512x512.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 44 KiB | 
| Before Width: | Height: | Size: 234 KiB After Width: | Height: | Size: 346 KiB | 
| Before Width: | Height: | Size: 238 KiB After Width: | Height: | Size: 354 KiB | 
| Before Width: | Height: | Size: 140 KiB After Width: | Height: | Size: 265 KiB | 
| Before Width: | Height: | Size: 139 KiB After Width: | Height: | Size: 227 KiB | 
| Before Width: | Height: | Size: 118 KiB After Width: | Height: | Size: 178 KiB | 
| Before Width: | Height: | Size: 262 KiB After Width: | Height: | Size: 264 KiB | 
							
								
								
									
										393
									
								
								assets/translations/ar.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,393 @@ | |||||||
|  | { | ||||||
|  |     "invalidURLForSource": "رابط تطبيق {} غير صالح", | ||||||
|  |     "noReleaseFound": "تعذر العثور على إصدار مناسب", | ||||||
|  |     "noVersionFound": "تعذر تحديد إصدار التطبيق", | ||||||
|  |     "urlMatchesNoSource": "الرابط لا يتطابق مع مصدر معروف", | ||||||
|  |     "cantInstallOlderVersion": "لا يمكن تثبيت إصدار أقدم من التطبيق", | ||||||
|  |     "appIdMismatch": "معرّف الحزمة المحملة لا يتطابق مع معرّف التطبيق الحالي", | ||||||
|  |     "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": "رابط مصدر التطبيق", | ||||||
|  |     "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": "مشاركة روابط التطبيقات المحددة", | ||||||
|  |     "resetInstallStatus": "إعادة تعيين حالة التثبيت", | ||||||
|  |     "more": "المزيد", | ||||||
|  |     "removeOutdatedFilter": "إزالة فلتر التطبيقات غير المحدثة", | ||||||
|  |     "showOutdatedOnly": "عرض التطبيقات غير المحدثة فقط", | ||||||
|  |     "filter": "تصفية", | ||||||
|  |     "filterApps": "تصفية التطبيقات", | ||||||
|  |     "appName": "اسم التطبيق", | ||||||
|  |     "author": "المؤلف", | ||||||
|  |     "upToDateApps": "تطبيقات محدثة", | ||||||
|  |     "nonInstalledApps": "تطبيقات غير مثبتة", | ||||||
|  |     "importExport": "استيراد/تصدير", | ||||||
|  |     "settings": "الإعدادات", | ||||||
|  |     "exportedTo": "تم التصدير إلى {}", | ||||||
|  |     "obtainiumExport": "تصدير Obtainium", | ||||||
|  |     "invalidInput": "إدخال غير صالح", | ||||||
|  |     "importedX": "تم استيراد {}", | ||||||
|  |     "obtainiumImport": "استيراد Obtainium", | ||||||
|  |     "importFromURLList": "استيراد من قائمة الروابط", | ||||||
|  |     "searchQuery": "استعلام البحث", | ||||||
|  |     "appURLList": "قائمة روابط التطبيقات", | ||||||
|  |     "line": "سطر", | ||||||
|  |     "searchX": "بحث {}", | ||||||
|  |     "noResults": "لا توجد نتائج", | ||||||
|  |     "importX": "استيراد {}", | ||||||
|  |     "importedAppsIdDisclaimer": "قد تظهر التطبيقات المستوردة بشكل غير صحيح كـ \"غير مثبتة\".\nلإصلاح هذا، قم بإعادة تثبيتها عبر Obtainium.\nهذا لا يؤثر على بيانات التطبيق.\n\nينطبق فقط على طرق الاستيراد عبر الروابط أو الطرف الثالث.", | ||||||
|  |     "importErrors": "أخطاء الاستيراد", | ||||||
|  |     "importedXOfYApps": "تم استيراد {} من {} تطبيقات.", | ||||||
|  |     "followingURLsHadErrors": "حدثت أخطاء في الروابط التالية:", | ||||||
|  |     "selectURL": "اختر رابطًا", | ||||||
|  |     "selectURLs": "اختر روابط", | ||||||
|  |     "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", | ||||||
|  |     "pickAnAPK": "اختر ملف APK", | ||||||
|  |     "appHasMoreThanOnePackage": "{} يحتوي على أكثر من حزمة:", | ||||||
|  |     "deviceSupportsXArch": "جهازك يدعم بنية المعالج {}.", | ||||||
|  |     "deviceSupportsFollowingArchs": "جهازك يدعم بنى المعالج التالية:", | ||||||
|  |     "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": "معرّف التطبيق أو الاسم", | ||||||
|  |     "appId": "معرّف التطبيق", | ||||||
|  |     "appWithIdOrNameNotFound": "لم يتم العثور على تطبيق بهذا المعرّف أو الاسم", | ||||||
|  |     "reposHaveMultipleApps": "قد تحتوي المستودعات على عدة تطبيقات", | ||||||
|  |     "fdroidThirdPartyRepo": "مستودع F-Droid طرف ثالث", | ||||||
|  |     "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": "استخدام تاريخ الإصدار كسلسلة إصدار", | ||||||
|  |     "releaseTitleAsVersion": "استخدام عنوان الإصدار كسلسلة إصدار", | ||||||
|  |     "releaseDateAsVersionExplanation": "يجب استخدام هذا الخيار فقط للتطبيقات التي لا يعمل فيها اكتشاف الإصدار بشكل صحيح، ولكن تاريخ الإصدار متاح.", | ||||||
|  |     "changes": "التغييرات", | ||||||
|  |     "releaseDate": "تاريخ الإصدار", | ||||||
|  |     "importFromURLsInFile": "استيراد من روابط في ملف (مثل OPML)", | ||||||
|  |     "versionDetectionExplanation": "تنسيق سلسلة الإصدار مع الإصدار المكتشف من النظام", | ||||||
|  |     "versionDetection": "اكتشاف الإصدار", | ||||||
|  |     "standardVersionDetection": "اكتشاف الإصدار القياسي", | ||||||
|  |     "groupByCategory": "التجميع حسب التصنيف", | ||||||
|  |     "autoApkFilterByArch": "محاولة تصفية ملفات APK حسب بنية المعالج إذا أمكن", | ||||||
|  |     "autoLinkFilterByArch": "محاولة تصفية الروابط حسب بنية المعالج إذا أمكن", | ||||||
|  |     "overrideSource": "تجاوز المصدر", | ||||||
|  |     "dontShowAgain": "عدم إظهار هذا مرة أخرى", | ||||||
|  |     "dontShowTrackOnlyWarnings": "عدم إظهار تحذيرات 'للتعقب فقط'", | ||||||
|  |     "dontShowAPKOriginWarnings": "عدم إظهار تحذيرات مصدر APK", | ||||||
|  |     "moveNonInstalledAppsToBottom": "نقل التطبيقات غير المثبتة إلى أسفل قائمة التطبيقات", | ||||||
|  |     "gitlabPATLabel": "رمز الوصول الشخصي لـ GitLab", | ||||||
|  |     "about": "حول", | ||||||
|  |     "requiresCredentialsInSettings": "{} يحتاج إلى بيانات اعتماد إضافية (في الإعدادات)", | ||||||
|  |     "checkOnStart": "التحقق من التحديثات عند البدء", | ||||||
|  |     "tryInferAppIdFromCode": "محاولة استنتاج معرّف التطبيق من الكود المصدري", | ||||||
|  |     "removeOnExternalUninstall": "إزالة التطبيقات المثبتة خارجيًا تلقائيًا", | ||||||
|  |     "pickHighestVersionCode": "تحديد أعلى إصدار تلقائيًا", | ||||||
|  |     "checkUpdateOnDetailPage": "التحقق من التحديثات عند فتح صفحة التفاصيل", | ||||||
|  |     "disablePageTransitions": "تعطيل انتقالات الصفحات", | ||||||
|  |     "reversePageTransitions": "عكس انتقالات الصفحات", | ||||||
|  |     "minStarCount": "الحد الأدنى لعدد النجوم", | ||||||
|  |     "addInfoBelow": "أضف هذه المعلومات أدناه.", | ||||||
|  |     "addInfoInSettings": "أضف هذه المعلومات في الإعدادات.", | ||||||
|  |     "githubSourceNote": "يمكن تجنب الحد المسموح لـ GitHub باستخدام مفتاح API.", | ||||||
|  |     "sortByLastLinkSegment": "الفرز حسب الجزء الأخير من الرابط فقط", | ||||||
|  |     "filterReleaseNotesByRegEx": "تصفية ملاحظات الإصدار باستخدام تعبير نمطي", | ||||||
|  |     "customLinkFilterRegex": "تصفية روابط APK المخصصة باستخدام تعبير نمطي (افتراضيًا '.apk$')", | ||||||
|  |     "appsPossiblyUpdated": "تمت محاولة تحديث التطبيقات", | ||||||
|  |     "appsPossiblyUpdatedNotifDescription": "يخطر المستخدم بأن تحديثات لواحد أو أكثر من التطبيقات قد تم تطبيقها في الخلفية", | ||||||
|  |     "xWasPossiblyUpdatedToY": "{} قد تم تحديثه إلى {}.", | ||||||
|  |     "enableBackgroundUpdates": "تمكين التحديثات في الخلفية", | ||||||
|  |     "backgroundUpdateReqsExplanation": "قد لا تكون التحديثات في الخلفية ممكنة لجميع التطبيقات.", | ||||||
|  |     "backgroundUpdateLimitsExplanation": "يمكن تحديد نجاح التثبيت في الخلفية فقط عند فتح Obtainium.", | ||||||
|  |     "verifyLatestTag": "التحقق من علامة 'الأحدث'", | ||||||
|  |     "intermediateLinkRegex": "تصفية للوصول إلى رابط 'وسيط'", | ||||||
|  |     "filterByLinkText": "تصفية الروابط حسب نص الرابط", | ||||||
|  |     "intermediateLinkNotFound": "لم يتم العثور على رابط وسيط", | ||||||
|  |     "intermediateLink": "رابط وسيط", | ||||||
|  |     "exemptFromBackgroundUpdates": "إعفاء من التحديثات في الخلفية (إذا تم تمكينها)", | ||||||
|  |     "bgUpdatesOnWiFiOnly": "تعطيل التحديثات في الخلفية عند عدم الاتصال بشبكة Wi-Fi", | ||||||
|  |     "bgUpdatesWhileChargingOnly": "تعطيل التحديثات في الخلفية عند عدم الشحن", | ||||||
|  |     "autoSelectHighestVersionCode": "تحديد أعلى إصدار تلقائيًا", | ||||||
|  |     "versionExtractionRegEx": "تعبير نمطي لاستخراج الإصدار", | ||||||
|  |     "trimVersionString": "قص سلسلة الإصدار باستخدام تعبير نمطي", | ||||||
|  |     "matchGroupToUseForX": "مجموعة المطابقة لاستخدامها لـ \"{}\"", | ||||||
|  |     "matchGroupToUse": "مجموعة المطابقة لاستخدامها في تعبير استخراج الإصدار", | ||||||
|  |     "highlightTouchTargets": "تمييز أهداف اللمس الأقل وضوحًا", | ||||||
|  |     "pickExportDir": "اختر دليل التصدير", | ||||||
|  |     "autoExportOnChanges": "تصدير تلقائي عند التغييرات", | ||||||
|  |     "includeSettings": "تضمين الإعدادات", | ||||||
|  |     "filterVersionsByRegEx": "تصفية الإصدارات باستخدام تعبير نمطي", | ||||||
|  |     "trySelectingSuggestedVersionCode": "محاولة تحديد الإصدار المقترح تلقائيًا", | ||||||
|  |     "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 يعمل على إصدار أندرويد < 8.1 مع ADB - قم بتحديث أندرويد أو استخدم Sui بدلاً من ذلك", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "تعيين Google Play كمصدر التثبيت (إذا تم استخدام Shizuku)", | ||||||
|  |     "useSystemFont": "استخدم خط النظام", | ||||||
|  |     "useVersionCodeAsOSVersion": "استخدم versionCode كإصدار مكتشف من النظام", | ||||||
|  |     "requestHeader": "رأس الطلب", | ||||||
|  |     "useLatestAssetDateAsReleaseDate": "استخدام أحدث تاريخ تحميل كتاريخ إصدار", | ||||||
|  |     "defaultPseudoVersioningMethod": "طريقة الإصدار الوهمي الافتراضية", | ||||||
|  |     "partialAPKHash": "جزء من تجزئة APK", | ||||||
|  |     "APKLinkHash": "تجزئة رابط APK", | ||||||
|  |     "directAPKLink": "رابط APK مباشر", | ||||||
|  |     "pseudoVersionInUse": "يتم استخدام إصدار وهمي", | ||||||
|  |     "installed": "مثبت", | ||||||
|  |     "latest": "الأحدث", | ||||||
|  |     "invertRegEx": "عكس التعبير النمطي", | ||||||
|  |     "note": "ملاحظة", | ||||||
|  |     "selfHostedNote": "يمكن استخدام القائمة المنسدلة \"{}\" للوصول إلى نسخ مخصصة/مستضافة ذاتيًا لأي مصدر.", | ||||||
|  |     "badDownload": "تعذر تحليل ملف APK (تنزيل غير متوافق أو غير مكتمل)", | ||||||
|  |     "beforeNewInstallsShareToAppVerifier": "مشاركة التطبيقات الجديدة مع AppVerifier (إذا كان متاحًا)", | ||||||
|  |     "appVerifierInstructionToast": "شارك مع AppVerifier، ثم عد هنا عندما تكون جاهزًا.", | ||||||
|  |     "wiki": "المساعدة/ويكي", | ||||||
|  |     "crowdsourcedConfigsLabel": "تكوينات التطبيقات المقدمة من الجمهور (استخدم على مسؤوليتك)", | ||||||
|  |     "crowdsourcedConfigsShort": "تكوينات التطبيقات المقدمة من الجمهور", | ||||||
|  |     "allowInsecure": "السماح بطلبات HTTP غير الآمنة", | ||||||
|  |     "stayOneVersionBehind": "البقاء بإصدار واحد خلف الأحدث", | ||||||
|  |     "useFirstApkOfVersion": "التحديد التلقائي الأول من بين عدة ملفات APK", | ||||||
|  |     "refreshBeforeDownload": "تحديث تفاصيل التطبيق قبل التنزيل", | ||||||
|  |     "tencentAppStore": "متجر Tencent للتطبيقات", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "متجر vivo للتطبيقات (CN)", | ||||||
|  |     "name": "الاسم", | ||||||
|  |     "smartname": "الاسم (ذكي)", | ||||||
|  |     "sortMethod": "طريقة الفرز", | ||||||
|  |     "welcome": "مرحبًا", | ||||||
|  |     "documentationLinksNote": "تحتوي صفحة Obtainium على GitHub المرتبطة أدناه على روابط لمقاطع فيديو، مقالات، مناقشات وموارد أخرى ستساعدك على فهم كيفية استخدام التطبيق.", | ||||||
|  |     "batteryOptimizationNote": "لاحظ أن التنزيلات في الخلفية قد تعمل بشكل أكثر موثوقية إذا قمت بتعطيل تحسينات بطارية النظام لـ Obtainium.", | ||||||
|  |     "fileDeletionError": "فشل حذف الملف (حاول حذفه يدويًا ثم حاول مرة أخرى): \"{}\"", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "إزالة التطبيق؟", | ||||||
|  |         "other": "إزالة التطبيقات؟" | ||||||
|  |     }, | ||||||
|  |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|  |         "one": "طلبات كثيرة جدًا (تم تحديد المعدل) - حاول مرة أخرى بعد {} دقيقة", | ||||||
|  |         "other": "طلبات كثيرة جدًا (تم تحديد المعدل) - حاول مرة أخرى بعد {} دقائق" | ||||||
|  |     }, | ||||||
|  |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|  |         "one": "واجه التحقق من التحديثات في الخلفية {}، سيتم جدولة محاولة أخرى بعد {} دقيقة", | ||||||
|  |         "other": "واجه التحقق من التحديثات في الخلفية {}، سيتم جدولة محاولة أخرى بعد {} دقائق" | ||||||
|  |     }, | ||||||
|  |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|  |         "one": "وجد التحقق من التحديثات في الخلفية {} تحديث - سيتم إخطار المستخدم إذا لزم الأمر", | ||||||
|  |         "other": "وجد التحقق من التحديثات في الخلفية {} تحديثات - سيتم إخطار المستخدم إذا لزم الأمر" | ||||||
|  |     }, | ||||||
|  |     "apps": { | ||||||
|  |         "one": "{} تطبيق", | ||||||
|  |         "other": "{} تطبيقات" | ||||||
|  |     }, | ||||||
|  |     "url": { | ||||||
|  |         "one": "{} رابط", | ||||||
|  |         "other": "{} روابط" | ||||||
|  |     }, | ||||||
|  |     "minute": { | ||||||
|  |         "one": "{} دقيقة", | ||||||
|  |         "other": "{} دقائق" | ||||||
|  |     }, | ||||||
|  |     "hour": { | ||||||
|  |         "one": "{} ساعة", | ||||||
|  |         "other": "{} ساعات" | ||||||
|  |     }, | ||||||
|  |     "day": { | ||||||
|  |         "one": "{} يوم", | ||||||
|  |         "other": "{} أيام" | ||||||
|  |     }, | ||||||
|  |     "clearedNLogsBeforeXAfterY": { | ||||||
|  |         "one": "تم مسح {n} سجل (قبل = {before}, بعد = {after})", | ||||||
|  |         "other": "تم مسح {n} سجلات (قبل = {before}, بعد = {after})" | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesAvailable": { | ||||||
|  |         "one": "{} وتطبيق واحد آخر لديه تحديثات.", | ||||||
|  |         "other": "{} و{} تطبيقات أخرى لديها تحديثات." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesInstalled": { | ||||||
|  |         "one": "{} وتطبيق واحد آخر تم تحديثه.", | ||||||
|  |         "other": "{} و{} تطبيقات أخرى تم تحديثها." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "فشل تحديث {} وتطبيق واحد آخر.", | ||||||
|  |         "other": "فشل تحديث {} و{} تطبيقات أخرى." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|  |         "one": "{} وتطبيق واحد آخر قد تم تحديثه.", | ||||||
|  |         "other": "{} و{} تطبيقات أخرى قد تم تحديثها." | ||||||
|  |     }, | ||||||
|  |     "apk": { | ||||||
|  |         "one": "{} APK", | ||||||
|  |         "other": "{} APKs" | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Detekcija standardne verzije", |     "standardVersionDetection": "Detekcija standardne verzije", | ||||||
|     "groupByCategory": "Grupiši po kategoriji", |     "groupByCategory": "Grupiši po kategoriji", | ||||||
|     "autoApkFilterByArch": "Pokušajte filtrirati APK-ove po arhitekturi procesora ako je moguće", |     "autoApkFilterByArch": "Pokušajte filtrirati APK-ove po arhitekturi procesora ako je moguće", | ||||||
|  |     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||||
|     "overrideSource": "Premosti izvor", |     "overrideSource": "Premosti izvor", | ||||||
|     "dontShowAgain": "Ne prikazuj ovo ponovo", |     "dontShowAgain": "Ne prikazuj ovo ponovo", | ||||||
|     "dontShowTrackOnlyWarnings": "Ne prikazuj upozorenja „Samo za  praćenje”", |     "dontShowTrackOnlyWarnings": "Ne prikazuj upozorenja „Samo za  praćenje”", | ||||||
| @@ -255,7 +256,7 @@ | |||||||
|     "intermediateLinkNotFound": "Intermediate veza nije nađena", |     "intermediateLinkNotFound": "Intermediate veza nije nađena", | ||||||
|     "intermediateLink": "Intermediate veza", |     "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 Wi-Fi-ju", | ||||||
|     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", |     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", | ||||||
|     "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", | ||||||
| @@ -314,11 +315,21 @@ | |||||||
|     "appVerifierInstructionToast": "Dijeli sa AppVerifier-om, zatim se vratite kada ste spremni.", |     "appVerifierInstructionToast": "Dijeli sa AppVerifier-om, zatim se vratite kada ste spremni.", | ||||||
|     "wiki": "Pomoć/Wiki", |     "wiki": "Pomoć/Wiki", | ||||||
|     "crowdsourcedConfigsLabel": "Konfiguracije aplikacije obezbeđene pomoću velikog broja ljudi (crowdsourcing) (koristite na svoju odgovornost)", |     "crowdsourcedConfigsLabel": "Konfiguracije aplikacije obezbeđene pomoću velikog broja ljudi (crowdsourcing) (koristite na svoju odgovornost)", | ||||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", |     "crowdsourcedConfigsShort": "Crowdsourced App Configurations", | ||||||
|     "allowInsecure": "Allow insecure HTTP requests", |     "allowInsecure": "Allow insecure HTTP requests", | ||||||
|     "stayOneVersionBehind": "Stay one version behind latest", |     "stayOneVersionBehind": "Stay one version behind latest", | ||||||
|  |     "useFirstApkOfVersion": "Auto-select first of multiple APKs", | ||||||
|     "refreshBeforeDownload": "Refresh app details before download", |     "refreshBeforeDownload": "Refresh app details before download", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Name", | ||||||
|  |     "smartname": "Name (Smart)", | ||||||
|  |     "sortMethod": "Sort Method", | ||||||
|  |     "welcome": "Welcome", | ||||||
|  |     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||||
|  |     "batteryOptimizationNote": "Note that background downloads may work more reliably if you disable OS battery optimizations for Obtainium.", | ||||||
|  |     "fileDeletionError": "Failed to delete file (try deleting it manually then try again): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Želite li ukloniti aplikaciju?", |         "one": "Želite li ukloniti aplikaciju?", | ||||||
|         "other": "Želite li ukloniti aplikacije?" |         "other": "Želite li ukloniti aplikacije?" | ||||||
|   | |||||||
							
								
								
									
										393
									
								
								assets/translations/ca.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,393 @@ | |||||||
|  | { | ||||||
|  |     "invalidURLForSource": "L'URL de l'aplicació {} no es vàlid", | ||||||
|  |     "noReleaseFound": "No s'ha pogut trobar una versió adequada", | ||||||
|  |     "noVersionFound": "No s'ha pogut determinar la versió", | ||||||
|  |     "urlMatchesNoSource": "L'URL no coincideix amb cap font coneguda", | ||||||
|  |     "cantInstallOlderVersion": "No és possible instal·lar una versió més antiga de l'aplicació", | ||||||
|  |     "appIdMismatch": "L'ID del paquet descarregat no coincideix amb l'ID de l'aplicació instal·lada", | ||||||
|  |     "functionNotImplemented": "Aquesta classe no ha implentat aquesta funció", | ||||||
|  |     "placeholder": "Espai reservat", | ||||||
|  |     "someErrors": "S'han produït alguns errors", | ||||||
|  |     "unexpectedError": "Error inesperat", | ||||||
|  |     "ok": "Accepta", | ||||||
|  |     "and": "i", | ||||||
|  |     "githubPATLabel": "Token d'accés personal a GitHub (augmenta el límit d'accés)", | ||||||
|  |     "includePrereleases": "Inclou les versions preliminars", | ||||||
|  |     "fallbackToOlderReleases": "Torna a les versions anteriors", | ||||||
|  |     "filterReleaseTitlesByRegEx": "Filtra el títol de la versió per una expressió regular", | ||||||
|  |     "invalidRegEx": "Expressió regular invàlida", | ||||||
|  |     "noDescription": "Sense descripció", | ||||||
|  |     "cancel": "Cancel·la", | ||||||
|  |     "continue": "Continua", | ||||||
|  |     "requiredInBrackets": "(requerit)", | ||||||
|  |     "dropdownNoOptsError": "ERROR: EL DESPLEGABLE HA DE TENIR ALMENYS UNA OPCIÓ", | ||||||
|  |     "colour": "Color", | ||||||
|  |     "standard": "Estàndard", | ||||||
|  |     "custom": "Personalitzat", | ||||||
|  |     "useMaterialYou": "Usa 'Material You'", | ||||||
|  |     "githubStarredRepos": "Repositoris favorits de GitHub", | ||||||
|  |     "uname": "Nom d'usuari", | ||||||
|  |     "wrongArgNum": "Nombre d'arguments proveït invàlid", | ||||||
|  |     "xIsTrackOnly": "{} és només per a seguiment", | ||||||
|  |     "source": "Font", | ||||||
|  |     "app": "Aplicació", | ||||||
|  |     "appsFromSourceAreTrackOnly": "Les aplicacions d'aquesta font són només per a seguiment.", | ||||||
|  |     "youPickedTrackOnly": "Has seleccionat l'opció 'només per a seguiment'.", | ||||||
|  |     "trackOnlyAppDescription": "Farem el seguiment de les actualitzacions per a l'aplicació, però Obtainium no podrà descarregar-la ni actualitzar-la.", | ||||||
|  |     "cancelled": "Cancel·lat", | ||||||
|  |     "appAlreadyAdded": "Aplicació ja afegida", | ||||||
|  |     "alreadyUpToDateQuestion": "Aplicació ja actualitzada?", | ||||||
|  |     "addApp": "Afegeix l'aplicació", | ||||||
|  |     "appSourceURL": "URL font de l'aplicació", | ||||||
|  |     "error": "Error", | ||||||
|  |     "add": "Afegeix", | ||||||
|  |     "searchSomeSourcesLabel": "Cerca (només algunes fonts)", | ||||||
|  |     "search": "Cerca", | ||||||
|  |     "additionalOptsFor": "Opcions addicionals per a {}", | ||||||
|  |     "supportedSources": "Fonts suportades", | ||||||
|  |     "trackOnlyInBrackets": "(només per a seguiment)", | ||||||
|  |     "searchableInBrackets": "(permet la cerca)", | ||||||
|  |     "appsString": "Aplicacions", | ||||||
|  |     "noApps": "No hi ha aplicacions", | ||||||
|  |     "noAppsForFilter": "No hi ha aplicacions per filtrar", | ||||||
|  |     "byX": "Per: {}", | ||||||
|  |     "percentProgress": "Progrés: {} %", | ||||||
|  |     "pleaseWait": "Espera...", | ||||||
|  |     "updateAvailable": "Actualització disponible", | ||||||
|  |     "notInstalled": "No instal·lat", | ||||||
|  |     "pseudoVersion": "pseudo-versió", | ||||||
|  |     "selectAll": "Selecciona-ho tot", | ||||||
|  |     "deselectX": "Desselecciona {}", | ||||||
|  |     "xWillBeRemovedButRemainInstalled": "{} s'eliminarà d'Obtainium però romandrà instal·lada al dispositiu.", | ||||||
|  |     "removeSelectedAppsQuestion": "Elimino les aplicacions seleccionades?", | ||||||
|  |     "removeSelectedApps": "Elimina les aplicacions seleccionades", | ||||||
|  |     "updateX": "Actualitza {}", | ||||||
|  |     "installX": "Instal·la {}", | ||||||
|  |     "markXTrackOnlyAsUpdated": "Marca {}\n(només per a seguiment)\ncom a actualitzada", | ||||||
|  |     "changeX": "Canvia {}", | ||||||
|  |     "installUpdateApps": "Instal·la/actualitza les aplicacions", | ||||||
|  |     "installUpdateSelectedApps": "Instal·la/actualitza les aplicacions seleccionades", | ||||||
|  |     "markXSelectedAppsAsUpdated": "Marco {} les aplicaciones seleccionades com a actualitzades?", | ||||||
|  |     "no": "No", | ||||||
|  |     "yes": "Sí", | ||||||
|  |     "markSelectedAppsUpdated": "Marca les aplicacions seleccionades com a actualitzades", | ||||||
|  |     "pinToTop": "Ancora-la al capdamunt", | ||||||
|  |     "unpinFromTop": "Desancora-la del capdamunt", | ||||||
|  |     "resetInstallStatusForSelectedAppsQuestion": "Restableixo l'estat d'instal·lació per a les aplicacions seleccionades?", | ||||||
|  |     "installStatusOfXWillBeResetExplanation": "Es restablirà l'estat d'instal·lació de les aplicacions seleccionades.\n\nAçò pot ser útil quan la versió de l'aplicació mostrada per Obtainium és incorrecta a conseqüència d'una actualització no reeixida o d'algun altre problema.", | ||||||
|  |     "customLinkMessage": "Aquests enllaços funcionen en dispositius amb Obtainium instal·lat", | ||||||
|  |     "shareAppConfigLinks": "Comparteix la configuració de l'aplicació com a enllaç HTML", | ||||||
|  |     "shareSelectedAppURLs": "Comparteix els URL de les aplicacions seleccionades", | ||||||
|  |     "resetInstallStatus": "Restableix l'estat de la instal·lació", | ||||||
|  |     "more": "Més", | ||||||
|  |     "removeOutdatedFilter": "Elimina el filtre d'aplicacions desactualitzades", | ||||||
|  |     "showOutdatedOnly": "Mostra només les aplicacions desactualitzades", | ||||||
|  |     "filter": "Filtra", | ||||||
|  |     "filterApps": "Filtra les aplicacions", | ||||||
|  |     "appName": "Nom de l'aplicació", | ||||||
|  |     "author": "Autor", | ||||||
|  |     "upToDateApps": "Aplicacions actualizades", | ||||||
|  |     "nonInstalledApps": "Aplicacions no instal·lades", | ||||||
|  |     "importExport": "Importa/exporta", | ||||||
|  |     "settings": "Paràmetres", | ||||||
|  |     "exportedTo": "Exportat a {}", | ||||||
|  |     "obtainiumExport": "Exporta Obtainium", | ||||||
|  |     "invalidInput": "Entrada no vàlida", | ||||||
|  |     "importedX": "Importat {}", | ||||||
|  |     "obtainiumImport": "Importa Obtainium", | ||||||
|  |     "importFromURLList": "Importa des de la llista d'URL", | ||||||
|  |     "searchQuery": "Terme de cerca", | ||||||
|  |     "appURLList": "Llista d'URL d'aplicacions", | ||||||
|  |     "line": "Línia", | ||||||
|  |     "searchX": "Cerca {}", | ||||||
|  |     "noResults": "No hi ha resultats", | ||||||
|  |     "importX": "Importa des de {}", | ||||||
|  |     "importedAppsIdDisclaimer": "Les aplicacions importades podrien mostrar-se incorrectament com a «No instal·lada».\nPer solventar-ho reinstal·la-les a través d'Obtainium.\nAixò no hauria d'afectar les dades de les aplicacions.\n\nNomés afecta els URL i els mètodes d'importació de tercers.", | ||||||
|  |     "importErrors": "Errors d'importació", | ||||||
|  |     "importedXOfYApps": "{} de {} aplicacions importades.", | ||||||
|  |     "followingURLsHadErrors": "Els següents URLs han tingut problemes:", | ||||||
|  |     "selectURL": "Selecciona URL", | ||||||
|  |     "selectURLs": "Selecciona URLs", | ||||||
|  |     "pick": "Escull", | ||||||
|  |     "theme": "Tema", | ||||||
|  |     "dark": "Fosc", | ||||||
|  |     "light": "Clar", | ||||||
|  |     "followSystem": "Segueix el sistema", | ||||||
|  |     "followSystemThemeExplanation": "Seguir el tema del sistema només és possible si uses aplicacions de tercers", | ||||||
|  |     "useBlackTheme": "Fes servir el negre pur en el tema fosc", | ||||||
|  |     "appSortBy": "Ordena les aplicacions per", | ||||||
|  |     "authorName": "Autor/nom", | ||||||
|  |     "nameAuthor": "Nom/Autor", | ||||||
|  |     "asAdded": "Per l'ordre en què es van afegir", | ||||||
|  |     "appSortOrder": "Per ordre de classificació", | ||||||
|  |     "ascending": "Ascendent", | ||||||
|  |     "descending": "Descendent", | ||||||
|  |     "bgUpdateCheckInterval": "Comprova les actualitzacions en segon pla", | ||||||
|  |     "neverManualOnly": "Mai, només manual", | ||||||
|  |     "appearance": "Aparença", | ||||||
|  |     "showWebInAppView": "Mostra el web d'origen en la vista de l'aplicació", | ||||||
|  |     "pinUpdates": "Ancora les actualitzacions al capdamunt de les aplicacions", | ||||||
|  |     "updates": "Actualitzacions", | ||||||
|  |     "sourceSpecific": "Font específica", | ||||||
|  |     "appSource": "Font de l'aplicació", | ||||||
|  |     "noLogs": "Cap registre", | ||||||
|  |     "appLogs": "Registres de l'aplicació", | ||||||
|  |     "close": "Tanca", | ||||||
|  |     "share": "Comparteix", | ||||||
|  |     "appNotFound": "No s'ha trobat l'aplicació", | ||||||
|  |     "obtainiumExportHyphenatedLowercase": "Exportació d'Obtainium", | ||||||
|  |     "pickAnAPK": "Escull una APK", | ||||||
|  |     "appHasMoreThanOnePackage": "{} té més d'un paquet:", | ||||||
|  |     "deviceSupportsXArch": "Aquest dispositiu admet l'aquitectura de CPU: {}.", | ||||||
|  |     "deviceSupportsFollowingArchs": "Aquest dispositiu admet les següents arquitectures de CPU:", | ||||||
|  |     "warning": "Avís", | ||||||
|  |     "sourceIsXButPackageFromYPrompt": "La font de l'aplicació és «{}» però el paquet de l'actualització ve de «{}». Vols continuar?", | ||||||
|  |     "updatesAvailable": "Actualitzacions disponibles", | ||||||
|  |     "updatesAvailableNotifDescription": "Notifica l'usuari que hi ha actualitzacions per a una o més aplicacions seguides per Obtainium", | ||||||
|  |     "noNewUpdates": "No hi ha noves actualitzacions.", | ||||||
|  |     "xHasAnUpdate": "{} té una actualització.", | ||||||
|  |     "appsUpdated": "Aplicacions actualitzades", | ||||||
|  |     "appsNotUpdated": "Error en actualitzar les aplicacions", | ||||||
|  |     "appsUpdatedNotifDescription": "Notifica l'usuari que una o més aplicacions s'han actualitzat en segon pla", | ||||||
|  |     "xWasUpdatedToY": "{} s'ha actualitzat a {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Error en actualitzar {} a {}.", | ||||||
|  |     "errorCheckingUpdates": "Error en cercar actualitzacions", | ||||||
|  |     "errorCheckingUpdatesNotifDescription": "Una notificació que es mostra quan la comprovació d'actualizacions en segon pla ha fallat", | ||||||
|  |     "appsRemoved": "Aplicacions suprimides", | ||||||
|  |     "appsRemovedNotifDescription": "Notifica l'usuari que una o més aplicacions s'han suprimit per errors en carregar-les", | ||||||
|  |     "xWasRemovedDueToErrorY": "{} s'ha suprimit per aquest error: {}", | ||||||
|  |     "completeAppInstallation": "Instal·lació completa de l'aplicació", | ||||||
|  |     "obtainiumMustBeOpenToInstallApps": "Obtainium ha d'estar obert per poder instal·lar aplicacions", | ||||||
|  |     "completeAppInstallationNotifDescription": "Demana l'usuari de tornar a Obtainium per acabar d'instal·lar una aplicació", | ||||||
|  |     "checkingForUpdates": "S'estan cercant actualitzacions...", | ||||||
|  |     "checkingForUpdatesNotifDescription": "Notificació temporal que apareix en cercar actualitzacions", | ||||||
|  |     "pleaseAllowInstallPerm": "Permet que Obtainium instal·li aplicacions", | ||||||
|  |     "trackOnly": "Només per a seguiment", | ||||||
|  |     "errorWithHttpStatusCode": "Error {}", | ||||||
|  |     "versionCorrectionDisabled": "Correcció de versions desactivada (el plugin sembla que no funciona)", | ||||||
|  |     "unknown": "Desconegut", | ||||||
|  |     "none": "Cap", | ||||||
|  |     "never": "Mai", | ||||||
|  |     "latestVersionX": "Última versió: {}", | ||||||
|  |     "installedVersionX": "Versió instal·lada: {}", | ||||||
|  |     "lastUpdateCheckX": "Última comprovació d'actualització: {}", | ||||||
|  |     "remove": "Suprimeix", | ||||||
|  |     "yesMarkUpdated": "Sí, marca com a actualitzada", | ||||||
|  |     "fdroid": "Repositori oficial F-Droid", | ||||||
|  |     "appIdOrName": "ID o nom de l'aplicació", | ||||||
|  |     "appId": "ID de l'aplicació", | ||||||
|  |     "appWithIdOrNameNotFound": "No s'han trobat aplicacions amb aquest ID o nom", | ||||||
|  |     "reposHaveMultipleApps": "Els repositoris poden contenir diverses aplicacions", | ||||||
|  |     "fdroidThirdPartyRepo": "Repositori F-Droid de tercers", | ||||||
|  |     "install": "Instal·la", | ||||||
|  |     "markInstalled": "Marca com a instal·lada", | ||||||
|  |     "update": "Actualitza", | ||||||
|  |     "markUpdated": "Marca com a actualitzada", | ||||||
|  |     "additionalOptions": "Opcions addicionals", | ||||||
|  |     "disableVersionDetection": "Desactiva la detecció de versions", | ||||||
|  |     "noVersionDetectionExplanation": "Només has d'usar aquesta opció en les aplicacions en què la detecció de versions no funcioni correctament.", | ||||||
|  |     "downloadingX": "Descarregant {}", | ||||||
|  |     "downloadX": "Descarrega {}", | ||||||
|  |     "downloadedX": "Descarregada {}", | ||||||
|  |     "releaseAsset": "Recurs actualitzat", | ||||||
|  |     "downloadNotifDescription": "Notifica l'usuari del progrés de la descàrrega d'una aplicació", | ||||||
|  |     "noAPKFound": "No s'ha trobat l'APK", | ||||||
|  |     "noVersionDetection": "No s'han detectat versions", | ||||||
|  |     "categorize": "Categoritza", | ||||||
|  |     "categories": "Categories", | ||||||
|  |     "category": "Categoria", | ||||||
|  |     "noCategory": "No hi ha la categoria", | ||||||
|  |     "noCategories": "No hi ha les categories", | ||||||
|  |     "deleteCategoriesQuestion": "Suprimeixo les categories?", | ||||||
|  |     "categoryDeleteWarning": "Totes les aplicacions de les categories suprimides es marcaran com a no categoritzades.", | ||||||
|  |     "addCategory": "Afegeix una categoria", | ||||||
|  |     "label": "Nom", | ||||||
|  |     "language": "Idioma", | ||||||
|  |     "copiedToClipboard": "Copiat al porta-retalls", | ||||||
|  |     "storagePermissionDenied": "Permís d'emmagatzematge denegat", | ||||||
|  |     "selectedCategorizeWarning": "Açò substituirà els paràmetres de categorització per a les aplicacions selecionades.", | ||||||
|  |     "filterAPKsByRegEx": "Filtra les APKs per l'expressió regular", | ||||||
|  |     "removeFromObtainium": "Suprimeix d'Obtainium", | ||||||
|  |     "uninstallFromDevice": "Desinstal·la del dispositiu", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "Només funciona per a aplicacions amb la detecció de versions desactivada.", | ||||||
|  |     "releaseDateAsVersion": "Usa la data de llançament com a cadena de la versió", | ||||||
|  |     "releaseTitleAsVersion": "Usa el títol com a cadena de la versió", | ||||||
|  |     "releaseDateAsVersionExplanation": "Aquest opció només s'hauria d'usar per a aplicacions en què la detecció de la versió no funciona correctament però disposem de la data de publicació.", | ||||||
|  |     "changes": "Canvis", | ||||||
|  |     "releaseDate": "Data de publicació", | ||||||
|  |     "importFromURLsInFile": "Importa els URLs des d'un fitxer (com ara OPML)", | ||||||
|  |     "versionDetectionExplanation": "Concilia la cadena de la versió amb la versió detectada del Sistema Operatiu", | ||||||
|  |     "versionDetection": "Detecció de la versió", | ||||||
|  |     "standardVersionDetection": "Detecció de la versió estàndard", | ||||||
|  |     "groupByCategory": "Agrupa per categories", | ||||||
|  |     "autoApkFilterByArch": "Intenta filtrar les APKs per l'aquitectura de la CPU, si és possible", | ||||||
|  |     "autoLinkFilterByArch": "Intenta filtrar els enllaços per l'aquitectura de la CPU, si és possible", | ||||||
|  |     "overrideSource": "Força la font", | ||||||
|  |     "dontShowAgain": "No ho tornis a mostrar", | ||||||
|  |     "dontShowTrackOnlyWarnings": "No mostris avisos de les aplicacions 'només per a seguiment'", | ||||||
|  |     "dontShowAPKOriginWarnings": "No mostris avisos dels orígens de les APKs", | ||||||
|  |     "moveNonInstalledAppsToBottom": "Desplaça les aplicacions no instal·lades al capdavall de les aplicacions", | ||||||
|  |     "gitlabPATLabel": "Token d'accés personal a GitLab", | ||||||
|  |     "about": "Quant a", | ||||||
|  |     "requiresCredentialsInSettings": "{} requereix credencials addicionals (a Paràmetres)", | ||||||
|  |     "checkOnStart": "Comprova si hi ha actualitzacions en iniciar Obtainium", | ||||||
|  |     "tryInferAppIdFromCode": "Intenta deduir l'ID de l'aplicació des del codi font", | ||||||
|  |     "removeOnExternalUninstall": "Suprimeix de forma automàtica les aplicacions desinstal·lades externament", | ||||||
|  |     "pickHighestVersionCode": "Selecciona de forma automàtica la versió superior de l'APK", | ||||||
|  |     "checkUpdateOnDetailPage": "Comprova les actualitzacions en obrir la pàgina de detalls de l'aplicació", | ||||||
|  |     "disablePageTransitions": "Inhabilita les animacions de transició de pàgina", | ||||||
|  |     "reversePageTransitions": "Inverteix les animacions de transició de pàgina", | ||||||
|  |     "minStarCount": "Nombre mínim d'estrelles", | ||||||
|  |     "addInfoBelow": "Afegeix aquesta informació a sota.", | ||||||
|  |     "addInfoInSettings": "Afegeix aquesta informació a Paràmetres.", | ||||||
|  |     "githubSourceNote": "La limitació de peticions a GitHub es pot evitar amb una clau API.", | ||||||
|  |     "sortByLastLinkSegment": "Ordena per 'només el darrer fragment de l'enllaç'", | ||||||
|  |     "filterReleaseNotesByRegEx": "Filtra les notes de la publicació de la versió per una expressió regular", | ||||||
|  |     "customLinkFilterRegex": "Filtre personalitzat de l'enllaç de l'APK per una expressió regular (Per_defecte '.apk$')", | ||||||
|  |     "appsPossiblyUpdated": "S'ha intentat l'actualització de l'aplicació", | ||||||
|  |     "appsPossiblyUpdatedNotifDescription": "Notifica l'usuari que les actualitzacions per a una o més aplicacions podrien haver-se fet en segon pla", | ||||||
|  |     "xWasPossiblyUpdatedToY": "{} podria haver-se actualitzat a {}.", | ||||||
|  |     "enableBackgroundUpdates": "Habilita les actualizacions en segon pla", | ||||||
|  |     "backgroundUpdateReqsExplanation": "Les actualitzacions en segon pla és possible que no estiguin disponibles per a totes les aplicacions.", | ||||||
|  |     "backgroundUpdateLimitsExplanation": "Les instal·lacions en segon pla reexides només es poden comprovar amb Obtainium obert.", | ||||||
|  |     "verifyLatestTag": "Comprova l'etiqueta 'Latest' (última versió)", | ||||||
|  |     "intermediateLinkRegex": "Filtra per un enllaç 'intermediari' per anar-hi", | ||||||
|  |     "filterByLinkText": "Filtra els enllaços pel text de l'enllaç", | ||||||
|  |     "intermediateLinkNotFound": "No s'ha trobat l'enllaç intermediari", | ||||||
|  |     "intermediateLink": "Enllaç intermediari", | ||||||
|  |     "exemptFromBackgroundUpdates": "Exempta d'actualitzacions en segon pla (si han estat habilitades)", | ||||||
|  |     "bgUpdatesOnWiFiOnly": "Inhabilita les actualitzacions en segon pla sense Wi-Fi", | ||||||
|  |     "bgUpdatesWhileChargingOnly": "Inhabilita les actualitzacions en segon pla quan no s'estigui carregant el mòbil", | ||||||
|  |     "autoSelectHighestVersionCode": "Selecciona de forma automàtica la versió més recent de l'APK", | ||||||
|  |     "versionExtractionRegEx": "Extracció de la cadena de la versió amb una expressió regular", | ||||||
|  |     "trimVersionString": "Retalla la cadena de la versió amb una expressió regular", | ||||||
|  |     "matchGroupToUseForX": "Grup de coincidència a usar per a \"{}\"", | ||||||
|  |     "matchGroupToUse": "Grup de coincidència a usar per a l'extracció de la cadena de la versió amb una expressió regular", | ||||||
|  |     "highlightTouchTargets": "Ressalta els elements de selecció menys obvis", | ||||||
|  |     "pickExportDir": "Selecciona el directori d'exportació", | ||||||
|  |     "autoExportOnChanges": "Exporta automàticament quan hi hagi canvis", | ||||||
|  |     "includeSettings": "Inclou paràmetres", | ||||||
|  |     "filterVersionsByRegEx": "Filtra les versions per una expressió regular", | ||||||
|  |     "trySelectingSuggestedVersionCode": "Prova a seleccionar la versió de l'APK suggerida", | ||||||
|  |     "dontSortReleasesList": "Mantén l'ordre de publicació de l'API", | ||||||
|  |     "reverseSort": "Ordre invers", | ||||||
|  |     "takeFirstLink": "Usa el primer enllaç", | ||||||
|  |     "skipSort": "Omet l'ordre", | ||||||
|  |     "debugMenu": "Menú de depuració", | ||||||
|  |     "bgTaskStarted": "S'ha iniciat la tasca en segon pla (revisa-ho als registres).", | ||||||
|  |     "runBgCheckNow": "Executa la comprovació d'actualitzacions en segon pla", | ||||||
|  |     "versionExtractWholePage": "Aplica l'extracció de la cadena de la versió amb una expressió regular a tota la pàgina", | ||||||
|  |     "installing": "Instal·lant", | ||||||
|  |     "skipUpdateNotifications": "No notifiquis les actualitzacions", | ||||||
|  |     "updatesAvailableNotifChannel": "Actualitzacions disponibles", | ||||||
|  |     "appsUpdatedNotifChannel": "Aplicacions actualitzades", | ||||||
|  |     "appsPossiblyUpdatedNotifChannel": "S'ha intentat actualitzar l'aplicació", | ||||||
|  |     "errorCheckingUpdatesNotifChannel": "Error en cercar actualitzacions", | ||||||
|  |     "appsRemovedNotifChannel": "Aplicacions suprimides", | ||||||
|  |     "downloadingXNotifChannel": "Descarregant {}", | ||||||
|  |     "completeAppInstallationNotifChannel": "Instal·lació finalitzada", | ||||||
|  |     "checkingForUpdatesNotifChannel": "S'estan cercant actualitzacions", | ||||||
|  |     "onlyCheckInstalledOrTrackOnlyApps": "Comprova les actualitzacions només per a aplicacions instal·lades o en seguiment", | ||||||
|  |     "supportFixedAPKURL": "Suport per als URLs fixos de l'APK", | ||||||
|  |     "selectX": "Selecciona {}", | ||||||
|  |     "parallelDownloads": "Permet les descàrregues paralel·les", | ||||||
|  |     "useShizuku": "Usa Shizuku o Sui per instal·lar", | ||||||
|  |     "shizukuBinderNotFound": "Shizuku no s'està executant", | ||||||
|  |     "shizukuOld": "Versió antiga de Shizuku (<11) - Actualitza-la", | ||||||
|  |     "shizukuOldAndroidWithADB": "Shizuku s'executa en Android < 8.1 amb ADB - Actualitza Android o usa Sui com a alternativa", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Defineix Google Play com a font d'instal·lació (si uses Shizuku)", | ||||||
|  |     "useSystemFont": "Usa la font del sistema", | ||||||
|  |     "useVersionCodeAsOSVersion": "Usa la versió de l'aplicació com a versió detectada del Sistema Operatiu", | ||||||
|  |     "requestHeader": "Capçalera de sol·licitud", | ||||||
|  |     "useLatestAssetDateAsReleaseDate": "Usa el darrer recurs carregat com a data de llançament", | ||||||
|  |     "defaultPseudoVersioningMethod": "Mètode de pseudo-versionat predeterminat", | ||||||
|  |     "partialAPKHash": "Hash de l'APK parcial", | ||||||
|  |     "APKLinkHash": "Hash de l'enllaç de l'APK", | ||||||
|  |     "directAPKLink": "Enllaç de l'APK directe", | ||||||
|  |     "pseudoVersionInUse": "S'està usant una pseudo-versió", | ||||||
|  |     "installed": "Instal·lada", | ||||||
|  |     "latest": "Versió més recent", | ||||||
|  |     "invertRegEx": "Inverteix l'expressió regular", | ||||||
|  |     "note": "Nota", | ||||||
|  |     "selfHostedNote": "El desplegable «{}» es pot usar per accedir a instàncies autoallotjades/personalitzades de qualsevol font.", | ||||||
|  |     "badDownload": "L'APK no s'ha pogut analitzar (incompatible o descàrrega parcial)", | ||||||
|  |     "beforeNewInstallsShareToAppVerifier": "Comparteix les aplicacions noves amb AppVerifier (si està instal·lat)", | ||||||
|  |     "appVerifierInstructionToast": "Comparteix amb AppVerifier i torna aquí quan estigui fet.", | ||||||
|  |     "wiki": "Ajuda/Wiki", | ||||||
|  |     "crowdsourcedConfigsLabel": "Configuració de les aplicacions crowdsourcing (usa-ho sota la teva responsabilitat)", | ||||||
|  |     "crowdsourcedConfigsShort": "Configuració de les aplicacions crowdsourcing", | ||||||
|  |     "allowInsecure": "Permet les sol·licituds HTTP insegures", | ||||||
|  |     "stayOneVersionBehind": "Roman a la versió anterior a l'última", | ||||||
|  |     "useFirstApkOfVersion": "Auto-select first of multiple APKs", | ||||||
|  |     "refreshBeforeDownload": "Actualitza les dades de l'aplicació abans de descarregar-la", | ||||||
|  |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Nom", | ||||||
|  |     "smartname": "Nom (smart)", | ||||||
|  |     "sortMethod": "Mètode d'ordenació", | ||||||
|  |     "welcome": "Benvinguda", | ||||||
|  |     "documentationLinksNote": "La pàgina GitHub d'Obtainium enllaçada a sota conté enllaços a vídeos, articles, debats i altres recursos que t'ajudaran a entendre com usar l'aplicació.", | ||||||
|  |     "batteryOptimizationNote": "Tingues present que les descàrregues en segon pla funcionaran millor si inhabilites l'optimització de bateria per a Obtainium.", | ||||||
|  |     "fileDeletionError": "Failed to delete file (try deleting it manually then try again): \"{}\"", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "¿Suprimeixo l'aplicació?", | ||||||
|  |         "other": "¿Suprimeixo les aplicacions?" | ||||||
|  |     }, | ||||||
|  |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|  |         "one": "Massa peticions (límit excedit), torna-hi en {} minut", | ||||||
|  |         "other": "Massa peticions (límit excedit), torna-hi en {} minuts" | ||||||
|  |     }, | ||||||
|  |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|  |         "one": "La comprovació d'actualitzacions en segon pla ha trobat un {}, es tornarà a provar en {} minut", | ||||||
|  |         "other": "La comprovació d'actualitzacions en segon pla ha trobat un {}, es tornarà a provar en {} minuts" | ||||||
|  |     }, | ||||||
|  |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|  |         "one": "La comprovació d'actualitzacions en segon pla ha trobat {} actualització, t'ho notificarem si cal", | ||||||
|  |         "other": "La comprovació d'actualitzacions en segon pla ha trobat {} actualitzacions, t'ho notificarem si cal" | ||||||
|  |     }, | ||||||
|  |     "apps": { | ||||||
|  |         "one": "{} Aplicació", | ||||||
|  |         "other": "{} Aplicacions" | ||||||
|  |     }, | ||||||
|  |     "url": { | ||||||
|  |         "one": "{} URL", | ||||||
|  |         "other": "{} URLs" | ||||||
|  |     }, | ||||||
|  |     "minute": { | ||||||
|  |         "one": "{} Minut", | ||||||
|  |         "other": "{} Minuts" | ||||||
|  |     }, | ||||||
|  |     "hour": { | ||||||
|  |         "one": "{} Hora", | ||||||
|  |         "other": "{} Hores" | ||||||
|  |     }, | ||||||
|  |     "day": { | ||||||
|  |         "one": "{} Dia", | ||||||
|  |         "other": "{} Dies" | ||||||
|  |     }, | ||||||
|  |     "clearedNLogsBeforeXAfterY": { | ||||||
|  |         "one": "Suprimit {n} registre (anterior a = {before}, posterior a = {after})", | ||||||
|  |         "other": "Suprimits {n} registres (anteriors a = {before}, posteriors a = {after})" | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesAvailable": { | ||||||
|  |         "one": "{} i 1 aplicació més tenen actualitzacions.", | ||||||
|  |         "other": "{} i {} aplicacions més tenen actualitzacions." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesInstalled": { | ||||||
|  |         "one": "{} i 1 aplicació més s'han actualitzat.", | ||||||
|  |         "other": "{} i {} aplicacions més s'han actualitzat." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Error en actualitzar {} i 1 aplicació més.", | ||||||
|  |         "other": "No s'ha pogut actualizar {} i {} aplicacions més." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|  |         "one": "{} i 1 aplicació més podrien haver estat actualizades.", | ||||||
|  |         "other": "{} i {} aplicacions més podrien haver estat actualitzades." | ||||||
|  |     }, | ||||||
|  |     "apk": { | ||||||
|  |         "one": "{} APK", | ||||||
|  |         "other": "{} APKs" | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -36,7 +36,7 @@ | |||||||
|     "trackOnlyAppDescription": "Aplikace je sledována kvůli aktualizacím, ale Obtainium ji nebude stahovat ani instalovat.", |     "trackOnlyAppDescription": "Aplikace je sledována kvůli aktualizacím, ale Obtainium ji nebude stahovat ani instalovat.", | ||||||
|     "cancelled": "Zrušeno", |     "cancelled": "Zrušeno", | ||||||
|     "appAlreadyAdded": "Aplikace již přidána", |     "appAlreadyAdded": "Aplikace již přidána", | ||||||
|     "alreadyUpToDateQuestion": "App already up to date?", |     "alreadyUpToDateQuestion": "Aplikace je již aktualizovaná?", | ||||||
|     "addApp": "Přidat aplikaci", |     "addApp": "Přidat aplikaci", | ||||||
|     "appSourceURL": "Zdrojová adresa URL aplikace", |     "appSourceURL": "Zdrojová adresa URL aplikace", | ||||||
|     "error": "Chyba", |     "error": "Chyba", | ||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Standardní detekce verze", |     "standardVersionDetection": "Standardní detekce verze", | ||||||
|     "groupByCategory": "Seskupit podle kategorie", |     "groupByCategory": "Seskupit podle kategorie", | ||||||
|     "autoApkFilterByArch": "Pokud je to možné, pokuste se filtrovat soubory APK podle architektury procesoru", |     "autoApkFilterByArch": "Pokud je to možné, pokuste se filtrovat soubory APK podle architektury procesoru", | ||||||
|  |     "autoLinkFilterByArch": "Pokus o filtrování odkazů podle architektury procesoru, pokud je to možné.", | ||||||
|     "overrideSource": "Přepsat zdroj", |     "overrideSource": "Přepsat zdroj", | ||||||
|     "dontShowAgain": "Nezobrazovat znovu", |     "dontShowAgain": "Nezobrazovat znovu", | ||||||
|     "dontShowTrackOnlyWarnings": "Nezobrazovat varování pro 'Jen sledované'", |     "dontShowTrackOnlyWarnings": "Nezobrazovat varování pro 'Jen sledované'", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Konfigurace aplikací s využitím crowdsourcingu", |     "crowdsourcedConfigsShort": "Konfigurace aplikací s využitím crowdsourcingu", | ||||||
|     "allowInsecure": "Povolení nezabezpečených požadavků HTTP", |     "allowInsecure": "Povolení nezabezpečených požadavků HTTP", | ||||||
|     "stayOneVersionBehind": "Zůstaňte o jednu verzi pozadu za nejnovější", |     "stayOneVersionBehind": "Zůstaňte o jednu verzi pozadu za nejnovější", | ||||||
|  |     "useFirstApkOfVersion": "Automatický výběr prvního z více souborů APK", | ||||||
|     "refreshBeforeDownload": "Obnovení údajů o aplikaci před stažením", |     "refreshBeforeDownload": "Obnovení údajů o aplikaci před stažením", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Název", | ||||||
|  |     "smartname": "Název (Smart)", | ||||||
|  |     "sortMethod": "Metoda třídění", | ||||||
|  |     "welcome": "Vítejte na", | ||||||
|  |     "documentationLinksNote": "Níže odkazovaná stránka Obtainium GitHub obsahuje odkazy na videa, články, diskuse a další zdroje, které vám pomohou pochopit, jak aplikaci používat.", | ||||||
|  |     "batteryOptimizationNote": "Všimněte si, že stahování na pozadí může fungovat spolehlivěji, pokud vypnete optimalizaci baterie operačního systému pro Obtainium.", | ||||||
|  |     "fileDeletionError": "Soubor se nepodařilo odstranit (zkuste jej odstranit ručně a pak to zkuste znovu): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Odstranit Apku?", |         "one": "Odstranit Apku?", | ||||||
|         "other": "Odstranit Apky?" |         "other": "Odstranit Apky?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Standard versionsregistrering", |     "standardVersionDetection": "Standard versionsregistrering", | ||||||
|     "groupByCategory": "Gruppér efter kategori", |     "groupByCategory": "Gruppér efter kategori", | ||||||
|     "autoApkFilterByArch": "Forsøg at filtrere APK'er efter CPU-arkitektur, hvis muligt", |     "autoApkFilterByArch": "Forsøg at filtrere APK'er efter CPU-arkitektur, hvis muligt", | ||||||
|  |     "autoLinkFilterByArch": "Forsøg at filtrere links efter CPU-arkitektur, hvis det er muligt", | ||||||
|     "overrideSource": "Tilsidesæt kilde", |     "overrideSource": "Tilsidesæt kilde", | ||||||
|     "dontShowAgain": "Vis ikke igen", |     "dontShowAgain": "Vis ikke igen", | ||||||
|     "dontShowTrackOnlyWarnings": "Vis ikke 'Følg Kun'-advarsler", |     "dontShowTrackOnlyWarnings": "Vis ikke 'Følg Kun'-advarsler", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Crowdsourcede app-konfigurationer", |     "crowdsourcedConfigsShort": "Crowdsourcede app-konfigurationer", | ||||||
|     "allowInsecure": "Tillad usikre HTTP-anmodninger", |     "allowInsecure": "Tillad usikre HTTP-anmodninger", | ||||||
|     "stayOneVersionBehind": "Forbliv én version bagud den seneste", |     "stayOneVersionBehind": "Forbliv én version bagud den seneste", | ||||||
|  |     "useFirstApkOfVersion": "Vælg automatisk den første af flere APK'er", | ||||||
|     "refreshBeforeDownload": "Opdater app-detaljer før download", |     "refreshBeforeDownload": "Opdater app-detaljer før download", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Navn", | ||||||
|  |     "smartname": "Navn (Smart)", | ||||||
|  |     "sortMethod": "Sorteringsmetode", | ||||||
|  |     "welcome": "Velkommen", | ||||||
|  |     "documentationLinksNote": "Obtainiums GitHub-side, som der linkes til nedenfor, indeholder links til videoer, artikler, diskussioner og andre ressourcer, som kan hjælpe dig med at forstå, hvordan du bruger appen.", | ||||||
|  |     "batteryOptimizationNote": "Bemærk, at baggrundsdownloads kan fungere mere pålideligt, hvis du deaktiverer OS-batterioptimering for Obtainium.", | ||||||
|  |     "fileDeletionError": "Kunne ikke slette filen (prøv at slette den manuelt og prøv igen): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Fjern app?", |         "one": "Fjern app?", | ||||||
|         "other": "Fjern apps?" |         "other": "Fjern apps?" | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ | |||||||
|     "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 zur Versionsüberwachung.", |     "appsFromSourceAreTrackOnly": "Apps aus dieser Quelle sind nur zur Nachverfolgung.", | ||||||
|     "youPickedTrackOnly": "Du hast die Option „Nur nachverfolgen“ gewählt.", |     "youPickedTrackOnly": "Du hast die Option „Nur nachverfolgen“ gewählt.", | ||||||
|     "trackOnlyAppDescription": "Die App wird auf neue verfügbare Versionen überwacht, aber Obtainium wird sie nicht herunterladen oder installieren.", |     "trackOnlyAppDescription": "Die App wird auf neue verfügbare Versionen überwacht, aber Obtainium wird sie nicht herunterladen oder installieren.", | ||||||
|     "cancelled": "Abgebrochen", |     "cancelled": "Abgebrochen", | ||||||
| @@ -85,7 +85,7 @@ | |||||||
|     "filter": "Filter", |     "filter": "Filter", | ||||||
|     "filterApps": "Apps filtern", |     "filterApps": "Apps filtern", | ||||||
|     "appName": "App-Name", |     "appName": "App-Name", | ||||||
|     "author": "Autor:in", |     "author": "Herausgebende", | ||||||
|     "upToDateApps": "Apps mit aktuellster Version", |     "upToDateApps": "Apps mit aktuellster Version", | ||||||
|     "nonInstalledApps": "Nicht installierte Apps", |     "nonInstalledApps": "Nicht installierte Apps", | ||||||
|     "importExport": "Import/Export", |     "importExport": "Import/Export", | ||||||
| @@ -109,15 +109,15 @@ | |||||||
|     "selectURL": "URL auswählen", |     "selectURL": "URL auswählen", | ||||||
|     "selectURLs": "URLs auswählen", |     "selectURLs": "URLs auswählen", | ||||||
|     "pick": "Auswählen", |     "pick": "Auswählen", | ||||||
|     "theme": "Theme", |     "theme": "Design", | ||||||
|     "dark": "Dunkel", |     "dark": "Dunkel", | ||||||
|     "light": "Hell", |     "light": "Hell", | ||||||
|     "followSystem": "Systemstandard", |     "followSystem": "Systemstandard", | ||||||
|     "followSystemThemeExplanation": "Das Abrufen des Systemdesigns ist unter Android < 10 nur mit Hilfe von Drittanbieterapps möglich", |     "followSystemThemeExplanation": "Das Abrufen des Systemdesigns ist unter Android < 10 nur mit Hilfe von Drittanbieterapps möglich", | ||||||
|     "useBlackTheme": "Rein schwarzen Hintergrund verwenden", |     "useBlackTheme": "Rein schwarzen Hintergrund verwenden", | ||||||
|     "appSortBy": "App sortieren nach", |     "appSortBy": "App sortieren nach", | ||||||
|     "authorName": "Autor:in/Name", |     "authorName": "Hrsg./Name", | ||||||
|     "nameAuthor": "Name/Autor:in", |     "nameAuthor": "Name/Hrsg.", | ||||||
|     "asAdded": "Wie hinzugefügt", |     "asAdded": "Wie hinzugefügt", | ||||||
|     "appSortOrder": "App sortieren nach", |     "appSortOrder": "App sortieren nach", | ||||||
|     "ascending": "Aufsteigend", |     "ascending": "Aufsteigend", | ||||||
| @@ -152,9 +152,9 @@ | |||||||
|     "xWasUpdatedToY": "{} wurde auf {} aktualisiert.", |     "xWasUpdatedToY": "{} wurde auf {} aktualisiert.", | ||||||
|     "xWasNotUpdatedToY": "Die Aktualisierung von {} auf {} ist fehlgeschlagen.", |     "xWasNotUpdatedToY": "Die Aktualisierung von {} auf {} ist fehlgeschlagen.", | ||||||
|     "errorCheckingUpdates": "Fehler beim Prüfen auf Aktualisierungen", |     "errorCheckingUpdates": "Fehler beim Prüfen auf Aktualisierungen", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Weist darauf hin, dass die Prüfung der Hintergrundaktualisierung fehlgeschlagen ist", |     "errorCheckingUpdatesNotifDescription": "Benachrichtigt, wenn die Prüfung der Hintergrundaktualisierung fehlgeschlagen ist", | ||||||
|     "appsRemoved": "Apps entfernt", |     "appsRemoved": "Apps entfernt", | ||||||
|     "appsRemovedNotifDescription": "Weist darauf hin, dass eine oder mehrere Apps aufgrund von Fehlern beim Laden entfernt wurden", |     "appsRemovedNotifDescription": "Benachrichtigt, wenn 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", | ||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "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", | ||||||
|  |     "autoLinkFilterByArch": "Versuchen Sie, Links nach CPU-Architektur zu filtern, wenn möglich", | ||||||
|     "overrideSource": "Quelle überschreiben", |     "overrideSource": "Quelle überschreiben", | ||||||
|     "dontShowAgain": "Nicht noch einmal anzeigen", |     "dontShowAgain": "Nicht noch einmal anzeigen", | ||||||
|     "dontShowTrackOnlyWarnings": "Warnung für „Nur nachverfolgen“ nicht anzeigen", |     "dontShowTrackOnlyWarnings": "Warnung für „Nur nachverfolgen“ nicht anzeigen", | ||||||
| @@ -244,10 +245,10 @@ | |||||||
|     "filterReleaseNotesByRegEx": "Versionshinweise nach regulärem Ausdruck\nfiltern", |     "filterReleaseNotesByRegEx": "Versionshinweise nach regulärem Ausdruck\nfiltern", | ||||||
|     "customLinkFilterRegex": "Benutzerdefinierter APK-Linkfilter durch regulären 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, dass Aktualisierungen für eine oder mehrere Apps möglicherweise im Hintergrund durchgeführt wurden", |     "appsPossiblyUpdatedNotifDescription": "Benachrichtigt, dass möglicherweise eine oder mehrere Apps im Hintergrund aktualisiert wurden", | ||||||
|     "xWasPossiblyUpdatedToY": "{} wurde möglicherweise aktualisiert auf {}.", |     "xWasPossiblyUpdatedToY": "{} wurde eventuell auf Version {} aktualisiert.", | ||||||
|     "enableBackgroundUpdates": "Hintergrundaktualisierungen aktivieren", |     "enableBackgroundUpdates": "Hintergrundaktualisierungen aktivieren", | ||||||
|     "backgroundUpdateReqsExplanation": "Die Hintergrundaktualisierung ist möglicherweise nicht für alle Apps möglich.", |     "backgroundUpdateReqsExplanation": "Die Hintergrundaktualisierung ist unter Umständen 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": "„Latest“-Tag überprüfen", |     "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", | ||||||
| @@ -292,8 +293,8 @@ | |||||||
|     "parallelDownloads": "Parallele Downloads erlauben", |     "parallelDownloads": "Parallele Downloads erlauben", | ||||||
|     "useShizuku": "Shizuku oder Sui zur Installation verwenden", |     "useShizuku": "Shizuku oder Sui zur Installation verwenden", | ||||||
|     "shizukuBinderNotFound": "Kompatibler Shizuku-Dienst wurde nicht gefunden", |     "shizukuBinderNotFound": "Kompatibler Shizuku-Dienst wurde nicht gefunden", | ||||||
|     "shizukuOld": "Alte Shizuku-Version (< 11) - aktualisieren Sie sie", |     "shizukuOld": "Veraltete Shizuku-Version (< 11) - bitte aktualisiere sie", | ||||||
|     "shizukuOldAndroidWithADB": "Shizuku läuft auf Android < 8.1 mit ADB - aktualisieren Sie Android oder verwenden Sie stattdessen Sui", |     "shizukuOldAndroidWithADB": "Shizuku läuft auf Android < 8.1 mit ADB - aktualisiere die Android-Version oder verwende stattdessen Sui", | ||||||
|     "shizukuPretendToBeGooglePlay": "(Mittels Shizuku) Google Play als Installationsquelle registrieren", |     "shizukuPretendToBeGooglePlay": "(Mittels Shizuku) Google Play als Installationsquelle registrieren", | ||||||
|     "useSystemFont": "Systemschriftart verwenden", |     "useSystemFont": "Systemschriftart verwenden", | ||||||
|     "useVersionCodeAsOSVersion": "Versionscode (versionCode) als erkannte Version vom Betriebssystem verwenden", |     "useVersionCodeAsOSVersion": "Versionscode (versionCode) als erkannte Version vom Betriebssystem verwenden", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Crowdsourced App-Konfigurationen", |     "crowdsourcedConfigsShort": "Crowdsourced App-Konfigurationen", | ||||||
|     "allowInsecure": "Unsichere HTTP-Anfragen zulassen", |     "allowInsecure": "Unsichere HTTP-Anfragen zulassen", | ||||||
|     "stayOneVersionBehind": "Eine Version hinter der neuesten Version bleiben", |     "stayOneVersionBehind": "Eine Version hinter der neuesten Version bleiben", | ||||||
|  |     "useFirstApkOfVersion": "Automatische Auswahl der ersten von mehreren APKs", | ||||||
|     "refreshBeforeDownload": "App-Details vor dem Download aktualisieren", |     "refreshBeforeDownload": "App-Details vor dem Download aktualisieren", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Name", | ||||||
|  |     "smartname": "Name (Smart)", | ||||||
|  |     "sortMethod": "Sortierverfahren", | ||||||
|  |     "welcome": "Willkommen", | ||||||
|  |     "documentationLinksNote": "Die unten verlinkte GitHub-Seite von Obtainium enthält Links zu Videos, Artikeln, Diskussionen und anderen Ressourcen, die Ihnen helfen werden, die Verwendung der App zu verstehen.", | ||||||
|  |     "batteryOptimizationNote": "Beachten Sie, dass Downloads im Hintergrund möglicherweise zuverlässiger funktionieren, wenn Sie die Batterieoptimierung des Betriebssystems für Obtainium deaktivieren.", | ||||||
|  |     "fileDeletionError": "Die Datei konnte nicht gelöscht werden (versuchen Sie, sie manuell zu löschen und versuchen Sie es dann erneut): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "App entfernen?", |         "one": "App entfernen?", | ||||||
|         "other": "Apps entfernen?" |         "other": "Apps entfernen?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Norma versiodetekto", |     "standardVersionDetection": "Norma versiodetekto", | ||||||
|     "groupByCategory": "Grupigi per kategorio", |     "groupByCategory": "Grupigi per kategorio", | ||||||
|     "autoApkFilterByArch": "Provi filtri APKj per CPU-arkitekturo se ebla", |     "autoApkFilterByArch": "Provi filtri APKj per CPU-arkitekturo se ebla", | ||||||
|  |     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||||
|     "overrideSource": "Substitui la fonton", |     "overrideSource": "Substitui la fonton", | ||||||
|     "dontShowAgain": "Ne plu montri", |     "dontShowAgain": "Ne plu montri", | ||||||
|     "dontShowTrackOnlyWarnings": "Ne plu montri 'Nur sekvita' avertojn", |     "dontShowTrackOnlyWarnings": "Ne plu montri 'Nur sekvita' avertojn", | ||||||
| @@ -314,11 +315,21 @@ | |||||||
|     "appVerifierInstructionToast": "Diskonigu kun AppVerifier, poste revenu ĉi tie kiam preta.", |     "appVerifierInstructionToast": "Diskonigu kun AppVerifier, poste revenu ĉi tie kiam preta.", | ||||||
|     "wiki": "Helpo/Vikio", |     "wiki": "Helpo/Vikio", | ||||||
|     "crowdsourcedConfigsLabel": "Komunumaj apo-agordoj (uzu kun singardo)", |     "crowdsourcedConfigsLabel": "Komunumaj apo-agordoj (uzu kun singardo)", | ||||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", |     "crowdsourcedConfigsShort": "Crowdsourced App Configurations", | ||||||
|     "allowInsecure": "Allow insecure HTTP requests", |     "allowInsecure": "Allow insecure HTTP requests", | ||||||
|     "stayOneVersionBehind": "Stay one version behind latest", |     "stayOneVersionBehind": "Stay one version behind latest", | ||||||
|  |     "useFirstApkOfVersion": "Auto-select first of multiple APKs", | ||||||
|     "refreshBeforeDownload": "Refresh app details before download", |     "refreshBeforeDownload": "Refresh app details before download", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Name", | ||||||
|  |     "smartname": "Name (Smart)", | ||||||
|  |     "sortMethod": "Sort Method", | ||||||
|  |     "welcome": "Welcome", | ||||||
|  |     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||||
|  |     "batteryOptimizationNote": "Note that background downloads may work more reliably if you disable OS battery optimizations for Obtainium.", | ||||||
|  |     "fileDeletionError": "Failed to delete file (try deleting it manually then try again): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Forigi la aplikaĵon?", |         "one": "Forigi la aplikaĵon?", | ||||||
|         "other": "Forigi la aplikaĵojn?" |         "other": "Forigi la aplikaĵojn?" | ||||||
|   | |||||||
| @@ -1,110 +1,110 @@ | |||||||
| { | { | ||||||
|     "invalidURLForSource": "Not a valid {} App URL", |     "invalidURLForSource": "Not a valid {} app URL", | ||||||
|     "noReleaseFound": "Could not find a suitable release", |     "noReleaseFound": "Could not find a suitable release", | ||||||
|     "noVersionFound": "Could not determine release version", |     "noVersionFound": "Could not determine release version", | ||||||
|     "urlMatchesNoSource": "URL does not match a known source", |     "urlMatchesNoSource": "URL does not match a known source", | ||||||
|     "cantInstallOlderVersion": "Cannot install an older version of an App", |     "cantInstallOlderVersion": "Cannot install an older version of an app", | ||||||
|     "appIdMismatch": "Downloaded package ID does not match existing App ID", |     "appIdMismatch": "Downloaded package ID does not match existing app ID", | ||||||
|     "functionNotImplemented": "This class has not implemented this function", |     "functionNotImplemented": "This class has not implemented this function", | ||||||
|     "placeholder": "Placeholder", |     "placeholder": "Placeholder", | ||||||
|     "someErrors": "Some Errors Occurred", |     "someErrors": "Some errors occurred", | ||||||
|     "unexpectedError": "Unexpected Error", |     "unexpectedError": "Unexpected error", | ||||||
|     "ok": "Okay", |     "ok": "Okay", | ||||||
|     "and": "and", |     "and": "and", | ||||||
|     "githubPATLabel": "GitHub Personal Access Token (Increases Rate Limit)", |     "githubPATLabel": "GitHub personal access token (increases rate limit)", | ||||||
|     "includePrereleases": "Include prereleases", |     "includePrereleases": "Include prereleases", | ||||||
|     "fallbackToOlderReleases": "Fallback to older releases", |     "fallbackToOlderReleases": "Fallback to older releases", | ||||||
|     "filterReleaseTitlesByRegEx": "Filter Release Titles by Regular Expression", |     "filterReleaseTitlesByRegEx": "Filter release titles by regular expression", | ||||||
|     "invalidRegEx": "Invalid regular expression", |     "invalidRegEx": "Invalid regular expression", | ||||||
|     "noDescription": "No description", |     "noDescription": "No description", | ||||||
|     "cancel": "Cancel", |     "cancel": "Cancel", | ||||||
|     "continue": "Continue", |     "continue": "Continue", | ||||||
|     "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", |     "standard": "Standard", | ||||||
|     "custom": "Custom", |     "custom": "Custom", | ||||||
|     "useMaterialYou": "Use Material You", |     "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", | ||||||
|     "xIsTrackOnly": "{} is Track-Only", |     "xIsTrackOnly": "{} is track-only", | ||||||
|     "source": "Source", |     "source": "Source", | ||||||
|     "app": "App", |     "app": "App", | ||||||
|     "appsFromSourceAreTrackOnly": "Apps from this source are 'Track-Only'.", |     "appsFromSourceAreTrackOnly": "Apps from this source are 'track-only'.", | ||||||
|     "youPickedTrackOnly": "You have selected the 'Track-Only' option.", |     "youPickedTrackOnly": "You have selected the 'track-only' option.", | ||||||
|     "trackOnlyAppDescription": "The App will be tracked for updates, but Obtainium will not be able to download or install it.", |     "trackOnlyAppDescription": "The app will be tracked for updates, but Obtainium will not be able to download or install it.", | ||||||
|     "cancelled": "Cancelled", |     "cancelled": "Cancelled", | ||||||
|     "appAlreadyAdded": "App already added", |     "appAlreadyAdded": "App already added", | ||||||
|     "alreadyUpToDateQuestion": "App Already up to Date?", |     "alreadyUpToDateQuestion": "App already up to date?", | ||||||
|     "addApp": "Add App", |     "addApp": "Add App", | ||||||
|     "appSourceURL": "App Source URL", |     "appSourceURL": "App source URL", | ||||||
|     "error": "Error", |     "error": "Error", | ||||||
|     "add": "Add", |     "add": "Add", | ||||||
|     "searchSomeSourcesLabel": "Search (Some Sources Only)", |     "searchSomeSourcesLabel": "Search (some sources only)", | ||||||
|     "search": "Search", |     "search": "Search", | ||||||
|     "additionalOptsFor": "Additional Options for {}", |     "additionalOptsFor": "Additional options for {}", | ||||||
|     "supportedSources": "Supported Sources", |     "supportedSources": "Supported sources", | ||||||
|     "trackOnlyInBrackets": "(Track-Only)", |     "trackOnlyInBrackets": "(track-only)", | ||||||
|     "searchableInBrackets": "(Searchable)", |     "searchableInBrackets": "(searchable)", | ||||||
|     "appsString": "Apps", |     "appsString": "Apps", | ||||||
|     "noApps": "No Apps", |     "noApps": "No apps", | ||||||
|     "noAppsForFilter": "No Apps for Filter", |     "noAppsForFilter": "No apps for filter", | ||||||
|     "byX": "By {}", |     "byX": "By {}", | ||||||
|     "percentProgress": "Progress: {}%", |     "percentProgress": "Progress: {}%", | ||||||
|     "pleaseWait": "Please Wait", |     "pleaseWait": "Please wait", | ||||||
|     "updateAvailable": "Update Available", |     "updateAvailable": "Update available", | ||||||
|     "notInstalled": "Not Installed", |     "notInstalled": "Not installed", | ||||||
|     "pseudoVersion": "pseudo-version", |     "pseudoVersion": "pseudo-version", | ||||||
|     "selectAll": "Select All", |     "selectAll": "Select all", | ||||||
|     "deselectX": "Deselect {}", |     "deselectX": "Deselect {}", | ||||||
|     "xWillBeRemovedButRemainInstalled": "{} will be removed from Obtainium but remain installed on device.", |     "xWillBeRemovedButRemainInstalled": "{} will be removed from Obtainium but remain installed on device.", | ||||||
|     "removeSelectedAppsQuestion": "Remove Selected Apps?", |     "removeSelectedAppsQuestion": "Remove selected apps?", | ||||||
|     "removeSelectedApps": "Remove Selected Apps", |     "removeSelectedApps": "Remove selected apps", | ||||||
|     "updateX": "Update {}", |     "updateX": "Update {}", | ||||||
|     "installX": "Install {}", |     "installX": "Install {}", | ||||||
|     "markXTrackOnlyAsUpdated": "Mark {}\n(Track-Only)\nas Updated", |     "markXTrackOnlyAsUpdated": "Mark {}\n(track-only)\nas updated", | ||||||
|     "changeX": "Change {}", |     "changeX": "Change {}", | ||||||
|     "installUpdateApps": "Install/Update Apps", |     "installUpdateApps": "Install/update apps", | ||||||
|     "installUpdateSelectedApps": "Install/Update Selected Apps", |     "installUpdateSelectedApps": "Install/update selected apps", | ||||||
|     "markXSelectedAppsAsUpdated": "Mark {} Selected Apps as Updated?", |     "markXSelectedAppsAsUpdated": "Mark {} selected apps as updated?", | ||||||
|     "no": "No", |     "no": "No", | ||||||
|     "yes": "Yes", |     "yes": "Yes", | ||||||
|     "markSelectedAppsUpdated": "Mark Selected Apps as Updated", |     "markSelectedAppsUpdated": "Mark selected apps as updated", | ||||||
|     "pinToTop": "Pin to top", |     "pinToTop": "Pin to top", | ||||||
|     "unpinFromTop": "Unpin from top", |     "unpinFromTop": "Unpin from top", | ||||||
|     "resetInstallStatusForSelectedAppsQuestion": "Reset Install Status for Selected Apps?", |     "resetInstallStatusForSelectedAppsQuestion": "Reset install status for selected apps?", | ||||||
|     "installStatusOfXWillBeResetExplanation": "The install status of any selected Apps will be reset.\n\nThis can help when the App version shown in Obtainium is incorrect due to failed updates or other issues.", |     "installStatusOfXWillBeResetExplanation": "The install status of any selected apps will be reset.\n\nThis can help when the app version shown in Obtainium is incorrect due to failed updates or other issues.", | ||||||
|     "customLinkMessage": "These links work on devices with Obtainium installed", |     "customLinkMessage": "These links work on devices with Obtainium installed", | ||||||
|     "shareAppConfigLinks": "Share app configuration as HTML link", |     "shareAppConfigLinks": "Share app configuration as HTML link", | ||||||
|     "shareSelectedAppURLs": "Share Selected App URLs", |     "shareSelectedAppURLs": "Share selected app URLs", | ||||||
|     "resetInstallStatus": "Reset Install Status", |     "resetInstallStatus": "Reset install status", | ||||||
|     "more": "More", |     "more": "More", | ||||||
|     "removeOutdatedFilter": "Remove Out-of-Date App Filter", |     "removeOutdatedFilter": "Remove out-of-date app filter", | ||||||
|     "showOutdatedOnly": "Show Out-of-Date Apps Only", |     "showOutdatedOnly": "Show out-of-date apps only", | ||||||
|     "filter": "Filter", |     "filter": "Filter", | ||||||
|     "filterApps": "Filter Apps", |     "filterApps": "Filter apps", | ||||||
|     "appName": "App Name", |     "appName": "App name", | ||||||
|     "author": "Author", |     "author": "Author", | ||||||
|     "upToDateApps": "Up to Date Apps", |     "upToDateApps": "Up to date apps", | ||||||
|     "nonInstalledApps": "Non-Installed Apps", |     "nonInstalledApps": "Non-installed apps", | ||||||
|     "importExport": "Import/Export", |     "importExport": "Import/Export", | ||||||
|     "settings": "Settings", |     "settings": "Settings", | ||||||
|     "exportedTo": "Exported to {}", |     "exportedTo": "Exported to {}", | ||||||
|     "obtainiumExport": "Obtainium Export", |     "obtainiumExport": "Obtainium export", | ||||||
|     "invalidInput": "Invalid input", |     "invalidInput": "Invalid input", | ||||||
|     "importedX": "Imported {}", |     "importedX": "Imported {}", | ||||||
|     "obtainiumImport": "Obtainium Import", |     "obtainiumImport": "Obtainium import", | ||||||
|     "importFromURLList": "Import from URL List", |     "importFromURLList": "Import from URL list", | ||||||
|     "searchQuery": "Search Query", |     "searchQuery": "Search query", | ||||||
|     "appURLList": "App URL List", |     "appURLList": "App URL list", | ||||||
|     "line": "Line", |     "line": "Line", | ||||||
|     "searchX": "Search {}", |     "searchX": "Search {}", | ||||||
|     "noResults": "No results found", |     "noResults": "No results found", | ||||||
|     "importX": "Import {}", |     "importX": "Import {}", | ||||||
|     "importedAppsIdDisclaimer": "Imported Apps may incorrectly show as \"Not Installed\".\nTo fix this, re-install them through Obtainium.\nThis should not affect App data.\n\nOnly affects URL and third-party import methods.", |     "importedAppsIdDisclaimer": "Imported apps may incorrectly show as \"not installed\".\nTo fix this, re-install them through Obtainium.\nThis should not affect app data.\n\nOnly affects URL and third-party import methods.", | ||||||
|     "importErrors": "Import Errors", |     "importErrors": "Import errors", | ||||||
|     "importedXOfYApps": "{} of {} Apps imported.", |     "importedXOfYApps": "{} of {} apps imported.", | ||||||
|     "followingURLsHadErrors": "The following URLs had errors:", |     "followingURLsHadErrors": "The following URLs had errors:", | ||||||
|     "selectURL": "Select URL", |     "selectURL": "Select URL", | ||||||
|     "selectURLs": "Select URLs", |     "selectURLs": "Select URLs", | ||||||
| @@ -112,26 +112,26 @@ | |||||||
|     "theme": "Theme", |     "theme": "Theme", | ||||||
|     "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", |     "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", | ||||||
|     "nameAuthor": "Name/Author", |     "nameAuthor": "Name/author", | ||||||
|     "asAdded": "As Added", |     "asAdded": "As added", | ||||||
|     "appSortOrder": "App Sort Order", |     "appSortOrder": "App sort order", | ||||||
|     "ascending": "Ascending", |     "ascending": "Ascending", | ||||||
|     "descending": "Descending", |     "descending": "Descending", | ||||||
|     "bgUpdateCheckInterval": "Background Update Checking Interval", |     "bgUpdateCheckInterval": "Background update checking interval", | ||||||
|     "neverManualOnly": "Never - Manual Only", |     "neverManualOnly": "Never - manual only", | ||||||
|     "appearance": "Appearance", |     "appearance": "Appearance", | ||||||
|     "showWebInAppView": "Show Source webpage in App view", |     "showWebInAppView": "Show source webpage in app view", | ||||||
|     "pinUpdates": "Pin updates to top of Apps view", |     "pinUpdates": "Pin updates to top of apps view", | ||||||
|     "updates": "Updates", |     "updates": "Updates", | ||||||
|     "sourceSpecific": "Source-Specific", |     "sourceSpecific": "Source-specific", | ||||||
|     "appSource": "App Source", |     "appSource": "App source", | ||||||
|     "noLogs": "No Logs", |     "noLogs": "No logs", | ||||||
|     "appLogs": "App Logs", |     "appLogs": "App logs", | ||||||
|     "close": "Close", |     "close": "Close", | ||||||
|     "share": "Share", |     "share": "Share", | ||||||
|     "appNotFound": "App not found", |     "appNotFound": "App not found", | ||||||
| @@ -141,28 +141,28 @@ | |||||||
|     "deviceSupportsXArch": "Your device supports the {} CPU architecture.", |     "deviceSupportsXArch": "Your device supports the {} CPU architecture.", | ||||||
|     "deviceSupportsFollowingArchs": "Your device supports the following CPU architectures:", |     "deviceSupportsFollowingArchs": "Your device supports the following CPU architectures:", | ||||||
|     "warning": "Warning", |     "warning": "Warning", | ||||||
|     "sourceIsXButPackageFromYPrompt": "The App source is '{}' but the release package comes from '{}'. Continue?", |     "sourceIsXButPackageFromYPrompt": "The app source is '{}' but the release package comes from '{}'. Continue?", | ||||||
|     "updatesAvailable": "Updates Available", |     "updatesAvailable": "Updates available", | ||||||
|     "updatesAvailableNotifDescription": "Notifies the user that updates are available for one or more Apps tracked by Obtainium", |     "updatesAvailableNotifDescription": "Notifies the user that updates are available for one or more apps tracked by Obtainium", | ||||||
|     "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", |     "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 {}.", |     "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", | ||||||
|     "appsRemovedNotifDescription": "Notifies the user that one or more Apps were removed due to errors while loading them", |     "appsRemovedNotifDescription": "Notifies the user that one or more apps were removed due to errors while loading them", | ||||||
|     "xWasRemovedDueToErrorY": "{} was removed due to this error: {}", |     "xWasRemovedDueToErrorY": "{} was removed due to this error: {}", | ||||||
|     "completeAppInstallation": "Complete App Installation", |     "completeAppInstallation": "Complete app installation", | ||||||
|     "obtainiumMustBeOpenToInstallApps": "Obtainium must be open to install Apps", |     "obtainiumMustBeOpenToInstallApps": "Obtainium must be open to install apps", | ||||||
|     "completeAppInstallationNotifDescription": "Asks the user to return to Obtainium to finish installing an App", |     "completeAppInstallationNotifDescription": "Asks the user to return to Obtainium to finish installing an app", | ||||||
|     "checkingForUpdates": "Checking for Updates", |     "checkingForUpdates": "Checking for updates", | ||||||
|     "checkingForUpdatesNotifDescription": "Transient notification that appears when checking for updates", |     "checkingForUpdatesNotifDescription": "Transient notification that appears when checking for updates", | ||||||
|     "pleaseAllowInstallPerm": "Please allow Obtainium to install Apps", |     "pleaseAllowInstallPerm": "Please allow Obtainium to install apps", | ||||||
|     "trackOnly": "Track-Only", |     "trackOnly": "Track-only", | ||||||
|     "errorWithHttpStatusCode": "Error {}", |     "errorWithHttpStatusCode": "Error {}", | ||||||
|     "versionCorrectionDisabled": "Version correction disabled (plugin doesn't seem to work)", |     "versionCorrectionDisabled": "Version correction disabled (plugin doesn't seem to work)", | ||||||
|     "unknown": "Unknown", |     "unknown": "Unknown", | ||||||
| @@ -170,123 +170,124 @@ | |||||||
|     "never": "Never", |     "never": "Never", | ||||||
|     "latestVersionX": "Latest: {}", |     "latestVersionX": "Latest: {}", | ||||||
|     "installedVersionX": "Installed: {}", |     "installedVersionX": "Installed: {}", | ||||||
|     "lastUpdateCheckX": "Last Update Check: {}", |     "lastUpdateCheckX": "Last update check: {}", | ||||||
|     "remove": "Remove", |     "remove": "Remove", | ||||||
|     "yesMarkUpdated": "Yes, Mark as Updated", |     "yesMarkUpdated": "Yes, mark as updated", | ||||||
|     "fdroid": "F-Droid Official", |     "fdroid": "F-Droid official", | ||||||
|     "appIdOrName": "App ID or Name", |     "appIdOrName": "App ID or name", | ||||||
|     "appId": "App ID", |     "appId": "App ID", | ||||||
|     "appWithIdOrNameNotFound": "No App was found with that ID or Name", |     "appWithIdOrNameNotFound": "No app was found with that ID or name", | ||||||
|     "reposHaveMultipleApps": "Repos may contain multiple Apps", |     "reposHaveMultipleApps": "Repos may contain multiple apps", | ||||||
|     "fdroidThirdPartyRepo": "F-Droid Third-Party Repo", |     "fdroidThirdPartyRepo": "F-Droid third-party repo", | ||||||
|     "install": "Install", |     "install": "Install", | ||||||
|     "markInstalled": "Mark Installed", |     "markInstalled": "Mark installed", | ||||||
|     "update": "Update", |     "update": "Update", | ||||||
|     "markUpdated": "Mark Updated", |     "markUpdated": "Mark updated", | ||||||
|     "additionalOptions": "Additional Options", |     "additionalOptions": "Additional options", | ||||||
|     "disableVersionDetection": "Disable Version Detection", |     "disableVersionDetection": "Disable version detection", | ||||||
|     "noVersionDetectionExplanation": "This option should only be used for Apps where version detection does not work correctly.", |     "noVersionDetectionExplanation": "This option should only be used for apps where version detection does not work correctly.", | ||||||
|     "downloadingX": "Downloading {}", |     "downloadingX": "Downloading {}", | ||||||
|     "downloadX": "Download {}", |     "downloadX": "Download {}", | ||||||
|     "downloadedX": "Downloaded {}", |     "downloadedX": "Downloaded {}", | ||||||
|     "releaseAsset": "Release Asset", |     "releaseAsset": "Release asset", | ||||||
|     "downloadNotifDescription": "Notifies the user of the progress in downloading an App", |     "downloadNotifDescription": "Notifies the user of the progress in downloading an app", | ||||||
|     "noAPKFound": "No APK found", |     "noAPKFound": "No APK found", | ||||||
|     "noVersionDetection": "No version detection", |     "noVersionDetection": "No version detection", | ||||||
|     "categorize": "Categorize", |     "categorize": "Categorize", | ||||||
|     "categories": "Categories", |     "categories": "Categories", | ||||||
|     "category": "Category", |     "category": "Category", | ||||||
|     "noCategory": "No Category", |     "noCategory": "No category", | ||||||
|     "noCategories": "No Categories", |     "noCategories": "No categories", | ||||||
|     "deleteCategoriesQuestion": "Delete Categories?", |     "deleteCategoriesQuestion": "Delete categories?", | ||||||
|     "categoryDeleteWarning": "All Apps in deleted categories will be set to uncategorized.", |     "categoryDeleteWarning": "All apps in deleted categories will be set to uncategorized.", | ||||||
|     "addCategory": "Add Category", |     "addCategory": "Add category", | ||||||
|     "label": "Label", |     "label": "Label", | ||||||
|     "language": "Language", |     "language": "Language", | ||||||
|     "copiedToClipboard": "Copied to Clipboard", |     "copiedToClipboard": "Copied to clipboard", | ||||||
|     "storagePermissionDenied": "Storage permission denied", |     "storagePermissionDenied": "Storage permission denied", | ||||||
|     "selectedCategorizeWarning": "This will replace any existing category settings for the selected Apps.", |     "selectedCategorizeWarning": "This will replace any existing category settings for the selected apps.", | ||||||
|     "filterAPKsByRegEx": "Filter APKs by Regular Expression", |     "filterAPKsByRegEx": "Filter APKs by regular expression", | ||||||
|     "removeFromObtainium": "Remove from Obtainium", |     "removeFromObtainium": "Remove from Obtainium", | ||||||
|     "uninstallFromDevice": "Uninstall from Device", |     "uninstallFromDevice": "Uninstall from device", | ||||||
|     "onlyWorksWithNonVersionDetectApps": "Only works for Apps with version detection disabled.", |     "onlyWorksWithNonVersionDetectApps": "Only works for apps with version detection disabled.", | ||||||
|     "releaseDateAsVersion": "Use release date as version string", |     "releaseDateAsVersion": "Use release date as version string", | ||||||
|     "releaseTitleAsVersion": "Use release title as version string", |     "releaseTitleAsVersion": "Use release title as version string", | ||||||
|     "releaseDateAsVersionExplanation": "This option should only be used for Apps where version detection does not work correctly, but a release date is available.", |     "releaseDateAsVersionExplanation": "This option should only be used for apps where version detection does not work correctly, but a release date is available.", | ||||||
|     "changes": "Changes", |     "changes": "Changes", | ||||||
|     "releaseDate": "Release Date", |     "releaseDate": "Release date", | ||||||
|     "importFromURLsInFile": "Import from URLs in File (like OPML)", |     "importFromURLsInFile": "Import from URLs in file (like OPML)", | ||||||
|     "versionDetectionExplanation": "Reconcile version string with version detected from OS", |     "versionDetectionExplanation": "Reconcile version string with version detected from OS", | ||||||
|     "versionDetection": "Version Detection", |     "versionDetection": "Version detection", | ||||||
|     "standardVersionDetection": "Standard version detection", |     "standardVersionDetection": "Standard version detection", | ||||||
|     "groupByCategory": "Group by Category", |     "groupByCategory": "Group by category", | ||||||
|     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", |     "autoApkFilterByArch": "Attempt to filter APKs by CPU architecture if possible", | ||||||
|     "overrideSource": "Override Source", |     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||||
|  |     "overrideSource": "Override source", | ||||||
|     "dontShowAgain": "Don't show this again", |     "dontShowAgain": "Don't show this again", | ||||||
|     "dontShowTrackOnlyWarnings": "Don't show 'Track-Only' warnings", |     "dontShowTrackOnlyWarnings": "Don't show 'track-only' warnings", | ||||||
|     "dontShowAPKOriginWarnings": "Don't show APK origin warnings", |     "dontShowAPKOriginWarnings": "Don't show APK origin warnings", | ||||||
|     "moveNonInstalledAppsToBottom": "Move non-installed Apps to bottom of Apps view", |     "moveNonInstalledAppsToBottom": "Move non-installed apps to bottom of apps view", | ||||||
|     "gitlabPATLabel": "GitLab Personal Access Token", |     "gitlabPATLabel": "GitLab personal access token", | ||||||
|     "about": "About", |     "about": "About", | ||||||
|     "requiresCredentialsInSettings": "{} needs additional credentials (in Settings)", |     "requiresCredentialsInSettings": "{} needs additional credentials (in Settings)", | ||||||
|     "checkOnStart": "Check for updates on startup", |     "checkOnStart": "Check for updates on startup", | ||||||
|     "tryInferAppIdFromCode": "Try inferring App ID from source code", |     "tryInferAppIdFromCode": "Try inferring app ID from source code", | ||||||
|     "removeOnExternalUninstall": "Automatically remove externally uninstalled Apps", |     "removeOnExternalUninstall": "Automatically remove externally uninstalled apps", | ||||||
|     "pickHighestVersionCode": "Auto-select highest version code APK", |     "pickHighestVersionCode": "Auto-select highest version code APK", | ||||||
|     "checkUpdateOnDetailPage": "Check for updates on opening an App detail page", |     "checkUpdateOnDetailPage": "Check for updates on opening an app detail page", | ||||||
|     "disablePageTransitions": "Disable page transition animations", |     "disablePageTransitions": "Disable page transition animations", | ||||||
|     "reversePageTransitions": "Reverse page transition animations", |     "reversePageTransitions": "Reverse page transition animations", | ||||||
|     "minStarCount": "Minimum Star Count", |     "minStarCount": "Minimum star count", | ||||||
|     "addInfoBelow": "Add this info below.", |     "addInfoBelow": "Add this info below.", | ||||||
|     "addInfoInSettings": "Add this info in the Settings.", |     "addInfoInSettings": "Add this info in the Settings.", | ||||||
|     "githubSourceNote": "GitHub rate limiting can be avoided using an API key.", |     "githubSourceNote": "GitHub rate limiting can be avoided using an API key.", | ||||||
|     "sortByLastLinkSegment": "Sort by only the last segment of the link", |     "sortByLastLinkSegment": "Sort by only the last segment of the link", | ||||||
|     "filterReleaseNotesByRegEx": "Filter Release Notes by Regular Expression", |     "filterReleaseNotesByRegEx": "Filter release notes by regular expression", | ||||||
|     "customLinkFilterRegex": "Custom APK Link Filter by Regular Expression (Default '.apk$')", |     "customLinkFilterRegex": "Custom APK link filter by regular expression (default '.apk$')", | ||||||
|     "appsPossiblyUpdated": "App Updates Attempted", |     "appsPossiblyUpdated": "App updates attempted", | ||||||
|     "appsPossiblyUpdatedNotifDescription": "Notifies the user that updates to one or more Apps were potentially applied in the background", |     "appsPossiblyUpdatedNotifDescription": "Notifies the user that updates to one or more apps were potentially applied in the background", | ||||||
|     "xWasPossiblyUpdatedToY": "{} may have been updated to {}.", |     "xWasPossiblyUpdatedToY": "{} may have been updated to {}.", | ||||||
|     "enableBackgroundUpdates": "Enable background updates", |     "enableBackgroundUpdates": "Enable background updates", | ||||||
|     "backgroundUpdateReqsExplanation": "Background updates may not be possible for all apps.", |     "backgroundUpdateReqsExplanation": "Background updates may not be possible for all apps.", | ||||||
|     "backgroundUpdateLimitsExplanation": "The success of a background install can only be determined when Obtainium is opened.", |     "backgroundUpdateLimitsExplanation": "The success of a background install can only be determined when Obtainium is opened.", | ||||||
|     "verifyLatestTag": "Verify the 'latest' tag", |     "verifyLatestTag": "Verify the 'latest' tag", | ||||||
|     "intermediateLinkRegex": "Filter for an 'Intermediate' Link to Visit", |     "intermediateLinkRegex": "Filter for an 'intermediate' link to visit", | ||||||
|     "filterByLinkText": "Filter links by link text", |     "filterByLinkText": "Filter links by link text", | ||||||
|     "intermediateLinkNotFound": "Intermediate link not found", |     "intermediateLinkNotFound": "Intermediate link not found", | ||||||
|     "intermediateLink": "Intermediate link", |     "intermediateLink": "Intermediate link", | ||||||
|     "exemptFromBackgroundUpdates": "Exempt from background updates (if enabled)", |     "exemptFromBackgroundUpdates": "Exempt from background updates (if enabled)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Disable background updates when not on WiFi", |     "bgUpdatesOnWiFiOnly": "Disable background updates when not on Wi-Fi", | ||||||
|     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", |     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", | ||||||
|     "autoSelectHighestVersionCode": "Auto-select highest versionCode APK", |     "autoSelectHighestVersionCode": "Auto-select highest versionCode APK", | ||||||
|     "versionExtractionRegEx": "Version String Extraction RegEx", |     "versionExtractionRegEx": "Version string extraction RegEx", | ||||||
|     "trimVersionString": "Trim Version String With RegEx", |     "trimVersionString": "Trim version string with RegEx", | ||||||
|     "matchGroupToUseForX": "Match Group to Use for \"{}\"", |     "matchGroupToUseForX": "Match group to use for \"{}\"", | ||||||
|     "matchGroupToUse": "Match Group to Use for Version String Extraction RegEx", |     "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": "Automatically export on changes", | ||||||
|     "includeSettings": "Include settings", |     "includeSettings": "Include settings", | ||||||
|     "filterVersionsByRegEx": "Filter Versions by Regular Expression", |     "filterVersionsByRegEx": "Filter versions by regular expression", | ||||||
|     "trySelectingSuggestedVersionCode": "Try selecting suggested versionCode APK", |     "trySelectingSuggestedVersionCode": "Try selecting suggested versionCode APK", | ||||||
|     "dontSortReleasesList": "Retain release order from API", |     "dontSortReleasesList": "Retain release order from API", | ||||||
|     "reverseSort": "Reverse sorting", |     "reverseSort": "Reverse sorting", | ||||||
|     "takeFirstLink": "Take first link", |     "takeFirstLink": "Take first link", | ||||||
|     "skipSort": "Skip sorting", |     "skipSort": "Skip sorting", | ||||||
|     "debugMenu": "Debug Menu", |     "debugMenu": "Debug menu", | ||||||
|     "bgTaskStarted": "Background task started - check logs.", |     "bgTaskStarted": "Background task started - check logs.", | ||||||
|     "runBgCheckNow": "Run Background Update Check Now", |     "runBgCheckNow": "Run background update check now", | ||||||
|     "versionExtractWholePage": "Apply version string extraction Regex to entire page", |     "versionExtractWholePage": "Apply version string extraction Regex to entire page", | ||||||
|     "installing": "Installing", |     "installing": "Installing", | ||||||
|     "skipUpdateNotifications": "Skip update notifications", |     "skipUpdateNotifications": "Skip update notifications", | ||||||
|     "updatesAvailableNotifChannel": "Updates Available", |     "updatesAvailableNotifChannel": "Updates available", | ||||||
|     "appsUpdatedNotifChannel": "Apps Updated", |     "appsUpdatedNotifChannel": "Apps updated", | ||||||
|     "appsPossiblyUpdatedNotifChannel": "App Updates Attempted", |     "appsPossiblyUpdatedNotifChannel": "App updates attempted", | ||||||
|     "errorCheckingUpdatesNotifChannel": "Error Checking for Updates", |     "errorCheckingUpdatesNotifChannel": "Error checking for updates", | ||||||
|     "appsRemovedNotifChannel": "Apps Removed", |     "appsRemovedNotifChannel": "Apps removed", | ||||||
|     "downloadingXNotifChannel": "Downloading {}", |     "downloadingXNotifChannel": "Downloading {}", | ||||||
|     "completeAppInstallationNotifChannel": "Complete App Installation", |     "completeAppInstallationNotifChannel": "Complete app installation", | ||||||
|     "checkingForUpdatesNotifChannel": "Checking for Updates", |     "checkingForUpdatesNotifChannel": "Checking for updates", | ||||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Only check installed and Track-Only apps for updates", |     "onlyCheckInstalledOrTrackOnlyApps": "Only check installed and track-only apps for updates", | ||||||
|     "supportFixedAPKURL": "Support fixed APK URLs", |     "supportFixedAPKURL": "Support fixed APK URLs", | ||||||
|     "selectX": "Select {}", |     "selectX": "Select {}", | ||||||
|     "parallelDownloads": "Allow parallel downloads", |     "parallelDownloads": "Allow parallel downloads", | ||||||
| @@ -299,29 +300,39 @@ | |||||||
|     "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", | ||||||
|     "defaultPseudoVersioningMethod": "Default Pseudo-Versioning Method", |     "defaultPseudoVersioningMethod": "Default pseudo-versioning method", | ||||||
|     "partialAPKHash": "Partial APK Hash", |     "partialAPKHash": "Partial APK hash", | ||||||
|     "APKLinkHash": "APK Link Hash", |     "APKLinkHash": "APK link hash", | ||||||
|     "directAPKLink": "Direct APK Link", |     "directAPKLink": "Direct APK link", | ||||||
|     "pseudoVersionInUse": "A Pseudo-Version is in Use", |     "pseudoVersionInUse": "A pseudo-version is in use", | ||||||
|     "installed": "Installed", |     "installed": "Installed", | ||||||
|     "latest": "Latest", |     "latest": "Latest", | ||||||
|     "invertRegEx": "Invert regular expression", |     "invertRegEx": "Invert regular expression", | ||||||
|     "note": "Note", |     "note": "Note", | ||||||
|     "selfHostedNote": "The \"{}\" dropdown can be used to reach self-hosted/custom instances of any source.", |     "selfHostedNote": "The \"{}\" dropdown can be used to reach self-hosted/custom instances of any source.", | ||||||
|     "badDownload": "The APK could not be parsed (incompatible or partial download)", |     "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", |     "wiki": "Help/Wiki", | ||||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", |     "crowdsourcedConfigsLabel": "Crowdsourced app configurations (use at your own risk)", | ||||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", |     "crowdsourcedConfigsShort": "Crowdsourced app configurations", | ||||||
|     "allowInsecure": "Allow insecure HTTP requests", |     "allowInsecure": "Allow insecure HTTP requests", | ||||||
|     "stayOneVersionBehind": "Stay one version behind latest", |     "stayOneVersionBehind": "Stay one version behind latest", | ||||||
|  |     "useFirstApkOfVersion": "Auto-select first of multiple APKs", | ||||||
|     "refreshBeforeDownload": "Refresh app details before download", |     "refreshBeforeDownload": "Refresh app details before download", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Name", | ||||||
|  |     "smartname": "Name (smart)", | ||||||
|  |     "sortMethod": "Sort method", | ||||||
|  |     "welcome": "Welcome", | ||||||
|  |     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||||
|  |     "batteryOptimizationNote": "Note that background downloads may work more reliably if you disable OS battery optimizations for Obtainium.", | ||||||
|  |     "fileDeletionError": "Failed to delete file (try deleting it manually then try again): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Remove App?", |         "one": "Remove app?", | ||||||
|         "other": "Remove Apps?" |         "other": "Remove apps?" | ||||||
|     }, |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Too many requests (rate limited) - try again in {} minute", |         "one": "Too many requests (rate limited) - try again in {} minute", | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Por versión", |     "standardVersionDetection": "Por versión", | ||||||
|     "groupByCategory": "Agrupar por categorías", |     "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)", | ||||||
|  |     "autoLinkFilterByArch": "Intente filtrar los enlaces por arquitectura de CPU si es posible", | ||||||
|     "overrideSource": "Forzar desde la fuente", |     "overrideSource": "Forzar desde la fuente", | ||||||
|     "dontShowAgain": "No mostrar de nuevo", |     "dontShowAgain": "No mostrar de nuevo", | ||||||
|     "dontShowTrackOnlyWarnings": "No mostrar avisos sobre apps 'solo para seguimiento'", |     "dontShowTrackOnlyWarnings": "No mostrar avisos sobre apps 'solo para seguimiento'", | ||||||
| @@ -255,7 +256,7 @@ | |||||||
|     "intermediateLinkNotFound": "Enlace intermedio no encontrado", |     "intermediateLinkNotFound": "Enlace intermedio no encontrado", | ||||||
|     "intermediateLink": "Enlace intermedio", |     "intermediateLink": "Enlace intermedio", | ||||||
|     "exemptFromBackgroundUpdates": "Exenta de actualizciones en segundo plano (si están habilitadas)", |     "exemptFromBackgroundUpdates": "Exenta de actualizciones en segundo plano (si están habilitadas)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Deshabilitar las actualizaciones en segundo plano sin WiFi", |     "bgUpdatesOnWiFiOnly": "Deshabilitar las actualizaciones en segundo plano sin Wi-Fi", | ||||||
|     "bgUpdatesWhileChargingOnly": "Desactiva las actualizaciones en segundo plano cuando no estés cargando", |     "bgUpdatesWhileChargingOnly": "Desactiva las actualizaciones en segundo plano cuando no estés cargando", | ||||||
|     "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", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Configuración de aplicaciones por crowdsourcing", |     "crowdsourcedConfigsShort": "Configuración de aplicaciones por crowdsourcing", | ||||||
|     "allowInsecure": "Permitir peticiones HTTP inseguras", |     "allowInsecure": "Permitir peticiones HTTP inseguras", | ||||||
|     "stayOneVersionBehind": "Mantenerse una versión por detrás de la última", |     "stayOneVersionBehind": "Mantenerse una versión por detrás de la última", | ||||||
|  |     "useFirstApkOfVersion": "Auto-selección del primero de múltiples APKs", | ||||||
|     "refreshBeforeDownload": "Actualiza los datos de la aplicación antes de descargarla", |     "refreshBeforeDownload": "Actualiza los datos de la aplicación antes de descargarla", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Nombre", | ||||||
|  |     "smartname": "Nombre (Smart)", | ||||||
|  |     "sortMethod": "Método de clasificación", | ||||||
|  |     "welcome": "Bienvenido", | ||||||
|  |     "documentationLinksNote": "La página GitHub de Obtainium enlazada a continuación contiene enlaces a vídeos, artículos, debates y otros recursos que te ayudarán a entender cómo utilizar la aplicación.", | ||||||
|  |     "batteryOptimizationNote": "Ten en cuenta que las descargas en segundo plano pueden funcionar de forma más fiable si desactivas las optimizaciones de batería del sistema operativo para Obtainium.", | ||||||
|  |     "fileDeletionError": "No se ha podido eliminar el archivo (intente eliminarlo manualmente y vuelva a intentarlo): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "¿Eliminar aplicación?", |         "one": "¿Eliminar aplicación?", | ||||||
|         "other": "¿Eliminar aplicaciones?" |         "other": "¿Eliminar aplicaciones?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "تشخیص نسخه استاندارد", |     "standardVersionDetection": "تشخیص نسخه استاندارد", | ||||||
|     "groupByCategory": "گروه بر اساس دسته", |     "groupByCategory": "گروه بر اساس دسته", | ||||||
|     "autoApkFilterByArch": "در صورت امکان سعی کنید APKها را بر اساس معماری CPU فیلتر کنید", |     "autoApkFilterByArch": "در صورت امکان سعی کنید APKها را بر اساس معماری CPU فیلتر کنید", | ||||||
|  |     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||||
|     "overrideSource": "نادیده گرفتن منبع", |     "overrideSource": "نادیده گرفتن منبع", | ||||||
|     "dontShowAgain": "دوباره این را نشان نده", |     "dontShowAgain": "دوباره این را نشان نده", | ||||||
|     "dontShowTrackOnlyWarnings": "هشدار 'فقط ردیابی' را نشان ندهید", |     "dontShowTrackOnlyWarnings": "هشدار 'فقط ردیابی' را نشان ندهید", | ||||||
| @@ -255,7 +256,7 @@ | |||||||
|     "intermediateLinkNotFound": "لینک میانی پیدا نشد", |     "intermediateLinkNotFound": "لینک میانی پیدا نشد", | ||||||
|     "intermediateLink": "پیوند میانی", |     "intermediateLink": "پیوند میانی", | ||||||
|     "exemptFromBackgroundUpdates": "معاف از بهروزرسانیهای پسزمینه (در صورت فعال بودن)", |     "exemptFromBackgroundUpdates": "معاف از بهروزرسانیهای پسزمینه (در صورت فعال بودن)", | ||||||
|     "bgUpdatesOnWiFiOnly": "بهروزرسانیهای پسزمینه را در صورت عدم اتصال به WiFi غیرفعال کنید", |     "bgUpdatesOnWiFiOnly": "بهروزرسانیهای پسزمینه را در صورت عدم اتصال به Wi-Fi غیرفعال کنید", | ||||||
|     "bgUpdatesWhileChargingOnly": "بهروزرسانیهای پسزمینه را هنگام شارژ نشدن غیرفعال کنید", |     "bgUpdatesWhileChargingOnly": "بهروزرسانیهای پسزمینه را هنگام شارژ نشدن غیرفعال کنید", | ||||||
|     "autoSelectHighestVersionCode": "انتخاب خودکار بالاترین نسخه کد APK", |     "autoSelectHighestVersionCode": "انتخاب خودکار بالاترین نسخه کد APK", | ||||||
|     "versionExtractionRegEx": "نسخه استخراج RegEx", |     "versionExtractionRegEx": "نسخه استخراج RegEx", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "تنظیمات برنامه های مشارکت جمعی", |     "crowdsourcedConfigsShort": "تنظیمات برنامه های مشارکت جمعی", | ||||||
|     "allowInsecure": "درخواست های HTTP ناامن را مجاز کنید", |     "allowInsecure": "درخواست های HTTP ناامن را مجاز کنید", | ||||||
|     "stayOneVersionBehind": "یک نسخه از آخرین نسخه پشت سر بگذارید", |     "stayOneVersionBehind": "یک نسخه از آخرین نسخه پشت سر بگذارید", | ||||||
|  |     "useFirstApkOfVersion": "Auto-select first of multiple APKs", | ||||||
|     "refreshBeforeDownload": "قبل از دانلود، جزئیات برنامه را بازخوانی کنید", |     "refreshBeforeDownload": "قبل از دانلود، جزئیات برنامه را بازخوانی کنید", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Name", | ||||||
|  |     "smartname": "Name (Smart)", | ||||||
|  |     "sortMethod": "Sort Method", | ||||||
|  |     "welcome": "Welcome", | ||||||
|  |     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||||
|  |     "batteryOptimizationNote": "Note that background downloads may work more reliably if you disable OS battery optimizations for Obtainium.", | ||||||
|  |     "fileDeletionError": "Failed to delete file (try deleting it manually then try again): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "برنامه حذف شود؟", |         "one": "برنامه حذف شود؟", | ||||||
|         "other": "برنامه ها حذف شوند؟" |         "other": "برنامه ها حذف شوند؟" | ||||||
|   | |||||||
| @@ -9,11 +9,11 @@ | |||||||
|     "placeholder": "Espace réservé", |     "placeholder": "Espace réservé", | ||||||
|     "someErrors": "Des erreurs sont survenues", |     "someErrors": "Des erreurs sont survenues", | ||||||
|     "unexpectedError": "Erreur inattendue", |     "unexpectedError": "Erreur inattendue", | ||||||
|     "ok": "Ok", |     "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 versions préliminaires", |     "includePrereleases": "Inclure les versions préliminaires", | ||||||
|     "fallbackToOlderReleases": "Retour aux anciennes versions", |     "fallbackToOlderReleases": "Revenir 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": "Aucune description", |     "noDescription": "Aucune description", | ||||||
| @@ -30,13 +30,13 @@ | |||||||
|     "wrongArgNum": "Nombre incorrect des arguments fournis", |     "wrongArgNum": "Nombre incorrect des arguments fournis", | ||||||
|     "xIsTrackOnly": "{} en Suivi uniquement", |     "xIsTrackOnly": "{} en Suivi uniquement", | ||||||
|     "source": "source", |     "source": "source", | ||||||
|     "app": "Application", |     "app": "Appli", | ||||||
|     "appsFromSourceAreTrackOnly": "Les applications de cette source sont en 'Suivi uniquement'.", |     "appsFromSourceAreTrackOnly": "Les applications de cette source sont en 'Suivi uniquement'.", | ||||||
|     "youPickedTrackOnly": "Vous avez sélectionné l'option 'Suivi uniquement'.", |     "youPickedTrackOnly": "Vous avez sélectionné l'option 'Suivi uniquement'.", | ||||||
|     "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": "L'application est à jour?", |     "alreadyUpToDateQuestion": "L'application est déjà à jour?", | ||||||
|     "addApp": "Ajouter appli", |     "addApp": "Ajouter appli", | ||||||
|     "appSourceURL": "URL source de l'application", |     "appSourceURL": "URL source de l'application", | ||||||
|     "error": "Erreur", |     "error": "Erreur", | ||||||
| @@ -47,30 +47,30 @@ | |||||||
|     "supportedSources": "Sources prises en charge", |     "supportedSources": "Sources prises en charge", | ||||||
|     "trackOnlyInBrackets": "(Suivi uniquement)", |     "trackOnlyInBrackets": "(Suivi uniquement)", | ||||||
|     "searchableInBrackets": "(Interrogeable)", |     "searchableInBrackets": "(Interrogeable)", | ||||||
|     "appsString": "Applications", |     "appsString": "Applis", | ||||||
|     "noApps": "Aucune application", |     "noApps": "Aucune application", | ||||||
|     "noAppsForFilter": "Aucune application à filtrer", |     "noAppsForFilter": "Aucune application correspondant au filtre", | ||||||
|     "byX": "Par {}", |     "byX": "Par {}", | ||||||
|     "percentProgress": "Progression : {}%", |     "percentProgress": "Progression : {}%", | ||||||
|     "pleaseWait": "Veuillez patienter", |     "pleaseWait": "Veuillez patienter", | ||||||
|     "updateAvailable": "Mise à jour disponible", |     "updateAvailable": "Mise à jour disponible", | ||||||
|     "notInstalled": "Non installé", |     "notInstalled": "Non installée", | ||||||
|     "pseudoVersion": "Version fictive", |     "pseudoVersion": "Version fictive", | ||||||
|     "selectAll": "Tout sélectionner", |     "selectAll": "Tout sélectionner", | ||||||
|     "deselectX": "Déselectionner {}", |     "deselectX": "Désélectionner {}", | ||||||
|     "xWillBeRemovedButRemainInstalled": "{} sera supprimée d'Obtainium mais restera installée 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": "Supprimer les applications sélectionnées", | ||||||
|     "updateX": "Mettre à jour {}", |     "updateX": "Mettre à jour {}", | ||||||
|     "installX": "Installer {}", |     "installX": "Installer {}", | ||||||
|     "markXTrackOnlyAsUpdated": "Marquer {}\n(Suivi uniquement)\ncomme étant à jour", |     "markXTrackOnlyAsUpdated": "Marquer {}\n(Suivi uniquement)\ncomme étant à jour", | ||||||
|     "changeX": "Changer {}", |     "changeX": "Modifier {}", | ||||||
|     "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 étant à jour ?", |     "markXSelectedAppsAsUpdated": "Marquer les {} applications sélectionnées comme étant à jour ?", | ||||||
|     "no": "Non", |     "no": "Non", | ||||||
|     "yes": "Oui", |     "yes": "Oui", | ||||||
|     "markSelectedAppsUpdated": "Marquer les application sélectionnées comme étant à jour", |     "markSelectedAppsUpdated": "Marquer les applications sélectionnées comme étant à jour", | ||||||
|     "pinToTop": "Épingler en haut", |     "pinToTop": "Épingler en haut", | ||||||
|     "unpinFromTop": "Désépingler 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 ?", | ||||||
| @@ -88,7 +88,7 @@ | |||||||
|     "author": "Auteur", |     "author": "Auteur", | ||||||
|     "upToDateApps": "Applications à jour", |     "upToDateApps": "Applications à jour", | ||||||
|     "nonInstalledApps": "Applications non installées", |     "nonInstalledApps": "Applications non installées", | ||||||
|     "importExport": "Importer/exporter", |     "importExport": "Import/Export", | ||||||
|     "settings": "Paramètres", |     "settings": "Paramètres", | ||||||
|     "exportedTo": "Exporté vers {}", |     "exportedTo": "Exporté vers {}", | ||||||
|     "obtainiumExport": "Exporter la configuration d'Obtainium", |     "obtainiumExport": "Exporter la configuration d'Obtainium", | ||||||
| @@ -97,24 +97,24 @@ | |||||||
|     "obtainiumImport": "Importer la configuration sur Obtainium", |     "obtainiumImport": "Importer la configuration sur Obtainium", | ||||||
|     "importFromURLList": "Importer depuis une liste d'URL", |     "importFromURLList": "Importer depuis une liste d'URL", | ||||||
|     "searchQuery": "Requête de recherche", |     "searchQuery": "Requête de recherche", | ||||||
|     "appURLList": "Liste d'URL de l'application", |     "appURLList": "Liste d'URL des applications", | ||||||
|     "line": "Ligne", |     "line": "Ligne", | ||||||
|     "searchX": "Sélectionner {}", |     "searchX": "Rechercher {}", | ||||||
|     "noResults": "Aucun résultat", |     "noResults": "Aucun résultat", | ||||||
|     "importX": "Importation de {}", |     "importX": "Importation de {}", | ||||||
|     "importedAppsIdDisclaimer": "Les applications importées peuvent s'afficher de manière incorrecte comme étant \"Non installées\".\nPour résoudre ce problème, réinstallez-les via Obtainium.\nCela n'affectera pas les données des applications.\n\nN'affecte que les méthodes d'importation d'URL et par des tiers.", |     "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\nCela n'affecte que les méthodes d'importation d'URL et par des tiers.", | ||||||
|     "importErrors": "Erreurs lors de l'importation", |     "importErrors": "Erreurs lors de l'importation", | ||||||
|     "importedXOfYApps": "{} applications sur {} ont été importés.", |     "importedXOfYApps": "{} applications sur {} ont été importées.", | ||||||
|     "followingURLsHadErrors": "Les URL suivants comportent des erreurs :", |     "followingURLsHadErrors": "Les URL suivantes comportent des erreurs :", | ||||||
|     "selectURL": "Sélectionner l'URL", |     "selectURL": "Sélectionner l'URL", | ||||||
|     "selectURLs": "Sélectionner les URL", |     "selectURLs": "Sélectionner les URL", | ||||||
|     "pick": "Choisir", |     "pick": "Choisir", | ||||||
|     "theme": "Thème", |     "theme": "Thème", | ||||||
|     "dark": "Sombre", |     "dark": "Sombre", | ||||||
|     "light": "Clair", |     "light": "Clair", | ||||||
|     "followSystem": "Correspondre au système", |     "followSystem": "Suivre le système", | ||||||
|     "followSystemThemeExplanation": "Correspondre au thème du système est possible en utilisant des applications tierces.", |     "followSystemThemeExplanation": "Suivre le thème du système est possible en utilisant des applications tierces.", | ||||||
|     "useBlackTheme": "Utiliser un thème Noir", |     "useBlackTheme": "Utiliser un thème noir", | ||||||
|     "appSortBy": "Trier les applications par", |     "appSortBy": "Trier les applications par", | ||||||
|     "authorName": "Auteur/Nom", |     "authorName": "Auteur/Nom", | ||||||
|     "nameAuthor": "Nom/Auteur", |     "nameAuthor": "Nom/Auteur", | ||||||
| @@ -136,10 +136,10 @@ | |||||||
|     "share": "Partager", |     "share": "Partager", | ||||||
|     "appNotFound": "Application introuvable", |     "appNotFound": "Application introuvable", | ||||||
|     "obtainiumExportHyphenatedLowercase": "export-obtainium", |     "obtainiumExportHyphenatedLowercase": "export-obtainium", | ||||||
|     "pickAnAPK": "Selectionner un APK", |     "pickAnAPK": "Sélectionner un APK", | ||||||
|     "appHasMoreThanOnePackage": "{} a plus d'un paquet:", |     "appHasMoreThanOnePackage": "{} a plus d'un paquet :", | ||||||
|     "deviceSupportsXArch": "Votre appareil prend en charge l'architecture CPU {}.", |     "deviceSupportsXArch": "Votre appareil prend en charge l'architecture CPU {}.", | ||||||
|     "deviceSupportsFollowingArchs": "Votre appareil prend en charge les architectures CPU suivants: ", |     "deviceSupportsFollowingArchs": "Votre appareil prend en charge les architectures CPU suivantes : ", | ||||||
|     "warning": "Avertissement", |     "warning": "Avertissement", | ||||||
|     "sourceIsXButPackageFromYPrompt": "La source de l'application est '{}' mais le paquet de mise à jour 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", | ||||||
| @@ -149,7 +149,7 @@ | |||||||
|     "appsUpdated": "Applications mises à jour", |     "appsUpdated": "Applications mises à jour", | ||||||
|     "appsNotUpdated": "Échec de la mise à jour des applications", |     "appsNotUpdated": "Échec de la mise à jour des applications", | ||||||
|     "appsUpdatedNotifDescription": "Notifie à l'utilisateur que des mises à jour d'une ou plusieurs applications ont été installées en arrière-plan.", |     "appsUpdatedNotifDescription": "Notifie à l'utilisateur que des mises à jour d'une ou plusieurs applications ont été installées en arrière-plan.", | ||||||
|     "xWasUpdatedToY": "{} a été mis à jour en {}.", |     "xWasUpdatedToY": "{} a été mise à jour vers {}.", | ||||||
|     "xWasNotUpdatedToY": "Échec de la mise à jour de {} vers {}.", |     "xWasNotUpdatedToY": "Échec de la mise à jour de {} vers {}.", | ||||||
|     "errorCheckingUpdates": "Erreur lors de la recherche de mises à jour", |     "errorCheckingUpdates": "Erreur lors de la recherche de mises à jour", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Notifie l'utilisateur lorsque la recherche de mises à jour en arrière-plan échoue.", |     "errorCheckingUpdatesNotifDescription": "Notifie l'utilisateur lorsque la recherche de mises à jour en arrière-plan échoue.", | ||||||
| @@ -180,26 +180,26 @@ | |||||||
|     "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", | ||||||
|     "install": "Installer", |     "install": "Installer", | ||||||
|     "markInstalled": "Marquer comme étant installé", |     "markInstalled": "Marquer comme installée", | ||||||
|     "update": "Mettre à jour", |     "update": "Mettre à jour", | ||||||
|     "markUpdated": "Marquer comme étant à jour", |     "markUpdated": "Marquer comme à jour", | ||||||
|     "additionalOptions": "Options supplémentaires", |     "additionalOptions": "Options supplémentaires", | ||||||
|     "disableVersionDetection": "Désactiver la détection de la 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 la 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 de {}", | ||||||
|     "downloadX": "Télécharger {}", |     "downloadX": "Télécharger {}", | ||||||
|     "downloadedX": "Téléchargé {}", |     "downloadedX": "Téléchargé {}", | ||||||
|     "releaseAsset": "Version active", |     "releaseAsset": "Élément de version", | ||||||
|     "downloadNotifDescription": "Notifie l'utilisateur sur l'avancement 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": "Aucune version trouvée", |     "noVersionDetection": "Aucune détection de version", | ||||||
|     "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égories", |     "noCategories": "Aucune catégorie", | ||||||
|     "deleteCategoriesQuestion": "Supprimer les catégories?", |     "deleteCategoriesQuestion": "Supprimer les catégories ?", | ||||||
|     "categoryDeleteWarning": "Toutes les applications des catégories supprimées seront définies comme 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": "Nom", |     "label": "Nom", | ||||||
|     "language": "Langue", |     "language": "Langue", | ||||||
| @@ -209,26 +209,27 @@ | |||||||
|     "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": "Ne fonctionne que pour les applications dont la détection de la version est désactivée.", |     "onlyWorksWithNonVersionDetectApps": "Ne fonctionne qu'avec 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", | ||||||
|     "releaseTitleAsVersion": "Utiliser le titre de la version comme chaîne de caractères de la version", |     "releaseTitleAsVersion": "Utiliser le titre de la version comme chaîne de version", | ||||||
|     "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.", |     "releaseDateAsVersionExplanation": "Cette option ne doit être utilisée que pour les applications pour lesquelles la détection de la version ne fonctionne pas correctement, mais dont une date de sortie est disponible.", | ||||||
|     "changes": "Modifications", |     "changes": "Modifications", | ||||||
|     "releaseDate": "Date de sortie", |     "releaseDate": "Date de sortie", | ||||||
|     "importFromURLsInFile": "Importer à partir des URLs d'un fichier (comme OPML)", |     "importFromURLsInFile": "Importer à partir des URLs d'un fichier (comme OPML)", | ||||||
|     "versionDetectionExplanation": "Reporter la chaîne de version selon la version détectée par le système d'exploitation", |     "versionDetectionExplanation": "Réconcilier la chaîne de version avec la version détectée par le système d'exploitation", | ||||||
|     "versionDetection": "Détection de la version", |     "versionDetection": "Détection de la version", | ||||||
|     "standardVersionDetection": "Détection de la version standard", |     "standardVersionDetection": "Détection de la version standard", | ||||||
|     "groupByCategory": "Grouper par catégorie", |     "groupByCategory": "Grouper par catégorie", | ||||||
|     "autoApkFilterByArch": "Essayer de filtrer les APKs par architecture CPU si possible", |     "autoApkFilterByArch": "Essayer de filtrer les APKs par architecture CPU si possible", | ||||||
|  |     "autoLinkFilterByArch": "Essayer de filtrer les liens en fonction de l'architecture de l'unité centrale si possible", | ||||||
|     "overrideSource": "Remplacer la source", |     "overrideSource": "Remplacer la source", | ||||||
|     "dontShowAgain": "Ne plus afficher", |     "dontShowAgain": "Ne plus afficher", | ||||||
|     "dontShowTrackOnlyWarnings": "Ne plus afficher les erreurs 'Suivi uniquement'", |     "dontShowTrackOnlyWarnings": "Ne plus afficher les avertissements 'Suivi uniquement'", | ||||||
|     "dontShowAPKOriginWarnings": "Ne plus afficher les erreurs sur l'origine de l'APK", |     "dontShowAPKOriginWarnings": "Ne plus afficher les avertissements 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", |     "about": "À propos", | ||||||
|     "requiresCredentialsInSettings": "{} a besoin d'un complément d'information (dans les Paramètres)", |     "requiresCredentialsInSettings": "{} a besoin d'informations d'identification supplémentaires (dans les Paramètres)", | ||||||
|     "checkOnStart": "Rechercher les mises à jour au démarrage", |     "checkOnStart": "Rechercher les mises à jour au démarrage", | ||||||
|     "tryInferAppIdFromCode": "Essayer de déduire l'identifiant 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", | ||||||
| @@ -243,67 +244,67 @@ | |||||||
|     "sortByLastLinkSegment": "Trier par 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 des applications", |     "appsPossiblyUpdated": "Tentatives de mise à jour d'applications", | ||||||
|     "appsPossiblyUpdatedNotifDescription": "Notifie à l'utilisateur que des mises à jour d'une ou plusieurs applications ont potentiellement été 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": "{} peut être mis à jour en {}.", |     "xWasPossiblyUpdatedToY": "{} a peut-être été mise à jour vers {}.", | ||||||
|     "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 résultat d'une installation en arrière-plan ne peut être déterminé qu'à l'ouverture d'Obtainium.", |     "backgroundUpdateLimitsExplanation": "Le succès d'une installation en arrière-plan ne peut être déterminé qu'à l'ouverture d'Obtainium.", | ||||||
|     "verifyLatestTag": "Vérifier la balise 'latest'", |     "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 du 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": "Exclure de la mise à jour en arrière-plan (si activé)", |     "exemptFromBackgroundUpdates": "Exclure des mises à jour en arrière-plan (si activées)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Désactiver les mises à jour en arrière-plan lorsque vous n'êtes pas en WiFi", |     "bgUpdatesOnWiFiOnly": "Désactiver les mises à jour en arrière-plan lorsque vous n'êtes pas en Wi-Fi", | ||||||
|     "bgUpdatesWhileChargingOnly": "Désactiver les mises à jour en arrière-plan lorsque le véhicule n'est pas en charge", |     "bgUpdatesWhileChargingOnly": "Désactiver les mises à jour en arrière-plan lorsque l'appareil n'est pas en charge", | ||||||
|     "autoSelectHighestVersionCode": "Sélectionner automatiquement la version la plus récente du code APK", |     "autoSelectHighestVersionCode": "Sélectionner automatiquement la version la plus récente du code APK", | ||||||
|     "versionExtractionRegEx": "Extraire la version par Expression régulière", |     "versionExtractionRegEx": "Expression régulière d'extraction de version", | ||||||
|     "trimVersionString": "Découper la version par Expression régulière", |     "trimVersionString": "Découper la chaîne de version avec une expression régulière", | ||||||
|     "matchGroupToUseForX": "Groupe de correspondance à utiliser pour \"{}\"", |     "matchGroupToUseForX": "Groupe de correspondance à utiliser pour \"{}\"", | ||||||
|     "matchGroupToUse": "Groupe de correspondance à utiliser pour l'extraction de la version par Expression régulière", |     "matchGroupToUse": "Groupe de correspondance à utiliser pour l'extraction de la version par expression régulière", | ||||||
|     "highlightTouchTargets": "Mettre en évidence les touches moins évidentes", |     "highlightTouchTargets": "Mettre en évidence les zones tactiles moins évidentes", | ||||||
|     "pickExportDir": "Selectionner le dossier d'exportation", |     "pickExportDir": "Sélectionner le dossier d'exportation", | ||||||
|     "autoExportOnChanges": "Exporter automatiquement lors de modifications", |     "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": "Essayer de sélectionner la version suggérée du code APK", |     "trySelectingSuggestedVersionCode": "Essayer de sélectionner le code de version APK suggéré", | ||||||
|     "dontSortReleasesList": "Conserver l'ordre de la version de l'API", |     "dontSortReleasesList": "Conserver l'ordre de version de l'API", | ||||||
|     "reverseSort": "Tri inversé", |     "reverseSort": "Tri inversé", | ||||||
|     "takeFirstLink": "Utiliser le premier lien", |     "takeFirstLink": "Utiliser le premier lien", | ||||||
|     "skipSort": "Ignorer le tri", |     "skipSort": "Ignorer le tri", | ||||||
|     "debugMenu": "Menu de déboggage", |     "debugMenu": "Menu de débogage", | ||||||
|     "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 recherche de mise à jour en arrière-plan maintenant", |     "runBgCheckNow": "Exécuter la recherche de mises à jour en arrière-plan maintenant", | ||||||
|     "versionExtractWholePage": "Appliquer l'extraction de la version par expression régulière à l'ensemble de la page", |     "versionExtractWholePage": "Appliquer l'expression régulière d'extraction de version à l'ensemble de la page", | ||||||
|     "installing": "Installation", |     "installing": "Installation", | ||||||
|     "skipUpdateNotifications": "Ignorer les notifications de mise à jour", |     "skipUpdateNotifications": "Désactiver les notifications de mise à jour", | ||||||
|     "updatesAvailableNotifChannel": "Mises à jour disponibles", |     "updatesAvailableNotifChannel": "Mises à jour disponibles", | ||||||
|     "appsUpdatedNotifChannel": "Applications mises à jour", |     "appsUpdatedNotifChannel": "Applications mises à jour", | ||||||
|     "appsPossiblyUpdatedNotifChannel": "Essayer de mettre à jour les applications", |     "appsPossiblyUpdatedNotifChannel": "Tentatives de mise à jour d'applications", | ||||||
|     "errorCheckingUpdatesNotifChannel": "Erreur lors de la recherche de 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 de {}", | ||||||
|     "completeAppInstallationNotifChannel": "Installation complète de l'application", |     "completeAppInstallationNotifChannel": "Terminer l'installation de l'application", | ||||||
|     "checkingForUpdatesNotifChannel": "Recherche de mises à jour", |     "checkingForUpdatesNotifChannel": "Recherche de mises à jour", | ||||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Rechercher uniquement les mises à jour des applications installées et des applications '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 simultanés", |     "parallelDownloads": "Autoriser les téléchargements simultanés", | ||||||
|     "useShizuku": "Utiliser Shizuku ou Sui pour l'installation", |     "useShizuku": "Utiliser Shizuku ou Sui pour l'installation", | ||||||
|     "shizukuBinderNotFound": "Le service Shizuku est introuvable", |     "shizukuBinderNotFound": "Le service Shizuku n'est pas en cours d'exécution", | ||||||
|     "shizukuOld": "Ancienne version de Shizuku (<11) - veuillez le mettre à jour", |     "shizukuOld": "Ancienne version de Shizuku (<11) - veuillez le mettre à jour", | ||||||
|     "shizukuOldAndroidWithADB": "Shizuku fonctionne sur Android < 8.1 avec ADB - veuillez mettre à jour Android ou utiliser Sui à la place", |     "shizukuOldAndroidWithADB": "Shizuku fonctionne sur Android < 8.1 avec ADB - veuillez mettre à jour Android ou utiliser Sui à la place", | ||||||
|     "shizukuPretendToBeGooglePlay": "Définir Google Play comme source d'installation (si Shizuku est utilisé)", |     "shizukuPretendToBeGooglePlay": "Définir Google Play comme source d'installation (si Shizuku est utilisé)", | ||||||
|     "useSystemFont": "Utiliser la police du système", |     "useSystemFont": "Utiliser la police du système", | ||||||
|     "useVersionCodeAsOSVersion": "Utiliser le code de version de l'application 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": "Intitulé de la demande", |     "requestHeader": "En-tête de requête", | ||||||
|     "useLatestAssetDateAsReleaseDate": "Utiliser le dernier élément mis en ligne comme date de sortie", |     "useLatestAssetDateAsReleaseDate": "Utiliser la date du dernier élément mis en ligne comme date de sortie", | ||||||
|     "defaultPseudoVersioningMethod": "Méthode de version fictive par défaut", |     "defaultPseudoVersioningMethod": "Méthode de versionnage fictif par défaut", | ||||||
|     "partialAPKHash": "Hash partiel de l'APK", |     "partialAPKHash": "Hash partiel de l'APK", | ||||||
|     "APKLinkHash": "Hash du lien APK", |     "APKLinkHash": "Hash du lien APK", | ||||||
|     "directAPKLink": "Lien direct de l'APK", |     "directAPKLink": "Lien direct de l'APK", | ||||||
|     "pseudoVersionInUse": "Version fictive utilisé", |     "pseudoVersionInUse": "Une version fictive est utilisée", | ||||||
|     "installed": "Installée", |     "installed": "Installée", | ||||||
|     "latest": "Dernière version", |     "latest": "Dernière version", | ||||||
|     "invertRegEx": "Inverser l'expression régulière", |     "invertRegEx": "Inverser l'expression régulière", | ||||||
| @@ -313,27 +314,37 @@ | |||||||
|     "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 tout est prêt.", |     "appVerifierInstructionToast": "Partagez avec AppVerifier, puis revenez ici lorsque tout est prêt.", | ||||||
|     "wiki": "Aide/Wiki", |     "wiki": "Aide/Wiki", | ||||||
|     "crowdsourcedConfigsLabel": "Configuration d'applis communautaires (à utiliser à vos risques et périls)", |     "crowdsourcedConfigsLabel": "Configurations d'applications communautaires (à utiliser à vos risques et périls)", | ||||||
|     "crowdsourcedConfigsShort": "Applis communautaires", |     "crowdsourcedConfigsShort": "Applications communautaires", | ||||||
|     "allowInsecure": "Autoriser les requêtes HTTP non sécurisées", |     "allowInsecure": "Autoriser les requêtes HTTP non sécurisées", | ||||||
|     "stayOneVersionBehind": "Rester à une version de la dernière", |     "stayOneVersionBehind": "Rester une version en arrière de la dernière", | ||||||
|  |     "useFirstApkOfVersion": "Sélection automatique du premier APK parmi plusieurs", | ||||||
|     "refreshBeforeDownload": "Actualiser les détails de l'application avant de la télécharger", |     "refreshBeforeDownload": "Actualiser les détails de l'application avant de la télécharger", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Nom", | ||||||
|  |     "smartname": "Nom (Smart)", | ||||||
|  |     "sortMethod": "Méthode de tri", | ||||||
|  |     "welcome": "Bienvenue", | ||||||
|  |     "documentationLinksNote": "La page GitHub d'Obtainium, dont le lien figure ci-dessous, contient des liens vers des vidéos, des articles, des discussions et d'autres ressources qui vous aideront à comprendre comment utiliser l'application.", | ||||||
|  |     "batteryOptimizationNote": "Notez que les téléchargements en arrière-plan peuvent fonctionner de manière plus fiable si vous désactivez les optimisations de la batterie du système d'exploitation pour Obtainium.", | ||||||
|  |     "fileDeletionError": "Échec de la suppression du fichier (essayez de le supprimer manuellement puis réessayez) : \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Supprimer l'application ?", |         "one": "Supprimer l'application ?", | ||||||
|         "other": "Supprimer les applications ?" |         "other": "Supprimer les applications ?" | ||||||
|     }, |     }, | ||||||
|     "tooManyRequestsTryAgainInMinutes": { |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|         "one": "Trop de requêtes (taux limité) - réessayez dans {} minute", |         "one": "Trop de requêtes (taux limité) - réessayez dans {} minute", | ||||||
|         "other": "Trop de requêtes (taux limité) - réessayez dans {} minutes" |         "other": "Trop de requêtes (taux limité) - réessayez dans {} minutes" | ||||||
|     }, |     }, | ||||||
|     "bgUpdateGotErrorRetryInMinutes": { |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|         "one": "La recherche de mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative programmée dans {} minute", |         "one": "La recherche de mise à jour en arrière-plan a rencontré une erreur {}, une nouvelle tentative programmée dans {} minute", | ||||||
|         "other": "La recherche de mise à jour en arrière-plan a rencontré un {}, une nouvelle tentative programmée dans {} minutes" |         "other": "La recherche de mise à jour en arrière-plan a rencontré une erreur {}, une nouvelle tentative programmée dans {} minutes" | ||||||
|     }, |     }, | ||||||
|     "bgCheckFoundUpdatesWillNotifyIfNeeded": { |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|         "one": "La recherche de mises à jour en arrière-plan à trouvée {} mise à jour - l'utilisateur sera notifié si nécessaire", |         "one": "La recherche de mises à jour en arrière-plan a trouvé {} mise à jour - l'utilisateur sera notifié si nécessaire", | ||||||
|         "other": "La recherche de mises à jour en arrière-plan à trouvée {} mises à jour - l'utilisateur sera notifié si nécessaire" |         "other": "La recherche de mises à jour en arrière-plan a trouvé {} mises à jour - l'utilisateur sera notifié si nécessaire" | ||||||
|     }, |     }, | ||||||
|     "apps": { |     "apps": { | ||||||
|         "one": "{} Application", |         "one": "{} Application", | ||||||
| @@ -341,7 +352,7 @@ | |||||||
|     }, |     }, | ||||||
|     "url": { |     "url": { | ||||||
|         "one": "{} URL", |         "one": "{} URL", | ||||||
|         "other": "{} URL" |         "other": "{} URLs" | ||||||
|     }, |     }, | ||||||
|     "minute": { |     "minute": { | ||||||
|         "one": "{} Minute", |         "one": "{} Minute", | ||||||
| @@ -365,7 +376,7 @@ | |||||||
|     }, |     }, | ||||||
|     "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és mis à jour." |         "other": "{} et {} autres applications ont été mises à jour." | ||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesFailed": { |     "xAndNMoreUpdatesFailed": { | ||||||
|         "one": "Échec de la mise à jour de {} et 1 autre application.", |         "one": "Échec de la mise à jour de {} et 1 autre application.", | ||||||
| @@ -373,7 +384,7 @@ | |||||||
|     }, |     }, | ||||||
|     "xAndNMoreUpdatesPossiblyInstalled": { |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|         "one": "{} et 1 autre application ont peut-être été mises à jour.", |         "one": "{} et 1 autre application ont peut-être été mises à jour.", | ||||||
|         "other": "{} et {} autres applications ont peut-être étés mis à jour." |         "other": "{} et {} autres applications ont peut-être été mises à jour." | ||||||
|     }, |     }, | ||||||
|     "apk": { |     "apk": { | ||||||
|         "one": "{} APK", |         "one": "{} APK", | ||||||
|   | |||||||
| @@ -28,11 +28,11 @@ | |||||||
|     "githubStarredRepos": "Csillagozott GitHub tárolók", |     "githubStarredRepos": "Csillagozott GitHub tárolók", | ||||||
|     "uname": "Felhasználónév", |     "uname": "Felhasználónév", | ||||||
|     "wrongArgNum": "A megadott argumentumok száma nem megfelelő", |     "wrongArgNum": "A megadott argumentumok száma nem megfelelő", | ||||||
|     "xIsTrackOnly": "A(z) {} csak nyomonkövethető", |     "xIsTrackOnly": "A(z) {} csak nyomon kö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 nyomonkövethetők”.", |     "appsFromSourceAreTrackOnly": "Az ebből a forrásból származó alkalmazások „csak nyomon követhetők”.", | ||||||
|     "youPickedTrackOnly": "„Csak nyomonkövetés” opciót választotta.", |     "youPickedTrackOnly": "„Csak nyomon kö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": "Visszavonva", |     "cancelled": "Visszavonva", | ||||||
|     "appAlreadyAdded": "Az alkalmazás már hozzá van adva", |     "appAlreadyAdded": "Az alkalmazás már hozzá van adva", | ||||||
| @@ -45,7 +45,7 @@ | |||||||
|     "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 nyomon követés)", | ||||||
|     "searchableInBrackets": "(Kereshető)", |     "searchableInBrackets": "(Kereshető)", | ||||||
|     "appsString": "Alkalmazások", |     "appsString": "Alkalmazások", | ||||||
|     "noApps": "Nincsenek alkalmazások", |     "noApps": "Nincsenek alkalmazások", | ||||||
| @@ -63,7 +63,7 @@ | |||||||
|     "removeSelectedApps": "A kiválasztott alkalmazások eltávolítása", |     "removeSelectedApps": "A kiválasztott alkalmazások eltávolítása", | ||||||
|     "updateX": "{} frissítése", |     "updateX": "{} frissítése", | ||||||
|     "installX": "{} telepítése", |     "installX": "{} telepítése", | ||||||
|     "markXTrackOnlyAsUpdated": "Megjelölés: {}\n(Csak nyomonkövetés)\nFrissítettként", |     "markXTrackOnlyAsUpdated": "Megjelölés: {}\n(Csak nyomon követés)\nFrissítettként", | ||||||
|     "changeX": "{}-változás", |     "changeX": "{}-változás", | ||||||
|     "installUpdateApps": "Alkalmazások telepítése/frissítése", |     "installUpdateApps": "Alkalmazások telepítése/frissítése", | ||||||
|     "installUpdateSelectedApps": "A kiválasztott alkalmazások telepítése/frissítése", |     "installUpdateSelectedApps": "A kiválasztott alkalmazások telepítése/frissítése", | ||||||
| @@ -88,13 +88,13 @@ | |||||||
|     "author": "Szerző", |     "author": "Szerző", | ||||||
|     "upToDateApps": "Naprakész alkalmazások", |     "upToDateApps": "Naprakész alkalmazások", | ||||||
|     "nonInstalledApps": "Nem telepített alkalmazások", |     "nonInstalledApps": "Nem telepített alkalmazások", | ||||||
|     "importExport": "Import/Export", |     "importExport": "Adatmozgatás", | ||||||
|     "settings": "Beállítások", |     "settings": "Beállítások", | ||||||
|     "exportedTo": "Exportálva ide: {}", |     "exportedTo": "Exportálva ide: {}", | ||||||
|     "obtainiumExport": "Obtainium adatok exportálása", |     "obtainiumExport": "Obtainium-adatok exportálása", | ||||||
|     "invalidInput": "Hibás bemenet", |     "invalidInput": "Hibás bemenet", | ||||||
|     "importedX": "Importálva innen: {}", |     "importedX": "Importálva innen: {}", | ||||||
|     "obtainiumImport": "Obtainium adatok importálása", |     "obtainiumImport": "Obtainium-adatok importálása", | ||||||
|     "importFromURLList": "Importálás webcímlistából", |     "importFromURLList": "Importálás webcímlistából", | ||||||
|     "searchQuery": "Keresési lekérdezés", |     "searchQuery": "Keresési lekérdezés", | ||||||
|     "appURLList": "Alkalmazás-webcímlista", |     "appURLList": "Alkalmazás-webcímlista", | ||||||
| @@ -143,7 +143,7 @@ | |||||||
|     "warning": "Figyelem", |     "warning": "Figyelem", | ||||||
|     "sourceIsXButPackageFromYPrompt": "Az alkalmazás forrása a(z) „{}” tároló, 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ők el", | ||||||
|     "updatesAvailableNotifDescription": "Értesíti a felhasználót, hogy egy vagy több, az Obtainium által nyomonkövetett alkalmazáshoz frissítések állnak rendelkezésre", |     "updatesAvailableNotifDescription": "Értesíti a felhasználót, hogy egy vagy több, az Obtainium által nyomon kö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", | ||||||
| @@ -154,15 +154,15 @@ | |||||||
|     "errorCheckingUpdates": "Hiba a frissítések ellenőrzésekor", |     "errorCheckingUpdates": "Hiba a frissítések ellenőrzésekor", | ||||||
|     "errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, amikor a frissítések ellenőrzése a háttérben nem sikerül", |     "errorCheckingUpdatesNotifDescription": "Értesítés, amely akkor jelenik meg, amikor a frissítések ellenőrzése a háttérben nem sikerül", | ||||||
|     "appsRemoved": "Alkalmazások eltávolítva", |     "appsRemoved": "Alkalmazások eltávolítva", | ||||||
|     "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", |     "appsRemovedNotifDescription": "Értesíti a felhasználót, hogy egy vagy több alkalmazás egy betöltés közbeni hiba miatt el lesz(nek) távolítva", | ||||||
|     "xWasRemovedDueToErrorY": "A(z) {} eltávolításra került a következő hiba miatt: {}", |     "xWasRemovedDueToErrorY": "A(z) {} el lett távolítva a következő hiba miatt: {}", | ||||||
|     "completeAppInstallation": "Teljes alkalmazástelepítés", |     "completeAppInstallation": "Teljes alkalmazástelepítés", | ||||||
|     "obtainiumMustBeOpenToInstallApps": "Az alkalmazások telepítéséhez az Obtainiumnak megnyitva kell lennie", |     "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 ellenőrzése", |     "checkingForUpdates": "Frissítések ellenőrzése", | ||||||
|     "checkingForUpdatesNotifDescription": "Átmeneti értesítés, amely a frissítések ellenőrzésekor jelenik meg", |     "checkingForUpdatesNotifDescription": "Átmeneti értesítés, amely a frissítések ellenőrzésekor jelenik meg", | ||||||
|     "pleaseAllowInstallPerm": "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 nyomonkövetés", |     "trackOnly": "Csak nyomon követés", | ||||||
|     "errorWithHttpStatusCode": "Hiba {}", |     "errorWithHttpStatusCode": "Hiba {}", | ||||||
|     "versionCorrectionDisabled": "Verzió-korrekció letiltva (úgy tűnik, hogy a bővítmény 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", | ||||||
| @@ -220,10 +220,11 @@ | |||||||
|     "versionDetection": "Verzió-érzékelés", |     "versionDetection": "Verzió-érzékelés", | ||||||
|     "standardVersionDetection": "Alapértelmezett 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 meg az APK-kat CPU-architektúra szerint szűrni", | ||||||
|  |     "autoLinkFilterByArch": "Ha lehetséges, próbálja meg a hivatkozásokat CPU-architektúra szerint szűrni", | ||||||
|     "overrideSource": "Forrás felülírása", |     "overrideSource": "Forrás felülírása", | ||||||
|     "dontShowAgain": "Ne jelenítse meg ezt többé", |     "dontShowAgain": "Ne jelenítse meg ezt többé", | ||||||
|     "dontShowTrackOnlyWarnings": "Ne jelenítse meg a „Csak nyomonkövetés” figyelmeztetést", |     "dontShowTrackOnlyWarnings": "Ne jelenítse meg a „Csak nyomon követés” figyelmeztetést", | ||||||
|     "dontShowAPKOriginWarnings": "Ne jelenítse 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 alkalmazásokat az alkalmazásnézet aljára", |     "moveNonInstalledAppsToBottom": "Helyezze át a nem telepített alkalmazásokat az alkalmazásnézet aljára", | ||||||
|     "gitlabPATLabel": "GitLab személyes hozzáférési token", |     "gitlabPATLabel": "GitLab személyes hozzáférési token", | ||||||
| @@ -261,7 +262,7 @@ | |||||||
|     "versionExtractionRegEx": "Verzió-karakterlánc kivonatolása reguláris kifejezéssel", |     "versionExtractionRegEx": "Verzió-karakterlánc kivonatolása reguláris kifejezéssel", | ||||||
|     "trimVersionString": "Verzió-karakterlánc levágása reguláris kifejezéssel", |     "trimVersionString": "Verzió-karakterlánc levágása reguláris kifejezéssel", | ||||||
|     "matchGroupToUseForX": "A(z) „{}” esetén használandó csoport egyeztetése", |     "matchGroupToUseForX": "A(z) „{}” esetén használandó csoport egyeztetése", | ||||||
|     "matchGroupToUse": "Verziókarakterlánc-kivonatoláshoz használandó csoport reguláris kifejezéssel való egyeztetése", |     "matchGroupToUse": "A verzió-karakterlánc kivonatolásához használandó csoport reguláris kifejezéssel való egyeztetése", | ||||||
|     "highlightTouchTargets": "A kevésbé nyilvánvaló érintési pontok kiemelése", |     "highlightTouchTargets": "A kevésbé nyilvánvaló érintési pontok kiemelése", | ||||||
|     "pickExportDir": "Válassza ki a könyvtárat, ahová exportálni szeretne", |     "pickExportDir": "Válassza ki a könyvtárat, ahová exportálni szeretne", | ||||||
|     "autoExportOnChanges": "Automatikus exportálás a változtatások után", |     "autoExportOnChanges": "Automatikus exportálás a változtatások után", | ||||||
| @@ -286,7 +287,7 @@ | |||||||
|     "downloadingXNotifChannel": "A(z) {} letöltése", |     "downloadingXNotifChannel": "A(z) {} letöltése", | ||||||
|     "completeAppInstallationNotifChannel": "Teljes alkalmazás telepítés", |     "completeAppInstallationNotifChannel": "Teljes alkalmazás telepítés", | ||||||
|     "checkingForUpdatesNotifChannel": "Frissítések ellenőrzése", |     "checkingForUpdatesNotifChannel": "Frissítések ellenőrzése", | ||||||
|     "onlyCheckInstalledOrTrackOnlyApps": "Csak a telepített és a csak nyomonkövethető alkalmazások frissítéseinek ellenőrzése", |     "onlyCheckInstalledOrTrackOnlyApps": "Csak a telepített és a csak nyomon követhető alkalmazások frissítéseinek ellenőrzése", | ||||||
|     "supportFixedAPKURL": "Támogatja a rögzített APK webcímeket", |     "supportFixedAPKURL": "Támogatja a rögzített APK webcímeket", | ||||||
|     "selectX": "{} kiválasztása", |     "selectX": "{} kiválasztása", | ||||||
|     "parallelDownloads": "Párhuzamos letöltések engedélyezése", |     "parallelDownloads": "Párhuzamos letöltések engedélyezése", | ||||||
| @@ -294,7 +295,7 @@ | |||||||
|     "shizukuBinderNotFound": "A Shizuku szolgáltatás nem fut", |     "shizukuBinderNotFound": "A Shizuku szolgáltatás nem fut", | ||||||
|     "shizukuOld": "Régi Shizuku verzió (<11) - frissítse", |     "shizukuOld": "Régi Shizuku verzió (<11) - frissítse", | ||||||
|     "shizukuOldAndroidWithADB": "A Shizuku csak Android < 8.1 ADB-vel fut - frissítse az Androidot vagy használja a Sui-t helyette", |     "shizukuOldAndroidWithADB": "A Shizuku csak Android < 8.1 ADB-vel fut - frissítse az Androidot vagy használja a Sui-t helyette", | ||||||
|     "shizukuPretendToBeGooglePlay": "Állítsa be a Google Playt telepítési forrásként (ha Shizukut használ)", |     "shizukuPretendToBeGooglePlay": "A Google Play beállítása telepítési forrásként (ha Shizukut használ)", | ||||||
|     "useSystemFont": "A rendszer betűtípusának használata", |     "useSystemFont": "A rendszer betűtípusának használata", | ||||||
|     "useVersionCodeAsOSVersion": "Az alkalmazás verziókódjának használata a rendszer által érzékelt verzióként", |     "useVersionCodeAsOSVersion": "Az alkalmazás verziókódjának használata a rendszer által érzékelt verzióként", | ||||||
|     "requestHeader": "Kérelemfejléc", |     "requestHeader": "Kérelemfejléc", | ||||||
| @@ -313,12 +314,22 @@ | |||||||
|     "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", |     "wiki": "Súgó/Wiki", | ||||||
|     "crowdsourcedConfigsLabel": "Crowdsource-ből származó alkalmazások beállítása (saját felelősségére használja)", |     "crowdsourcedConfigsLabel": "Közreműködők által összeállított alkalmazásbeállítások (saját felelősségére használja)", | ||||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", |     "crowdsourcedConfigsShort": "Alkalmazáslista", | ||||||
|     "allowInsecure": "Nem biztonságos HTTP-kérések engedélyezése", |     "allowInsecure": "Nem biztonságos HTTP-kérések engedélyezése", | ||||||
|     "stayOneVersionBehind": "Maradjon egy verzióval a legújabb mögött", |     "stayOneVersionBehind": "Maradjon egy verzióval a legújabb mögött", | ||||||
|  |     "useFirstApkOfVersion": "A legelső APK automatikus kiválasztása, ha több APK is található", | ||||||
|     "refreshBeforeDownload": "Az alkalmazás adatainak frissítése a letöltés előtt", |     "refreshBeforeDownload": "Az alkalmazás adatainak frissítése a letöltés előtt", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent Appstore", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Név", | ||||||
|  |     "smartname": "Név (Okos)", | ||||||
|  |     "sortMethod": "Rendezési eljárás", | ||||||
|  |     "welcome": "Üdvözöljük!", | ||||||
|  |     "documentationLinksNote": "Az alábbi hivatkozás az Obtainium GitHub oldalára vezet, amely további videók, cikkek, beszélgetések és egyéb források hivatkozásait tartalmazza, amelyek segítenek megérteni az alkalmazás használatát.", | ||||||
|  |     "batteryOptimizationNote": "Megjegyzés: A háttérfrissítések megbízhatóbban működhetnek, ha kikapcsolja a rendszer akkumulátor-optimalizálását az Obtainium számára.", | ||||||
|  |     "fileDeletionError": "Nem sikerült törölni a fájlt (próbálja meg kézzel törölni, majd próbálja meg újra): \"{}\"", | ||||||
|     "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?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Pendeteksi versi standar", |     "standardVersionDetection": "Pendeteksi versi standar", | ||||||
|     "groupByCategory": "Kelompokkan berdasarkan kategori", |     "groupByCategory": "Kelompokkan berdasarkan kategori", | ||||||
|     "autoApkFilterByArch": "Coba filter APK berdasarkan arsitektur CPU jika memungkinkan", |     "autoApkFilterByArch": "Coba filter APK berdasarkan arsitektur CPU jika memungkinkan", | ||||||
|  |     "autoLinkFilterByArch": "Mencoba memfilter tautan berdasarkan arsitektur CPU jika memungkinkan", | ||||||
|     "overrideSource": "Ganti sumber", |     "overrideSource": "Ganti sumber", | ||||||
|     "dontShowAgain": "Jangan tampilkan lagi", |     "dontShowAgain": "Jangan tampilkan lagi", | ||||||
|     "dontShowTrackOnlyWarnings": "Jangan tampilkan peringatan 'Pelacakan Saja'", |     "dontShowTrackOnlyWarnings": "Jangan tampilkan peringatan 'Pelacakan Saja'", | ||||||
| @@ -255,7 +256,7 @@ | |||||||
|     "intermediateLinkNotFound": "Tautan perantara tidak ditemukan", |     "intermediateLinkNotFound": "Tautan perantara tidak ditemukan", | ||||||
|     "intermediateLink": "Tautan perantara", |     "intermediateLink": "Tautan perantara", | ||||||
|     "exemptFromBackgroundUpdates": "Dikecualikan dari pembaruan latar belakang (jika diaktifkan)", |     "exemptFromBackgroundUpdates": "Dikecualikan dari pembaruan latar belakang (jika diaktifkan)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Nonaktifkan pembaruan latar belakang saat tidak menggunakan WiFi", |     "bgUpdatesOnWiFiOnly": "Nonaktifkan pembaruan latar belakang saat tidak menggunakan Wi-Fi", | ||||||
|     "bgUpdatesWhileChargingOnly": "Menonaktifkan pembaruan latar belakang saat tidak mengisi daya", |     "bgUpdatesWhileChargingOnly": "Menonaktifkan pembaruan latar belakang saat tidak mengisi daya", | ||||||
|     "autoSelectHighestVersionCode": "Pilih otomatis APK dengan versi kode tertinggi", |     "autoSelectHighestVersionCode": "Pilih otomatis APK dengan versi kode tertinggi", | ||||||
|     "versionExtractionRegEx": "Reguler ekspresi terkait ekstraksi versi string", |     "versionExtractionRegEx": "Reguler ekspresi terkait ekstraksi versi string", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Konfigurasi Aplikasi Crowdsourced", |     "crowdsourcedConfigsShort": "Konfigurasi Aplikasi Crowdsourced", | ||||||
|     "allowInsecure": "Izinkan permintaan HTTP yang tidak aman", |     "allowInsecure": "Izinkan permintaan HTTP yang tidak aman", | ||||||
|     "stayOneVersionBehind": "Tetap satu versi di belakang versi terbaru", |     "stayOneVersionBehind": "Tetap satu versi di belakang versi terbaru", | ||||||
|  |     "useFirstApkOfVersion": "Pilih otomatis yang pertama dari beberapa APK", | ||||||
|     "refreshBeforeDownload": "Segarkan detail aplikasi sebelum mengunduh", |     "refreshBeforeDownload": "Segarkan detail aplikasi sebelum mengunduh", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Nama", | ||||||
|  |     "smartname": "Nama (Cerdas)", | ||||||
|  |     "sortMethod": "Metode Penyortiran", | ||||||
|  |     "welcome": "Selamat datang.", | ||||||
|  |     "documentationLinksNote": "Halaman GitHub Obtainium yang ditautkan di bawah ini berisi tautan ke video, artikel, diskusi, dan sumber daya lain yang akan membantu Anda memahami cara menggunakan aplikasi.", | ||||||
|  |     "batteryOptimizationNote": "Perhatikan bahwa unduhan latar belakang dapat bekerja lebih andal jika Anda menonaktifkan optimasi baterai OS untuk Obtainium.", | ||||||
|  |     "fileDeletionError": "Gagal menghapus file (coba hapus secara manual, lalu coba lagi): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Hapus aplikasi?", |         "one": "Hapus aplikasi?", | ||||||
|         "other": "Hapus aplikasi?" |         "other": "Hapus aplikasi?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Rilevamento di versione standard", |     "standardVersionDetection": "Rilevamento di versione standard", | ||||||
|     "groupByCategory": "Raggruppa per categoria", |     "groupByCategory": "Raggruppa per categoria", | ||||||
|     "autoApkFilterByArch": "Tenta di filtrare gli APK in base all'architettura della CPU, se possibile", |     "autoApkFilterByArch": "Tenta di filtrare gli APK in base all'architettura della CPU, se possibile", | ||||||
|  |     "autoLinkFilterByArch": "Tentare di filtrare i collegamenti in base all'architettura della CPU, se possibile.", | ||||||
|     "overrideSource": "Sovrascrivi fonte", |     "overrideSource": "Sovrascrivi fonte", | ||||||
|     "dontShowAgain": "Non mostrarlo più", |     "dontShowAgain": "Non mostrarlo più", | ||||||
|     "dontShowTrackOnlyWarnings": "Non mostrare gli avvisi 'Solo-Monitoraggio'", |     "dontShowTrackOnlyWarnings": "Non mostrare gli avvisi 'Solo-Monitoraggio'", | ||||||
| @@ -255,7 +256,7 @@ | |||||||
|     "intermediateLinkNotFound": "Link intermedio non trovato", |     "intermediateLinkNotFound": "Link intermedio non trovato", | ||||||
|     "intermediateLink": "Collegamento intermedio", |     "intermediateLink": "Collegamento intermedio", | ||||||
|     "exemptFromBackgroundUpdates": "Esente da aggiornamenti in secondo piano (se attivo)", |     "exemptFromBackgroundUpdates": "Esente da aggiornamenti in secondo piano (se attivo)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Disattiva aggiornamenti in secondo piano quando non si usa il WiFi", |     "bgUpdatesOnWiFiOnly": "Disattiva aggiornamenti in secondo piano quando non si usa il Wi-Fi", | ||||||
|     "bgUpdatesWhileChargingOnly": "Disabilita gli aggiornamenti in background quando non è in carica", |     "bgUpdatesWhileChargingOnly": "Disabilita gli aggiornamenti in background quando non è in carica", | ||||||
|     "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", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Configurazioni di app in crowdsourcing", |     "crowdsourcedConfigsShort": "Configurazioni di app in crowdsourcing", | ||||||
|     "allowInsecure": "Consentire le richieste HTTP non sicure", |     "allowInsecure": "Consentire le richieste HTTP non sicure", | ||||||
|     "stayOneVersionBehind": "Rimanere una versione indietro rispetto alla più recente", |     "stayOneVersionBehind": "Rimanere una versione indietro rispetto alla più recente", | ||||||
|  |     "useFirstApkOfVersion": "Selezione automatica del primo di più APK", | ||||||
|     "refreshBeforeDownload": "Aggiornare i dettagli dell'app prima del download", |     "refreshBeforeDownload": "Aggiornare i dettagli dell'app prima del download", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Nome", | ||||||
|  |     "smartname": "Nome (intelligente)", | ||||||
|  |     "sortMethod": "Metodo di ordinamento", | ||||||
|  |     "welcome": "Benvenuti", | ||||||
|  |     "documentationLinksNote": "La pagina GitHub di Obtainium collegata qui sotto contiene collegamenti a video, articoli, discussioni e altre risorse che vi aiuteranno a capire come utilizzare l'applicazione.", | ||||||
|  |     "batteryOptimizationNote": "Si noti che i download in background potrebbero funzionare in modo più affidabile se si disabilita l'ottimizzazione della batteria del sistema operativo per Obtainium.", | ||||||
|  |     "fileDeletionError": "Errore nell'eliminazione del file (provare a cancellarlo manualmente e poi riprovare): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Rimuovere l'app?", |         "one": "Rimuovere l'app?", | ||||||
|         "other": "Rimuovere le app?" |         "other": "Rimuovere le app?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "標準のバージョン検出", |     "standardVersionDetection": "標準のバージョン検出", | ||||||
|     "groupByCategory": "カテゴリ別にグループ化する", |     "groupByCategory": "カテゴリ別にグループ化する", | ||||||
|     "autoApkFilterByArch": "可能であれば、CPUアーキテクチャによるAPKのフィルタリングを試みる", |     "autoApkFilterByArch": "可能であれば、CPUアーキテクチャによるAPKのフィルタリングを試みる", | ||||||
|  |     "autoLinkFilterByArch": "可能であれば、CPUアーキテクチャによるリンクのフィルタリングを試みる。", | ||||||
|     "overrideSource": "ソースの上書き", |     "overrideSource": "ソースの上書き", | ||||||
|     "dontShowAgain": "二度と表示しない", |     "dontShowAgain": "二度と表示しない", | ||||||
|     "dontShowTrackOnlyWarnings": "「追跡のみ」の警告を表示しない", |     "dontShowTrackOnlyWarnings": "「追跡のみ」の警告を表示しない", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "クラウドソーシングによるアプリの設定", |     "crowdsourcedConfigsShort": "クラウドソーシングによるアプリの設定", | ||||||
|     "allowInsecure": "安全でないHTTPリクエストを許可する", |     "allowInsecure": "安全でないHTTPリクエストを許可する", | ||||||
|     "stayOneVersionBehind": "最新のバージョンから1つ前のものを使用する", |     "stayOneVersionBehind": "最新のバージョンから1つ前のものを使用する", | ||||||
|  |     "useFirstApkOfVersion": "複数のAPKから最初のAPKを自動選択する", | ||||||
|     "refreshBeforeDownload": "ダウンロード前にアプリの詳細を更新する", |     "refreshBeforeDownload": "ダウンロード前にアプリの詳細を更新する", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "クールApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "名称", | ||||||
|  |     "smartname": "名前(スマート)", | ||||||
|  |     "sortMethod": "ソート方法", | ||||||
|  |     "welcome": "ようこそ", | ||||||
|  |     "documentationLinksNote": "以下のリンクにあるObtainium GitHubページには、ビデオ、記事、ディスカッション、その他のリソースへのリンクがあり、アプリの使い方を理解するのに役立ちます。", | ||||||
|  |     "batteryOptimizationNote": "ObtainiumのOSバッテリー最適化を無効にすると、バックグラウンドダウンロードがより確実に動作するようになります。", | ||||||
|  |     "fileDeletionError": "ファイルの削除に失敗しました(手動で削除してから再試行してください):\"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "アプリを削除しますか?", |         "one": "アプリを削除しますか?", | ||||||
|         "other": "アプリを削除しますか?" |         "other": "アプリを削除しますか?" | ||||||
|   | |||||||
							
								
								
									
										393
									
								
								assets/translations/ko.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,393 @@ | |||||||
|  | { | ||||||
|  |     "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\nURL 및 타사 가져오기 방법에만 영향을 미칩니다.", | ||||||
|  |     "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 타사 저장소", | ||||||
|  |     "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": "릴리스 날짜를 버전 문자열로 사용", | ||||||
|  |     "releaseTitleAsVersion": "릴리스 제목을 버전 문자열로 사용", | ||||||
|  |     "releaseDateAsVersionExplanation": "이 옵션은 버전 감지가 올바르게 작동하지 않지만 릴리스 날짜가 있는 앱에만 사용해야 합니다.", | ||||||
|  |     "changes": "변경 사항", | ||||||
|  |     "releaseDate": "릴리스 날짜", | ||||||
|  |     "importFromURLsInFile": "파일의 URL에서 가져오기 (OPML과 같은)", | ||||||
|  |     "versionDetectionExplanation": "OS에서 감지된 버전과 버전 문자열 조정", | ||||||
|  |     "versionDetection": "버전 감지", | ||||||
|  |     "standardVersionDetection": "표준 버전 감지", | ||||||
|  |     "groupByCategory": "카테고리별 그룹화", | ||||||
|  |     "autoApkFilterByArch": "가능한 경우 CPU 아키텍처별로 APK 필터링 시도", | ||||||
|  |     "autoLinkFilterByArch": "가능하면 CPU 아키텍처별로 링크를 필터링해 보세요.", | ||||||
|  |     "overrideSource": "소스 재정의", | ||||||
|  |     "dontShowAgain": "다시 표시하지 않기", | ||||||
|  |     "dontShowTrackOnlyWarnings": "'추적 전용' 경고 표시 안 함", | ||||||
|  |     "dontShowAPKOriginWarnings": "APK 출처 경고 표시 안 함", | ||||||
|  |     "moveNonInstalledAppsToBottom": "설치되지 않은 앱을 앱 보기 하단으로 이동", | ||||||
|  |     "gitlabPATLabel": "GitLab 개인 액세스 토큰", | ||||||
|  |     "about": "정보", | ||||||
|  |     "requiresCredentialsInSettings": "{}는 추가 자격 증명이 필요합니다 (설정에서)", | ||||||
|  |     "checkOnStart": "시작 시 업데이트 확인", | ||||||
|  |     "tryInferAppIdFromCode": "소스 코드에서 앱 ID 추론 시도", | ||||||
|  |     "removeOnExternalUninstall": "외부에서 제거된 앱 자동 제거", | ||||||
|  |     "pickHighestVersionCode": "가장 높은 버전 코드 APK 자동 선택", | ||||||
|  |     "checkUpdateOnDetailPage": "앱 세부 정보 페이지 열 때 업데이트 확인", | ||||||
|  |     "disablePageTransitions": "페이지 전환 애니메이션 비활성화", | ||||||
|  |     "reversePageTransitions": "페이지 전환 애니메이션 반전", | ||||||
|  |     "minStarCount": "최소 별 개수", | ||||||
|  |     "addInfoBelow": "아래에 이 정보를 추가하십시오.", | ||||||
|  |     "addInfoInSettings": "설정에 이 정보를 추가하십시오.", | ||||||
|  |     "githubSourceNote": "GitHub 속도 제한은 API 키를 사용하여 피할 수 있습니다.", | ||||||
|  |     "sortByLastLinkSegment": "링크의 마지막 세그먼트로만 정렬", | ||||||
|  |     "filterReleaseNotesByRegEx": "정규 표현식으로 릴리스 노트 필터링", | ||||||
|  |     "customLinkFilterRegex": "정규 표현식으로 사용자 정의 APK 링크 필터링 (기본값 '.apk$')", | ||||||
|  |     "appsPossiblyUpdated": "앱 업데이트 시도됨", | ||||||
|  |     "appsPossiblyUpdatedNotifDescription": "백그라운드에서 하나 이상의 앱에 대한 업데이트가 잠재적으로 적용되었음을 사용자에게 알립니다", | ||||||
|  |     "xWasPossiblyUpdatedToY": "{}가 {}로 업데이트되었을 수 있습니다.", | ||||||
|  |     "enableBackgroundUpdates": "백그라운드 업데이트 활성화", | ||||||
|  |     "backgroundUpdateReqsExplanation": "모든 앱에 대해 백그라운드 업데이트가 가능하지 않을 수 있습니다.", | ||||||
|  |     "backgroundUpdateLimitsExplanation": "백그라운드 설치의 성공 여부는 Obtainium이 열릴 때만 확인할 수 있습니다.", | ||||||
|  |     "verifyLatestTag": "'최신' 태그 확인", | ||||||
|  |     "intermediateLinkRegex": "'중간' 링크 방문 필터", | ||||||
|  |     "filterByLinkText": "링크 텍스트로 링크 필터링", | ||||||
|  |     "intermediateLinkNotFound": "중간 링크를 찾을 수 없습니다", | ||||||
|  |     "intermediateLink": "중간 링크", | ||||||
|  |     "exemptFromBackgroundUpdates": "백그라운드 업데이트에서 제외 (활성화된 경우)", | ||||||
|  |     "bgUpdatesOnWiFiOnly": "WiFi가 아닐 때 백그라운드 업데이트 비활성화", | ||||||
|  |     "bgUpdatesWhileChargingOnly": "충전 중이 아닐 때 백그라운드 업데이트 비활성화", | ||||||
|  |     "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 URL 지원", | ||||||
|  |     "selectX": "{} 선택", | ||||||
|  |     "parallelDownloads": "병렬 다운로드 허용", | ||||||
|  |     "useShizuku": "Shizuku 또는 Sui를 사용하여 설치", | ||||||
|  |     "shizukuBinderNotFound": "Shizuku 서비스가 실행 중이 아닙니다", | ||||||
|  |     "shizukuOld": "오래된 Shizuku 버전 (<11) - 업데이트 필요", | ||||||
|  |     "shizukuOldAndroidWithADB": "ADB로 Android < 8.1에서 실행 중인 Shizuku - Android를 업데이트하거나 대신 Sui를 사용하세요", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "설치 소스로 Google Play 설정 (Shizuku 사용 시)", | ||||||
|  |     "useSystemFont": "시스템 글꼴 사용", | ||||||
|  |     "useVersionCodeAsOSVersion": "앱 versionCode를 OS에서 감지된 버전으로 사용", | ||||||
|  |     "requestHeader": "요청 헤더", | ||||||
|  |     "useLatestAssetDateAsReleaseDate": "최신 자산 업로드를 릴리스 날짜로 사용", | ||||||
|  |     "defaultPseudoVersioningMethod": "기본 의사 버전 관리 방법", | ||||||
|  |     "partialAPKHash": "부분 APK 해시", | ||||||
|  |     "APKLinkHash": "APK 링크 해시", | ||||||
|  |     "directAPKLink": "직접 APK 링크", | ||||||
|  |     "pseudoVersionInUse": "의사 버전 사용 중", | ||||||
|  |     "installed": "설치됨", | ||||||
|  |     "latest": "최신", | ||||||
|  |     "invertRegEx": "정규 표현식 반전", | ||||||
|  |     "note": "노트", | ||||||
|  |     "selfHostedNote": "\"{}\" 드롭다운을 사용하여 소스의 자체 호스팅/사용자 정의 인스턴스에 도달할 수 있습니다.", | ||||||
|  |     "badDownload": "APK를 구문 분석할 수 없습니다 (호환되지 않거나 부분 다운로드)", | ||||||
|  |     "beforeNewInstallsShareToAppVerifier": "새 앱을 AppVerifier와 공유 (가능한 경우)", | ||||||
|  |     "appVerifierInstructionToast": "AppVerifier에 공유한 후 준비가 되면 여기로 돌아오세요.", | ||||||
|  |     "wiki": "도움말/위키", | ||||||
|  |     "crowdsourcedConfigsLabel": "크라우드소싱 앱 구성 (자신의 책임 하에 사용)", | ||||||
|  |     "crowdsourcedConfigsShort": "크라우드소싱 앱 구성", | ||||||
|  |     "allowInsecure": "안전하지 않은 HTTP 요청 허용", | ||||||
|  |     "stayOneVersionBehind": "최신 버전보다 한 버전 뒤에 머무르기", | ||||||
|  |     "useFirstApkOfVersion": "여러 앱 중 첫 번째 앱 자동 선택", | ||||||
|  |     "refreshBeforeDownload": "다운로드 전에 앱 세부 정보 새로 고침", | ||||||
|  |     "tencentAppStore": "텐센트 앱 스토어", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "이름", | ||||||
|  |     "smartname": "이름(스마트)", | ||||||
|  |     "sortMethod": "정렬 방법", | ||||||
|  |     "welcome": "환영", | ||||||
|  |     "documentationLinksNote": "아래에 링크된 Obtainium 깃허브 페이지에는 앱 사용 방법을 이해하는 데 도움이 되는 동영상, 기사, 토론 및 기타 리소스에 대한 링크가 포함되어 있습니다.", | ||||||
|  |     "batteryOptimizationNote": "Obtainium의 OS 배터리 최적화를 비활성화하면 백그라운드 다운로드가 더 안정적으로 작동할 수 있습니다.", | ||||||
|  |     "fileDeletionError": "파일을 삭제하지 못했습니다(수동으로 삭제한 후 다시 시도하세요): \"{}\"", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "앱을 제거하시겠습니까?", | ||||||
|  |         "other": "앱을 제거하시겠습니까?" | ||||||
|  |     }, | ||||||
|  |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|  |         "one": "요청이 너무 많습니다 (속도 제한) - {}분 후에 다시 시도하세요", | ||||||
|  |         "other": "요청이 너무 많습니다 (속도 제한) - {}분 후에 다시 시도하세요" | ||||||
|  |     }, | ||||||
|  |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|  |         "one": "BG 업데이트 확인 중 {} 오류가 발생했습니다. {}분 후에 다시 확인을 예약합니다", | ||||||
|  |         "other": "BG 업데이트 확인 중 {} 오류가 발생했습니다. {}분 후에 다시 확인을 예약합니다" | ||||||
|  |     }, | ||||||
|  |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|  |         "one": "BG 업데이트 확인에서 {}개의 업데이트를 발견했습니다 - 필요 시 사용자에게 알립니다", | ||||||
|  |         "other": "BG 업데이트 확인에서 {}개의 업데이트를 발견했습니다 - 필요 시 사용자에게 알립니다" | ||||||
|  |     }, | ||||||
|  |     "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" | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Standaard versieherkenning", |     "standardVersionDetection": "Standaard versieherkenning", | ||||||
|     "groupByCategory": "Groeperen op categorie", |     "groupByCategory": "Groeperen op categorie", | ||||||
|     "autoApkFilterByArch": "Probeer APK's te filteren op CPU-architectuur, indien mogelijk", |     "autoApkFilterByArch": "Probeer APK's te filteren op CPU-architectuur, indien mogelijk", | ||||||
|  |     "autoLinkFilterByArch": "Probeer links zo mogelijk te filteren op CPU-architectuur", | ||||||
|     "overrideSource": "Bron overschrijven", |     "overrideSource": "Bron overschrijven", | ||||||
|     "dontShowAgain": "Laat dit niet meer zien", |     "dontShowAgain": "Laat dit niet meer zien", | ||||||
|     "dontShowTrackOnlyWarnings": "Geen waarschuwingen weergeven voor 'Alleen volgen'", |     "dontShowTrackOnlyWarnings": "Geen waarschuwingen weergeven voor 'Alleen volgen'", | ||||||
| @@ -255,7 +256,7 @@ | |||||||
|     "intermediateLinkNotFound": "Intermediaire link niet gevonden", |     "intermediateLinkNotFound": "Intermediaire link niet gevonden", | ||||||
|     "intermediateLink": "Intermediaire link", |     "intermediateLink": "Intermediaire link", | ||||||
|     "exemptFromBackgroundUpdates": "Vrijgesteld van achtergrond-updates (indien ingeschakeld)", |     "exemptFromBackgroundUpdates": "Vrijgesteld van achtergrond-updates (indien ingeschakeld)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Achtergrond-updates uitschakelen wanneer niet verbonden met WiFi", |     "bgUpdatesOnWiFiOnly": "Achtergrond-updates uitschakelen wanneer niet verbonden met Wi-Fi", | ||||||
|     "bgUpdatesWhileChargingOnly": "Achtergrondupdates uitschakelen als er niet wordt opgeladen", |     "bgUpdatesWhileChargingOnly": "Achtergrondupdates uitschakelen als er niet wordt opgeladen", | ||||||
|     "autoSelectHighestVersionCode": "De APK met de hoogste versiecode automatisch selecteren", |     "autoSelectHighestVersionCode": "De APK met de hoogste versiecode automatisch selecteren", | ||||||
|     "versionExtractionRegEx": "Reguliere expressie voor versie-extractie", |     "versionExtractionRegEx": "Reguliere expressie voor versie-extractie", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "App-configuraties door menigte", |     "crowdsourcedConfigsShort": "App-configuraties door menigte", | ||||||
|     "allowInsecure": "Onveilige HTTP-verzoeken toestaan", |     "allowInsecure": "Onveilige HTTP-verzoeken toestaan", | ||||||
|     "stayOneVersionBehind": "Blijf een versie achter op de nieuwste", |     "stayOneVersionBehind": "Blijf een versie achter op de nieuwste", | ||||||
|  |     "useFirstApkOfVersion": "Automatisch de eerste van meerdere APK's selecteren", | ||||||
|     "refreshBeforeDownload": "Vernieuw app details voor download", |     "refreshBeforeDownload": "Vernieuw app details voor download", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Naam", | ||||||
|  |     "smartname": "Naam (Slim)", | ||||||
|  |     "sortMethod": "Sorteermethode", | ||||||
|  |     "welcome": "Welkom", | ||||||
|  |     "documentationLinksNote": "De GitHub pagina van Obtainium waarnaar hieronder wordt gelinkt bevat links naar video's, artikelen, discussies en andere bronnen die je zullen helpen begrijpen hoe je de app kunt gebruiken.", | ||||||
|  |     "batteryOptimizationNote": "Merk op dat downloads op de achtergrond mogelijk betrouwbaarder werken als je de batterijoptimalisatie van het besturingssysteem voor Obtainium uitschakelt.", | ||||||
|  |     "fileDeletionError": "Bestand is niet verwijderd (probeer het handmatig te verwijderen en probeer het opnieuw): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "App verwijderen?", |         "one": "App verwijderen?", | ||||||
|         "other": "Apps verwijderen?" |         "other": "Apps verwijderen?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Standardowe wykrywanie wersji", |     "standardVersionDetection": "Standardowe wykrywanie wersji", | ||||||
|     "groupByCategory": "Grupuj według kategorii", |     "groupByCategory": "Grupuj według kategorii", | ||||||
|     "autoApkFilterByArch": "Spróbuj filtrować pliki APK według architektury procesora, jeśli to możliwe", |     "autoApkFilterByArch": "Spróbuj filtrować pliki APK według architektury procesora, jeśli to możliwe", | ||||||
|  |     "autoLinkFilterByArch": "Spróbuj filtrować linki według architektury procesora, jeśli to możliwe.", | ||||||
|     "overrideSource": "Nadpisz źródło", |     "overrideSource": "Nadpisz źródło", | ||||||
|     "dontShowAgain": "Nie pokazuj tego ponownie", |     "dontShowAgain": "Nie pokazuj tego ponownie", | ||||||
|     "dontShowTrackOnlyWarnings": "Nie pokazuj ostrzeżeń \"Tylko obserwowana\"", |     "dontShowTrackOnlyWarnings": "Nie pokazuj ostrzeżeń \"Tylko obserwowana\"", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Baza konfiguracji", |     "crowdsourcedConfigsShort": "Baza konfiguracji", | ||||||
|     "allowInsecure": "Zezwalaj na niezabezpieczone żądania HTTP", |     "allowInsecure": "Zezwalaj na niezabezpieczone żądania HTTP", | ||||||
|     "stayOneVersionBehind": "Pozostań jedną wersję w tyle za najnowszą", |     "stayOneVersionBehind": "Pozostań jedną wersję w tyle za najnowszą", | ||||||
|  |     "useFirstApkOfVersion": "Automatyczny wybór pierwszego z wielu plików APK", | ||||||
|     "refreshBeforeDownload": "Odśwież szczegóły aplikacji przed pobraniem", |     "refreshBeforeDownload": "Odśwież szczegóły aplikacji przed pobraniem", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Nazwa", | ||||||
|  |     "smartname": "Nazwa (Smart)", | ||||||
|  |     "sortMethod": "Metoda sortowania", | ||||||
|  |     "welcome": "Witamy", | ||||||
|  |     "documentationLinksNote": "Strona Obtainium GitHub, do której link znajduje się poniżej, zawiera linki do filmów, artykułów, dyskusji i innych zasobów, które pomogą ci zrozumieć, jak korzystać z aplikacji.", | ||||||
|  |     "batteryOptimizationNote": "Należy pamiętać, że pobieranie w tle może działać bardziej niezawodnie po wyłączeniu optymalizacji baterii systemu operacyjnego dla Obtainium.", | ||||||
|  |     "fileDeletionError": "Nie udało się usunąć pliku (spróbuj usunąć go ręcznie, a następnie spróbuj ponownie): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Usunąć aplikację?", |         "one": "Usunąć aplikację?", | ||||||
|         "few": "Usunąć aplikacje?", |         "few": "Usunąć aplikacje?", | ||||||
|   | |||||||
							
								
								
									
										393
									
								
								assets/translations/pt-BR.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,393 @@ | |||||||
|  | { | ||||||
|  |     "invalidURLForSource": "Não é uma URL de app válida de {}", | ||||||
|  |     "noReleaseFound": "Não foi possível encontrar um lançamento adequado", | ||||||
|  |     "noVersionFound": "Não foi possível determinar a versão do lançamento", | ||||||
|  |     "urlMatchesNoSource": "A URL não corresponde com nenhuma fonte conhecida", | ||||||
|  |     "cantInstallOlderVersion": "Não é possível instalar uma versão mais antiga de um app", | ||||||
|  |     "appIdMismatch": "O ID do pacote baixado não corresponde ao existente", | ||||||
|  |     "functionNotImplemented": "Essa classe não implementou esse recurso ainda", | ||||||
|  |     "placeholder": "Espaço reservado", | ||||||
|  |     "someErrors": "Ocorreram alguns erros", | ||||||
|  |     "unexpectedError": "Erro inesperado", | ||||||
|  |     "ok": "Ok", | ||||||
|  |     "and": "e", | ||||||
|  |     "githubPATLabel": "Token de acesso pessoal do GitHub (aumenta o limite de taxa)", | ||||||
|  |     "includePrereleases": "Incluir pré-lançamentos", | ||||||
|  |     "fallbackToOlderReleases": "Recorrer à lançamentos mais antigos", | ||||||
|  |     "filterReleaseTitlesByRegEx": "Filtrar títulos de lançamentos por expressão regular", | ||||||
|  |     "invalidRegEx": "Expressão regular inválida", | ||||||
|  |     "noDescription": "Sem descrição", | ||||||
|  |     "cancel": "Cancelar", | ||||||
|  |     "continue": "Continuar", | ||||||
|  |     "requiredInBrackets": "(obrigatório)", | ||||||
|  |     "dropdownNoOptsError": "ERRO: O MENU DEVE TER PELO MENOS UMA OPÇÃO", | ||||||
|  |     "colour": "Cor", | ||||||
|  |     "standard": "Padrão", | ||||||
|  |     "custom": "Personalizado", | ||||||
|  |     "useMaterialYou": "Usar Material You", | ||||||
|  |     "githubStarredRepos": "Repositórios com estrela do GitHub", | ||||||
|  |     "uname": "Nome de usuário", | ||||||
|  |     "wrongArgNum": "Número errado de argumentos fornecidos", | ||||||
|  |     "xIsTrackOnly": "{} é somente de rastreio", | ||||||
|  |     "source": "Fonte", | ||||||
|  |     "app": "Aplicativo", | ||||||
|  |     "appsFromSourceAreTrackOnly": "Apps desta fonte são somente para rastreamento.", | ||||||
|  |     "youPickedTrackOnly": "Você selecionou a opção de somente rastreamento.", | ||||||
|  |     "trackOnlyAppDescription": "As atualizações do app serão rastreadas, mas o Obtainium não baixará ou instalará elas.", | ||||||
|  |     "cancelled": "Cancelado", | ||||||
|  |     "appAlreadyAdded": "O app já foi adicionado", | ||||||
|  |     "alreadyUpToDateQuestion": "O app já está atualizado?", | ||||||
|  |     "addApp": "Adicionar app", | ||||||
|  |     "appSourceURL": "URL da fonte do app", | ||||||
|  |     "error": "Erro", | ||||||
|  |     "add": "Adicionar", | ||||||
|  |     "searchSomeSourcesLabel": "Pesquisar (somente algumas fontes)", | ||||||
|  |     "search": "Pesquisar", | ||||||
|  |     "additionalOptsFor": "Opções adicionais de {}", | ||||||
|  |     "supportedSources": "Fontes suportadas", | ||||||
|  |     "trackOnlyInBrackets": "(somente rastreamento)", | ||||||
|  |     "searchableInBrackets": "(pesquisável)", | ||||||
|  |     "appsString": "Aplicativos", | ||||||
|  |     "noApps": "Nenhum app", | ||||||
|  |     "noAppsForFilter": "Nenhum app pro filtro", | ||||||
|  |     "byX": "Por {}", | ||||||
|  |     "percentProgress": "Progresso: {}%", | ||||||
|  |     "pleaseWait": "Por favor aguarde", | ||||||
|  |     "updateAvailable": "Atualização disponível", | ||||||
|  |     "notInstalled": "Não instalado", | ||||||
|  |     "pseudoVersion": "pseudo-versão", | ||||||
|  |     "selectAll": "Selecionar tudo", | ||||||
|  |     "deselectX": "Desselecionar {}", | ||||||
|  |     "xWillBeRemovedButRemainInstalled": "{} será removido do Obtainium mas continuará instalado no dispositivo.", | ||||||
|  |     "removeSelectedAppsQuestion": "Remover os apps selecionados?", | ||||||
|  |     "removeSelectedApps": "Remover apps selecionados", | ||||||
|  |     "updateX": "Atualizar {}", | ||||||
|  |     "installX": "Instalar {}", | ||||||
|  |     "markXTrackOnlyAsUpdated": "Marcar {}\n(somente rastreamento)\ncomo atualizado", | ||||||
|  |     "changeX": "Alterar {}", | ||||||
|  |     "installUpdateApps": "Instalar/atualizar apps", | ||||||
|  |     "installUpdateSelectedApps": "Instalar/atualizar apps selecionados", | ||||||
|  |     "markXSelectedAppsAsUpdated": "Marcar os {} apps selecionados como atualizados?", | ||||||
|  |     "no": "Não", | ||||||
|  |     "yes": "Sim", | ||||||
|  |     "markSelectedAppsUpdated": "Marcar apps selecionados como atualizados", | ||||||
|  |     "pinToTop": "Fixar ao topo", | ||||||
|  |     "unpinFromTop": "Desfixar do topo", | ||||||
|  |     "resetInstallStatusForSelectedAppsQuestion": "Redefinir o estado de instalação dos apps selecionados?", | ||||||
|  |     "installStatusOfXWillBeResetExplanation": "Os estados de instalação dos apps selecionados serão redefinidos.\n\nIsso pode ajudar quando a versão exibida no Obtainium está incorreta devido a atualizações malsucedidas ou outros problemas.", | ||||||
|  |     "customLinkMessage": "Esses links funcionarão em dispositivos com o Obtainium instalado", | ||||||
|  |     "shareAppConfigLinks": "Compartilhar configuração do app como um link HTML", | ||||||
|  |     "shareSelectedAppURLs": "Compartilhar as URLs dos apps selecionados", | ||||||
|  |     "resetInstallStatus": "Redefinir estado de instalação", | ||||||
|  |     "more": "Mais", | ||||||
|  |     "removeOutdatedFilter": "Remover filtro de apps desatualizados", | ||||||
|  |     "showOutdatedOnly": "Mostrar somente apps desatualizados", | ||||||
|  |     "filter": "Filtro", | ||||||
|  |     "filterApps": "Filtrar apps", | ||||||
|  |     "appName": "Nome do app", | ||||||
|  |     "author": "Autor", | ||||||
|  |     "upToDateApps": "Apps atualizados", | ||||||
|  |     "nonInstalledApps": "Apps não instalados", | ||||||
|  |     "importExport": "Importar/Exportar", | ||||||
|  |     "settings": "Configurações", | ||||||
|  |     "exportedTo": "Exportado para {}", | ||||||
|  |     "obtainiumExport": "Exportação do Obtainium", | ||||||
|  |     "invalidInput": "Entrada inválida", | ||||||
|  |     "importedX": "{} importado(s)", | ||||||
|  |     "obtainiumImport": "Importação do Obtainium", | ||||||
|  |     "importFromURLList": "Importar da lista de URLs", | ||||||
|  |     "searchQuery": "Consulta de pesquisa", | ||||||
|  |     "appURLList": "Lista de URLs dos apps", | ||||||
|  |     "line": "Linha", | ||||||
|  |     "searchX": "Pesquisar {}", | ||||||
|  |     "noResults": "Nenhum resultado encontrado", | ||||||
|  |     "importX": "Importar {}", | ||||||
|  |     "importedAppsIdDisclaimer": "Os apps importados podem ser exibidos incorretamente como se não estivessem instalados.\nPara resolver isso, reinstale eles pelo Obtainium.\nIsso não afetará os dados dos apps.\n\nIsso somente afeta a URL e os métodos de importação de terceiros.", | ||||||
|  |     "importErrors": "Erros de importação", | ||||||
|  |     "importedXOfYApps": "{} de {} foram importados.", | ||||||
|  |     "followingURLsHadErrors": "As seguintes URLs tiveram erros:", | ||||||
|  |     "selectURL": "Selecionar URL", | ||||||
|  |     "selectURLs": "Selecionar URLs", | ||||||
|  |     "pick": "Escolher", | ||||||
|  |     "theme": "Tema", | ||||||
|  |     "dark": "Escuro", | ||||||
|  |     "light": "Claro", | ||||||
|  |     "followSystem": "Seguir o sistema", | ||||||
|  |     "followSystemThemeExplanation": "Só é possível seguir o tema do sistema ao usar aplicativos de terceiros", | ||||||
|  |     "useBlackTheme": "Usar o tema escuro de preto profundo", | ||||||
|  |     "appSortBy": "Ordenar apps por", | ||||||
|  |     "authorName": "Autor/nome", | ||||||
|  |     "nameAuthor": "Nome/autor", | ||||||
|  |     "asAdded": "Como adicionados", | ||||||
|  |     "appSortOrder": "Ordem dos apps", | ||||||
|  |     "ascending": "Crescente", | ||||||
|  |     "descending": "Decrescente", | ||||||
|  |     "bgUpdateCheckInterval": "Intervalo de busca por atualizações em segundo plano", | ||||||
|  |     "neverManualOnly": "Nunca - somente manualmente", | ||||||
|  |     "appearance": "Aparência", | ||||||
|  |     "showWebInAppView": "Mostrar a fonte da pagina web na tela de apps", | ||||||
|  |     "pinUpdates": "Fixar atualizações no topo da tela de apps", | ||||||
|  |     "updates": "Atualizações", | ||||||
|  |     "sourceSpecific": "Específico à fonte", | ||||||
|  |     "appSource": "Fonte do app", | ||||||
|  |     "noLogs": "Nenhum registro", | ||||||
|  |     "appLogs": "Registros do app", | ||||||
|  |     "close": "Fechar", | ||||||
|  |     "share": "Compartilhar", | ||||||
|  |     "appNotFound": "O app não foi encontrado", | ||||||
|  |     "obtainiumExportHyphenatedLowercase": "obtainium-export", | ||||||
|  |     "pickAnAPK": "Selecione um APK", | ||||||
|  |     "appHasMoreThanOnePackage": "{} tem mais de um pacote:", | ||||||
|  |     "deviceSupportsXArch": "Seu dispositivo suporta a arquitetura de CPU {}.", | ||||||
|  |     "deviceSupportsFollowingArchs": "Seu dispositivo suporta as seguintes arquiteturas de CPU:", | ||||||
|  |     "warning": "Alerta", | ||||||
|  |     "sourceIsXButPackageFromYPrompt": "A fonte do app é '{}' mas o pacote de lançamento vem de '{}'. Continuar mesmo assim?", | ||||||
|  |     "updatesAvailable": "Atualizações disponíveis", | ||||||
|  |     "updatesAvailableNotifDescription": "Notifica o usuário que atualizações estão disponíveis para um ou mais apps rastreados pelo Obtainium", | ||||||
|  |     "noNewUpdates": "Nenhuma atualização disponível.", | ||||||
|  |     "xHasAnUpdate": "{} tem uma atualização.", | ||||||
|  |     "appsUpdated": "Apps atualizados", | ||||||
|  |     "appsNotUpdated": "Falhou ao atualizar os aplicativos", | ||||||
|  |     "appsUpdatedNotifDescription": "Notifica o usuário que atualizações de um ou mais apps foram aplicadas em segundo plano", | ||||||
|  |     "xWasUpdatedToY": "{} foi atualizado para a versão {}.", | ||||||
|  |     "xWasNotUpdatedToY": "Falha ao atualizar {} para a versão {}.", | ||||||
|  |     "errorCheckingUpdates": "Ocorreu um erro ao buscar atualizações", | ||||||
|  |     "errorCheckingUpdatesNotifDescription": "Uma notificação que mostra quando a busca de atualizações em segundo plano falha", | ||||||
|  |     "appsRemoved": "Apps removidos", | ||||||
|  |     "appsRemovedNotifDescription": "Notifica o usuário que um ou mais apps foram removidos devido a erros ao carregá-los", | ||||||
|  |     "xWasRemovedDueToErrorY": "{} for removido devido ao erro: {}", | ||||||
|  |     "completeAppInstallation": "Concluir instalação do app", | ||||||
|  |     "obtainiumMustBeOpenToInstallApps": "O Obtainium precisa estar aberto para instalar apps", | ||||||
|  |     "completeAppInstallationNotifDescription": "Pede pro usuário voltar ao Obtainium para concluir a instalação de um app", | ||||||
|  |     "checkingForUpdates": "Buscando atualizações", | ||||||
|  |     "checkingForUpdatesNotifDescription": "Notificação transitória que aparece ao buscar atualizações", | ||||||
|  |     "pleaseAllowInstallPerm": "Permita que o Obtainium instale apps", | ||||||
|  |     "trackOnly": "Somente rastreamento", | ||||||
|  |     "errorWithHttpStatusCode": "Erro {}", | ||||||
|  |     "versionCorrectionDisabled": "Correção de versão desativada (o plugin parece não funcionar)", | ||||||
|  |     "unknown": "Desconhecido", | ||||||
|  |     "none": "Nenhum", | ||||||
|  |     "never": "Nunca", | ||||||
|  |     "latestVersionX": "Mais recente: {}", | ||||||
|  |     "installedVersionX": "Instalado: {}", | ||||||
|  |     "lastUpdateCheckX": "Última busca por atualizações: {}", | ||||||
|  |     "remove": "Remover", | ||||||
|  |     "yesMarkUpdated": "Sim, marcar como atualizado", | ||||||
|  |     "fdroid": "Oficial do F-Droid", | ||||||
|  |     "appIdOrName": "ID do app ou nome", | ||||||
|  |     "appId": "ID do app", | ||||||
|  |     "appWithIdOrNameNotFound": "Nenhum app foi encontrado com aquele ID ou nome", | ||||||
|  |     "reposHaveMultipleApps": "Repositórios podem conter vários apps", | ||||||
|  |     "fdroidThirdPartyRepo": "Repositório de terceiros do F-Droid", | ||||||
|  |     "install": "Instalar", | ||||||
|  |     "markInstalled": "Marcar como instalado", | ||||||
|  |     "update": "Atualizar", | ||||||
|  |     "markUpdated": "Marcar como atualizado", | ||||||
|  |     "additionalOptions": "Opções adicionais", | ||||||
|  |     "disableVersionDetection": "Desativar detecção de versão", | ||||||
|  |     "noVersionDetectionExplanation": "Essa opção só seve ser usada para apps aonde a detecção de versão não funciona corretamente.", | ||||||
|  |     "downloadingX": "Baixando {}", | ||||||
|  |     "downloadX": "Baixar {}", | ||||||
|  |     "downloadedX": "{} foi baixado", | ||||||
|  |     "releaseAsset": "Item de lançamento", | ||||||
|  |     "downloadNotifDescription": "Notifica o usuário do progresso ao baixar um app", | ||||||
|  |     "noAPKFound": "Nenhum APK encontrado", | ||||||
|  |     "noVersionDetection": "Sem detecção de versão", | ||||||
|  |     "categorize": "Categorizar", | ||||||
|  |     "categories": "Categorias", | ||||||
|  |     "category": "Categoria", | ||||||
|  |     "noCategory": "Nenhuma categoria", | ||||||
|  |     "noCategories": "Nenhuma categoria", | ||||||
|  |     "deleteCategoriesQuestion": "Excluir categorias?", | ||||||
|  |     "categoryDeleteWarning": "Todos os apps em categorias excluídas ficarão sem categoria.", | ||||||
|  |     "addCategory": "Adicionar categoria", | ||||||
|  |     "label": "Rótulo", | ||||||
|  |     "language": "Idioma", | ||||||
|  |     "copiedToClipboard": "Copiado para a área de transferência", | ||||||
|  |     "storagePermissionDenied": "Permissão de armazenamento negada", | ||||||
|  |     "selectedCategorizeWarning": "Isso substituirá a configuração de categoria existente dos apps selecionados.", | ||||||
|  |     "filterAPKsByRegEx": "Filtrar APKs por expressão regular", | ||||||
|  |     "removeFromObtainium": "Remover do Obtainium", | ||||||
|  |     "uninstallFromDevice": "Desinstalar do dispositivo", | ||||||
|  |     "onlyWorksWithNonVersionDetectApps": "Funciona somente em apps com a detecção de versão desativada.", | ||||||
|  |     "releaseDateAsVersion": "Usar data de lançamento como número da versão", | ||||||
|  |     "releaseTitleAsVersion": "Usar título do lançamento como número da versão", | ||||||
|  |     "releaseDateAsVersionExplanation": "Essa opção só deve ser usada para apps quais a detecção de versão não funciona corretamente, mas uma data de lançamento está disponível.", | ||||||
|  |     "changes": "Alterações", | ||||||
|  |     "releaseDate": "Data de lançamento", | ||||||
|  |     "importFromURLsInFile": "Importar das URLs em arquivo (como OPML)", | ||||||
|  |     "versionDetectionExplanation": "Combinar o número da versão com a versão detectada pelo sistema", | ||||||
|  |     "versionDetection": "Detecção de versão", | ||||||
|  |     "standardVersionDetection": "Detecção de versão padrão", | ||||||
|  |     "groupByCategory": "Agrupar por categoria", | ||||||
|  |     "autoApkFilterByArch": "Tentar filtrar APKs pela arquitetura da CPU quando possível", | ||||||
|  |     "autoLinkFilterByArch": "Tentar filtrar links pela arquitetura da CPU quando possível", | ||||||
|  |     "overrideSource": "Sobrescrever fonte", | ||||||
|  |     "dontShowAgain": "Não mostrar isso novamente", | ||||||
|  |     "dontShowTrackOnlyWarnings": "Não mostrar alertas de \"somente rastreamento\"", | ||||||
|  |     "dontShowAPKOriginWarnings": "Não mostrar alertas de origem dos APKs", | ||||||
|  |     "moveNonInstalledAppsToBottom": "Mover apps não instalados ao final da tela de apps", | ||||||
|  |     "gitlabPATLabel": "Token de acesso pessoal do GitLab", | ||||||
|  |     "about": "Sobre", | ||||||
|  |     "requiresCredentialsInSettings": "{} precisa de credenciais adicionais (nas Configurações)", | ||||||
|  |     "checkOnStart": "Buscar atualizações ao abrir o app", | ||||||
|  |     "tryInferAppIdFromCode": "Tentar inferir o ID do app pelo código fonte", | ||||||
|  |     "removeOnExternalUninstall": "Remover automaticamente apps desinstalados externamente", | ||||||
|  |     "pickHighestVersionCode": "Selecionar APK de versão mais alta automaticamente", | ||||||
|  |     "checkUpdateOnDetailPage": "Buscar atualizações ao abrir a tela de detalhes de um app", | ||||||
|  |     "disablePageTransitions": "Desativar animações de transição de tela", | ||||||
|  |     "reversePageTransitions": "Inverter animações de transição de tela", | ||||||
|  |     "minStarCount": "Número de estrelas mínimo", | ||||||
|  |     "addInfoBelow": "Adicione essa informação abaixo.", | ||||||
|  |     "addInfoInSettings": "Adicione essa informação nas Configurações.", | ||||||
|  |     "githubSourceNote": "O limite de taxa do GitHub pode ser evitado ao usar uma chave de API.", | ||||||
|  |     "sortByLastLinkSegment": "Ordenar somente pelo ultimo segmento do link", | ||||||
|  |     "filterReleaseNotesByRegEx": "Filtrar notas de lançamento por expressão regular", | ||||||
|  |     "customLinkFilterRegex": "Filtro de link de APK personalizado por expressão regular (padrão '.apk$')", | ||||||
|  |     "appsPossiblyUpdated": "Tentativas de atualização de apps", | ||||||
|  |     "appsPossiblyUpdatedNotifDescription": "Notifica o usuário que atualizações de um ou mais apps podem ter sido aplicadas em segundo plano", | ||||||
|  |     "xWasPossiblyUpdatedToY": "{} pode ter sido atualizado para a versão {}.", | ||||||
|  |     "enableBackgroundUpdates": "Ativar atualizações em segundo plano", | ||||||
|  |     "backgroundUpdateReqsExplanation": "Atualizações em segundo plano podem não funcionar com todos os apps.", | ||||||
|  |     "backgroundUpdateLimitsExplanation": "O sucesso de uma instalação em segundo plano só pode ser determinada ao abrir o Obtainium.", | ||||||
|  |     "verifyLatestTag": "Verificar a tag 'mais recente'", | ||||||
|  |     "intermediateLinkRegex": "Filtrar por um link 'intermediário' para visitar", | ||||||
|  |     "filterByLinkText": "Filtrar links por texto do link", | ||||||
|  |     "intermediateLinkNotFound": "Link intermediário não encontrado", | ||||||
|  |     "intermediateLink": "Link intermediário", | ||||||
|  |     "exemptFromBackgroundUpdates": "Isento de atualizações em segundo plano (caso ativadas)", | ||||||
|  |     "bgUpdatesOnWiFiOnly": "Desativar atualizações em segundo plano fora do Wi-Fi", | ||||||
|  |     "bgUpdatesWhileChargingOnly": "Desativar atualizações em segundo plano fora do carregador", | ||||||
|  |     "autoSelectHighestVersionCode": "Selecionar automaticamente APK com o código de versão mais alto", | ||||||
|  |     "versionExtractionRegEx": "ExReg de extração do número da versão", | ||||||
|  |     "trimVersionString": "Cortar número da versal com ExReg", | ||||||
|  |     "matchGroupToUseForX": "Corresponder grupo para o uso em \"{}\"", | ||||||
|  |     "matchGroupToUse": "Corresponder grupo para o uso para a extração do número da versão por ExReg", | ||||||
|  |     "highlightTouchTargets": "Acentuar alvos de toque menos óbvios", | ||||||
|  |     "pickExportDir": "Selecionar pasta de exportação", | ||||||
|  |     "autoExportOnChanges": "Exportar automaticamente ao ocorrer alterações", | ||||||
|  |     "includeSettings": "Incluir configurações", | ||||||
|  |     "filterVersionsByRegEx": "Filtrar versões por expressão regular", | ||||||
|  |     "trySelectingSuggestedVersionCode": "Tente selecionar o APK com o código de versão sugerido", | ||||||
|  |     "dontSortReleasesList": "Manter ordem de lançamento da API", | ||||||
|  |     "reverseSort": "Ordem inversa", | ||||||
|  |     "takeFirstLink": "Usar o primeiro link", | ||||||
|  |     "skipSort": "Pular ordenação", | ||||||
|  |     "debugMenu": "Menu de depuração", | ||||||
|  |     "bgTaskStarted": "Tarefa em segundo plano iniada - verifique os registros.", | ||||||
|  |     "runBgCheckNow": "Executar busca por atualizações em segundo plano agora", | ||||||
|  |     "versionExtractWholePage": "Aplicar ExReg de extração de número de versão à página inteira", | ||||||
|  |     "installing": "Instalando", | ||||||
|  |     "skipUpdateNotifications": "Pular notificações de atualização", | ||||||
|  |     "updatesAvailableNotifChannel": "Atualizações disponíveis", | ||||||
|  |     "appsUpdatedNotifChannel": "Apps atualizados", | ||||||
|  |     "appsPossiblyUpdatedNotifChannel": "Tentativas de atualização de apps", | ||||||
|  |     "errorCheckingUpdatesNotifChannel": "Erro ao buscar atualizações", | ||||||
|  |     "appsRemovedNotifChannel": "Apps removidos", | ||||||
|  |     "downloadingXNotifChannel": "Baixando {}", | ||||||
|  |     "completeAppInstallationNotifChannel": "Concluir instalação do app", | ||||||
|  |     "checkingForUpdatesNotifChannel": "Buscando atualizações", | ||||||
|  |     "onlyCheckInstalledOrTrackOnlyApps": "Buscar atualizações somente para apps instalados e de somente rastreamento", | ||||||
|  |     "supportFixedAPKURL": "Suportar URLs de APK fixas", | ||||||
|  |     "selectX": "Selecionar {}", | ||||||
|  |     "parallelDownloads": "Permitir downloads em paralelo", | ||||||
|  |     "useShizuku": "Usar Shizuku ou Sui para instalação", | ||||||
|  |     "shizukuBinderNotFound": "Serviço Shizuku não está em execução", | ||||||
|  |     "shizukuOld": "Versão do Shizuku antiga (<11) - atualize", | ||||||
|  |     "shizukuOldAndroidWithADB": "Shizuku sendo executado no Android < 8.1 com ADB - atualize o Android ou use o Sui", | ||||||
|  |     "shizukuPretendToBeGooglePlay": "Definir Google Play como a fonte de instalação (se o Shizuku é usado)", | ||||||
|  |     "useSystemFont": "Usar a fonte do sistema", | ||||||
|  |     "useVersionCodeAsOSVersion": "Usar código de versão do app como a versão detectada pelo sistema", | ||||||
|  |     "requestHeader": "Cabeçalho da solicitação", | ||||||
|  |     "useLatestAssetDateAsReleaseDate": "Usar o envio de item mais recente como a data de lançamento", | ||||||
|  |     "defaultPseudoVersioningMethod": "Método de pseudo-versão padrão", | ||||||
|  |     "partialAPKHash": "Hash do APK parcial", | ||||||
|  |     "APKLinkHash": "Hash do link do APK", | ||||||
|  |     "directAPKLink": "Link direto ao APK", | ||||||
|  |     "pseudoVersionInUse": "Uma pseudo-versão está em uso", | ||||||
|  |     "installed": "Instalado", | ||||||
|  |     "latest": "Mais recente", | ||||||
|  |     "invertRegEx": "Inverter expressão regular", | ||||||
|  |     "note": "Observação", | ||||||
|  |     "selfHostedNote": "O menu de opções \"{}\" pode ser usado para alcançar instâncias hospedadas-por-você/personalizadas de qualquer fonte.", | ||||||
|  |     "badDownload": "O APK não pode ser interpretado (incompatível ou baixado parcialmente)", | ||||||
|  |     "beforeNewInstallsShareToAppVerifier": "Compartilhar apps novos com o AppVerifier (se disponível)", | ||||||
|  |     "appVerifierInstructionToast": "Compartilhe com o AppVerifier, e volte aqui ao estar pronto.", | ||||||
|  |     "wiki": "Ajuda/Wiki", | ||||||
|  |     "crowdsourcedConfigsLabel": "Configurações de app pela comunidade (use ao seu próprio risco)", | ||||||
|  |     "crowdsourcedConfigsShort": "Configurações de app da comunidade", | ||||||
|  |     "allowInsecure": "Permitir solicitações de HTTP inseguras", | ||||||
|  |     "stayOneVersionBehind": "Ficar uma versão antes da mais recente", | ||||||
|  |     "useFirstApkOfVersion": "Seleção automática do primeiro de vários APKs", | ||||||
|  |     "refreshBeforeDownload": "Atualizar detalhes do app antes de baixar", | ||||||
|  |     "tencentAppStore": "Loja de Apps da Tencent", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "Vivo App Store (CN)", | ||||||
|  |     "name": "Nome", | ||||||
|  |     "smartname": "Nome (inteligente)", | ||||||
|  |     "sortMethod": "Método de ordenação", | ||||||
|  |     "welcome": "Boas vindas", | ||||||
|  |     "documentationLinksNote": "A página do Obtainium no GitHub visível abaixo contém links de vídeos, artigos, discussões, e outros recursos que podem te ajudar ao usar o app.", | ||||||
|  |     "batteryOptimizationNote": "Observe que os downloads em segundo plano podem funcionar de forma mais confiável se você desativar as otimizações de bateria do sistema operacional para o Obtainium.", | ||||||
|  |     "fileDeletionError": "Falha ao excluir o arquivo (tente excluí-lo manualmente e tente novamente): \"{}\"", | ||||||
|  |     "removeAppQuestion": { | ||||||
|  |         "one": "Remover app?", | ||||||
|  |         "other": "Remover apps?" | ||||||
|  |     }, | ||||||
|  |     "tooManyRequestsTryAgainInMinutes": { | ||||||
|  |         "one": "Muitas solicitações (limitado) - tente novamente em {} minuto", | ||||||
|  |         "other": "Muitas solicitações (limitado) - tente novamente em {} minutos" | ||||||
|  |     }, | ||||||
|  |     "bgUpdateGotErrorRetryInMinutes": { | ||||||
|  |         "one": "A busca de atualizações em segundo plano encontrou um {}, será agendado uma nova tentativa em {} minuto", | ||||||
|  |         "other": "A busca de atualizações em segundo plano encontrou um {}, será agendado uma nova tentativa em {} minutos" | ||||||
|  |     }, | ||||||
|  |     "bgCheckFoundUpdatesWillNotifyIfNeeded": { | ||||||
|  |         "one": "Verificação da atualização do BG encontrada {} atualização - notificará o usuário se necessário", | ||||||
|  |         "other": "Verificação de atualização do BG encontrou atualizações {} - notificará o usuário se necessário" | ||||||
|  |     }, | ||||||
|  |     "apps": { | ||||||
|  |         "one": "{} app", | ||||||
|  |         "other": "{} apps" | ||||||
|  |     }, | ||||||
|  |     "url": { | ||||||
|  |         "one": "{} URL", | ||||||
|  |         "other": "{} URLs" | ||||||
|  |     }, | ||||||
|  |     "minute": { | ||||||
|  |         "one": "{} minuto", | ||||||
|  |         "other": "{} minutos" | ||||||
|  |     }, | ||||||
|  |     "hour": { | ||||||
|  |         "one": "{} hora", | ||||||
|  |         "other": "{} horas" | ||||||
|  |     }, | ||||||
|  |     "day": { | ||||||
|  |         "one": "{} dia", | ||||||
|  |         "other": "{} dias" | ||||||
|  |     }, | ||||||
|  |     "clearedNLogsBeforeXAfterY": { | ||||||
|  |         "one": "Limpou {n} registros (before = {before}, after = {after})", | ||||||
|  |         "other": "Limpou {n} registros (antes = {before}, depois = {after})" | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesAvailable": { | ||||||
|  |         "one": "{} e mais 1 app têm atualizações.", | ||||||
|  |         "other": "{} e mais {} apps têm atualizações." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesInstalled": { | ||||||
|  |         "one": "{} e mais 1 app foram atualizados.", | ||||||
|  |         "other": "{} e mais {} apps foram atualizados." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesFailed": { | ||||||
|  |         "one": "Falha ao atualizar {} e mais 1 app.", | ||||||
|  |         "other": "Falha ao atualizar {} e mais {} apps." | ||||||
|  |     }, | ||||||
|  |     "xAndNMoreUpdatesPossiblyInstalled": { | ||||||
|  |         "one": "{} e mais 1 app podem ter sido atualizados.", | ||||||
|  |         "other": "{} e mais {} apps podem ter sido atualizados." | ||||||
|  |     }, | ||||||
|  |     "apk": { | ||||||
|  |         "one": "{} APK", | ||||||
|  |         "other": "{} APKs" | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Detecção de versão padrão", |     "standardVersionDetection": "Detecção de versão padrão", | ||||||
|     "groupByCategory": "Agroupar por categoria", |     "groupByCategory": "Agroupar por categoria", | ||||||
|     "autoApkFilterByArch": "Tente filtrar APKs por arquitetura de CPU, se possível", |     "autoApkFilterByArch": "Tente filtrar APKs por arquitetura de CPU, se possível", | ||||||
|  |     "autoLinkFilterByArch": "Tentativa de filtrar as ligações por arquitetura de CPU, se possível", | ||||||
|     "overrideSource": "Substituir fonte", |     "overrideSource": "Substituir fonte", | ||||||
|     "dontShowAgain": "Não mostrar isso novamente", |     "dontShowAgain": "Não mostrar isso novamente", | ||||||
|     "dontShowTrackOnlyWarnings": "Não mostrar avisos 'Apenas monitorar'", |     "dontShowTrackOnlyWarnings": "Não mostrar avisos 'Apenas monitorar'", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Configurações de aplicações com base em crowdsourcing", |     "crowdsourcedConfigsShort": "Configurações de aplicações com base em crowdsourcing", | ||||||
|     "allowInsecure": "Permitir pedidos HTTP inseguros", |     "allowInsecure": "Permitir pedidos HTTP inseguros", | ||||||
|     "stayOneVersionBehind": "Manter-se uma versão atrás da mais recente", |     "stayOneVersionBehind": "Manter-se uma versão atrás da mais recente", | ||||||
|  |     "useFirstApkOfVersion": "Seleção automática do primeiro de vários APKs", | ||||||
|     "refreshBeforeDownload": "Atualizar os detalhes da aplicação antes da transferência", |     "refreshBeforeDownload": "Atualizar os detalhes da aplicação antes da transferência", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Nome", | ||||||
|  |     "smartname": "Nome (Smart)", | ||||||
|  |     "sortMethod": "Método de ordenação", | ||||||
|  |     "welcome": "Bem-vindo", | ||||||
|  |     "documentationLinksNote": "A página do Obtainium no GitHub com a ligação abaixo contém ligações para vídeos, artigos, discussões e outros recursos que o ajudarão a compreender como utilizar a aplicação.", | ||||||
|  |     "batteryOptimizationNote": "Note que os downloads em segundo plano podem funcionar de forma mais fiável se desativar as optimizações da bateria do SO para o Obtainium.", | ||||||
|  |     "fileDeletionError": "Falha ao eliminar o ficheiro (tente eliminá-lo manualmente e depois tente novamente): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Remover aplicativo?", |         "one": "Remover aplicativo?", | ||||||
|         "other": "Remover aplicativos?" |         "other": "Remover aplicativos?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Стандартное", |     "standardVersionDetection": "Стандартное", | ||||||
|     "groupByCategory": "Группировать по категориям", |     "groupByCategory": "Группировать по категориям", | ||||||
|     "autoApkFilterByArch": "Попытаться отфильтровать APK-файлы по архитектуре процессора", |     "autoApkFilterByArch": "Попытаться отфильтровать APK-файлы по архитектуре процессора", | ||||||
|  |     "autoLinkFilterByArch": "Попытайтесь отфильтровать ссылки по архитектуре процессора, если это возможно", | ||||||
|     "overrideSource": "Переопределить источник", |     "overrideSource": "Переопределить источник", | ||||||
|     "dontShowAgain": "Не показывать снова", |     "dontShowAgain": "Не показывать снова", | ||||||
|     "dontShowTrackOnlyWarnings": "Не показывать предупреждения о только отслеживаемых приложениях", |     "dontShowTrackOnlyWarnings": "Не показывать предупреждения о только отслеживаемых приложениях", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Конфиги приложений с помощью краудсорсинга", |     "crowdsourcedConfigsShort": "Конфиги приложений с помощью краудсорсинга", | ||||||
|     "allowInsecure": "Разрешить небезопасные HTTP-запросы", |     "allowInsecure": "Разрешить небезопасные HTTP-запросы", | ||||||
|     "stayOneVersionBehind": "Не отставайте от последней версии", |     "stayOneVersionBehind": "Не отставайте от последней версии", | ||||||
|  |     "useFirstApkOfVersion": "Автоматический выбор первого из нескольких APK", | ||||||
|     "refreshBeforeDownload": "Обновляйте информацию о приложении перед загрузкой", |     "refreshBeforeDownload": "Обновляйте информацию о приложении перед загрузкой", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Имя", | ||||||
|  |     "smartname": "Имя (умное)", | ||||||
|  |     "sortMethod": "Метод сортировки", | ||||||
|  |     "welcome": "Добро пожаловать", | ||||||
|  |     "documentationLinksNote": "На странице Obtainium GitHub, ссылка на которую приведена ниже, содержатся ссылки на видео, статьи, обсуждения и другие ресурсы, которые помогут вам понять, как пользоваться приложением.", | ||||||
|  |     "batteryOptimizationNote": "Обратите внимание, что фоновая загрузка может работать более надежно, если отключить оптимизацию батареи ОС для Obtainium.", | ||||||
|  |     "fileDeletionError": "Не удалось удалить файл (попробуйте удалить его вручную, а затем повторите попытку): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Удалить приложение?", |         "one": "Удалить приложение?", | ||||||
|         "other": "Удалить приложения?" |         "other": "Удалить приложения?" | ||||||
|   | |||||||
| @@ -13,7 +13,14 @@ const neverAutoTranslate = { | |||||||
|     obtainiumExportHyphenatedLowercase: ['*'], |     obtainiumExportHyphenatedLowercase: ['*'], | ||||||
|     theme: ['de'], |     theme: ['de'], | ||||||
|     appId: ['de'], |     appId: ['de'], | ||||||
|     placeholder: ['pl'] |     placeholder: ['pl'], | ||||||
|  |     importExport: ['fr'], | ||||||
|  |     url: ['fr'], | ||||||
|  |     vivoAppStore: ['cs', 'ja', 'ko'], | ||||||
|  |     coolapk: ['ja'], | ||||||
|  |     obtainiumImport: ['nl'], | ||||||
|  |     appLogs: ['nl'], | ||||||
|  |     tencentAppStore: ['*'] | ||||||
| } | } | ||||||
|  |  | ||||||
| const translateText = async (text, targetLang, authKey) => { | const translateText = async (text, targetLang, authKey) => { | ||||||
| @@ -76,40 +83,49 @@ const main = async () => { | |||||||
|         const translationKeys = Object.keys(templateTranslation) |         const translationKeys = Object.keys(templateTranslation) | ||||||
|         for (let j in translationKeys) { |         for (let j in translationKeys) { | ||||||
|             const k = translationKeys[j] |             const k = translationKeys[j] | ||||||
|             if (JSON.stringify(thisTranslation[k]) == JSON.stringify(templateTranslation[k])) { |             try { | ||||||
|                 const lang = file.split('/').pop().split('.')[0] |                 if (JSON.stringify(thisTranslation[k]) == JSON.stringify(templateTranslation[k])) { | ||||||
|                 if (!neverAutoTranslate[k] || (neverAutoTranslate[k].indexOf('*') < 0 && neverAutoTranslate[k].indexOf(lang) < 0)) { |                     const lang = file.split('/').pop().split('.')[0] | ||||||
|                     const reportLine = `${file} :::: ${k} :::: ${JSON.stringify(thisTranslation[k])}` |                     if (!neverAutoTranslate[k] || (neverAutoTranslate[k].indexOf('*') < 0 && neverAutoTranslate[k].indexOf(lang) < 0)) { | ||||||
|                     if (deeplAPIKey) { |                         const reportLine = `${file} :::: ${k} :::: ${JSON.stringify(thisTranslation[k])}` | ||||||
|                         const translateFunc = async (str) => { |                         if (deeplAPIKey) { | ||||||
|                             const response = await translateText(str, lang, deeplAPIKey) |                             const translateFunc = async (str) => { | ||||||
|                             if (response.translations) { |                                 await new Promise((resolve, reject) => { | ||||||
|                                 return response.translations[0].text |                                     setTimeout(() => { | ||||||
|                             } else { |                                         resolve() | ||||||
|                                 throw JSON.stringify(response) |                                     }, Math.random() * 10000); // Try to avoid rate limit | ||||||
|                             } |                                 }) | ||||||
|                         } |                                 const response = await translateText(str, lang, deeplAPIKey) | ||||||
|                         try { |                                 if (response.translations) { | ||||||
|                             if (typeof templateTranslation[k] == 'string') { |                                     return response.translations[0].text | ||||||
|                                 thisTranslation[k] = await translateFunc(thisTranslation[k]) |                                 } else { | ||||||
|                             } else { |                                     throw JSON.stringify(response) | ||||||
|                                 const subKeys = Object.keys(templateTranslation[k]) |  | ||||||
|                                 for (let n in subKeys) { |  | ||||||
|                                     const kk = subKeys[n] |  | ||||||
|                                     thisTranslation[k][kk] = await translateFunc(thisTranslation[k][kk]) |  | ||||||
|                                 } |                                 } | ||||||
|                             } |                             } | ||||||
|                         } catch (e) { |                             try { | ||||||
|                             if (typeof e == 'string') { |                                 if (typeof templateTranslation[k] == 'string') { | ||||||
|                                 console.log(`${reportLine} :::: ${e}`) |                                     thisTranslation[k] = await translateFunc(thisTranslation[k]) | ||||||
|                             } else { |                                 } else { | ||||||
|                                 throw e |                                     const subKeys = Object.keys(templateTranslation[k]) | ||||||
|  |                                     for (let n in subKeys) { | ||||||
|  |                                         const kk = subKeys[n] | ||||||
|  |                                         thisTranslation[k][kk] = await translateFunc(thisTranslation[k][kk]) | ||||||
|  |                                     } | ||||||
|  |                                 } | ||||||
|  |                             } catch (e) { | ||||||
|  |                                 if (typeof e == 'string') { | ||||||
|  |                                     console.log(`${reportLine} :::: ${e}`) | ||||||
|  |                                 } else { | ||||||
|  |                                     throw e | ||||||
|  |                                 } | ||||||
|                             } |                             } | ||||||
|  |                         } else { | ||||||
|  |                             console.log(reportLine) | ||||||
|                         } |                         } | ||||||
|                     } else { |  | ||||||
|                         console.log(reportLine) |  | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|  |             } catch (err) { | ||||||
|  |                 console.error(err) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         fs.writeFileSync(file, `${JSON.stringify(thisTranslation, null, '    ')}\n`) |         fs.writeFileSync(file, `${JSON.stringify(thisTranslation, null, '    ')}\n`) | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Standardversionsdetektering", |     "standardVersionDetection": "Standardversionsdetektering", | ||||||
|     "groupByCategory": "Gruppera via Kategori", |     "groupByCategory": "Gruppera via Kategori", | ||||||
|     "autoApkFilterByArch": "Försök att filtrera APK-filer efter CPU-arkitektur om möjligt", |     "autoApkFilterByArch": "Försök att filtrera APK-filer efter CPU-arkitektur om möjligt", | ||||||
|  |     "autoLinkFilterByArch": "Försök att filtrera länkar efter CPU-arkitektur om möjligt", | ||||||
|     "overrideSource": "Överskrid Källa", |     "overrideSource": "Överskrid Källa", | ||||||
|     "dontShowAgain": "Visa inte detta igen", |     "dontShowAgain": "Visa inte detta igen", | ||||||
|     "dontShowTrackOnlyWarnings": "Visa inte 'Följ-Endast' varningar", |     "dontShowTrackOnlyWarnings": "Visa inte 'Följ-Endast' varningar", | ||||||
| @@ -255,7 +256,7 @@ | |||||||
|     "intermediateLinkNotFound": "Mellanlänk hittades inte", |     "intermediateLinkNotFound": "Mellanlänk hittades inte", | ||||||
|     "intermediateLink": "Mellanlänk", |     "intermediateLink": "Mellanlänk", | ||||||
|     "exemptFromBackgroundUpdates": "Undta från bakgrundsuppdateringar (om aktiverad)", |     "exemptFromBackgroundUpdates": "Undta från bakgrundsuppdateringar (om aktiverad)", | ||||||
|     "bgUpdatesOnWiFiOnly": "Inaktivera Bakgrundsuppdateringar utan WiFi", |     "bgUpdatesOnWiFiOnly": "Inaktivera Bakgrundsuppdateringar utan Wi-Fi", | ||||||
|     "bgUpdatesWhileChargingOnly": "Inaktivera bakgrundsuppdateringar när du inte laddar", |     "bgUpdatesWhileChargingOnly": "Inaktivera bakgrundsuppdateringar när du inte laddar", | ||||||
|     "autoSelectHighestVersionCode": "Välj automatiskt högsta versionskod APK", |     "autoSelectHighestVersionCode": "Välj automatiskt högsta versionskod APK", | ||||||
|     "versionExtractionRegEx": "Version Extraction RegEx", |     "versionExtractionRegEx": "Version Extraction RegEx", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Appkonfigurationer med hjälp av crowdsourcing", |     "crowdsourcedConfigsShort": "Appkonfigurationer med hjälp av crowdsourcing", | ||||||
|     "allowInsecure": "Tillåt osäkra HTTP-förfrågningar", |     "allowInsecure": "Tillåt osäkra HTTP-förfrågningar", | ||||||
|     "stayOneVersionBehind": "Håll dig en version bakom den senaste", |     "stayOneVersionBehind": "Håll dig en version bakom den senaste", | ||||||
|  |     "useFirstApkOfVersion": "Välj automatiskt den första av flera APK:er", | ||||||
|     "refreshBeforeDownload": "Uppdatera appdetaljerna före nedladdning", |     "refreshBeforeDownload": "Uppdatera appdetaljerna före nedladdning", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Namn", | ||||||
|  |     "smartname": "Namn (Smart)", | ||||||
|  |     "sortMethod": "Sorteringsmetod", | ||||||
|  |     "welcome": "Välkommen", | ||||||
|  |     "documentationLinksNote": "Obtainium GitHub-sidan som länkas nedan innehåller länkar till videor, artiklar, diskussioner och andra resurser som hjälper dig att förstå hur du använder appen.", | ||||||
|  |     "batteryOptimizationNote": "Observera att nedladdningar i bakgrunden kan fungera mer tillförlitligt om du inaktiverar OS-batterioptimeringar för Obtainium.", | ||||||
|  |     "fileDeletionError": "Misslyckades med att radera filen (försök radera den manuellt och försök sedan igen): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Ta Bort App?", |         "one": "Ta Bort App?", | ||||||
|         "other": "Ta Bort Appar?" |         "other": "Ta Bort Appar?" | ||||||
|   | |||||||
| @@ -221,6 +221,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", | ||||||
|  |     "autoLinkFilterByArch": "Mümkünse bağlantıları CPU mimarisine göre filtrelemeye çalışın", | ||||||
|     "overrideSource": "Öncelenecek Kaynak", |     "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", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Kitle Kaynaklı Uygulama Yapılandırmaları", |     "crowdsourcedConfigsShort": "Kitle Kaynaklı Uygulama Yapılandırmaları", | ||||||
|     "allowInsecure": "Güvensiz HTTP isteklerine izin ver", |     "allowInsecure": "Güvensiz HTTP isteklerine izin ver", | ||||||
|     "stayOneVersionBehind": "En son sürümün bir sürüm gerisinde kalın", |     "stayOneVersionBehind": "En son sürümün bir sürüm gerisinde kalın", | ||||||
|  |     "useFirstApkOfVersion": "Birden fazla APK arasından ilkini otomatik seçme", | ||||||
|     "refreshBeforeDownload": "İndirmeden önce uygulama ayrıntılarını yenileyin", |     "refreshBeforeDownload": "İndirmeden önce uygulama ayrıntılarını yenileyin", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "İsim", | ||||||
|  |     "smartname": "İsim (Akıllı)", | ||||||
|  |     "sortMethod": "Sıralama Yöntemi", | ||||||
|  |     "welcome": "Hoş geldiniz", | ||||||
|  |     "documentationLinksNote": "Aşağıda bağlantısı verilen Obtainium GitHub sayfası, uygulamayı nasıl kullanacağınızı anlamanıza yardımcı olacak videolara, makalelere, tartışmalara ve diğer kaynaklara bağlantılar içerir.", | ||||||
|  |     "batteryOptimizationNote": "Obtainium için işletim sistemi pil optimizasyonlarını devre dışı bırakırsanız arka planda indirmelerin daha güvenilir şekilde çalışabileceğini unutmayın.", | ||||||
|  |     "fileDeletionError": "Dosya silinemedi (elle silmeyi deneyin ve sonra tekrar deneyin): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Uygulamayı Kaldır?", |         "one": "Uygulamayı Kaldır?", | ||||||
|         "other": "Uygulamaları Kaldır?" |         "other": "Uygulamaları Kaldır?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Стандартне визначення версії", |     "standardVersionDetection": "Стандартне визначення версії", | ||||||
|     "groupByCategory": "Групувати за категоріями", |     "groupByCategory": "Групувати за категоріями", | ||||||
|     "autoApkFilterByArch": "Спробувати фільтрувати APK за архітектурою ЦП, якщо можливо", |     "autoApkFilterByArch": "Спробувати фільтрувати APK за архітектурою ЦП, якщо можливо", | ||||||
|  |     "autoLinkFilterByArch": "Спробуйте відфільтрувати посилання за архітектурою процесора, якщо це можливо", | ||||||
|     "overrideSource": "Перевизначити джерело", |     "overrideSource": "Перевизначити джерело", | ||||||
|     "dontShowAgain": "Не показувати це знову", |     "dontShowAgain": "Не показувати це знову", | ||||||
|     "dontShowTrackOnlyWarnings": "Не показувати попередження про 'Тільки відстеження'", |     "dontShowTrackOnlyWarnings": "Не показувати попередження про 'Тільки відстеження'", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "Налаштування краудсорсингових додатків", |     "crowdsourcedConfigsShort": "Налаштування краудсорсингових додатків", | ||||||
|     "allowInsecure": "Дозволити незахищені HTTP-запити", |     "allowInsecure": "Дозволити незахищені HTTP-запити", | ||||||
|     "stayOneVersionBehind": "Залишайтеся на одну версію актуальнішою", |     "stayOneVersionBehind": "Залишайтеся на одну версію актуальнішою", | ||||||
|  |     "useFirstApkOfVersion": "Автоматичний вибір першого з декількох APK", | ||||||
|     "refreshBeforeDownload": "Оновіть інформацію про програму перед завантаженням", |     "refreshBeforeDownload": "Оновіть інформацію про програму перед завантаженням", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Ім'я", | ||||||
|  |     "smartname": "Ім'я (Smart)", | ||||||
|  |     "sortMethod": "Метод сортування", | ||||||
|  |     "welcome": "Ласкаво просимо.", | ||||||
|  |     "documentationLinksNote": "Сторінка Obtainium на GitHub, посилання на яку наведено нижче, містить посилання на відео, статті, дискусії та інші ресурси, які допоможуть вам зрозуміти, як користуватися додатком.", | ||||||
|  |     "batteryOptimizationNote": "Зауважте, що фонові завантаження можуть працювати надійніше, якщо ви вимкнете оптимізацію батареї ОС для Obtainium.", | ||||||
|  |     "fileDeletionError": "Не вдалося видалити файл (спробуйте видалити його вручну, а потім спробуйте ще раз): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Видалити застосунок?", |         "one": "Видалити застосунок?", | ||||||
|         "other": "Видалити застосунки?" |         "other": "Видалити застосунки?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "Phát hiện phiên bản tiêu chuẩn", |     "standardVersionDetection": "Phát hiện phiên bản tiêu chuẩn", | ||||||
|     "groupByCategory": "Nhóm theo danh mục", |     "groupByCategory": "Nhóm theo danh mục", | ||||||
|     "autoApkFilterByArch": "Cố gắng lọc APK theo kiến trúc CPU nếu có thể", |     "autoApkFilterByArch": "Cố gắng lọc APK theo kiến trúc CPU nếu có thể", | ||||||
|  |     "autoLinkFilterByArch": "Attempt to filter links by CPU architecture if possible", | ||||||
|     "overrideSource": "Ghi đè nguồn", |     "overrideSource": "Ghi đè nguồn", | ||||||
|     "dontShowAgain": "Đừng hiển thị thông tin này nữa", |     "dontShowAgain": "Đừng hiển thị thông tin này nữa", | ||||||
|     "dontShowTrackOnlyWarnings": "Không hiển thị cảnh báo 'Chỉ theo dõi'", |     "dontShowTrackOnlyWarnings": "Không hiển thị cảnh báo 'Chỉ theo dõi'", | ||||||
| @@ -255,7 +256,7 @@ | |||||||
|     "intermediateLinkNotFound": "Không tìm thấy liên kết trung gian", |     "intermediateLinkNotFound": "Không tìm thấy liên kết trung gian", | ||||||
|     "intermediateLink": "Liên kết trung gian", |     "intermediateLink": "Liên kết trung gian", | ||||||
|     "exemptFromBackgroundUpdates": "Miễn cập nhật nền (nếu được bật)", |     "exemptFromBackgroundUpdates": "Miễn cập nhật nền (nếu được bật)", | ||||||
|     "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ó Wi-Fi", | ||||||
|     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", |     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", | ||||||
|     "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", | ||||||
| @@ -314,11 +315,21 @@ | |||||||
|     "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", |     "wiki": "Trợ giúp/Wiki", | ||||||
|     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", |     "crowdsourcedConfigsLabel": "Crowdsourced App Configurations (use at your own risk)", | ||||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", |     "crowdsourcedConfigsShort": "Crowdsourced App Configurations", | ||||||
|     "allowInsecure": "Allow insecure HTTP requests", |     "allowInsecure": "Allow insecure HTTP requests", | ||||||
|     "stayOneVersionBehind": "Stay one version behind latest", |     "stayOneVersionBehind": "Stay one version behind latest", | ||||||
|  |     "useFirstApkOfVersion": "Auto-select first of multiple APKs", | ||||||
|     "refreshBeforeDownload": "Refresh app details before download", |     "refreshBeforeDownload": "Refresh app details before download", | ||||||
|     "tencentAppStore": "Tencent App Store", |     "tencentAppStore": "Tencent App Store", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "Name", | ||||||
|  |     "smartname": "Name (Smart)", | ||||||
|  |     "sortMethod": "Sort Method", | ||||||
|  |     "welcome": "Welcome", | ||||||
|  |     "documentationLinksNote": "The Obtainium GitHub page linked below contains links to videos, articles, discussions and other resources that will help you understand how to use the app.", | ||||||
|  |     "batteryOptimizationNote": "Note that background downloads may work more reliably if you disable OS battery optimizations for Obtainium.", | ||||||
|  |     "fileDeletionError": "Failed to delete file (try deleting it manually then try again): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "Gỡ ứng dụng?", |         "one": "Gỡ ứng dụng?", | ||||||
|         "other": "Gỡ ứng dụng?" |         "other": "Gỡ ứng dụng?" | ||||||
|   | |||||||
| @@ -211,7 +211,7 @@ | |||||||
|     "uninstallFromDevice": "從裝置解除安裝", |     "uninstallFromDevice": "從裝置解除安裝", | ||||||
|     "onlyWorksWithNonVersionDetectApps": "僅適用於停用版本偵測的應用程式。", |     "onlyWorksWithNonVersionDetectApps": "僅適用於停用版本偵測的應用程式。", | ||||||
|     "releaseDateAsVersion": "使用發佈日期作為版本字串", |     "releaseDateAsVersion": "使用發佈日期作為版本字串", | ||||||
|     "releaseTitleAsVersion": "Use release title as version string", |     "releaseTitleAsVersion": "使用發佈標題作為版本字串", | ||||||
|     "releaseDateAsVersionExplanation": "此選項僅應用於版本偵測無法正確工作但有發佈日期的應用程式。", |     "releaseDateAsVersionExplanation": "此選項僅應用於版本偵測無法正確工作但有發佈日期的應用程式。", | ||||||
|     "changes": "變更", |     "changes": "變更", | ||||||
|     "releaseDate": "發佈日期", |     "releaseDate": "發佈日期", | ||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "標準版本偵測", |     "standardVersionDetection": "標準版本偵測", | ||||||
|     "groupByCategory": "按類別分組", |     "groupByCategory": "按類別分組", | ||||||
|     "autoApkFilterByArch": "如果可能,嘗試按 CPU 架構過濾 APK", |     "autoApkFilterByArch": "如果可能,嘗試按 CPU 架構過濾 APK", | ||||||
|  |     "autoLinkFilterByArch": "若可能,自動根據 CPU 架構篩選連結", | ||||||
|     "overrideSource": "覆蓋來源", |     "overrideSource": "覆蓋來源", | ||||||
|     "dontShowAgain": "不要再顯示", |     "dontShowAgain": "不要再顯示", | ||||||
|     "dontShowTrackOnlyWarnings": "不要顯示「僅追蹤」警告", |     "dontShowTrackOnlyWarnings": "不要顯示「僅追蹤」警告", | ||||||
| @@ -255,8 +256,8 @@ | |||||||
|     "intermediateLinkNotFound": "沒有找到中間連結", |     "intermediateLinkNotFound": "沒有找到中間連結", | ||||||
|     "intermediateLink": "中間連結", |     "intermediateLink": "中間連結", | ||||||
|     "exemptFromBackgroundUpdates": "免除背景更新(若已啟用)", |     "exemptFromBackgroundUpdates": "免除背景更新(若已啟用)", | ||||||
|     "bgUpdatesOnWiFiOnly": "停用非 WiFi 的背景更新", |     "bgUpdatesOnWiFiOnly": "停用非 Wi-Fi 的背景更新", | ||||||
|     "bgUpdatesWhileChargingOnly": "Disable background updates when not charging", |     "bgUpdatesWhileChargingOnly": "未充電時停用背景更新", | ||||||
|     "autoSelectHighestVersionCode": "自動選擇最高 versionCode 的 APK", |     "autoSelectHighestVersionCode": "自動選擇最高 versionCode 的 APK", | ||||||
|     "versionExtractionRegEx": "版本字串提取正則表達式", |     "versionExtractionRegEx": "版本字串提取正則表達式", | ||||||
|     "trimVersionString": "用正則表達式修剪版本字串", |     "trimVersionString": "用正則表達式修剪版本字串", | ||||||
| @@ -314,11 +315,21 @@ | |||||||
|     "appVerifierInstructionToast": "分享至 AppVerifier,然後準備好時回到此處。", |     "appVerifierInstructionToast": "分享至 AppVerifier,然後準備好時回到此處。", | ||||||
|     "wiki": "幫助/維基", |     "wiki": "幫助/維基", | ||||||
|     "crowdsourcedConfigsLabel": "群眾外包的應用程式設定(使用風險自負)", |     "crowdsourcedConfigsLabel": "群眾外包的應用程式設定(使用風險自負)", | ||||||
|     "crowdsourcedConfigsShort": "Crowdsourced App Configs", |     "crowdsourcedConfigsShort": "群眾外包的應用程式設定", | ||||||
|     "allowInsecure": "Allow insecure HTTP requests", |     "allowInsecure": "允許不安全的 HTTP 請求", | ||||||
|     "stayOneVersionBehind": "Stay one version behind latest", |     "stayOneVersionBehind": "保持比最新版本落後一個版本", | ||||||
|     "refreshBeforeDownload": "Refresh app details before download", |     "useFirstApkOfVersion": "Auto-select first of multiple APKs", | ||||||
|  |     "refreshBeforeDownload": "下載前刷新應用程式詳細資訊", | ||||||
|     "tencentAppStore": "騰訊應用寶", |     "tencentAppStore": "騰訊應用寶", | ||||||
|  |     "coolApk": "CoolApk", | ||||||
|  |     "vivoAppStore": "vivo App Store (CN)", | ||||||
|  |     "name": "名稱", | ||||||
|  |     "smartname": "名稱(智慧)", | ||||||
|  |     "sortMethod": "排序方式", | ||||||
|  |     "welcome": "歡迎", | ||||||
|  |     "documentationLinksNote": "下方連結的 Obtainium GitHub 頁面包含影片、文章、討論及其他資源,能幫助你瞭解如何使用這款應用程式。", | ||||||
|  |     "batteryOptimizationNote": "Note that background downloads may work more reliably if you disable OS battery optimizations for Obtainium.", | ||||||
|  |     "fileDeletionError": "Failed to delete file (try deleting it manually then try again): \"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "移除應用程式?", |         "one": "移除應用程式?", | ||||||
|         "other": "移除應用程式?" |         "other": "移除應用程式?" | ||||||
|   | |||||||
| @@ -221,6 +221,7 @@ | |||||||
|     "standardVersionDetection": "常规版本检测", |     "standardVersionDetection": "常规版本检测", | ||||||
|     "groupByCategory": "按类别分组显示", |     "groupByCategory": "按类别分组显示", | ||||||
|     "autoApkFilterByArch": "如果可能,尝试按设备支持的 CPU 架构筛选 APK 文件", |     "autoApkFilterByArch": "如果可能,尝试按设备支持的 CPU 架构筛选 APK 文件", | ||||||
|  |     "autoLinkFilterByArch": "尽可能按 CPU 架构过滤链接", | ||||||
|     "overrideSource": "覆盖来源", |     "overrideSource": "覆盖来源", | ||||||
|     "dontShowAgain": "不再显示", |     "dontShowAgain": "不再显示", | ||||||
|     "dontShowTrackOnlyWarnings": "忽略“仅追踪”模式警告", |     "dontShowTrackOnlyWarnings": "忽略“仅追踪”模式警告", | ||||||
| @@ -317,8 +318,18 @@ | |||||||
|     "crowdsourcedConfigsShort": "众包应用程序配置", |     "crowdsourcedConfigsShort": "众包应用程序配置", | ||||||
|     "allowInsecure": "允许不安全的 HTTP 请求", |     "allowInsecure": "允许不安全的 HTTP 请求", | ||||||
|     "stayOneVersionBehind": "比最新版本晚一个版本", |     "stayOneVersionBehind": "比最新版本晚一个版本", | ||||||
|  |     "useFirstApkOfVersion": "从多个 APK 中自动选择第一个", | ||||||
|     "refreshBeforeDownload": "下载前刷新应用程序详细信息", |     "refreshBeforeDownload": "下载前刷新应用程序详细信息", | ||||||
|     "tencentAppStore": "腾讯应用宝", |     "tencentAppStore": "腾讯应用宝", | ||||||
|  |     "coolApk": "酷安", | ||||||
|  |     "vivoAppStore": "vivo 应用商店(中国)", | ||||||
|  |     "name": "名称", | ||||||
|  |     "smartname": "姓名(智能)", | ||||||
|  |     "sortMethod": "排序方法", | ||||||
|  |     "welcome": "欢迎光临", | ||||||
|  |     "documentationLinksNote": "下面链接的 Obtainium GitHub 页面包含视频、文章、讨论和其他资源的链接,可帮助您了解如何使用该应用程序。", | ||||||
|  |     "batteryOptimizationNote": "请注意,如果为 Obtainium 禁用操作系统电池优化功能,后台下载可能会更稳定。", | ||||||
|  |     "fileDeletionError": "删除文件失败(尝试手动删除,然后再试一次):\"{}\"", | ||||||
|     "removeAppQuestion": { |     "removeAppQuestion": { | ||||||
|         "one": "是否删除应用?", |         "one": "是否删除应用?", | ||||||
|         "other": "是否删除应用?" |         "other": "是否删除应用?" | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								build.sh
									
									
									
									
									
								
							
							
						
						| @@ -2,15 +2,35 @@ | |||||||
| # Convenience script | # Convenience script | ||||||
|  |  | ||||||
| CURR_DIR="$(pwd)" | CURR_DIR="$(pwd)" | ||||||
| trap "cd "$CURR_DIR"" EXIT | SCRIPT_DIR="$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")" &>/dev/null && pwd)" | ||||||
|  | trap "cd \"$CURR_DIR\"" EXIT | ||||||
|  | cd "$SCRIPT_DIR" | ||||||
|  |  | ||||||
| if [ -z "$1" ]; then | if [ -z "$1" ]; then | ||||||
|     git fetch && git merge origin/main && git push # Typically run after a PR to main, so bring dev up to date |     git fetch && git merge origin/main && git push # Typically run after a PR to main, so bring dev up to date | ||||||
| fi | fi | ||||||
|  |  | ||||||
|  | # Update local Flutter | ||||||
|  | git submodule update --remote | ||||||
| cd .flutter | cd .flutter | ||||||
| git fetch | git fetch | ||||||
| git checkout "$(flutter --version | head -2 | tail -1 | awk '{print $4}')" # Ensure included Flutter submodule version equals my environment | git checkout stable | ||||||
|  | git pull | ||||||
|  | FLUTTER_GIT_URL="https://github.com/flutter/flutter/" ./bin/flutter upgrade | ||||||
| cd .. | cd .. | ||||||
|  |  | ||||||
|  | # Keep global Flutter, if any, in sync | ||||||
|  | if [ -f ~/flutter/bin/flutter ]; then | ||||||
|  |     cd ~/flutter | ||||||
|  |     ./bin/flutter channel stable | ||||||
|  |     ./bin/flutter upgrade | ||||||
|  |     cd "$SCRIPT_DIR" | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | if [ -z "$(which flutter)" ]; then | ||||||
|  |     export PATH="$PATH:$SCRIPT_DIR/.flutter/bin" | ||||||
|  | fi | ||||||
|  |  | ||||||
| rm ./build/app/outputs/flutter-apk/* 2>/dev/null                                       # Get rid of older builds if any | rm ./build/app/outputs/flutter-apk/* 2>/dev/null                                       # Get rid of older builds if any | ||||||
| flutter build apk --flavor normal && flutter build apk --split-per-abi --flavor normal # Build (both split and combined APKs) | flutter build apk --flavor normal && flutter build apk --split-per-abi --flavor normal # Build (both split and combined APKs) | ||||||
| for file in ./build/app/outputs/flutter-apk/app-*normal*.apk*; do mv "$file" "${file//-normal/}"; done | for file in ./build/app/outputs/flutter-apk/app-*normal*.apk*; do mv "$file" "${file//-normal/}"; done | ||||||
|   | |||||||
							
								
								
									
										86
									
								
								docker/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,86 @@ | |||||||
|  | FROM ubuntu:22.04 | ||||||
|  |  | ||||||
|  | ENV DEBIAN_FRONTEND noninteractive | ||||||
|  | ENV ANDROID_SDK_ROOT /opt/android-sdk-linux | ||||||
|  | ENV PATH "${PATH}:/opt/flutter/bin:/root/.pub-cache/bin:${ANDROID_SDK_ROOT}/cmdline-tools/latest/bin:/opt/android-sdk-linux/platform-tools" | ||||||
|  |  | ||||||
|  | ENV HOME=/root | ||||||
|  |  | ||||||
|  | # Prereqs | ||||||
|  | RUN \ | ||||||
|  |     apt-get update &&\ | ||||||
|  |     apt-get install -y --no-install-recommends \ | ||||||
|  |         bash \ | ||||||
|  |         curl \ | ||||||
|  |         file \ | ||||||
|  |         git 2.x \ | ||||||
|  |         unzip \ | ||||||
|  |         xz-utils \ | ||||||
|  |         zip \ | ||||||
|  |         libglu1-mesa \ | ||||||
|  |         libxi-dev \ | ||||||
|  |         libxmu-dev \ | ||||||
|  |         libglu1-mesa-dev \ | ||||||
|  |         git-lfs \ | ||||||
|  |         openssl \ | ||||||
|  |         wget | ||||||
|  |  | ||||||
|  | # Build prereqs | ||||||
|  | RUN \ | ||||||
|  |     apt-get install -y \ | ||||||
|  |         cmake curl git wget unzip libgconf-2-4 gdb libstdc++6 libglu1-mesa fonts-droid-fallback lib32stdc++6 python3 sed \ | ||||||
|  |         cmake ninja-build build-essential libgl1-mesa-dev libegl1-mesa-dev libgles2-mesa-dev clang pkg-config libgtk-3-dev \ | ||||||
|  |         liblzma-dev libmount-dev libblkid-dev libgcrypt20-dev libgpg-error-dev libssl-dev libpng-dev libjpeg-dev \ | ||||||
|  |         libtiff-dev libgif-dev libgtk-3-dev | ||||||
|  |  | ||||||
|  | # Android SDK prequisites | ||||||
|  | # https://developer.android.com/studio#command-tools | ||||||
|  | RUN \ | ||||||
|  |     apt-get install -y --no-install-recommends \ | ||||||
|  |         git \ | ||||||
|  |         git-lfs \ | ||||||
|  |         openssl \ | ||||||
|  |         wget \ | ||||||
|  |         unzip | ||||||
|  |  | ||||||
|  | # Android SDK | ||||||
|  | RUN \ | ||||||
|  |     wget --quiet  https://dl.google.com/android/repository/commandlinetools-linux-13114758_latest.zip -O /tmp/tools.zip && \ | ||||||
|  |     mkdir -p ${ANDROID_SDK_ROOT}/cmdline-tools && \ | ||||||
|  |     unzip -q /tmp/tools.zip -d ${ANDROID_SDK_ROOT}/cmdline-tools && \ | ||||||
|  |     mv ${ANDROID_SDK_ROOT}/cmdline-tools/cmdline-tools ${ANDROID_SDK_ROOT}/cmdline-tools/latest && \ | ||||||
|  |     rm -v /tmp/tools.zip && \ | ||||||
|  |     mkdir -p /root/.android/ && touch /root/.android/repositories.cfg &&\ | ||||||
|  |     apt-get install -y --no-install-recommends openjdk-21-jdk openjdk-21-jre &&\ | ||||||
|  |     yes | sdkmanager --licenses &&\ | ||||||
|  |     sdkmanager --update | ||||||
|  |  | ||||||
|  |     # Platform tools | ||||||
|  | # Get latest with sdkmanager --list | ||||||
|  | RUN sdkmanager --install "build-tools;33.0.1" | ||||||
|  | RUN sdkmanager --install "ndk;26.3.11579264" | ||||||
|  | RUN sdkmanager --install "cmake;4.0.2" | ||||||
|  | RUN sdkmanager --install platform-tools | ||||||
|  | RUN sdkmanager --install emulator | ||||||
|  | RUN sdkmanager --install tools | ||||||
|  | RUN sdkmanager --install "platforms;android-28" | ||||||
|  | RUN sdkmanager --install "platforms;android-31" | ||||||
|  | RUN sdkmanager --install "platforms;android-32" | ||||||
|  | RUN sdkmanager --install "platforms;android-33" | ||||||
|  | RUN sdkmanager --install "platforms;android-34" | ||||||
|  | RUN sdkmanager --install "platforms;android-35" | ||||||
|  |  | ||||||
|  | # Flutter | ||||||
|  | ARG DEV_UID=0 | ||||||
|  | RUN \ | ||||||
|  |     wget --quiet https://storage.googleapis.com/flutter_infra_release/releases/stable/linux/flutter_linux_3.29.3-stable.tar.xz -O /tmp/flutter.tar.xz &&\ | ||||||
|  |     mkdir -p /opt &&\ | ||||||
|  |     cd /opt &&\ | ||||||
|  |     tar xf /tmp/flutter.tar.xz &&\ | ||||||
|  |     rm /tmp/flutter.tar.xz &&\ | ||||||
|  |     git config --global --add safe.directory /opt/flutter &&\ | ||||||
|  |     dart pub global activate cider &&\ | ||||||
|  |     chown -R ${DEV_UID} /opt/flutter | ||||||
|  | RUN flutter --disable-analytics | ||||||
|  | RUN flutter upgrade | ||||||
|  | RUN chmod a+w /opt/flutter/packages -R | ||||||
							
								
								
									
										21
									
								
								docker/builder.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  |  | ||||||
|  | SCRIPT_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) | ||||||
|  |  | ||||||
|  | cd ${SCRIPT_DIR}/.. | ||||||
|  | mkdir -p ./data/home | ||||||
|  | docker run \ | ||||||
|  |     --rm \ | ||||||
|  |     -ti \ | ||||||
|  |     --net host \ | ||||||
|  |     -v "${PWD}/../:${PWD}/../":z \ | ||||||
|  |     -w "${PWD}" \ | ||||||
|  |     --name flutter-dev-obtainium \ | ||||||
|  |     --user $(id -u) \ | ||||||
|  |     -v ./data/home:/home/${USER}:z \ | ||||||
|  |     -e USER=${USER} \ | ||||||
|  |     -e HOME=/home/${USER} \ | ||||||
|  |     -e ANDROID_USER_HOME=${HOME}/.android \ | ||||||
|  |     -e GRADLE_USER_HOME=${HOME}/.gradle \ | ||||||
|  |     -e PS1="${debian_chroot:+($debian_chroot)}${USER}@\h:\w\$ " \ | ||||||
|  |     flutter-builder-obtainium | ||||||
							
								
								
									
										14
									
								
								docker/mkbuilder.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  |  | ||||||
|  | SCRIPT_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) | ||||||
|  | D=$(date +'%Y%m%d.%H%M%S%3N') | ||||||
|  |  | ||||||
|  | set -e | ||||||
|  |  | ||||||
|  | cd "${SCRIPT_DIR}/.." | ||||||
|  | # Create the builder image | ||||||
|  | docker build \ | ||||||
|  |     -t flutter-builder-obtainium \ | ||||||
|  |     -f ./docker/Dockerfile \ | ||||||
|  |     --build-arg="DEV_UID=$(id -u)" \ | ||||||
|  |     . | ||||||
| @@ -13,7 +13,6 @@ | |||||||
| 			<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> | ||||||
| 			<li>SourceForge</li> |  | ||||||
| 			<li>SourceHut</li> | 			<li>SourceHut</li> | ||||||
| 		</ul> | 		</ul> | ||||||
| 	</li> | 	</li> | ||||||
| @@ -22,24 +21,19 @@ | |||||||
| 		<ul> | 		<ul> | ||||||
| 			<li>APKPure</li> | 			<li>APKPure</li> | ||||||
| 			<li>Aptoide</li> | 			<li>Aptoide</li> | ||||||
| 			<li>Uptodowng</li> | 			<li>Uptodown</li> | ||||||
| 			<li>APKMirror (Track-Only)</li> | 			<li>APKMirror (Track-Only)</li> | ||||||
| 			<li>Huawei AppGallery</li> | 			<li>Huawei AppGallery</li> | ||||||
|  | 			<li>Tencent App Store</li> | ||||||
|  | 			<li>CoolApk</li> | ||||||
|  | 			<li>vivo App Store (CN)</li> | ||||||
| 			<li>Jenkins Jobs</li> | 			<li>Jenkins Jobs</li> | ||||||
| 		</ul> | 			<li>RuStore</li> | ||||||
| 	</li> |  | ||||||
| 	<li> |  | ||||||
| 		<p>Open Source - App-Specific:</p> |  | ||||||
| 		<ul> |  | ||||||
| 			<li>Mullvad</li> |  | ||||||
| 			<li>Signal</li> |  | ||||||
| 			<li>VLC</li> |  | ||||||
| 		</ul> | 		</ul> | ||||||
| 	</li> | 	</li> | ||||||
| 	<li> | 	<li> | ||||||
| 		<p>Other - App-Specific:</p> | 		<p>Other - App-Specific:</p> | ||||||
| 		<ul> | 		<ul> | ||||||
| 			<li>WhatsApp</li> |  | ||||||
| 			<li>Telegram App</li> | 			<li>Telegram App</li> | ||||||
| 			<li>Neutron Code</li> | 			<li>Neutron Code</li> | ||||||
| 		</ul> | 		</ul> | ||||||
|   | |||||||
| Before Width: | Height: | Size: 66 KiB After Width: | Height: | Size: 41 B | 
							
								
								
									
										1
									
								
								fastlane/metadata/android/en-US/images/featureGraphic.png
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ../../../../../assets/graphics/banner.png | ||||||
| Before Width: | Height: | Size: 66 KiB After Width: | Height: | Size: 41 B | 
| Before Width: | Height: | Size: 44 KiB After Width: | Height: | Size: 47 B | 
							
								
								
									
										1
									
								
								fastlane/metadata/android/en-US/images/icon.png
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ../../../../../assets/graphics/icon-512x512.png | ||||||
| Before Width: | Height: | Size: 44 KiB After Width: | Height: | Size: 47 B | 
| Before Width: | Height: | Size: 234 KiB After Width: | Height: | Size: 47 B | 
							
								
								
									
										1
									
								
								fastlane/metadata/android/en-US/images/phoneScreenshots/1.png
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ../../../../../../assets/screenshots/1.apps.png | ||||||
| Before Width: | Height: | Size: 234 KiB After Width: | Height: | Size: 47 B | 
| Before Width: | Height: | Size: 238 KiB After Width: | Height: | Size: 53 B | 
							
								
								
									
										1
									
								
								fastlane/metadata/android/en-US/images/phoneScreenshots/2.png
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ../../../../../../assets/screenshots/2.dark_theme.png | ||||||
| Before Width: | Height: | Size: 238 KiB After Width: | Height: | Size: 53 B | 
| Before Width: | Height: | Size: 140 KiB After Width: | Height: | Size: 55 B | 
							
								
								
									
										1
									
								
								fastlane/metadata/android/en-US/images/phoneScreenshots/3.png
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ../../../../../../assets/screenshots/3.material_you.png | ||||||
| Before Width: | Height: | Size: 140 KiB After Width: | Height: | Size: 55 B | 
| Before Width: | Height: | Size: 139 KiB After Width: | Height: | Size: 46 B | 
							
								
								
									
										1
									
								
								fastlane/metadata/android/en-US/images/phoneScreenshots/4.png
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ../../../../../../assets/screenshots/4.app.png | ||||||
| Before Width: | Height: | Size: 139 KiB After Width: | Height: | Size: 46 B | 
| Before Width: | Height: | Size: 118 KiB After Width: | Height: | Size: 51 B | 
							
								
								
									
										1
									
								
								fastlane/metadata/android/en-US/images/phoneScreenshots/5.png
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ../../../../../../assets/screenshots/5.app_opts.png | ||||||
| Before Width: | Height: | Size: 118 KiB After Width: | Height: | Size: 51 B | 
| Before Width: | Height: | Size: 262 KiB After Width: | Height: | Size: 54 B | 
							
								
								
									
										1
									
								
								fastlane/metadata/android/en-US/images/phoneScreenshots/6.png
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ../../../../../../assets/screenshots/6.app_webview.png | ||||||
| Before Width: | Height: | Size: 262 KiB After Width: | Height: | Size: 54 B | 
| @@ -13,7 +13,6 @@ | |||||||
| 			<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> | ||||||
| 			<li>SourceForge</li> |  | ||||||
| 			<li>SourceHut</li> | 			<li>SourceHut</li> | ||||||
| 		</ul> | 		</ul> | ||||||
| 	</li> | 	</li> | ||||||
| @@ -22,24 +21,19 @@ | |||||||
| 		<ul> | 		<ul> | ||||||
| 			<li>APKPure</li> | 			<li>APKPure</li> | ||||||
| 			<li>Aptoide</li> | 			<li>Aptoide</li> | ||||||
| 			<li>Uptodowng</li> | 			<li>Uptodown</li> | ||||||
| 			<li>APKMirror (Track-Only)</li> | 			<li>APKMirror (Track-Only)</li> | ||||||
| 			<li>Huawei AppGallery</li> | 			<li>Huawei AppGallery</li> | ||||||
|  | 			<li>Tencent App Store</li> | ||||||
|  | 			<li>CoolApk</li> | ||||||
|  | 			<li>vivo App Store (CN)</li> | ||||||
| 			<li>Jenkins Jobs</li> | 			<li>Jenkins Jobs</li> | ||||||
| 		</ul> | 			<li>RuStore</li> | ||||||
| 	</li> |  | ||||||
| 	<li> |  | ||||||
| 		<p>Свободное ПО - Для отдельных приложений:</p> |  | ||||||
| 		<ul> |  | ||||||
| 			<li>Mullvad</li> |  | ||||||
| 			<li>Signal</li> |  | ||||||
| 			<li>VLC</li> |  | ||||||
| 		</ul> | 		</ul> | ||||||
| 	</li> | 	</li> | ||||||
| 	<li> | 	<li> | ||||||
| 		<p>Другие - Для отдельных приложений:</p> | 		<p>Другие - Для отдельных приложений:</p> | ||||||
| 		<ul> | 		<ul> | ||||||
| 			<li>WhatsApp</li> |  | ||||||
| 			<li>Telegram App</li> | 			<li>Telegram App</li> | ||||||
| 			<li>Neutron Code</li> | 			<li>Neutron Code</li> | ||||||
| 		</ul> | 		</ul> | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								fastlane/metadata/android/ru/title.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | Obtainium | ||||||
| @@ -72,7 +72,8 @@ class APKMirror extends AppSource { | |||||||
|                 true |                 true | ||||||
|             ? additionalSettings['filterReleaseTitlesByRegEx'] |             ? additionalSettings['filterReleaseTitlesByRegEx'] | ||||||
|             : null; |             : null; | ||||||
|     Response res = await sourceRequest('$standardUrl/feed', additionalSettings); |     Response res = | ||||||
|  |         await sourceRequest('$standardUrl/feed/', additionalSettings); | ||||||
|     if (res.statusCode == 200) { |     if (res.statusCode == 200) { | ||||||
|       var items = parse(res.body).querySelectorAll('item'); |       var items = parse(res.body).querySelectorAll('item'); | ||||||
|       dynamic targetRelease; |       dynamic targetRelease; | ||||||
|   | |||||||
| @@ -1,24 +1,18 @@ | |||||||
|  | import 'dart:convert'; | ||||||
|  |  | ||||||
| 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/parser.dart'; |  | ||||||
| import 'package:obtainium/app_sources/html.dart'; |  | ||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.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'; | ||||||
|  |  | ||||||
| parseDateTimeMMMddCommayyyy(String? dateString) { | extension Unique<E, Id> on List<E> { | ||||||
|   DateTime? releaseDate; |   List<E> unique([Id Function(E element)? id, bool inplace = true]) { | ||||||
|   try { |     final ids = Set(); | ||||||
|     releaseDate = dateString != null |     var list = inplace ? this : List<E>.from(this); | ||||||
|         ? DateFormat('MMM dd, yyyy').parse(dateString) |     list.retainWhere((x) => ids.add(id != null ? id(x) : x as Id)); | ||||||
|         : null; |     return list; | ||||||
|     releaseDate = dateString != null && releaseDate == null |  | ||||||
|         ? DateFormat('MMMM dd, yyyy').parse(dateString) |  | ||||||
|         : releaseDate; |  | ||||||
|   } catch (err) { |  | ||||||
|     // ignore |  | ||||||
|   } |   } | ||||||
|   return releaseDate; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| class APKPure extends AppSource { | class APKPure extends AppSource { | ||||||
| @@ -35,6 +29,10 @@ class APKPure extends AppSource { | |||||||
|       [ |       [ | ||||||
|         GeneratedFormSwitch('stayOneVersionBehind', |         GeneratedFormSwitch('stayOneVersionBehind', | ||||||
|             label: tr('stayOneVersionBehind'), defaultValue: false) |             label: tr('stayOneVersionBehind'), defaultValue: false) | ||||||
|  |       ], | ||||||
|  |       [ | ||||||
|  |         GeneratedFormSwitch('useFirstApkOfVersion', | ||||||
|  |             label: tr('useFirstApkOfVersion'), defaultValue: true) | ||||||
|       ] |       ] | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
| @@ -65,95 +63,73 @@ class APKPure extends AppSource { | |||||||
|     return Uri.parse(standardUrl).pathSegments.last; |     return Uri.parse(standardUrl).pathSegments.last; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   getDetailsForVersionLink( |   getDetailsForVersion( | ||||||
|       String standardUrl, |       List<Map<String, dynamic>> versionVariants, | ||||||
|       String appId, |  | ||||||
|       String host, |  | ||||||
|       List<String> supportedArchs, |       List<String> supportedArchs, | ||||||
|       String link, |  | ||||||
|       Map<String, dynamic> additionalSettings) async { |       Map<String, dynamic> additionalSettings) async { | ||||||
|     var res = await sourceRequest(link, additionalSettings); |     var apkUrls = versionVariants | ||||||
|     if (res.statusCode == 200) { |         .map((e) { | ||||||
|       var html = parse(res.body); |           String appId = e['package_name']; | ||||||
|       var apksDiv = |           String versionCode = e['version_code']; | ||||||
|           html.querySelector('#version-list div div.show-more-content'); |  | ||||||
|       DateTime? topReleaseDate; |           List<String> architectures = e['native_code']?.cast<String>(); | ||||||
|       var apkUrls = apksDiv |           String architectureString = architectures.join(','); | ||||||
|               ?.querySelectorAll('div.group-title') |           if (architectures.contains("universal") || | ||||||
|               .map((e) { |               architectures.contains("unlimited")) { | ||||||
|                 String architectureString = e.text.trim(); |             architectures = []; | ||||||
|                 if (architectureString.toLowerCase() == 'unlimited' || |           } | ||||||
|                     architectureString.toLowerCase() == 'universal') { |           if (additionalSettings['autoApkFilterByArch'] == true && | ||||||
|                   architectureString = ''; |               architectures.isNotEmpty && | ||||||
|                 } |               architectures.where((a) => supportedArchs.contains(a)).isEmpty) { | ||||||
|                 List<String> architectures = architectureString |             return null; | ||||||
|                     .split(',') |           } | ||||||
|                     .map((e) => e.trim()) |  | ||||||
|                     .where((e) => e.isNotEmpty) |           String type = e['asset']['type']; | ||||||
|                     .toList(); |           String downloadUri = e['asset']['url']; | ||||||
|                 // 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 |           return MapEntry( | ||||||
|                 var apkInfo = e.nextElementSibling?.querySelector('div.info'); |               '$appId-$versionCode-$architectureString.${type.toLowerCase()}', | ||||||
|                 String? versionCode = RegExp('[0-9]+') |               downloadUri); | ||||||
|                     .firstMatch( |         }) | ||||||
|                         apkInfo?.querySelector('div.info-top .code')?.text ?? |         .nonNulls | ||||||
|                             '') |         .toList() | ||||||
|                     ?.group(0) |         .unique((e) => e.key); | ||||||
|                     ?.trim(); |  | ||||||
|                 var types = apkInfo |     if (apkUrls.isEmpty) { | ||||||
|                         ?.querySelectorAll('div.info-top span.tag') |       throw NoAPKError(); | ||||||
|                         .map((e) => e.text.trim()) |     } | ||||||
|                         .map((t) => t == 'APKs' ? 'APK' : t) ?? |  | ||||||
|                     []; |     // get version details from first variant | ||||||
|                 String type = types.isEmpty ? 'APK' : types.first; |     var v = versionVariants.first; | ||||||
|                 String? dateString = apkInfo |     String version = v['version_name']; | ||||||
|                     ?.querySelector('div.info-bottom span.time') |     String author = v['developer']; | ||||||
|                     ?.text |     String appName = v['title']; | ||||||
|                     .trim(); |     DateTime releaseDate = DateTime.parse(v['update_date']); | ||||||
|                 DateTime? releaseDate = parseDateTimeMMMddCommayyyy(dateString); |     String? changeLog = v['whatsnew']; | ||||||
|                 if (additionalSettings['autoApkFilterByArch'] == true && |     if (changeLog != null && changeLog.isEmpty) { | ||||||
|                     architectures.isNotEmpty && |       changeLog = null; | ||||||
|                     architectures |     } | ||||||
|                         .where((a) => supportedArchs.contains(a)) |  | ||||||
|                         .isEmpty) { |     if (additionalSettings['useFirstApkOfVersion'] == true) { | ||||||
|                   return const MapEntry('', ''); |       apkUrls = [apkUrls.first]; | ||||||
|                 } |     } | ||||||
|                 topReleaseDate ??= |  | ||||||
|                     releaseDate; // Just use the release date of the first APK in the list as the release date for this version |     return APKDetails(version, apkUrls, AppNames(author, appName), | ||||||
|                 return MapEntry( |         releaseDate: releaseDate, changeLog: changeLog); | ||||||
|                     '$appId-$versionCode-$architectureString.${type.toLowerCase()}', |   } | ||||||
|                     'https://d.${hosts.contains(host) ? 'cdnpure.com' : host}/b/$type/$appId?versionCode=$versionCode'); |  | ||||||
|               }) |   @override | ||||||
|               .where((e) => e.key.isNotEmpty) |   Future<Map<String, String>?> getRequestHeaders( | ||||||
|               .toList() ?? |       Map<String, dynamic> additionalSettings, | ||||||
|           []; |       {bool forAPKDownload = false}) async { | ||||||
|       if (apkUrls.isEmpty) { |     if (forAPKDownload) { | ||||||
|         throw NoAPKError(); |       return null; | ||||||
|       } |  | ||||||
|       String version = Uri.parse(link).pathSegments.last; |  | ||||||
|       String? author; |  | ||||||
|       try { |  | ||||||
|         author = html |  | ||||||
|                 .querySelector('span.info-sdk') |  | ||||||
|                 ?.text |  | ||||||
|                 .trim() |  | ||||||
|                 .substring(version.length + 4) ?? |  | ||||||
|             Uri.parse(standardUrl).pathSegments.reversed.last; |  | ||||||
|       } catch (e) { |  | ||||||
|         author = html.querySelector('span.info-sdk')?.text.trim() ?? |  | ||||||
|             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 { |     } else { | ||||||
|       throw getObtainiumHttpError(res); |       return { | ||||||
|  |         "Ual-Access-Businessid": "projecta", | ||||||
|  |         "Ual-Access-ProjectA": | ||||||
|  |             '{"device_info":{"os_ver":"${((await DeviceInfoPlugin().androidInfo).version.sdkInt)}"}}', | ||||||
|  |       }; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -163,41 +139,50 @@ class APKPure extends AppSource { | |||||||
|     Map<String, dynamic> additionalSettings, |     Map<String, dynamic> additionalSettings, | ||||||
|   ) async { |   ) async { | ||||||
|     String appId = (await tryInferringAppId(standardUrl))!; |     String appId = (await tryInferringAppId(standardUrl))!; | ||||||
|     String host = Uri.parse(standardUrl).host; |  | ||||||
|  |  | ||||||
|     var res0 = await sourceRequest('$standardUrl/versions', additionalSettings); |     List<String> supportedArchs = | ||||||
|     var decodedStandardUrl = standardUrl; |         (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||||
|     try { |  | ||||||
|       decodedStandardUrl = Uri.decodeFull(decodedStandardUrl); |     // request versions from API | ||||||
|     } catch (e) { |     var res = await sourceRequest( | ||||||
|       // |         "https://tapi.pureapk.com/v3/get_app_his_version?package_name=$appId&hl=en", | ||||||
|  |         additionalSettings); | ||||||
|  |     if (res.statusCode != 200) { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|     } |     } | ||||||
|     var versionLinks = await grabLinksCommon(res0, { |     List<Map<String, dynamic>> apks = | ||||||
|       'skipSort': true, |         jsonDecode(res.body)['version_list'].cast<Map<String, dynamic>>(); | ||||||
|       'customLinkFilterRegex': '$decodedStandardUrl/download/[^/]+\$' |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     var supportedArchs = (await DeviceInfoPlugin().androidInfo).supportedAbis; |     // group by version | ||||||
|  |     List<List<Map<String, dynamic>>> versions = apks | ||||||
|  |         .fold<Map<String, List<Map<String, dynamic>>>>({}, | ||||||
|  |             (Map<String, List<Map<String, dynamic>>> val, | ||||||
|  |                 Map<String, dynamic> element) { | ||||||
|  |           String v = element['version_name']; | ||||||
|  |           if (!val.containsKey(v)) { | ||||||
|  |             val[v] = []; | ||||||
|  |           } | ||||||
|  |           val[v]?.add(element); | ||||||
|  |           return val; | ||||||
|  |         }) | ||||||
|  |         .values | ||||||
|  |         .toList(); | ||||||
|  |  | ||||||
|     if (additionalSettings['autoApkFilterByArch'] != true) { |     if (versions.isEmpty) { | ||||||
|       // No need to request multiple versions when we're not going to filter them (always pick the top one) |  | ||||||
|       versionLinks = versionLinks.sublist(0, 1); |  | ||||||
|     } |  | ||||||
|     if (versionLinks.isEmpty) { |  | ||||||
|       throw NoReleasesError(); |       throw NoReleasesError(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     for (var i = 0; i < versionLinks.length; i++) { |     for (var i = 0; i < versions.length; i++) { | ||||||
|       var link = versionLinks[i]; |       var v = versions[i]; | ||||||
|       try { |       try { | ||||||
|         if (i == 0 && additionalSettings['stayOneVersionBehind'] == true) { |         if (i == 0 && additionalSettings['stayOneVersionBehind'] == true) { | ||||||
|           throw NoReleasesError(); |           throw NoReleasesError(); | ||||||
|         } |         } | ||||||
|         return await getDetailsForVersionLink(standardUrl, appId, host, |         return await getDetailsForVersion( | ||||||
|             supportedArchs, link.key, additionalSettings); |             v, supportedArchs, additionalSettings); | ||||||
|       } catch (e) { |       } catch (e) { | ||||||
|         if (additionalSettings['fallbackToOlderReleases'] != true || |         if (additionalSettings['fallbackToOlderReleases'] != true || | ||||||
|             i == versionLinks.length - 1) { |             i == versions.length - 1) { | ||||||
|           rethrow; |           rethrow; | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|   | |||||||
							
								
								
									
										174
									
								
								lib/app_sources/coolapk.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,174 @@ | |||||||
|  | import 'dart:convert'; | ||||||
|  | import 'package:bcrypt/bcrypt.dart'; | ||||||
|  | import 'package:crypto/crypto.dart'; | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  | import 'dart:math'; | ||||||
|  |  | ||||||
|  | // kanged from https://github.com/DUpdateSystem/UpgradeAll/blob/b2f92c9/core-websdk/src/main/java/net/xzos/upgradeall/core/websdk/api/client_proxy/hubs/CoolApk.kt | ||||||
|  | class CoolApk extends AppSource { | ||||||
|  |   CoolApk() { | ||||||
|  |     name = tr('coolApk'); | ||||||
|  |     hosts = ['www.coolapk.com', 'api2.coolapk.com']; | ||||||
|  |     allowSubDomains = true; | ||||||
|  |     naiveStandardVersionDetection = true; | ||||||
|  |     allowOverride = false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|  |     RegExp standardUrlRegEx = RegExp( | ||||||
|  |         r'^https?://(www\.)?coolapk\.com/apk/[^/]+', | ||||||
|  |         caseSensitive: false); | ||||||
|  |     var match = standardUrlRegEx.firstMatch(url); | ||||||
|  |     if (match == null) { | ||||||
|  |       throw InvalidURLError(name); | ||||||
|  |     } | ||||||
|  |     String standardizedUrl = match.group(0)!; | ||||||
|  |     return standardizedUrl; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<String?> tryInferringAppId(String standardUrl, | ||||||
|  |       {Map<String, dynamic> additionalSettings = const {}}) async { | ||||||
|  |     String appId = Uri.parse(standardUrl).pathSegments.last; | ||||||
|  |     return appId; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |       String standardUrl, | ||||||
|  |       Map<String, dynamic> additionalSettings, | ||||||
|  |       ) async { | ||||||
|  |     String appId = (await tryInferringAppId(standardUrl))!; | ||||||
|  |     String apiUrl = 'https://api2.coolapk.com'; | ||||||
|  |  | ||||||
|  |     // get latest | ||||||
|  |     var detailUrl = '$apiUrl/v6/apk/detail?id=$appId'; | ||||||
|  |     var headers = await getRequestHeaders(additionalSettings); | ||||||
|  |     var res = await sourceRequest(detailUrl, additionalSettings); | ||||||
|  |  | ||||||
|  |     if (res.statusCode != 200) { | ||||||
|  |       throw getObtainiumHttpError(res); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var json = jsonDecode(res.body); | ||||||
|  |     if (json['status'] == -2 || json['data'] == null) { | ||||||
|  |       throw NoReleasesError(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var detail = json['data']; | ||||||
|  |     String version = detail['apkversionname'].toString(); | ||||||
|  |     String appName = detail['title'].toString(); | ||||||
|  |     String author = detail['developername']?.toString() ?? 'CoolApk'; | ||||||
|  |     String changelog = detail['changelog']?.toString() ?? ''; | ||||||
|  |     int? releaseDate = detail['lastupdate'] != null | ||||||
|  |         ? (detail['lastupdate'] is int | ||||||
|  |         ? detail['lastupdate'] * 1000 | ||||||
|  |         : int.parse(detail['lastupdate'].toString()) * 1000) | ||||||
|  |         : null; | ||||||
|  |     String aid = detail['id'].toString(); | ||||||
|  |  | ||||||
|  |     // get apk url | ||||||
|  |     String apkUrl = await _getLatestApkUrl(apiUrl, appId, aid, version, headers); | ||||||
|  |     if (apkUrl.isEmpty) { | ||||||
|  |       throw NoAPKError(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     String apkName = '${appId}_$version.apk'; | ||||||
|  |  | ||||||
|  |     return APKDetails( | ||||||
|  |       version, | ||||||
|  |       [MapEntry(apkName, apkUrl)], | ||||||
|  |       AppNames(author, appName), | ||||||
|  |       releaseDate: releaseDate != null | ||||||
|  |           ? DateTime.fromMillisecondsSinceEpoch(releaseDate) | ||||||
|  |           : null, | ||||||
|  |       changeLog: changelog, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Future<String> _getLatestApkUrl(String apiUrl, String appId, String aid, | ||||||
|  |       String version, Map<String, String>? headers) async { | ||||||
|  |     String url = '$apiUrl/v6/apk/download?pn=$appId&aid=$aid'; | ||||||
|  |     var res = await sourceRequest(url, {}, followRedirects: false); | ||||||
|  |     if (res.statusCode >= 300 && res.statusCode < 400) { | ||||||
|  |       String location = res.headers['location'] ?? ''; | ||||||
|  |       return location; | ||||||
|  |     } | ||||||
|  |     return ''; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<Map<String, String>?> getRequestHeaders( | ||||||
|  |       Map<String, dynamic> additionalSettings, | ||||||
|  |       {bool forAPKDownload = false}) async { | ||||||
|  |     var tokenPair = _getToken(); | ||||||
|  |     // CoolAPK header | ||||||
|  |     return { | ||||||
|  |       'User-Agent': | ||||||
|  |       'Dalvik/2.1.0 (Linux; U; Android 9; MI 8 SE MIUI/9.5.9) (#Build; Xiaomi; MI 8 SE; PKQ1.181121.001; 9) +CoolMarket/12.4.2-2208241-universal', | ||||||
|  |       'X-App-Id': 'com.coolapk.market', | ||||||
|  |       'X-Requested-With': 'XMLHttpRequest', | ||||||
|  |       'X-Sdk-Int': '30', | ||||||
|  |       'X-App-Mode': 'universal', | ||||||
|  |       'X-App-Channel': 'coolapk', | ||||||
|  |       'X-Sdk-Locale': 'zh-CN', | ||||||
|  |       'X-App-Version': '12.4.2', | ||||||
|  |       'X-Api-Supported': '2208241', | ||||||
|  |       'X-App-Code': '2208241', | ||||||
|  |       'X-Api-Version': '12', | ||||||
|  |       'X-App-Device': tokenPair['deviceCode']!, | ||||||
|  |       'X-Dark-Mode': '0', | ||||||
|  |       'X-App-Token': tokenPair['token']!, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Map<String, String> _getToken() { | ||||||
|  |     final rand = Random(); | ||||||
|  |  | ||||||
|  |     String randHexString(int n) => | ||||||
|  |         List.generate(n, (_) => rand.nextInt(256).toRadixString(16).padLeft(2, '0')) | ||||||
|  |             .join() | ||||||
|  |             .toUpperCase(); | ||||||
|  |  | ||||||
|  |     String randMacAddress() => | ||||||
|  |         List.generate(6, (_) => rand.nextInt(256).toRadixString(16).padLeft(2, '0')) | ||||||
|  |             .join(':'); | ||||||
|  |  | ||||||
|  |     // 加密算法来自 https://github.com/XiaoMengXinX/FuckCoolapkTokenV2、https://github.com/Coolapk-UWP/Coolapk-UWP | ||||||
|  |     // device | ||||||
|  |     String aid = randHexString(16); | ||||||
|  |     String mac = randMacAddress(); | ||||||
|  |     const manufactor = 'Google'; | ||||||
|  |     const brand = 'Google'; | ||||||
|  |     const model = 'Pixel 5a'; | ||||||
|  |     const buildNumber = 'SQ1D.220105.007'; | ||||||
|  |  | ||||||
|  |     // generate deviceCode | ||||||
|  |     String deviceCode = | ||||||
|  |     base64.encode('$aid; ; ; $mac; $manufactor; $brand; $model; $buildNumber'.codeUnits); | ||||||
|  |  | ||||||
|  |     // generate timestamp | ||||||
|  |     String timeStamp = (DateTime.now().millisecondsSinceEpoch ~/ 1000).toString(); | ||||||
|  |     String base64TimeStamp = base64.encode(timeStamp.codeUnits); | ||||||
|  |     String md5TimeStamp = md5.convert(timeStamp.codeUnits).toString(); | ||||||
|  |     String md5DeviceCode = md5.convert(deviceCode.codeUnits).toString(); | ||||||
|  |  | ||||||
|  |     // generate token | ||||||
|  |     String token = | ||||||
|  |         'token://com.coolapk.market/dcf01e569c1e3db93a3d0fcf191a622c?$md5TimeStamp\$$md5DeviceCode&com.coolapk.market'; | ||||||
|  |     String base64Token = base64.encode(token.codeUnits); | ||||||
|  |     String md5Base64Token = md5.convert(base64Token.codeUnits).toString(); | ||||||
|  |     String md5Token = md5.convert(token.codeUnits).toString(); | ||||||
|  |  | ||||||
|  |     // generate salt and hash | ||||||
|  |     String bcryptSalt = '\$2a\$10\$${base64TimeStamp.substring(0, 14)}/${md5Token.substring(0, 6)}u'; | ||||||
|  |     String bcryptResult = BCrypt.hashpw(md5Base64Token, bcryptSalt); | ||||||
|  |     String reBcryptResult = bcryptResult.replaceRange(0, 3, '\$2y'); | ||||||
|  |     String finalToken = 'v2${base64.encode(reBcryptResult.codeUnits)}'; | ||||||
|  |  | ||||||
|  |     return {'deviceCode': deviceCode, 'token': finalToken}; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,5 +1,6 @@ | |||||||
| 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/components/generated_form.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'; | ||||||
|  |  | ||||||
| @@ -8,12 +9,23 @@ class DirectAPKLink extends AppSource { | |||||||
|  |  | ||||||
|   DirectAPKLink() { |   DirectAPKLink() { | ||||||
|     name = tr('directAPKLink'); |     name = tr('directAPKLink'); | ||||||
|     additionalSourceAppSpecificSettingFormItems = html |     additionalSourceAppSpecificSettingFormItems = [ | ||||||
|         .additionalSourceAppSpecificSettingFormItems |       ...html.additionalSourceAppSpecificSettingFormItems | ||||||
|         .where((element) => element |           .where((element) => element | ||||||
|             .where((element) => element.key == 'requestHeader') |               .where((element) => element.key == 'requestHeader') | ||||||
|             .isNotEmpty) |               .isNotEmpty) | ||||||
|         .toList(); |           .toList(), | ||||||
|  |       [ | ||||||
|  |         GeneratedFormDropdown( | ||||||
|  |             'defaultPseudoVersioningMethod', | ||||||
|  |             [ | ||||||
|  |               MapEntry('partialAPKHash', tr('partialAPKHash')), | ||||||
|  |               MapEntry('ETag', 'ETag') | ||||||
|  |             ], | ||||||
|  |             label: tr('defaultPseudoVersioningMethod'), | ||||||
|  |             defaultValue: 'partialAPKHash') | ||||||
|  |       ] | ||||||
|  |     ]; | ||||||
|     excludeCommonSettingKeys = [ |     excludeCommonSettingKeys = [ | ||||||
|       'versionExtractionRegEx', |       'versionExtractionRegEx', | ||||||
|       'matchGroupToUse', |       'matchGroupToUse', | ||||||
| @@ -57,9 +69,8 @@ class DirectAPKLink extends AppSource { | |||||||
|         additionalSettingsNew[s] = additionalSettings[s]; |         additionalSettingsNew[s] = additionalSettings[s]; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     additionalSettingsNew['defaultPseudoVersioningMethod'] = 'partialAPKHash'; |  | ||||||
|     additionalSettingsNew['directAPKLink'] = true; |     additionalSettingsNew['directAPKLink'] = true; | ||||||
|     additionalSettings['versionDetection'] = false; |     additionalSettingsNew['versionDetection'] = false; | ||||||
|     return html.getLatestAPKDetails(standardUrl, additionalSettingsNew); |     return html.getLatestAPKDetails(standardUrl, additionalSettingsNew); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -75,8 +75,18 @@ class GitHub extends AppSource { | |||||||
|       ], |       ], | ||||||
|       [GeneratedFormSwitch('verifyLatestTag', label: tr('verifyLatestTag'))], |       [GeneratedFormSwitch('verifyLatestTag', label: tr('verifyLatestTag'))], | ||||||
|       [ |       [ | ||||||
|         GeneratedFormSwitch('dontSortReleasesList', |         GeneratedFormDropdown( | ||||||
|             label: tr('dontSortReleasesList')) |             'sortMethodChoice', | ||||||
|  |             [ | ||||||
|  |               MapEntry('date', tr('releaseDate')), | ||||||
|  |               MapEntry('smartname', tr('smartname')), | ||||||
|  |               MapEntry('none', tr('none')), | ||||||
|  |               MapEntry('smartname-datefallback', | ||||||
|  |                   '${tr('smartname')} x ${tr('releaseDate')}'), | ||||||
|  |               MapEntry('name', tr('name')), | ||||||
|  |             ], | ||||||
|  |             label: tr('sortMethod'), | ||||||
|  |             defaultValue: 'date') | ||||||
|       ], |       ], | ||||||
|       [ |       [ | ||||||
|         GeneratedFormSwitch('useLatestAssetDateAsReleaseDate', |         GeneratedFormSwitch('useLatestAssetDateAsReleaseDate', | ||||||
| @@ -244,10 +254,10 @@ class GitHub extends AppSource { | |||||||
|             ? additionalSettings['filterReleaseNotesByRegEx'] |             ? additionalSettings['filterReleaseNotesByRegEx'] | ||||||
|             : null; |             : null; | ||||||
|     bool verifyLatestTag = additionalSettings['verifyLatestTag'] == true; |     bool verifyLatestTag = additionalSettings['verifyLatestTag'] == true; | ||||||
|     bool dontSortReleasesList = |  | ||||||
|         additionalSettings['dontSortReleasesList'] == true; |  | ||||||
|     bool useLatestAssetDateAsReleaseDate = |     bool useLatestAssetDateAsReleaseDate = | ||||||
|         additionalSettings['useLatestAssetDateAsReleaseDate'] == true; |         additionalSettings['useLatestAssetDateAsReleaseDate'] == true; | ||||||
|  |     String sortMethod = | ||||||
|  |         additionalSettings['sortMethodChoice'] ?? 'smartname-datefallback'; | ||||||
|     dynamic latestRelease; |     dynamic latestRelease; | ||||||
|     if (verifyLatestTag) { |     if (verifyLatestTag) { | ||||||
|       var temp = requestUrl.split('?'); |       var temp = requestUrl.split('?'); | ||||||
| @@ -316,7 +326,7 @@ class GitHub extends AppSource { | |||||||
|               ? getPublishDateFromRelease(rel) |               ? getPublishDateFromRelease(rel) | ||||||
|               : getNewestAssetDateFromRelease(rel); |               : getNewestAssetDateFromRelease(rel); | ||||||
|  |  | ||||||
|       if (dontSortReleasesList) { |       if (sortMethod == 'none') { | ||||||
|         releases = releases.reversed.toList(); |         releases = releases.reversed.toList(); | ||||||
|       } else { |       } else { | ||||||
|         releases.sort((a, b) { |         releases.sort((a, b) { | ||||||
| @@ -330,22 +340,30 @@ class GitHub extends AppSource { | |||||||
|           } else { |           } else { | ||||||
|             var nameA = a['tag_name'] ?? a['name']; |             var nameA = a['tag_name'] ?? a['name']; | ||||||
|             var nameB = b['tag_name'] ?? b['name']; |             var nameB = b['tag_name'] ?? b['name']; | ||||||
|             var stdFormats = findStandardFormatsForVersion(nameA, true) |             var stdFormats = findStandardFormatsForVersion(nameA, false) | ||||||
|                 .intersection(findStandardFormatsForVersion(nameB, true)); |                 .intersection(findStandardFormatsForVersion(nameB, false)); | ||||||
|             if (stdFormats.isNotEmpty) { |             if (sortMethod == 'date' || | ||||||
|               var reg = RegExp(stdFormats.first); |                 (sortMethod == 'smartname-datefallback' && | ||||||
|               var matchA = reg.firstMatch(nameA); |                     stdFormats.isEmpty)) { | ||||||
|               var matchB = reg.firstMatch(nameB); |  | ||||||
|               return compareAlphaNumeric( |  | ||||||
|                   (nameA as String).substring(matchA!.start, matchA.end), |  | ||||||
|                   (nameB as String).substring(matchB!.start, matchB.end)); |  | ||||||
|             } else { |  | ||||||
|               return (getReleaseDateFromRelease( |               return (getReleaseDateFromRelease( | ||||||
|                           a, useLatestAssetDateAsReleaseDate) ?? |                           a, useLatestAssetDateAsReleaseDate) ?? | ||||||
|                       DateTime(1)) |                       DateTime(1)) | ||||||
|                   .compareTo(getReleaseDateFromRelease( |                   .compareTo(getReleaseDateFromRelease( | ||||||
|                           b, useLatestAssetDateAsReleaseDate) ?? |                           b, useLatestAssetDateAsReleaseDate) ?? | ||||||
|                       DateTime(0)); |                       DateTime(0)); | ||||||
|  |             } else { | ||||||
|  |               if (sortMethod != 'name' && stdFormats.isNotEmpty) { | ||||||
|  |                 var reg = RegExp(stdFormats.last); | ||||||
|  |                 var matchA = reg.firstMatch(nameA); | ||||||
|  |                 var matchB = reg.firstMatch(nameB); | ||||||
|  |                 return compareAlphaNumeric( | ||||||
|  |                     (nameA as String).substring(matchA!.start, matchA.end), | ||||||
|  |                     (nameB as String).substring(matchB!.start, matchB.end)); | ||||||
|  |               } else { | ||||||
|  |                 // 'name' | ||||||
|  |                 return compareAlphaNumeric( | ||||||
|  |                     (nameA as String), (nameB as String)); | ||||||
|  |               } | ||||||
|             } |             } | ||||||
|           } |           } | ||||||
|         }); |         }); | ||||||
| @@ -501,7 +519,7 @@ class GitHub extends AppSource { | |||||||
|   AppNames getAppNames(String standardUrl) { |   AppNames getAppNames(String standardUrl) { | ||||||
|     String temp = standardUrl.substring(standardUrl.indexOf('://') + 3); |     String temp = standardUrl.substring(standardUrl.indexOf('://') + 3); | ||||||
|     List<String> names = temp.substring(temp.indexOf('/') + 1).split('/'); |     List<String> names = temp.substring(temp.indexOf('/') + 1).split('/'); | ||||||
|     return AppNames(names[0], names[1]); |     return AppNames(names[0], names.sublist(1).join('/')); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<Map<String, List<String>>> searchCommon( |   Future<Map<String, List<String>>> searchCommon( | ||||||
|   | |||||||
| @@ -53,8 +53,12 @@ class GitLab extends AppSource { | |||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|  |     var urlSegments = url.split('/'); | ||||||
|  |     var cutOffIndex = urlSegments.indexWhere((s) => s == '-'); | ||||||
|  |     url = | ||||||
|  |         urlSegments.sublist(0, cutOffIndex <= 0 ? null : cutOffIndex).join('/'); | ||||||
|     RegExp standardUrlRegEx = RegExp( |     RegExp standardUrlRegEx = RegExp( | ||||||
|         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+/[^/]+', |         '^https?://(www\\.)?${getSourceRegex(hosts)}/[^/]+(/[^((\b/\b)|(\b/-/\b))]+){1,20}', | ||||||
|         caseSensitive: false); |         caseSensitive: false); | ||||||
|     RegExpMatch? match = standardUrlRegEx.firstMatch(url); |     RegExpMatch? match = standardUrlRegEx.firstMatch(url); | ||||||
|     if (match == null) { |     if (match == null) { | ||||||
| @@ -116,7 +120,7 @@ class GitLab extends AppSource { | |||||||
|       Map<String, dynamic> additionalSettings) async { |       Map<String, dynamic> additionalSettings) async { | ||||||
|     String? PAT = await getPATIfAny(hostChanged ? additionalSettings : {}); |     String? PAT = await getPATIfAny(hostChanged ? additionalSettings : {}); | ||||||
|     String optionalAuth = (PAT != null) ? 'private_token=$PAT' : ''; |     String optionalAuth = (PAT != null) ? 'private_token=$PAT' : ''; | ||||||
|     return '$apkUrl?$optionalAuth'; |     return '$apkUrl${(Uri.parse(apkUrl).query.isEmpty ? '?' : '&')}$optionalAuth'; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
| @@ -126,6 +130,8 @@ class GitLab extends AppSource { | |||||||
|   ) async { |   ) async { | ||||||
|     // Prepare request params |     // Prepare request params | ||||||
|     var names = GitHub().getAppNames(standardUrl); |     var names = GitHub().getAppNames(standardUrl); | ||||||
|  |     String projectUriComponent = | ||||||
|  |         '${Uri.encodeComponent(names.author)}%2F${Uri.encodeComponent(names.name)}'; | ||||||
|     String? PAT = await getPATIfAny(hostChanged ? additionalSettings : {}); |     String? PAT = await getPATIfAny(hostChanged ? additionalSettings : {}); | ||||||
|     String optionalAuth = (PAT != null) ? 'private_token=$PAT' : ''; |     String optionalAuth = (PAT != null) ? 'private_token=$PAT' : ''; | ||||||
|  |  | ||||||
| @@ -133,7 +139,7 @@ class GitLab extends AppSource { | |||||||
|  |  | ||||||
|     // Get project ID |     // Get project ID | ||||||
|     Response res0 = await sourceRequest( |     Response res0 = await sourceRequest( | ||||||
|         'https://${hosts[0]}/api/v4/projects/${names.author}%2F${names.name}?$optionalAuth', |         'https://${hosts[0]}/api/v4/projects/$projectUriComponent?$optionalAuth', | ||||||
|         additionalSettings); |         additionalSettings); | ||||||
|     if (res0.statusCode != 200) { |     if (res0.statusCode != 200) { | ||||||
|       throw getObtainiumHttpError(res0); |       throw getObtainiumHttpError(res0); | ||||||
| @@ -145,7 +151,7 @@ class GitLab extends AppSource { | |||||||
|  |  | ||||||
|     // 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/$projectUriComponent/${trackOnly ? 'repository/tags' : 'releases'}?$optionalAuth', | ||||||
|         additionalSettings); |         additionalSettings); | ||||||
|     if (res.statusCode != 200) { |     if (res.statusCode != 200) { | ||||||
|       throw getObtainiumHttpError(res); |       throw getObtainiumHttpError(res); | ||||||
| @@ -157,30 +163,40 @@ class GitLab extends AppSource { | |||||||
|     apkDetailsList = json.map((e) { |     apkDetailsList = json.map((e) { | ||||||
|       var apkUrlsFromAssets = (e['assets']?['links'] as List<dynamic>? ?? []) |       var apkUrlsFromAssets = (e['assets']?['links'] as List<dynamic>? ?? []) | ||||||
|           .map((e) { |           .map((e) { | ||||||
|             return (e['direct_asset_url'] ?? e['url'] ?? '') as String; |             var url = (e['direct_asset_url'] ?? e['url'] ?? '') as String; | ||||||
|  |             var parsedUrl = url.isNotEmpty ? Uri.parse(url) : null; | ||||||
|  |             return MapEntry( | ||||||
|  |                 (e['name'] ?? | ||||||
|  |                     (parsedUrl != null && parsedUrl.pathSegments.isNotEmpty | ||||||
|  |                         ? parsedUrl.pathSegments.last | ||||||
|  |                         : 'unknown')) as String, | ||||||
|  |                 (e['direct_asset_url'] ?? e['url'] ?? '') as String); | ||||||
|           }) |           }) | ||||||
|           .where((s) => s.isNotEmpty) |           .where((s) => s.key.isNotEmpty) | ||||||
|           .toList(); |           .toList(); | ||||||
|       List<String> uploadedAPKsFromDescription = |       var uploadedAPKsFromDescription = ((e['description'] ?? '') as String) | ||||||
|           ((e['description'] ?? '') as String) |           .split('](') | ||||||
|               .split('](') |           .join('\n') | ||||||
|               .join('\n') |           .split('.apk)') | ||||||
|               .split('.apk)') |           .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) => 'https://${hosts[0]}/-/project/$projectId$s') | ||||||
|               .map((s) => 'https://${hosts[0]}/-/project/$projectId$s') |           .map((l) => MapEntry(Uri.parse(l).pathSegments.last, l)) | ||||||
|               .toList(); |           .toList(); | ||||||
|       var apkUrlsSet = apkUrlsFromAssets.toSet(); |       Map<String, String> apkUrls = {}; | ||||||
|       apkUrlsSet.addAll(uploadedAPKsFromDescription); |       for (var entry in apkUrlsFromAssets) { | ||||||
|  |         apkUrls[entry.key] = entry.value; | ||||||
|  |       } | ||||||
|  |       for (var entry in uploadedAPKsFromDescription) { | ||||||
|  |         apkUrls[entry.key] = entry.value; | ||||||
|  |       } | ||||||
|       var releaseDateString = |       var releaseDateString = | ||||||
|           e['released_at'] ?? e['created_at'] ?? e['commit']?['created_at']; |           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(e['tag_name'] ?? e['name'], apkUrls.entries.toList(), | ||||||
|           e['tag_name'] ?? e['name'], |           AppNames(names.author, names.name.split('/').last), | ||||||
|           getApkUrlsFromUrls(apkUrlsSet.toList()), |  | ||||||
|           GitHub().getAppNames(standardUrl), |  | ||||||
|           releaseDate: releaseDate); |           releaseDate: releaseDate); | ||||||
|     }); |     }); | ||||||
|     if (apkDetailsList.isEmpty) { |     if (apkDetailsList.isEmpty) { | ||||||
|   | |||||||
| @@ -1,3 +1,5 @@ | |||||||
|  | import 'dart:convert'; | ||||||
|  |  | ||||||
| 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:http/http.dart'; | import 'package:http/http.dart'; | ||||||
| @@ -7,32 +9,7 @@ import 'package:obtainium/providers/apps_provider.dart'; | |||||||
| import 'package:obtainium/providers/source_provider.dart'; | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
| String ensureAbsoluteUrl(String ambiguousUrl, Uri referenceAbsoluteUrl) { | String ensureAbsoluteUrl(String ambiguousUrl, Uri referenceAbsoluteUrl) { | ||||||
|   if (ambiguousUrl.startsWith('//')) { |   return referenceAbsoluteUrl.resolve(ambiguousUrl).toString(); | ||||||
|     ambiguousUrl = '${referenceAbsoluteUrl.scheme}:$ambiguousUrl'; |  | ||||||
|   } |  | ||||||
|   try { |  | ||||||
|     Uri.parse(ambiguousUrl).origin; |  | ||||||
|     return ambiguousUrl; |  | ||||||
|   } catch (err) { |  | ||||||
|     // is relative |  | ||||||
|   } |  | ||||||
|   var currPathSegments = referenceAbsoluteUrl.path |  | ||||||
|       .split('/') |  | ||||||
|       .where((element) => element.trim().isNotEmpty) |  | ||||||
|       .toList(); |  | ||||||
|   String absoluteUrl; |  | ||||||
|   if (ambiguousUrl.startsWith('/')) { |  | ||||||
|     absoluteUrl = '${referenceAbsoluteUrl.origin}$ambiguousUrl'; |  | ||||||
|   } else if (currPathSegments.isEmpty) { |  | ||||||
|     absoluteUrl = '${referenceAbsoluteUrl.origin}/$ambiguousUrl'; |  | ||||||
|   } else if (ambiguousUrl.split('/').where((e) => e.isNotEmpty).length == 1) { |  | ||||||
|     absoluteUrl = |  | ||||||
|         '${referenceAbsoluteUrl.origin}/${currPathSegments.join('/')}/$ambiguousUrl'; |  | ||||||
|   } else { |  | ||||||
|     absoluteUrl = |  | ||||||
|         '${referenceAbsoluteUrl.origin}/${currPathSegments.sublist(0, currPathSegments.length - (currPathSegments.last.contains('.') ? 1 : 0)).join('/')}/$ambiguousUrl'; |  | ||||||
|   } |  | ||||||
|   return Uri.parse(absoluteUrl).toString(); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| int compareAlphaNumeric(String a, String b) { | int compareAlphaNumeric(String a, String b) { | ||||||
| @@ -67,6 +44,27 @@ int compareAlphaNumeric(String a, String b) { | |||||||
|   return aParts.length.compareTo(bParts.length); |   return aParts.length.compareTo(bParts.length); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | List<String> collectAllStringsFromJSONObject(dynamic obj) { | ||||||
|  |   List<String> extractor(dynamic obj) { | ||||||
|  |     final results = <String>[]; | ||||||
|  |     if (obj is String) { | ||||||
|  |       results.add(obj); | ||||||
|  |     } else if (obj is List) { | ||||||
|  |       for (final item in obj) { | ||||||
|  |         results.addAll(extractor(item)); | ||||||
|  |       } | ||||||
|  |     } else if (obj is Map<String, dynamic>) { | ||||||
|  |       for (final value in obj.values) { | ||||||
|  |         results.addAll(extractor(value)); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return results; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return extractor(obj); | ||||||
|  | } | ||||||
|  |  | ||||||
| List<String> _splitAlphaNumeric(String s) { | List<String> _splitAlphaNumeric(String s) { | ||||||
|   List<String> parts = []; |   List<String> parts = []; | ||||||
|   StringBuffer sb = StringBuffer(); |   StringBuffer sb = StringBuffer(); | ||||||
| @@ -95,6 +93,13 @@ bool _isNumeric(String s) { | |||||||
|   return s.codeUnitAt(0) >= 48 && s.codeUnitAt(0) <= 57; |   return s.codeUnitAt(0) >= 48 && s.codeUnitAt(0) <= 57; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | List<MapEntry<String, String>> getLinksInLines(String lines) => RegExp( | ||||||
|  |         r'(http|ftp|https)://([\w_-]+(?:(?:\.[\w_-]+)+))([\w.,@?^=%&:/~+#-]*[\w@?^=%&/~+#-])?') | ||||||
|  |     .allMatches(lines) | ||||||
|  |     .map((match) => | ||||||
|  |         MapEntry(match.group(0)!, match.group(0)?.split('/').last ?? '')) | ||||||
|  |     .toList(); | ||||||
|  |  | ||||||
| // Given an HTTP response, grab some links according to the common additional settings | // Given an HTTP response, grab some links according to the common additional settings | ||||||
| // (those that apply to intermediate and final steps) | // (those that apply to intermediate and final steps) | ||||||
| Future<List<MapEntry<String, String>>> grabLinksCommon( | Future<List<MapEntry<String, String>>> grabLinksCommon( | ||||||
| @@ -114,12 +119,21 @@ Future<List<MapEntry<String, String>>> grabLinksCommon( | |||||||
|       .map((e) => MapEntry(ensureAbsoluteUrl(e.key, res.request!.url), e.value)) |       .map((e) => MapEntry(ensureAbsoluteUrl(e.key, res.request!.url), e.value)) | ||||||
|       .toList(); |       .toList(); | ||||||
|   if (allLinks.isEmpty) { |   if (allLinks.isEmpty) { | ||||||
|     allLinks = RegExp( |     allLinks = getLinksInLines(res.body); | ||||||
|             r'(http|ftp|https)://([\w_-]+(?:(?:\.[\w_-]+)+))([\w.,@?^=%&:/~+#-]*[\w@?^=%&/~+#-])?') |   } | ||||||
|         .allMatches(res.body) |   if (allLinks.isEmpty) { | ||||||
|         .map((match) => |     // Getting desperate | ||||||
|             MapEntry(match.group(0)!, match.group(0)?.split('/').last ?? '')) |     try { | ||||||
|         .toList(); |       var jsonStrings = collectAllStringsFromJSONObject(jsonDecode(res.body)); | ||||||
|  |       allLinks = getLinksInLines(jsonStrings.join('\n')); | ||||||
|  |       if (allLinks.isEmpty) { | ||||||
|  |         allLinks = getLinksInLines(jsonStrings.map((l) { | ||||||
|  |           return ensureAbsoluteUrl(l, res.request!.url); | ||||||
|  |         }).join('\n')); | ||||||
|  |       } | ||||||
|  |     } catch (e) { | ||||||
|  |       // | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|   List<MapEntry<String, String>> links = []; |   List<MapEntry<String, String>> links = []; | ||||||
|   bool skipSort = additionalSettings['skipSort'] == true; |   bool skipSort = additionalSettings['skipSort'] == true; | ||||||
| @@ -212,6 +226,10 @@ class HTML extends AppSource { | |||||||
|           required: true, |           required: true, | ||||||
|           additionalValidators: [(value) => regExValidator(value)]) |           additionalValidators: [(value) => regExValidator(value)]) | ||||||
|     ], |     ], | ||||||
|  |     [ | ||||||
|  |       GeneratedFormSwitch('autoLinkFilterByArch', | ||||||
|  |           label: tr('autoLinkFilterByArch'), defaultValue: false) | ||||||
|  |     ], | ||||||
|   ]; |   ]; | ||||||
|   HTML() { |   HTML() { | ||||||
|     additionalSourceAppSpecificSettingFormItems = [ |     additionalSourceAppSpecificSettingFormItems = [ | ||||||
| @@ -259,7 +277,8 @@ class HTML extends AppSource { | |||||||
|             'defaultPseudoVersioningMethod', |             'defaultPseudoVersioningMethod', | ||||||
|             [ |             [ | ||||||
|               MapEntry('partialAPKHash', tr('partialAPKHash')), |               MapEntry('partialAPKHash', tr('partialAPKHash')), | ||||||
|               MapEntry('APKLinkHash', tr('APKLinkHash')) |               MapEntry('APKLinkHash', tr('APKLinkHash')), | ||||||
|  |               MapEntry('ETag', 'ETag') | ||||||
|             ], |             ], | ||||||
|             label: tr('defaultPseudoVersioningMethod'), |             label: tr('defaultPseudoVersioningMethod'), | ||||||
|             defaultValue: 'partialAPKHash') |             defaultValue: 'partialAPKHash') | ||||||
| @@ -313,8 +332,12 @@ class HTML extends AppSource { | |||||||
|           await sourceRequest(currentUrl, additionalSettings), |           await sourceRequest(currentUrl, additionalSettings), | ||||||
|           additionalSettings['intermediateLink'][i]); |           additionalSettings['intermediateLink'][i]); | ||||||
|       if (intLinks.isEmpty) { |       if (intLinks.isEmpty) { | ||||||
|         throw NoReleasesError(); |         throw NoReleasesError(note: currentUrl); | ||||||
|       } else { |       } else { | ||||||
|  |         if (additionalSettings['intermediateLink'][i]['autoLinkFilterByArch'] == | ||||||
|  |             true) { | ||||||
|  |           intLinks = await filterApksByArch(intLinks); | ||||||
|  |         } | ||||||
|         currentUrl = intLinks.last.key; |         currentUrl = intLinks.last.key; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| @@ -329,7 +352,7 @@ class HTML extends AppSource { | |||||||
|       links = filterApks(links, additionalSettings['apkFilterRegEx'], |       links = filterApks(links, additionalSettings['apkFilterRegEx'], | ||||||
|           additionalSettings['invertAPKFilter']); |           additionalSettings['invertAPKFilter']); | ||||||
|       if (links.isEmpty) { |       if (links.isEmpty) { | ||||||
|         throw NoReleasesError(); |         throw NoReleasesError(note: currentUrl); | ||||||
|       } |       } | ||||||
|     } else { |     } else { | ||||||
|       links = [MapEntry(currentUrl, currentUrl)]; |       links = [MapEntry(currentUrl, currentUrl)]; | ||||||
| @@ -348,20 +371,32 @@ class HTML extends AppSource { | |||||||
|         additionalSettings['versionExtractWholePage'] == true |         additionalSettings['versionExtractWholePage'] == true | ||||||
|             ? versionExtractionWholePageString |             ? versionExtractionWholePageString | ||||||
|             : relDecoded); |             : relDecoded); | ||||||
|     version ??= additionalSettings['defaultPseudoVersioningMethod'] == |     var apkReqHeaders = | ||||||
|             'APKLinkHash' |         await getRequestHeaders(additionalSettings, forAPKDownload: true); | ||||||
|         ? rel.hashCode.toString() |     if (version == null && | ||||||
|         : (await checkPartialDownloadHashDynamic(rel, |         additionalSettings['defaultPseudoVersioningMethod'] == 'ETag') { | ||||||
|                 headers: await getRequestHeaders(additionalSettings, |       version = await checkETagHeader(rel, | ||||||
|                     forAPKDownload: true), |           headers: apkReqHeaders, | ||||||
|                 allowInsecure: additionalSettings['allowInsecure'] == true)) |           allowInsecure: additionalSettings['allowInsecure'] == true); | ||||||
|             .toString(); |       if (version == null) { | ||||||
|  |         throw NoVersionError(); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     version ??= | ||||||
|  |         additionalSettings['defaultPseudoVersioningMethod'] == 'APKLinkHash' | ||||||
|  |             ? rel.hashCode.toString() | ||||||
|  |             : (await checkPartialDownloadHashDynamic(rel, | ||||||
|  |                     headers: apkReqHeaders, | ||||||
|  |                     allowInsecure: additionalSettings['allowInsecure'] == true)) | ||||||
|  |                 .toString(); | ||||||
|     return APKDetails( |     return APKDetails( | ||||||
|         version, |         version, | ||||||
|         [rel] |         [rel].map((e) { | ||||||
|             .map((e) => |           var uri = Uri.parse(e); | ||||||
|                 MapEntry('${e.hashCode}-${Uri.parse(e).pathSegments.last}', e)) |           var fileName = | ||||||
|             .toList(), |               uri.pathSegments.isNotEmpty ? uri.pathSegments.last : uri.origin; | ||||||
|  |           return MapEntry('${e.hashCode}-$fileName', e); | ||||||
|  |         }).toList(), | ||||||
|         AppNames(uri.host, tr('app'))); |         AppNames(uri.host, tr('app'))); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										100
									
								
								lib/app_sources/rustore.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,100 @@ | |||||||
|  | import 'dart:convert'; | ||||||
|  | import 'dart:typed_data'; | ||||||
|  |  | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:flutter_charset_detector/flutter_charset_detector.dart'; | ||||||
|  | import 'package:http/http.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class RuStore extends AppSource { | ||||||
|  |   RuStore() { | ||||||
|  |     hosts = ['rustore.ru']; | ||||||
|  |     name = 'RuStore'; | ||||||
|  |     naiveStandardVersionDetection = true; | ||||||
|  |     showReleaseDateAsVersionToggle = true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|  |     RegExp standardUrlRegEx = RegExp( | ||||||
|  |         '^https?://(www\\.)?${getSourceRegex(hosts)}/catalog/app/+[^/]+', | ||||||
|  |         caseSensitive: false); | ||||||
|  |     RegExpMatch? 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; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Future<String> decodeString(String str) async { | ||||||
|  |     try { | ||||||
|  |       return (await CharsetDetector.autoDecode( | ||||||
|  |               Uint8List.fromList(str.codeUnits))) | ||||||
|  |           .string; | ||||||
|  |     } catch (e) { | ||||||
|  |       return str; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |     String standardUrl, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |   ) async { | ||||||
|  |     String? appId = await tryInferringAppId(standardUrl); | ||||||
|  |     Response res0 = await sourceRequest( | ||||||
|  |         'https://backapi.rustore.ru/applicationData/overallInfo/$appId', | ||||||
|  |         additionalSettings); | ||||||
|  |     if (res0.statusCode != 200) { | ||||||
|  |       throw getObtainiumHttpError(res0); | ||||||
|  |     } | ||||||
|  |     var appDetails = jsonDecode(res0.body)['body']; | ||||||
|  |     if (appDetails['appId'] == null) { | ||||||
|  |       throw NoReleasesError(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     String appName = appDetails['appName'] ?? tr('app'); | ||||||
|  |     String author = appDetails['companyName'] ?? name; | ||||||
|  |     String? dateStr = appDetails['updatedAt']; | ||||||
|  |     String? version = appDetails['versionName']; | ||||||
|  |     String? changeLog = appDetails['whatsNew']; | ||||||
|  |     if (version == null) { | ||||||
|  |       throw NoVersionError(); | ||||||
|  |     } | ||||||
|  |     DateTime? relDate; | ||||||
|  |     if (dateStr != null) { | ||||||
|  |       relDate = DateTime.parse(dateStr); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Response res1 = await sourceRequest( | ||||||
|  |         'https://backapi.rustore.ru/applicationData/download-link', | ||||||
|  |         additionalSettings, | ||||||
|  |         followRedirects: false, | ||||||
|  |         postBody: {"appId": appDetails['appId'], "firstInstall": true}); | ||||||
|  |     var downloadDetails = jsonDecode(res1.body)['body']; | ||||||
|  |     if (res1.statusCode != 200 || downloadDetails['apkUrl'] == null) { | ||||||
|  |       throw NoAPKError(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     appName = await decodeString(appName); | ||||||
|  |     author = await decodeString(author); | ||||||
|  |     changeLog = changeLog != null ? await decodeString(changeLog) : null; | ||||||
|  |  | ||||||
|  |     return APKDetails( | ||||||
|  |         version, | ||||||
|  |         getApkUrlsFromUrls([ | ||||||
|  |           (downloadDetails['apkUrl'] as String) | ||||||
|  |               .replaceAll(RegExp('\\.zip\$'), '.apk') | ||||||
|  |         ]), | ||||||
|  |         AppNames(author, appName), | ||||||
|  |         releaseDate: relDate, | ||||||
|  |         changeLog: changeLog); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -64,11 +64,11 @@ class Tencent extends AppSource { | |||||||
|       var author = json['app_detail_records'][appId]['app_info']['author']; |       var author = json['app_detail_records'][appId]['app_info']['author']; | ||||||
|       var releaseDate = |       var releaseDate = | ||||||
|           json['app_detail_records'][appId]['app_info']['update_time']; |           json['app_detail_records'][appId]['app_info']['update_time']; | ||||||
|  |       var apkName = Uri.parse(apkUrl).queryParameters['fsname'] ?? | ||||||
|  |           '${appId}_$version.apk'; | ||||||
|  |  | ||||||
|       return APKDetails( |       return APKDetails( | ||||||
|           version, |           version, [MapEntry(apkName, apkUrl)], AppNames(author, appName), | ||||||
|           [MapEntry(Uri.parse(apkUrl).queryParameters['fsname']!, apkUrl)], |  | ||||||
|           AppNames(author, appName), |  | ||||||
|           releaseDate: releaseDate != null |           releaseDate: releaseDate != null | ||||||
|               ? DateTime.fromMillisecondsSinceEpoch(releaseDate * 1000) |               ? DateTime.fromMillisecondsSinceEpoch(releaseDate * 1000) | ||||||
|               : null); |               : null); | ||||||
|   | |||||||
| @@ -1,9 +1,23 @@ | |||||||
| 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/apkpure.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'; | ||||||
|  |  | ||||||
|  | parseDateTimeMMMddCommayyyy(String? dateString) { | ||||||
|  |   DateTime? releaseDate; | ||||||
|  |   try { | ||||||
|  |     releaseDate = dateString != null | ||||||
|  |         ? DateFormat('MMM dd, yyyy').parse(dateString) | ||||||
|  |         : null; | ||||||
|  |     releaseDate = dateString != null && releaseDate == null | ||||||
|  |         ? DateFormat('MMMM dd, yyyy').parse(dateString) | ||||||
|  |         : releaseDate; | ||||||
|  |   } catch (err) { | ||||||
|  |     // ignore | ||||||
|  |   } | ||||||
|  |   return releaseDate; | ||||||
|  | } | ||||||
|  |  | ||||||
| class Uptodown extends AppSource { | class Uptodown extends AppSource { | ||||||
|   Uptodown() { |   Uptodown() { | ||||||
|     hosts = ['uptodown.com']; |     hosts = ['uptodown.com']; | ||||||
|   | |||||||
							
								
								
									
										98
									
								
								lib/app_sources/vivoappstore.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,98 @@ | |||||||
|  | import 'dart:convert'; | ||||||
|  |  | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:obtainium/custom_errors.dart'; | ||||||
|  | import 'package:obtainium/providers/source_provider.dart'; | ||||||
|  |  | ||||||
|  | class VivoAppStore extends AppSource { | ||||||
|  |   static const appDetailUrl = | ||||||
|  |       'https://h5coml.vivo.com.cn/h5coml/appdetail_h5/browser_v2/index.html?appId='; | ||||||
|  |  | ||||||
|  |   VivoAppStore() { | ||||||
|  |     name = tr('vivoAppStore'); | ||||||
|  |     hosts = ['h5.appstore.vivo.com.cn', 'h5coml.vivo.com.cn']; | ||||||
|  |     naiveStandardVersionDetection = true; | ||||||
|  |     canSearch = true; | ||||||
|  |     allowOverride = false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String sourceSpecificStandardizeURL(String url, {bool forSelection = false}) { | ||||||
|  |     var vivoAppId = parseVivoAppId(url); | ||||||
|  |     return '$appDetailUrl$vivoAppId'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<String?> tryInferringAppId(String standardUrl, | ||||||
|  |       {Map<String, dynamic> additionalSettings = const {}}) async { | ||||||
|  |     var json = await getDetailJson(standardUrl, additionalSettings); | ||||||
|  |     return json['package_name']; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<APKDetails> getLatestAPKDetails( | ||||||
|  |       String standardUrl, Map<String, dynamic> additionalSettings) async { | ||||||
|  |     var json = await getDetailJson(standardUrl, additionalSettings); | ||||||
|  |     var appName = json['title_zh'].toString(); | ||||||
|  |     var packageName = json['package_name'].toString(); | ||||||
|  |     var versionName = json['version_name'].toString(); | ||||||
|  |     var versionCode = json['version_code'].toString(); | ||||||
|  |     var developer = json['developer'].toString(); | ||||||
|  |     var uploadTime = json['upload_time'].toString(); | ||||||
|  |     var apkUrl = json['download_url'].toString(); | ||||||
|  |     var apkName = '${packageName}_$versionCode.apk'; | ||||||
|  |     return APKDetails( | ||||||
|  |         versionName, [MapEntry(apkName, apkUrl)], AppNames(developer, appName), | ||||||
|  |         releaseDate: DateTime.parse(uploadTime)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Future<Map<String, List<String>>> search(String query, | ||||||
|  |       {Map<String, dynamic> querySettings = const {}}) async { | ||||||
|  |     var apiBaseUrl = | ||||||
|  |         'https://h5-api.appstore.vivo.com.cn/h5appstore/search/result-list?app_version=2100&page_index=1&apps_per_page=20&target=local&cfrom=2&key='; | ||||||
|  |     var searchUrl = '$apiBaseUrl${Uri.encodeQueryComponent(query)}'; | ||||||
|  |     var response = await sourceRequest(searchUrl, {}); | ||||||
|  |     if (response.statusCode != 200) { | ||||||
|  |       throw getObtainiumHttpError(response); | ||||||
|  |     } | ||||||
|  |     var json = jsonDecode(response.body); | ||||||
|  |     if (json['code'] != 0 || !json['data']['appSearchResponse']['result']) { | ||||||
|  |       throw NoReleasesError(); | ||||||
|  |     } | ||||||
|  |     Map<String, List<String>> results = {}; | ||||||
|  |     var resultsJson = json['data']['appSearchResponse']['value']; | ||||||
|  |     for (var item in (resultsJson as List<dynamic>)) { | ||||||
|  |       results['$appDetailUrl${item['id']}'] = [ | ||||||
|  |         item['title_zh'].toString(), | ||||||
|  |         item['developer'].toString() | ||||||
|  |       ]; | ||||||
|  |     } | ||||||
|  |     return results; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Future<Map<String, dynamic>> getDetailJson( | ||||||
|  |       String standardUrl, Map<String, dynamic> additionalSettings) async { | ||||||
|  |     var vivoAppId = parseVivoAppId(standardUrl); | ||||||
|  |     var apiBaseUrl = 'https://h5-api.appstore.vivo.com.cn/detail/'; | ||||||
|  |     var params = '?frompage=messageh5&app_version=2100'; | ||||||
|  |     var detailUrl = '$apiBaseUrl$vivoAppId$params'; | ||||||
|  |     var response = await sourceRequest(detailUrl, additionalSettings); | ||||||
|  |     if (response.statusCode != 200) { | ||||||
|  |       throw getObtainiumHttpError(response); | ||||||
|  |     } | ||||||
|  |     var json = jsonDecode(response.body); | ||||||
|  |     if (json['id'] == null) { | ||||||
|  |       throw NoReleasesError(); | ||||||
|  |     } | ||||||
|  |     return json; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   String parseVivoAppId(String url) { | ||||||
|  |     var appId = Uri.parse(url.replaceAll('/#', '')).queryParameters['appId']; | ||||||
|  |     if (appId == null || appId.isEmpty) { | ||||||
|  |       throw InvalidURLError(name); | ||||||
|  |     } | ||||||
|  |     return appId; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -154,7 +154,7 @@ String list2FriendlyString(List<String> list) { | |||||||
|               (e.key == list.length - 1 |               (e.key == list.length - 1 | ||||||
|                   ? '' |                   ? '' | ||||||
|                   : e.key == list.length - 2 |                   : e.key == list.length - 2 | ||||||
|                       ? ', and ' |                       ? ' and ' | ||||||
|                       : ', ')) |                       : ', ')) | ||||||
|           .join(''); |           .join(''); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -23,7 +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('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'), | ||||||
| @@ -34,7 +34,8 @@ List<MapEntry<Locale, String>> supportedLocales = const [ | |||||||
|   MapEntry(Locale('pl'), 'Polski'), |   MapEntry(Locale('pl'), 'Polski'), | ||||||
|   MapEntry(Locale('ru'), 'Русский'), |   MapEntry(Locale('ru'), 'Русский'), | ||||||
|   MapEntry(Locale('bs'), 'Bosanski'), |   MapEntry(Locale('bs'), 'Bosanski'), | ||||||
|   MapEntry(Locale('pt'), 'Brasileiro'), |   MapEntry(Locale('pt'), 'Português'), | ||||||
|  |   MapEntry(Locale('pt', 'BR'), 'Brasileiro'), | ||||||
|   MapEntry(Locale('cs'), 'Česky'), |   MapEntry(Locale('cs'), 'Česky'), | ||||||
|   MapEntry(Locale('sv'), 'Svenska'), |   MapEntry(Locale('sv'), 'Svenska'), | ||||||
|   MapEntry(Locale('nl'), 'Nederlands'), |   MapEntry(Locale('nl'), 'Nederlands'), | ||||||
| @@ -44,7 +45,10 @@ List<MapEntry<Locale, String>> supportedLocales = const [ | |||||||
|   MapEntry(Locale('da'), 'Dansk'), |   MapEntry(Locale('da'), 'Dansk'), | ||||||
|   MapEntry(Locale('en', 'EO'), |   MapEntry(Locale('en', 'EO'), | ||||||
|       'Esperanto'), // https://github.com/aissat/easy_localization/issues/220#issuecomment-846035493 |       'Esperanto'), // https://github.com/aissat/easy_localization/issues/220#issuecomment-846035493 | ||||||
|   MapEntry(Locale('in'), 'Bahasa Indonesia') |   MapEntry(Locale('in'), 'Bahasa Indonesia'), | ||||||
|  |   MapEntry(Locale('ko'), '한국어'), | ||||||
|  |   MapEntry(Locale('ca'), 'Català'), | ||||||
|  |   MapEntry(Locale('ar'), 'العربية'), | ||||||
| ]; | ]; | ||||||
| const fallbackLocale = Locale('en'); | const fallbackLocale = Locale('en'); | ||||||
| const localeDir = 'assets/translations'; | const localeDir = 'assets/translations'; | ||||||
| @@ -60,11 +64,11 @@ Future<void> loadTranslations() async { | |||||||
|   var forceLocale = s.forcedLocale; |   var forceLocale = s.forcedLocale; | ||||||
|   final controller = EasyLocalizationController( |   final controller = EasyLocalizationController( | ||||||
|     saveLocale: true, |     saveLocale: true, | ||||||
|     forceLocale: forceLocale != null ? Locale(forceLocale) : null, |     forceLocale: forceLocale, | ||||||
|     fallbackLocale: fallbackLocale, |     fallbackLocale: fallbackLocale, | ||||||
|     supportedLocales: supportedLocales.map((e) => e.key).toList(), |     supportedLocales: supportedLocales.map((e) => e.key).toList(), | ||||||
|     assetLoader: const RootBundleAssetLoader(), |     assetLoader: const RootBundleAssetLoader(), | ||||||
|     useOnlyLangCode: true, |     useOnlyLangCode: false, | ||||||
|     useFallbackTranslations: true, |     useFallbackTranslations: true, | ||||||
|     path: localeDir, |     path: localeDir, | ||||||
|     onLoadError: (FlutterError e) { |     onLoadError: (FlutterError e) { | ||||||
| @@ -107,18 +111,20 @@ void main() async { | |||||||
|     ); |     ); | ||||||
|     SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge); |     SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge); | ||||||
|   } |   } | ||||||
|  |   final np = NotificationsProvider(); | ||||||
|  |   await np.initialize(); | ||||||
|   runApp(MultiProvider( |   runApp(MultiProvider( | ||||||
|     providers: [ |     providers: [ | ||||||
|       ChangeNotifierProvider(create: (context) => AppsProvider()), |       ChangeNotifierProvider(create: (context) => AppsProvider()), | ||||||
|       ChangeNotifierProvider(create: (context) => SettingsProvider()), |       ChangeNotifierProvider(create: (context) => SettingsProvider()), | ||||||
|       Provider(create: (context) => NotificationsProvider()), |       Provider(create: (context) => np), | ||||||
|       Provider(create: (context) => LogsProvider()) |       Provider(create: (context) => LogsProvider()) | ||||||
|     ], |     ], | ||||||
|     child: EasyLocalization( |     child: EasyLocalization( | ||||||
|         supportedLocales: supportedLocales.map((e) => e.key).toList(), |         supportedLocales: supportedLocales.map((e) => e.key).toList(), | ||||||
|         path: localeDir, |         path: localeDir, | ||||||
|         fallbackLocale: fallbackLocale, |         fallbackLocale: fallbackLocale, | ||||||
|         useOnlyLangCode: true, |         useOnlyLangCode: false, | ||||||
|         child: const Obtainium()), |         child: const Obtainium()), | ||||||
|   )); |   )); | ||||||
|   BackgroundFetch.registerHeadlessTask(backgroundFetchHeadlessTask); |   BackgroundFetch.registerHeadlessTask(backgroundFetchHeadlessTask); | ||||||
| @@ -166,6 +172,7 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|     SettingsProvider settingsProvider = context.watch<SettingsProvider>(); |     SettingsProvider settingsProvider = context.watch<SettingsProvider>(); | ||||||
|     AppsProvider appsProvider = context.read<AppsProvider>(); |     AppsProvider appsProvider = context.read<AppsProvider>(); | ||||||
|     LogsProvider logs = context.read<LogsProvider>(); |     LogsProvider logs = context.read<LogsProvider>(); | ||||||
|  |     NotificationsProvider notifs = context.read<NotificationsProvider>(); | ||||||
|  |  | ||||||
|     if (settingsProvider.prefs == null) { |     if (settingsProvider.prefs == null) { | ||||||
|       settingsProvider.initializeSettings(); |       settingsProvider.initializeSettings(); | ||||||
| @@ -202,16 +209,17 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|           }); |           }); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       if (!supportedLocales |       if (!supportedLocales.map((e) => e.key).contains(context.locale) || | ||||||
|               .map((e) => e.key.languageCode) |  | ||||||
|               .contains(context.locale.languageCode) || |  | ||||||
|           (settingsProvider.forcedLocale == null && |           (settingsProvider.forcedLocale == null && | ||||||
|               context.deviceLocale.languageCode != |               context.deviceLocale != context.locale)) { | ||||||
|                   context.locale.languageCode)) { |  | ||||||
|         settingsProvider.resetLocaleSafe(context); |         settingsProvider.resetLocaleSafe(context); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     WidgetsBinding.instance.addPostFrameCallback((_) { | ||||||
|  |       notifs.checkLaunchByNotif(); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|     return DynamicColorBuilder( |     return DynamicColorBuilder( | ||||||
|         builder: (ColorScheme? lightDynamic, ColorScheme? darkDynamic) { |         builder: (ColorScheme? lightDynamic, ColorScheme? darkDynamic) { | ||||||
|       // Decide on a colour/brightness scheme based on OS and user settings |       // Decide on a colour/brightness scheme based on OS and user settings | ||||||
| @@ -244,22 +252,21 @@ class _ObtainiumState extends State<Obtainium> { | |||||||
|           supportedLocales: context.supportedLocales, |           supportedLocales: context.supportedLocales, | ||||||
|           locale: context.locale, |           locale: context.locale, | ||||||
|           navigatorKey: globalNavigatorKey, |           navigatorKey: globalNavigatorKey, | ||||||
|  |           debugShowCheckedModeBanner: false, | ||||||
|           theme: ThemeData( |           theme: ThemeData( | ||||||
|               useMaterial3: true, |               useMaterial3: true, | ||||||
|               colorScheme: settingsProvider.theme == ThemeSettings.dark |               colorScheme: settingsProvider.theme == ThemeSettings.dark | ||||||
|                   ? darkColorScheme |                   ? darkColorScheme | ||||||
|                   : lightColorScheme, |                   : lightColorScheme, | ||||||
|               fontFamily: settingsProvider.useSystemFont |               fontFamily: | ||||||
|                   ? 'SystemFont' |                   settingsProvider.useSystemFont ? 'SystemFont' : 'Montserrat'), | ||||||
|                   : '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: settingsProvider.useSystemFont |               fontFamily: | ||||||
|                   ? 'SystemFont' |                   settingsProvider.useSystemFont ? 'SystemFont' : 'Montserrat'), | ||||||
|                   : '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())); | ||||||
|   | |||||||
| @@ -407,8 +407,14 @@ class AddAppPageState extends State<AddAppPage> { | |||||||
|                         defaultValue: pickedSourceOverride ?? '', |                         defaultValue: pickedSourceOverride ?? '', | ||||||
|                         [ |                         [ | ||||||
|                           MapEntry('', tr('none')), |                           MapEntry('', tr('none')), | ||||||
|                           ...sourceProvider.sources.map( |                           ...sourceProvider.sources | ||||||
|                               (s) => MapEntry(s.runtimeType.toString(), s.name)) |                               .where((s) => | ||||||
|  |                                   s.allowOverride || | ||||||
|  |                                   (pickedSource != null && | ||||||
|  |                                       pickedSource.runtimeType == | ||||||
|  |                                           s.runtimeType)) | ||||||
|  |                               .map((s) => | ||||||
|  |                                   MapEntry(s.runtimeType.toString(), s.name)) | ||||||
|                         ], |                         ], | ||||||
|                         label: tr('overrideSource')) |                         label: tr('overrideSource')) | ||||||
|                   ] |                   ] | ||||||
| @@ -575,8 +581,10 @@ class AddAppPageState extends State<AddAppPage> { | |||||||
|  |  | ||||||
|     Widget getSourcesListWidget() => Padding( |     Widget getSourcesListWidget() => Padding( | ||||||
|           padding: const EdgeInsets.all(16), |           padding: const EdgeInsets.all(16), | ||||||
|           child: Row( |           child: Wrap( | ||||||
|             mainAxisAlignment: MainAxisAlignment.spaceBetween, |             direction: Axis.horizontal, | ||||||
|  |             alignment: WrapAlignment.spaceBetween, | ||||||
|  |             spacing: 12, | ||||||
|             children: [ |             children: [ | ||||||
|               GestureDetector( |               GestureDetector( | ||||||
|                   onTap: () { |                   onTap: () { | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter/services.dart'; | import 'package:flutter/services.dart'; | ||||||
|  | import 'package:flutter_markdown/flutter_markdown.dart'; | ||||||
| import 'package:obtainium/components/generated_form_modal.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'; | ||||||
| @@ -12,6 +13,7 @@ import 'package:obtainium/providers/source_provider.dart'; | |||||||
| import 'package:url_launcher/url_launcher_string.dart'; | import 'package:url_launcher/url_launcher_string.dart'; | ||||||
| import 'package:webview_flutter/webview_flutter.dart'; | import 'package:webview_flutter/webview_flutter.dart'; | ||||||
| import 'package:provider/provider.dart'; | import 'package:provider/provider.dart'; | ||||||
|  | import 'package:markdown/markdown.dart' as md; | ||||||
|  |  | ||||||
| class AppPage extends StatefulWidget { | class AppPage extends StatefulWidget { | ||||||
|   const AppPage({super.key, required this.appId}); |   const AppPage({super.key, required this.appId}); | ||||||
| @@ -23,9 +25,32 @@ class AppPage extends StatefulWidget { | |||||||
| } | } | ||||||
|  |  | ||||||
| class _AppPageState extends State<AppPage> { | class _AppPageState extends State<AppPage> { | ||||||
|  |   late final WebViewController _webViewController; | ||||||
|  |   bool _wasWebViewOpened = false; | ||||||
|   AppInMemory? prevApp; |   AppInMemory? prevApp; | ||||||
|   bool updating = false; |   bool updating = false; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void initState() { | ||||||
|  |     super.initState(); | ||||||
|  |     _webViewController = WebViewController() | ||||||
|  |       ..setJavaScriptMode(JavaScriptMode.unrestricted) | ||||||
|  |       ..setNavigationDelegate( | ||||||
|  |         NavigationDelegate( | ||||||
|  |           onWebResourceError: (WebResourceError error) { | ||||||
|  |             if (error.isForMainFrame == true) { | ||||||
|  |               showError( | ||||||
|  |                   ObtainiumError(error.description, unexpected: true), context); | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           onNavigationRequest: (NavigationRequest request) => | ||||||
|  |               request.url.startsWith("rustore://") | ||||||
|  |                   ? NavigationDecision.prevent | ||||||
|  |                   : NavigationDecision.navigate, | ||||||
|  |         ), | ||||||
|  |       ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     var appsProvider = context.watch<AppsProvider>(); |     var appsProvider = context.watch<AppsProvider>(); | ||||||
| @@ -75,9 +100,12 @@ class _AppPageState extends State<AppPage> { | |||||||
|     bool isVersionDetectionStandard = |     bool isVersionDetectionStandard = | ||||||
|         app?.app.additionalSettings['versionDetection'] == true; |         app?.app.additionalSettings['versionDetection'] == true; | ||||||
|  |  | ||||||
|     bool installedVersionIsEstimate = trackOnly || |     bool installedVersionIsEstimate = app?.app != null ? isVersionPseudo(app!.app) : false; | ||||||
|         (app?.app.installedVersion != null && |  | ||||||
|             app?.app.additionalSettings['versionDetection'] != true); |     if (app != null && !_wasWebViewOpened) { | ||||||
|  |       _wasWebViewOpened = true; | ||||||
|  |       _webViewController.loadRequest(Uri.parse(app.app.url)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     getInfoColumn() { |     getInfoColumn() { | ||||||
|       String versionLines = ''; |       String versionLines = ''; | ||||||
| @@ -221,24 +249,41 @@ class _AppPageState extends State<AppPage> { | |||||||
|           if (app?.app.additionalSettings['about'] is String && |           if (app?.app.additionalSettings['about'] is String && | ||||||
|               app?.app.additionalSettings['about'].isNotEmpty) |               app?.app.additionalSettings['about'].isNotEmpty) | ||||||
|             Column( |             Column( | ||||||
|  |               mainAxisSize: MainAxisSize.min, | ||||||
|               children: [ |               children: [ | ||||||
|                 const SizedBox( |                 const SizedBox( | ||||||
|                   height: 48, |                   height: 48, | ||||||
|                 ), |                 ), | ||||||
|                 GestureDetector( |                 GestureDetector( | ||||||
|                   onLongPress: () { |                     onLongPress: () { | ||||||
|                     Clipboard.setData(ClipboardData( |                       Clipboard.setData(ClipboardData( | ||||||
|                         text: app?.app.additionalSettings['about'] ?? '')); |                           text: app?.app.additionalSettings['about'] ?? '')); | ||||||
|                     ScaffoldMessenger.of(context).showSnackBar(SnackBar( |                       ScaffoldMessenger.of(context).showSnackBar(SnackBar( | ||||||
|                       content: Text(tr('copiedToClipboard')), |                         content: Text(tr('copiedToClipboard')), | ||||||
|                     )); |                       )); | ||||||
|                   }, |                     }, | ||||||
|                   child: Text( |                     child: Markdown( | ||||||
|                     app?.app.additionalSettings['about'], |                       physics: NeverScrollableScrollPhysics(), | ||||||
|                     textAlign: TextAlign.center, |                       shrinkWrap: true, | ||||||
|                     style: const TextStyle(fontStyle: FontStyle.italic), |                       styleSheet: MarkdownStyleSheet( | ||||||
|                   ), |                           blockquoteDecoration: | ||||||
|                 ) |                               BoxDecoration(color: Theme.of(context).cardColor), | ||||||
|  |                           textAlign: WrapAlignment.center), | ||||||
|  |                       data: app?.app.additionalSettings['about'], | ||||||
|  |                       onTapLink: (text, href, title) { | ||||||
|  |                         if (href != null) { | ||||||
|  |                           launchUrlString(href, | ||||||
|  |                               mode: LaunchMode.externalApplication); | ||||||
|  |                         } | ||||||
|  |                       }, | ||||||
|  |                       extensionSet: md.ExtensionSet( | ||||||
|  |                         md.ExtensionSet.gitHubFlavored.blockSyntaxes, | ||||||
|  |                         [ | ||||||
|  |                           md.EmojiSyntax(), | ||||||
|  |                           ...md.ExtensionSet.gitHubFlavored.inlineSyntaxes | ||||||
|  |                         ], | ||||||
|  |                       ), | ||||||
|  |                     )) | ||||||
|               ], |               ], | ||||||
|             ), |             ), | ||||||
|         ], |         ], | ||||||
| @@ -281,7 +326,7 @@ class _AppPageState extends State<AppPage> { | |||||||
|                   ? Theme.of(context).textTheme.displaySmall |                   ? Theme.of(context).textTheme.displaySmall | ||||||
|                   : Theme.of(context).textTheme.displayLarge, |                   : Theme.of(context).textTheme.displayLarge, | ||||||
|             ), |             ), | ||||||
|             Text(tr('byX', args: [app?.app.author ?? tr('unknown')]), |             Text(tr('byX', args: [app?.author ?? tr('unknown')]), | ||||||
|                 textAlign: TextAlign.center, |                 textAlign: TextAlign.center, | ||||||
|                 style: small |                 style: small | ||||||
|                     ? Theme.of(context).textTheme.headlineSmall |                     ? Theme.of(context).textTheme.headlineSmall | ||||||
| @@ -321,22 +366,9 @@ class _AppPageState extends State<AppPage> { | |||||||
|  |  | ||||||
|     getAppWebView() => app != null |     getAppWebView() => app != null | ||||||
|         ? WebViewWidget( |         ? WebViewWidget( | ||||||
|             controller: WebViewController() |             key: ObjectKey(_webViewController), | ||||||
|               ..setJavaScriptMode(JavaScriptMode.unrestricted) |             controller: _webViewController | ||||||
|               ..setBackgroundColor(Theme.of(context).colorScheme.surface) |               ..setBackgroundColor(Theme.of(context).colorScheme.surface)) | ||||||
|               ..setJavaScriptMode(JavaScriptMode.unrestricted) |  | ||||||
|               ..setNavigationDelegate( |  | ||||||
|                 NavigationDelegate( |  | ||||||
|                   onWebResourceError: (WebResourceError error) { |  | ||||||
|                     if (error.isForMainFrame == true) { |  | ||||||
|                       showError( |  | ||||||
|                           ObtainiumError(error.description, unexpected: true), |  | ||||||
|                           context); |  | ||||||
|                     } |  | ||||||
|                   }, |  | ||||||
|                 ), |  | ||||||
|               ) |  | ||||||
|               ..loadRequest(Uri.parse(app.app.url))) |  | ||||||
|         : Container(); |         : Container(); | ||||||
|  |  | ||||||
|     showMarkUpdatedDialog() { |     showMarkUpdatedDialog() { | ||||||
|   | |||||||
| @@ -60,6 +60,9 @@ showChangeLogDialog(BuildContext context, App app, String? changesUrl, | |||||||
|                     width: MediaQuery.of(context).size.width, |                     width: MediaQuery.of(context).size.width, | ||||||
|                     height: MediaQuery.of(context).size.height - 350, |                     height: MediaQuery.of(context).size.height - 350, | ||||||
|                     child: Markdown( |                     child: Markdown( | ||||||
|  |                       styleSheet: MarkdownStyleSheet( | ||||||
|  |                           blockquoteDecoration: BoxDecoration( | ||||||
|  |                               color: Theme.of(context).cardColor)), | ||||||
|                       data: changeLog, |                       data: changeLog, | ||||||
|                       onTapLink: (text, href, title) { |                       onTapLink: (text, href, title) { | ||||||
|                         if (href != null) { |                         if (href != null) { | ||||||
| @@ -213,7 +216,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|           } |           } | ||||||
|         } |         } | ||||||
|         for (var t in authorTokens) { |         for (var t in authorTokens) { | ||||||
|           if (!app.app.author.toLowerCase().contains(t.toLowerCase())) { |           if (!app.author.toLowerCase().contains(t.toLowerCase())) { | ||||||
|             return false; |             return false; | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
| @@ -244,11 +247,11 @@ class AppsPageState extends State<AppsPage> { | |||||||
|     listedApps.sort((a, b) { |     listedApps.sort((a, b) { | ||||||
|       int result = 0; |       int result = 0; | ||||||
|       if (settingsProvider.sortColumn == SortColumnSettings.authorName) { |       if (settingsProvider.sortColumn == SortColumnSettings.authorName) { | ||||||
|         result = ((a.app.author + a.name).toLowerCase()) |         result = ((a.author + a.name).toLowerCase()) | ||||||
|             .compareTo((b.app.author + b.name).toLowerCase()); |             .compareTo((b.author + b.name).toLowerCase()); | ||||||
|       } else if (settingsProvider.sortColumn == SortColumnSettings.nameAuthor) { |       } else if (settingsProvider.sortColumn == SortColumnSettings.nameAuthor) { | ||||||
|         result = ((a.name + a.app.author).toLowerCase()) |         result = ((a.name + a.author).toLowerCase()) | ||||||
|             .compareTo((b.name + b.app.author).toLowerCase()); |             .compareTo((b.name + b.author).toLowerCase()); | ||||||
|       } else if (settingsProvider.sortColumn == |       } else if (settingsProvider.sortColumn == | ||||||
|           SortColumnSettings.releaseDate) { |           SortColumnSettings.releaseDate) { | ||||||
|         result = (a.app.releaseDate)?.compareTo( |         result = (a.app.releaseDate)?.compareTo( | ||||||
| @@ -444,7 +447,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     getVersionText(int appIndex) { |     getVersionText(int appIndex) { | ||||||
|       return '${listedApps[appIndex].app.installedVersion ?? tr('notInstalled')}${listedApps[appIndex].app.additionalSettings['trackOnly'] == true ? ' ${tr('pseudoVersion')}' : ''}'; |       return listedApps[appIndex].app.installedVersion ?? tr('notInstalled'); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     getChangesButtonString(int appIndex, bool hasChangeLogFn) { |     getChangesButtonString(int appIndex, bool hasChangeLogFn) { | ||||||
| @@ -500,7 +503,10 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                                     MediaQuery.of(context).size.width / 4), |                                     MediaQuery.of(context).size.width / 4), | ||||||
|                             child: Text(getVersionText(index), |                             child: Text(getVersionText(index), | ||||||
|                                 overflow: TextOverflow.ellipsis, |                                 overflow: TextOverflow.ellipsis, | ||||||
|                                 textAlign: TextAlign.end)), |                                 textAlign: TextAlign.end, | ||||||
|  |                                 style: isVersionPseudo(listedApps[index].app) | ||||||
|  |                                     ? TextStyle(fontStyle: FontStyle.italic) | ||||||
|  |                                     : null)), | ||||||
|                       ]), |                       ]), | ||||||
|                       Row( |                       Row( | ||||||
|                         mainAxisSize: MainAxisSize.min, |                         mainAxisSize: MainAxisSize.min, | ||||||
| @@ -524,12 +530,12 @@ class AppsPageState extends State<AppsPage> { | |||||||
|       var transparent = |       var transparent = | ||||||
|           Theme.of(context).colorScheme.surface.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) => ((e.key / (listedApps[index].app.categories.length - 1)))), |             ((e.key / (listedApps[index].app.categories.length - 1)) - 0.0001)), | ||||||
|         1 |         1 | ||||||
|       ]; |       ]; | ||||||
|       if (stops.length == 2) { |       if (stops.length == 2) { | ||||||
|         stops[0] = 1; |         stops[0] = 0.9999; | ||||||
|       } |       } | ||||||
|       return Container( |       return Container( | ||||||
|           decoration: BoxDecoration( |           decoration: BoxDecoration( | ||||||
| @@ -567,7 +573,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                     : FontWeight.normal, |                     : FontWeight.normal, | ||||||
|               ), |               ), | ||||||
|             ), |             ), | ||||||
|             subtitle: Text(tr('byX', args: [listedApps[index].app.author]), |             subtitle: Text(tr('byX', args: [listedApps[index].author]), | ||||||
|                 maxLines: 1, |                 maxLines: 1, | ||||||
|                 style: TextStyle( |                 style: TextStyle( | ||||||
|                     overflow: TextOverflow.ellipsis, |                     overflow: TextOverflow.ellipsis, | ||||||
| @@ -668,6 +674,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                 formItems.add(GeneratedFormSwitch('updates', |                 formItems.add(GeneratedFormSwitch('updates', | ||||||
|                     label: tr('updateX', args: [ |                     label: tr('updateX', args: [ | ||||||
|                       plural('apps', existingUpdateIdsAllOrSelected.length) |                       plural('apps', existingUpdateIdsAllOrSelected.length) | ||||||
|  |                           .toLowerCase() | ||||||
|                     ]), |                     ]), | ||||||
|                     defaultValue: true)); |                     defaultValue: true)); | ||||||
|               } |               } | ||||||
| @@ -675,6 +682,7 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                 formItems.add(GeneratedFormSwitch('installs', |                 formItems.add(GeneratedFormSwitch('installs', | ||||||
|                     label: tr('installX', args: [ |                     label: tr('installX', args: [ | ||||||
|                       plural('apps', newInstallIdsAllOrSelected.length) |                       plural('apps', newInstallIdsAllOrSelected.length) | ||||||
|  |                           .toLowerCase() | ||||||
|                     ]), |                     ]), | ||||||
|                     defaultValue: existingUpdateIdsAllOrSelected.isEmpty)); |                     defaultValue: existingUpdateIdsAllOrSelected.isEmpty)); | ||||||
|               } |               } | ||||||
| @@ -693,7 +701,8 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                         newInstallIdsAllOrSelected.length + |                         newInstallIdsAllOrSelected.length + | ||||||
|                         trackOnlyUpdateIdsAllOrSelected.length; |                         trackOnlyUpdateIdsAllOrSelected.length; | ||||||
|                     return GeneratedFormModal( |                     return GeneratedFormModal( | ||||||
|                       title: tr('changeX', args: [plural('apps', totalApps)]), |                       title: tr('changeX', | ||||||
|  |                           args: [plural('apps', totalApps).toLowerCase()]), | ||||||
|                       items: formItems.map((e) => [e]).toList(), |                       items: formItems.map((e) => [e]).toList(), | ||||||
|                       initValid: true, |                       initValid: true, | ||||||
|                     ); |                     ); | ||||||
| @@ -908,13 +917,15 @@ class AppsPageState extends State<AppsPage> { | |||||||
|                                               .map((e) => e.id) |                                               .map((e) => e.id) | ||||||
|                                               .toList(), |                                               .toList(), | ||||||
|                                           overrideExportSettings: false)); |                                           overrideExportSettings: false)); | ||||||
|  |                                   String fn = | ||||||
|  |                                       '${tr('obtainiumExportHyphenatedLowercase')}-${DateTime.now().toIso8601String().replaceAll(':', '-')}-count-${selectedApps.length}'; | ||||||
|                                   XFile f = XFile.fromData( |                                   XFile f = XFile.fromData( | ||||||
|                                       Uint8List.fromList( |                                       Uint8List.fromList( | ||||||
|                                           utf8.encode(exportJSON)), |                                           utf8.encode(exportJSON)), | ||||||
|                                       mimeType: 'application/json', |                                       mimeType: 'application/json', | ||||||
|                                       name: |                                       name: fn); | ||||||
|                                           '${tr('obtainiumExportHyphenatedLowercase')}-${selectedApps.length}-${DateTime.now().millisecondsSinceEpoch}'); |                                   Share.shareXFiles([f], | ||||||
|                                   Share.shareXFiles([f]); |                                       fileNameOverrides: ['$fn.json']); | ||||||
|                                 }, |                                 }, | ||||||
|                           child: Text( |                           child: Text( | ||||||
|                               '${tr('share')} - ${tr('obtainiumExport')}')), |                               '${tr('share')} - ${tr('obtainiumExport')}')), | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| import 'dart:async'; | import 'dart:async'; | ||||||
|  |  | ||||||
|  | import 'package:android_intent_plus/android_intent.dart'; | ||||||
| import 'package:animations/animations.dart'; | import 'package:animations/animations.dart'; | ||||||
| import 'package:app_links/app_links.dart'; | import 'package:app_links/app_links.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| @@ -15,6 +16,7 @@ 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:obtainium/providers/source_provider.dart'; | ||||||
| import 'package:provider/provider.dart'; | import 'package:provider/provider.dart'; | ||||||
|  | import 'package:url_launcher/url_launcher_string.dart'; | ||||||
|  |  | ||||||
| class HomePage extends StatefulWidget { | class HomePage extends StatefulWidget { | ||||||
|   const HomePage({super.key}); |   const HomePage({super.key}); | ||||||
| @@ -54,6 +56,69 @@ class _HomePageState extends State<HomePage> { | |||||||
|   void initState() { |   void initState() { | ||||||
|     super.initState(); |     super.initState(); | ||||||
|     initDeepLinks(); |     initDeepLinks(); | ||||||
|  |     WidgetsBinding.instance.addPostFrameCallback((_) async { | ||||||
|  |       var sp = context.read<SettingsProvider>(); | ||||||
|  |       if (!sp.welcomeShown) { | ||||||
|  |         await showDialog( | ||||||
|  |             context: context, | ||||||
|  |             builder: (BuildContext ctx) { | ||||||
|  |               return AlertDialog( | ||||||
|  |                 title: Text(tr('welcome')), | ||||||
|  |                 content: Column( | ||||||
|  |                   mainAxisSize: MainAxisSize.min, | ||||||
|  |                   spacing: 20, | ||||||
|  |                   children: [ | ||||||
|  |                     Text(tr('documentationLinksNote')), | ||||||
|  |                     GestureDetector( | ||||||
|  |                         onTap: () { | ||||||
|  |                           launchUrlString( | ||||||
|  |                               'https://github.com/ImranR98/Obtainium/blob/main/README.md', | ||||||
|  |                               mode: LaunchMode.externalApplication); | ||||||
|  |                         }, | ||||||
|  |                         child: Text( | ||||||
|  |                           'https://github.com/ImranR98/Obtainium/blob/main/README.md', | ||||||
|  |                           style: const TextStyle( | ||||||
|  |                               decoration: TextDecoration.underline, | ||||||
|  |                               fontWeight: FontWeight.bold), | ||||||
|  |                         )), | ||||||
|  |                     Column( | ||||||
|  |                       crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|  |                       children: [ | ||||||
|  |                         Text(tr('batteryOptimizationNote')), | ||||||
|  |                         GestureDetector( | ||||||
|  |                           onTap: () { | ||||||
|  |                             final intent = AndroidIntent( | ||||||
|  |                               action: | ||||||
|  |                                   'android.settings.IGNORE_BATTERY_OPTIMIZATION_SETTINGS', | ||||||
|  |                               package: | ||||||
|  |                                   obtainiumId, // Replace with your app's package name | ||||||
|  |                             ); | ||||||
|  |  | ||||||
|  |                             intent.launch(); | ||||||
|  |                           }, | ||||||
|  |                           child: Text( | ||||||
|  |                             tr('settings'), | ||||||
|  |                             style: const TextStyle( | ||||||
|  |                                 decoration: TextDecoration.underline, | ||||||
|  |                                 fontWeight: FontWeight.bold), | ||||||
|  |                           ), | ||||||
|  |                         ) | ||||||
|  |                       ], | ||||||
|  |                     ) | ||||||
|  |                   ], | ||||||
|  |                 ), | ||||||
|  |                 actions: [ | ||||||
|  |                   TextButton( | ||||||
|  |                       onPressed: () { | ||||||
|  |                         sp.welcomeShown = true; | ||||||
|  |                         Navigator.of(context).pop(null); | ||||||
|  |                       }, | ||||||
|  |                       child: Text(tr('ok'))), | ||||||
|  |                 ], | ||||||
|  |               ); | ||||||
|  |             }); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> initDeepLinks() async { |   Future<void> initDeepLinks() async { | ||||||
|   | |||||||
| @@ -490,8 +490,9 @@ class _ImportExportPageState extends State<ImportExportPage> { | |||||||
|                                                       searchSource[0]); |                                                       searchSource[0]); | ||||||
|                                                 } |                                                 } | ||||||
|                                               }, |                                               }, | ||||||
|                                         child: Text(tr('searchX', |                                         child: Text(tr('searchX', args: [ | ||||||
|                                             args: [tr('source')])))), |                                           tr('source').toLowerCase() | ||||||
|  |                                         ])))), | ||||||
|                               ], |                               ], | ||||||
|                             ), |                             ), | ||||||
|                             const SizedBox(height: 8), |                             const SizedBox(height: 8), | ||||||
|   | |||||||
| @@ -262,14 +262,14 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|             child: Text(tr('followSystem')), |             child: Text(tr('followSystem')), | ||||||
|           ), |           ), | ||||||
|           ...supportedLocales.map((e) => DropdownMenuItem( |           ...supportedLocales.map((e) => DropdownMenuItem( | ||||||
|                 value: e.key.toLanguageTag(), |                 value: e.key, | ||||||
|                 child: Text(e.value), |                 child: Text(e.value), | ||||||
|               )) |               )) | ||||||
|         ], |         ], | ||||||
|         onChanged: (value) { |         onChanged: (value) { | ||||||
|           settingsProvider.forcedLocale = value; |           settingsProvider.forcedLocale = value; | ||||||
|           if (value != null) { |           if (value != null) { | ||||||
|             context.setLocale(Locale(value)); |             context.setLocale(value); | ||||||
|           } else { |           } else { | ||||||
|             settingsProvider.resetLocaleSafe(context); |             settingsProvider.resetLocaleSafe(context); | ||||||
|           } |           } | ||||||
|   | |||||||
| @@ -7,8 +7,8 @@ import 'dart:io'; | |||||||
| import 'dart:math'; | import 'dart:math'; | ||||||
| import 'package:battery_plus/battery_plus.dart'; | import 'package:battery_plus/battery_plus.dart'; | ||||||
| import 'package:fluttertoast/fluttertoast.dart'; | import 'package:fluttertoast/fluttertoast.dart'; | ||||||
| import 'package:http/http.dart' as http; |  | ||||||
| import 'package:crypto/crypto.dart'; | import 'package:crypto/crypto.dart'; | ||||||
|  | import 'dart:typed_data'; | ||||||
|  |  | ||||||
| import 'package:android_intent_plus/flag.dart'; | import 'package:android_intent_plus/flag.dart'; | ||||||
| import 'package:android_package_installer/android_package_installer.dart'; | import 'package:android_package_installer/android_package_installer.dart'; | ||||||
| @@ -53,6 +53,7 @@ class AppInMemory { | |||||||
|       AppInMemory(app.deepCopy(), downloadProgress, installedInfo, icon); |       AppInMemory(app.deepCopy(), downloadProgress, installedInfo, icon); | ||||||
|  |  | ||||||
|   String get name => app.overrideName ?? app.finalName; |   String get name => app.overrideName ?? app.finalName; | ||||||
|  |   String get author => app.overrideAuthor ?? app.finalAuthor; | ||||||
| } | } | ||||||
|  |  | ||||||
| class DownloadedApk { | class DownloadedApk { | ||||||
| @@ -69,7 +70,6 @@ class DownloadedXApkDir { | |||||||
| } | } | ||||||
|  |  | ||||||
| List<String> generateStandardVersionRegExStrings() { | List<String> generateStandardVersionRegExStrings() { | ||||||
|   // TODO: Look into RegEx for non-Latin characters / non-Arabic numerals |  | ||||||
|   var basics = [ |   var basics = [ | ||||||
|     '[0-9]+', |     '[0-9]+', | ||||||
|     '[0-9]+\\.[0-9]+', |     '[0-9]+\\.[0-9]+', | ||||||
| @@ -77,7 +77,7 @@ List<String> generateStandardVersionRegExStrings() { | |||||||
|     '[0-9]+\\.[0-9]+\\.[0-9]+\\.[0-9]+' |     '[0-9]+\\.[0-9]+\\.[0-9]+\\.[0-9]+' | ||||||
|   ]; |   ]; | ||||||
|   var preSuffixes = ['-', '\\+']; |   var preSuffixes = ['-', '\\+']; | ||||||
|   var suffixes = ['alpha', 'beta', 'ose']; |   var suffixes = ['alpha', 'beta', 'ose', '[0-9]+']; | ||||||
|   var finals = ['\\+[0-9]+', '[0-9]+']; |   var finals = ['\\+[0-9]+', '[0-9]+']; | ||||||
|   List<String> results = []; |   List<String> results = []; | ||||||
|   for (var b in basics) { |   for (var b in basics) { | ||||||
| @@ -151,13 +151,15 @@ Future<File> downloadFileWithRetry(String url, String fileName, | |||||||
|     {bool useExisting = true, |     {bool useExisting = true, | ||||||
|     Map<String, String>? headers, |     Map<String, String>? headers, | ||||||
|     int retries = 3, |     int retries = 3, | ||||||
|     bool allowInsecure = false}) async { |     bool allowInsecure = false, | ||||||
|  |     LogsProvider? logs}) async { | ||||||
|   try { |   try { | ||||||
|     return await downloadFile( |     return await downloadFile( | ||||||
|         url, fileName, fileNameHasExt, onProgress, destDir, |         url, fileName, fileNameHasExt, onProgress, destDir, | ||||||
|         useExisting: useExisting, |         useExisting: useExisting, | ||||||
|         headers: headers, |         headers: headers, | ||||||
|         allowInsecure: allowInsecure); |         allowInsecure: allowInsecure, | ||||||
|  |         logs: logs); | ||||||
|   } catch (e) { |   } catch (e) { | ||||||
|     if (retries > 0 && e is ClientException) { |     if (retries > 0 && e is ClientException) { | ||||||
|       await Future.delayed(const Duration(seconds: 5)); |       await Future.delayed(const Duration(seconds: 5)); | ||||||
| @@ -166,7 +168,8 @@ Future<File> downloadFileWithRetry(String url, String fileName, | |||||||
|           useExisting: useExisting, |           useExisting: useExisting, | ||||||
|           headers: headers, |           headers: headers, | ||||||
|           retries: (retries - 1), |           retries: (retries - 1), | ||||||
|           allowInsecure: allowInsecure); |           allowInsecure: allowInsecure, | ||||||
|  |           logs: logs); | ||||||
|     } else { |     } else { | ||||||
|       rethrow; |       rethrow; | ||||||
|     } |     } | ||||||
| @@ -215,11 +218,8 @@ Future<String> checkPartialDownloadHash(String url, int bytesToGrab, | |||||||
|   return hashListOfLists(bytes); |   return hashListOfLists(bytes); | ||||||
| } | } | ||||||
|  |  | ||||||
| Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | Future<String?> checkETagHeader(String url, | ||||||
|     Function? onProgress, String destDir, |     {Map<String, String>? headers, bool allowInsecure = false}) async { | ||||||
|     {bool useExisting = true, |  | ||||||
|     Map<String, String>? headers, |  | ||||||
|     bool allowInsecure = false}) async { |  | ||||||
|   // Send the initial request but cancel it as soon as you have the headers |   // Send the initial request but cancel it as soon as you have the headers | ||||||
|   var reqHeaders = headers ?? {}; |   var reqHeaders = headers ?? {}; | ||||||
|   var req = Request('GET', Uri.parse(url)); |   var req = Request('GET', Uri.parse(url)); | ||||||
| @@ -227,6 +227,34 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | |||||||
|   var client = IOClient(createHttpClient(allowInsecure)); |   var client = IOClient(createHttpClient(allowInsecure)); | ||||||
|   StreamedResponse response = await client.send(req); |   StreamedResponse response = await client.send(req); | ||||||
|   var resHeaders = response.headers; |   var resHeaders = response.headers; | ||||||
|  |   client.close(); | ||||||
|  |   return resHeaders[HttpHeaders.etagHeader] | ||||||
|  |       ?.replaceAll('"', '') | ||||||
|  |       .hashCode | ||||||
|  |       .toString(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | deleteFile(File file) { | ||||||
|  |   try { | ||||||
|  |     file.deleteSync(recursive: true); | ||||||
|  |   } on PathAccessException catch (e) { | ||||||
|  |     throw ObtainiumError(tr('fileDeletionError', args: [e.path ?? tr('unknown')])); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | ||||||
|  |     Function? onProgress, String destDir, | ||||||
|  |     {bool useExisting = true, | ||||||
|  |     Map<String, String>? headers, | ||||||
|  |     bool allowInsecure = false, | ||||||
|  |     LogsProvider? logs}) async { | ||||||
|  |   // Send the initial request but cancel it as soon as you have the headers | ||||||
|  |   var reqHeaders = headers ?? {}; | ||||||
|  |   var req = Request('GET', Uri.parse(url)); | ||||||
|  |   req.headers.addAll(reqHeaders); | ||||||
|  |   var headersClient = IOClient(createHttpClient(allowInsecure)); | ||||||
|  |   StreamedResponse headersResponse = await headersClient.send(req); | ||||||
|  |   var resHeaders = headersResponse.headers; | ||||||
|  |  | ||||||
|   // Use the headers to decide what the file extension is, and |   // Use the headers to decide what the file extension is, and | ||||||
|   // whether it supports partial downloads (range request), and |   // whether it supports partial downloads (range request), and | ||||||
| @@ -254,21 +282,20 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | |||||||
|     rangeFeatureEnabled = |     rangeFeatureEnabled = | ||||||
|         resHeaders['accept-ranges']?.trim().toLowerCase() == 'bytes'; |         resHeaders['accept-ranges']?.trim().toLowerCase() == 'bytes'; | ||||||
|   } |   } | ||||||
|  |   headersClient.close(); | ||||||
|  |  | ||||||
|   // If you have an existing file that is usable, |   // If you have an existing file that is usable, | ||||||
|   // decide whether you can use it (either return full or resume partial) |   // decide whether you can use it (either return full or resume partial) | ||||||
|   var fullContentLength = response.contentLength; |   var fullContentLength = headersResponse.contentLength; | ||||||
|   if (useExisting && downloadedFile.existsSync()) { |   if (useExisting && downloadedFile.existsSync()) { | ||||||
|     var length = downloadedFile.lengthSync(); |     var length = downloadedFile.lengthSync(); | ||||||
|     if (fullContentLength == null || !rangeFeatureEnabled) { |     if (fullContentLength == null || !rangeFeatureEnabled) { | ||||||
|       // If there is no content length reported, assume it the existing file is fully downloaded |       // If there is no content length reported, assume it the existing file is fully downloaded | ||||||
|       // Also if the range feature is not supported, don't trust the content length if any (#1542) |       // Also if the range feature is not supported, don't trust the content length if any (#1542) | ||||||
|       client.close(); |  | ||||||
|       return downloadedFile; |       return downloadedFile; | ||||||
|     } else { |     } else { | ||||||
|       // Check if resume needed/possible |       // Check if resume needed/possible | ||||||
|       if (length == fullContentLength) { |       if (length == fullContentLength) { | ||||||
|         client.close(); |  | ||||||
|         return downloadedFile; |         return downloadedFile; | ||||||
|       } |       } | ||||||
|       if (length > fullContentLength) { |       if (length > fullContentLength) { | ||||||
| @@ -280,6 +307,41 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | |||||||
|   // Download to a '.temp' file (to distinguish btn. complete/incomplete files) |   // Download to a '.temp' file (to distinguish btn. complete/incomplete files) | ||||||
|   File tempDownloadedFile = File('${downloadedFile.path}.part'); |   File tempDownloadedFile = File('${downloadedFile.path}.part'); | ||||||
|  |  | ||||||
|  |   // If there is already a temp file, a download may already be in progress - account for this (see #2073) | ||||||
|  |   bool tempFileExists = tempDownloadedFile.existsSync(); | ||||||
|  |   if (tempFileExists && useExisting) { | ||||||
|  |     logs?.add( | ||||||
|  |         'Partial download exists - will wait: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||||
|  |     bool isDownloading = true; | ||||||
|  |     int currentTempFileSize = await tempDownloadedFile.length(); | ||||||
|  |     bool shouldReturn = false; | ||||||
|  |     while (isDownloading) { | ||||||
|  |       await Future.delayed(Duration(seconds: 7)); | ||||||
|  |       if (tempDownloadedFile.existsSync()) { | ||||||
|  |         int newTempFileSize = await tempDownloadedFile.length(); | ||||||
|  |         if (newTempFileSize > currentTempFileSize) { | ||||||
|  |           currentTempFileSize = newTempFileSize; | ||||||
|  |           logs?.add( | ||||||
|  |               'Existing partial download still in progress: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||||
|  |         } else { | ||||||
|  |           logs?.add( | ||||||
|  |               'Ignoring existing partial download: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||||
|  |           break; | ||||||
|  |         } | ||||||
|  |       } else { | ||||||
|  |         shouldReturn = downloadedFile.existsSync(); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (shouldReturn) { | ||||||
|  |       logs?.add( | ||||||
|  |           'Existing partial download completed - not repeating: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||||
|  |       return downloadedFile; | ||||||
|  |     } else { | ||||||
|  |       logs?.add( | ||||||
|  |           'Existing partial download not in progress: ${tempDownloadedFile.uri.pathSegments.last}'); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // If the range feature is not available (or you need to start a ranged req from 0), |   // If the range feature is not available (or you need to start a ranged req from 0), | ||||||
|   // complete the already-started request, else cancel it and start a ranged request, |   // complete the already-started request, else cancel it and start a ranged request, | ||||||
|   // and open the file for writing in the appropriate mode |   // and open the file for writing in the appropriate mode | ||||||
| @@ -288,66 +350,78 @@ Future<File> downloadFile(String url, String fileName, bool fileNameHasExt, | |||||||
|       : null; |       : null; | ||||||
|   int rangeStart = targetFileLength ?? 0; |   int rangeStart = targetFileLength ?? 0; | ||||||
|   IOSink? sink; |   IOSink? sink; | ||||||
|  |   req = Request('GET', Uri.parse(url)); | ||||||
|  |   req.headers.addAll(reqHeaders); | ||||||
|   if (rangeFeatureEnabled && fullContentLength != null && rangeStart > 0) { |   if (rangeFeatureEnabled && fullContentLength != null && rangeStart > 0) { | ||||||
|     client.close(); |     reqHeaders.addAll({'range': 'bytes=$rangeStart-${fullContentLength - 1}'}); | ||||||
|     client = IOClient(createHttpClient(allowInsecure)); |  | ||||||
|     req = Request('GET', Uri.parse(url)); |  | ||||||
|     req.headers.addAll(reqHeaders); |  | ||||||
|     req.headers.addAll({'range': 'bytes=$rangeStart-${fullContentLength - 1}'}); |  | ||||||
|     response = await client.send(req); |  | ||||||
|     sink = tempDownloadedFile.openWrite(mode: FileMode.writeOnlyAppend); |     sink = tempDownloadedFile.openWrite(mode: FileMode.writeOnlyAppend); | ||||||
|   } else if (tempDownloadedFile.existsSync()) { |   } else if (tempDownloadedFile.existsSync()) { | ||||||
|     tempDownloadedFile.deleteSync(recursive: true); |     deleteFile(tempDownloadedFile); | ||||||
|   } |   } | ||||||
|  |   var responseWithClient = | ||||||
|  |       await sourceRequestStreamResponse('GET', url, reqHeaders, {}); | ||||||
|  |   HttpClient responseClient = responseWithClient.key; | ||||||
|  |   HttpClientResponse response = responseWithClient.value; | ||||||
|   sink ??= tempDownloadedFile.openWrite(mode: FileMode.writeOnly); |   sink ??= tempDownloadedFile.openWrite(mode: FileMode.writeOnly); | ||||||
|  |  | ||||||
|   // Perform the download |   // Perform the download | ||||||
|   var received = 0; |   var received = 0; | ||||||
|   double? progress; |   double? progress; | ||||||
|  |   DateTime? lastProgressUpdate; // Track last progress update time | ||||||
|   if (rangeStart > 0 && fullContentLength != null) { |   if (rangeStart > 0 && fullContentLength != null) { | ||||||
|     received = rangeStart; |     received = rangeStart; | ||||||
|   } |   } | ||||||
|   await response.stream.map((s) { |   const downloadUIUpdateInterval = Duration(milliseconds: 500); | ||||||
|     received += s.length; |   const downloadBufferSize = 32 * 1024; // 32KB | ||||||
|     progress = |   final downloadBuffer = BytesBuilder(); | ||||||
|         (fullContentLength != null ? (received / fullContentLength) * 100 : 30); |   await response | ||||||
|     if (onProgress != null) { |       .asBroadcastStream() | ||||||
|       onProgress(progress); |       .map((chunk) { | ||||||
|     } |         received += chunk.length; | ||||||
|     return s; |         final now = DateTime.now(); | ||||||
|   }).pipe(sink); |         if (onProgress != null && | ||||||
|  |             (lastProgressUpdate == null || | ||||||
|  |                 now.difference(lastProgressUpdate!) >= | ||||||
|  |                     downloadUIUpdateInterval)) { | ||||||
|  |           progress = fullContentLength != null | ||||||
|  |               ? (received / fullContentLength) * 100 | ||||||
|  |               : 30; | ||||||
|  |           onProgress(progress); | ||||||
|  |           lastProgressUpdate = now; | ||||||
|  |         } | ||||||
|  |         return chunk; | ||||||
|  |       }) | ||||||
|  |       .transform(StreamTransformer<List<int>, List<int>>.fromHandlers( | ||||||
|  |         handleData: (List<int> data, EventSink<List<int>> s) { | ||||||
|  |           downloadBuffer.add(data); | ||||||
|  |           if (downloadBuffer.length >= downloadBufferSize) { | ||||||
|  |             s.add(downloadBuffer.takeBytes()); | ||||||
|  |           } | ||||||
|  |         }, | ||||||
|  |         handleDone: (EventSink<List<int>> s) { | ||||||
|  |           if (downloadBuffer.isNotEmpty) { | ||||||
|  |             s.add(downloadBuffer.takeBytes()); | ||||||
|  |           } | ||||||
|  |           s.close(); | ||||||
|  |         }, | ||||||
|  |       )) | ||||||
|  |       .pipe(sink); | ||||||
|   await sink.close(); |   await sink.close(); | ||||||
|   progress = null; |   progress = null; | ||||||
|   if (onProgress != null) { |   if (onProgress != null) { | ||||||
|     onProgress(progress); |     onProgress(progress); | ||||||
|   } |   } | ||||||
|   if (response.statusCode < 200 || response.statusCode > 299) { |   if (response.statusCode < 200 || response.statusCode > 299) { | ||||||
|     tempDownloadedFile.deleteSync(recursive: true); |     deleteFile(tempDownloadedFile); | ||||||
|     throw response.reasonPhrase ?? tr('unexpectedError'); |     throw response.reasonPhrase; | ||||||
|   } |   } | ||||||
|   if (tempDownloadedFile.existsSync()) { |   if (tempDownloadedFile.existsSync()) { | ||||||
|     tempDownloadedFile.renameSync(downloadedFile.path); |     tempDownloadedFile.renameSync(downloadedFile.path); | ||||||
|   } |   } | ||||||
|   client.close(); |   responseClient.close(); | ||||||
|   return downloadedFile; |   return downloadedFile; | ||||||
| } | } | ||||||
|  |  | ||||||
| Future<Map<String, String>> getHeaders(String url, |  | ||||||
|     {Map<String, String>? headers, bool allowInsecure = false}) async { |  | ||||||
|   var req = http.Request('GET', Uri.parse(url)); |  | ||||||
|   if (headers != null) { |  | ||||||
|     req.headers.addAll(headers); |  | ||||||
|   } |  | ||||||
|   var client = IOClient(createHttpClient(allowInsecure)); |  | ||||||
|   var response = await client.send(req); |  | ||||||
|   if (response.statusCode < 200 || response.statusCode > 299) { |  | ||||||
|     throw ObtainiumError(response.reasonPhrase ?? tr('unexpectedError')); |  | ||||||
|   } |  | ||||||
|   var returnHeaders = response.headers; |  | ||||||
|   client.close(); |  | ||||||
|   return returnHeaders; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| Future<List<PackageInfo>> getAllInstalledInfo() async { | Future<List<PackageInfo>> getAllInstalledInfo() async { | ||||||
|   return await pm.getInstalledPackages() ?? []; |   return await pm.getInstalledPackages() ?? []; | ||||||
| } | } | ||||||
| @@ -419,9 +493,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|         // Delete any partial APKs (if safe to do so) |         // Delete any partial APKs (if safe to do so) | ||||||
|         var cutoff = DateTime.now().subtract(const Duration(days: 7)); |         var cutoff = DateTime.now().subtract(const Duration(days: 7)); | ||||||
|         APKDir.listSync() |         APKDir.listSync() | ||||||
|             .where((element) => |             .where((element) => element.statSync().modified.isBefore(cutoff)) | ||||||
|                 element.path.endsWith('.part') || |  | ||||||
|                 element.statSync().modified.isBefore(cutoff)) |  | ||||||
|             .forEach((partialApk) { |             .forEach((partialApk) { | ||||||
|           if (!areDownloadsRunning()) { |           if (!areDownloadsRunning()) { | ||||||
|             partialApk.delete(recursive: true); |             partialApk.delete(recursive: true); | ||||||
| @@ -495,7 +567,8 @@ class AppsProvider with ChangeNotifier { | |||||||
|         prevProg = prog; |         prevProg = prog; | ||||||
|       }, APKDir.path, |       }, APKDir.path, | ||||||
|           useExisting: useExisting, |           useExisting: useExisting, | ||||||
|           allowInsecure: app.additionalSettings['allowInsecure'] == true); |           allowInsecure: app.additionalSettings['allowInsecure'] == true, | ||||||
|  |           logs: logs); | ||||||
|       // Set to 90 for remaining steps, will make null in 'finally' |       // Set to 90 for remaining steps, will make null in 'finally' | ||||||
|       if (apps[app.id] != null) { |       if (apps[app.id] != null) { | ||||||
|         apps[app.id]!.downloadProgress = -1; |         apps[app.id]!.downloadProgress = -1; | ||||||
| @@ -732,8 +805,10 @@ class AppsProvider with ChangeNotifier { | |||||||
|         await pm.getPackageArchiveInfo(archiveFilePath: file.file.path); |         await pm.getPackageArchiveInfo(archiveFilePath: file.file.path); | ||||||
|     if (newInfo == null) { |     if (newInfo == null) { | ||||||
|       try { |       try { | ||||||
|         file.file.deleteSync(recursive: true); |         deleteFile(file.file); | ||||||
|         additionalAPKs.forEach((a) => a.file.deleteSync(recursive: true)); |         for (var a in additionalAPKs) { | ||||||
|  |           deleteFile(a.file); | ||||||
|  |         } | ||||||
|       } catch (e) { |       } catch (e) { | ||||||
|         // |         // | ||||||
|       } finally { |       } finally { | ||||||
| @@ -772,7 +847,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|     bool installed = false; |     bool installed = false; | ||||||
|     if (code != null && code != 0 && code != 3) { |     if (code != null && code != 0 && code != 3) { | ||||||
|       try { |       try { | ||||||
|         file.file.deleteSync(recursive: true); |         deleteFile(file.file); | ||||||
|       } catch (e) { |       } catch (e) { | ||||||
|         // |         // | ||||||
|       } finally { |       } finally { | ||||||
| @@ -1124,7 +1199,8 @@ class AppsProvider with ChangeNotifier { | |||||||
|                     forAPKDownload: |                     forAPKDownload: | ||||||
|                         fileUrl.key.endsWith('.apk') ? true : false), |                         fileUrl.key.endsWith('.apk') ? true : false), | ||||||
|             useExisting: false, |             useExisting: false, | ||||||
|             allowInsecure: app.additionalSettings['allowInsecure'] == true); |             allowInsecure: app.additionalSettings['allowInsecure'] == true, | ||||||
|  |             logs: logs); | ||||||
|         notificationsProvider |         notificationsProvider | ||||||
|             .notify(DownloadedNotification(fileUrl.key, fileUrl.value)); |             .notify(DownloadedNotification(fileUrl.key, fileUrl.value)); | ||||||
|       } catch (e) { |       } catch (e) { | ||||||
| @@ -1267,7 +1343,11 @@ class AppsProvider with ChangeNotifier { | |||||||
|     var templateVersionFormats = |     var templateVersionFormats = | ||||||
|         findStandardFormatsForVersion(templateVersion, true); |         findStandardFormatsForVersion(templateVersion, true); | ||||||
|     var comparisonVersionFormats = |     var comparisonVersionFormats = | ||||||
|         findStandardFormatsForVersion(comparisonVersion, false); |         findStandardFormatsForVersion(comparisonVersion, true); | ||||||
|  |     if (comparisonVersionFormats.isEmpty) { | ||||||
|  |       comparisonVersionFormats = | ||||||
|  |           findStandardFormatsForVersion(comparisonVersion, false); | ||||||
|  |     } | ||||||
|     var commonStandardFormats = |     var commonStandardFormats = | ||||||
|         templateVersionFormats.intersection(comparisonVersionFormats); |         templateVersionFormats.intersection(comparisonVersionFormats); | ||||||
|     if (commonStandardFormats.isEmpty) { |     if (commonStandardFormats.isEmpty) { | ||||||
| @@ -1414,8 +1494,10 @@ class AppsProvider with ChangeNotifier { | |||||||
|         app = getCorrectedInstallStatusAppIfPossible(app, info) ?? app; |         app = getCorrectedInstallStatusAppIfPossible(app, info) ?? app; | ||||||
|       } |       } | ||||||
|       if (!onlyIfExists || this.apps.containsKey(app.id)) { |       if (!onlyIfExists || this.apps.containsKey(app.id)) { | ||||||
|         File('${(await getAppsDir()).path}/${app.id}.json') |         String filePath = '${(await getAppsDir()).path}/${app.id}.json'; | ||||||
|             .writeAsStringSync(jsonEncode(app.toJson())); |         File('$filePath.tmp') | ||||||
|  |             .writeAsStringSync(jsonEncode(app.toJson())); // #2089 | ||||||
|  |         File('$filePath.tmp').renameSync(filePath); | ||||||
|       } |       } | ||||||
|       try { |       try { | ||||||
|         this.apps.update(app.id, |         this.apps.update(app.id, | ||||||
| @@ -1437,7 +1519,7 @@ class AppsProvider with ChangeNotifier { | |||||||
|     await Future.wait(appIds.map((appId) async { |     await Future.wait(appIds.map((appId) async { | ||||||
|       File file = File('${(await getAppsDir()).path}/$appId.json'); |       File file = File('${(await getAppsDir()).path}/$appId.json'); | ||||||
|       if (file.existsSync()) { |       if (file.existsSync()) { | ||||||
|         file.deleteSync(recursive: true); |         deleteFile(file); | ||||||
|       } |       } | ||||||
|       apkFiles |       apkFiles | ||||||
|           .where( |           .where( | ||||||
|   | |||||||
| @@ -2,7 +2,9 @@ | |||||||
| // Contains a set of pre-defined ObtainiumNotification objects that should be used throughout the app | // Contains a set of pre-defined ObtainiumNotification objects that should be used throughout the app | ||||||
|  |  | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_local_notifications/flutter_local_notifications.dart'; | import 'package:flutter_local_notifications/flutter_local_notifications.dart'; | ||||||
|  | import 'package:obtainium/main.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'; | ||||||
|  |  | ||||||
| @@ -16,10 +18,11 @@ class ObtainiumNotification { | |||||||
|   Importance importance; |   Importance importance; | ||||||
|   int? progPercent; |   int? progPercent; | ||||||
|   bool onlyAlertOnce; |   bool onlyAlertOnce; | ||||||
|  |   String? payload; | ||||||
|  |  | ||||||
|   ObtainiumNotification(this.id, this.title, this.message, this.channelCode, |   ObtainiumNotification(this.id, this.title, this.message, this.channelCode, | ||||||
|       this.channelName, this.channelDescription, this.importance, |       this.channelName, this.channelDescription, this.importance, | ||||||
|       {this.onlyAlertOnce = false, this.progPercent}); |       {this.onlyAlertOnce = false, this.progPercent, this.payload}); | ||||||
| } | } | ||||||
|  |  | ||||||
| class UpdateNotification extends ObtainiumNotification { | class UpdateNotification extends ObtainiumNotification { | ||||||
| @@ -88,7 +91,8 @@ class ErrorCheckingUpdatesNotification extends ObtainiumNotification { | |||||||
|             'BG_UPDATE_CHECK_ERROR', |             'BG_UPDATE_CHECK_ERROR', | ||||||
|             tr('errorCheckingUpdatesNotifChannel'), |             tr('errorCheckingUpdatesNotifChannel'), | ||||||
|             tr('errorCheckingUpdatesNotifDescription'), |             tr('errorCheckingUpdatesNotifDescription'), | ||||||
|             Importance.high); |             Importance.high, | ||||||
|  |             payload: "${tr('errorCheckingUpdates')}\n$error"); | ||||||
| } | } | ||||||
|  |  | ||||||
| class AppsRemovedNotification extends ObtainiumNotification { | class AppsRemovedNotification extends ObtainiumNotification { | ||||||
| @@ -173,11 +177,50 @@ class NotificationsProvider { | |||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   Future<void> initialize() async { |   Future<void> initialize() async { | ||||||
|     isInitialized = await notifications.initialize(const InitializationSettings( |     isInitialized = await notifications.initialize( | ||||||
|             android: AndroidInitializationSettings('ic_notification'))) ?? |           const InitializationSettings( | ||||||
|  |               android: AndroidInitializationSettings('ic_notification')), | ||||||
|  |           onDidReceiveNotificationResponse: (NotificationResponse response) { | ||||||
|  |             _showNotificationPayload(response.payload); | ||||||
|  |           }, | ||||||
|  |         ) ?? | ||||||
|         false; |         false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   checkLaunchByNotif() async { | ||||||
|  |     final NotificationAppLaunchDetails? launchDetails = | ||||||
|  |         await notifications.getNotificationAppLaunchDetails(); | ||||||
|  |     if (launchDetails?.didNotificationLaunchApp ?? false) { | ||||||
|  |       _showNotificationPayload(launchDetails!.notificationResponse?.payload, | ||||||
|  |           doublePop: true); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   _showNotificationPayload(String? payload, {bool doublePop = false}) { | ||||||
|  |     if (payload?.isNotEmpty == true) { | ||||||
|  |       var title = (payload ?? '\n\n').split('\n').first; | ||||||
|  |       var content = (payload ?? '\n\n').split('\n').sublist(1).join('\n'); | ||||||
|  |       globalNavigatorKey.currentState?.push( | ||||||
|  |         PageRouteBuilder( | ||||||
|  |           pageBuilder: (context, _, __) => AlertDialog( | ||||||
|  |             title: Text(title), | ||||||
|  |             content: Text(content), | ||||||
|  |             actions: [ | ||||||
|  |               TextButton( | ||||||
|  |                   onPressed: () { | ||||||
|  |                     Navigator.of(context).pop(null); | ||||||
|  |                     if (doublePop) { | ||||||
|  |                       Navigator.of(context).pop(null); | ||||||
|  |                     } | ||||||
|  |                   }, | ||||||
|  |                   child: Text(tr('ok'))), | ||||||
|  |             ], | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   Future<void> cancel(int id) async { |   Future<void> cancel(int id) async { | ||||||
|     if (!isInitialized) { |     if (!isInitialized) { | ||||||
|       await initialize(); |       await initialize(); | ||||||
| @@ -195,7 +238,8 @@ class NotificationsProvider { | |||||||
|       Importance importance, |       Importance importance, | ||||||
|       {bool cancelExisting = false, |       {bool cancelExisting = false, | ||||||
|       int? progPercent, |       int? progPercent, | ||||||
|       bool onlyAlertOnce = false}) async { |       bool onlyAlertOnce = false, | ||||||
|  |       String? payload}) async { | ||||||
|     if (cancelExisting) { |     if (cancelExisting) { | ||||||
|       await cancel(id); |       await cancel(id); | ||||||
|     } |     } | ||||||
| @@ -216,7 +260,8 @@ class NotificationsProvider { | |||||||
|                 maxProgress: 100, |                 maxProgress: 100, | ||||||
|                 showProgress: progPercent != null, |                 showProgress: progPercent != null, | ||||||
|                 onlyAlertOnce: onlyAlertOnce, |                 onlyAlertOnce: onlyAlertOnce, | ||||||
|                 indeterminate: progPercent != null && progPercent < 0))); |                 indeterminate: progPercent != null && progPercent < 0)), | ||||||
|  |         payload: payload); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> notify(ObtainiumNotification notif, |   Future<void> notify(ObtainiumNotification notif, | ||||||
| @@ -225,5 +270,6 @@ class NotificationsProvider { | |||||||
|           notif.channelName, notif.channelDescription, notif.importance, |           notif.channelName, notif.channelDescription, notif.importance, | ||||||
|           cancelExisting: cancelExisting, |           cancelExisting: cancelExisting, | ||||||
|           onlyAlertOnce: notif.onlyAlertOnce, |           onlyAlertOnce: notif.onlyAlertOnce, | ||||||
|           progPercent: notif.progPercent); |           progPercent: notif.progPercent, | ||||||
|  |           payload: notif.payload); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -150,6 +150,15 @@ class SettingsProvider with ChangeNotifier { | |||||||
|     return result; |     return result; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   bool get welcomeShown { | ||||||
|  |     return prefs?.getBool('welcomeShown') ?? false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   set welcomeShown(bool welcomeShown) { | ||||||
|  |     prefs?.setBool('welcomeShown', welcomeShown); | ||||||
|  |     notifyListeners(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   bool checkJustStarted() { |   bool checkJustStarted() { | ||||||
|     if (justStarted) { |     if (justStarted) { | ||||||
|       justStarted = false; |       justStarted = false; | ||||||
| @@ -261,22 +270,24 @@ class SettingsProvider with ChangeNotifier { | |||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   String? get forcedLocale { |   Locale? get forcedLocale { | ||||||
|     var fl = prefs?.getString('forcedLocale'); |     var flSegs = prefs?.getString('forcedLocale')?.split('-'); | ||||||
|     return supportedLocales |     var fl = flSegs != null && flSegs.isNotEmpty | ||||||
|             .where((element) => element.key.toLanguageTag() == fl) |         ? Locale(flSegs[0], flSegs.length > 1 ? flSegs[1] : null) | ||||||
|             .isNotEmpty |         : null; | ||||||
|  |     var set = supportedLocales.where((element) => element.key == fl).isNotEmpty | ||||||
|         ? fl |         ? fl | ||||||
|         : null; |         : null; | ||||||
|  |     return set; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   set forcedLocale(String? fl) { |   set forcedLocale(Locale? fl) { | ||||||
|     if (fl == null) { |     if (fl == null) { | ||||||
|       prefs?.remove('forcedLocale'); |       prefs?.remove('forcedLocale'); | ||||||
|     } else if (supportedLocales |     } else if (supportedLocales | ||||||
|         .where((element) => element.key.toLanguageTag() == fl) |         .where((element) => element.key == fl) | ||||||
|         .isNotEmpty) { |         .isNotEmpty) { | ||||||
|       prefs?.setString('forcedLocale', fl); |       prefs?.setString('forcedLocale', fl.toLanguageTag()); | ||||||
|     } |     } | ||||||
|     notifyListeners(); |     notifyListeners(); | ||||||
|   } |   } | ||||||
| @@ -285,9 +296,7 @@ class SettingsProvider with ChangeNotifier { | |||||||
|       a.length == b.length && a.union(b).length == a.length; |       a.length == b.length && a.union(b).length == a.length; | ||||||
|  |  | ||||||
|   void resetLocaleSafe(BuildContext context) { |   void resetLocaleSafe(BuildContext context) { | ||||||
|     if (context.supportedLocales |     if (context.supportedLocales.contains(context.deviceLocale)) { | ||||||
|         .map((e) => e.languageCode) |  | ||||||
|         .contains(context.deviceLocale.languageCode)) { |  | ||||||
|       context.resetLocale(); |       context.resetLocale(); | ||||||
|     } else { |     } else { | ||||||
|       context.setLocale(context.fallbackLocale!); |       context.setLocale(context.fallbackLocale!); | ||||||
| @@ -461,7 +470,7 @@ class SettingsProvider with ChangeNotifier { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   List<String> get searchDeselected { |   List<String> get searchDeselected { | ||||||
|     return prefs?.getStringList('searchDeselected') ?? []; |     return prefs?.getStringList('searchDeselected') ?? SourceProvider().sources.map((s) => s.name).toList(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   set searchDeselected(List<String> list) { |   set searchDeselected(List<String> list) { | ||||||
|   | |||||||
| @@ -3,16 +3,18 @@ | |||||||
|  |  | ||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
| import 'dart:io'; | import 'dart:io'; | ||||||
|  | import 'package:http/http.dart' as http; | ||||||
|  | import 'dart:typed_data'; | ||||||
|  |  | ||||||
| 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'; | ||||||
| import 'package:obtainium/app_sources/codeberg.dart'; | import 'package:obtainium/app_sources/codeberg.dart'; | ||||||
|  | import 'package:obtainium/app_sources/coolapk.dart'; | ||||||
| import 'package:obtainium/app_sources/directAPKLink.dart'; | import 'package:obtainium/app_sources/directAPKLink.dart'; | ||||||
| import 'package:obtainium/app_sources/fdroid.dart'; | import 'package:obtainium/app_sources/fdroid.dart'; | ||||||
| import 'package:obtainium/app_sources/fdroidrepo.dart'; | import 'package:obtainium/app_sources/fdroidrepo.dart'; | ||||||
| @@ -23,11 +25,13 @@ import 'package:obtainium/app_sources/izzyondroid.dart'; | |||||||
| import 'package:obtainium/app_sources/html.dart'; | import 'package:obtainium/app_sources/html.dart'; | ||||||
| import 'package:obtainium/app_sources/jenkins.dart'; | import 'package:obtainium/app_sources/jenkins.dart'; | ||||||
| import 'package:obtainium/app_sources/neutroncode.dart'; | import 'package:obtainium/app_sources/neutroncode.dart'; | ||||||
|  | import 'package:obtainium/app_sources/rustore.dart'; | ||||||
| import 'package:obtainium/app_sources/sourceforge.dart'; | 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/telegramapp.dart'; | import 'package:obtainium/app_sources/telegramapp.dart'; | ||||||
| import 'package:obtainium/app_sources/tencent.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/vivoappstore.dart'; | ||||||
| import 'package:obtainium/components/generated_form.dart'; | import 'package:obtainium/components/generated_form.dart'; | ||||||
| import 'package:obtainium/custom_errors.dart'; | import 'package:obtainium/custom_errors.dart'; | ||||||
| import 'package:obtainium/mass_app_sources/githubstars.dart'; | import 'package:obtainium/mass_app_sources/githubstars.dart'; | ||||||
| @@ -151,6 +155,10 @@ appJSONCompatibilityModifiers(Map<String, dynamic> json) { | |||||||
|   if (additionalSettings['autoApkFilterByArch'] == null) { |   if (additionalSettings['autoApkFilterByArch'] == null) { | ||||||
|     additionalSettings['autoApkFilterByArch'] = false; |     additionalSettings['autoApkFilterByArch'] = false; | ||||||
|   } |   } | ||||||
|  |   // GitHub "don't sort" option to new dropdown format | ||||||
|  |   if (additionalSettings['dontSortReleasesList'] == true) { | ||||||
|  |     additionalSettings['sortMethodChoice'] = 'none'; | ||||||
|  |   } | ||||||
|   if (source.runtimeType == HTML().runtimeType) { |   if (source.runtimeType == HTML().runtimeType) { | ||||||
|     // HTML key rename |     // HTML key rename | ||||||
|     if (originalAdditionalSettings['sortByFileNamesNotLinks'] != null) { |     if (originalAdditionalSettings['sortByFileNamesNotLinks'] != null) { | ||||||
| @@ -329,6 +337,15 @@ class App { | |||||||
|     return overrideName ?? name; |     return overrideName ?? name; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   String? get overrideAuthor => | ||||||
|  |       additionalSettings['appAuthor']?.toString().trim().isNotEmpty == true | ||||||
|  |           ? additionalSettings['appAuthor'] | ||||||
|  |           : null; | ||||||
|  |  | ||||||
|  |   String get finalAuthor { | ||||||
|  |     return overrideAuthor ?? author; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   App deepCopy() => App( |   App deepCopy() => App( | ||||||
|       id, |       id, | ||||||
|       url, |       url, | ||||||
| @@ -349,7 +366,7 @@ class App { | |||||||
|       otherAssetUrls: otherAssetUrls); |       otherAssetUrls: otherAssetUrls); | ||||||
|  |  | ||||||
|   factory App.fromJson(Map<String, dynamic> json) { |   factory App.fromJson(Map<String, dynamic> json) { | ||||||
|     Map<String, dynamic> originalJSON = new Map.from(json); |     Map<String, dynamic> originalJSON = Map.from(json); | ||||||
|     try { |     try { | ||||||
|       json = appJSONCompatibilityModifiers(json); |       json = appJSONCompatibilityModifiers(json); | ||||||
|     } catch (e) { |     } catch (e) { | ||||||
| @@ -464,6 +481,23 @@ List<MapEntry<String, String>> getApkUrlsFromUrls(List<String> urls) => | |||||||
|       return MapEntry(apkSegs.isNotEmpty ? apkSegs.last : segments.last, e); |       return MapEntry(apkSegs.isNotEmpty ? apkSegs.last : segments.last, e); | ||||||
|     }).toList(); |     }).toList(); | ||||||
|  |  | ||||||
|  | Future<List<MapEntry<String, String>>> filterApksByArch( | ||||||
|  |     List<MapEntry<String, String>> apkUrls) async { | ||||||
|  |   if (apkUrls.length > 1) { | ||||||
|  |     var abis = (await DeviceInfoPlugin().androidInfo).supportedAbis; | ||||||
|  |     for (var abi in abis) { | ||||||
|  |       var urls2 = apkUrls | ||||||
|  |           .where((element) => RegExp('.*$abi.*').hasMatch(element.key)) | ||||||
|  |           .toList(); | ||||||
|  |       if (urls2.isNotEmpty && urls2.length < apkUrls.length) { | ||||||
|  |         apkUrls = urls2; | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return apkUrls; | ||||||
|  | } | ||||||
|  |  | ||||||
| getSourceRegex(List<String> hosts) { | getSourceRegex(List<String> hosts) { | ||||||
|   return '(${hosts.join('|').replaceAll('.', '\\.')})'; |   return '(${hosts.join('|').replaceAll('.', '\\.')})'; | ||||||
| } | } | ||||||
| @@ -477,6 +511,72 @@ HttpClient createHttpClient(bool insecure) { | |||||||
|   return client; |   return client; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | Future<MapEntry<HttpClient, HttpClientResponse>> sourceRequestStreamResponse( | ||||||
|  |     String method, | ||||||
|  |     String url, | ||||||
|  |     Map<String, String>? requestHeaders, | ||||||
|  |     Map<String, dynamic> additionalSettings, | ||||||
|  |     {bool followRedirects = true, | ||||||
|  |     Object? postBody}) async { | ||||||
|  |   var currentUrl = Uri.parse(url); | ||||||
|  |   var redirectCount = 0; | ||||||
|  |   const maxRedirects = 10; | ||||||
|  |   List<Cookie> cookies = []; | ||||||
|  |   while (redirectCount < maxRedirects) { | ||||||
|  |     var httpClient = | ||||||
|  |         createHttpClient(additionalSettings['allowInsecure'] == true); | ||||||
|  |     var request = await httpClient.openUrl(method, currentUrl); | ||||||
|  |     if (requestHeaders != null) { | ||||||
|  |       requestHeaders.forEach((key, value) { | ||||||
|  |         request.headers.set(key, value); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |     request.cookies.addAll(cookies); | ||||||
|  |     request.followRedirects = false; | ||||||
|  |     if (postBody != null) { | ||||||
|  |       request.headers.contentType = ContentType.json; | ||||||
|  |       request.write(jsonEncode(postBody)); | ||||||
|  |     } | ||||||
|  |     final response = await request.close(); | ||||||
|  |  | ||||||
|  |     if (followRedirects && | ||||||
|  |         (response.statusCode >= 300 && response.statusCode <= 399)) { | ||||||
|  |       final location = response.headers.value(HttpHeaders.locationHeader); | ||||||
|  |       if (location != null) { | ||||||
|  |         currentUrl = Uri.parse(ensureAbsoluteUrl(location, currentUrl)); | ||||||
|  |         redirectCount++; | ||||||
|  |         cookies = response.cookies; | ||||||
|  |         httpClient.close(); | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return MapEntry(httpClient, response); | ||||||
|  |   } | ||||||
|  |   throw ObtainiumError('Too many redirects ($maxRedirects)'); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Future<Response> httpClientResponseStreamToFinalResponse(HttpClient httpClient, | ||||||
|  |     String method, String url, HttpClientResponse response) async { | ||||||
|  |   final bytes = | ||||||
|  |       (await response.fold<BytesBuilder>(BytesBuilder(), (b, d) => b..add(d))) | ||||||
|  |           .toBytes(); | ||||||
|  |  | ||||||
|  |   final headers = <String, String>{}; | ||||||
|  |   response.headers.forEach((name, values) { | ||||||
|  |     headers[name] = values.join(', '); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   httpClient.close(); | ||||||
|  |  | ||||||
|  |   return http.Response.bytes( | ||||||
|  |     bytes, | ||||||
|  |     response.statusCode, | ||||||
|  |     headers: headers, | ||||||
|  |     request: http.Request(method, Uri.parse(url)), | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  |  | ||||||
| abstract class AppSource { | abstract class AppSource { | ||||||
|   List<String> hosts = []; |   List<String> hosts = []; | ||||||
|   bool hostChanged = false; |   bool hostChanged = false; | ||||||
| @@ -487,6 +587,7 @@ abstract class AppSource { | |||||||
|   bool appIdInferIsOptional = false; |   bool appIdInferIsOptional = false; | ||||||
|   bool allowSubDomains = false; |   bool allowSubDomains = false; | ||||||
|   bool naiveStandardVersionDetection = false; |   bool naiveStandardVersionDetection = false; | ||||||
|  |   bool allowOverride = true; | ||||||
|   bool neverAutoSelect = false; |   bool neverAutoSelect = false; | ||||||
|   bool showReleaseDateAsVersionToggle = false; |   bool showReleaseDateAsVersionToggle = false; | ||||||
|   bool versionDetectionDisallowed = false; |   bool versionDetectionDisallowed = false; | ||||||
| @@ -534,25 +635,16 @@ abstract class AppSource { | |||||||
|   Future<Response> sourceRequest( |   Future<Response> sourceRequest( | ||||||
|       String url, Map<String, dynamic> additionalSettings, |       String url, Map<String, dynamic> additionalSettings, | ||||||
|       {bool followRedirects = true, Object? postBody}) async { |       {bool followRedirects = true, Object? postBody}) async { | ||||||
|  |     var method = postBody == null ? 'GET' : 'POST'; | ||||||
|     var requestHeaders = await getRequestHeaders(additionalSettings); |     var requestHeaders = await getRequestHeaders(additionalSettings); | ||||||
|     if (requestHeaders != null || followRedirects == false) { |     var streamedResponseAndClient = await sourceRequestStreamResponse( | ||||||
|       var req = Request(postBody == null ? 'GET' : 'POST', Uri.parse(url)); |         method, url, requestHeaders, additionalSettings, | ||||||
|       req.followRedirects = followRedirects; |         followRedirects: followRedirects, postBody: postBody); | ||||||
|       if (requestHeaders != null) { |     return await httpClientResponseStreamToFinalResponse( | ||||||
|         req.headers.addAll(requestHeaders); |         streamedResponseAndClient.key, | ||||||
|       } |         method, | ||||||
|       if (postBody != null) { |         url, | ||||||
|         req.headers[HttpHeaders.contentTypeHeader] = 'application/json'; |         streamedResponseAndClient.value); | ||||||
|         req.body = jsonEncode(postBody); |  | ||||||
|       } |  | ||||||
|       return Response.fromStream(await IOClient( |  | ||||||
|               createHttpClient(additionalSettings['allowInsecure'] == true)) |  | ||||||
|           .send(req)); |  | ||||||
|     } else { |  | ||||||
|       return postBody == null |  | ||||||
|           ? get(Uri.parse(url)) |  | ||||||
|           : post(Uri.parse(url), body: jsonEncode(postBody)); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void runOnAddAppInputChange(String inputUrl) { |   void runOnAddAppInputChange(String inputUrl) { | ||||||
| @@ -621,6 +713,7 @@ 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)], | ||||||
|  |     [GeneratedFormTextField('appAuthor', label: tr('author'), required: false)], | ||||||
|     [ |     [ | ||||||
|       GeneratedFormSwitch('shizukuPretendToBeGooglePlay', |       GeneratedFormSwitch('shizukuPretendToBeGooglePlay', | ||||||
|           label: tr('shizukuPretendToBeGooglePlay'), defaultValue: false) |           label: tr('shizukuPretendToBeGooglePlay'), defaultValue: false) | ||||||
| @@ -847,6 +940,11 @@ List<MapEntry<String, String>> filterApks( | |||||||
|   return apkUrls; |   return apkUrls; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | isVersionPseudo(App app) => | ||||||
|  |   app.additionalSettings['trackOnly'] == true || | ||||||
|  |       (app.installedVersion != null && | ||||||
|  |           app.additionalSettings['versionDetection'] != true); | ||||||
|  |  | ||||||
| class SourceProvider { | class SourceProvider { | ||||||
|   // Add more source classes here so they are available via the service |   // Add more source classes here so they are available via the service | ||||||
|   List<AppSource> get sources => [ |   List<AppSource> get sources => [ | ||||||
| @@ -862,8 +960,11 @@ class SourceProvider { | |||||||
|         Uptodown(), |         Uptodown(), | ||||||
|         HuaweiAppGallery(), |         HuaweiAppGallery(), | ||||||
|         Tencent(), |         Tencent(), | ||||||
|  |         CoolApk(), | ||||||
|  |         VivoAppStore(), | ||||||
|         Jenkins(), |         Jenkins(), | ||||||
|         APKMirror(), |         APKMirror(), | ||||||
|  |         RuStore(), | ||||||
|         TelegramApp(), |         TelegramApp(), | ||||||
|         NeutronCode(), |         NeutronCode(), | ||||||
|         DirectAPKLink(), |         DirectAPKLink(), | ||||||
| @@ -972,18 +1073,8 @@ class SourceProvider { | |||||||
|     if (apk.apkUrls.isEmpty && !trackOnly) { |     if (apk.apkUrls.isEmpty && !trackOnly) { | ||||||
|       throw NoAPKError(); |       throw NoAPKError(); | ||||||
|     } |     } | ||||||
|     if (apk.apkUrls.length > 1 && |     if (additionalSettings['autoApkFilterByArch'] == true) { | ||||||
|         additionalSettings['autoApkFilterByArch'] == true) { |       apk.apkUrls = await filterApksByArch(apk.apkUrls); | ||||||
|       var abis = (await DeviceInfoPlugin().androidInfo).supportedAbis; |  | ||||||
|       for (var abi in abis) { |  | ||||||
|         var urls2 = apk.apkUrls |  | ||||||
|             .where((element) => RegExp('.*$abi.*').hasMatch(element.key)) |  | ||||||
|             .toList(); |  | ||||||
|         if (urls2.isNotEmpty && urls2.length < apk.apkUrls.length) { |  | ||||||
|           apk.apkUrls = urls2; |  | ||||||
|           break; |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |     } | ||||||
|     var name = currentApp != null ? currentApp.name.trim() : ''; |     var name = currentApp != null ? currentApp.name.trim() : ''; | ||||||
|     name = name.isNotEmpty ? name : apk.names.name; |     name = name.isNotEmpty ? name : apk.names.name; | ||||||
|   | |||||||
							
								
								
									
										398
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						| @@ -5,16 +5,16 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: android_intent_plus |       name: android_intent_plus | ||||||
|       sha256: "53136214d506d3128c9f4e5bfce3d026abe7e8038958629811a8d3223b1757c1" |       sha256: dfc1fd3a577205ae8f11e990fb4ece8c90cceabbee56fcf48e463ecf0bd6aae3 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "5.2.1" |     version: "5.3.0" | ||||||
|   android_package_installer: |   android_package_installer: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       path: "." |       path: "." | ||||||
|       ref: main |       ref: main | ||||||
|       resolved-ref: "6109d0b1dc015c016e113ded3dd80ac823f790ab" |       resolved-ref: c9e144fae63231b5d1d7b88af779c314afcb6e83 | ||||||
|       url: "https://github.com/ImranR98/android_package_installer" |       url: "https://github.com/ImranR98/android_package_installer" | ||||||
|     source: git |     source: git | ||||||
|     version: "0.0.1" |     version: "0.0.1" | ||||||
| @@ -48,10 +48,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: app_links |       name: app_links | ||||||
|       sha256: "433df2e61b10519407475d7f69e470789d23d593f28224c38ba1068597be7950" |       sha256: "85ed8fc1d25a76475914fff28cc994653bd900bc2c26e4b57a49e097febb54ba" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.3.3" |     version: "6.4.0" | ||||||
|   app_links_linux: |   app_links_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -76,38 +76,30 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.0.4" |     version: "1.0.4" | ||||||
|   archive: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: archive |  | ||||||
|       sha256: "6199c74e3db4fbfbd04f66d739e72fe11c8a8957d5f219f1f4482dbde6420b5a" |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "4.0.2" |  | ||||||
|   args: |   args: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: args |       name: args | ||||||
|       sha256: bf9f5caeea8d8fe6721a9c358dd8a5c1947b27f1cfaa18b39c301273594919e6 |       sha256: d0481093c50b1da8910eb0bb301626d4d8eb7284aa739614d2b394ee09e3ea04 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.6.0" |     version: "2.7.0" | ||||||
|   async: |   async: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: async |       name: async | ||||||
|       sha256: "947bfcf187f74dbc5e146c9eb9c0f10c9f8b30743e341481c1e2ed3ecc18c20c" |       sha256: "758e6d74e971c3e5aceb4110bfd6698efc7f501675bcfe0c775459a8140750eb" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.11.0" |     version: "2.13.0" | ||||||
|   background_fetch: |   background_fetch: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: background_fetch |       name: background_fetch | ||||||
|       sha256: e9f26ae54d88310b7ac2a68f2f9fcee0081a4d5f11100f233a70702021e7ac4f |       sha256: "442e82f508708be89fd0cc7e1dc3b27bc7c6c8c39a47967ccb7ed1c57b9108b5" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.3.7" |     version: "1.3.8" | ||||||
|   battery_plus: |   battery_plus: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -124,62 +116,54 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.1" |     version: "2.0.1" | ||||||
|  |   bcrypt: | ||||||
|  |     dependency: "direct main" | ||||||
|  |     description: | ||||||
|  |       name: bcrypt | ||||||
|  |       sha256: "9dc3f234d5935a76917a6056613e1a6d9b53f7fa56f98e24cd49b8969307764b" | ||||||
|  |       url: "https://pub.dev" | ||||||
|  |     source: hosted | ||||||
|  |     version: "1.1.3" | ||||||
|   boolean_selector: |   boolean_selector: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: boolean_selector |       name: boolean_selector | ||||||
|       sha256: "6cfb5af12253eaf2b368f07bacc5a80d1301a071c73360d746b7f2e32d762c66" |       sha256: "8aab1771e1243a5063b8b0ff68042d67334e3feab9e95b9490f9a6ebf73b42ea" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.1" |     version: "2.1.2" | ||||||
|   characters: |   characters: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: characters |       name: characters | ||||||
|       sha256: "04a925763edad70e8443c99234dc3328f442e811f1d8fd1a72f1c8ad0f69a605" |       sha256: f71061c654a3380576a52b451dd5532377954cf9dbd272a78fc8479606670803 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.3.0" |     version: "1.4.0" | ||||||
|   checked_yaml: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: checked_yaml |  | ||||||
|       sha256: feb6bed21949061731a7a75fc5d2aa727cf160b91af9a3e464c5e3a32e28b5ff |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "2.0.3" |  | ||||||
|   cli_util: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: cli_util |  | ||||||
|       sha256: ff6785f7e9e3c38ac98b2fb035701789de90154024a75b6cb926445e83197d1c |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "0.4.2" |  | ||||||
|   clock: |   clock: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: clock |       name: clock | ||||||
|       sha256: cb6d7f03e1de671e34607e909a7213e31d7752be4fb66a86d29fe1eb14bfb5cf |       sha256: fddb70d9b5277016c77a80201021d40a2247104d9f4aa7bab7157b7e3f05b84b | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.1.1" |     version: "1.1.2" | ||||||
|   collection: |   collection: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: collection |       name: collection | ||||||
|       sha256: a1ace0a119f20aabc852d165077c036cd864315bd99b7eaa10a60100341941bf |       sha256: "2f5709ae4d3d59dd8f7cd309b4e023046b57d8a6c82130785d2b0e5868084e76" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.19.0" |     version: "1.19.1" | ||||||
|   connectivity_plus: |   connectivity_plus: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: connectivity_plus |       name: connectivity_plus | ||||||
|       sha256: e0817759ec6d2d8e57eb234e6e57d2173931367a865850c7acea40d4b4f9c27d |       sha256: "051849e2bd7c7b3bc5844ea0d096609ddc3a859890ec3a9ac4a65a2620cc1f99" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.1.1" |     version: "6.1.4" | ||||||
|   connectivity_plus_platform_interface: |   connectivity_plus_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -224,18 +208,18 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: dbus |       name: dbus | ||||||
|       sha256: "365c771ac3b0e58845f39ec6deebc76e3276aa9922b0cc60840712094d9047ac" |       sha256: "79e0c23480ff85dc68de79e2cd6334add97e48f7f4865d17686dd6ea81a47e8c" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.7.10" |     version: "0.7.11" | ||||||
|   device_info_plus: |   device_info_plus: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: device_info_plus |       name: device_info_plus | ||||||
|       sha256: "4fa68e53e26ab17b70ca39f072c285562cfc1589df5bb1e9295db90f6645f431" |       sha256: "0c6396126421b590089447154c5f98a5de423b70cfb15b1578fd018843ee6f53" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "11.2.0" |     version: "11.4.0" | ||||||
|   device_info_plus_platform_interface: |   device_info_plus_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -256,10 +240,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: easy_localization |       name: easy_localization | ||||||
|       sha256: fa59bcdbbb911a764aa6acf96bbb6fa7a5cf8234354fc45ec1a43a0349ef0201 |       sha256: "0f5239c7b8ab06c66440cfb0e9aa4b4640429c6668d5a42fe389c5de42220b12" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.7" |     version: "3.0.7+1" | ||||||
|   easy_logger: |   easy_logger: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -280,18 +264,18 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: fake_async |       name: fake_async | ||||||
|       sha256: "511392330127add0b769b75a987850d136345d9227c6b94c96a04cf4a391bf78" |       sha256: "5368f224a74523e8d2e7399ea1638b37aecfca824a3cc4dfdf77bf1fa905ac44" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.3.1" |     version: "1.3.3" | ||||||
|   ffi: |   ffi: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: ffi |       name: ffi | ||||||
|       sha256: "16ed7b077ef01ad6170a3d0c57caa4a112a38d7a2ed5602e0aca9ca6f3d98da6" |       sha256: "289279317b4b16eb2bb7e271abccd4bf84ec9bdcbe999e278a94b804f5630418" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.3" |     version: "2.1.4" | ||||||
|   file: |   file: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -304,10 +288,10 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: file_picker |       name: file_picker | ||||||
|       sha256: c904b4ab56d53385563c7c39d8e9fa9af086f91495dfc48717ad84a42c3cf204 |       sha256: "77f8e81d22d2a07d0dee2c62e1dda71dc1da73bf43bb2d45af09727406167964" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "8.1.7" |     version: "10.1.9" | ||||||
|   fixnum: |   fixnum: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -320,18 +304,18 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: flex_color_picker |       name: flex_color_picker | ||||||
|       sha256: c083b79f1c57eaeed9f464368be376951230b3cb1876323b784626152a86e480 |       sha256: "8f753a1a026a13ea5cc5eddbae3ceb886f2537569ab2e5208efb1e3bb5af72ff" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.7.0" |     version: "3.7.1" | ||||||
|   flex_seed_scheme: |   flex_seed_scheme: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: flex_seed_scheme |       name: flex_seed_scheme | ||||||
|       sha256: d3ba3c5c92d2d79d45e94b4c6c71d01fac3c15017da1545880c53864da5dfeb0 |       sha256: b06d8b367b84cbf7ca5c5603c858fa5edae88486c4e4da79ac1044d73b6c62ec | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.5.0" |     version: "3.5.1" | ||||||
|   flutter: |   flutter: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: flutter |     description: flutter | ||||||
| @@ -345,14 +329,54 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.0.3" |     version: "6.0.3" | ||||||
|  |   flutter_charset_detector: | ||||||
|  |     dependency: "direct main" | ||||||
|  |     description: | ||||||
|  |       name: flutter_charset_detector | ||||||
|  |       sha256: "21f6fe8172fbfe3ba9d2fe0dba3702ba07f682315e829a68d49185a0c80d5ad0" | ||||||
|  |       url: "https://pub.dev" | ||||||
|  |     source: hosted | ||||||
|  |     version: "5.0.0" | ||||||
|  |   flutter_charset_detector_android: | ||||||
|  |     dependency: transitive | ||||||
|  |     description: | ||||||
|  |       name: flutter_charset_detector_android | ||||||
|  |       sha256: "443145e8fc8515b3b32aee375691e40dd59197a86a2ae153166bc88c8200d83b" | ||||||
|  |       url: "https://pub.dev" | ||||||
|  |     source: hosted | ||||||
|  |     version: "3.0.0" | ||||||
|  |   flutter_charset_detector_darwin: | ||||||
|  |     dependency: transitive | ||||||
|  |     description: | ||||||
|  |       name: flutter_charset_detector_darwin | ||||||
|  |       sha256: daac20390275efb92fbb14350fe11286c5e29c7b80d6b0867f52d760f0d69763 | ||||||
|  |       url: "https://pub.dev" | ||||||
|  |     source: hosted | ||||||
|  |     version: "1.1.0" | ||||||
|  |   flutter_charset_detector_platform_interface: | ||||||
|  |     dependency: transitive | ||||||
|  |     description: | ||||||
|  |       name: flutter_charset_detector_platform_interface | ||||||
|  |       sha256: "1c09ed7b314a5a9dde76057b98b7d35458ba881eed03d5e5b6f7f74b4869d18c" | ||||||
|  |       url: "https://pub.dev" | ||||||
|  |     source: hosted | ||||||
|  |     version: "1.1.0" | ||||||
|  |   flutter_charset_detector_web: | ||||||
|  |     dependency: transitive | ||||||
|  |     description: | ||||||
|  |       name: flutter_charset_detector_web | ||||||
|  |       sha256: e3ac65f94b12f4887937b21a19365d7927db816840cb93274e3861241cb0e9f2 | ||||||
|  |       url: "https://pub.dev" | ||||||
|  |     source: hosted | ||||||
|  |     version: "2.0.0" | ||||||
|   flutter_fgbg: |   flutter_fgbg: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: flutter_fgbg |       name: flutter_fgbg | ||||||
|       sha256: e02ad0738ba5fc7f331b62acb0d74aa540626a6441ae18fad685faa5ac4ad7a5 |       sha256: eb6da9b2047372566a6e17b505975fe5bace94af01f6fc825c4b6f81baa6c447 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.6.0" |     version: "0.7.1" | ||||||
|   flutter_keyboard_visibility: |   flutter_keyboard_visibility: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -401,22 +425,14 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.0.0" |     version: "1.0.0" | ||||||
|   flutter_launcher_icons: |  | ||||||
|     dependency: "direct dev" |  | ||||||
|     description: |  | ||||||
|       name: flutter_launcher_icons |  | ||||||
|       sha256: "31cd0885738e87c72d6f055564d37fabcdacee743b396b78c7636c169cac64f5" |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "0.14.2" |  | ||||||
|   flutter_lints: |   flutter_lints: | ||||||
|     dependency: "direct dev" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: flutter_lints |       name: flutter_lints | ||||||
|       sha256: "5398f14efa795ffb7a33e9b6a08798b26a180edac4ad7db3f231e40f82ce11e1" |       sha256: "3105dc8492f6183fb076ccf1f351ac3d60564bff92e20bfc4af9cc1651f4e7e1" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "5.0.0" |     version: "6.0.0" | ||||||
|   flutter_local_notifications: |   flutter_local_notifications: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -450,20 +466,20 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: flutter_markdown |       name: flutter_markdown | ||||||
|       sha256: e37f4c69a07b07bb92622ef6b131a53c9aae48f64b176340af9e8e5238718487 |       sha256: "08fb8315236099ff8e90cb87bb2b935e0a724a3af1623000a9cec930468e0f27" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.7.5" |     version: "0.7.7+1" | ||||||
|   flutter_plugin_android_lifecycle: |   flutter_plugin_android_lifecycle: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: flutter_plugin_android_lifecycle |       name: flutter_plugin_android_lifecycle | ||||||
|       sha256: "615a505aef59b151b46bbeef55b36ce2b6ed299d160c51d84281946f0aa0ce0e" |       sha256: f948e346c12f8d5480d2825e03de228d0eb8c3a737e4cdaa122267b89c022b5e | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.0.24" |     version: "2.0.28" | ||||||
|   flutter_test: |   flutter_test: | ||||||
|     dependency: "direct dev" |     dependency: transitive | ||||||
|     description: flutter |     description: flutter | ||||||
|     source: sdk |     source: sdk | ||||||
|     version: "0.0.0" |     version: "0.0.0" | ||||||
| @@ -484,18 +500,18 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: fluttertoast |       name: fluttertoast | ||||||
|       sha256: "24467dc20bbe49fd63e57d8e190798c4d22cbbdac30e54209d153a15273721d1" |       sha256: "25e51620424d92d3db3832464774a6143b5053f15e382d8ffbfd40b6e795dcf1" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "8.2.10" |     version: "8.2.12" | ||||||
|   fraction: |   fraction: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: fraction |       name: fraction | ||||||
|       sha256: "7804c9a73d26bd3d5ccf52b7225eecd0af4e33b310729726dc8f8bb14c217716" |       sha256: dd487c01c0bfdcccc44d15250b24b4bee94b6981e01e8c41007a1b02f395ea01 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "5.0.4" |     version: "5.0.5" | ||||||
|   gtk: |   gtk: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -516,18 +532,18 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: html |       name: html | ||||||
|       sha256: "1fc58edeaec4307368c60d59b7e15b9d658b57d7f3125098b6294153c75337ec" |       sha256: "6d1264f2dffa1b1101c25a91dff0dc2daee4c18e87cd8538729773c073dbf602" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.15.5" |     version: "0.15.6" | ||||||
|   http: |   http: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: http |       name: http | ||||||
|       sha256: b9c29a161230ee03d3ccf545097fccd9b87a5264228c5d348202e0f0c28f9010 |       sha256: "2c11f3f94c687ee9bad77c171151672986360b2b001d109814ee7140b2cf261b" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.2.2" |     version: "1.4.0" | ||||||
|   http_parser: |   http_parser: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -536,46 +552,30 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "4.1.2" |     version: "4.1.2" | ||||||
|   image: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: image |  | ||||||
|       sha256: "8346ad4b5173924b5ddddab782fc7d8a6300178c8b1dc427775405a01701c4a6" |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "4.5.2" |  | ||||||
|   intl: |   intl: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: intl |       name: intl | ||||||
|       sha256: d6f56758b7d3014a48af9701c085700aac781a92a87a62b1333b46d8879661cf |       sha256: "3df61194eb431efc39c4ceba583b95633a403f46c9fd341e550ce0bfa50e9aa5" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.19.0" |     version: "0.20.2" | ||||||
|   json_annotation: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: json_annotation |  | ||||||
|       sha256: "1ce844379ca14835a50d2f019a3099f419082cfdd231cd86a142af94dd5c6bb1" |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "4.9.0" |  | ||||||
|   leak_tracker: |   leak_tracker: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: leak_tracker |       name: leak_tracker | ||||||
|       sha256: "7bb2830ebd849694d1ec25bf1f44582d6ac531a57a365a803a6034ff751d2d06" |       sha256: "6bb818ecbdffe216e81182c2f0714a2e62b593f4a4f13098713ff1685dfb6ab0" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "10.0.7" |     version: "10.0.9" | ||||||
|   leak_tracker_flutter_testing: |   leak_tracker_flutter_testing: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: leak_tracker_flutter_testing |       name: leak_tracker_flutter_testing | ||||||
|       sha256: "9491a714cca3667b60b5c420da8217e6de0d1ba7a5ec322fab01758f6998f379" |       sha256: f8b613e7e6a13ec79cfdc0e97638fddb3ab848452eff057653abd3edba760573 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.0.8" |     version: "3.0.9" | ||||||
|   leak_tracker_testing: |   leak_tracker_testing: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -588,26 +588,26 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: lints |       name: lints | ||||||
|       sha256: c35bb79562d980e9a453fc715854e1ed39e24e7d0297a880ef54e17f9874a9d7 |       sha256: a5e2b223cb7c9c8efdc663ef484fdd95bb243bff242ef5b13e26883547fce9a0 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "5.1.1" |     version: "6.0.0" | ||||||
|   markdown: |   markdown: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: markdown |       name: markdown | ||||||
|       sha256: ef2a1298144e3f985cc736b22e0ccdaf188b5b3970648f2d9dc13efd1d9df051 |       sha256: "935e23e1ff3bc02d390bad4d4be001208ee92cc217cb5b5a6c19bc14aaa318c1" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "7.2.2" |     version: "7.3.0" | ||||||
|   matcher: |   matcher: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: matcher |       name: matcher | ||||||
|       sha256: d2323aa2060500f906aa31a895b4030b6da3ebdcc5619d14ce1aada65cd161cb |       sha256: dc58c723c3c24bf8d3e2d3ad3f2f9d7bd9cf43ec6feaa64181775e60190153f2 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.12.16+1" |     version: "0.12.17" | ||||||
|   material_color_utilities: |   material_color_utilities: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -620,10 +620,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: meta |       name: meta | ||||||
|       sha256: bdb68674043280c3428e9ec998512fb681678676b3c54e773629ffe74419f8c7 |       sha256: e3641ec5d63ebf0d9b41bd43201a66e3fc79a65db5f61fc181f04cd27aab950c | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.15.0" |     version: "1.16.0" | ||||||
|   mime: |   mime: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -652,10 +652,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path |       name: path | ||||||
|       sha256: "087ce49c3f0dc39180befefc60fdb4acd8f8620e5682fe2476afd0b3688bb4af" |       sha256: "75cca69d1490965be98c73ceaea117e8a04dd21217b37b292c9ddbec0d955bc5" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.9.0" |     version: "1.9.1" | ||||||
|   path_provider: |   path_provider: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
| @@ -668,10 +668,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: path_provider_android |       name: path_provider_android | ||||||
|       sha256: "4adf4fd5423ec60a29506c76581bc05854c55e3a0b72d35bb28d661c9686edf2" |       sha256: d0d310befe2c8ab9e7f393288ccbb11b60c019c6b5afc21973eeee4dda2b35e9 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.2.15" |     version: "2.2.17" | ||||||
|   path_provider_foundation: |   path_provider_foundation: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -708,26 +708,26 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: permission_handler |       name: permission_handler | ||||||
|       sha256: "18bf33f7fefbd812f37e72091a15575e72d5318854877e0e4035a24ac1113ecb" |       sha256: "2d070d8684b68efb580a5997eb62f675e8a885ef0be6e754fb9ef489c177470f" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "11.3.1" |     version: "12.0.0+1" | ||||||
|   permission_handler_android: |   permission_handler_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: permission_handler_android |       name: permission_handler_android | ||||||
|       sha256: "71bbecfee799e65aff7c744761a57e817e73b738fedf62ab7afd5593da21f9f1" |       sha256: "1e3bc410ca1bf84662104b100eb126e066cb55791b7451307f9708d4007350e6" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "12.0.13" |     version: "13.0.1" | ||||||
|   permission_handler_apple: |   permission_handler_apple: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: permission_handler_apple |       name: permission_handler_apple | ||||||
|       sha256: e6f6d73b12438ef13e648c4ae56bd106ec60d17e90a59c4545db6781229082a0 |       sha256: f000131e755c54cf4d84a5d8bd6e4149e262cc31c5a8b1d698de1ac85fa41023 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "9.4.5" |     version: "9.4.7" | ||||||
|   permission_handler_html: |   permission_handler_html: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -740,10 +740,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: permission_handler_platform_interface |       name: permission_handler_platform_interface | ||||||
|       sha256: e9c8eadee926c4532d0305dff94b85bf961f16759c3af791486613152af4b4f9 |       sha256: eb99b295153abce5d683cac8c02e22faab63e50679b937fa1bf67d58bb282878 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "4.2.3" |     version: "4.3.0" | ||||||
|   permission_handler_windows: |   permission_handler_windows: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -808,54 +808,46 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.10.2+1" |     version: "0.10.2+1" | ||||||
|   posix: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: posix |  | ||||||
|       sha256: a0117dc2167805aa9125b82eee515cc891819bac2f538c83646d355b16f58b9a |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "6.0.1" |  | ||||||
|   provider: |   provider: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: provider |       name: provider | ||||||
|       sha256: c8a055ee5ce3fd98d6fc872478b03823ffdb448699c6ebdbbc71d59b596fd48c |       sha256: "4abbd070a04e9ddc287673bf5a030c7ca8b685ff70218720abab8b092f53dd84" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.1.2" |     version: "6.1.5" | ||||||
|   share_plus: |   share_plus: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: share_plus |       name: share_plus | ||||||
|       sha256: "6327c3f233729374d0abaafd61f6846115b2a481b4feddd8534211dc10659400" |       sha256: b2961506569e28948d75ec346c28775bb111986bb69dc6a20754a457e3d97fa0 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "10.1.3" |     version: "11.0.0" | ||||||
|   share_plus_platform_interface: |   share_plus_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: share_plus_platform_interface |       name: share_plus_platform_interface | ||||||
|       sha256: cc012a23fc2d479854e6c80150696c4a5f5bb62cb89af4de1c505cf78d0a5d0b |       sha256: "1032d392bc5d2095a77447a805aa3f804d2ae6a4d5eef5e6ebb3bd94c1bc19ef" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "5.0.2" |     version: "6.0.0" | ||||||
|   shared_preferences: |   shared_preferences: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences |       name: shared_preferences | ||||||
|       sha256: a752ce92ea7540fc35a0d19722816e04d0e72828a4200e83a98cf1a1eb524c9a |       sha256: "6e8bf70b7fef813df4e9a36f658ac46d107db4b4cfe1048b477d4e453a8159f5" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.3.5" |     version: "2.5.3" | ||||||
|   shared_preferences_android: |   shared_preferences_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_android |       name: shared_preferences_android | ||||||
|       sha256: bf808be89fe9dc467475e982c1db6c2faf3d2acf54d526cd5ec37d86c99dbd84 |       sha256: "20cbd561f743a342c76c151d6ddb93a9ce6005751e7aa458baad3858bfbfb6ac" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.4.1" |     version: "2.4.10" | ||||||
|   shared_preferences_foundation: |   shared_preferences_foundation: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -884,10 +876,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: shared_preferences_web |       name: shared_preferences_web | ||||||
|       sha256: d2ca4132d3946fec2184261726b355836a82c33d7d5b67af32692aff18a4684e |       sha256: c49bd060261c9a3f0ff445892695d6212ff603ef3115edbb448509d407600019 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.4.2" |     version: "2.4.3" | ||||||
|   shared_preferences_windows: |   shared_preferences_windows: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -901,7 +893,7 @@ packages: | |||||||
|     description: |     description: | ||||||
|       path: "." |       path: "." | ||||||
|       ref: master |       ref: master | ||||||
|       resolved-ref: "8784c39b909324df8913dd30fa416b8a50d55f49" |       resolved-ref: "89cdb5434a7ac7510f6bcdb60e1d51a27ee2f40b" | ||||||
|       url: "https://github.com/AlexBacich/shared-storage" |       url: "https://github.com/AlexBacich/shared-storage" | ||||||
|     source: git |     source: git | ||||||
|     version: "0.7.0" |     version: "0.7.0" | ||||||
| @@ -910,8 +902,8 @@ packages: | |||||||
|     description: |     description: | ||||||
|       path: "." |       path: "." | ||||||
|       ref: master |       ref: master | ||||||
|       resolved-ref: "25acc02612c2e0fcae40d312e047ac48106f8f6b" |       resolved-ref: c4349ceb03ae7293987cc0290ef06761e62c082e | ||||||
|       url: "https://github.com/re7gog/shizuku_apk_installer" |       url: "https://github.com/wilver06w/shizuku_apk_installer" | ||||||
|     source: git |     source: git | ||||||
|     version: "0.0.1" |     version: "0.0.1" | ||||||
|   sky_engine: |   sky_engine: | ||||||
| @@ -923,10 +915,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: source_span |       name: source_span | ||||||
|       sha256: "53e943d4206a5e30df338fd4c6e7a077e02254531b138a15aec3bd143c1a8b3c" |       sha256: "254ee5351d6cb365c859e20ee823c3bb479bf4a293c22d17a9f1bf144ce86f7c" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.10.0" |     version: "1.10.1" | ||||||
|   sprintf: |   sprintf: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -939,34 +931,34 @@ packages: | |||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: sqflite |       name: sqflite | ||||||
|       sha256: "2d7299468485dca85efeeadf5d38986909c5eb0cd71fd3db2c2f000e6c9454bb" |       sha256: e2297b1da52f127bc7a3da11439985d9b536f75070f3325e62ada69a5c585d03 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.4.1" |     version: "2.4.2" | ||||||
|   sqflite_android: |   sqflite_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: sqflite_android |       name: sqflite_android | ||||||
|       sha256: "78f489aab276260cdd26676d2169446c7ecd3484bbd5fead4ca14f3ed4dd9ee3" |       sha256: "2b3070c5fa881839f8b402ee4a39c1b4d561704d4ebbbcfb808a119bc2a1701b" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.4.0" |     version: "2.4.1" | ||||||
|   sqflite_common: |   sqflite_common: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: sqflite_common |       name: sqflite_common | ||||||
|       sha256: "761b9740ecbd4d3e66b8916d784e581861fd3c3553eda85e167bc49fdb68f709" |       sha256: "84731e8bfd8303a3389903e01fb2141b6e59b5973cacbb0929021df08dddbe8b" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.5.4+6" |     version: "2.5.5" | ||||||
|   sqflite_darwin: |   sqflite_darwin: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: sqflite_darwin |       name: sqflite_darwin | ||||||
|       sha256: "22adfd9a2c7d634041e96d6241e6e1c8138ca6817018afc5d443fef91dcefa9c" |       sha256: "279832e5cde3fe99e8571879498c9211f3ca6391b0d818df4e17d9fff5c6ccb3" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.4.1+1" |     version: "2.4.2" | ||||||
|   sqflite_platform_interface: |   sqflite_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -979,58 +971,58 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: stack_trace |       name: stack_trace | ||||||
|       sha256: "9f47fd3630d76be3ab26f0ee06d213679aa425996925ff3feffdec504931c377" |       sha256: "8b27215b45d22309b5cddda1aa2b19bdfec9df0e765f2de506401c071d38d1b1" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.12.0" |     version: "1.12.1" | ||||||
|   stream_channel: |   stream_channel: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: stream_channel |       name: stream_channel | ||||||
|       sha256: ba2aa5d8cc609d96bbb2899c28934f9e1af5cddbd60a827822ea467161eb54e7 |       sha256: "969e04c80b8bcdf826f8f16579c7b14d780458bd97f56d107d3950fdbeef059d" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.1.2" |     version: "2.1.4" | ||||||
|   string_scanner: |   string_scanner: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: string_scanner |       name: string_scanner | ||||||
|       sha256: "688af5ed3402a4bde5b3a6c15fd768dbf2621a614950b17f04626c431ab3c4c3" |       sha256: "921cd31725b72fe181906c6a94d987c78e3b98c2e205b397ea399d4054872b43" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.3.0" |     version: "1.4.1" | ||||||
|   synchronized: |   synchronized: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: synchronized |       name: synchronized | ||||||
|       sha256: "69fe30f3a8b04a0be0c15ae6490fc859a78ef4c43ae2dd5e8a623d45bfcf9225" |       sha256: "0669c70faae6270521ee4f05bffd2919892d42d1276e6c495be80174b6bc0ef6" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.3.0+3" |     version: "3.3.1" | ||||||
|   term_glyph: |   term_glyph: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: term_glyph |       name: term_glyph | ||||||
|       sha256: a29248a84fbb7c79282b40b8c72a1209db169a2e0542bce341da992fe1bc7e84 |       sha256: "7f554798625ea768a7518313e58f83891c7f5024f88e46e7182a4558850a4b8e" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.2.1" |     version: "1.2.2" | ||||||
|   test_api: |   test_api: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: test_api |       name: test_api | ||||||
|       sha256: "664d3a9a64782fcdeb83ce9c6b39e78fd2971d4e37827b9b06c3aa1edc5e760c" |       sha256: fb31f383e2ee25fbbfe06b40fe21e1e458d14080e3c67e7ba0acfde4df4e0bbd | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.7.3" |     version: "0.7.4" | ||||||
|   timezone: |   timezone: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: timezone |       name: timezone | ||||||
|       sha256: ffc9d5f4d1193534ef051f9254063fa53d588609418c84299956c3db9383587d |       sha256: dd14a3b83cfd7cb19e7888f1cbc20f258b8d71b54c06f79ac585f14093a287d1 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "0.10.0" |     version: "0.10.1" | ||||||
|   typed_data: |   typed_data: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -1059,18 +1051,18 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_android |       name: url_launcher_android | ||||||
|       sha256: "6fc2f56536ee873eeb867ad176ae15f304ccccc357848b351f6f0d8d4a40d193" |       sha256: "8582d7f6fe14d2652b4c45c9b6c14c0b678c2af2d083a11b604caeba51930d79" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.3.14" |     version: "6.3.16" | ||||||
|   url_launcher_ios: |   url_launcher_ios: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_ios |       name: url_launcher_ios | ||||||
|       sha256: "16a513b6c12bb419304e72ea0ae2ab4fed569920d1c7cb850263fe3acc824626" |       sha256: "7f2022359d4c099eea7df3fdf739f7d3d3b9faf3166fb1dd390775176e0b76cb" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.3.2" |     version: "6.3.3" | ||||||
|   url_launcher_linux: |   url_launcher_linux: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -1099,10 +1091,10 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: url_launcher_web |       name: url_launcher_web | ||||||
|       sha256: "3ba963161bd0fe395917ba881d320b9c4f6dd3c4a233da62ab18a5025c85f1e9" |       sha256: "4bd2b7b4dc4d4d0b94e5babfffbca8eac1a126c7f3d6ecbc1a11013faa3abba2" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.4.0" |     version: "2.4.1" | ||||||
|   url_launcher_windows: |   url_launcher_windows: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -1131,66 +1123,66 @@ packages: | |||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: vm_service |       name: vm_service | ||||||
|       sha256: f6be3ed8bd01289b34d679c2b62226f63c0e69f9fd2e50a6b3c1c729a961041b |       sha256: ddfa8d30d89985b96407efce8acbdd124701f96741f2d981ca860662f1c0dc02 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "14.3.0" |     version: "15.0.0" | ||||||
|   web: |   web: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: web |       name: web | ||||||
|       sha256: cd3543bd5798f6ad290ea73d210f423502e71900302dde696f8bff84bf89a1cb |       sha256: "868d88a33d8a87b18ffc05f9f030ba328ffefba92d6c127917a2ba740f9cfe4a" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.1.0" |     version: "1.1.1" | ||||||
|   webview_flutter: |   webview_flutter: | ||||||
|     dependency: "direct main" |     dependency: "direct main" | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter |       name: webview_flutter | ||||||
|       sha256: "889a0a678e7c793c308c68739996227c9661590605e70b1f6cf6b9a6634f7aec" |       sha256: c3e4fe614b1c814950ad07186007eff2f2e5dd2935eba7b9a9a1af8e5885f1ba | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "4.10.0" |     version: "4.13.0" | ||||||
|   webview_flutter_android: |   webview_flutter_android: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_android |       name: webview_flutter_android | ||||||
|       sha256: "3d535126f7244871542b2f0b0fcf94629c9a14883250461f9abe1a6644c1c379" |       sha256: f6e6afef6e234801da77170f7a1847ded8450778caf2fe13979d140484be3678 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "4.2.0" |     version: "4.7.0" | ||||||
|   webview_flutter_platform_interface: |   webview_flutter_platform_interface: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_platform_interface |       name: webview_flutter_platform_interface | ||||||
|       sha256: d937581d6e558908d7ae3dc1989c4f87b786891ab47bb9df7de548a151779d8d |       sha256: "7cb32b21825bd65569665c32bb00a34ded5779786d6201f5350979d2d529940d" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "2.10.0" |     version: "2.13.0" | ||||||
|   webview_flutter_wkwebview: |   webview_flutter_wkwebview: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: webview_flutter_wkwebview |       name: webview_flutter_wkwebview | ||||||
|       sha256: "4adc14ea9a770cc9e2c8f1ac734536bd40e82615bd0fa6b94be10982de656cc7" |       sha256: a3d461fe3467014e05f3ac4962e5fdde2a4bf44c561cb53e9ae5c586600fdbc3 | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "3.17.0" |     version: "3.22.0" | ||||||
|   win32: |   win32: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: win32 |       name: win32 | ||||||
|       sha256: "154360849a56b7b67331c21f09a386562d88903f90a1099c5987afc1912e1f29" |       sha256: "329edf97fdd893e0f1e3b9e88d6a0e627128cc17cc316a8d67fda8f1451178ba" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "5.10.0" |     version: "5.13.0" | ||||||
|   win32_registry: |   win32_registry: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
|       name: win32_registry |       name: win32_registry | ||||||
|       sha256: "21ec76dfc731550fd3e2ce7a33a9ea90b828fdf19a5c3bcf556fa992cfa99852" |       sha256: "6f1b564492d0147b330dd794fee8f512cec4977957f310f9951b5f9d83618dae" | ||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "1.1.5" |     version: "2.1.0" | ||||||
|   xdg_directories: |   xdg_directories: | ||||||
|     dependency: transitive |     dependency: transitive | ||||||
|     description: |     description: | ||||||
| @@ -1207,14 +1199,6 @@ packages: | |||||||
|       url: "https://pub.dev" |       url: "https://pub.dev" | ||||||
|     source: hosted |     source: hosted | ||||||
|     version: "6.3.0" |     version: "6.3.0" | ||||||
|   yaml: |  | ||||||
|     dependency: transitive |  | ||||||
|     description: |  | ||||||
|       name: yaml |  | ||||||
|       sha256: b9da305ac7c39faa3f030eccd175340f968459dae4af175130b3fc47e40d76ce |  | ||||||
|       url: "https://pub.dev" |  | ||||||
|     source: hosted |  | ||||||
|     version: "3.1.3" |  | ||||||
| sdks: | sdks: | ||||||
|   dart: ">=3.6.0 <4.0.0" |   dart: ">=3.8.1 <4.0.0" | ||||||
|   flutter: ">=3.27.0" |   flutter: ">=3.27.0" | ||||||
|   | |||||||