Compare commits
	
		
			142 Commits
		
	
	
		
			8b2be5c55c
			...
			v122113131
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | a4527940b8 | ||
|  | 9e8a25ed3e | ||
|  | 8ea46e146b | ||
|  | 5ecf3c3f87 | ||
|  | 325f103417 | ||
|  | ab4b1ae644 | ||
|  | 87ea44754e | ||
|  | 04dec50808 | ||
|  | e36189e2e7 | ||
|  | d6bdf510a4 | ||
|  | a464e93370 | ||
| 4b63afe62a | |||
| ac4c4b9441 | |||
|  | 16b10dc1b7 | ||
| 02d734eee8 | |||
| c5cdfc0d53 | |||
| 6d610ed61a | |||
| 792950be7c | |||
|  | af8969ce4a | ||
|  | 27c55e59a1 | ||
|  | 94a0747947 | ||
|  | d862bfba4f | ||
|  | b0d1d9c29a | ||
|  | 7b40a31979 | ||
|  | 823a8c3692 | ||
|  | 5494978db8 | ||
|  | 6076eb1cee | ||
|  | 131101d2ee | ||
|  | 62ad1f45ba | ||
|  | 402d18b889 | ||
|  | e32699c93f | ||
|  | 059a237b99 | ||
|  | d2bdbae6c8 | ||
|  | 510fcbe47e | ||
| 667e9c1a5d | |||
| 53b1d1f8b2 | |||
| c25e8889a4 | |||
|  | 8b0bbe71c9 | ||
| 8bfe14c019 | |||
| 208babbce3 | |||
| 02098a7aa9 | |||
| d0a982f385 | |||
| 1d1c121aab | |||
| fe12819163 | |||
|  | 023a30c008 | ||
|  | a2862a2587 | ||
|  | 054e936657 | ||
| 1d2e5069b8 | |||
| a147646743 | |||
| 32e7fc0038 | |||
| c15bf44032 | |||
| 0bcd55bd4e | |||
| ebef0b3511 | |||
| 713ceb05bf | |||
| dc8381b661 | |||
| b5b820c64b | |||
| f7055626d9 | |||
|  | 6ec3e96909 | ||
| 22da30eaa8 | |||
| 79fd115f5e | |||
| 8dc3d319cd | |||
| 27bb056397 | |||
| f9ba13dc32 | |||
| 6f60ef4346 | |||
| 28b950f467 | |||
| a9c7ec3dc1 | |||
| 920d4ac1ef | |||
| 0e96d313ec | |||
| 7211fdb1a3 | |||
|  | 381d6acc82 | ||
| d311c2cdeb | |||
| 219cae5d74 | |||
| 2968aee309 | |||
| 6cb4b35c93 | |||
| 15ec0f2d26 | |||
| 4781e30da2 | |||
| c8759cc035 | |||
| cb4f2f02ef | |||
| 7517626ab7 | |||
| 41c951b659 | |||
|  | ad279c6683 | ||
|  | 5f0817ddb7 | ||
|  | 7124cbcacd | ||
|  | 2a710a1a08 | ||
|  | 82ec2445a1 | ||
|  | cabb6d494d | ||
|  | 5c12481813 | ||
|  | b16f86dda1 | ||
|  | 2bc28db2cc | ||
|  | bf1b680b4a | ||
| e2afff0b8e | |||
| a382fc89ea | |||
| 3f0a3903ae | |||
| f46f98cef0 | |||
| bf6f1a917e | |||
| 71c0a4d340 | |||
| 63c550ead3 | |||
|  | d81fb79b4f | ||
|  | 144067d5b6 | ||
|  | 8106faa45c | ||
|  | d9ef301e0f | ||
|  | 90b52232ab | ||
|  | 0f000ea359 | ||
|  | fb8f81a4c8 | ||
|  | a76b3dd2a9 | ||
|  | 91aed5a777 | ||
| 366b2e10f1 | |||
|  | 8823cc6c6c | ||
| d2436bb976 | |||
|  | 74ef4da15b | ||
|  | bd96c67788 | ||
|  | da71de6806 | ||
|  | 0264da8ccc | ||
|  | 270d959ee0 | ||
|  | 6d11dfb80c | ||
|  | 4184bbb900 | ||
| ef994460c1 | |||
| 758708e18d | |||
| c0381144d1 | |||
| cda3ba6cb4 | |||
| a4636cc0c8 | |||
|  | 4c12c9d570 | ||
|  | f4db02521d | ||
|  | 01763556b1 | ||
| 60c24fc75a | |||
| 5853a19937 | |||
| 99f2c04bf6 | |||
|  | e2411c00d8 | ||
|  | 0473a5f7bc | ||
|  | d0d6a4378c | ||
|  | 1dfa3c9f07 | ||
|  | 815f00e764 | ||
| bdc77ab8ef | |||
| 6bd06cb458 | |||
|  | e9e8bee6c9 | ||
|  | ff6038dbd4 | ||
|  | 8146cff011 | ||
|  | fc4c48dd12 | ||
|  | 94f1ec943c | ||
|  | 0278540fb2 | ||
|  | 109050d0cf | ||
|  | 0392e1b406 | 
							
								
								
									
										115
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										115
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -1,21 +1,124 @@ | ||||
| kind: pipeline | ||||
| type: docker | ||||
| name: android | ||||
| name: test | ||||
|  | ||||
| steps: | ||||
|   - name: code-analysis | ||||
|   - name: AnylyseBuildTest | ||||
|     image: mingc/android-build-box:latest | ||||
|     failure: ignore | ||||
|     commands: | ||||
|       - ls -la | ||||
|       - ./gradlew sonarqube -Dsonar.projectKey=RFS2 -Dsonar.host.url=$SONAR_HOST_URL -Dsonar.login=$SONAR_LOGIN -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Configure gradle..." | ||||
|       - mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=true\nappLoginUrl=\"URL\"\nappLoginUsername=\"LOGIN\"\nappLoginPassword=\"PASS\"\npushCache=false\nsystemProp.org.gradle.internal.http.connectionTimeout=180000\nsystemProp.org.gradle.internal.http.socketTimeout=180000" >> ~/.gradle/gradle.properties | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Analysing..." | ||||
|       - ./gradlew sonarqube -Dsonar.projectKey=RFS2 -Dsonar.host.url=$SONAR_HOST_URL -Dsonar.login=$SONAR_LOGIN | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Building..." | ||||
|       - ./gradlew build | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Testing..." | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - ./gradlew test | ||||
|     environment: | ||||
|       SONAR_HOST_URL: | ||||
|         from_secret: sonarScannerHostUrl | ||||
|       SONAR_LOGIN: | ||||
|         from_secret: sonarScannerLogin | ||||
| trigger: | ||||
|   event: | ||||
|     - push | ||||
|     - pull_request | ||||
|  | ||||
| --- | ||||
| kind: pipeline | ||||
| type: docker | ||||
| name: Publish | ||||
|  | ||||
| steps: | ||||
|   - name: createTag | ||||
|     image: ubuntu:latest | ||||
|     commands: | ||||
|       - apt-get update && apt-get install -y git | ||||
|       - ./build.sh --publish --from-ci | ||||
|       - git remote add pushing https://$GITEA_USR:$GITEA_PASS@gitea.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform.git | ||||
|       - git push pushing --tags | ||||
|     environment: | ||||
|       GITEA_USR: | ||||
|         from_secret: giteaUsr | ||||
|       GITEA_PASS: | ||||
|         from_secret: giteaPass | ||||
|  | ||||
|   - name: scpFiles | ||||
|     image: appleboy/drone-scp | ||||
|     settings: | ||||
|       host: amine-louveau.fr | ||||
|       username: ubuntu | ||||
|       key: | ||||
|         from_secret: privateKey | ||||
|       port: 22 | ||||
|       target: /home/ubuntu/ | ||||
|       source: version.txt | ||||
|  | ||||
|   - name: deploy | ||||
|     image: appleboy/drone-ssh | ||||
|     settings: | ||||
|       host: amine-louveau.fr | ||||
|       user: ubuntu | ||||
|       key: | ||||
|         from_secret: privateKey | ||||
|       command_timeout: 2m | ||||
|       script: | ||||
|         - cd /home/ubuntu | ||||
|         - sudo rm -rf /var/www/amine/version.txt | ||||
|         - sudo chown www-data:www-data ./version.txt | ||||
|         - sudo mv version.txt /var/www/amine/ | ||||
|  | ||||
| trigger: | ||||
|   event: | ||||
|     - promote | ||||
|   target: | ||||
|     - production | ||||
|  | ||||
| --- | ||||
| kind: pipeline | ||||
| type: docker | ||||
| name: Release | ||||
|  | ||||
| steps: | ||||
|   - name: build | ||||
|     image: mingc/android-build-box:latest | ||||
|     commands: | ||||
|       - ./gradlew  :androidApp:build -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" -P pushCache=false | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Configure gradle..." | ||||
|       - mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=true\nappLoginUrl=\"URL\"\nappLoginUsername=\"LOGIN\"\nappLoginPassword=\"PASS\"\npushCache=false\nsystemProp.org.gradle.internal.http.connectionTimeout=180000\nsystemProp.org.gradle.internal.http.socketTimeout=180000" >> ~/.gradle/gradle.properties | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Generate APK" | ||||
|       - ./gradlew :androidApp:assembleGithubConfigRelease  -P pushCache=false | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Get Key" | ||||
|       - wget https://amine-louveau.fr/key | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Zipalign" | ||||
|       - $ANDROID_HOME/build-tools/31.0.0/zipalign -f -v 4 androidApp/build/outputs/apk/githubConfig/release/androidApp-githubConfig-release-unsigned.apk androidApp/build/outputs/apk/githubConfig/release/android-prod-released-ziped.apk | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Sign" | ||||
|       - $ANDROID_HOME/build-tools/31.0.0/apksigner sign -v --out signed.apk --ks ./key --ks-key-alias $YOUR_KEY_ALIAS --ks-pass pass:$YOUR_KEYSTORE_PASSWORD --v1-signing-enabled true --v2-signing-enabled true androidApp/build/outputs/apk/githubConfig/release/android-prod-released-ziped.apk | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Verify" | ||||
|       - $ANDROID_HOME/build-tools/31.0.0/apksigner verify signed.apk | ||||
|     environment: | ||||
|       YOUR_KEYSTORE_PASSWORD: | ||||
|         from_secret: keyPass | ||||
|       YOUR_KEY_ALIAS: | ||||
|         from_secret: keyAlias | ||||
|  | ||||
|   - name: gitea_release | ||||
|     image: plugins/gitea-release | ||||
|     settings: | ||||
|       api_key: | ||||
|         from_secret: giteaAPI | ||||
|       base_url: https://gitea.amine-louveau.fr | ||||
|       files: signed.apk | ||||
| trigger: | ||||
|   event: | ||||
|     - tag | ||||
							
								
								
									
										11
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,3 +1,14 @@ | ||||
| # V2/Multiplatform rewrite | ||||
|  | ||||
| **v1** | ||||
|  | ||||
| - The app has the same functionalities as before. | ||||
|  | ||||
|  | ||||
| -------------------------------------------------------------------- | ||||
|                                                                        | ||||
| # Old version changes | ||||
|  | ||||
| **1.7.x** | ||||
|  | ||||
| - Hiding tags with 0 articles | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| # ReaderForSelfoss-multiplatform | ||||
| # ReaderForSelfoss-multiplatform [](https://build.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform) | ||||
|  | ||||
| [](https://crowdin.com/project/readerforselfoss) | ||||
|  | ||||
|   | ||||
| @@ -6,6 +6,7 @@ plugins { | ||||
|     id("com.android.application") | ||||
|     kotlin("android") | ||||
|     kotlin("kapt") | ||||
|     id("com.mikepenz.aboutlibraries.plugin") | ||||
| } | ||||
|  | ||||
| fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | ||||
| @@ -30,7 +31,7 @@ fun gitVersion(): String { | ||||
|         println("Tag found on current commit") | ||||
|         execWithOutput("git -C ../ describe --contains HEAD") | ||||
|     } | ||||
|     return process.replace("'", "").substring(1).replace("\\.", "").trim() | ||||
|     return process.replace("^0", "").replace("'", "").substring(1).replace("\\.", "").trim() | ||||
| } | ||||
|  | ||||
| fun versionCodeFromGit(): Int { | ||||
| @@ -54,11 +55,15 @@ fun versionNameFromGit(): String { | ||||
| android { | ||||
|     compileOptions { | ||||
|         // Flag to enable support for the new language APIs | ||||
|         isCoreLibraryDesugaringEnabled = true | ||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
|         targetCompatibility = JavaVersion.VERSION_1_8 | ||||
|         sourceCompatibility = JavaVersion.VERSION_11 | ||||
|         targetCompatibility = JavaVersion.VERSION_11 | ||||
|     } | ||||
|     compileSdk = 31 | ||||
|  | ||||
|     // For Kotlin projects | ||||
|     kotlinOptions { | ||||
|         jvmTarget = "11" | ||||
|     } | ||||
|     compileSdk = 33 | ||||
|     buildToolsVersion = "31.0.0" | ||||
|     buildFeatures { | ||||
|         viewBinding = true | ||||
| @@ -66,7 +71,7 @@ android { | ||||
|     defaultConfig { | ||||
|         applicationId = "bou.amine.apps.readerforselfossv2.android" | ||||
|         minSdk = 21 | ||||
|         targetSdk = 31 | ||||
|         targetSdk = 33 | ||||
|         versionCode = versionCodeFromGit() | ||||
|         versionName = versionNameFromGit() | ||||
|  | ||||
| @@ -79,6 +84,11 @@ android { | ||||
|         // tests | ||||
|         testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" | ||||
|     } | ||||
|     packagingOptions { | ||||
|         resources { | ||||
|             excludes += "/META-INF/{AL2.0,LGPL2.1}" | ||||
|         } | ||||
|     } | ||||
|     buildTypes { | ||||
|         getByName("release") { | ||||
|             isMinifyEnabled = true | ||||
| @@ -93,14 +103,12 @@ android { | ||||
|     } | ||||
|     flavorDimensions.add("build") | ||||
|     productFlavors { | ||||
|         create("github") { | ||||
|         create("githubConfig") { | ||||
|             versionNameSuffix = "-github" | ||||
|             dimension = "build" | ||||
|         } | ||||
|     } | ||||
|     kotlinOptions { | ||||
|         jvmTarget = "1.8" | ||||
|     } | ||||
|     namespace = "bou.amine.apps.readerforselfossv2.android" | ||||
|  | ||||
| } | ||||
|  | ||||
| @@ -113,13 +121,6 @@ dependencies { | ||||
|  | ||||
|     implementation("androidx.preference:preference-ktx:1.1.1") | ||||
|  | ||||
|     // Testing | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-core:3.4.0-alpha02") | ||||
|     androidTestImplementation("androidx.test:runner:1.3.1-alpha02") | ||||
|     // Espresso-contrib for DatePicker, RecyclerView, Drawer actions, Accessibility checks, CountingIdlingResource | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-contrib:3.4.0-alpha02") | ||||
|     // Espresso-intents for validation and stubbing of Intents | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-intents:3.4.0-alpha02") | ||||
|     implementation(fileTree(mapOf("include" to listOf("*.jar"), "dir" to "libs"))) | ||||
|  | ||||
|     // Android Support | ||||
| @@ -128,31 +129,18 @@ dependencies { | ||||
|     implementation("androidx.recyclerview:recyclerview:1.3.0-alpha01") | ||||
|     implementation("androidx.legacy:legacy-support-v4:1.0.0") | ||||
|     implementation("androidx.vectordrawable:vectordrawable:1.2.0-alpha02") | ||||
|     implementation("androidx.browser:browser:1.4.0") | ||||
|     implementation("androidx.cardview:cardview:1.0.0") | ||||
|     implementation("androidx.annotation:annotation:1.3.0") | ||||
|     implementation("androidx.work:work-runtime-ktx:2.7.1") | ||||
|     implementation("androidx.constraintlayout:constraintlayout:2.1.3") | ||||
|     implementation("org.jsoup:jsoup:1.14.3") | ||||
|  | ||||
|     coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:1.1.5") | ||||
|  | ||||
|     //multidex | ||||
|     implementation("androidx.multidex:multidex:2.0.1") | ||||
|  | ||||
|     // About | ||||
|     implementation("com.mikepenz:aboutlibraries-core:8.9.4") | ||||
|     implementation("com.mikepenz:aboutlibraries:8.9.4") | ||||
|     implementation("com.mikepenz:aboutlibraries-definitions:8.9.4") | ||||
|  | ||||
|     // Async | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.6.0") | ||||
|  | ||||
|     // Retrofit + http logging + okhttp | ||||
|     implementation("com.squareup.retrofit2:retrofit:2.9.0") | ||||
|     implementation("com.squareup.okhttp3:logging-interceptor:5.0.0-alpha.3") | ||||
|     implementation("com.squareup.retrofit2:converter-gson:2.9.0") | ||||
|     implementation("com.burgstaller:okhttp-digest:2.5") | ||||
|     implementation("com.mikepenz:aboutlibraries-core:10.5.1") | ||||
|     implementation("com.mikepenz:aboutlibraries:10.5.1") | ||||
|  | ||||
|     // Material-ish things | ||||
|     implementation("com.ashokvarma.android:bottom-navigation-bar:2.2.0") | ||||
| @@ -166,8 +154,6 @@ dependencies { | ||||
|     implementation("com.mikepenz:materialdrawer:8.4.5") | ||||
|  | ||||
|     // Themes | ||||
|     implementation("com.52inc:scoops:1.0.0") | ||||
|     implementation("com.jaredrummler:colorpicker:1.1.0") | ||||
|     implementation("com.github.rubensousa:floatingtoolbar:1.5.1") | ||||
|  | ||||
|     // Pager | ||||
| @@ -190,14 +176,41 @@ dependencies { | ||||
|  | ||||
|     implementation("androidx.core:core-ktx:1.8.0") | ||||
|  | ||||
|     // implementation("androidx.lifecycle:lifecycle-livedata-ktx:2.5.1") | ||||
|     // implementation("androidx.lifecycle:lifecycle-common-java8:2.5.1") | ||||
|     // implementation("androidx.lifecycle:lifecycle-runtime:2.5.1") | ||||
|     implementation("androidx.lifecycle:lifecycle-extensions:2.2.0") | ||||
|  | ||||
|     // Network information | ||||
|      implementation("com.github.ln-12:multiplatform-connectivity-status:1.3.0") | ||||
|  | ||||
|     // SQLDELIGHT | ||||
|     implementation("com.squareup.sqldelight:android-driver:1.5.3") | ||||
|     implementation("com.squareup.sqldelight:android-driver:1.5.4") | ||||
|  | ||||
|     //test | ||||
|     testImplementation("junit:junit:4.13.2") | ||||
|     testImplementation("io.mockk:mockk:1.12.0") | ||||
|     testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.0") | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") | ||||
| } | ||||
|  | ||||
| tasks.withType<Test> { | ||||
|     outputs.upToDateWhen { false } | ||||
|     useJUnit() | ||||
|     testLogging { | ||||
|         exceptionFormat = org.gradle.api.tasks.testing.logging.TestExceptionFormat.FULL | ||||
|         events = setOf( | ||||
|             org.gradle.api.tasks.testing.logging.TestLogEvent.PASSED, | ||||
|             org.gradle.api.tasks.testing.logging.TestLogEvent.FAILED, | ||||
|             org.gradle.api.tasks.testing.logging.TestLogEvent.STANDARD_ERROR | ||||
|         ) | ||||
|         showStandardStreams = true | ||||
|     } | ||||
| } | ||||
|  | ||||
| aboutLibraries { | ||||
|     offlineMode = true | ||||
|     fetchRemoteLicense = false | ||||
|     fetchRemoteFunding = false | ||||
|     includePlatform = false | ||||
|     strictMode = com.mikepenz.aboutlibraries.plugin.StrictMode.FAIL | ||||
|     duplicationMode = com.mikepenz.aboutlibraries.plugin.DuplicateMode.MERGE | ||||
|     duplicationRule = com.mikepenz.aboutlibraries.plugin.DuplicateRule.GROUP | ||||
| } | ||||
							
								
								
									
										40
									
								
								androidApp/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										40
									
								
								androidApp/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							| @@ -30,15 +30,8 @@ | ||||
|     <fields>; | ||||
| } | ||||
|  | ||||
| -dontwarn okio.** | ||||
| -dontwarn retrofit2.Platform$Java8 | ||||
| -keep class retrofit.** { *; } | ||||
| -keepclasseswithmembers class * { | ||||
|     @retrofit.http.* <methods>; | ||||
| } | ||||
| -keepattributes *Annotation*,Signature | ||||
| -keepattributes Exceptions | ||||
| -dontwarn okio.** | ||||
| -dontwarn javax.annotation.Nullable | ||||
| -dontwarn javax.annotation.ParametersAreNonnullByDefault | ||||
|  | ||||
| @@ -55,11 +48,40 @@ | ||||
|   public *; | ||||
| } | ||||
|  | ||||
| -dontwarn com.anupcowkur.reservoir.** | ||||
|  | ||||
| -dontwarn javax.annotation.** | ||||
|  | ||||
| -keep class android.support.v7.widget.SearchView { *; } | ||||
|  | ||||
| # maybe remove later ? | ||||
| -keep class * extends androidx.fragment.app.Fragment | ||||
|  | ||||
|  | ||||
| # Keep `Companion` object fields of serializable classes. | ||||
| # This avoids serializer lookup through `getDeclaredClasses` as done for named companion objects. | ||||
| -if @kotlinx.serialization.Serializable class ** | ||||
| -keepclassmembers class <1> { | ||||
|    static <1>$Companion Companion; | ||||
| } | ||||
|  | ||||
| # Keep `serializer()` on companion objects (both default and named) of serializable classes. | ||||
| -if @kotlinx.serialization.Serializable class ** { | ||||
|    static **$* *; | ||||
| } | ||||
| -keepclassmembers class <2>$<3> { | ||||
|    kotlinx.serialization.KSerializer serializer(...); | ||||
| } | ||||
|  | ||||
| # Keep `INSTANCE.serializer()` of serializable objects. | ||||
| -if @kotlinx.serialization.Serializable class ** { | ||||
|    public static ** INSTANCE; | ||||
| } | ||||
| -keepclassmembers class <1> { | ||||
|    public static <1> INSTANCE; | ||||
|    kotlinx.serialization.KSerializer serializer(...); | ||||
| } | ||||
|  | ||||
| # @Serializable and @Polymorphic are used at runtime for polymorphic serialization. | ||||
| -keepattributes RuntimeVisibleAnnotations,AnnotationDefault | ||||
|  | ||||
| -dontwarn io.mockk.** | ||||
| -keep class io.mockk.** { *; } | ||||
|   | ||||
| @@ -1,11 +0,0 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| # NOTE: This is copy/pasted in jenkins | ||||
|  | ||||
| rm -f version.txt | ||||
| printf "versionName=$1-github\nversionCode=$1" >> version.txt | ||||
|  | ||||
| # You'll need to change server as your server and define a VERSION_PATH. | ||||
| scp version.txt server:$VERSION_PATH | ||||
|  | ||||
| rm version.txt | ||||
| @@ -1,18 +1,22 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     package="bou.amine.apps.readerforselfossv2.android"> | ||||
|     xmlns:tools="http://schemas.android.com/tools"> | ||||
|  | ||||
|     <uses-permission android:name="android.permission.INTERNET" /> | ||||
|     <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> | ||||
|  | ||||
|     <application | ||||
|         android:allowBackup="false" | ||||
|         android:fullBackupContent="false" | ||||
|         tools:replace="android:allowBackup" | ||||
|         android:name=".MyApp" | ||||
|         android:allowBackup="true" | ||||
|         android:icon="@mipmap/ic_launcher" | ||||
|         android:label="@string/app_name" | ||||
|         android:supportsRtl="true" | ||||
|         android:networkSecurityConfig="@xml/network_security_config" | ||||
|         android:theme="@style/NoBar"> | ||||
|         android:theme="@style/NoBar" | ||||
|         android:dataExtractionRules="@xml/data_extraction_rules" | ||||
|         android:configChanges="uiMode"> | ||||
|         <activity | ||||
|             android:name=".MainActivity" | ||||
|             android:theme="@style/SplashTheme" | ||||
|   | ||||
| @@ -7,13 +7,10 @@ import android.widget.* | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.constraintlayout.widget.ConstraintLayout | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityAddSourceBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.Toppings | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlValid | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlInvalid | ||||
| import bou.amine.apps.readerforselfossv2.model.NetworkUnavailableException | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.ftinc.scoop.Scoop | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| @@ -26,7 +23,6 @@ class AddSourceActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private var mSpoutsValue: String? = null | ||||
|  | ||||
|     private lateinit var appColors: AppColors | ||||
|     private lateinit var binding: ActivityAddSourceBinding | ||||
|  | ||||
|     override val di by closestDI() | ||||
| @@ -34,43 +30,18 @@ class AddSourceActivity : AppCompatActivity(), DIAware { | ||||
|     private val appSettingsService : AppSettingsService by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         appColors = AppColors(this@AddSourceActivity) | ||||
|  | ||||
|         super.onCreate(savedInstanceState) | ||||
|         binding = ActivityAddSourceBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
|         setContentView(view) | ||||
|  | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.bind(this, Toppings.PRIMARY.value, binding.toolbar) | ||||
|         scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||
|  | ||||
|         val drawable = binding.nameInput.background | ||||
|         drawable.setTint(appColors.colorAccent) | ||||
|  | ||||
|  | ||||
|         // TODO: clean | ||||
|         binding.nameInput.background = drawable | ||||
|  | ||||
|         val drawable1 = binding.sourceUri.background | ||||
|         drawable1.setTint(appColors.colorAccent) | ||||
|  | ||||
|         binding.sourceUri.background = drawable1 | ||||
|  | ||||
|         val drawable2 = binding.tags.background | ||||
|         drawable2.setTint(appColors.colorAccent) | ||||
|  | ||||
|         binding.tags.background = drawable2 | ||||
|  | ||||
|         setSupportActionBar(binding.toolbar) | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||
|  | ||||
|         maybeGetDetailsFromIntentSharing(intent, binding.sourceUri, binding.nameInput) | ||||
|  | ||||
|         binding.saveBtn.setTextColor(appColors.colorAccent) | ||||
|  | ||||
|         binding.saveBtn.setOnClickListener { | ||||
|             handleSaveSource( | ||||
|                 binding.tags, | ||||
| @@ -84,7 +55,7 @@ class AddSourceActivity : AppCompatActivity(), DIAware { | ||||
|         super.onResume() | ||||
|  | ||||
|         val baseUrl = appSettingsService.getBaseUrl() | ||||
|         if (baseUrl.isEmpty() || !baseUrl.isBaseUrlValid(this@AddSourceActivity)) { | ||||
|         if (baseUrl.isEmpty() || baseUrl.isBaseUrlInvalid()) { | ||||
|             mustLoginToAddSource() | ||||
|         } else { | ||||
|             handleSpoutsSpinner(binding.spoutsSpinner, binding.progress, binding.formContainer) | ||||
| @@ -123,7 +94,7 @@ class AddSourceActivity : AppCompatActivity(), DIAware { | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             try { | ||||
|                 val items = repository.getSpouts() | ||||
|                 if (items != null) { | ||||
|                 if (items.isNotEmpty()) { | ||||
|                     val itemsStrings = items.map { it.value.name } | ||||
|                     for ((key, value) in items) { | ||||
|                         spoutsKV[value.name] = key | ||||
|   | ||||
| @@ -11,12 +11,14 @@ import android.view.MenuItem | ||||
| import android.view.View | ||||
| import android.widget.ImageView | ||||
| import android.widget.Toast | ||||
| import androidx.activity.result.contract.ActivityResultContracts | ||||
| import androidx.appcompat.app.ActionBarDrawerToggle | ||||
| import androidx.appcompat.app.AlertDialog | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.appcompat.widget.SearchView | ||||
| import androidx.core.view.doOnNextLayout | ||||
| import androidx.drawerlayout.widget.DrawerLayout | ||||
| import androidx.lifecycle.lifecycleScope | ||||
| import androidx.recyclerview.widget.* | ||||
| import androidx.work.Constraints | ||||
| import androidx.work.ExistingPeriodicWorkPolicy | ||||
| @@ -28,11 +30,8 @@ import bou.amine.apps.readerforselfossv2.android.adapters.ItemsAdapter | ||||
| import bou.amine.apps.readerforselfossv2.android.background.LoadingWorker | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityHomeBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.settings.SettingsActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.Toppings | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.bottombar.maybeShow | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.bottombar.removeBadge | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.CustomTabActivityHelper | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| @@ -42,21 +41,18 @@ import com.ashokvarma.bottomnavigation.BottomNavigationItem | ||||
| import com.ashokvarma.bottomnavigation.TextBadgeItem | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
| import com.ftinc.scoop.Scoop | ||||
| import com.mikepenz.aboutlibraries.LibsBuilder | ||||
| import com.mikepenz.materialdrawer.holder.BadgeStyle | ||||
| import com.mikepenz.materialdrawer.holder.ColorHolder | ||||
| import com.mikepenz.materialdrawer.holder.StringHolder | ||||
| import com.mikepenz.materialdrawer.model.DividerDrawerItem | ||||
| import com.mikepenz.materialdrawer.model.PrimaryDrawerItem | ||||
| import com.mikepenz.materialdrawer.model.ProfileDrawerItem | ||||
| import com.mikepenz.materialdrawer.model.SecondaryDrawerItem | ||||
| import com.mikepenz.materialdrawer.model.interfaces.* | ||||
| import com.mikepenz.materialdrawer.util.AbstractDrawerImageLoader | ||||
| import com.mikepenz.materialdrawer.util.DrawerImageLoader | ||||
| import com.mikepenz.materialdrawer.util.addStickyFooterItem | ||||
| import com.mikepenz.materialdrawer.util.updateBadge | ||||
| import com.mikepenz.materialdrawer.widget.AccountHeaderView | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| @@ -64,12 +60,10 @@ import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
| import java.util.concurrent.TimeUnit | ||||
| import kotlin.concurrent.thread | ||||
|  | ||||
|  | ||||
| class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAware { | ||||
|  | ||||
|     private val SETTINGS_ACTIVITY: Int = 101111 | ||||
|     private val DRAWER_ID_TAGS = 100101L | ||||
|     private val DRAWER_ID_HIDDEN_TAGS = 101100L | ||||
|     private val DRAWER_ID_SOURCES = 100110L | ||||
| @@ -83,8 +77,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|     private lateinit var tabNewBadge: TextBadgeItem | ||||
|     private lateinit var tabArchiveBadge: TextBadgeItem | ||||
|     private lateinit var tabStarredBadge: TextBadgeItem | ||||
|     private lateinit var customTabActivityHelper: CustomTabActivityHelper | ||||
|     private lateinit var appColors: AppColors | ||||
|     private var offset: Int = 0 | ||||
|     private var firstVisible: Int = 0 | ||||
|     private lateinit var recyclerViewScrollListener: RecyclerView.OnScrollListener | ||||
| @@ -96,22 +88,15 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|     private lateinit var tagsBadge: Map<Long, Int> | ||||
|  | ||||
|     private val settingsLauncher = registerForActivityResult(ActivityResultContracts.StartActivityForResult()) { | ||||
|         appSettingsService.refreshUserSettings() | ||||
|     } | ||||
|  | ||||
|     override val di by closestDI() | ||||
|     private val repository : Repository by instance() | ||||
|     private val appSettingsService : AppSettingsService by instance() | ||||
|  | ||||
|     data class DrawerData(val tags: List<SelfossModel.Tag>?, val sources: List<SelfossModel.Source>?) | ||||
|  | ||||
|     override fun onStart() { | ||||
|         super.onStart() | ||||
|         customTabActivityHelper.bindCustomTabsService(this) | ||||
|     } | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|  | ||||
|         // Add appcolors to DI | ||||
|         appColors = AppColors(this@HomeActivity) | ||||
|  | ||||
|         super.onCreate(savedInstanceState) | ||||
|         binding = ActivityHomeBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
| @@ -125,8 +110,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|         setContentView(view) | ||||
|  | ||||
|         handleThemeBinding() | ||||
|  | ||||
|         setSupportActionBar(binding.toolBar) | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|         supportActionBar?.setHomeButtonEnabled(true) | ||||
| @@ -134,20 +117,17 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|         binding.drawerContainer.addDrawerListener(mDrawerToggle) | ||||
|         mDrawerToggle.syncState() | ||||
|  | ||||
|         customTabActivityHelper = CustomTabActivityHelper() | ||||
|  | ||||
|         handleBottomBar() | ||||
|         handleDrawer() | ||||
|         initDrawer() | ||||
|  | ||||
|         handleSwipeRefreshLayout() | ||||
|  | ||||
|         getElementsAccordingToTab() | ||||
|  | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             repository.tryToCacheItemsAndGetNewOnes() | ||||
|         if (appSettingsService.isItemCachingEnabled()) { | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|                 repository.tryToCacheItemsAndGetNewOnes() | ||||
|             } | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     private fun handleSwipeRefreshLayout() { | ||||
| @@ -199,8 +179,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|                         adapter.handleItemAtIndex(position) | ||||
|  | ||||
|                         reloadBadgeContent() | ||||
|  | ||||
|                         val tagHashes = i.tags.map { it.longHash() } | ||||
|                         tagsBadge = tagsBadge.map { | ||||
|                             if (tagHashes.contains(it.key)) { | ||||
| @@ -228,32 +206,61 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|         ItemTouchHelper(simpleItemTouchCallback).attachToRecyclerView(binding.recyclerView) | ||||
|     } | ||||
|  | ||||
|     private fun updateBottomBarBadgeCount(badge: TextBadgeItem, count: Int) { | ||||
|         if (count > 0) { | ||||
|             badge | ||||
|                 .setText(count.toString()) | ||||
|                 .maybeShow() | ||||
|         } else { | ||||
|             badge.removeBadge() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleBottomBar() { | ||||
|  | ||||
|         tabNewBadge = TextBadgeItem() | ||||
|             .setText("") | ||||
|             .setHideOnSelect(false).hide(false) | ||||
|             .setBackgroundColor(appColors.colorPrimary) | ||||
|         tabArchiveBadge = TextBadgeItem() | ||||
|             .setText("") | ||||
|             .setHideOnSelect(false).hide(false) | ||||
|             .setBackgroundColor(appColors.colorPrimary) | ||||
|         tabStarredBadge = TextBadgeItem() | ||||
|             .setText("") | ||||
|             .setHideOnSelect(false).hide(false) | ||||
|             .setBackgroundColor(appColors.colorPrimary) | ||||
|  | ||||
|         if (appSettingsService.isDisplayUnreadCountEnabled()) { | ||||
|             lifecycleScope.launch { | ||||
|                 repository.badgeUnread.collect { | ||||
|                     updateBottomBarBadgeCount(tabNewBadge, it) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (appSettingsService.isDisplayAllCountEnabled()) { | ||||
|             lifecycleScope.launch { | ||||
|                 repository.badgeAll.collect { | ||||
|                     updateBottomBarBadgeCount(tabArchiveBadge, it) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             lifecycleScope.launch { | ||||
|                 repository.badgeStarred.collect { | ||||
|                     updateBottomBarBadgeCount(tabStarredBadge, it) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val tabNew = | ||||
|             BottomNavigationItem( | ||||
|                 R.drawable.ic_tab_fiber_new_black_24dp, | ||||
|                 getString(R.string.tab_new) | ||||
|             ).setActiveColor(appColors.colorAccent) | ||||
|             ) | ||||
|                 .setBadgeItem(tabNewBadge) | ||||
|         val tabArchive = | ||||
|             BottomNavigationItem( | ||||
|                 R.drawable.ic_tab_archive_black_24dp, | ||||
|                 getString(R.string.tab_read) | ||||
|             ).setActiveColor(appColors.colorAccentDark) | ||||
|             ) | ||||
|                 .setBadgeItem(tabArchiveBadge) | ||||
|         val tabStarred = | ||||
|             BottomNavigationItem( | ||||
| @@ -269,7 +276,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|             .setFirstSelectedPosition(0) | ||||
|             .initialise() | ||||
|         binding.bottomBar.setMode(BottomNavigationBar.MODE_SHIFTING) | ||||
|         binding.bottomBar.setBackgroundStyle(BottomNavigationBar.BACKGROUND_STYLE_STATIC) | ||||
|  | ||||
|         if (fromTabShortcut) { | ||||
|             binding.bottomBar.selectTab(elementsShown.position - 1) | ||||
| @@ -279,12 +285,8 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|     override fun onResume() { | ||||
|         super.onResume() | ||||
|  | ||||
|         appColors = AppColors(this@HomeActivity) | ||||
|  | ||||
|         handleDrawerItems() | ||||
|  | ||||
|         handleThemeUpdate() | ||||
|  | ||||
|         reloadLayoutManager() | ||||
|  | ||||
|         if (appSettingsService.isInfiniteLoadingEnabled()) { | ||||
| @@ -304,36 +306,17 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|         getElementsAccordingToTab() | ||||
|     } | ||||
|  | ||||
|     override fun onStop() { | ||||
|         super.onStop() | ||||
|         customTabActivityHelper.unbindCustomTabsService(this) | ||||
|     } | ||||
|  | ||||
|     private fun handleThemeBinding() { | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.bind(this, Toppings.PRIMARY.value, binding.toolBar) | ||||
|         scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||
|     } | ||||
|  | ||||
|     private fun handleThemeUpdate() { | ||||
|  | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.update(Toppings.PRIMARY.value, appColors.colorPrimary) | ||||
|  | ||||
|         scoop.update(Toppings.PRIMARY_DARK.value, appColors.colorPrimaryDark) | ||||
|     } | ||||
|  | ||||
|     private fun handleDrawer() { | ||||
|     private fun initDrawer() { | ||||
|         DrawerImageLoader.init(object : AbstractDrawerImageLoader() { | ||||
|             override fun set(imageView: ImageView, uri: Uri, placeholder: Drawable, tag: String?) { | ||||
|                 Glide.with(this@HomeActivity) | ||||
|                         .asBitmap() | ||||
|                         .load(uri) | ||||
|                         .apply(RequestOptions() | ||||
|                                 .placeholder(R.mipmap.ic_launcher) | ||||
|                                 .fallback(R.mipmap.ic_launcher) | ||||
|                                 .fitCenter()) | ||||
|                         .into(imageView) | ||||
|                     .asBitmap() | ||||
|                     .load(uri) | ||||
|                     .apply(RequestOptions() | ||||
|                         .placeholder(R.mipmap.ic_launcher) | ||||
|                         .fallback(R.mipmap.ic_launcher) | ||||
|                         .fitCenter()) | ||||
|                     .into(imageView) | ||||
|             } | ||||
|  | ||||
|             override fun cancel(imageView: ImageView) { | ||||
| @@ -362,285 +345,31 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|         binding.drawerContainer.addDrawerListener(drawerListener) | ||||
|  | ||||
|         binding.mainDrawer.addStickyFooterItem( | ||||
|             PrimaryDrawerItem().apply { | ||||
|                 nameRes = R.string.drawer_report_bug | ||||
|                 iconRes = R.drawable.ic_bug_report_black_24dp | ||||
|                 isIconTinted = true | ||||
|                 onDrawerItemClickListener = { _, _, _ -> | ||||
|                     val browserIntent = Intent(Intent.ACTION_VIEW, Uri.parse(AppSettingsService.trackerUrl)) | ||||
|                     startActivity(browserIntent) | ||||
|                     false | ||||
|                 } | ||||
|             }) | ||||
|  | ||||
|         binding.mainDrawer.addStickyFooterItem( | ||||
|             PrimaryDrawerItem().apply { | ||||
|                 nameRes = R.string.title_activity_settings | ||||
|                 iconRes = R.drawable.ic_settings_black_24dp | ||||
|                 isIconTinted = true | ||||
|                 onDrawerItemClickListener = { _, _, _ -> | ||||
|                     startActivityForResult(Intent(this@HomeActivity, SettingsActivity::class.java), SETTINGS_ACTIVITY) | ||||
|                     false | ||||
|                 } | ||||
|             }) | ||||
|  | ||||
|         if (appSettingsService.isDisplayAccountHeaderEnabled()) { | ||||
|             AccountHeaderView(this).apply { | ||||
|                 attachToSliderView(binding.mainDrawer) | ||||
|                 addProfiles( | ||||
|                     ProfileDrawerItem().apply { | ||||
|                         nameText = appSettingsService.getBaseUrl() | ||||
|                         setBackgroundResource(R.drawable.bg) | ||||
|                         iconRes = R.mipmap.ic_launcher | ||||
|                         selectionListEnabledForSingleProfile = false | ||||
|                     } | ||||
|                 ) | ||||
|             } | ||||
|         // Sticky items | ||||
|         addStickyPrimaryItem(R.string.drawer_report_bug, R.drawable.ic_bug_report_black_24dp) { _, _, _ -> | ||||
|             val browserIntent = | ||||
|                 Intent(Intent.ACTION_VIEW, Uri.parse(AppSettingsService.trackerUrl)) | ||||
|             startActivity(browserIntent) | ||||
|             false | ||||
|         } | ||||
|         addStickyPrimaryItem(R.string.title_activity_settings, R.drawable.ic_settings_black_24dp) { _, _, _ -> | ||||
|             settingsLauncher.launch(Intent(this, SettingsActivity::class.java)) | ||||
|             false | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // TODO: refactor this. | ||||
|     // TODO: use updateSources | ||||
|     private fun addStickyPrimaryItem(name: Int, icon: Int, clickListener: ((v: View?, item: IDrawerItem<*>, position: Int) -> Boolean)?) { | ||||
|         binding.mainDrawer.addStickyFooterItem( | ||||
|             PrimaryDrawerItem().apply { | ||||
|                 nameRes = name | ||||
|                 iconRes = icon | ||||
|                 isIconTinted = true | ||||
|                 onDrawerItemClickListener = clickListener | ||||
|             }) | ||||
|     } | ||||
|  | ||||
|     private fun handleDrawerItems() { | ||||
|         tagsBadge = emptyMap() | ||||
|         fun handleDrawerData(maybeDrawerData: DrawerData?, loadedFromCache: Boolean = false) { | ||||
|             fun createDrawerItem( | ||||
|                 it: SelfossModel.Tag | ||||
|             ) { | ||||
|                 val gd = GradientDrawable() | ||||
|                 val gdColor = try { | ||||
|                     Color.parseColor(it.color) | ||||
|                 } catch (e: IllegalArgumentException) { | ||||
|                     appColors.colorPrimary | ||||
|                 } | ||||
|  | ||||
|                 gd.setColor(gdColor) | ||||
|                 gd.shape = GradientDrawable.RECTANGLE | ||||
|                 gd.setSize(30, 30) | ||||
|                 gd.cornerRadius = 30F | ||||
|  | ||||
|                 val drawerItem = PrimaryDrawerItem() | ||||
|                     .apply { | ||||
|                         nameText = it.tag.getHtmlDecoded() | ||||
|                         identifier = it.tag.longHash() | ||||
|                         iconDrawable = gd | ||||
|                         badgeStyle = BadgeStyle().apply { | ||||
|                             textColor = ColorHolder.fromColor(Color.WHITE) | ||||
|                             color = ColorHolder.fromColor(appColors.colorAccent) | ||||
|                         } | ||||
|                         onDrawerItemClickListener = { _, _, _ -> | ||||
|                             repository.tagFilter = it | ||||
|                             repository.sourceFilter = null | ||||
|                             getElementsAccordingToTab() | ||||
|                             fetchOnEmptyList() | ||||
|                             false | ||||
|                         } | ||||
|                     } | ||||
|                 if (it.unread > 0) { | ||||
|                     drawerItem.badgeText = it.unread.toString() | ||||
|                 } | ||||
|  | ||||
|                 binding.mainDrawer.itemAdapter.add(drawerItem) | ||||
|             } | ||||
|  | ||||
|             fun handleTags(maybeTags: List<SelfossModel.Tag>?) { | ||||
|                 if (maybeTags == null) { | ||||
|                     if (loadedFromCache) { | ||||
|                         binding.mainDrawer.itemAdapter.add( | ||||
|                             SecondaryDrawerItem() | ||||
|                                 .apply { nameRes = R.string.drawer_error_loading_tags; isSelectable = false } | ||||
|                         ) | ||||
|                     } | ||||
|                 } else { | ||||
|                     val filteredTags = maybeTags | ||||
|                         .filterNot { appSettingsService.getHiddenTags().contains(it.tag) } | ||||
|                         .sortedBy { it.unread == 0 } | ||||
|                     tagsBadge = filteredTags.map { | ||||
|                         createDrawerItem(it) | ||||
|  | ||||
|                         (it.tag.longHash() to it.unread) | ||||
|                     }.toMap() | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             fun handleHiddenTags(maybeTags: List<SelfossModel.Tag>?) { | ||||
|                 if (maybeTags == null) { | ||||
|                     if (loadedFromCache) { | ||||
|                         binding.mainDrawer.itemAdapter.add( | ||||
|                             SecondaryDrawerItem().apply { | ||||
|                                 nameRes = R.string.drawer_error_loading_tags | ||||
|                                 isSelectable = false | ||||
|                             } | ||||
|                         ) | ||||
|                     } | ||||
|                 } else { | ||||
|                     val filteredHiddenTags: List<SelfossModel.Tag> = | ||||
|                         maybeTags.filter { appSettingsService.getHiddenTags().contains(it.tag) } | ||||
|                     tagsBadge = filteredHiddenTags.map { | ||||
|                         createDrawerItem(it) | ||||
|  | ||||
|                         (it.tag.longHash() to it.unread) | ||||
|                     }.toMap() | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             fun handleSources(maybeSources: List<SelfossModel.Source>?) { | ||||
|                 if (maybeSources == null) { | ||||
|                     if (loadedFromCache) { | ||||
|                         binding.mainDrawer.itemAdapter.add( | ||||
|                             SecondaryDrawerItem().apply { | ||||
|                                 nameRes = R.string.drawer_error_loading_sources | ||||
|                                 isSelectable = false | ||||
|                             } | ||||
|                         ) | ||||
|                     } | ||||
|                 } else { | ||||
|                     for (source in maybeSources) { | ||||
|                         val item = PrimaryDrawerItem().apply { | ||||
|                             nameText = source.title.getHtmlDecoded() | ||||
|                             identifier = source.id.toLong() | ||||
|                             iconUrl = source.getIcon(repository.baseUrl) | ||||
|                             onDrawerItemClickListener = { _,_,_ -> | ||||
|                                 repository.sourceFilter = source | ||||
|                                 repository.tagFilter = null | ||||
|                                 getElementsAccordingToTab() | ||||
|                                 fetchOnEmptyList() | ||||
|                                 false | ||||
|                             } | ||||
|                         } | ||||
|                         binding.mainDrawer.itemAdapter.add(item) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             binding.mainDrawer.itemAdapter.clear() | ||||
|             if (maybeDrawerData != null) { | ||||
|                 binding.mainDrawer.itemAdapter.add( | ||||
|                     SecondaryDrawerItem().apply { | ||||
|                         nameRes = R.string.drawer_item_filters | ||||
|                         isSelectable = false | ||||
|                         identifier = DRAWER_ID_FILTERS | ||||
|                         badgeRes = R.string.drawer_action_clear | ||||
|                         onDrawerItemClickListener = { _,_,_ -> | ||||
|                             repository.sourceFilter = null | ||||
|                             repository.tagFilter = null | ||||
|                             binding.mainDrawer.setSelectionAtPosition(-1) | ||||
|                             getElementsAccordingToTab() | ||||
|                             fetchOnEmptyList() | ||||
|                             false | ||||
|                         } | ||||
|                     } | ||||
|                 ) | ||||
|                 if (appSettingsService.getHiddenTags().isNotEmpty()) { | ||||
|                     binding.mainDrawer.itemAdapter.add( | ||||
|                         DividerDrawerItem(), | ||||
|                         SecondaryDrawerItem().apply { | ||||
|                             nameRes = R.string.drawer_item_hidden_tags | ||||
|                             identifier = DRAWER_ID_HIDDEN_TAGS | ||||
|                             isSelectable = false | ||||
|                         } | ||||
|                     ) | ||||
|                     handleHiddenTags(maybeDrawerData.tags) | ||||
|                 } | ||||
|                 binding.mainDrawer.itemAdapter.add( | ||||
|                     DividerDrawerItem(), | ||||
|                     SecondaryDrawerItem().apply { | ||||
|                         nameRes = R.string.drawer_item_tags | ||||
|                         identifier = DRAWER_ID_TAGS | ||||
|                         isSelectable = false | ||||
|                     } | ||||
|                 ) | ||||
|                 handleTags(maybeDrawerData.tags) | ||||
|                 binding.mainDrawer.itemAdapter.add( | ||||
|                     DividerDrawerItem(), | ||||
|                     SecondaryDrawerItem().apply { | ||||
|                         nameRes = R.string.drawer_item_sources | ||||
|                         identifier = DRAWER_ID_SOURCES | ||||
|                         isSelectable = false | ||||
|                         badgeRes = R.string.drawer_action_edit | ||||
|                         onDrawerItemClickListener = { v,_,_ -> | ||||
|                             startActivity(Intent(v!!.context, SourcesActivity::class.java)) | ||||
|                             false | ||||
|                         } | ||||
|                     } | ||||
|                 ) | ||||
|                 handleSources(maybeDrawerData.sources) | ||||
|                 binding.mainDrawer.itemAdapter.add( | ||||
|                     DividerDrawerItem(), | ||||
|                     PrimaryDrawerItem().apply { | ||||
|                         nameRes = R.string.action_about | ||||
|                         isSelectable = false | ||||
|                         iconRes = R.drawable.ic_info_outline_white_24dp | ||||
|                         isIconTinted = true | ||||
|                         onDrawerItemClickListener = { _,_,_ -> | ||||
|                             LibsBuilder() | ||||
|                                 .withAboutIconShown(true) | ||||
|                                 .withAboutVersionShown(true) | ||||
|                                 .start(this@HomeActivity) | ||||
|                             false | ||||
|                         } | ||||
|                     } | ||||
|                 ) | ||||
|  | ||||
|                 if (!loadedFromCache) { | ||||
|                     if (maybeDrawerData.tags != null) { | ||||
|                         thread { | ||||
|                             repository.resetDBTagsWithData(maybeDrawerData.tags) | ||||
|                         } | ||||
|                     } | ||||
|                     if (maybeDrawerData.sources != null) { | ||||
|                         thread { | ||||
|                             repository.resetDBSourcesWithData(maybeDrawerData.sources) | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } else { | ||||
|                 if (!loadedFromCache) { | ||||
|                     binding.mainDrawer.itemAdapter.add( | ||||
|                         PrimaryDrawerItem().apply { | ||||
|                             nameRes = R.string.no_tags_loaded | ||||
|                             identifier = DRAWER_ID_TAGS | ||||
|                             isSelectable = false | ||||
|                         }, | ||||
|                         PrimaryDrawerItem().apply { | ||||
|                             nameRes = R.string.no_sources_loaded | ||||
|                             identifier = DRAWER_ID_SOURCES | ||||
|                             isSelectable = false | ||||
|                         } | ||||
|                     ) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fun drawerApiCalls(maybeDrawerData: DrawerData?) { | ||||
|             var tags: List<SelfossModel.Tag>? = null | ||||
|             var sources: List<SelfossModel.Source>? | ||||
|  | ||||
|             fun sourcesApiCall() { | ||||
|                 CoroutineScope(Dispatchers.Main).launch { | ||||
|                     val response = repository.getSources() | ||||
|                     if (response != null) { | ||||
|                         sources = response | ||||
|                         val apiDrawerData = DrawerData(tags, sources) | ||||
|                         if ((maybeDrawerData != null && maybeDrawerData != apiDrawerData) || maybeDrawerData == null) { | ||||
|                             handleDrawerData(apiDrawerData) | ||||
|                         } | ||||
|                     } else { | ||||
|                         val apiDrawerData = DrawerData(tags, null) | ||||
|                         if ((maybeDrawerData != null && maybeDrawerData != apiDrawerData) || maybeDrawerData == null) { | ||||
|                             handleDrawerData(apiDrawerData) | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 tags = repository.getTags() | ||||
|                 sourcesApiCall() | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         binding.mainDrawer.itemAdapter.add( | ||||
|             PrimaryDrawerItem().apply { | ||||
|                 nameRes = R.string.drawer_loading | ||||
| @@ -648,16 +377,177 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|             } | ||||
|         ) | ||||
|  | ||||
|         thread { | ||||
|             val drawerData = DrawerData(repository.getDBTags().map { it.toView() }, | ||||
|                                         repository.getDBSources().map { it.toView() }) | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             val tags = repository.getTags() | ||||
|             val sources = repository.getSources() | ||||
|             runOnUiThread { | ||||
|                 handleDrawerData(drawerData, loadedFromCache = true) | ||||
|                 drawerApiCalls(drawerData) | ||||
|                 handleDrawerData(tags, sources) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleDrawerData(tags: List<SelfossModel.Tag>, sources: List<SelfossModel.Source>) { | ||||
|         binding.mainDrawer.itemAdapter.clear() | ||||
|  | ||||
|         // Filters title with clear action | ||||
|         secondaryItem(withDivider = false, R.string.drawer_item_filters, DRAWER_ID_FILTERS, R.string.drawer_action_clear) { _,_,_ -> | ||||
|             repository.sourceFilter = null | ||||
|             repository.tagFilter = null | ||||
|             binding.mainDrawer.setSelectionAtPosition(-1) | ||||
|             getElementsAccordingToTab() | ||||
|             fetchOnEmptyList() | ||||
|             false | ||||
|         } | ||||
|  | ||||
|         // Hidden tags | ||||
|         if (tags.isNotEmpty() && appSettingsService.getHiddenTags().isNotEmpty()) { | ||||
|             secondaryItem( | ||||
|                 withDivider = true, | ||||
|                 R.string.drawer_item_hidden_tags, | ||||
|                 DRAWER_ID_HIDDEN_TAGS | ||||
|             ) | ||||
|             handleHiddenTags(tags) | ||||
|         } | ||||
|  | ||||
|         // Tags | ||||
|         secondaryItem(withDivider = true, R.string.drawer_item_tags, DRAWER_ID_TAGS) | ||||
|         if (tags.isEmpty()) { | ||||
|             binding.mainDrawer.itemAdapter.add( | ||||
|                 SecondaryDrawerItem() | ||||
|                     .apply { nameRes = R.string.drawer_error_loading_tags; isSelectable = false } | ||||
|             ) | ||||
|         } else { | ||||
|             handleTags(tags) | ||||
|         } | ||||
|  | ||||
|         // Sources | ||||
|         secondaryItem(withDivider = true, R.string.drawer_item_sources, DRAWER_ID_SOURCES, R.string.drawer_action_edit) { v, _, _ -> | ||||
|             startActivity(Intent(v!!.context, SourcesActivity::class.java)) | ||||
|             false | ||||
|         } | ||||
|         if (sources.isEmpty()) { | ||||
|             binding.mainDrawer.itemAdapter.add( | ||||
|                 SecondaryDrawerItem().apply { | ||||
|                     nameRes = R.string.drawer_error_loading_sources | ||||
|                     isSelectable = false | ||||
|                 } | ||||
|             ) | ||||
|         } else { | ||||
|             handleSources(sources) | ||||
|         } | ||||
|  | ||||
|         // About action | ||||
|         binding.mainDrawer.itemAdapter.add( | ||||
|             DividerDrawerItem(), | ||||
|             PrimaryDrawerItem().apply { | ||||
|                 nameRes = R.string.action_about | ||||
|                 isSelectable = false | ||||
|                 iconRes = R.drawable.ic_info_outline_white_24dp | ||||
|                 isIconTinted = true | ||||
|                 onDrawerItemClickListener = { _,_,_ -> | ||||
|                     LibsBuilder() | ||||
|                         .withAboutIconShown(true) | ||||
|                         .withAboutVersionShown(true) | ||||
|                         .start(this@HomeActivity) | ||||
|                     false | ||||
|                 } | ||||
|             } | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     private fun secondaryItem(withDivider: Boolean, name: Int, id: Long, badgeId: Int? = null, clickListener: ((v: View?, item: IDrawerItem<*>, position: Int) -> Boolean)? = null) { | ||||
|         if (withDivider) { | ||||
|             binding.mainDrawer.itemAdapter.add(DividerDrawerItem()) | ||||
|         } | ||||
|  | ||||
|         binding.mainDrawer.itemAdapter.add( | ||||
|             SecondaryDrawerItem().apply { | ||||
|                 nameRes = name | ||||
|                 identifier = id | ||||
|                 isSelectable = false | ||||
|                 if (badgeId != null) { | ||||
|                     badgeRes = badgeId | ||||
|                 } | ||||
|                 onDrawerItemClickListener = clickListener | ||||
|             } | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     private fun createDrawerItem(it: SelfossModel.Tag) { | ||||
|         val gd = GradientDrawable() | ||||
|         val gdColor = try { | ||||
|             Color.parseColor(it.color) | ||||
|         } catch (e: IllegalArgumentException) { | ||||
|             resources.getColor(R.color.colorPrimary) | ||||
|         } | ||||
|         gd.setColor(gdColor) | ||||
|         gd.shape = GradientDrawable.RECTANGLE | ||||
|         gd.setSize(30, 30) | ||||
|         gd.cornerRadius = 30F | ||||
|  | ||||
|         val drawerItem = PrimaryDrawerItem() | ||||
|             .apply { | ||||
|                 nameText = it.tag.getHtmlDecoded() | ||||
|                 identifier = it.tag.longHash() | ||||
|                 iconDrawable = gd | ||||
|                 badgeStyle = BadgeStyle().apply { | ||||
|                     textColor = ColorHolder.fromColor(Color.WHITE) | ||||
|                     color = ColorHolder.fromColor(resources.getColor(R.color.colorAccent)) | ||||
|                 } | ||||
|                 onDrawerItemClickListener = { _, _, _ -> | ||||
|                     repository.tagFilter = it | ||||
|                     repository.sourceFilter = null | ||||
|                     getElementsAccordingToTab() | ||||
|                     fetchOnEmptyList() | ||||
|                     false | ||||
|                 } | ||||
|             } | ||||
|         if (it.unread > 0) { | ||||
|             drawerItem.badgeText = it.unread.toString() | ||||
|         } | ||||
|  | ||||
|         binding.mainDrawer.itemAdapter.add(drawerItem) | ||||
|     } | ||||
|  | ||||
|     private fun handleTags(tags: List<SelfossModel.Tag>) { | ||||
|         val filteredTags = tags | ||||
|             .filterNot { appSettingsService.getHiddenTags().contains(it.tag) } | ||||
|             .sortedBy { it.tag } | ||||
|         createTagItems(filteredTags) | ||||
|     } | ||||
|  | ||||
|     private fun handleHiddenTags(tags: List<SelfossModel.Tag>) { | ||||
|         val filteredHiddenTags: List<SelfossModel.Tag> = | ||||
|             tags.filter { appSettingsService.getHiddenTags().contains(it.tag) } | ||||
|         createTagItems(filteredHiddenTags) | ||||
|     } | ||||
|  | ||||
|     private fun createTagItems(tags: List<SelfossModel.Tag>) { | ||||
|         tagsBadge = tags.associate { | ||||
|             createDrawerItem(it) | ||||
|  | ||||
|             (it.tag.longHash() to it.unread) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleSources(sources: List<SelfossModel.Source>) { | ||||
|         for (source in sources) { | ||||
|             val item = PrimaryDrawerItem().apply { | ||||
|                 nameText = source.title.getHtmlDecoded() | ||||
|                 identifier = source.id.toLong() | ||||
|                 iconUrl = source.getIcon(repository.baseUrl) | ||||
|                 onDrawerItemClickListener = { _,_,_ -> | ||||
|                     repository.sourceFilter = source | ||||
|                     repository.tagFilter = null | ||||
|                     getElementsAccordingToTab() | ||||
|                     fetchOnEmptyList() | ||||
|                     false | ||||
|                 } | ||||
|             } | ||||
|             binding.mainDrawer.itemAdapter.add(item) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun reloadLayoutManager() { | ||||
|         val currentManager = binding.recyclerView.layoutManager | ||||
|         val layoutManager: RecyclerView.LayoutManager | ||||
| @@ -825,8 +715,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|                         ItemCardAdapter( | ||||
|                             this, | ||||
|                             items, | ||||
|                             customTabActivityHelper, | ||||
|                             appColors, | ||||
|                         ) { | ||||
|                             updateItems(it) | ||||
|                         } | ||||
| @@ -835,8 +723,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|                         ItemListAdapter( | ||||
|                             this, | ||||
|                             items, | ||||
|                             customTabActivityHelper, | ||||
|                             appColors, | ||||
|                         ) { | ||||
|                             updateItems(it) | ||||
|                         } | ||||
| @@ -859,29 +745,12 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|     private fun reloadBadges() { | ||||
|         if (appSettingsService.isDisplayUnreadCountEnabled() || appSettingsService.isDisplayAllCountEnabled()) { | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 repository.reloadBadges() | ||||
|                 reloadBadgeContent() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun reloadBadgeContent() { | ||||
|         if (appSettingsService.isDisplayUnreadCountEnabled()) { | ||||
|             tabNewBadge | ||||
|                 .setText(repository.badgeUnread.toString()) | ||||
|                 .maybeShow() | ||||
|         } | ||||
|         if (appSettingsService.isDisplayAllCountEnabled()) { | ||||
|             tabArchiveBadge | ||||
|                 .setText(repository.badgeAll.toString()) | ||||
|                 .maybeShow() | ||||
|             tabStarredBadge | ||||
|                 .setText(repository.badgeStarred.toString()) | ||||
|                 .maybeShow() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun reloadTagsBadges() { | ||||
|         tagsBadge.forEach { | ||||
|             binding.mainDrawer.updateBadge(it.key, StringHolder(it.value.toString())) | ||||
| @@ -1003,10 +872,10 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|     private fun maxItemNumber(): Int = | ||||
|         when (elementsShown) { | ||||
|             ItemType.UNREAD -> repository.badgeUnread | ||||
|             ItemType.ALL -> repository.badgeAll | ||||
|             ItemType.STARRED -> repository.badgeStarred | ||||
|             else -> repository.badgeUnread // if !elementsShown then unread are fetched. | ||||
|             ItemType.UNREAD -> repository.badgeUnread.value | ||||
|             ItemType.ALL -> repository.badgeAll.value | ||||
|             ItemType.STARRED -> repository.badgeStarred.value | ||||
|             else -> repository.badgeUnread.value // if !elementsShown then unread are fetched. | ||||
|         } | ||||
|  | ||||
|     private fun updateItems(adapterItems: ArrayList<SelfossModel.Item>) { | ||||
| @@ -1030,12 +899,5 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|             WorkManager.getInstance(baseContext).enqueueUniquePeriodicWork("selfoss-loading", ExistingPeriodicWorkPolicy.KEEP, backgroundWork) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { | ||||
|         super.onActivityResult(requestCode, resultCode, data) | ||||
|         if (requestCode == SETTINGS_ACTIVITY) { | ||||
|             appSettingsService.refreshUserSettings() | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -12,9 +12,9 @@ import android.view.inputmethod.EditorInfo | ||||
| import android.widget.TextView | ||||
| import androidx.appcompat.app.AlertDialog | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.appcompat.app.AppCompatDelegate | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityLoginBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlValid | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.isBaseUrlInvalid | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.mikepenz.aboutlibraries.LibsBuilder | ||||
| @@ -30,7 +30,6 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|     private var inValidCount: Int = 0 | ||||
|     private var isWithLogin = false | ||||
|  | ||||
|     private lateinit var appColors: AppColors | ||||
|     private lateinit var binding: ActivityLoginBinding | ||||
|  | ||||
|     override val di by closestDI() | ||||
| @@ -38,9 +37,9 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|     private val appSettingsService : AppSettingsService by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         appColors = AppColors(this@LoginActivity) | ||||
|  | ||||
|         super.onCreate(savedInstanceState) | ||||
|         AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_FOLLOW_SYSTEM) | ||||
|  | ||||
|         binding = ActivityLoginBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
| @@ -94,12 +93,15 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|     } | ||||
|  | ||||
|     private fun goToMain() { | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             repository.updateApiVersion() | ||||
|         } | ||||
|         val intent = Intent(this, HomeActivity::class.java) | ||||
|         startActivity(intent) | ||||
|         finish() | ||||
|     } | ||||
|  | ||||
|     private fun preferenceError(t: Throwable) { | ||||
|     private fun preferenceError() { | ||||
|         appSettingsService.resetLoginInformation() | ||||
|  | ||||
|         binding.urlView.error = getString(R.string.wrong_infos) | ||||
| @@ -115,14 +117,14 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|         binding.passwordView.error = null | ||||
|  | ||||
|         // Store values at the time of the login attempt. | ||||
|         val url = binding.urlView.text.toString() | ||||
|         val login = binding.loginView.text.toString() | ||||
|         val password = binding.passwordView.text.toString() | ||||
|         val url = binding.urlView.text.toString().trim() | ||||
|         val login = binding.loginView.text.toString().trim() | ||||
|         val password = binding.passwordView.text.toString().trim() | ||||
|  | ||||
|         var cancel = false | ||||
|         var focusView: View? = null | ||||
|  | ||||
|         if (!url.isBaseUrlValid(this@LoginActivity)) { | ||||
|         if (url.isBaseUrlInvalid()) { | ||||
|             binding.urlView.error = getString(R.string.login_url_problem) | ||||
|             focusView = binding.urlView | ||||
|             cancel = true | ||||
| @@ -164,11 +166,10 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 val result = repository.login() | ||||
|                 if (result) { | ||||
|                     repository.updateApiVersion() | ||||
|                     goToMain() | ||||
|                 } else { | ||||
|                     CoroutineScope(Dispatchers.Main).launch { | ||||
|                         preferenceError(Exception("Not success")) | ||||
|                         preferenceError() | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|   | ||||
| @@ -3,17 +3,21 @@ package bou.amine.apps.readerforselfossv2.android | ||||
| import android.app.NotificationChannel | ||||
| import android.app.NotificationManager | ||||
| import android.content.Context | ||||
| import android.content.res.Configuration | ||||
| import android.graphics.drawable.Drawable | ||||
| import android.net.Uri | ||||
| import android.os.Build | ||||
| import android.widget.ImageView | ||||
| import android.widget.Toast | ||||
| import androidx.appcompat.app.AppCompatDelegate | ||||
| import androidx.appcompat.app.AppCompatDelegate.* | ||||
| import androidx.lifecycle.DefaultLifecycleObserver | ||||
| import androidx.lifecycle.LifecycleOwner | ||||
| import androidx.lifecycle.ProcessLifecycleOwner | ||||
| import androidx.multidex.MultiDexApplication | ||||
| import androidx.preference.PreferenceManager | ||||
| import bou.amine.apps.readerforselfossv2.DI.networkModule | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.network.isNetworkAccessible | ||||
| import bou.amine.apps.readerforselfossv2.android.viewmodel.AppViewModel | ||||
| import bou.amine.apps.readerforselfossv2.dao.DriverFactory | ||||
| import bou.amine.apps.readerforselfossv2.dao.ReaderForSelfossDB | ||||
| @@ -21,7 +25,6 @@ import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
| import com.ftinc.scoop.Scoop | ||||
| import com.github.ln_12.library.ConnectivityStatus | ||||
| import com.mikepenz.materialdrawer.util.AbstractDrawerImageLoader | ||||
| import com.mikepenz.materialdrawer.util.DrawerImageLoader | ||||
| @@ -29,6 +32,7 @@ import io.github.aakira.napier.DebugAntilog | ||||
| import io.github.aakira.napier.Napier | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.flow.MutableStateFlow | ||||
| import kotlinx.coroutines.launch | ||||
| import org.kodein.di.* | ||||
|  | ||||
| @@ -38,7 +42,7 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|         import(networkModule) | ||||
|         bind<DriverFactory>() with singleton { DriverFactory(applicationContext) } | ||||
|         bind<ReaderForSelfossDB>() with singleton { ReaderForSelfossDB(driverFactory.createDriver()) } | ||||
|         bind<Repository>() with singleton { Repository(instance(), instance(), connectivityStatus, instance()) } | ||||
|         bind<Repository>() with singleton { Repository(instance(), instance(), isConnectionAvailable, instance()) } | ||||
|         bind<ConnectivityStatus>() with singleton { ConnectivityStatus(applicationContext) } | ||||
|         bind<AppViewModel>() with singleton { AppViewModel(repository = instance()) } | ||||
|     } | ||||
| @@ -47,6 +51,10 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|     private val viewModel: AppViewModel by instance() | ||||
|     private val connectivityStatus: ConnectivityStatus by instance() | ||||
|     private val driverFactory: DriverFactory by instance() | ||||
|     private val appSettingsService : AppSettingsService by instance() | ||||
|  | ||||
|     // TODO: handle with the "previous" way | ||||
|     private val isConnectionAvailable: MutableStateFlow<Boolean> = MutableStateFlow(true) | ||||
|  | ||||
|     override fun onCreate() { | ||||
|         super.onCreate() | ||||
| @@ -54,8 +62,6 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|  | ||||
|         initDrawerImageLoader() | ||||
|  | ||||
|         initTheme() | ||||
|  | ||||
|         tryToHandleBug() | ||||
|  | ||||
|         handleNotificationChannels() | ||||
| @@ -78,7 +84,6 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|                 ).show() | ||||
|             } | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     private fun handleNotificationChannels() { | ||||
| @@ -117,14 +122,6 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|         }) | ||||
|     } | ||||
|  | ||||
|     private fun initTheme() { | ||||
|         Scoop.waffleCone() | ||||
|             .addFlavor(getString(R.string.default_theme), R.style.NoBar, true) | ||||
|             .addFlavor(getString(R.string.default_dark_theme), R.style.NoBarDark, false) | ||||
|             .setSharedPreferences(PreferenceManager.getDefaultSharedPreferences(this)) | ||||
|             .initialize() | ||||
|     } | ||||
|  | ||||
|     private fun tryToHandleBug() { | ||||
|         val oldHandler = Thread.getDefaultUncaughtExceptionHandler() | ||||
|  | ||||
| @@ -139,6 +136,19 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun onConfigurationChanged(newConfig: Configuration) { | ||||
|         super.onConfigurationChanged(newConfig) | ||||
|         if (appSettingsService.getCurrentTheme() == MODE_NIGHT_FOLLOW_SYSTEM) { | ||||
|             var mode = when (newConfig.uiMode and Configuration.UI_MODE_NIGHT_MASK) { | ||||
|                 Configuration.UI_MODE_NIGHT_YES -> MODE_NIGHT_YES | ||||
|                 else -> MODE_NIGHT_NO | ||||
|             } | ||||
|             setDefaultNightMode(mode) | ||||
|         } | ||||
|  | ||||
|  | ||||
|     } | ||||
|  | ||||
|     class AppLifeCycleObserver(val connectivityStatus: ConnectivityStatus, val repository: Repository) : DefaultLifecycleObserver { | ||||
|  | ||||
|         override fun onResume(owner: LifecycleOwner) { | ||||
|   | ||||
| @@ -12,12 +12,9 @@ import androidx.viewpager2.adapter.FragmentStateAdapter | ||||
| import androidx.viewpager2.widget.ViewPager2 | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityReaderBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.fragments.ArticleFragment | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.Toppings | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.ftinc.scoop.Scoop | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| @@ -28,7 +25,6 @@ import org.kodein.di.instance | ||||
| class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private var currentItem: Int = 0 | ||||
|     private lateinit var appColors: AppColors | ||||
|  | ||||
|     private lateinit var toolbarMenu: Menu | ||||
|  | ||||
| @@ -40,9 +36,9 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private fun showMenuItem(willAddToFavorite: Boolean) { | ||||
|         if (willAddToFavorite) { | ||||
|             toolbarMenu.findItem(R.id.star).icon.setTint(Color.WHITE) | ||||
|             toolbarMenu.findItem(R.id.star).icon?.setTint(Color.WHITE) | ||||
|         } else { | ||||
|             toolbarMenu.findItem(R.id.star).icon.setTint(Color.RED) | ||||
|             toolbarMenu.findItem(R.id.star).icon?.setTint(Color.RED) | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -56,16 +52,11 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         appColors = AppColors(this) | ||||
|         binding = ActivityReaderBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
|         setContentView(view) | ||||
|  | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.bind(this, Toppings.PRIMARY.value, binding.toolBar) | ||||
|         scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||
|  | ||||
|         setSupportActionBar(binding.toolBar) | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||
|   | ||||
| @@ -8,11 +8,8 @@ import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.recyclerview.widget.LinearLayoutManager | ||||
| import bou.amine.apps.readerforselfossv2.android.adapters.SourcesListAdapter | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySourcesBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.Toppings | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import com.ftinc.scoop.Scoop | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| @@ -22,21 +19,15 @@ import org.kodein.di.instance | ||||
|  | ||||
| class SourcesActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private lateinit var appColors: AppColors | ||||
|     private lateinit var binding: ActivitySourcesBinding | ||||
|  | ||||
|     override val di by closestDI() | ||||
|     private val repository : Repository by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         appColors = AppColors(this@SourcesActivity) | ||||
|         binding = ActivitySourcesBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.bind(this, Toppings.PRIMARY.value, binding.toolbar) | ||||
|         scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||
|  | ||||
|         super.onCreate(savedInstanceState) | ||||
|  | ||||
|         setContentView(view) | ||||
| @@ -45,8 +36,8 @@ class SourcesActivity : AppCompatActivity(), DIAware { | ||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||
|  | ||||
|         binding.fab.rippleColor = appColors.colorAccentDark | ||||
|         binding.fab.backgroundTintList = ColorStateList.valueOf(appColors.colorAccent) | ||||
|         binding.fab.rippleColor = resources.getColor(R.color.colorAccentDark) | ||||
|         binding.fab.backgroundTintList = ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) | ||||
|     } | ||||
|  | ||||
|     override fun onStop() { | ||||
| @@ -65,20 +56,13 @@ class SourcesActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             val response = repository.getSources() | ||||
|             if (response != null) { | ||||
|             if (response.isNotEmpty()) { | ||||
|                 items = response | ||||
|                 val mAdapter = SourcesListAdapter( | ||||
|                     this@SourcesActivity, items | ||||
|                 ) | ||||
|                 binding.recyclerView.adapter = mAdapter | ||||
|                 mAdapter.notifyDataSetChanged() | ||||
|                 if (items.isEmpty()) { | ||||
|                     Toast.makeText( | ||||
|                         this@SourcesActivity, | ||||
|                         R.string.nothing_here, | ||||
|                         Toast.LENGTH_SHORT | ||||
|                     ).show() | ||||
|                 } | ||||
|             } else { | ||||
|                 Toast.makeText( | ||||
|                     this@SourcesActivity, | ||||
|   | ||||
| @@ -10,11 +10,12 @@ import androidx.recyclerview.widget.RecyclerView | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.CardItemBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.* | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.CustomTabActivityHelper | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.LinkOnTouchListener | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.bitmapCenterCrop | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularBitmapDrawable | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrl | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.shareLink | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| @@ -34,8 +35,6 @@ import org.kodein.di.instance | ||||
| class ItemCardAdapter( | ||||
|     override val app: Activity, | ||||
|     override var items: ArrayList<SelfossModel.Item>, | ||||
|     private val helper: CustomTabActivityHelper, | ||||
|     override val appColors: AppColors, | ||||
|     override val updateItems: (ArrayList<SelfossModel.Item>) -> Unit | ||||
| ) : ItemsAdapter<ItemCardAdapter.ViewHolder>() { | ||||
|     private val c: Context = app.baseContext | ||||
| @@ -44,8 +43,8 @@ class ItemCardAdapter( | ||||
|         c.resources.getDimension(R.dimen.card_image_max_height).toInt() | ||||
|  | ||||
|     override val di: DI by closestDI(app) | ||||
|     override val repository : Repository by instance() | ||||
|     override val appSettingsService : AppSettingsService by instance() | ||||
|     override val repository: Repository by instance() | ||||
|     override val appSettingsService: AppSettingsService by instance() | ||||
|  | ||||
|     override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { | ||||
|         val binding = CardItemBinding.inflate(LayoutInflater.from(parent.context), parent, false) | ||||
| @@ -61,9 +60,9 @@ class ItemCardAdapter( | ||||
|  | ||||
|             binding.title.setOnTouchListener(LinkOnTouchListener()) | ||||
|  | ||||
|             binding.title.setLinkTextColor(appColors.colorAccent) | ||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText(repository.dateUtils) | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText() | ||||
|  | ||||
|             if (!appSettingsService.isFullHeightCardsEnabled()) { | ||||
|                 binding.itemImage.maxHeight = imageMaxHeight | ||||
| @@ -83,10 +82,10 @@ class ItemCardAdapter( | ||||
|                 val color = generator.getColor(itm.title.getHtmlDecoded()) | ||||
|  | ||||
|                 val drawable = | ||||
|                         TextDrawable | ||||
|                                 .builder() | ||||
|                                 .round() | ||||
|                                 .build(itm.title.getHtmlDecoded().toTextDrawableString(), color) | ||||
|                     TextDrawable | ||||
|                         .builder() | ||||
|                         .round() | ||||
|                         .build(itm.title.getHtmlDecoded().toTextDrawableString(), color) | ||||
|                 binding.sourceImage.setImageDrawable(drawable) | ||||
|             } else { | ||||
|                 c.circularBitmapDrawable(itm.getIcon(repository.baseUrl), binding.sourceImage) | ||||
| @@ -101,7 +100,7 @@ class ItemCardAdapter( | ||||
|     inner class ViewHolder(val binding: CardItemBinding) : RecyclerView.ViewHolder(binding.root) { | ||||
|         init { | ||||
|             handleClickListeners() | ||||
|             handleCustomTabActions() | ||||
|             handleLinkOpening() | ||||
|         } | ||||
|  | ||||
|         private fun handleClickListeners() { | ||||
| @@ -111,41 +110,32 @@ class ItemCardAdapter( | ||||
|                 if (item.starred) { | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.unstarr(item) | ||||
|                         // TODO: Handle failure | ||||
|                     } | ||||
|                     item.starred = false | ||||
|                     binding.favButton.isSelected = false | ||||
|                 } else { | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.starr(item) | ||||
|                         // TODO: Handle failure | ||||
|                     } | ||||
|                     item.starred = true | ||||
|                     binding.favButton.isSelected = true | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|                 binding.shareBtn.setOnClickListener { | ||||
|                     val item = items[bindingAdapterPosition] | ||||
|                     c.shareLink(item.getLinkDecoded(), item.title.getHtmlDecoded()) | ||||
|                 } | ||||
|  | ||||
|                 binding.browserBtn.setOnClickListener { | ||||
|                     c.openInBrowserAsNewTask(items[bindingAdapterPosition]) | ||||
|                 } | ||||
|             binding.shareBtn.setOnClickListener { | ||||
|                 val item = items[bindingAdapterPosition] | ||||
|                 c.shareLink(item.getLinkDecoded(), item.title.getHtmlDecoded()) | ||||
|             } | ||||
|  | ||||
|         private fun handleCustomTabActions() { | ||||
|             val customTabsIntent = c.buildCustomTabsIntent() | ||||
|             helper.bindCustomTabsService(app) | ||||
|             binding.browserBtn.setOnClickListener { | ||||
|                 c.openInBrowserAsNewTask(items[bindingAdapterPosition]) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         private fun handleLinkOpening() { | ||||
|             binding.root.setOnClickListener { | ||||
|                 c.openItemUrl( | ||||
|                     items, | ||||
|                     bindingAdapterPosition, | ||||
|                     items[bindingAdapterPosition].getLinkDecoded(), | ||||
|                     customTabsIntent, | ||||
|                     appSettingsService.isInternalBrowserEnabled(), | ||||
|                     appSettingsService.isArticleViewerEnabled(), | ||||
|                     app | ||||
|                 ) | ||||
|   | ||||
| @@ -5,12 +5,10 @@ import android.content.Context | ||||
| import android.view.LayoutInflater | ||||
| import android.view.ViewGroup | ||||
| import androidx.recyclerview.widget.RecyclerView | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ListItemBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.LinkOnTouchListener | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.buildCustomTabsIntent | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.CustomTabActivityHelper | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.bitmapCenterCrop | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularBitmapDrawable | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrl | ||||
| @@ -29,8 +27,6 @@ import org.kodein.di.instance | ||||
| class ItemListAdapter( | ||||
|     override val app: Activity, | ||||
|     override var items: ArrayList<SelfossModel.Item>, | ||||
|     private val helper: CustomTabActivityHelper, | ||||
|     override val appColors: AppColors, | ||||
|     override val updateItems: (ArrayList<SelfossModel.Item>) -> Unit | ||||
| ) : ItemsAdapter<ItemListAdapter.ViewHolder>() { | ||||
|     private val generator: ColorGenerator = ColorGenerator.MATERIAL | ||||
| @@ -53,9 +49,9 @@ class ItemListAdapter( | ||||
|  | ||||
|             binding.title.setOnTouchListener(LinkOnTouchListener()) | ||||
|  | ||||
|             binding.title.setLinkTextColor(appColors.colorAccent) | ||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText(repository.dateUtils) | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText() | ||||
|  | ||||
|             if (itm.getThumbnail(repository.baseUrl).isEmpty()) { | ||||
|  | ||||
| @@ -83,20 +79,15 @@ class ItemListAdapter( | ||||
|     inner class ViewHolder(val binding: ListItemBinding) : RecyclerView.ViewHolder(binding.root) { | ||||
|  | ||||
|         init { | ||||
|             handleCustomTabActions() | ||||
|             handleLinkOpening() | ||||
|         } | ||||
|  | ||||
|         private fun handleCustomTabActions() { | ||||
|             val customTabsIntent = c.buildCustomTabsIntent() | ||||
|             helper.bindCustomTabsService(app) | ||||
|  | ||||
|         private fun handleLinkOpening() { | ||||
|             binding.root.setOnClickListener { | ||||
|                 c.openItemUrl( | ||||
|                     items, | ||||
|                     bindingAdapterPosition, | ||||
|                     items[bindingAdapterPosition].getLinkDecoded(), | ||||
|                     customTabsIntent, | ||||
|                     appSettingsService.isInternalBrowserEnabled(), | ||||
|                     appSettingsService.isArticleViewerEnabled(), | ||||
|                     app | ||||
|                 ) | ||||
|   | ||||
| @@ -5,7 +5,6 @@ import android.graphics.Color | ||||
| import android.widget.TextView | ||||
| import androidx.recyclerview.widget.RecyclerView | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| @@ -21,7 +20,6 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|     abstract val repository: Repository | ||||
|     abstract val appSettingsService: AppSettingsService | ||||
|     abstract val app: Activity | ||||
|     abstract val appColors: AppColors | ||||
|     abstract val updateItems: (ArrayList<SelfossModel.Item>) -> Unit | ||||
|  | ||||
|     fun updateAllItems(items: ArrayList<SelfossModel.Item>) { | ||||
| @@ -30,7 +28,7 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|         updateItems(this.items) | ||||
|     } | ||||
|  | ||||
|     private fun unmarkSnackbar(i: SelfossModel.Item, position: Int) { | ||||
|     private fun unmarkSnackbar(item: SelfossModel.Item, position: Int) { | ||||
|         val s = Snackbar | ||||
|             .make( | ||||
|                 app.findViewById(R.id.coordLayout), | ||||
| @@ -39,7 +37,7 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|             ) | ||||
|             .setAction(R.string.undo_string) { | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     unreadItemAtIndex(position, false) | ||||
|                     unreadItemAtIndex(item, position, false) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
| @@ -49,7 +47,7 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|         s.show() | ||||
|     } | ||||
|  | ||||
|     private fun markSnackbar(position: Int) { | ||||
|     private fun markSnackbar(item: SelfossModel.Item, position: Int) { | ||||
|         val s = Snackbar | ||||
|             .make( | ||||
|                 app.findViewById(R.id.coordLayout), | ||||
| @@ -57,7 +55,7 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|                 Snackbar.LENGTH_LONG | ||||
|             ) | ||||
|             .setAction(R.string.undo_string) { | ||||
|                 readItemAtIndex(position) | ||||
|                 readItemAtIndex(item, position, false) | ||||
|             } | ||||
|  | ||||
|         val view = s.view | ||||
| @@ -68,37 +66,36 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|  | ||||
|     fun handleItemAtIndex(position: Int) { | ||||
|         if (items[position].unread) { | ||||
|             readItemAtIndex(position) | ||||
|             readItemAtIndex(items[position], position) | ||||
|         } else { | ||||
|             unreadItemAtIndex(position) | ||||
|             unreadItemAtIndex(items[position], position) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun readItemAtIndex(position: Int, showSnackbar: Boolean = true) { | ||||
|         val i = items[position] | ||||
|     private fun readItemAtIndex(item: SelfossModel.Item, position: Int, showSnackbar: Boolean = true) { | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             repository.markAsRead(i) | ||||
|             repository.markAsRead(item) | ||||
|         } | ||||
|         if (repository.displayedItems == ItemType.UNREAD) { | ||||
|             items.remove(i) | ||||
|             items.remove(item) | ||||
|             notifyItemRemoved(position) | ||||
|             updateItems(items) | ||||
|         } else { | ||||
|             notifyItemChanged(position) | ||||
|         } | ||||
|         if (showSnackbar) { | ||||
|             unmarkSnackbar(i, position) | ||||
|             unmarkSnackbar(item, position) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun unreadItemAtIndex(position: Int, showSnackbar: Boolean = true) { | ||||
|     private fun unreadItemAtIndex(item: SelfossModel.Item, position: Int, showSnackbar: Boolean = true) { | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             repository.unmarkAsRead(items[position]) | ||||
|             repository.unmarkAsRead(item) | ||||
|  | ||||
|         } | ||||
|         notifyItemChanged(position) | ||||
|         if (showSnackbar) { | ||||
|             markSnackbar(position) | ||||
|             markSnackbar(item, position) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -61,9 +61,13 @@ class SourcesListAdapter( | ||||
|         binding.sourceTitle.text = itm.title.getHtmlDecoded() | ||||
|     } | ||||
|  | ||||
|     override fun getItemId(position: Int) = position.toLong() | ||||
|  | ||||
|     override fun getItemViewType(position: Int) = position | ||||
|  | ||||
|     override fun getItemCount(): Int = items.size | ||||
|  | ||||
|     inner class ViewHolder(internal val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) { | ||||
|     inner class ViewHolder(private val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) { | ||||
|  | ||||
|         init { | ||||
|             handleClickListeners() | ||||
| @@ -74,13 +78,13 @@ class SourcesListAdapter( | ||||
|             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) | ||||
|  | ||||
|             deleteBtn.setOnClickListener { | ||||
|                 val (id) = items[adapterPosition] | ||||
|                 val (id) = items[bindingAdapterPosition] | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     val successfullyDeletedSource = repository.deleteSource(id) | ||||
|                     if (successfullyDeletedSource) { | ||||
|                         items.removeAt(adapterPosition) | ||||
|                         notifyItemRemoved(adapterPosition) | ||||
|                         notifyItemRangeChanged(adapterPosition, itemCount) | ||||
|                         items.removeAt(bindingAdapterPosition) | ||||
|                         notifyItemRemoved(bindingAdapterPosition) | ||||
|                         notifyItemRangeChanged(bindingAdapterPosition, itemCount) | ||||
|                     } else { | ||||
|                         Toast.makeText( | ||||
|                             app, | ||||
|   | ||||
| @@ -1,35 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.api.mercury | ||||
|  | ||||
| import com.google.gson.GsonBuilder | ||||
| import okhttp3.OkHttpClient | ||||
| import okhttp3.logging.HttpLoggingInterceptor | ||||
| import retrofit2.Call | ||||
| import retrofit2.Retrofit | ||||
| import retrofit2.converter.gson.GsonConverterFactory | ||||
|  | ||||
| class MercuryApi() { | ||||
|     private val service: MercuryService | ||||
|  | ||||
|     init { | ||||
|  | ||||
|         val interceptor = HttpLoggingInterceptor() | ||||
|         interceptor.level = HttpLoggingInterceptor.Level.NONE | ||||
|         val client = OkHttpClient.Builder().addInterceptor(interceptor).build() | ||||
|  | ||||
|         val gson = GsonBuilder() | ||||
|             .setLenient() | ||||
|             .create() | ||||
|         val retrofit = | ||||
|             Retrofit | ||||
|                 .Builder() | ||||
|                 .baseUrl("https://www.amine-louveau.fr") | ||||
|                 .client(client) | ||||
|                 .addConverterFactory(GsonConverterFactory.create(gson)) | ||||
|                 .build() | ||||
|         service = retrofit.create(MercuryService::class.java) | ||||
|     } | ||||
|  | ||||
|     fun parseUrl(url: String): Call<ParsedContent> { | ||||
|         return service.parseUrl(url) | ||||
|     } | ||||
| } | ||||
| @@ -1,59 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.api.mercury | ||||
|  | ||||
| import android.os.Parcel | ||||
| import android.os.Parcelable | ||||
| import com.google.gson.annotations.SerializedName | ||||
|  | ||||
| class ParsedContent( | ||||
|     @SerializedName("title") val title: String, | ||||
|     @SerializedName("content") val content: String?, | ||||
|     @SerializedName("date_published") val date_published: String, | ||||
|     @SerializedName("lead_image_url") val lead_image_url: String?, | ||||
|     @SerializedName("dek") val dek: String, | ||||
|     @SerializedName("url") val url: String, | ||||
|     @SerializedName("domain") val domain: String, | ||||
|     @SerializedName("excerpt") val excerpt: String, | ||||
|     @SerializedName("total_pages") val total_pages: Int, | ||||
|     @SerializedName("rendered_pages") val rendered_pages: Int, | ||||
|     @SerializedName("next_page_url") val next_page_url: String | ||||
| ) : Parcelable { | ||||
|  | ||||
|     companion object { | ||||
|         @JvmField | ||||
|         val CREATOR: Parcelable.Creator<ParsedContent> = | ||||
|             object : Parcelable.Creator<ParsedContent> { | ||||
|                 override fun createFromParcel(source: Parcel): ParsedContent = ParsedContent(source) | ||||
|                 override fun newArray(size: Int): Array<ParsedContent?> = arrayOfNulls(size) | ||||
|             } | ||||
|     } | ||||
|  | ||||
|     constructor(source: Parcel) : this( | ||||
|         title = source.readString().orEmpty(), | ||||
|         content = source.readString(), | ||||
|         date_published = source.readString().orEmpty(), | ||||
|         lead_image_url = source.readString(), | ||||
|         dek = source.readString().orEmpty(), | ||||
|         url = source.readString().orEmpty(), | ||||
|         domain = source.readString().orEmpty(), | ||||
|         excerpt = source.readString().orEmpty(), | ||||
|         total_pages = source.readInt(), | ||||
|         rendered_pages = source.readInt(), | ||||
|         next_page_url = source.readString().orEmpty() | ||||
|     ) | ||||
|  | ||||
|     override fun describeContents() = 0 | ||||
|  | ||||
|     override fun writeToParcel(dest: Parcel, flags: Int) { | ||||
|         dest.writeString(title) | ||||
|         dest.writeString(content) | ||||
|         dest.writeString(date_published) | ||||
|         dest.writeString(lead_image_url) | ||||
|         dest.writeString(dek) | ||||
|         dest.writeString(url) | ||||
|         dest.writeString(domain) | ||||
|         dest.writeString(excerpt) | ||||
|         dest.writeInt(total_pages) | ||||
|         dest.writeInt(rendered_pages) | ||||
|         dest.writeString(next_page_url) | ||||
|     } | ||||
| } | ||||
| @@ -1,10 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.api.mercury | ||||
|  | ||||
| import retrofit2.Call | ||||
| import retrofit2.http.GET | ||||
| import retrofit2.http.Query | ||||
|  | ||||
| interface MercuryService { | ||||
|     @GET("parser.php") | ||||
|     fun parseUrl(@Query("link") link: String): Call<ParsedContent> | ||||
| } | ||||
| @@ -52,11 +52,13 @@ override fun doWork(): Result { | ||||
|  | ||||
|             repository.handleDBActions() | ||||
|  | ||||
|             val apiItems = repository.tryToCacheItemsAndGetNewOnes() | ||||
|             if (appSettingsService.isNotifyNewItemsEnabled()) { | ||||
|                 launch { | ||||
|                     handleNewItemsNotification(repository.tryToCacheItemsAndGetNewOnes(), notificationManager) | ||||
|                     handleNewItemsNotification(apiItems, notificationManager) | ||||
|                 } | ||||
|             } | ||||
|             apiItems.map { it.preloadImages(context) } | ||||
|         } | ||||
|     } | ||||
|     return Result.success() | ||||
| @@ -66,6 +68,7 @@ override fun doWork(): Result { | ||||
|         newItems: List<SelfossModel.Item>?, | ||||
|         notificationManager: NotificationManager | ||||
|     ) { | ||||
|         // TODO: Check if this coroutine is actually required | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|                 val apiItems = newItems.orEmpty() | ||||
|  | ||||
| @@ -102,7 +105,6 @@ override fun doWork(): Result { | ||||
|                         notificationManager.notify(2, newItemsNotification.build()) | ||||
|                     } | ||||
|                 } | ||||
|                 apiItems.map { it.preloadImages(context) } | ||||
|             Timer("", false).schedule(4000) { | ||||
|                 notificationManager.cancel(1) | ||||
|             } | ||||
|   | ||||
| @@ -8,6 +8,7 @@ import android.graphics.Typeface | ||||
| import android.graphics.drawable.ColorDrawable | ||||
| import android.net.Uri | ||||
| import android.os.Bundle | ||||
| import android.util.TypedValue | ||||
| import android.view.* | ||||
| import android.webkit.WebResourceResponse | ||||
| import android.webkit.WebSettings | ||||
| @@ -15,27 +16,21 @@ import android.webkit.WebView | ||||
| import android.webkit.WebViewClient | ||||
| import android.widget.Toast | ||||
| import androidx.appcompat.app.AlertDialog | ||||
| import androidx.browser.customtabs.CustomTabsIntent | ||||
| import androidx.core.content.res.ResourcesCompat | ||||
| import androidx.core.widget.NestedScrollView | ||||
| import androidx.fragment.app.Fragment | ||||
| import bou.amine.apps.readerforselfossv2.android.ImageActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.api.mercury.MercuryApi | ||||
| import bou.amine.apps.readerforselfossv2.android.api.mercury.ParsedContent | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.FragmentArticleBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.model.ParecelableItem | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toModel | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toParcelable | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.buildCustomTabsIntent | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.CustomTabActivityHelper | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.getBitmapInputStream | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrlInternalBrowser | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.shareLink | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.rest.MercuryApi | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import bou.amine.apps.readerforselfossv2.utils.getImages | ||||
| @@ -53,18 +48,15 @@ import org.kodein.di.DI | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.x.closestDI | ||||
| import org.kodein.di.instance | ||||
| import retrofit2.Call | ||||
| import retrofit2.Callback | ||||
| import retrofit2.Response | ||||
| import java.net.MalformedURLException | ||||
| import java.net.URL | ||||
| import java.util.* | ||||
| import java.util.concurrent.ExecutionException | ||||
|  | ||||
|  | ||||
| class ArticleFragment : Fragment(), DIAware { | ||||
|     private var fontSize: Int = 16 | ||||
|     private lateinit var item: SelfossModel.Item | ||||
|     private var mCustomTabActivityHelper: CustomTabActivityHelper? = null | ||||
|     private lateinit var url: String | ||||
|     private lateinit var contentText: String | ||||
|     private lateinit var contentSource: String | ||||
| @@ -72,7 +64,6 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|     private lateinit var contentTitle: String | ||||
|     private lateinit var allImages : ArrayList<String> | ||||
|     private lateinit var fab: FloatingActionButton | ||||
|     private lateinit var appColors: AppColors | ||||
|     private lateinit var textAlignment: String | ||||
|     private var _binding: FragmentArticleBinding? = null | ||||
|     private val binding get() = _binding!! | ||||
| @@ -86,16 +77,10 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|     private var font = "" | ||||
|     private var staticBar = false | ||||
|  | ||||
|     override fun onStop() { | ||||
|         super.onStop() | ||||
|         if (mCustomTabActivityHelper != null) { | ||||
|             mCustomTabActivityHelper!!.unbindCustomTabsService(activity) | ||||
|         } | ||||
|     } | ||||
|     private val mercuryApi : MercuryApi by instance() | ||||
|  | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         appColors = AppColors(requireActivity()) | ||||
|  | ||||
|         super.onCreate(savedInstanceState) | ||||
|  | ||||
|         val pi: ParecelableItem = requireArguments().getParcelable(ARG_ITEMS)!! | ||||
| @@ -115,7 +100,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|             contentText = item.content | ||||
|             contentTitle = item.title.getHtmlDecoded() | ||||
|             contentImage = item.getThumbnail(repository.baseUrl) | ||||
|             contentSource = item.sourceAndDateText(repository.dateUtils) | ||||
|             contentSource = item.sourceAndDateText() | ||||
|             allImages = item.getImages() | ||||
|  | ||||
|             fontSize = appSettingsService.getFontSize() | ||||
| @@ -136,25 +121,19 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|  | ||||
|             fab = binding.fab | ||||
|  | ||||
|             fab.backgroundTintList = ColorStateList.valueOf(appColors.colorAccent) | ||||
|             fab.backgroundTintList = ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             fab.rippleColor = appColors.colorAccentDark | ||||
|             fab.rippleColor = resources.getColor(R.color.colorAccentDark) | ||||
|  | ||||
|             val floatingToolbar: FloatingToolbar = binding.floatingToolbar | ||||
|             floatingToolbar.attachFab(fab) | ||||
|  | ||||
|             floatingToolbar.background = ColorDrawable(appColors.colorAccent) | ||||
|  | ||||
|             val customTabsIntent = requireActivity().buildCustomTabsIntent() | ||||
|             mCustomTabActivityHelper = CustomTabActivityHelper() | ||||
|             mCustomTabActivityHelper!!.bindCustomTabsService(activity) | ||||
|  | ||||
|             floatingToolbar.background = ColorDrawable(resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             floatingToolbar.setClickListener( | ||||
|                 object : FloatingToolbar.ItemClickListener { | ||||
|                     override fun onItemClick(item: MenuItem) { | ||||
|                         when (item.itemId) { | ||||
|                             R.id.more_action -> getContentFromMercury(customTabsIntent) | ||||
|                             R.id.share_action -> requireActivity().shareLink(url, contentTitle) | ||||
|                             R.id.open_action -> requireActivity().openInBrowserAsNewTask(this@ArticleFragment.item) | ||||
|                             R.id.unread_action -> if (context != null) { | ||||
| @@ -200,7 +179,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|             } | ||||
|  | ||||
|             if (contentText.isEmptyOrNullOrNullString()) { | ||||
|                 getContentFromMercury(customTabsIntent) | ||||
|                 getContentFromMercury() | ||||
|             } else { | ||||
|                 binding.titleView.text = contentTitle | ||||
|                 if (typeface != null) { | ||||
| @@ -266,96 +245,86 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun getContentFromMercury(customTabsIntent: CustomTabsIntent) { | ||||
|     private fun getContentFromMercury() { | ||||
|         if (repository.isNetworkAvailable()) { | ||||
|         binding.progressBar.visibility = View.VISIBLE | ||||
|         val parser = MercuryApi() | ||||
|             binding.progressBar.visibility = View.VISIBLE | ||||
|  | ||||
|             parser.parseUrl(url).enqueue( | ||||
|                 object : Callback<ParsedContent> { | ||||
|                     override fun onResponse( | ||||
|                         call: Call<ParsedContent>, | ||||
|                         response: Response<ParsedContent> | ||||
|                     ) { | ||||
|                         // TODO: clean all the following after finding the mercury content issue | ||||
|                         try { | ||||
|                             if (response.body() != null && response.body()!!.content != null && !response.body()!!.content.isNullOrEmpty()) { | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|                  val response = mercuryApi.query(url) | ||||
|                 if (response.success) { | ||||
|                     try { | ||||
|                         if (response.data != null && response.data!!.content != null && !response.data!!.content.isNullOrEmpty()) { | ||||
|                             try { | ||||
|                                 binding.titleView.text = response.data!!.title | ||||
|                                 if (typeface != null) { | ||||
|                                     binding.titleView.typeface = typeface | ||||
|                                 } | ||||
|                                 try { | ||||
|                                     binding.titleView.text = response.body()!!.title | ||||
|                                     if (typeface != null) { | ||||
|                                         binding.titleView.typeface = typeface | ||||
|                                     } | ||||
|                                     // Note: Mercury may return relative urls... If it does the url val will not be changed. | ||||
|                                     URL(response.data!!.url) | ||||
|                                     url = response.data!!.url | ||||
|                                 } catch (e: MalformedURLException) { | ||||
|                                     // Mercury returned a relative url. We do nothing. | ||||
|                                 } | ||||
|                             } catch (e: Exception) { | ||||
|                             } | ||||
|  | ||||
|                             try { | ||||
|                                 contentText = response.data!!.content.orEmpty() | ||||
|                                 htmlToWebview() | ||||
|                             } catch (e: Exception) { | ||||
|                             } | ||||
|  | ||||
|                             try { | ||||
|                                 if (response.data!!.lead_image_url != null && !response.data!!.lead_image_url.isNullOrEmpty() && context != null) { | ||||
|                                     binding.imageView.visibility = View.VISIBLE | ||||
|                                     try { | ||||
|                                         // Note: Mercury may return relative urls... If it does the url val will not be changed. | ||||
|                                         URL(response.body()!!.url) | ||||
|                                         url = response.body()!!.url | ||||
|                                     } catch (e: MalformedURLException) { | ||||
|                                         // Mercury returned a relative url. We do nothing. | ||||
|                                         Glide | ||||
|                                             .with(requireContext()) | ||||
|                                             .asBitmap() | ||||
|                                             .load( | ||||
|                                                 response.data!!.lead_image_url.orEmpty() | ||||
|                                             ) | ||||
|                                             .apply(RequestOptions.fitCenterTransform()) | ||||
|                                             .into(binding.imageView) | ||||
|                                     } catch (e: Exception) { | ||||
|                                     } | ||||
|                                 } catch (e: Exception) { | ||||
|                                 } else { | ||||
|                                     binding.imageView.visibility = View.GONE | ||||
|                                 } | ||||
|  | ||||
|                                 try { | ||||
|                                     contentText = response.body()!!.content.orEmpty() | ||||
|                                     htmlToWebview() | ||||
|                                 } catch (e: Exception) { | ||||
|                                 } | ||||
|  | ||||
|                                 try { | ||||
|                                     if (response.body()!!.lead_image_url != null && !response.body()!!.lead_image_url.isNullOrEmpty() && context != null) { | ||||
|                                         binding.imageView.visibility = View.VISIBLE | ||||
|                                         try { | ||||
|                                             Glide | ||||
|                                                 .with(requireContext()) | ||||
|                                                 .asBitmap() | ||||
|                                                 .load( | ||||
|                                                     response.body()!!.lead_image_url.orEmpty() | ||||
|                                                 ) | ||||
|                                                 .apply(RequestOptions.fitCenterTransform()) | ||||
|                                                 .into(binding.imageView) | ||||
|                                         } catch (e: Exception) { | ||||
|                                         } | ||||
|                                     } else { | ||||
|                                         binding.imageView.visibility = View.GONE | ||||
|                                     } | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|                                 } | ||||
|  | ||||
|                                 try { | ||||
|                                     binding.nestedScrollView.scrollTo(0, 0) | ||||
|  | ||||
|                                     binding.progressBar.visibility = View.GONE | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|                                 } | ||||
|                             } else { | ||||
|                                 try { | ||||
|                                     openInBrowserAfterFailing(customTabsIntent) | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|                             } catch (e: Exception) { | ||||
|                                 if (context != null) { | ||||
|                                 } | ||||
|                             } | ||||
|                         } catch (e: Exception) { | ||||
|                             if (context != null) { | ||||
|  | ||||
|                             try { | ||||
|                                 binding.nestedScrollView.scrollTo(0, 0) | ||||
|  | ||||
|                                 binding.progressBar.visibility = View.GONE | ||||
|                             } catch (e: Exception) { | ||||
|                                 if (context != null) { | ||||
|                                 } | ||||
|                             } | ||||
|                         } else { | ||||
|                             try { | ||||
|                                 openInBrowserAfterFailing() | ||||
|                             } catch (e: Exception) { | ||||
|                                 if (context != null) { | ||||
|                                 } | ||||
|                             } | ||||
|                         } | ||||
|                     } catch (e: Exception) { | ||||
|                         if (context != null) { | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     override fun onFailure( | ||||
|                         call: Call<ParsedContent>, | ||||
|                         t: Throwable | ||||
|                     ) = openInBrowserAfterFailing(customTabsIntent) | ||||
|                 } else { | ||||
|                     openInBrowserAfterFailing() | ||||
|                 } | ||||
|             ) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun htmlToWebview() { | ||||
|         val stringColor = String.format("#%06X", 0xFFFFFF and appColors.colorAccent) | ||||
|  | ||||
|         val attrs: IntArray = intArrayOf(android.R.attr.fontFamily) | ||||
|         val a: TypedArray = requireContext().obtainStyledAttributes(resId, attrs) | ||||
| @@ -364,18 +333,18 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         binding.webcontent.settings.standardFontFamily = a.getString(0) | ||||
|         binding.webcontent.visibility = View.VISIBLE | ||||
|  | ||||
|         // TODO: Set the color strings programmatically | ||||
|         val (stringTextColor, stringBackgroundColor) = if (appColors.isDarkTheme) { | ||||
|             Pair("#FFFFFF", "#303030") | ||||
|         } else { | ||||
|             Pair("#212121", "#FAFAFA") | ||||
|         } | ||||
|         val colorOnSurface = TypedValue() | ||||
|         requireContext().theme.resolveAttribute(R.attr.colorOnSurface, colorOnSurface, true) | ||||
|  | ||||
|         val colorSurface = TypedValue() | ||||
|         requireContext().theme.resolveAttribute(R.attr.colorSurface, colorSurface, true) | ||||
|  | ||||
|         binding.webcontent.settings.useWideViewPort = true | ||||
|         binding.webcontent.settings.loadWithOverviewMode = true | ||||
|         binding.webcontent.settings.javaScriptEnabled = false | ||||
|  | ||||
|         binding.webcontent.webViewClient = object : WebViewClient() { | ||||
|             @Deprecated("Deprecated in Java") | ||||
|             override fun shouldOverrideUrlLoading(view: WebView?, url : String): Boolean { | ||||
|                 if (binding.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||
|                     requireContext().startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) | ||||
| @@ -383,6 +352,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 return true | ||||
|             } | ||||
|  | ||||
|             @Deprecated("Deprecated in Java") | ||||
|             override fun shouldInterceptRequest(view: WebView?, url: String): WebResourceResponse? { | ||||
|                 val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||
|                 if (url.lowercase(Locale.US).contains(".jpg") || url.lowercase(Locale.US).contains(".jpeg")) { | ||||
| @@ -409,7 +379,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         } | ||||
|  | ||||
|         val gestureDetector = GestureDetector(activity, object : GestureDetector.SimpleOnGestureListener() { | ||||
|             override fun onSingleTapUp(e: MotionEvent?): Boolean { | ||||
|             override fun onSingleTapUp(e: MotionEvent): Boolean { | ||||
|                 return performClick() | ||||
|             } | ||||
|         }) | ||||
| @@ -430,6 +400,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         val fontName =  when (font) { | ||||
|             getString(R.string.open_sans_font_id) -> "Open Sans" | ||||
|             getString(R.string.roboto_font_id) -> "Roboto" | ||||
|             getString(R.string.source_code_pro_font_id) -> "Source Code Pro" | ||||
|             else -> "" | ||||
|         } | ||||
|  | ||||
| @@ -458,10 +429,10 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 |        max-width: 100%; | ||||
|                 |      } | ||||
|                 |      a { | ||||
|                 |        color: $stringColor !important; | ||||
|                 |        color: ${String.format("#%06X", 0xFFFFFF and resources.getColor(R.color.colorAccent))} !important; | ||||
|                 |      } | ||||
|                 |      *:not(a) { | ||||
|                 |        color: $stringTextColor; | ||||
|                 |        color: ${String.format("#%06X", 0xFFFFFF and colorOnSurface.data)}; | ||||
|                 |      } | ||||
|                 |      * { | ||||
|                 |        font-size: ${fontSize}px; | ||||
| @@ -469,11 +440,11 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 |        word-break: break-word; | ||||
|                 |        overflow:hidden; | ||||
|                 |        line-height: 1.5em; | ||||
|                 |        background-color: $stringBackgroundColor; | ||||
|                 |        background-color: ${String.format("#%06X", 0xFFFFFF and colorSurface.data)}; | ||||
|                 |      } | ||||
|                 |      body, html { | ||||
|                 |        background-color: $stringBackgroundColor !important; | ||||
|                 |        border-color: $stringBackgroundColor  !important; | ||||
|                 |        background-color: ${String.format("#%06X", 0xFFFFFF and colorSurface.data)} !important; | ||||
|                 |        border-color: ${String.format("#%06X", 0xFFFFFF and colorSurface.data)}  !important; | ||||
|                 |        padding: 0 !important; | ||||
|                 |        margin: 0 !important; | ||||
|                 |      } | ||||
| @@ -483,7 +454,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 |      pre, code { | ||||
|                 |        white-space: pre-wrap; | ||||
|                 |        width:100%; | ||||
|                 |        background-color: $stringBackgroundColor; | ||||
|                 |        background-color: ${String.format("#%06X", 0xFFFFFF and colorSurface.data)}; | ||||
|                 |      } | ||||
|                 |   </style> | ||||
|                 |   $fontLinkAndStyle | ||||
| @@ -507,13 +478,9 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         binding.nestedScrollView.smoothScrollBy(0, -height/2) | ||||
|     } | ||||
|  | ||||
|     private fun openInBrowserAfterFailing(customTabsIntent: CustomTabsIntent) { | ||||
|     private fun openInBrowserAfterFailing() { | ||||
|         binding.progressBar.visibility = View.GONE | ||||
|         requireActivity().openItemUrlInternalBrowser( | ||||
|                 url, | ||||
|                 customTabsIntent, | ||||
|                 requireActivity() | ||||
|         ) | ||||
|         requireActivity().openInBrowserAsNewTask(this@ArticleFragment.item) | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|   | ||||
| @@ -3,7 +3,6 @@ package bou.amine.apps.readerforselfossv2.android.model | ||||
| import android.os.Parcel | ||||
| import android.os.Parcelable | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import com.google.gson.annotations.SerializedName | ||||
|  | ||||
| fun SelfossModel.Item.toParcelable() : ParecelableItem = | ||||
|     ParecelableItem( | ||||
| @@ -34,17 +33,17 @@ fun ParecelableItem.toModel() : SelfossModel.Item = | ||||
|         this.tags.split(",") | ||||
|     ) | ||||
| data class ParecelableItem( | ||||
|     @SerializedName("id") val id: Int, | ||||
|     @SerializedName("datetime") val datetime: String, | ||||
|     @SerializedName("title") val title: String, | ||||
|     @SerializedName("content") val content: String, | ||||
|     @SerializedName("unread") var unread: Boolean, | ||||
|     @SerializedName("starred") var starred: Boolean, | ||||
|     @SerializedName("thumbnail") val thumbnail: String?, | ||||
|     @SerializedName("icon") val icon: String?, | ||||
|     @SerializedName("link") val link: String, | ||||
|     @SerializedName("sourcetitle") val sourcetitle: String, | ||||
|     @SerializedName("tags") val tags: String | ||||
|     val id: Int, | ||||
|     val datetime: String, | ||||
|     val title: String, | ||||
|     val content: String, | ||||
|     var unread: Boolean, | ||||
|     var starred: Boolean, | ||||
|     val thumbnail: String?, | ||||
|     val icon: String?, | ||||
|     val link: String, | ||||
|     val sourcetitle: String, | ||||
|     val tags: String | ||||
| ) : Parcelable { | ||||
|  | ||||
|     companion object { | ||||
|   | ||||
| @@ -7,22 +7,16 @@ import android.text.Editable | ||||
| import android.text.InputFilter | ||||
| import android.text.InputType | ||||
| import android.text.TextWatcher | ||||
| import android.view.Menu | ||||
| import android.view.MenuInflater | ||||
| import android.view.MenuItem | ||||
| import android.widget.Toast | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.appcompat.app.AppCompatDelegate | ||||
| import androidx.core.widget.addTextChangedListener | ||||
| import androidx.preference.EditTextPreference | ||||
| import androidx.preference.Preference | ||||
| import androidx.preference.PreferenceFragmentCompat | ||||
| import androidx.preference.PreferenceManager | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySettingsBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.AppColors | ||||
| import bou.amine.apps.readerforselfossv2.android.themes.Toppings | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import com.ftinc.scoop.Scoop | ||||
|  | ||||
| private const val TITLE_TAG = "settingsActivityTitle" | ||||
|  | ||||
| @@ -31,15 +25,8 @@ class SettingsActivity : AppCompatActivity(), | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         if (PreferenceManager.getDefaultSharedPreferences(this).getBoolean("dark_theme", false)) { | ||||
|             setTheme(R.style.NoBarDark) | ||||
|         } | ||||
|         val binding = ActivitySettingsBinding.inflate(layoutInflater) | ||||
|  | ||||
|         val scoop = Scoop.getInstance() | ||||
|         scoop.bind(this, Toppings.PRIMARY.value, binding.toolbar) | ||||
|         scoop.bindStatusBar(this, Toppings.PRIMARY_DARK.value) | ||||
|  | ||||
|         setContentView(binding.root) | ||||
|         if (savedInstanceState == null) { | ||||
|             supportFragmentManager | ||||
| @@ -69,11 +56,13 @@ class SettingsActivity : AppCompatActivity(), | ||||
|     } | ||||
|  | ||||
|     override fun onSupportNavigateUp(): Boolean { | ||||
|         if (supportFragmentManager.popBackStackImmediate()) { | ||||
|         return if (supportFragmentManager.popBackStackImmediate()) { | ||||
|             supportActionBar?.title = getText(R.string.title_activity_settings) | ||||
|             return true | ||||
|             false | ||||
|         } else { | ||||
|             super.onBackPressed() | ||||
|             true | ||||
|         } | ||||
|         return super.onSupportNavigateUp() | ||||
|     } | ||||
|  | ||||
|     override fun onPreferenceStartFragment( | ||||
| @@ -167,21 +156,11 @@ class SettingsActivity : AppCompatActivity(), | ||||
|     class ThemePreferenceFragment : PreferenceFragmentCompat() { | ||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||
|             setPreferencesFromResource(R.xml.pref_theme, rootKey) | ||||
|             setHasOptionsMenu(true) | ||||
|         } | ||||
|  | ||||
|         override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) { | ||||
|             super.onCreateOptionsMenu(menu, inflater) | ||||
|             inflater.inflate(R.menu.settings_theme, menu) | ||||
|         } | ||||
|  | ||||
|         override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||
|             val id = item.itemId | ||||
|             if (id == R.id.clear) { | ||||
|                 AppColors.resetColors() | ||||
|                 requireActivity().recreate() | ||||
|             preferenceManager.findPreference<Preference>("currentMode")?.onPreferenceChangeListener = Preference.OnPreferenceChangeListener { _, newValue -> | ||||
|                 AppCompatDelegate.setDefaultNightMode(newValue.toString().toInt()) // ListPreference Only takes string-arrays ¯\_(ツ)_/¯ | ||||
|                 true | ||||
|             } | ||||
|             return super.onOptionsItemSelected(item) | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -216,11 +195,4 @@ class SettingsActivity : AppCompatActivity(), | ||||
|             setPreferencesFromResource(R.xml.pref_experimental, rootKey) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||
|         when (item.itemId) { | ||||
|             android.R.id.home -> super.onBackPressed() | ||||
|         } | ||||
|         return super.onOptionsItemSelected(item) | ||||
|     } | ||||
| } | ||||
| @@ -1,72 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.themes | ||||
|  | ||||
| import android.app.Activity | ||||
| import androidx.annotation.ColorInt | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import com.russhwolf.settings.Settings | ||||
|  | ||||
| class AppColors(a: Activity) { | ||||
|  | ||||
|     @ColorInt val colorPrimary: Int | ||||
|     @ColorInt val colorPrimaryDark: Int | ||||
|     @ColorInt val colorAccent: Int | ||||
|     @ColorInt val colorAccentDark: Int | ||||
|     @ColorInt val colorBackground: Int | ||||
|     @ColorInt val textColor: Int | ||||
|     val isDarkTheme: Boolean | ||||
|  | ||||
|     init { | ||||
|         val settings = Settings() | ||||
|  | ||||
|         colorPrimary = | ||||
|                 settings.getInt( | ||||
|                     "color_primary", | ||||
|                     a.resources.getColor(R.color.colorPrimary) | ||||
|                 ) | ||||
|         colorPrimaryDark = | ||||
|                 settings.getInt( | ||||
|                     "color_primary_dark", | ||||
|                     a.resources.getColor(R.color.colorPrimaryDark) | ||||
|                 ) | ||||
|         colorAccent = | ||||
|                 settings.getInt( | ||||
|                     "color_accent", | ||||
|                     a.resources.getColor(R.color.colorAccent) | ||||
|                 ) | ||||
|         colorAccentDark = | ||||
|                 settings.getInt( | ||||
|                     "color_accent_dark", | ||||
|                     a.resources.getColor(R.color.colorAccentDark) | ||||
|                 ) | ||||
|         isDarkTheme = | ||||
|                 settings.getBoolean( | ||||
|                     "dark_theme", | ||||
|                     false | ||||
|                 ) | ||||
|  | ||||
|         colorBackground = if (isDarkTheme) { | ||||
|             a.setTheme(R.style.NoBarDark) | ||||
|             a.resources.getColor(R.color.darkBackground) | ||||
|         } else { | ||||
|             a.setTheme(R.style.NoBar) | ||||
|             a.resources.getColor(R.color.grey_50) | ||||
|         } | ||||
|  | ||||
|         textColor = if (isDarkTheme) { | ||||
|             a.resources.getColor(R.color.white) | ||||
|         } else { | ||||
|             a.resources.getColor(R.color.grey_900) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         fun resetColors() { | ||||
|             val settings = Settings() | ||||
|             settings.remove("color_primary") | ||||
|             settings.remove("color_primary_dark") | ||||
|             settings.remove("color_accent") | ||||
|             settings.remove("color_accent_dark") | ||||
|             settings.remove("dark_theme") | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.themes | ||||
|  | ||||
| enum class Toppings(val value: Int) { | ||||
|     PRIMARY(1), | ||||
|     PRIMARY_DARK(2), | ||||
|     ACCENT(3), | ||||
|     ACCENT_DARK(4) | ||||
| } | ||||
| @@ -1,43 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils | ||||
|  | ||||
| import okhttp3.OkHttpClient | ||||
| import java.security.cert.CertificateException | ||||
| import java.security.cert.X509Certificate | ||||
| import javax.net.ssl.SSLContext | ||||
| import javax.net.ssl.TrustManager | ||||
| import javax.net.ssl.X509TrustManager | ||||
|  | ||||
| fun getUnsafeHttpClient(): OkHttpClient.Builder = | ||||
|     try { | ||||
|         // Create a trust manager that does not validate certificate chains | ||||
|         val trustAllCerts = arrayOf<TrustManager>(object : X509TrustManager { | ||||
|             override fun getAcceptedIssuers(): Array<X509Certificate> = | ||||
|                 arrayOf() | ||||
|  | ||||
|             @Throws(CertificateException::class) | ||||
|             override fun checkClientTrusted( | ||||
|                 chain: Array<java.security.cert.X509Certificate>, | ||||
|                 authType: String | ||||
|             ) { | ||||
|             } | ||||
|  | ||||
|             @Throws(CertificateException::class) | ||||
|             override fun checkServerTrusted( | ||||
|                 chain: Array<java.security.cert.X509Certificate>, | ||||
|                 authType: String | ||||
|             ) { | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|         // Install the all-trusting trust manager | ||||
|         val sslContext = SSLContext.getInstance("SSL") | ||||
|         sslContext.init(null, trustAllCerts, java.security.SecureRandom()) | ||||
|  | ||||
|         val sslSocketFactory = sslContext.socketFactory | ||||
|  | ||||
|         OkHttpClient.Builder() | ||||
|             .sslSocketFactory(sslSocketFactory, trustAllCerts[0] as X509TrustManager) | ||||
|             .hostnameVerifier { _, _ -> true } | ||||
|     } catch (e: Exception) { | ||||
|         throw RuntimeException(e) | ||||
|     } | ||||
| @@ -15,109 +15,16 @@ import android.view.MotionEvent | ||||
| import android.view.View | ||||
| import android.widget.TextView | ||||
| import android.widget.Toast | ||||
| import androidx.browser.customtabs.CustomTabsIntent | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.ReaderActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.CustomTabActivityHelper | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | ||||
| import okhttp3.HttpUrl.Companion.toHttpUrlOrNull | ||||
|  | ||||
| fun Context.buildCustomTabsIntent(): CustomTabsIntent { | ||||
|  | ||||
|     val actionIntent = Intent(Intent.ACTION_SEND) | ||||
|     actionIntent.type = "text/plain" | ||||
|     val pflags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||
|         PendingIntent.FLAG_IMMUTABLE | ||||
|     } else { | ||||
|         0 | ||||
|     } | ||||
|     val createPendingShareIntent: PendingIntent = PendingIntent.getActivity( | ||||
|             this, | ||||
|             0, | ||||
|             actionIntent, | ||||
|             pflags | ||||
|         ) | ||||
|  | ||||
|     val intentBuilder = CustomTabsIntent.Builder() | ||||
|  | ||||
|     // TODO: change to primary when it's possible to customize custom tabs title color | ||||
|     //intentBuilder.setToolbarColor(c.getResources().getColor(R.color.colorPrimary)); | ||||
|     intentBuilder.setToolbarColor(resources.getColor(R.color.colorAccentDark)) | ||||
|     intentBuilder.setShowTitle(true) | ||||
|  | ||||
|  | ||||
|     intentBuilder.setStartAnimations( | ||||
|         this, | ||||
|         R.anim.slide_in_right, | ||||
|         R.anim.slide_out_left | ||||
|     ) | ||||
|     intentBuilder.setExitAnimations( | ||||
|         this, | ||||
|         android.R.anim.slide_in_left, | ||||
|         android.R.anim.slide_out_right | ||||
|     ) | ||||
|  | ||||
|     val closeicon = BitmapFactory.decodeResource(resources, R.drawable.ic_close_white_24dp) | ||||
|     intentBuilder.setCloseButtonIcon(closeicon) | ||||
|  | ||||
|     val shareLabel = this.getString(R.string.label_share) | ||||
|     val icon = BitmapFactory.decodeResource( | ||||
|         resources, | ||||
|         R.drawable.ic_share_white_24dp | ||||
|     ) | ||||
|     intentBuilder.setActionButton(icon, shareLabel, createPendingShareIntent) | ||||
|  | ||||
|     return intentBuilder.build() | ||||
| } | ||||
|  | ||||
| fun Context.openItemUrlInternally( | ||||
|     allItems: ArrayList<SelfossModel.Item>, | ||||
|     currentItem: Int, | ||||
|     linkDecoded: String, | ||||
|     customTabsIntent: CustomTabsIntent, | ||||
|     articleViewer: Boolean, | ||||
|     app: Activity | ||||
| ) { | ||||
|     if (articleViewer) { | ||||
|         ReaderActivity.allItems = allItems | ||||
|         val intent = Intent(this, ReaderActivity::class.java) | ||||
|         intent.putExtra("currentItem", currentItem) | ||||
|         app.startActivity(intent) | ||||
|     } else { | ||||
|         this.openItemUrlInternalBrowser( | ||||
|                 linkDecoded, | ||||
|                 customTabsIntent, | ||||
|                 app) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun Context.openItemUrlInternalBrowser( | ||||
|         linkDecoded: String, | ||||
|         customTabsIntent: CustomTabsIntent, | ||||
|         app: Activity | ||||
| ) { | ||||
|     try { | ||||
|         CustomTabActivityHelper.openCustomTab( | ||||
|                 app, | ||||
|                 customTabsIntent, | ||||
|                 Uri.parse(linkDecoded) | ||||
|         ) { _, uri -> | ||||
|             val intent = Intent(Intent.ACTION_VIEW, uri) | ||||
|             intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|             startActivity(intent) | ||||
|         } | ||||
|     } catch (e: Exception) { | ||||
|         openInBrowser(linkDecoded, app) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun Context.openItemUrl( | ||||
|     allItems: ArrayList<SelfossModel.Item>, | ||||
|     currentItem: Int, | ||||
|     linkDecoded: String, | ||||
|     customTabsIntent: CustomTabsIntent, | ||||
|     internalBrowser: Boolean, | ||||
|     articleViewer: Boolean, | ||||
|     app: Activity | ||||
| ) { | ||||
| @@ -129,41 +36,24 @@ fun Context.openItemUrl( | ||||
|             Toast.LENGTH_LONG | ||||
|         ).show() | ||||
|     } else { | ||||
|         if (!internalBrowser) { | ||||
|             openInBrowser(linkDecoded, app) | ||||
|         } else if (articleViewer) { | ||||
|             this.openItemUrlInternally( | ||||
|                 allItems, | ||||
|                 currentItem, | ||||
|                 linkDecoded, | ||||
|                 customTabsIntent, | ||||
|                 articleViewer, | ||||
|                 app | ||||
|             ) | ||||
|         if (articleViewer) { | ||||
|             ReaderActivity.allItems = allItems | ||||
|             val intent = Intent(this, ReaderActivity::class.java) | ||||
|             intent.putExtra("currentItem", currentItem) | ||||
|             app.startActivity(intent) | ||||
|         } else { | ||||
|             this.openItemUrlInternalBrowser( | ||||
|                     linkDecoded, | ||||
|                     customTabsIntent, | ||||
|                     app | ||||
|             ) | ||||
|             val intent = Intent(Intent.ACTION_VIEW) | ||||
|             intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK | ||||
|             intent.data = Uri.parse(linkDecoded.toStringUriWithHttp()) | ||||
|             startActivity(intent) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| private fun openInBrowser(linkDecoded: String, app: Activity) { | ||||
|     val intent = Intent(Intent.ACTION_VIEW) | ||||
|     intent.data = Uri.parse(linkDecoded) | ||||
|     try { | ||||
|         app.startActivity(intent) | ||||
|     } catch (e: ActivityNotFoundException) { | ||||
|         Toast.makeText(app.baseContext, e.message, Toast.LENGTH_LONG).show() | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun String.isUrlValid(): Boolean = | ||||
|     this.toHttpUrlOrNull() != null && Patterns.WEB_URL.matcher(this).matches() | ||||
|  | ||||
| fun String.isBaseUrlValid(ctx: Context): Boolean { | ||||
| fun String.isBaseUrlInvalid(): Boolean { | ||||
|     val baseUrl = this.toHttpUrlOrNull() | ||||
|     var existsAndEndsWithSlash = false | ||||
|     if (baseUrl != null) { | ||||
| @@ -171,7 +61,7 @@ fun String.isBaseUrlValid(ctx: Context): Boolean { | ||||
|         existsAndEndsWithSlash = "" == pathSegments[pathSegments.size - 1] | ||||
|     } | ||||
|  | ||||
|     return Patterns.WEB_URL.matcher(this).matches() && existsAndEndsWithSlash | ||||
|     return !(Patterns.WEB_URL.matcher(this).matches() && existsAndEndsWithSlash) | ||||
| } | ||||
|  | ||||
| fun Context.openInBrowserAsNewTask(i: SelfossModel.Item) { | ||||
| @@ -181,7 +71,7 @@ fun Context.openInBrowserAsNewTask(i: SelfossModel.Item) { | ||||
|     startActivity(intent) | ||||
| } | ||||
|  | ||||
| class LinkOnTouchListener: View.OnTouchListener { | ||||
| class LinkOnTouchListener : View.OnTouchListener { | ||||
|     override fun onTouch(v: View?, event: MotionEvent?): Boolean { | ||||
|         var ret = false | ||||
|         val widget: TextView = v as TextView | ||||
| @@ -191,7 +81,8 @@ class LinkOnTouchListener: View.OnTouchListener { | ||||
|         val action = event!!.action | ||||
|  | ||||
|         if (action == MotionEvent.ACTION_UP || | ||||
|             action == MotionEvent.ACTION_DOWN) { | ||||
|             action == MotionEvent.ACTION_DOWN | ||||
|         ) { | ||||
|             var x: Float = event.x | ||||
|             var y: Float = event.y | ||||
|  | ||||
|   | ||||
| @@ -1,153 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs; | ||||
|  | ||||
|  | ||||
| import android.app.Activity; | ||||
| import android.net.Uri; | ||||
| import android.os.Bundle; | ||||
| import androidx.browser.customtabs.CustomTabsClient; | ||||
| import androidx.browser.customtabs.CustomTabsIntent; | ||||
| import androidx.browser.customtabs.CustomTabsServiceConnection; | ||||
| import androidx.browser.customtabs.CustomTabsSession; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * This is a helper class to manage the connection to the Custom Tabs Service. | ||||
|  */ | ||||
| public class CustomTabActivityHelper implements ServiceConnectionCallback { | ||||
|     private CustomTabsSession mCustomTabsSession; | ||||
|     private CustomTabsClient mClient; | ||||
|     private CustomTabsServiceConnection mConnection; | ||||
|     private ConnectionCallback mConnectionCallback; | ||||
|  | ||||
|     /** | ||||
|      * Opens the URL on a Custom Tab if possible. Otherwise fallsback to opening it on a WebView. | ||||
|      * | ||||
|      * @param activity         The host activity. | ||||
|      * @param customTabsIntent a CustomTabsIntent to be used if Custom Tabs is available. | ||||
|      * @param uri              the Uri to be opened. | ||||
|      * @param fallback         a CustomTabFallback to be used if Custom Tabs is not available. | ||||
|      */ | ||||
|     public static void openCustomTab(Activity activity, | ||||
|                                      CustomTabsIntent customTabsIntent, | ||||
|                                      Uri uri, | ||||
|                                      CustomTabFallback fallback) { | ||||
|         String packageName = CustomTabsHelper.getPackageNameToUse(activity); | ||||
|  | ||||
|         //If we cant find a package name, it means theres no browser that supports | ||||
|         //Chrome Custom Tabs installed. So, we fallback to the webview | ||||
|         if (packageName == null) { | ||||
|             if (fallback != null) { | ||||
|                 fallback.openUri(activity, uri); | ||||
|             } | ||||
|         } else { | ||||
|             customTabsIntent.intent.setPackage(packageName); | ||||
|             customTabsIntent.launchUrl(activity, uri); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Unbinds the Activity from the Custom Tabs Service. | ||||
|      * | ||||
|      * @param activity the activity that is connected to the service. | ||||
|      */ | ||||
|     public void unbindCustomTabsService(Activity activity) { | ||||
|         if (mConnection == null) return; | ||||
|         activity.unbindService(mConnection); | ||||
|         mClient = null; | ||||
|         mCustomTabsSession = null; | ||||
|         mConnection = null; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Creates or retrieves an exiting CustomTabsSession. | ||||
|      * | ||||
|      * @return a CustomTabsSession. | ||||
|      */ | ||||
|     public CustomTabsSession getSession() { | ||||
|         if (mClient == null) { | ||||
|             mCustomTabsSession = null; | ||||
|         } else if (mCustomTabsSession == null) { | ||||
|             mCustomTabsSession = mClient.newSession(null); | ||||
|         } | ||||
|         return mCustomTabsSession; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Register a Callback to be called when connected or disconnected from the Custom Tabs Service. | ||||
|      * | ||||
|      * @param connectionCallback | ||||
|      */ | ||||
|     public void setConnectionCallback(ConnectionCallback connectionCallback) { | ||||
|         this.mConnectionCallback = connectionCallback; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Binds the Activity to the Custom Tabs Service. | ||||
|      * | ||||
|      * @param activity the activity to be binded to the service. | ||||
|      */ | ||||
|     public void bindCustomTabsService(Activity activity) { | ||||
|         if (mClient != null) return; | ||||
|  | ||||
|         String packageName = CustomTabsHelper.getPackageNameToUse(activity); | ||||
|         if (packageName == null) return; | ||||
|  | ||||
|         mConnection = new ServiceConnection(this); | ||||
|         CustomTabsClient.bindCustomTabsService(activity, packageName, mConnection); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return true if call to mayLaunchUrl was accepted. | ||||
|      * @see {@link CustomTabsSession#mayLaunchUrl(Uri, Bundle, List)}. | ||||
|      */ | ||||
|     public boolean mayLaunchUrl(Uri uri, Bundle extras, List<Bundle> otherLikelyBundles) { | ||||
|         if (mClient == null) return false; | ||||
|  | ||||
|         CustomTabsSession session = getSession(); | ||||
|         return session != null && session.mayLaunchUrl(uri, extras, otherLikelyBundles); | ||||
|  | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onServiceConnected(CustomTabsClient client) { | ||||
|         mClient = client; | ||||
|         mClient.warmup(0L); | ||||
|         if (mConnectionCallback != null) mConnectionCallback.onCustomTabsConnected(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onServiceDisconnected() { | ||||
|         mClient = null; | ||||
|         mCustomTabsSession = null; | ||||
|         if (mConnectionCallback != null) mConnectionCallback.onCustomTabsDisconnected(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * A Callback for when the service is connected or disconnected. Use those callbacks to | ||||
|      * handle UI changes when the service is connected or disconnected. | ||||
|      */ | ||||
|     public interface ConnectionCallback { | ||||
|         /** | ||||
|          * Called when the service is connected. | ||||
|          */ | ||||
|         void onCustomTabsConnected(); | ||||
|  | ||||
|         /** | ||||
|          * Called when the service is disconnected. | ||||
|          */ | ||||
|         void onCustomTabsDisconnected(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * To be used as a fallback to open the Uri when Custom Tabs is not available. | ||||
|      */ | ||||
|     public interface CustomTabFallback { | ||||
|         /** | ||||
|          * @param activity The Activity that wants to open the Uri. | ||||
|          * @param uri      The uri to be opened by the fallback. | ||||
|          */ | ||||
|         void openUri(Activity activity, Uri uri); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,129 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs; | ||||
|  | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.content.Intent; | ||||
| import android.content.IntentFilter; | ||||
| import android.content.pm.PackageManager; | ||||
| import android.content.pm.ResolveInfo; | ||||
| import android.net.Uri; | ||||
| import android.text.TextUtils; | ||||
| import android.util.Log; | ||||
| import androidx.browser.customtabs.CustomTabsService; | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.customtabs.helpers.KeepAliveService; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
|  | ||||
| @SuppressWarnings("ALL") | ||||
| class CustomTabsHelper { | ||||
|     private static final String TAG = "CustomTabsHelper"; | ||||
|     private static final String STABLE_PACKAGE = "com.android.chrome"; | ||||
|     private static final String BETA_PACKAGE = "com.chrome.beta"; | ||||
|     private static final String DEV_PACKAGE = "com.chrome.dev"; | ||||
|     private static final String LOCAL_PACKAGE = "com.google.android.apps.chrome"; | ||||
|     private static final String EXTRA_CUSTOM_TABS_KEEP_ALIVE = | ||||
|             "android.support.customtabs.extra.KEEP_ALIVE"; | ||||
|  | ||||
|     private static String sPackageNameToUse; | ||||
|  | ||||
|     private CustomTabsHelper() { | ||||
|     } | ||||
|  | ||||
|     public static void addKeepAliveExtra(Context context, Intent intent) { | ||||
|         Intent keepAliveIntent = new Intent().setClassName( | ||||
|                 context.getPackageName(), KeepAliveService.class.getCanonicalName()); | ||||
|         intent.putExtra(EXTRA_CUSTOM_TABS_KEEP_ALIVE, keepAliveIntent); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Goes through all apps that handle VIEW intents and have a warmup service. Picks | ||||
|      * the one chosen by the user if there is one, otherwise makes a best effort to return a | ||||
|      * valid package name. | ||||
|      * <p> | ||||
|      * This is <strong>not</strong> threadsafe. | ||||
|      * | ||||
|      * @param context {@link Context} to use for accessing {@link PackageManager}. | ||||
|      * @return The package name recommended to use for connecting to custom tabs related components. | ||||
|      */ | ||||
|     public static String getPackageNameToUse(Context context) { | ||||
|         if (sPackageNameToUse != null) return sPackageNameToUse; | ||||
|  | ||||
|         PackageManager pm = context.getPackageManager(); | ||||
|         // Get default VIEW intent handler. | ||||
|         Intent activityIntent = new Intent(Intent.ACTION_VIEW, Uri.parse("http://www.example.com")); | ||||
|         ResolveInfo defaultViewHandlerInfo = pm.resolveActivity(activityIntent, 0); | ||||
|         String defaultViewHandlerPackageName = null; | ||||
|         if (defaultViewHandlerInfo != null) { | ||||
|             defaultViewHandlerPackageName = defaultViewHandlerInfo.activityInfo.packageName; | ||||
|         } | ||||
|  | ||||
|         // Get all apps that can handle VIEW intents. | ||||
|         List<ResolveInfo> resolvedActivityList = pm.queryIntentActivities(activityIntent, 0); | ||||
|         List<String> packagesSupportingCustomTabs = new ArrayList<>(); | ||||
|         for (ResolveInfo info : resolvedActivityList) { | ||||
|             Intent serviceIntent = new Intent(); | ||||
|             serviceIntent.setAction(CustomTabsService.ACTION_CUSTOM_TABS_CONNECTION); | ||||
|             serviceIntent.setPackage(info.activityInfo.packageName); | ||||
|             if (pm.resolveService(serviceIntent, 0) != null) { | ||||
|                 packagesSupportingCustomTabs.add(info.activityInfo.packageName); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Now packagesSupportingCustomTabs contains all apps that can handle both VIEW intents | ||||
|         // and service calls. | ||||
|         if (packagesSupportingCustomTabs.isEmpty()) { | ||||
|             sPackageNameToUse = null; | ||||
|         } else if (packagesSupportingCustomTabs.size() == 1) { | ||||
|             sPackageNameToUse = packagesSupportingCustomTabs.get(0); | ||||
|         } else if (!TextUtils.isEmpty(defaultViewHandlerPackageName) | ||||
|                 && !hasSpecializedHandlerIntents(context, activityIntent) | ||||
|                 && packagesSupportingCustomTabs.contains(defaultViewHandlerPackageName)) { | ||||
|             sPackageNameToUse = defaultViewHandlerPackageName; | ||||
|         } else if (packagesSupportingCustomTabs.contains(STABLE_PACKAGE)) { | ||||
|             sPackageNameToUse = STABLE_PACKAGE; | ||||
|         } else if (packagesSupportingCustomTabs.contains(BETA_PACKAGE)) { | ||||
|             sPackageNameToUse = BETA_PACKAGE; | ||||
|         } else if (packagesSupportingCustomTabs.contains(DEV_PACKAGE)) { | ||||
|             sPackageNameToUse = DEV_PACKAGE; | ||||
|         } else if (packagesSupportingCustomTabs.contains(LOCAL_PACKAGE)) { | ||||
|             sPackageNameToUse = LOCAL_PACKAGE; | ||||
|         } | ||||
|         return sPackageNameToUse; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Used to check whether there is a specialized handler for a given intent. | ||||
|      * | ||||
|      * @param intent The intent to check with. | ||||
|      * @return Whether there is a specialized handler for the given intent. | ||||
|      */ | ||||
|     private static boolean hasSpecializedHandlerIntents(Context context, Intent intent) { | ||||
|         try { | ||||
|             PackageManager pm = context.getPackageManager(); | ||||
|             List<ResolveInfo> handlers = pm.queryIntentActivities( | ||||
|                     intent, | ||||
|                     PackageManager.GET_RESOLVED_FILTER); | ||||
|             if (handlers == null || handlers.isEmpty()) { | ||||
|                 return false; | ||||
|             } | ||||
|             for (ResolveInfo resolveInfo : handlers) { | ||||
|                 IntentFilter filter = resolveInfo.filter; | ||||
|                 if (filter == null) continue; | ||||
|                 if (filter.countDataAuthorities() == 0 || filter.countDataPaths() == 0) continue; | ||||
|                 if (resolveInfo.activityInfo == null) continue; | ||||
|                 return true; | ||||
|             } | ||||
|         } catch (RuntimeException e) { | ||||
|             Log.e(TAG, "Runtime exception while getting specialized handlers"); | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return All possible chrome package names that provide custom tabs feature. | ||||
|      */ | ||||
|     public static String[] getPackages() { | ||||
|         return new String[]{"", STABLE_PACKAGE, BETA_PACKAGE, DEV_PACKAGE, LOCAL_PACKAGE}; | ||||
|     } | ||||
| } | ||||
| @@ -1,33 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs; | ||||
|  | ||||
|  | ||||
| import android.content.ComponentName; | ||||
| import androidx.browser.customtabs.CustomTabsClient; | ||||
| import androidx.browser.customtabs.CustomTabsServiceConnection; | ||||
|  | ||||
| import java.lang.ref.WeakReference; | ||||
|  | ||||
| /** | ||||
|  * Implementation for the CustomTabsServiceConnection that avoids leaking the | ||||
|  * ServiceConnectionCallback | ||||
|  */ | ||||
| public class ServiceConnection extends CustomTabsServiceConnection { | ||||
|     // A weak reference to the ServiceConnectionCallback to avoid leaking it. | ||||
|     private WeakReference<ServiceConnectionCallback> mConnectionCallback; | ||||
|  | ||||
|     public ServiceConnection(ServiceConnectionCallback connectionCallback) { | ||||
|         mConnectionCallback = new WeakReference<>(connectionCallback); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onCustomTabsServiceConnected(ComponentName name, CustomTabsClient client) { | ||||
|         ServiceConnectionCallback connectionCallback = mConnectionCallback.get(); | ||||
|         if (connectionCallback != null) connectionCallback.onServiceConnected(client); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onServiceDisconnected(ComponentName name) { | ||||
|         ServiceConnectionCallback connectionCallback = mConnectionCallback.get(); | ||||
|         if (connectionCallback != null) connectionCallback.onServiceDisconnected(); | ||||
|     } | ||||
| } | ||||
| @@ -1,19 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs; | ||||
|  | ||||
|  | ||||
| import androidx.browser.customtabs.CustomTabsClient; | ||||
|  | ||||
|  | ||||
| public interface ServiceConnectionCallback { | ||||
|     /** | ||||
|      * Called when the service is connected. | ||||
|      * | ||||
|      * @param client a CustomTabsClient | ||||
|      */ | ||||
|     void onServiceConnected(CustomTabsClient client); | ||||
|  | ||||
|     /** | ||||
|      * Called when the service is disconnected. | ||||
|      */ | ||||
|     void onServiceDisconnected(); | ||||
| } | ||||
| @@ -1,15 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils.customtabs.helpers; | ||||
|  | ||||
| import android.app.Service; | ||||
| import android.content.Intent; | ||||
| import android.os.Binder; | ||||
| import android.os.IBinder; | ||||
|  | ||||
| public class KeepAliveService extends Service { | ||||
|     private static final Binder sBinder = new Binder(); | ||||
|  | ||||
|     @Override | ||||
|     public IBinder onBind(Intent intent) { | ||||
|         return sBinder; | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +1,11 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <layer-list xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|  | ||||
|     <item | ||||
|         android:drawable="@color/ic_launcher_background"/> | ||||
|     <item> | ||||
|         <shape android:shape="rectangle" > | ||||
|             <solid android:color="?attr/colorSurface" /> | ||||
|         </shape> | ||||
|     </item> | ||||
|  | ||||
|     <item> | ||||
|         <bitmap | ||||
|   | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 406 B | 
| @@ -1,9 +0,0 @@ | ||||
| <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|         android:width="24dp" | ||||
|         android:height="24dp" | ||||
|         android:viewportWidth="24.0" | ||||
|         android:viewportHeight="24.0"> | ||||
|     <path | ||||
|         android:fillColor="#FFFFFFFF" | ||||
|         android:pathData="M13,12h7v1.5h-7zM13,9.5h7L20,11h-7zM13,14.5h7L20,16h-7zM21,4L3,4c-1.1,0 -2,0.9 -2,2v13c0,1.1 0.9,2 2,2h18c1.1,0 2,-0.9 2,-2L23,6c0,-1.1 -0.9,-2 -2,-2zM21,19h-9L12,6h9v13z"/> | ||||
| </vector> | ||||
							
								
								
									
										7
									
								
								androidApp/src/main/res/font/source_code_pro_medium.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								androidApp/src/main/res/font/source_code_pro_medium.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <font-family xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|         app:fontProviderAuthority="com.google.android.gms.fonts" | ||||
|         app:fontProviderPackage="com.google.android.gms" | ||||
|         app:fontProviderQuery="name=Source Code Pro&weight=500" | ||||
|         app:fontProviderCerts="@array/com_google_android_gms_fonts_certs"> | ||||
| </font-family> | ||||
| @@ -14,12 +14,12 @@ | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content"> | ||||
|  | ||||
|             <androidx.appcompat.widget.Toolbar | ||||
|             <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|                 android:id="@+id/toolbar" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="?attr/actionBarSize" | ||||
|                 app:theme="@style/ToolBarStyle" | ||||
|                 app:popupTheme="?attr/toolbarPopupTheme" /> | ||||
|  | ||||
|                  /> | ||||
|  | ||||
|         </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
| @@ -107,8 +107,7 @@ | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/tags" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 android:layout_height="40dp" | ||||
|                 android:theme="@style/App.Spinner"/> | ||||
|                 android:layout_height="40dp"/> | ||||
|  | ||||
|             <Button | ||||
|                 android:text="@string/add_source_save" | ||||
| @@ -116,7 +115,7 @@ | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:id="@+id/saveBtn" | ||||
|                 android:elevation="5dp" | ||||
|                 android:textColor="?attr/colorAccent" | ||||
|                 android:textColor="?android:textColorPrimary" | ||||
|                 android:layout_marginEnd="16dp" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|                 android:layout_marginRight="16dp" | ||||
|   | ||||
| @@ -28,12 +28,12 @@ | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="wrap_content"> | ||||
|  | ||||
|                     <androidx.appcompat.widget.Toolbar | ||||
|                     <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|                         android:id="@+id/toolBar" | ||||
|                         android:layout_width="match_parent" | ||||
|                         android:layout_height="?attr/actionBarSize" | ||||
|                         app:theme="@style/ToolBarStyle" | ||||
|                         app:popupTheme="?attr/toolbarPopupTheme" /> | ||||
|  | ||||
|                          /> | ||||
|  | ||||
|                 </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
| @@ -78,10 +78,12 @@ | ||||
|  | ||||
|         </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||
|         <com.ashokvarma.bottomnavigation.BottomNavigationBar | ||||
|             android:layout_gravity="bottom" | ||||
|             android:id="@+id/bottomBar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="60dp"/> | ||||
|             android:layout_height="60dp" | ||||
|             android:layout_gravity="bottom" | ||||
|             app:bnbActiveColor="@color/colorAccent" | ||||
|             app:bnbBackgroundColor="?attr/bottomBarBackground" /> | ||||
|     </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||
|  | ||||
|     <com.mikepenz.materialdrawer.widget.MaterialDrawerSliderView | ||||
|   | ||||
| @@ -16,8 +16,8 @@ | ||||
|             android:id="@+id/toolBar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             app:popupTheme="?attr/toolbarPopupTheme" | ||||
|             app:theme="@style/ToolBarStyle" /> | ||||
|  | ||||
|              /> | ||||
|  | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
|   | ||||
| @@ -10,12 +10,12 @@ | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content"> | ||||
|  | ||||
|         <androidx.appcompat.widget.Toolbar | ||||
|         <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|             android:id="@+id/toolbar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             app:theme="@style/ToolBarStyle" | ||||
|             app:popupTheme="?attr/toolbarPopupTheme" /> | ||||
|  | ||||
|              /> | ||||
|  | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|     <LinearLayout | ||||
|   | ||||
| @@ -17,8 +17,10 @@ | ||||
|             android:id="@+id/toolBar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             android:theme="@style/ToolBarStyle" | ||||
|             app:popupTheme="?attr/toolbarPopupTheme" | ||||
|             app:theme="@style/ToolBarStyle" /> | ||||
|  | ||||
|             /> | ||||
|  | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     android:id="@+id/layout" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
| @@ -8,11 +9,11 @@ | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content"> | ||||
|  | ||||
|         <androidx.appcompat.widget.Toolbar | ||||
|         <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|             android:id="@+id/toolbar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             android:theme="@style/ToolBarStyle" /> | ||||
|              /> | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
|     <FrameLayout | ||||
|   | ||||
| @@ -10,12 +10,12 @@ | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content"> | ||||
|  | ||||
|         <androidx.appcompat.widget.Toolbar | ||||
|         <androidx.appcompat.widget.Toolbar app:popupTheme="?attr/toolbarPopupTheme" android:theme="@style/ToolBarStyle" | ||||
|             android:id="@+id/toolbar" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="?attr/actionBarSize" | ||||
|             app:theme="@style/ToolBarStyle" | ||||
|             app:popupTheme="?attr/toolbarPopupTheme" /> | ||||
|  | ||||
|              /> | ||||
|  | ||||
|     </com.google.android.material.appbar.AppBarLayout> | ||||
|  | ||||
|   | ||||
| @@ -16,8 +16,7 @@ | ||||
|     app:layout_constraintTop_toTopOf="parent" | ||||
|     card_view:cardElevation="2dp" | ||||
|     card_view:cardUseCompatPadding="true" | ||||
|     card_view:layout_constraintBottom_toBottomOf="parent" | ||||
|     app:cardBackgroundColor="?cardBackgroundColor"> | ||||
|     card_view:layout_constraintBottom_toBottomOf="parent"> | ||||
|  | ||||
|     <androidx.constraintlayout.widget.ConstraintLayout | ||||
|         android:layout_width="match_parent" | ||||
|   | ||||
| @@ -66,7 +66,7 @@ | ||||
|                 android:layout_marginRight="16dp" | ||||
|                 android:layout_marginTop="24dp" | ||||
|                 android:paddingBottom="48dp" | ||||
|                 android:background="?android:colorBackground" | ||||
|                 android:background="?attr/webviewBackground" | ||||
|                 app:layout_constraintHorizontal_bias="0.0" | ||||
|                 app:layout_constraintLeft_toLeftOf="parent" | ||||
|                 app:layout_constraintRight_toRightOf="parent" | ||||
|   | ||||
| @@ -8,12 +8,6 @@ | ||||
|         android:title="@string/unmark" | ||||
|         app:showAsAction="ifRoom" /> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/more_action" | ||||
|         android:icon="@drawable/ic_chrome_reader_mode_white_24dp" | ||||
|         android:title="@string/reader_action_more" | ||||
|         app:showAsAction="ifRoom" /> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/open_action" | ||||
|         android:icon="@drawable/ic_open_in_browser_white_24dp" | ||||
|   | ||||
| @@ -1,10 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <menu xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|  | ||||
|     <item | ||||
|         android:id="@+id/clear" | ||||
|         android:icon="@drawable/ic_history_white_24dp" | ||||
|         android:title="@string/drawer_action_clear" | ||||
|         app:showAsAction="ifRoom" /> | ||||
| </menu> | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Recompte d'articles no llegits"</string> | ||||
|     <string name="display_all_counts_title">"Recompte d'articles llegits i preferits"</string> | ||||
|     <string name="text_wrong_url">"Sembla que esteu utilitzant un URL no vàlid. Assegureu-vos que és correcte, i si el problema persisteix, poseu-vos en contacte amb mi (a través de l'enllaç de contacte que hi ha a la Botiga). Tingueu en compte que per utilitzar aquesta aplicació cal que també utilitzeu Selfoss. Si no, no podreu accedir a canals RSS."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Obre els enllaços dins de l'aplicació"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Els articles s'obriran dins de l'aplicació"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Els articles s'obriran amb el navegador predeterminat"</string> | ||||
|     <string name="prefer_article_viewer_title">"Obre el visualitzador d'articles"</string> | ||||
|     <string name="prefer_article_viewer_on">"S'obrirà el visualitzador d'articles en lloc del navegador intern"</string> | ||||
|     <string name="prefer_article_viewer_off">"S'obrirà el navegador intern en lloc del visualitzador d'articles"</string> | ||||
|     <string name="pref_article_viewer_title">"Obre els enllaços dins de l'aplicació"</string> | ||||
|     <string name="pref_article_viewer_on">"Els articles s'obriran dins de l'aplicació"</string> | ||||
|     <string name="pref_article_viewer_off">"Els articles s'obriran amb el navegador predeterminat"</string> | ||||
|     <string name="pref_general_category_links">"Gestió d'enllaços"</string> | ||||
|     <string name="pref_general_category_displaying">"Visualització"</string> | ||||
|     <string name="pref_switch_card_view_on">"Els articles es mostraran com a targetes"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">L\'alçada de les targetes serà fixa</string> | ||||
|     <string name="source_code">Codi font</string> | ||||
|     <string name="drawer_error_loading_tags">S\'ha produït un error en carregar les etiquetes</string> | ||||
|     <string name="drawer_error_loading_sources">S\'ha produït un error en carregar les fonts</string> | ||||
|     <string name="drawer_item_filters">Filtres</string> | ||||
|     <string name="drawer_action_clear">Esborra</string> | ||||
|     <string name="drawer_item_tags">Etiquetes</string> | ||||
|     <string name="drawer_item_sources">Fonts</string> | ||||
|     <string name="drawer_action_edit">Edita</string> | ||||
|     <string name="no_tags_loaded">No s\'ha carregat cap etiqueta</string> | ||||
|     <string name="no_sources_loaded">No s\'ha carregat cap font</string> | ||||
|     <string name="drawer_loading">S\'està carregant…</string> | ||||
|     <string name="menu_home_search">Cerca</string> | ||||
|     <string name="can_delete_source">No es pot suprimir la font</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">API de Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Nombre d\'elements carregats</string> | ||||
|     <string name="pref_hidden_tags">Etiquetes ocultes</string> | ||||
|     <string name="display_header_drawer_summary">Mostra una capçalera amb la instància URL de Selfoss al panell lateral.</string> | ||||
|     <string name="display_header_drawer_title">Capçalera de menú</string> | ||||
|     <string name="pref_general_infinite_loading_title">Carrega articles en desplaçar</string> | ||||
|     <string name="translation">Traducció</string> | ||||
|     <string name="cant_open_invalid_url">L\'element URL no és vàlid. Estic intentant solucionar aquest problema perquè l\'aplicació no falli.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Zeige Anzahl ungelesener Artikel"</string> | ||||
|     <string name="display_all_counts_title">"Zeige Anzahl der Favoriten und gelesenen Artikel"</string> | ||||
|     <string name="text_wrong_url">"Sie scheinen eine ungültige URL verwenden. Stellen Sie sicher, dass die URL richtig ist. Sollte das Problem weiterhin bestehen kontaktieren Sie mich (über den Playstore-Kontakt-Link). Bitte beachten Sie, dass Sie Selfoss benötigen um RSS-Feeds zu lesen."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Öffne Links innerhalb der App"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Artikel werden innerhalb der App geöffnet"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Artikel werden mit deinem Standard-Browser geöffnet"</string> | ||||
|     <string name="prefer_article_viewer_title">"Verwenden Sie den Artikel-viewer"</string> | ||||
|     <string name="prefer_article_viewer_on">"Artikel-Viewer wird anstelle des internen Browser verwendet"</string> | ||||
|     <string name="prefer_article_viewer_off">"Der internen Browser wird anstelle des Artikel-Viewer verwendet"</string> | ||||
|     <string name="pref_article_viewer_title">"Öffne Links innerhalb der App"</string> | ||||
|     <string name="pref_article_viewer_on">"Artikel werden innerhalb der App geöffnet"</string> | ||||
|     <string name="pref_article_viewer_off">"Artikel werden mit deinem Standard-Browser geöffnet"</string> | ||||
|     <string name="pref_general_category_links">"Umgang mit Links"</string> | ||||
|     <string name="pref_general_category_displaying">"Ansicht"</string> | ||||
|     <string name="pref_switch_card_view_on">"Artikel werden als Kacheln angezeigt"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Kartenhöhe ist fix</string> | ||||
|     <string name="source_code">Quellcode</string> | ||||
|     <string name="drawer_error_loading_tags">Fehler beim Laden der Tags…</string> | ||||
|     <string name="drawer_error_loading_sources">Fehler beim Laden der Quellen…</string> | ||||
|     <string name="drawer_item_filters">Filter</string> | ||||
|     <string name="drawer_action_clear">leeren</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Quellen</string> | ||||
|     <string name="drawer_action_edit">bearbeiten</string> | ||||
|     <string name="no_tags_loaded">No tags loaded</string> | ||||
|     <string name="no_sources_loaded">Keine Quellen geladen</string> | ||||
|     <string name="drawer_loading">Lade…</string> | ||||
|     <string name="menu_home_search">Suche</string> | ||||
|     <string name="can_delete_source">Can\'t delete the source…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">selfoss API</string> | ||||
|     <string name="pref_api_items_number_title">Loaded items number</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">Display a header with the selfoss instance url on the lateral drawer.</string> | ||||
|     <string name="display_header_drawer_title">Account header</string> | ||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||
|     <string name="translation">Übersetzung</string> | ||||
|     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Mostrar recuento no leído"</string> | ||||
|     <string name="display_all_counts_title">"Mostrar recuento de favoritos y leídos"</string> | ||||
|     <string name="text_wrong_url">"Parece estar tratando de utilizar una dirección URL inválida. Asegúrese de que sea correcta y si el problema persiste, póngase en contacto conmigo (mediante el enlace de contacto de la tienda). Tenga en cuenta que la aplicación necesita utilizar Selfoss. No se puede acceder al contenido RSS sin él."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Abrir enlaces dentro de la aplicación"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Los artículos se abrirán dentro de la aplicación"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Los artículos se abrirán con tu navegador predeterminado"</string> | ||||
|     <string name="prefer_article_viewer_title">"Utilizar el visor de artículo"</string> | ||||
|     <string name="prefer_article_viewer_on">"Se usará el visor de artículos en lugar del navegador interno"</string> | ||||
|     <string name="prefer_article_viewer_off">"Se utilizará el navegador interno en lugar del visor de artículo"</string> | ||||
|     <string name="pref_article_viewer_title">"Abrir enlaces dentro de la aplicación"</string> | ||||
|     <string name="pref_article_viewer_on">"Los artículos se abrirán dentro de la aplicación"</string> | ||||
|     <string name="pref_article_viewer_off">"Los artículos se abrirán con tu navegador predeterminado"</string> | ||||
|     <string name="pref_general_category_links">"Control de enlaces"</string> | ||||
|     <string name="pref_general_category_displaying">"Mostrando"</string> | ||||
|     <string name="pref_switch_card_view_on">"Los artículos se mostrarán como tarjetas"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Se fijará la altura de la tarjeta</string> | ||||
|     <string name="source_code">Código fuente</string> | ||||
|     <string name="drawer_error_loading_tags">Error al cargar etiquetas…</string> | ||||
|     <string name="drawer_error_loading_sources">Error al cargar fuentes…</string> | ||||
|     <string name="drawer_item_filters">Filtros</string> | ||||
|     <string name="drawer_action_clear">limpiar</string> | ||||
|     <string name="drawer_item_tags">Etiquetas</string> | ||||
|     <string name="drawer_item_sources">Fuentes</string> | ||||
|     <string name="drawer_action_edit">editar</string> | ||||
|     <string name="no_tags_loaded">No hay etiquetas cargadas</string> | ||||
|     <string name="no_sources_loaded">No hay fuentes cargadas</string> | ||||
|     <string name="drawer_loading">Cargando…</string> | ||||
|     <string name="menu_home_search">Buscar</string> | ||||
|     <string name="can_delete_source">No se puede eliminar la fuente…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Api de Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Número de artículos cargados</string> | ||||
|     <string name="pref_hidden_tags">Etiquetas ocultas</string> | ||||
|     <string name="display_header_drawer_summary">Mostrar una cabecera con la url de instancia de selfoss en el cajón lateral.</string> | ||||
|     <string name="display_header_drawer_title">Cabecera de cuenta</string> | ||||
|     <string name="pref_general_infinite_loading_title">Cargar más artículos en desplazamiento</string> | ||||
|     <string name="translation">Traducción</string> | ||||
|     <string name="cant_open_invalid_url">La url del elemento no es válida. Estoy buscando resolver este problema para que la aplicación no colapse.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Display unread count"</string> | ||||
|     <string name="display_all_counts_title">"Display count for favorite and read"</string> | ||||
|     <string name="text_wrong_url">"You seem to be trying to use an invalid URL. Make sure it is correct, and if the problem persists, contact me (via the store contact link). Please note that the app needs you to be using Selfoss. You can't access RSS feeds without it."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Open links inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Articles will open with your default browser"</string> | ||||
|     <string name="prefer_article_viewer_title">"Use the article viewer"</string> | ||||
|     <string name="prefer_article_viewer_on">"Will use the article viewer instead of the internal browser"</string> | ||||
|     <string name="prefer_article_viewer_off">"Will use the internal browser instead of the article viewer"</string> | ||||
|     <string name="pref_article_viewer_title">"Open links inside the app"</string> | ||||
|     <string name="pref_article_viewer_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_article_viewer_off">"Articles will open with your default browser"</string> | ||||
|     <string name="pref_general_category_links">"Link handling"</string> | ||||
|     <string name="pref_general_category_displaying">"Displaying"</string> | ||||
|     <string name="pref_switch_card_view_on">"The articles will be displayed as cards"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Card height will be fixed</string> | ||||
|     <string name="source_code">Source code</string> | ||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="drawer_item_filters">Filters</string> | ||||
|     <string name="drawer_action_clear">clear</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Sources</string> | ||||
|     <string name="drawer_action_edit">edit</string> | ||||
|     <string name="no_tags_loaded">No tags loaded</string> | ||||
|     <string name="no_sources_loaded">No sources loaded</string> | ||||
|     <string name="drawer_loading">Loading …</string> | ||||
|     <string name="menu_home_search">Search</string> | ||||
|     <string name="can_delete_source">Can\'t delete the source…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Loaded items number</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">Display a header with the selfoss instance url on the lateral drawer.</string> | ||||
|     <string name="display_header_drawer_title">Account header</string> | ||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||
|     <string name="translation">Translation</string> | ||||
|     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|     <string name="error_field_required">"Champ requis"</string> | ||||
|     <string name="prompt_url">"Url Selfoss"</string> | ||||
|     <string name="withLoginSwitch">"Avec login ?"</string> | ||||
|     <string name="login_url_problem">"Petit souci. Il manque peut être un / à la fin ?"</string> | ||||
|     <string name="login_url_problem">"Petit souci. Il manque peut-être un / à la fin ?"</string> | ||||
|     <string name="prompt_login">"Utilisateur"</string> | ||||
|     <string name="label_share">"Partager"</string> | ||||
|     <string name="readAll">"Tout lire"</string> | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Afficher le nombre de non lus"</string> | ||||
|     <string name="display_all_counts_title">"Afficher le nombre de favoris et d'articles lus"</string> | ||||
|     <string name="text_wrong_url">"Vous semblez essayer de vous connecter avec une URL invalide. Assurez-vous que c'est la bonne, et si le problème persiste, contactez-moi via le lien du play store. Notez aussi que l'application ne peut fonctionner sans l'application web Selfoss. Vous ne pouvez pas utiliser l'application pour accéder directement aux flux RSS."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Ouvrir les liens dans l'application"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Les articles s'ouvriront dans l'application"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Les articles s'ouvriront dans votre naviguateur par défaut"</string> | ||||
|     <string name="prefer_article_viewer_title">"Utiliser le visionneur d'articles"</string> | ||||
|     <string name="prefer_article_viewer_on">"Utiliser le naviguateur interne"</string> | ||||
|     <string name="prefer_article_viewer_off">"Utiliser le naviguateur interne au lieu du visionneur d'articles"</string> | ||||
|     <string name="pref_article_viewer_title">"Ouvrir les liens dans l'application"</string> | ||||
|     <string name="pref_article_viewer_on">"Les articles s'ouvriront dans l'application"</string> | ||||
|     <string name="pref_article_viewer_off">"Les articles s'ouvriront dans votre naviguateur par défaut"</string> | ||||
|     <string name="pref_general_category_links">"Gestion des liens"</string> | ||||
|     <string name="pref_general_category_displaying">"Affichage"</string> | ||||
|     <string name="pref_switch_card_view_on">"Les articles seront affichés en forme de carte"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">La taille de la carte sera fixe</string> | ||||
|     <string name="source_code">Code source</string> | ||||
|     <string name="drawer_error_loading_tags">Erreur lors du chargement des tags…</string> | ||||
|     <string name="drawer_error_loading_sources">Erreur lors du chargement des sources…</string> | ||||
|     <string name="drawer_item_filters">Filtres</string> | ||||
|     <string name="drawer_action_clear">raz</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Sources</string> | ||||
|     <string name="drawer_action_edit">éditer</string> | ||||
|     <string name="no_tags_loaded">Pas de tags chargés</string> | ||||
|     <string name="no_sources_loaded">Pas de sources chargés</string> | ||||
|     <string name="drawer_loading">Chargement …</string> | ||||
|     <string name="menu_home_search">Rechercher</string> | ||||
|     <string name="can_delete_source">Impossible de supprimer la source…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Api Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Nombre d\'articles chargés</string> | ||||
|     <string name="pref_hidden_tags">Tags Cachés</string> | ||||
|     <string name="display_header_drawer_summary">Afficher une entête avec l\'url de votre instance de Selfoss en haut du drawer lateral.</string> | ||||
|     <string name="display_header_drawer_title">Entête de compte</string> | ||||
|     <string name="pref_general_infinite_loading_title">Charger plus d\'articles au scroll</string> | ||||
|     <string name="translation">Traduction</string> | ||||
|     <string name="cant_open_invalid_url">L’url de l’élément n’est pas valide. En attendant la résolution du problème, le lien ne s\'ouvrira pas.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">La barre sera affichée</string> | ||||
|     <string name="reader_static_bar_off">La barre sera affichée grâce au bouton</string> | ||||
|     <string name="remove_source">Supprimer la source</string> | ||||
|     <string name="pref_theme_title">Thème Clair/Sombre</string> | ||||
|     <string name="mode_dark">Thème sombre</string> | ||||
|     <string name="mode_system">Utiliser les paramètres système</string> | ||||
|     <string name="mode_light">Thème clair</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Mostrar reconto de artigos non lidos"</string> | ||||
|     <string name="display_all_counts_title">"Mostrar reconto de artigos lidos e favoritos"</string> | ||||
|     <string name="text_wrong_url">"Semella que intentas usar unha URL non válida. Asegúrate de que é correcta, e se o problema persiste, ponte en contacto conmigo (a través da ligazón de contacto na tenda). Por favor ten en conta que a aplicación precisa que uses Selfoss. Non podes acceder a canles RSS se non o tes."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Abrir ligazóns dentro da aplicación"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Os artigos abriranse dentro da aplicación"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Os artigos abriranse co teu navegador prederminado"</string> | ||||
|     <string name="prefer_article_viewer_title">"Usar o visor de artigos"</string> | ||||
|     <string name="prefer_article_viewer_on">"Usarase o visor de artigos en lugar do navegador interno"</string> | ||||
|     <string name="prefer_article_viewer_off">"Usarase o navegador interno en lugar do visor de artigos"</string> | ||||
|     <string name="pref_article_viewer_title">"Abrir ligazóns dentro da aplicación"</string> | ||||
|     <string name="pref_article_viewer_on">"Os artigos abriranse dentro da aplicación"</string> | ||||
|     <string name="pref_article_viewer_off">"Os artigos abriranse co teu navegador prederminado"</string> | ||||
|     <string name="pref_general_category_links">"Xestión de ligazóns"</string> | ||||
|     <string name="pref_general_category_displaying">"Visualización"</string> | ||||
|     <string name="pref_switch_card_view_on">"Os artigos amosaranse coma tarxetas"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">A altura das tarxetas será fixa</string> | ||||
|     <string name="source_code">Código fonte</string> | ||||
|     <string name="drawer_error_loading_tags">Produciuse un erro ao cargar as etiquetas…</string> | ||||
|     <string name="drawer_error_loading_sources">Produciuse un erro ao cargar as fontes…</string> | ||||
|     <string name="drawer_item_filters">Filtros</string> | ||||
|     <string name="drawer_action_clear">limpar</string> | ||||
|     <string name="drawer_item_tags">Etiquetas</string> | ||||
|     <string name="drawer_item_sources">Fontes</string> | ||||
|     <string name="drawer_action_edit">editar</string> | ||||
|     <string name="no_tags_loaded">Non se cargou ningunha etiqueta</string> | ||||
|     <string name="no_sources_loaded">Non se cargou ningunha fonte</string> | ||||
|     <string name="drawer_loading">Cargando…</string> | ||||
|     <string name="menu_home_search">Procurar</string> | ||||
|     <string name="can_delete_source">Non se puido eliminar a fonte…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">API de Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Número de elementos cargados</string> | ||||
|     <string name="pref_hidden_tags">Etiquetas ocultas</string> | ||||
|     <string name="display_header_drawer_summary">Amosar unha cabeceira coa URL da instancia de Selfoss no panel lateral.</string> | ||||
|     <string name="display_header_drawer_title">Cabeceira da conta</string> | ||||
|     <string name="pref_general_infinite_loading_title">Cargar máis artigos ao desprazarse</string> | ||||
|     <string name="translation">Traducción</string> | ||||
|     <string name="cant_open_invalid_url">A URL do elemento non é válida. Estou tratando de solucionar isto pra que a aplicación non falle.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">A barra inferior mostrarase sempre</string> | ||||
|     <string name="reader_static_bar_off">A barra inferior pode mostrarse a través do botón flotante</string> | ||||
|     <string name="remove_source">Eliminar fonte</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Tampilkan jumlah item yang belum dibaca"</string> | ||||
|     <string name="display_all_counts_title">"Tampilkan jumlah item untuk favorit dan sudah dibaca"</string> | ||||
|     <string name="text_wrong_url">"Sepertinya Anda mencoba menggunakan URL yang tidak valid. Pastikan itu benar, jika masalah terus berlanjut, hubungi saya (melalui link kontak toko). Harap dicatat bahwa aplikasi ini mengharuskan Anda menggunakan Selfoss. Tanpa itu, Anda tidak bisa mengakses umpan RSS."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Buka tautan dalam aplikasi"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Artikel akan dibuka di dalam aplikasi"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Artikel akan dibuka dalam peramban bawaan Anda"</string> | ||||
|     <string name="prefer_article_viewer_title">"Gunakan pratinjau artikel"</string> | ||||
|     <string name="prefer_article_viewer_on">"Lihat artikel di penampil daripada peramban internal"</string> | ||||
|     <string name="prefer_article_viewer_off">"Gunakan peramban internal dan bukan penampil artikel"</string> | ||||
|     <string name="pref_article_viewer_title">"Buka tautan dalam aplikasi"</string> | ||||
|     <string name="pref_article_viewer_on">"Artikel akan dibuka di dalam aplikasi"</string> | ||||
|     <string name="pref_article_viewer_off">"Artikel akan dibuka dalam peramban bawaan Anda"</string> | ||||
|     <string name="pref_general_category_links">"Pengolahan tautan"</string> | ||||
|     <string name="pref_general_category_displaying">"Tampilan"</string> | ||||
|     <string name="pref_switch_card_view_on">"Artikel ini akan ditampilkan dalam bentuk kartu"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Ukuran kartu akan tetap</string> | ||||
|     <string name="source_code">Kode sumber</string> | ||||
|     <string name="drawer_error_loading_tags">Kesalahan saat memuat tag…</string> | ||||
|     <string name="drawer_error_loading_sources">Kesalahan saat memuat sumber…</string> | ||||
|     <string name="drawer_item_filters">Filter</string> | ||||
|     <string name="drawer_action_clear">kosongkan</string> | ||||
|     <string name="drawer_item_tags">Tag</string> | ||||
|     <string name="drawer_item_sources">Sumber</string> | ||||
|     <string name="drawer_action_edit">suntung</string> | ||||
|     <string name="no_tags_loaded">Tidak ada tag yang dimuat</string> | ||||
|     <string name="no_sources_loaded">Tak ada sumber yang dimuat</string> | ||||
|     <string name="drawer_loading">Memuat …</string> | ||||
|     <string name="menu_home_search">Cari</string> | ||||
|     <string name="can_delete_source">Tidak dapat menghapus sumber…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Item nomor dimuat</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">Kop dengan alamat link Selfoss ditampilkan di laci lateral.</string> | ||||
|     <string name="display_header_drawer_title">Kop akun</string> | ||||
|     <string name="pref_general_infinite_loading_title">Muat lebih banyak artikel saat membalik halaman</string> | ||||
|     <string name="translation">Terjemahan</string> | ||||
|     <string name="cant_open_invalid_url">Alamat tautan proyek tidak valid. Saya mencoba memecahkan masalah ini untuk menghindari aplikasi berhenti.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Display unread count"</string> | ||||
|     <string name="display_all_counts_title">"Display count for favorite and read"</string> | ||||
|     <string name="text_wrong_url">"You seem to be trying to use an invalid URL. Make sure it is correct, and if the problem persists, contact me (via the store contact link). Please note that the app needs you to be using Selfoss. You can't access RSS feeds without it."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Open links inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Articles will open with your default browser"</string> | ||||
|     <string name="prefer_article_viewer_title">"Use the article viewer"</string> | ||||
|     <string name="prefer_article_viewer_on">"Will use the article viewer instead of the internal browser"</string> | ||||
|     <string name="prefer_article_viewer_off">"Will use the internal browser instead of the article viewer"</string> | ||||
|     <string name="pref_article_viewer_title">"Open links inside the app"</string> | ||||
|     <string name="pref_article_viewer_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_article_viewer_off">"Articles will open with your default browser"</string> | ||||
|     <string name="pref_general_category_links">"Link handling"</string> | ||||
|     <string name="pref_general_category_displaying">"Displaying"</string> | ||||
|     <string name="pref_switch_card_view_on">"The articles will be displayed as cards"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Card height will be fixed</string> | ||||
|     <string name="source_code">Codice sorgente</string> | ||||
|     <string name="drawer_error_loading_tags">Errore nel caricamento dei tag…</string> | ||||
|     <string name="drawer_error_loading_sources">Errore nel caricamento delle fonti…</string> | ||||
|     <string name="drawer_item_filters">Filtri</string> | ||||
|     <string name="drawer_action_clear">cancella</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Fonti</string> | ||||
|     <string name="drawer_action_edit">modifica</string> | ||||
|     <string name="no_tags_loaded">Nessun tag caricato</string> | ||||
|     <string name="no_sources_loaded">No sources loaded</string> | ||||
|     <string name="drawer_loading">Caricamento…</string> | ||||
|     <string name="menu_home_search">Cerca</string> | ||||
|     <string name="can_delete_source">Non è possibile eliminare la fonte…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Api di Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Numero di elementi caricati</string> | ||||
|     <string name="pref_hidden_tags">Tag nascosti</string> | ||||
|     <string name="display_header_drawer_summary">Display a header with the selfoss instance url on the lateral drawer.</string> | ||||
|     <string name="display_header_drawer_title">Account header</string> | ||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||
|     <string name="translation">Traduzioni</string> | ||||
|     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Display unread count"</string> | ||||
|     <string name="display_all_counts_title">"Display count for favorite and read"</string> | ||||
|     <string name="text_wrong_url">"You seem to be trying to use an invalid URL. Make sure it is correct, and if the problem persists, contact me (via the store contact link). Please note that the app needs you to be using Selfoss. You can't access RSS feeds without it."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Open links inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Articles will open with your default browser"</string> | ||||
|     <string name="prefer_article_viewer_title">"Use the article viewer"</string> | ||||
|     <string name="prefer_article_viewer_on">"Will use the article viewer instead of the internal browser"</string> | ||||
|     <string name="prefer_article_viewer_off">"Will use the internal browser instead of the article viewer"</string> | ||||
|     <string name="pref_article_viewer_title">"Open links inside the app"</string> | ||||
|     <string name="pref_article_viewer_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_article_viewer_off">"Articles will open with your default browser"</string> | ||||
|     <string name="pref_general_category_links">"Link handling"</string> | ||||
|     <string name="pref_general_category_displaying">"Displaying"</string> | ||||
|     <string name="pref_switch_card_view_on">"The articles will be displayed as cards"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Card height will be fixed</string> | ||||
|     <string name="source_code">Source code</string> | ||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="drawer_item_filters">Filters</string> | ||||
|     <string name="drawer_action_clear">clear</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Sources</string> | ||||
|     <string name="drawer_action_edit">edit</string> | ||||
|     <string name="no_tags_loaded">No tags loaded</string> | ||||
|     <string name="no_sources_loaded">No sources loaded</string> | ||||
|     <string name="drawer_loading">Loading …</string> | ||||
|     <string name="menu_home_search">Search</string> | ||||
|     <string name="can_delete_source">Can\'t delete the source…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Loaded items number</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">Display a header with the selfoss instance url on the lateral drawer.</string> | ||||
|     <string name="display_header_drawer_title">Account header</string> | ||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||
|     <string name="translation">Translation</string> | ||||
|     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
							
								
								
									
										13
									
								
								androidApp/src/main/res/values-night/styles.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								androidApp/src/main/res/values-night/styles.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| <resources> | ||||
|     <style name="NoBar" parent="Theme.MaterialComponents.DayNight.NoActionBar"> | ||||
|         <item name="colorPrimary">@color/colorPrimary</item> | ||||
|         <item name="colorPrimaryDark">@color/colorPrimaryDark</item> | ||||
|         <item name="colorAccent">@color/colorAccent</item> | ||||
|         <item name="colorAccentDark">@color/colorAccentDark</item> | ||||
|         <item name="preferenceTheme">@style/PreferenceStyle</item> | ||||
|         <item name="android:statusBarColor">@color/dark</item> | ||||
|         <item name="bottomBarBackground">@color/dark</item> | ||||
|         <item name="toolbarPopupTheme">@style/ThemeOverlay.AppCompat.Dark</item> | ||||
|         <item name="webviewBackground">@color/dark</item> | ||||
|     </style> | ||||
| </resources> | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Geef aantal ongelezen weer"</string> | ||||
|     <string name="display_all_counts_title">"Geef aantal weer bij favorieten en gelezen"</string> | ||||
|     <string name="text_wrong_url">"De gebruikte link lijkt onjuist. Controleer deze. Mocht het probleem blijven, neem dan contact met me op (via de contact link in de store). Om deze app te kunnen gebruiken heb je Selfoss nodig."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Links opnemen in interne browser"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Artikelen worden in de interne browser geopend"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Artikelen worden geopend in de standaard browser"</string> | ||||
|     <string name="prefer_article_viewer_title">"Gebruik artikel viewer"</string> | ||||
|     <string name="prefer_article_viewer_on">"Artikelen in viewer weergeven in plaats van de interne browser"</string> | ||||
|     <string name="prefer_article_viewer_off">"Artikelen in interne browser weergeven in plaats van viewer"</string> | ||||
|     <string name="pref_article_viewer_title">"Links opnemen in interne browser"</string> | ||||
|     <string name="pref_article_viewer_on">"Artikelen worden in de interne browser geopend"</string> | ||||
|     <string name="pref_article_viewer_off">"Artikelen worden geopend in de standaard browser"</string> | ||||
|     <string name="pref_general_category_links">"Links"</string> | ||||
|     <string name="pref_general_category_displaying">"Weergave"</string> | ||||
|     <string name="pref_switch_card_view_on">"De artikelen worden als kaarten weergegeven"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Vaste hoogte</string> | ||||
|     <string name="source_code">Broncode</string> | ||||
|     <string name="drawer_error_loading_tags">Fout bij het laden van tags…</string> | ||||
|     <string name="drawer_error_loading_sources">Fout bij laden van bronnen…</string> | ||||
|     <string name="drawer_item_filters">Filters</string> | ||||
|     <string name="drawer_action_clear">wissen</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Bronnen</string> | ||||
|     <string name="drawer_action_edit">bewerken</string> | ||||
|     <string name="no_tags_loaded">Geen tags geladen</string> | ||||
|     <string name="no_sources_loaded">Geen bronnen geladen</string> | ||||
|     <string name="drawer_loading">Bezig met laden …</string> | ||||
|     <string name="menu_home_search">Zoeken</string> | ||||
|     <string name="can_delete_source">Kan de bron niet verwijderen…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Geladen items nummer</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">Laat een koptekst weergeven met de url van de selfoss instantie in de zijlade.</string> | ||||
|     <string name="display_header_drawer_title">Account titel</string> | ||||
|     <string name="pref_general_infinite_loading_title">Laad meer artikelen door te bladeren</string> | ||||
|     <string name="translation">Vertaling</string> | ||||
|     <string name="cant_open_invalid_url">De URL is ongeldig. Ik probeer dit probleem op te lossen, zodat de toepassing niet wordt afgesloten.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Exibir contagem de artigos não lidos"</string> | ||||
|     <string name="display_all_counts_title">"Exibir contagem de lidos e favoritos"</string> | ||||
|     <string name="text_wrong_url">"Parece que você está tentando utilizar uma URL inválida. Certifique-se de que está correto, e se o problema persistir, entre em contato comigo (através do link de contato da loja). Por favor, note que o aplicativo precisa que você esteja usando o Selfoss. Você não pode acessar feeds RSS sem ele."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Abrir links dentro do aplicativo"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Os artigos serão abertos dentro do aplicativo"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Os artigos serão abertos com seu navegador padrão"</string> | ||||
|     <string name="prefer_article_viewer_title">"Use o visualizador de artigos"</string> | ||||
|     <string name="prefer_article_viewer_on">"Usará o visualizador de artigos em vez do navegador"</string> | ||||
|     <string name="prefer_article_viewer_off">"Utilizará o navegador em vez do visualizador de artigos"</string> | ||||
|     <string name="pref_article_viewer_title">"Abrir links dentro do aplicativo"</string> | ||||
|     <string name="pref_article_viewer_on">"Os artigos serão abertos dentro do aplicativo"</string> | ||||
|     <string name="pref_article_viewer_off">"Os artigos serão abertos com seu navegador padrão"</string> | ||||
|     <string name="pref_general_category_links">"Manipulação de links"</string> | ||||
|     <string name="pref_general_category_displaying">"Mostrando"</string> | ||||
|     <string name="pref_switch_card_view_on">"Os artigos serão exibidos no formato de cards"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Cards com altura de tamanho fixo</string> | ||||
|     <string name="source_code">Código fonte</string> | ||||
|     <string name="drawer_error_loading_tags">Erro ao carregar as tags…</string> | ||||
|     <string name="drawer_error_loading_sources">Erro ao carregar as fontes…</string> | ||||
|     <string name="drawer_item_filters">Filtros</string> | ||||
|     <string name="drawer_action_clear">limpar</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Fontes</string> | ||||
|     <string name="drawer_action_edit">editar</string> | ||||
|     <string name="no_tags_loaded">Nenhuma tag carregada</string> | ||||
|     <string name="no_sources_loaded">Nenhuma fonte carregada</string> | ||||
|     <string name="drawer_loading">Carregando …</string> | ||||
|     <string name="menu_home_search">Procurar</string> | ||||
|     <string name="can_delete_source">Não foi possível apagar a fonte…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Quantidade de itens carregados</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">Exibir um cabeçalho com o URL da instância do Selfoss na barra lateral.</string> | ||||
|     <string name="display_header_drawer_title">Cabeçalho da conta</string> | ||||
|     <string name="pref_general_infinite_loading_title">Carregar mais artigos ao realizar o scroll</string> | ||||
|     <string name="translation">Traduções</string> | ||||
|     <string name="cant_open_invalid_url">A url está inválida. Estou tentando resolver esse problema para que o aplicativo não encerre.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Exibir a contagem não lida"</string> | ||||
|     <string name="display_all_counts_title">"Exibir a contagem para o favorito e leitura"</string> | ||||
|     <string name="text_wrong_url">"Você parece estar tentando usar um URL inválido. Certifique-se de que está correto, e se o problema persistir, entre em contato comigo (através do link de contato da loja). Por favor, note que o aplicativo precisa que você esteja usando o Selfoss. Você não pode acessar feeds RSS sem ele."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Abrir links dentro do app"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Artigos serão aberto dentro do aplicativo"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Artigos serão aberto com o seu navegador padrão"</string> | ||||
|     <string name="prefer_article_viewer_title">"Use o Visualizador de artigo"</string> | ||||
|     <string name="prefer_article_viewer_on">"Vai usar o Visualizador de artigo em vez do navegador interno"</string> | ||||
|     <string name="prefer_article_viewer_off">"Vai usar o navegador interno em vez do Visualizador de artigo"</string> | ||||
|     <string name="pref_article_viewer_title">"Abrir links dentro do app"</string> | ||||
|     <string name="pref_article_viewer_on">"Artigos serão aberto dentro do aplicativo"</string> | ||||
|     <string name="pref_article_viewer_off">"Artigos serão aberto com o seu navegador padrão"</string> | ||||
|     <string name="pref_general_category_links">"Manipulação de ligações"</string> | ||||
|     <string name="pref_general_category_displaying">"Mostrando"</string> | ||||
|     <string name="pref_switch_card_view_on">"Os artigos serão exibidos como cartões"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Altura do cartão será corrigida</string> | ||||
|     <string name="source_code">Código fonte</string> | ||||
|     <string name="drawer_error_loading_tags">Erro ao carregar etiquetas…</string> | ||||
|     <string name="drawer_error_loading_sources">Erro ao carregar fontes…</string> | ||||
|     <string name="drawer_item_filters">Filtros</string> | ||||
|     <string name="drawer_action_clear">limpar</string> | ||||
|     <string name="drawer_item_tags">Etiquetas</string> | ||||
|     <string name="drawer_item_sources">Fontes</string> | ||||
|     <string name="drawer_action_edit">editar</string> | ||||
|     <string name="no_tags_loaded">Não tags carregado</string> | ||||
|     <string name="no_sources_loaded">Não há fontes carregadas</string> | ||||
|     <string name="drawer_loading">A carregar…</string> | ||||
|     <string name="menu_home_search">Buscar</string> | ||||
|     <string name="can_delete_source">Não é possível excluir a fonte…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Api de Selfoss</string> | ||||
|     <string name="pref_api_items_number_title">Número de itens carregados</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">Exibir um cabeçalho com o url de instância de selfoss na gaveta lateral.</string> | ||||
|     <string name="display_header_drawer_title">Cabeçalho de conta</string> | ||||
|     <string name="pref_general_infinite_loading_title">Carregar mais artigos no pergaminho</string> | ||||
|     <string name="translation">Tradução</string> | ||||
|     <string name="cant_open_invalid_url">A url do item é inválido. Eu estou olhando para resolver esta questão, para que o app não vai falhar.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Display unread count"</string> | ||||
|     <string name="display_all_counts_title">"Display count for favorite and read"</string> | ||||
|     <string name="text_wrong_url">"You seem to be trying to use an invalid URL. Make sure it is correct, and if the problem persists, contact me (via the store contact link). Please note that the app needs you to be using Selfoss. You can't access RSS feeds without it."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Open links inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Articles will open with your default browser"</string> | ||||
|     <string name="prefer_article_viewer_title">"Use the article viewer"</string> | ||||
|     <string name="prefer_article_viewer_on">"Will use the article viewer instead of the internal browser"</string> | ||||
|     <string name="prefer_article_viewer_off">"Will use the internal browser instead of the article viewer"</string> | ||||
|     <string name="pref_article_viewer_title">"Open links inside the app"</string> | ||||
|     <string name="pref_article_viewer_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_article_viewer_off">"Articles will open with your default browser"</string> | ||||
|     <string name="pref_general_category_links">"Link handling"</string> | ||||
|     <string name="pref_general_category_displaying">"Displaying"</string> | ||||
|     <string name="pref_switch_card_view_on">"The articles will be displayed as cards"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Card height will be fixed</string> | ||||
|     <string name="source_code">Source code</string> | ||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="drawer_item_filters">Filters</string> | ||||
|     <string name="drawer_action_clear">clear</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Sources</string> | ||||
|     <string name="drawer_action_edit">edit</string> | ||||
|     <string name="no_tags_loaded">No tags loaded</string> | ||||
|     <string name="no_sources_loaded">No sources loaded</string> | ||||
|     <string name="drawer_loading">Loading …</string> | ||||
|     <string name="menu_home_search">Search</string> | ||||
|     <string name="can_delete_source">Can\'t delete the source…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Loaded items number</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">Display a header with the selfoss instance url on the lateral drawer.</string> | ||||
|     <string name="display_header_drawer_title">Account header</string> | ||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||
|     <string name="translation">Translation</string> | ||||
|     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"Okunmamış sayıyı görüntüle"</string> | ||||
|     <string name="display_all_counts_title">"Favori ve okunan sayıları göster"</string> | ||||
|     <string name="text_wrong_url">"Geçersiz bir URL kullanmaya çalışıyormuş gibi görünüyorsunuz. Doğru olduğundan emin olun ve sorun devam ederse, bana ulaşın (mağaza iletişim bağlantısıyla). Uygulamanın, Selfoss'u kullanmanız gerektiğini lütfen unutmayın. RSS özet akışlarına olmadan erişemezsiniz."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Uygulamadaki bağlantıları açın"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Makale, uygulama içinde açılacaktır"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Makaleler varsayılan tarayıcınızla açılır"</string> | ||||
|     <string name="prefer_article_viewer_title">"Makale görüntüleyiciyi kullanın"</string> | ||||
|     <string name="prefer_article_viewer_on">"Dahili tarayıcı yerine makale görüntüleyicisini kullanacak"</string> | ||||
|     <string name="prefer_article_viewer_off">"Makale görüntüleyicisi yerine dahili tarayıcıyı kullanacak"</string> | ||||
|     <string name="pref_article_viewer_title">"Uygulamadaki bağlantıları açın"</string> | ||||
|     <string name="pref_article_viewer_on">"Makale, uygulama içinde açılacaktır"</string> | ||||
|     <string name="pref_article_viewer_off">"Makaleler varsayılan tarayıcınızla açılır"</string> | ||||
|     <string name="pref_general_category_links">"Bağlantı açma şekli"</string> | ||||
|     <string name="pref_general_category_displaying">"Gösteriliyor"</string> | ||||
|     <string name="pref_switch_card_view_on">"Makaleler kart olarak gösterilecek"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">Kart yüksekliği sabit olacak</string> | ||||
|     <string name="source_code">Kaynak kodu</string> | ||||
|     <string name="drawer_error_loading_tags">Etiketler yükleme hatası…</string> | ||||
|     <string name="drawer_error_loading_sources">Kaynaklar yüklenirken hata oluştu…</string> | ||||
|     <string name="drawer_item_filters">Filtreler</string> | ||||
|     <string name="drawer_action_clear">temizle</string> | ||||
|     <string name="drawer_item_tags">Etiketler</string> | ||||
|     <string name="drawer_item_sources">Kaynaklar</string> | ||||
|     <string name="drawer_action_edit">düzenle</string> | ||||
|     <string name="no_tags_loaded">Yüklenen görüntü yok</string> | ||||
|     <string name="no_sources_loaded">Yüklenen kaynak yok</string> | ||||
|     <string name="drawer_loading">Yükleniyor…</string> | ||||
|     <string name="menu_home_search">Ara</string> | ||||
|     <string name="can_delete_source">Kaynak silinemiyor…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">Selfoss Uygulaması</string> | ||||
|     <string name="pref_api_items_number_title">Yüklenen öğe numarası</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">Selfoss örneği url\'li bir üstbilgi, yan çekmece üzerine gösterin.</string> | ||||
|     <string name="display_header_drawer_title">Hesap başlığı</string> | ||||
|     <string name="pref_general_infinite_loading_title">Kaydırma üzerine daha fazla makale yükleyin</string> | ||||
|     <string name="translation">Çeviri</string> | ||||
|     <string name="cant_open_invalid_url">Öğe url geçersiz. Uygulama çökmeyeceği için bu sorunu çözmeye çalışıyorum.</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"显示未读数"</string> | ||||
|     <string name="display_all_counts_title">"显示收藏和已读的计数"</string> | ||||
|     <string name="text_wrong_url">"您似乎试图使用无效的 URL。确保它是正确的,如果问题仍然存在,请与我联系 (通过商店的联系链接)。请注意,该应用程序需要您使用 Selfoss。没有它,您无法访问 RSS 源。"</string> | ||||
|     <string name="pref_general_internal_browser_title">"打开应用程序中的链接"</string> | ||||
|     <string name="pref_general_internal_browser_on">"文章将在应用程序内打开"</string> | ||||
|     <string name="pref_general_internal_browser_off">"文章将使用默认浏览器打开"</string> | ||||
|     <string name="prefer_article_viewer_title">"使用文章查看器"</string> | ||||
|     <string name="prefer_article_viewer_on">"将使用文章查看器而不是内部浏览器"</string> | ||||
|     <string name="prefer_article_viewer_off">"将使用内部浏览器而不是文章查看器"</string> | ||||
|     <string name="pref_article_viewer_title">"在应用内打开链接"</string> | ||||
|     <string name="pref_article_viewer_on">"文章将在应用程序内打开"</string> | ||||
|     <string name="pref_article_viewer_off">"文章将使用默认浏览器打开"</string> | ||||
|     <string name="pref_general_category_links">"链接处理"</string> | ||||
|     <string name="pref_general_category_displaying">"显示"</string> | ||||
|     <string name="pref_switch_card_view_on">"这些文章将以卡片形式显示"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">卡片高度将被固定</string> | ||||
|     <string name="source_code">源代码</string> | ||||
|     <string name="drawer_error_loading_tags">加载标记时出错..。</string> | ||||
|     <string name="drawer_error_loading_sources">加载源时出错..。</string> | ||||
|     <string name="drawer_item_filters">搜索条件</string> | ||||
|     <string name="drawer_action_clear">清空</string> | ||||
|     <string name="drawer_item_tags">标签</string> | ||||
|     <string name="drawer_item_sources">来源</string> | ||||
|     <string name="drawer_action_edit">编辑</string> | ||||
|     <string name="no_tags_loaded">未加载标签</string> | ||||
|     <string name="no_sources_loaded">未加载数据源</string> | ||||
|     <string name="drawer_loading">正在载入…</string> | ||||
|     <string name="menu_home_search">搜索</string> | ||||
|     <string name="can_delete_source">无法删除数据源…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">塞尔福斯 Api</string> | ||||
|     <string name="pref_api_items_number_title">已加载项目编号</string> | ||||
|     <string name="pref_hidden_tags">隐藏标签</string> | ||||
|     <string name="display_header_drawer_summary">在侧边栏中显示带有 Selfoss 链接地址的页眉。</string> | ||||
|     <string name="display_header_drawer_title">帐户页眉</string> | ||||
|     <string name="pref_general_infinite_loading_title">翻页时载入更多文章</string> | ||||
|     <string name="translation">翻译</string> | ||||
|     <string name="cant_open_invalid_url">项目链接地址无效。我正在设法解决这个问题,以避免应用程序崩溃。</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">底部栏将始终显示</string> | ||||
|     <string name="reader_static_bar_off">底部栏可以通过浮动按钮显示</string> | ||||
|     <string name="remove_source">删除源</string> | ||||
|     <string name="pref_theme_title">浅色/深色模式</string> | ||||
|     <string name="mode_dark">深色模式</string> | ||||
|     <string name="mode_system">遵循系统设置</string> | ||||
|     <string name="mode_light">浅色模式</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -47,12 +47,9 @@ | ||||
|     <string name="switch_unread_count_title">"显示未读数"</string> | ||||
|     <string name="display_all_counts_title">"显示收藏和已读的计数"</string> | ||||
|     <string name="text_wrong_url">"您似乎试图使用无效的 URL。确保它是正确的,如果问题仍然存在,请与我联系 (通过商店的联系链接)。请注意,该应用程序需要您使用 Selfoss。没有它,您无法访问 RSS 源。"</string> | ||||
|     <string name="pref_general_internal_browser_title">"打开应用程序中的链接"</string> | ||||
|     <string name="pref_general_internal_browser_on">"文章将在应用程序内打开"</string> | ||||
|     <string name="pref_general_internal_browser_off">"文章将使用默认浏览器打开"</string> | ||||
|     <string name="prefer_article_viewer_title">"使用文章查看器"</string> | ||||
|     <string name="prefer_article_viewer_on">"将使用文章查看器而不是内部浏览器"</string> | ||||
|     <string name="prefer_article_viewer_off">"将使用内部浏览器而不是文章查看器"</string> | ||||
|     <string name="pref_article_viewer_title">"打开应用程序中的链接"</string> | ||||
|     <string name="pref_article_viewer_on">"文章将在应用程序内打开"</string> | ||||
|     <string name="pref_article_viewer_off">"文章将使用默认浏览器打开"</string> | ||||
|     <string name="pref_general_category_links">"链接处理"</string> | ||||
|     <string name="pref_general_category_displaying">"显示"</string> | ||||
|     <string name="pref_switch_card_view_on">"这些文章将以卡片形式显示"</string> | ||||
| @@ -66,14 +63,11 @@ | ||||
|     <string name="card_height_off">卡片高度将被固定</string> | ||||
|     <string name="source_code">源代码</string> | ||||
|     <string name="drawer_error_loading_tags">加载标记时出错..。</string> | ||||
|     <string name="drawer_error_loading_sources">加载源时出错..。</string> | ||||
|     <string name="drawer_item_filters">搜索条件</string> | ||||
|     <string name="drawer_action_clear">清空</string> | ||||
|     <string name="drawer_item_tags">标签</string> | ||||
|     <string name="drawer_item_sources">来源</string> | ||||
|     <string name="drawer_action_edit">编辑</string> | ||||
|     <string name="no_tags_loaded">未加载标签</string> | ||||
|     <string name="no_sources_loaded">未加载数据源</string> | ||||
|     <string name="drawer_loading">正在载入…</string> | ||||
|     <string name="menu_home_search">搜索</string> | ||||
|     <string name="can_delete_source">无法删除数据源…</string> | ||||
| @@ -84,8 +78,6 @@ | ||||
|     <string name="pref_selfoss_category">塞尔福斯 Api</string> | ||||
|     <string name="pref_api_items_number_title">已加载项目编号</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string name="display_header_drawer_summary">在侧边栏中显示带有 Selfoss 链接地址的页眉。</string> | ||||
|     <string name="display_header_drawer_title">帐户页眉</string> | ||||
|     <string name="pref_general_infinite_loading_title">翻页时载入更多文章</string> | ||||
|     <string name="translation">翻译</string> | ||||
|     <string name="cant_open_invalid_url">项目链接地址无效。我正在设法解决这个问题,以避免应用程序崩溃。</string> | ||||
| @@ -136,4 +128,9 @@ | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -2,5 +2,8 @@ | ||||
| <resources> | ||||
|     <declare-styleable name="Theme"> | ||||
|         <attr name="colorAccentDark" format="reference|color" /> | ||||
|         <attr name="bottomBarBackground" format="reference|color" /> | ||||
|         <attr name="toolbarPopupTheme" format="reference|color" /> | ||||
|         <attr name="webviewBackground" format="reference|color" /> | ||||
|     </declare-styleable> | ||||
| </resources> | ||||
| @@ -11,6 +11,5 @@ | ||||
|     <color name="refresh_progress_1">@color/colorAccentDark</color> | ||||
|     <color name="refresh_progress_2">@color/colorAccent</color> | ||||
|     <color name="refresh_progress_3">@color/pink</color> | ||||
|  | ||||
|     <color name="darkBackground">#303030</color> | ||||
|     <color name="dark">#FF282828</color> | ||||
| </resources> | ||||
|   | ||||
							
								
								
									
										32
									
								
								androidApp/src/main/res/values/mode_settings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								androidApp/src/main/res/values/mode_settings.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources> | ||||
|     <string-array name="ModeTitles"> | ||||
|         <item>@string/mode_light</item> | ||||
|         <item>@string/mode_dark</item> | ||||
|         <item>@string/mode_system</item> | ||||
|     </string-array> | ||||
|  | ||||
|     <string-array name="ModeValues"> | ||||
|         <item>1</item> <!--MODE_NIGHT_NO--> | ||||
|         <item>2</item> <!--MODE_NIGHT_YES--> | ||||
|         <item>-1</item> <!--MODE_NIGHT_FOLLOW_SYSTEM--> | ||||
|     </string-array> | ||||
|  | ||||
|     <string-array name="Voice"> | ||||
|         <item>Male</item> | ||||
|         <item>Female</item> | ||||
|     </string-array> | ||||
|  | ||||
|     <string-array name="VoiceAlias"> | ||||
|         <item>"usenglishmale"</item> | ||||
|         <item>"usenglishfemale"</item> | ||||
|         <item>"ukenglishmale"</item> | ||||
|         <item>"ukenglishfemale"</item> | ||||
|         <item>"eurfrenchmale"</item> | ||||
|         <item>"eurfrenchfemale"</item> | ||||
|         <item>"eurspanishmale"</item> | ||||
|         <item>"eurspanishfemale"</item> | ||||
|         <item>"euritalianmale"</item> | ||||
|         <item>"euritalianfemale"</item> | ||||
|     </string-array> | ||||
| </resources> | ||||
| @@ -3,5 +3,6 @@ | ||||
|     <array name="preloaded_fonts" translatable="false"> | ||||
|         <item>@font/open_sans</item> | ||||
|         <item>@font/roboto</item> | ||||
|         <item>@font/source_code_pro_medium</item> | ||||
|     </array> | ||||
| </resources> | ||||
|   | ||||
| @@ -4,5 +4,6 @@ | ||||
|         <item></item> | ||||
|         <item>@string/open_sans_font_id</item> | ||||
|         <item>@string/roboto_font_id</item> | ||||
|         <item>@string/source_code_pro_font_id</item> | ||||
|     </array> | ||||
| </resources> | ||||
| @@ -4,5 +4,6 @@ | ||||
|         <item>Systems</item> | ||||
|         <item>Open Sans</item> | ||||
|         <item>Roboto</item> | ||||
|         <item>Source Code Pro</item> | ||||
|     </array> | ||||
| </resources> | ||||
| @@ -46,12 +46,9 @@ | ||||
|     <string name="switch_unread_count_title">"Display unread count"</string> | ||||
|     <string name="display_all_counts_title">"Display count for favorite and read"</string> | ||||
|     <string name="text_wrong_url">"You seem to be trying to use an invalid URL. Make sure it is correct, and if the problem persists, contact me (via the store contact link). Please note that the app needs you to be using Selfoss. You can't access RSS feeds without it."</string> | ||||
|     <string name="pref_general_internal_browser_title">"Open links inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_general_internal_browser_off">"Articles will open with your default browser"</string> | ||||
|     <string name="prefer_article_viewer_title">"Use the article viewer"</string> | ||||
|     <string name="prefer_article_viewer_on">"Will use the article viewer instead of the internal browser"</string> | ||||
|     <string name="prefer_article_viewer_off">"Will use the internal browser instead of the article viewer"</string> | ||||
|     <string name="pref_article_viewer_title">"Open links inside the app"</string> | ||||
|     <string name="pref_article_viewer_on">"Articles will open inside the app"</string> | ||||
|     <string name="pref_article_viewer_off">"Articles will open with your default browser"</string> | ||||
|     <string name="pref_general_category_links">"Link handling"</string> | ||||
|     <string name="pref_general_category_displaying">"Displaying"</string> | ||||
|     <string name="pref_switch_card_view_on">"The articles will be displayed as cards"</string> | ||||
| @@ -72,8 +69,6 @@ | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
|     <string name="drawer_item_sources">Sources</string> | ||||
|     <string name="drawer_action_edit">edit</string> | ||||
|     <string name="no_tags_loaded">No tags loaded</string> | ||||
|     <string name="no_sources_loaded">No sources loaded</string> | ||||
|     <string name="drawer_loading">Loading …</string> | ||||
|     <string name="menu_home_search">Search</string> | ||||
|     <string name="can_delete_source">Can\'t delete the source…</string> | ||||
| @@ -84,9 +79,6 @@ | ||||
|     <string name="pref_selfoss_category">Selfoss Api</string> | ||||
|     <string name="pref_api_items_number_title">Loaded items number</string> | ||||
|     <string name="pref_hidden_tags">Hidden Tags</string> | ||||
|     <string | ||||
|         name="display_header_drawer_summary">Display a header with the selfoss instance url on the lateral drawer.</string> | ||||
|     <string name="display_header_drawer_title">Account header</string> | ||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||
|     <string name="translation">Translation</string> | ||||
|     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> | ||||
| @@ -118,7 +110,7 @@ | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> | ||||
|     <string name="loading_notification_title">Loading ...</string> | ||||
|     <string name="loading_notification_title">Loading …</string> | ||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</string> | ||||
|     <string name="notification_channel_sync">Sync notification</string> | ||||
|     <string name="new_items_channel_sync">New items notification</string> | ||||
| @@ -133,10 +125,15 @@ | ||||
|     <string name="reader_text_align_left">Align left</string> | ||||
|     <string name="reader_text_align_justify">Justify</string> | ||||
|     <string name="settings_reader_font">Reader font</string> | ||||
|     <string name="source_code_pro_font_id" translatable="false">source_code_pro_medium</string> | ||||
|     <string name="open_sans_font_id" translatable="false">open_sans</string> | ||||
|     <string name="roboto_font_id" translatable="false">roboto</string> | ||||
|     <string name="reader_static_bar_title">Static bottom bar in the article viewer</string> | ||||
|     <string name="reader_static_bar_on">The bottom bar will always be displayed</string> | ||||
|     <string name="reader_static_bar_off">The bottom bar can be shown through the floating button</string> | ||||
|     <string name="remove_source">Remove source</string> | ||||
|     <string name="pref_theme_title">Light/Dark mode</string> | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
| </resources> | ||||
|   | ||||
| @@ -4,41 +4,23 @@ | ||||
|         <item name="android:windowBackground">@drawable/background_splash</item> | ||||
|     </style> | ||||
|  | ||||
|     <style name="NoBar" parent="Theme.MaterialComponents.Light.NoActionBar"> | ||||
|     <style name="NoBar" parent="Theme.MaterialComponents.DayNight.NoActionBar"> | ||||
|         <item name="colorPrimary">@color/colorPrimary</item> | ||||
|         <item name="colorPrimaryDark">@color/colorPrimaryDark</item> | ||||
|         <item name="colorAccent">@color/colorAccent</item> | ||||
|         <item name="colorAccentDark">@color/colorAccentDark</item> | ||||
|         <item name="cardBackgroundColor">@color/white</item> | ||||
|         <item name="android:colorBackground">@color/grey_50</item> | ||||
|         <item name="colorSurface">@color/grey_50</item> | ||||
|         <item name="android:textColorPrimary">@color/grey_900</item> | ||||
|         <item name="android:textColorSecondary">@color/grey_400</item> | ||||
|         <item name="materialDrawerStyle">@style/App.materialDrawerStyle</item> | ||||
|         <item name="materialDrawerHeaderStyle">@style/Widget.MaterialDrawerHeaderStyle</item> | ||||
|         <item name="preferenceTheme">@style/PreferenceStyle</item> | ||||
|         <item name="android:statusBarColor">?attr/colorPrimary</item> | ||||
|         <item name="bottomBarBackground">@color/white</item> | ||||
|         <item name="toolbarPopupTheme">@style/ThemeOverlay.AppCompat.Light</item> | ||||
|         <item name="preferenceTheme">@style/PreferenceStyle</item> | ||||
|         <item name="webviewBackground">@color/white</item> | ||||
|     </style> | ||||
|  | ||||
|     <style name="NoBarDark" parent="Theme.MaterialComponents.DayNight.NoActionBar"> | ||||
|         <item name="colorPrimary">@color/colorPrimary</item> | ||||
|         <item name="colorPrimaryDark">@color/colorPrimaryDark</item> | ||||
|         <item name="colorAccent">@color/colorAccent</item> | ||||
|         <item name="colorAccentDark">@color/colorAccentDark</item> | ||||
|         <item name="cardBackgroundColor">@color/grey_800</item> | ||||
|         <item name="android:colorBackground">@color/darkBackground</item> | ||||
|         <item name="colorSurface">@color/darkBackground</item> | ||||
|         <item name="alertDialogTheme">@style/AlertDialogDark</item> | ||||
|         <item name="bnbBackgroundColor">@color/grey_900</item> | ||||
|         <item name="android:textColorPrimary">@color/white</item> | ||||
|         <item name="android:textColorSecondary">@color/grey_600</item> | ||||
|         <item name="materialDrawerStyle">@style/App.materialDrawerStyle</item> | ||||
|         <item name="materialDrawerHeaderStyle">@style/Widget.MaterialDrawerHeaderStyle</item> | ||||
|         <item name="toolbarPopupTheme">@style/ThemeOverlay.AppCompat.Dark</item> | ||||
|         <item name="preferenceTheme">@style/PreferenceStyle</item> | ||||
|     <!-- Preference Theme --> | ||||
|     <style name="PreferenceStyle" parent="@style/PreferenceThemeOverlay"> | ||||
|         <item name="android:tint">?attr/colorOnSurface</item> | ||||
|     </style> | ||||
|  | ||||
|     <!-- ToolBar --> | ||||
|     <style name="ToolBarStyle" parent="Theme.AppCompat"> | ||||
|         <item name="android:textColorPrimary">@color/white</item> | ||||
|         <item name="android:textColorSecondary">@color/white</item> | ||||
| @@ -46,28 +28,4 @@ | ||||
|         <!--<item name="actionOverflowButtonStyle">@style/ActionButtonOverflowStyle</item> | ||||
|         <item name="drawerArrowStyle">@style/DrawerArrowStyle</item>--> | ||||
|     </style> | ||||
|  | ||||
|     <!-- Material Drawer Theme --> | ||||
|     <style name="App.materialDrawerStyle" parent="@style/Widget.MaterialDrawerStyle"> | ||||
|         <item name="materialDrawerPrimaryIcon">?android:textColorPrimary</item> | ||||
|         <item name="materialDrawerSecondaryIcon">?android:textColorPrimary</item> | ||||
|         <item name="materialDrawerSecondaryText">?android:textColorPrimary</item> | ||||
|     </style> | ||||
|        | ||||
|     <!-- Preference Theme --> | ||||
|     <style name="PreferenceStyle" parent="@style/PreferenceThemeOverlay"> | ||||
|         <item name="android:tint">?android:textColorPrimary</item> | ||||
|     </style> | ||||
|  | ||||
|     <!-- Spinner Theme --> | ||||
|     <style name="App.Spinner" parent="Widget.AppCompat.Light.DropDownItem.Spinner"> | ||||
|         <item name="android:textColor">?android:textColorPrimary</item> | ||||
|     </style> | ||||
|  | ||||
|     <!-- Alert dialog Theme --> | ||||
|  | ||||
|     <style name="AlertDialogDark" parent="Theme.MaterialComponents.Dialog"> | ||||
|         <item name="android:background">@color/darkBackground</item> | ||||
|     </style> | ||||
|  | ||||
| </resources> | ||||
|   | ||||
							
								
								
									
										17
									
								
								androidApp/src/main/res/xml/data_extraction_rules.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								androidApp/src/main/res/xml/data_extraction_rules.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <data-extraction-rules> | ||||
|     <cloud-backup> | ||||
|         <exclude domain="root" /> | ||||
|         <exclude domain="file" /> | ||||
|         <exclude domain="database" /> | ||||
|         <exclude domain="sharedpref" /> | ||||
|         <exclude domain="external" /> | ||||
|     </cloud-backup> | ||||
|     <device-transfer> | ||||
|         <exclude domain="root" /> | ||||
|         <exclude domain="file" /> | ||||
|         <exclude domain="database" /> | ||||
|         <exclude domain="sharedpref" /> | ||||
|         <exclude domain="external" /> | ||||
|     </device-transfer> | ||||
| </data-extraction-rules> | ||||
| @@ -6,7 +6,7 @@ | ||||
|  | ||||
|     </PreferenceCategory> | ||||
|     <EditTextPreference | ||||
|         android:defaultValue="200" | ||||
|         android:defaultValue="20" | ||||
|         android:inputType="number" | ||||
|         android:key="prefer_api_items_number" | ||||
|         android:selectAllOnFocus="true" | ||||
| @@ -34,18 +34,10 @@ | ||||
|     </PreferenceCategory> | ||||
|     <SwitchPreference | ||||
|         android:defaultValue="true" | ||||
|         android:key="prefer_internal_browser" | ||||
|         android:summaryOff="@string/pref_general_internal_browser_off" | ||||
|         android:summaryOn="@string/pref_general_internal_browser_on" | ||||
|         android:title="@string/pref_general_internal_browser_title" | ||||
|         app:iconSpaceReserved="false"/> | ||||
|     <SwitchPreference | ||||
|         android:defaultValue="true" | ||||
|         android:dependency="prefer_internal_browser" | ||||
|         android:key="prefer_article_viewer" | ||||
|         android:summaryOff="@string/prefer_article_viewer_off" | ||||
|         android:summaryOn="@string/prefer_article_viewer_on" | ||||
|         android:title="@string/prefer_article_viewer_title" | ||||
|         android:summaryOff="@string/pref_article_viewer_off" | ||||
|         android:summaryOn="@string/pref_article_viewer_on" | ||||
|         android:title="@string/pref_article_viewer_title" | ||||
|         app:iconSpaceReserved="false"/> | ||||
|     <SwitchPreference | ||||
|         android:defaultValue="false" | ||||
| @@ -60,12 +52,6 @@ | ||||
|         android:title="@string/pref_general_category_displaying"> | ||||
|  | ||||
|     </PreferenceCategory> | ||||
|     <SwitchPreference | ||||
|         android:defaultValue="false" | ||||
|         android:key="account_header_displaying" | ||||
|         android:summary="@string/display_header_drawer_summary" | ||||
|         android:title="@string/display_header_drawer_title" | ||||
|         app:iconSpaceReserved="false"/> | ||||
|     <SwitchPreference | ||||
|         android:defaultValue="false" | ||||
|         android:key="card_view_active" | ||||
|   | ||||
| @@ -1,35 +1,13 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto"> | ||||
|     <!-- TODO translate this file --> | ||||
|  | ||||
|     <SwitchPreference | ||||
|         android:defaultValue="false" | ||||
|         android:key="dark_theme" | ||||
|     <ListPreference | ||||
|         android:defaultValue="0" | ||||
|         android:entries="@array/ModeTitles" | ||||
|         android:entryValues="@array/ModeValues" | ||||
|         android:key="currentMode" | ||||
|         app:iconSpaceReserved="false" | ||||
|         android:title="Dark theme" /> | ||||
|  | ||||
|     <com.jaredrummler.android.colorpicker.ColorPreferenceCompat | ||||
|         android:defaultValue="@color/colorPrimary" | ||||
|         android:key="color_primary" | ||||
|         app:iconSpaceReserved="false" | ||||
|         android:title="Primary color"/> | ||||
|  | ||||
|     <com.jaredrummler.android.colorpicker.ColorPreferenceCompat | ||||
|         android:defaultValue="@color/colorPrimaryDark" | ||||
|         android:key="color_primary_dark" | ||||
|         app:iconSpaceReserved="false" | ||||
|         android:title="Primary dark color"/> | ||||
|  | ||||
|     <com.jaredrummler.android.colorpicker.ColorPreferenceCompat | ||||
|         android:defaultValue="@color/colorAccent" | ||||
|         android:key="color_accent" | ||||
|         app:iconSpaceReserved="false" | ||||
|         android:title="Accent color"/> | ||||
|  | ||||
|     <com.jaredrummler.android.colorpicker.ColorPreferenceCompat | ||||
|         android:defaultValue="@color/colorAccentDark" | ||||
|         android:key="color_accent_dark" | ||||
|         app:iconSpaceReserved="false" | ||||
|         android:title="Accent dark color"/> | ||||
|         android:title="@string/pref_theme_title" | ||||
|         app:useSimpleSummaryProvider="false" /> | ||||
| </PreferenceScreen> | ||||
							
								
								
									
										33
									
								
								androidApp/src/test/kotlin/DatesTest.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								androidApp/src/test/kotlin/DatesTest.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| package bou.amine.apps.readerforselfossv2.repository | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.utils.DateUtils | ||||
| import junit.framework.TestCase.assertEquals | ||||
| import kotlinx.datetime.LocalDateTime | ||||
| import kotlinx.datetime.TimeZone | ||||
| import kotlinx.datetime.toInstant | ||||
| import org.junit.Test | ||||
|  | ||||
| class DatesTest { | ||||
|  | ||||
|     private val v3Date = "2013-04-07T13:43:00+01:00" | ||||
|     private val v4Date = "2013-04-07 13:43:00" | ||||
|  | ||||
|     @Test | ||||
|     fun v3_date_should_be_parsed() { | ||||
|         val date = DateUtils.parseDate(v3Date) | ||||
|         val expected = LocalDateTime(2013, 4, 7, 13, 43, 0, 0).toInstant(TimeZone.of("UTC+1")) .toEpochMilliseconds() | ||||
|  | ||||
|         assertEquals(date, expected) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun v4_date_should_be_parsed() { | ||||
|         val date = DateUtils.parseDate(v4Date) | ||||
|         val expected = | ||||
|             LocalDateTime(2013, 4, 7, 13, 43, 0, 0).toInstant(TimeZone.currentSystemDefault()) | ||||
|                 .toEpochMilliseconds() | ||||
|  | ||||
|         assertEquals(date, expected) | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										1370
									
								
								androidApp/src/test/kotlin/RepositoryTest.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1370
									
								
								androidApp/src/test/kotlin/RepositoryTest.kt
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										57
									
								
								androidApp/src/test/kotlin/TestUtils.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								androidApp/src/test/kotlin/TestUtils.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | ||||
| package bou.amine.apps.readerforselfossv2.repository | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.dao.ITEM | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
|  | ||||
|  | ||||
| fun generateTestDBItems(item: FakeItemParameters = FakeItemParameters()): List<ITEM> { | ||||
|     return listOf( | ||||
|         ITEM( | ||||
|             id = item.id, | ||||
|             datetime = item.datetime, | ||||
|             title = item.title, | ||||
|             content = item.content, | ||||
|             unread = item.unread, | ||||
|             starred = item.starred, | ||||
|             thumbnail = item.thumbnail, | ||||
|             icon = item.icon, | ||||
|             link = item.link, | ||||
|             sourcetitle = item.sourcetitle, | ||||
|             tags = item.tags | ||||
|         ) | ||||
|     ) | ||||
| } | ||||
|  | ||||
| fun generateTestApiItem(item: FakeItemParameters = FakeItemParameters()): List<SelfossModel.Item> { | ||||
|     return listOf( | ||||
|         SelfossModel.Item( | ||||
|             id = item.id.toInt(), | ||||
|             datetime = item.datetime, | ||||
|             title = item.title, | ||||
|             content = item.content, | ||||
|             unread = item.unread, | ||||
|             starred = item.starred, | ||||
|             thumbnail = item.thumbnail, | ||||
|             icon = item.icon, | ||||
|             link = item.link, | ||||
|             sourcetitle = item.sourcetitle, | ||||
|             tags = item.tags.split(',') | ||||
|         ) | ||||
|     ) | ||||
| } | ||||
|  | ||||
| class FakeItemParameters { | ||||
|     var id = "20" | ||||
|     var datetime = "2022-09-09T03:32:01-04:00" | ||||
|     val title = "Etica della ricerca sotto i riflettori." | ||||
|     val content = | ||||
|         "<p><strong>Luigi Campanella, già Presidente SCI</strong></p>\n<p>L’etica della scienza è di certo ambito di cui continuiamo a scoprire nuovi aspetti e risvolti.</p>\n<p>L’ultimo è quello delle intelligenze artificiali capaci di creare opere complesse basate su immagini e parole memorizzate con il rischio di fake news e di contenuti disturbanti.</p>\n<p>Per evitare che ciò accada si sta procedendo filtrando secondo criteri di autocensura i dati da cui l’intelligenza artificiale parte.</p>\n<p>Comincia ad intravedersi un futuro prossimo di competizione fra autori umani ed artificiali nel quale sarà importante, quando i loro prodotti saranno indistinguibili, dichiararne l’origine.</p>\n<p>Come si comprende, si conferma che gli aspetti etici dell’innovazione e della ricerca si diversificato sempre di più.</p>\n<p>La biologia molecolare e la genetica già in passato hanno posto all’attenzione comune aspetti di etica della scienza che hanno indotto a nuove riflessioni circa i limiti delle ricerche.</p>\n<p>L’argomento, sempre attuale, torna sulle prime pagine a seguito della pubblicazione di una ricerca della Università di Cambridge che ha sviluppato una struttura cellulare di un topo con un cuore che batte regolarmente.</p>\n<img src=\"https://ilblogdellasci.files.wordpress.com/2022/09/image002-1.png?w=481\" alt=\"\" width=\"697\" height=\"430\" /><img src=\"https://ilblogdellasci.files.wordpress.com/2022/09/image003-1.png?w=906\" alt=\"\" /><p>Magdalena Zernicka-Goetz</p>\n<img src=\"https://ilblogdellasci.files.wordpress.com/2022/09/image004.jpg?w=474\" alt=\"\" width=\"622\" height=\"465\" /><p>Gianluca Amadei</p>\n<p>Del gruppo fa parte anche uno scienziato italiano Gianluca Amadei,che dinnanzi alle obiezioni di natura etica sulla realizzazione della vita artificiale si è affrettato a sostenere che non è creare nuove vite il fine primario della ricerca, ma quello di salvare quelle esistenti, di dare contributi essenziali alla medicina citando il caso del fallimento tuttora non interpretato di alcune gravidanze e di superare la sperimentazione animale, così contribuendo positivamente alla soluzione di un altro dilemma etico.</p>\n<p>L’embrione sintetico ha ovviamente come primo traguardo il contributo ai trapianti oggi drammaticamente carenti nell’offerta rispetto alla domanda, con attese fino a 4 anni per i trapianti di cuore ed a 2 anni per quelli di fegato. Il lavoro dovrebbe adesso continuare presso l’Ateneo di Padova per creare nuovi organi e nuovi farmaci.</p>" | ||||
|     var unread = true | ||||
|     var starred = true | ||||
|     val thumbnail = null | ||||
|     val icon = "ba79e238383ce83c23a169929c8906ef.png" | ||||
|     val link = | ||||
|         "https://ilblogdellasci.wordpress.com/2022/09/09/etica-della-ricerca-sotto-i-riflettori/" | ||||
|     var sourcetitle = "La Chimica e la Società" | ||||
|     var tags = "Chimica, Testing" | ||||
| } | ||||
| @@ -1,11 +0,0 @@ | ||||
| #!/bin/bash | ||||
| # You can pass --force as first parameter to force push and tag creation. | ||||
|  | ||||
| echo "Creating tag $@" | ||||
|  | ||||
| TAG="v$@" | ||||
| git tag ${TAG} | ||||
|  | ||||
| echo "Pushing tag" | ||||
|  | ||||
| git push origin ${TAG} | ||||
| @@ -1,21 +1,20 @@ | ||||
| buildscript { | ||||
|     repositories { | ||||
|         gradlePluginPortal() | ||||
|         google() | ||||
|         mavenCentral() | ||||
|     } | ||||
|     dependencies { | ||||
|         classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.6.10") | ||||
|         classpath("com.android.tools.build:gradle:7.2.2") | ||||
|  | ||||
|         // sonarquve | ||||
|         classpath("org.sonarsource.scanner.gradle:sonarqube-gradle-plugin:3.4.0.2513") | ||||
|  | ||||
|         // SqlDelight | ||||
|         classpath("com.squareup.sqldelight:gradle-plugin:1.5.3") | ||||
|         classpath("com.squareup.sqldelight:gradle-plugin:1.5.4") | ||||
|     } | ||||
| } | ||||
|  | ||||
| plugins { | ||||
|     //trick: for the same plugin versions in all sub-modules | ||||
|     id("com.android.application").version("7.3.1").apply(false) | ||||
|     id("com.android.library").version("7.3.1").apply(false) | ||||
|     kotlin("android").version("1.7.20").apply(false) | ||||
|     kotlin("multiplatform").version("1.7.20").apply(false) | ||||
|     id("org.sonarqube").version("3.4.0.2513").apply(false) | ||||
|     id("com.mikepenz.aboutlibraries.plugin").version("10.5.1").apply(false) | ||||
| } | ||||
|  | ||||
| apply(plugin = "org.sonarqube") | ||||
|  | ||||
| allprojects { | ||||
| @@ -27,6 +26,7 @@ allprojects { | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| tasks.register("clean", Delete::class) { | ||||
|     delete(rootProject.buildDir) | ||||
| } | ||||
| } | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| 
 | ||||
| git fetch --tags -p | ||||
| 
 | ||||
| BASE_VERSION="1.7" | ||||
| BASE_VERSION="1" | ||||
| LAST_TAG=$(git tag -l | sort -V | tail -1) | ||||
| 
 | ||||
| INITIAL_VERSION="${BASE_VERSION//./}$(date '+%y%m%j')" | ||||
| @@ -21,11 +21,11 @@ VERSION="${INITIAL_VERSION}${TODAYS_VERSION}" | ||||
| 
 | ||||
| PARAMS_EXCEPT_PUBLISH=$(echo $1 | sed 's/\-\-publish//') | ||||
| 
 | ||||
| ./version.sh ${VERSION} ${PARAMS_EXCEPT_PUBLISH} | ||||
| ./version.sh ${VERSION} ${PARAMS_EXCEPT_PUBLISH} $@ | ||||
| 
 | ||||
| if [[ "$@" == *'--publish'* ]] | ||||
| then | ||||
|     ./publish-version.sh ${VERSION} | ||||
|     ./publish-version.sh ${VERSION} $@ | ||||
| else | ||||
|     echo "Did not publish. If you wanted to do so, call the script with \"--publish\" or \"--publish-local\"." | ||||
| fi | ||||
| @@ -1,3 +1 @@ | ||||
| A new RSS reader for <a href="http://selfoss.aditu.de/">selfoss</a>. | ||||
|  | ||||
| It connects to your selfoss instance (works only with selfoss, and can't work without it), and you'll be able to read and manage all your RSS feeds. | ||||
|   | ||||
| @@ -11,14 +11,26 @@ | ||||
| # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects | ||||
| # org.gradle.parallel=true | ||||
| #Tue Mar 22 16:50:00 CET 2022 | ||||
| #Gradle | ||||
| org.gradle.jvmargs=-Xmx2048M -Dfile.encoding=UTF-8 -Dkotlin.daemon.jvm.options\="-Xmx2048M" | ||||
|  | ||||
| #Kotlin | ||||
| kotlin.code.style=official | ||||
| kotlin.mpp.enableCInteropCommonization=true | ||||
| org.gradle.jvmargs=-Xmx2048M -Dkotlin.daemon.jvm.options\="-Xmx2048M" | ||||
| kotlin.native.enableDependencyPropagation=false | ||||
|  | ||||
| #Android | ||||
| android.useAndroidX=true | ||||
| kotlin.native.enableDependencyPropagation=false | ||||
| #android.nonTransitiveRClass=true | ||||
| android.enableJetifier=true | ||||
|  | ||||
|  | ||||
| #MPP | ||||
| kotlin.mpp.enableCInteropCommonization=true | ||||
| kotlin.mpp.enableGranularSourceSetsMetadata=true | ||||
|  | ||||
|  | ||||
| org.gradle.parallel=true | ||||
| org.gradle.caching=true | ||||
| ignoreGitVersion=false | ||||
| kotlin.native.cacheKind.iosX64=none | ||||
| pushCache=true | ||||
|   | ||||
							
								
								
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| #Wed Feb 09 17:05:19 CET 2022 | ||||
| distributionBase=GRADLE_USER_HOME | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-7.3.3-bin.zip | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-7.4-bin.zip | ||||
| distributionPath=wrapper/dists | ||||
| zipStorePath=wrapper/dists | ||||
| zipStoreBase=GRADLE_USER_HOME | ||||
|   | ||||
| @@ -1,383 +1,383 @@ | ||||
|     // !$*UTF8*$! | ||||
|     { | ||||
|     	archiveVersion = 1; | ||||
|     	classes = { | ||||
|     	}; | ||||
|     	objectVersion = 50; | ||||
|     	objects = { | ||||
| // !$*UTF8*$! | ||||
| { | ||||
| 	archiveVersion = 1; | ||||
| 	classes = { | ||||
| 	}; | ||||
| 	objectVersion = 50; | ||||
| 	objects = { | ||||
|  | ||||
|     /* Begin PBXBuildFile section */ | ||||
| 058557BB273AAA24004C7B11 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 058557BA273AAA24004C7B11 /* Assets.xcassets */; }; | ||||
| 058557D9273AAEEB004C7B11 /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */; }; | ||||
|     		2152FB042600AC8F00CF470E /* iOSApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2152FB032600AC8F00CF470E /* iOSApp.swift */; }; | ||||
|     		7555FF83242A565900829871 /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7555FF82242A565900829871 /* ContentView.swift */; }; | ||||
|     /* End PBXBuildFile section */ | ||||
| /* Begin PBXBuildFile section */ | ||||
| 		058557BB273AAA24004C7B11 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 058557BA273AAA24004C7B11 /* Assets.xcassets */; }; | ||||
| 		058557D9273AAEEB004C7B11 /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */; }; | ||||
| 		2152FB042600AC8F00CF470E /* iOSApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2152FB032600AC8F00CF470E /* iOSApp.swift */; }; | ||||
| 		7555FF83242A565900829871 /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7555FF82242A565900829871 /* ContentView.swift */; }; | ||||
| /* End PBXBuildFile section */ | ||||
|  | ||||
|     /* Begin PBXCopyFilesBuildPhase section */ | ||||
|     		7555FFB4242A642300829871 /* Embed Frameworks */ = { | ||||
|     			isa = PBXCopyFilesBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			dstPath = ""; | ||||
|     			dstSubfolderSpec = 10; | ||||
|     			files = ( | ||||
|     			); | ||||
|     			name = "Embed Frameworks"; | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     		}; | ||||
|     /* End PBXCopyFilesBuildPhase section */ | ||||
| /* Begin PBXCopyFilesBuildPhase section */ | ||||
| 		7555FFB4242A642300829871 /* Embed Frameworks */ = { | ||||
| 			isa = PBXCopyFilesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			dstPath = ""; | ||||
| 			dstSubfolderSpec = 10; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			name = "Embed Frameworks"; | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| /* End PBXCopyFilesBuildPhase section */ | ||||
|  | ||||
|     /* Begin PBXFileReference section */ | ||||
|     		058557BA273AAA24004C7B11 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; }; | ||||
| 058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = "Preview Assets.xcassets"; sourceTree = "<group>"; }; | ||||
|     		2152FB032600AC8F00CF470E /* iOSApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = iOSApp.swift; sourceTree = "<group>"; }; | ||||
|     		7555FF7B242A565900829871 /* iosApp.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = iosApp.app; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
|     		7555FF82242A565900829871 /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = "<group>"; }; | ||||
|     		7555FF8C242A565B00829871 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; }; | ||||
|     /* End PBXFileReference section */ | ||||
| /* Begin PBXFileReference section */ | ||||
| 		058557BA273AAA24004C7B11 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; }; | ||||
| 		058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = "Preview Assets.xcassets"; sourceTree = "<group>"; }; | ||||
| 		2152FB032600AC8F00CF470E /* iOSApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = iOSApp.swift; sourceTree = "<group>"; }; | ||||
| 		7555FF7B242A565900829871 /* iosApp.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = iosApp.app; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
| 		7555FF82242A565900829871 /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = "<group>"; }; | ||||
| 		7555FF8C242A565B00829871 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; }; | ||||
| /* End PBXFileReference section */ | ||||
|  | ||||
|     /* Begin PBXFrameworksBuildPhase section */ | ||||
|     		7555FF78242A565900829871 /* Frameworks */ = { | ||||
|     			isa = PBXFrameworksBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			files = ( | ||||
|     			); | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     		}; | ||||
|     /* End PBXFrameworksBuildPhase section */ | ||||
| /* Begin PBXFrameworksBuildPhase section */ | ||||
| 		7555FF78242A565900829871 /* Frameworks */ = { | ||||
| 			isa = PBXFrameworksBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| /* End PBXFrameworksBuildPhase section */ | ||||
|  | ||||
|     /* Begin PBXGroup section */ | ||||
|     		058557D7273AAEEB004C7B11 /* Preview Content */ = { | ||||
| 	isa = PBXGroup; | ||||
| 	children = ( | ||||
| 		058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */, | ||||
| 	); | ||||
| 	path = "Preview Content"; | ||||
| 	sourceTree = "<group>"; | ||||
| }; | ||||
|     		7555FF72242A565900829871 = { | ||||
|     			isa = PBXGroup; | ||||
|     			children = ( | ||||
|     				7555FF7D242A565900829871 /* iosApp */, | ||||
|     				7555FF7C242A565900829871 /* Products */, | ||||
|     				7555FFB0242A642200829871 /* Frameworks */, | ||||
|     			); | ||||
|     			sourceTree = "<group>"; | ||||
|     		}; | ||||
|     		7555FF7C242A565900829871 /* Products */ = { | ||||
|     			isa = PBXGroup; | ||||
|     			children = ( | ||||
|     				7555FF7B242A565900829871 /* iosApp.app */, | ||||
|     			); | ||||
|     			name = Products; | ||||
|     			sourceTree = "<group>"; | ||||
|     		}; | ||||
|     		7555FF7D242A565900829871 /* iosApp */ = { | ||||
|     			isa = PBXGroup; | ||||
|     			children = ( | ||||
|     				058557BA273AAA24004C7B11 /* Assets.xcassets */, | ||||
|     				7555FF82242A565900829871 /* ContentView.swift */, | ||||
|     				7555FF8C242A565B00829871 /* Info.plist */, | ||||
|     				2152FB032600AC8F00CF470E /* iOSApp.swift */, | ||||
| 		058557D7273AAEEB004C7B11 /* Preview Content */, | ||||
|     			); | ||||
|     			path = iosApp; | ||||
|     			sourceTree = "<group>"; | ||||
|     		}; | ||||
|     		7555FFB0242A642200829871 /* Frameworks */ = { | ||||
|     			isa = PBXGroup; | ||||
|     			children = ( | ||||
|     			); | ||||
|     			name = Frameworks; | ||||
|     			sourceTree = "<group>"; | ||||
|     		}; | ||||
|     /* End PBXGroup section */ | ||||
| /* Begin PBXGroup section */ | ||||
| 		058557D7273AAEEB004C7B11 /* Preview Content */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */, | ||||
| 			); | ||||
| 			path = "Preview Content"; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		7555FF72242A565900829871 = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				7555FF7D242A565900829871 /* iosApp */, | ||||
| 				7555FF7C242A565900829871 /* Products */, | ||||
| 				7555FFB0242A642200829871 /* Frameworks */, | ||||
| 			); | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		7555FF7C242A565900829871 /* Products */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				7555FF7B242A565900829871 /* iosApp.app */, | ||||
| 			); | ||||
| 			name = Products; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		7555FF7D242A565900829871 /* iosApp */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				058557BA273AAA24004C7B11 /* Assets.xcassets */, | ||||
| 				7555FF82242A565900829871 /* ContentView.swift */, | ||||
| 				7555FF8C242A565B00829871 /* Info.plist */, | ||||
| 				2152FB032600AC8F00CF470E /* iOSApp.swift */, | ||||
| 				058557D7273AAEEB004C7B11 /* Preview Content */, | ||||
| 			); | ||||
| 			path = iosApp; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		7555FFB0242A642200829871 /* Frameworks */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 			); | ||||
| 			name = Frameworks; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| /* End PBXGroup section */ | ||||
|  | ||||
|     /* Begin PBXNativeTarget section */ | ||||
|     		7555FF7A242A565900829871 /* iosApp */ = { | ||||
|     			isa = PBXNativeTarget; | ||||
|     			buildConfigurationList = 7555FFA5242A565B00829871 /* Build configuration list for PBXNativeTarget "iosApp" */; | ||||
|     			buildPhases = ( | ||||
|     				7555FFB5242A651A00829871 /* ShellScript */, | ||||
|     				7555FF77242A565900829871 /* Sources */, | ||||
|     				7555FF78242A565900829871 /* Frameworks */, | ||||
|     				7555FF79242A565900829871 /* Resources */, | ||||
|     				7555FFB4242A642300829871 /* Embed Frameworks */, | ||||
|     			); | ||||
|     			buildRules = ( | ||||
|     			); | ||||
|     			dependencies = ( | ||||
|     			); | ||||
|     			name = iosApp; | ||||
|     			productName = iosApp; | ||||
|     			productReference = 7555FF7B242A565900829871 /* iosApp.app */; | ||||
|     			productType = "com.apple.product-type.application"; | ||||
|     		}; | ||||
|     /* End PBXNativeTarget section */ | ||||
| /* Begin PBXNativeTarget section */ | ||||
| 		7555FF7A242A565900829871 /* iosApp */ = { | ||||
| 			isa = PBXNativeTarget; | ||||
| 			buildConfigurationList = 7555FFA5242A565B00829871 /* Build configuration list for PBXNativeTarget "iosApp" */; | ||||
| 			buildPhases = ( | ||||
| 				7555FFB5242A651A00829871 /* ShellScript */, | ||||
| 				7555FF77242A565900829871 /* Sources */, | ||||
| 				7555FF78242A565900829871 /* Frameworks */, | ||||
| 				7555FF79242A565900829871 /* Resources */, | ||||
| 				7555FFB4242A642300829871 /* Embed Frameworks */, | ||||
| 			); | ||||
| 			buildRules = ( | ||||
| 			); | ||||
| 			dependencies = ( | ||||
| 			); | ||||
| 			name = iosApp; | ||||
| 			productName = iosApp; | ||||
| 			productReference = 7555FF7B242A565900829871 /* iosApp.app */; | ||||
| 			productType = "com.apple.product-type.application"; | ||||
| 		}; | ||||
| /* End PBXNativeTarget section */ | ||||
|  | ||||
|     /* Begin PBXProject section */ | ||||
|     		7555FF73242A565900829871 /* Project object */ = { | ||||
|     			isa = PBXProject; | ||||
|     			attributes = { | ||||
|     				LastSwiftUpdateCheck = 1130; | ||||
|     				LastUpgradeCheck = 1130; | ||||
|     				ORGANIZATIONNAME = orgName; | ||||
|     				TargetAttributes = { | ||||
|     					7555FF7A242A565900829871 = { | ||||
|     						CreatedOnToolsVersion = 11.3.1; | ||||
|     					}; | ||||
|     				}; | ||||
|     			}; | ||||
|     			buildConfigurationList = 7555FF76242A565900829871 /* Build configuration list for PBXProject "iosApp" */; | ||||
|     			compatibilityVersion = "Xcode 9.3"; | ||||
|     			developmentRegion = en; | ||||
|     			hasScannedForEncodings = 0; | ||||
|     			knownRegions = ( | ||||
|     				en, | ||||
|     				Base, | ||||
|     			); | ||||
|     			mainGroup = 7555FF72242A565900829871; | ||||
|     			productRefGroup = 7555FF7C242A565900829871 /* Products */; | ||||
|     			projectDirPath = ""; | ||||
|     			projectRoot = ""; | ||||
|     			targets = ( | ||||
|     				7555FF7A242A565900829871 /* iosApp */, | ||||
|     			); | ||||
|     		}; | ||||
|     /* End PBXProject section */ | ||||
| /* Begin PBXProject section */ | ||||
| 		7555FF73242A565900829871 /* Project object */ = { | ||||
| 			isa = PBXProject; | ||||
| 			attributes = { | ||||
| 				LastSwiftUpdateCheck = 1130; | ||||
| 				LastUpgradeCheck = 1130; | ||||
| 				ORGANIZATIONNAME = orgName; | ||||
| 				TargetAttributes = { | ||||
| 					7555FF7A242A565900829871 = { | ||||
| 						CreatedOnToolsVersion = 11.3.1; | ||||
| 					}; | ||||
| 				}; | ||||
| 			}; | ||||
| 			buildConfigurationList = 7555FF76242A565900829871 /* Build configuration list for PBXProject "iosApp" */; | ||||
| 			compatibilityVersion = "Xcode 9.3"; | ||||
| 			developmentRegion = en; | ||||
| 			hasScannedForEncodings = 0; | ||||
| 			knownRegions = ( | ||||
| 				en, | ||||
| 				Base, | ||||
| 			); | ||||
| 			mainGroup = 7555FF72242A565900829871; | ||||
| 			productRefGroup = 7555FF7C242A565900829871 /* Products */; | ||||
| 			projectDirPath = ""; | ||||
| 			projectRoot = ""; | ||||
| 			targets = ( | ||||
| 				7555FF7A242A565900829871 /* iosApp */, | ||||
| 			); | ||||
| 		}; | ||||
| /* End PBXProject section */ | ||||
|  | ||||
|     /* Begin PBXResourcesBuildPhase section */ | ||||
|     		7555FF79242A565900829871 /* Resources */ = { | ||||
|     			isa = PBXResourcesBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			files = ( | ||||
| 		058557D9273AAEEB004C7B11 /* Preview Assets.xcassets in Resources */, | ||||
| 		058557BB273AAA24004C7B11 /* Assets.xcassets in Resources */, | ||||
|     			); | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     		}; | ||||
|     /* End PBXResourcesBuildPhase section */ | ||||
| /* Begin PBXResourcesBuildPhase section */ | ||||
| 		7555FF79242A565900829871 /* Resources */ = { | ||||
| 			isa = PBXResourcesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 				058557D9273AAEEB004C7B11 /* Preview Assets.xcassets in Resources */, | ||||
| 				058557BB273AAA24004C7B11 /* Assets.xcassets in Resources */, | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| /* End PBXResourcesBuildPhase section */ | ||||
|  | ||||
|     /* Begin PBXShellScriptBuildPhase section */ | ||||
|     		7555FFB5242A651A00829871 /* ShellScript */ = { | ||||
|     			isa = PBXShellScriptBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			files = ( | ||||
|     			); | ||||
|     			inputFileListPaths = ( | ||||
|     			); | ||||
|     			inputPaths = ( | ||||
|     			); | ||||
|     			outputFileListPaths = ( | ||||
|     			); | ||||
|     			outputPaths = ( | ||||
|     			); | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     			shellPath = /bin/sh; | ||||
|     			shellScript = "cd \"$SRCROOT/..\"\n./gradlew :shared:embedAndSignAppleFrameworkForXcode\n"; | ||||
|     		}; | ||||
|     /* End PBXShellScriptBuildPhase section */ | ||||
| /* Begin PBXShellScriptBuildPhase section */ | ||||
| 		7555FFB5242A651A00829871 /* ShellScript */ = { | ||||
| 			isa = PBXShellScriptBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			inputFileListPaths = ( | ||||
| 			); | ||||
| 			inputPaths = ( | ||||
| 			); | ||||
| 			outputFileListPaths = ( | ||||
| 			); | ||||
| 			outputPaths = ( | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 			shellPath = /bin/sh; | ||||
| 			shellScript = "cd \"$SRCROOT/..\"\n./gradlew :shared:embedAndSignAppleFrameworkForXcode\n"; | ||||
| 		}; | ||||
| /* End PBXShellScriptBuildPhase section */ | ||||
|  | ||||
|     /* Begin PBXSourcesBuildPhase section */ | ||||
|     		7555FF77242A565900829871 /* Sources */ = { | ||||
|     			isa = PBXSourcesBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			files = ( | ||||
|     				2152FB042600AC8F00CF470E /* iOSApp.swift in Sources */, | ||||
|     				7555FF83242A565900829871 /* ContentView.swift in Sources */, | ||||
|     			); | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     		}; | ||||
|     /* End PBXSourcesBuildPhase section */ | ||||
| /* Begin PBXSourcesBuildPhase section */ | ||||
| 		7555FF77242A565900829871 /* Sources */ = { | ||||
| 			isa = PBXSourcesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 				2152FB042600AC8F00CF470E /* iOSApp.swift in Sources */, | ||||
| 				7555FF83242A565900829871 /* ContentView.swift in Sources */, | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| /* End PBXSourcesBuildPhase section */ | ||||
|  | ||||
|     /* Begin XCBuildConfiguration section */ | ||||
|     		7555FFA3242A565B00829871 /* Debug */ = { | ||||
|     			isa = XCBuildConfiguration; | ||||
|     			buildSettings = { | ||||
|     				ALWAYS_SEARCH_USER_PATHS = NO; | ||||
|     				CLANG_ANALYZER_NONNULL = YES; | ||||
|     				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
|     				CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; | ||||
|     				CLANG_CXX_LIBRARY = "libc++"; | ||||
|     				CLANG_ENABLE_MODULES = YES; | ||||
|     				CLANG_ENABLE_OBJC_ARC = YES; | ||||
|     				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
|     				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; | ||||
|     				CLANG_WARN_BOOL_CONVERSION = YES; | ||||
|     				CLANG_WARN_COMMA = YES; | ||||
|     				CLANG_WARN_CONSTANT_CONVERSION = YES; | ||||
|     				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; | ||||
|     				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; | ||||
|     				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
|     				CLANG_WARN_EMPTY_BODY = YES; | ||||
|     				CLANG_WARN_ENUM_CONVERSION = YES; | ||||
|     				CLANG_WARN_INFINITE_RECURSION = YES; | ||||
|     				CLANG_WARN_INT_CONVERSION = YES; | ||||
|     				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; | ||||
|     				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
|     				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
|     				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
|     				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
|     				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
|     				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
|     				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
|     				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
|     				CLANG_WARN_UNREACHABLE_CODE = YES; | ||||
|     				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; | ||||
|     				COPY_PHASE_STRIP = NO; | ||||
|     				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; | ||||
|     				ENABLE_STRICT_OBJC_MSGSEND = YES; | ||||
|     				ENABLE_TESTABILITY = YES; | ||||
|     				GCC_C_LANGUAGE_STANDARD = gnu11; | ||||
|     				GCC_DYNAMIC_NO_PIC = NO; | ||||
|     				GCC_NO_COMMON_BLOCKS = YES; | ||||
|     				GCC_OPTIMIZATION_LEVEL = 0; | ||||
|     				GCC_PREPROCESSOR_DEFINITIONS = ( | ||||
|     					"DEBUG=1", | ||||
|     					"$(inherited)", | ||||
|     				); | ||||
|     				GCC_WARN_64_TO_32_BIT_CONVERSION = YES; | ||||
|     				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; | ||||
|     				GCC_WARN_UNDECLARED_SELECTOR = YES; | ||||
|     				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||
|     				GCC_WARN_UNUSED_FUNCTION = YES; | ||||
|     				GCC_WARN_UNUSED_VARIABLE = YES; | ||||
|     				IPHONEOS_DEPLOYMENT_TARGET = 14.1; | ||||
|     				MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; | ||||
|     				MTL_FAST_MATH = YES; | ||||
|     				ONLY_ACTIVE_ARCH = YES; | ||||
|     				SDKROOT = iphoneos; | ||||
|     				SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; | ||||
|     				SWIFT_OPTIMIZATION_LEVEL = "-Onone"; | ||||
|     			}; | ||||
|     			name = Debug; | ||||
|     		}; | ||||
|     		7555FFA4242A565B00829871 /* Release */ = { | ||||
|     			isa = XCBuildConfiguration; | ||||
|     			buildSettings = { | ||||
|     				ALWAYS_SEARCH_USER_PATHS = NO; | ||||
|     				CLANG_ANALYZER_NONNULL = YES; | ||||
|     				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
|     				CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; | ||||
|     				CLANG_CXX_LIBRARY = "libc++"; | ||||
|     				CLANG_ENABLE_MODULES = YES; | ||||
|     				CLANG_ENABLE_OBJC_ARC = YES; | ||||
|     				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
|     				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; | ||||
|     				CLANG_WARN_BOOL_CONVERSION = YES; | ||||
|     				CLANG_WARN_COMMA = YES; | ||||
|     				CLANG_WARN_CONSTANT_CONVERSION = YES; | ||||
|     				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; | ||||
|     				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; | ||||
|     				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
|     				CLANG_WARN_EMPTY_BODY = YES; | ||||
|     				CLANG_WARN_ENUM_CONVERSION = YES; | ||||
|     				CLANG_WARN_INFINITE_RECURSION = YES; | ||||
|     				CLANG_WARN_INT_CONVERSION = YES; | ||||
|     				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; | ||||
|     				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
|     				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
|     				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
|     				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
|     				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
|     				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
|     				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
|     				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
|     				CLANG_WARN_UNREACHABLE_CODE = YES; | ||||
|     				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; | ||||
|     				COPY_PHASE_STRIP = NO; | ||||
|     				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; | ||||
|     				ENABLE_NS_ASSERTIONS = NO; | ||||
|     				ENABLE_STRICT_OBJC_MSGSEND = YES; | ||||
|     				GCC_C_LANGUAGE_STANDARD = gnu11; | ||||
|     				GCC_NO_COMMON_BLOCKS = YES; | ||||
|     				GCC_WARN_64_TO_32_BIT_CONVERSION = YES; | ||||
|     				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; | ||||
|     				GCC_WARN_UNDECLARED_SELECTOR = YES; | ||||
|     				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||
|     				GCC_WARN_UNUSED_FUNCTION = YES; | ||||
|     				GCC_WARN_UNUSED_VARIABLE = YES; | ||||
|     				IPHONEOS_DEPLOYMENT_TARGET = 14.1; | ||||
|     				MTL_ENABLE_DEBUG_INFO = NO; | ||||
|     				MTL_FAST_MATH = YES; | ||||
|     				SDKROOT = iphoneos; | ||||
|     				SWIFT_COMPILATION_MODE = wholemodule; | ||||
|     				SWIFT_OPTIMIZATION_LEVEL = "-O"; | ||||
|     				VALIDATE_PRODUCT = YES; | ||||
|     			}; | ||||
|     			name = Release; | ||||
|     		}; | ||||
|     		7555FFA6242A565B00829871 /* Debug */ = { | ||||
|     			isa = XCBuildConfiguration; | ||||
|     			buildSettings = { | ||||
|     				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
|     				CODE_SIGN_STYLE = Automatic; | ||||
| 		DEVELOPMENT_ASSET_PATHS = "\"iosApp/Preview Content\""; | ||||
|     				ENABLE_PREVIEWS = YES; | ||||
|     				FRAMEWORK_SEARCH_PATHS = "$(SRCROOT)/../shared/build/xcode-frameworks/$(CONFIGURATION)/$(SDK_NAME)"; | ||||
|     				INFOPLIST_FILE = iosApp/Info.plist; | ||||
|     				LD_RUNPATH_SEARCH_PATHS = ( | ||||
|     					"$(inherited)", | ||||
|     					"@executable_path/Frameworks", | ||||
|     				); | ||||
|     				OTHER_LDFLAGS = ( | ||||
|     					"$(inherited)", | ||||
|     					"-framework", | ||||
|     					shared, | ||||
|     				); | ||||
|     				PRODUCT_BUNDLE_IDENTIFIER = orgIdentifier.iosApp; | ||||
|     				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
|     				SWIFT_VERSION = 5.0; | ||||
|     				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
|     			}; | ||||
|     			name = Debug; | ||||
|     		}; | ||||
|     		7555FFA7242A565B00829871 /* Release */ = { | ||||
|     			isa = XCBuildConfiguration; | ||||
|     			buildSettings = { | ||||
|     				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
|     				CODE_SIGN_STYLE = Automatic; | ||||
|     				DEVELOPMENT_ASSET_PATHS = "\"iosApp/Preview Content\""; | ||||
|     				ENABLE_PREVIEWS = YES; | ||||
|     				FRAMEWORK_SEARCH_PATHS = "$(SRCROOT)/../shared/build/xcode-frameworks/$(CONFIGURATION)/$(SDK_NAME)"; | ||||
|     				INFOPLIST_FILE = iosApp/Info.plist; | ||||
|     				LD_RUNPATH_SEARCH_PATHS = ( | ||||
|     					"$(inherited)", | ||||
|     					"@executable_path/Frameworks", | ||||
|     				); | ||||
|     				OTHER_LDFLAGS = ( | ||||
|     					"$(inherited)", | ||||
|     					"-framework", | ||||
|     					shared, | ||||
|     				); | ||||
|     				PRODUCT_BUNDLE_IDENTIFIER = orgIdentifier.iosApp; | ||||
|     				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
|     				SWIFT_VERSION = 5.0; | ||||
|     				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
|     			}; | ||||
|     			name = Release; | ||||
|     		}; | ||||
|     /* End XCBuildConfiguration section */ | ||||
| /* Begin XCBuildConfiguration section */ | ||||
| 		7555FFA3242A565B00829871 /* Debug */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ALWAYS_SEARCH_USER_PATHS = NO; | ||||
| 				CLANG_ANALYZER_NONNULL = YES; | ||||
| 				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
| 				CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; | ||||
| 				CLANG_CXX_LIBRARY = "libc++"; | ||||
| 				CLANG_ENABLE_MODULES = YES; | ||||
| 				CLANG_ENABLE_OBJC_ARC = YES; | ||||
| 				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
| 				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; | ||||
| 				CLANG_WARN_BOOL_CONVERSION = YES; | ||||
| 				CLANG_WARN_COMMA = YES; | ||||
| 				CLANG_WARN_CONSTANT_CONVERSION = YES; | ||||
| 				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; | ||||
| 				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; | ||||
| 				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
| 				CLANG_WARN_EMPTY_BODY = YES; | ||||
| 				CLANG_WARN_ENUM_CONVERSION = YES; | ||||
| 				CLANG_WARN_INFINITE_RECURSION = YES; | ||||
| 				CLANG_WARN_INT_CONVERSION = YES; | ||||
| 				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
| 				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
| 				CLANG_WARN_UNREACHABLE_CODE = YES; | ||||
| 				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; | ||||
| 				COPY_PHASE_STRIP = NO; | ||||
| 				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; | ||||
| 				ENABLE_STRICT_OBJC_MSGSEND = YES; | ||||
| 				ENABLE_TESTABILITY = YES; | ||||
| 				GCC_C_LANGUAGE_STANDARD = gnu11; | ||||
| 				GCC_DYNAMIC_NO_PIC = NO; | ||||
| 				GCC_NO_COMMON_BLOCKS = YES; | ||||
| 				GCC_OPTIMIZATION_LEVEL = 0; | ||||
| 				GCC_PREPROCESSOR_DEFINITIONS = ( | ||||
| 					"DEBUG=1", | ||||
| 					"$(inherited)", | ||||
| 				); | ||||
| 				GCC_WARN_64_TO_32_BIT_CONVERSION = YES; | ||||
| 				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; | ||||
| 				GCC_WARN_UNDECLARED_SELECTOR = YES; | ||||
| 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||
| 				GCC_WARN_UNUSED_FUNCTION = YES; | ||||
| 				GCC_WARN_UNUSED_VARIABLE = YES; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 14.1; | ||||
| 				MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; | ||||
| 				MTL_FAST_MATH = YES; | ||||
| 				ONLY_ACTIVE_ARCH = YES; | ||||
| 				SDKROOT = iphoneos; | ||||
| 				SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; | ||||
| 				SWIFT_OPTIMIZATION_LEVEL = "-Onone"; | ||||
| 			}; | ||||
| 			name = Debug; | ||||
| 		}; | ||||
| 		7555FFA4242A565B00829871 /* Release */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ALWAYS_SEARCH_USER_PATHS = NO; | ||||
| 				CLANG_ANALYZER_NONNULL = YES; | ||||
| 				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
| 				CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; | ||||
| 				CLANG_CXX_LIBRARY = "libc++"; | ||||
| 				CLANG_ENABLE_MODULES = YES; | ||||
| 				CLANG_ENABLE_OBJC_ARC = YES; | ||||
| 				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
| 				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; | ||||
| 				CLANG_WARN_BOOL_CONVERSION = YES; | ||||
| 				CLANG_WARN_COMMA = YES; | ||||
| 				CLANG_WARN_CONSTANT_CONVERSION = YES; | ||||
| 				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; | ||||
| 				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; | ||||
| 				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
| 				CLANG_WARN_EMPTY_BODY = YES; | ||||
| 				CLANG_WARN_ENUM_CONVERSION = YES; | ||||
| 				CLANG_WARN_INFINITE_RECURSION = YES; | ||||
| 				CLANG_WARN_INT_CONVERSION = YES; | ||||
| 				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
| 				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
| 				CLANG_WARN_UNREACHABLE_CODE = YES; | ||||
| 				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; | ||||
| 				COPY_PHASE_STRIP = NO; | ||||
| 				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; | ||||
| 				ENABLE_NS_ASSERTIONS = NO; | ||||
| 				ENABLE_STRICT_OBJC_MSGSEND = YES; | ||||
| 				GCC_C_LANGUAGE_STANDARD = gnu11; | ||||
| 				GCC_NO_COMMON_BLOCKS = YES; | ||||
| 				GCC_WARN_64_TO_32_BIT_CONVERSION = YES; | ||||
| 				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; | ||||
| 				GCC_WARN_UNDECLARED_SELECTOR = YES; | ||||
| 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||
| 				GCC_WARN_UNUSED_FUNCTION = YES; | ||||
| 				GCC_WARN_UNUSED_VARIABLE = YES; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 14.1; | ||||
| 				MTL_ENABLE_DEBUG_INFO = NO; | ||||
| 				MTL_FAST_MATH = YES; | ||||
| 				SDKROOT = iphoneos; | ||||
| 				SWIFT_COMPILATION_MODE = wholemodule; | ||||
| 				SWIFT_OPTIMIZATION_LEVEL = "-O"; | ||||
| 				VALIDATE_PRODUCT = YES; | ||||
| 			}; | ||||
| 			name = Release; | ||||
| 		}; | ||||
| 		7555FFA6242A565B00829871 /* Debug */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				DEVELOPMENT_ASSET_PATHS = "\"iosApp/Preview Content\""; | ||||
| 				ENABLE_PREVIEWS = YES; | ||||
| 				FRAMEWORK_SEARCH_PATHS = "$(SRCROOT)/../shared/build/xcode-frameworks/$(CONFIGURATION)/$(SDK_NAME)"; | ||||
| 				INFOPLIST_FILE = iosApp/Info.plist; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 				); | ||||
| 				OTHER_LDFLAGS = ( | ||||
| 					"$(inherited)", | ||||
| 					"-framework", | ||||
| 					shared, | ||||
| 				); | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = orgIdentifier.iosApp; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 			}; | ||||
| 			name = Debug; | ||||
| 		}; | ||||
| 		7555FFA7242A565B00829871 /* Release */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				DEVELOPMENT_ASSET_PATHS = "\"iosApp/Preview Content\""; | ||||
| 				ENABLE_PREVIEWS = YES; | ||||
| 				FRAMEWORK_SEARCH_PATHS = "$(SRCROOT)/../shared/build/xcode-frameworks/$(CONFIGURATION)/$(SDK_NAME)"; | ||||
| 				INFOPLIST_FILE = iosApp/Info.plist; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 				); | ||||
| 				OTHER_LDFLAGS = ( | ||||
| 					"$(inherited)", | ||||
| 					"-framework", | ||||
| 					shared, | ||||
| 				); | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = orgIdentifier.iosApp; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 			}; | ||||
| 			name = Release; | ||||
| 		}; | ||||
| /* End XCBuildConfiguration section */ | ||||
|  | ||||
|     /* Begin XCConfigurationList section */ | ||||
|     		7555FF76242A565900829871 /* Build configuration list for PBXProject "iosApp" */ = { | ||||
|     			isa = XCConfigurationList; | ||||
|     			buildConfigurations = ( | ||||
|     				7555FFA3242A565B00829871 /* Debug */, | ||||
|     				7555FFA4242A565B00829871 /* Release */, | ||||
|     			); | ||||
|     			defaultConfigurationIsVisible = 0; | ||||
|     			defaultConfigurationName = Release; | ||||
|     		}; | ||||
|     		7555FFA5242A565B00829871 /* Build configuration list for PBXNativeTarget "iosApp" */ = { | ||||
|     			isa = XCConfigurationList; | ||||
|     			buildConfigurations = ( | ||||
|     				7555FFA6242A565B00829871 /* Debug */, | ||||
|     				7555FFA7242A565B00829871 /* Release */, | ||||
|     			); | ||||
|     			defaultConfigurationIsVisible = 0; | ||||
|     			defaultConfigurationName = Release; | ||||
|     		}; | ||||
|     /* End XCConfigurationList section */ | ||||
|     	}; | ||||
|     	rootObject = 7555FF73242A565900829871 /* Project object */; | ||||
|     } | ||||
| /* Begin XCConfigurationList section */ | ||||
| 		7555FF76242A565900829871 /* Build configuration list for PBXProject "iosApp" */ = { | ||||
| 			isa = XCConfigurationList; | ||||
| 			buildConfigurations = ( | ||||
| 				7555FFA3242A565B00829871 /* Debug */, | ||||
| 				7555FFA4242A565B00829871 /* Release */, | ||||
| 			); | ||||
| 			defaultConfigurationIsVisible = 0; | ||||
| 			defaultConfigurationName = Release; | ||||
| 		}; | ||||
| 		7555FFA5242A565B00829871 /* Build configuration list for PBXNativeTarget "iosApp" */ = { | ||||
| 			isa = XCConfigurationList; | ||||
| 			buildConfigurations = ( | ||||
| 				7555FFA6242A565B00829871 /* Debug */, | ||||
| 				7555FFA7242A565B00829871 /* Release */, | ||||
| 			); | ||||
| 			defaultConfigurationIsVisible = 0; | ||||
| 			defaultConfigurationName = Release; | ||||
| 		}; | ||||
| /* End XCConfigurationList section */ | ||||
| 	}; | ||||
| 	rootObject = 7555FF73242A565900829871 /* Project object */; | ||||
| } | ||||
|   | ||||
| @@ -2,12 +2,9 @@ import SwiftUI | ||||
| import shared | ||||
|  | ||||
| struct ContentView: View { | ||||
|     let greet = Greeting().greeting() | ||||
|  | ||||
|     let toto = SelfossApi().getItems() | ||||
|  | ||||
| 	var body: some View { | ||||
| 		Text(greet) | ||||
| 		Text("ototot") | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										15
									
								
								publish-version.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										15
									
								
								publish-version.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| # NOTE: This is copy/pasted in jenkins | ||||
|  | ||||
| rm -f version.txt | ||||
| printf "versionName=$1-github\nversionCode=$1" >> version.txt | ||||
|  | ||||
| if [[ "$@" == *'--from-ci'* ]] | ||||
| then | ||||
|         echo "File created. HANDLE IN CI" | ||||
| else | ||||
|   # You'll need to change server as your server and define a VERSION_PATH. | ||||
|   scp version.txt server:$VERSION_PATH | ||||
|   rm version.txt | ||||
| fi | ||||
| @@ -8,6 +8,14 @@ pluginManagement { | ||||
|     } | ||||
| } | ||||
|  | ||||
| dependencyResolutionManagement { | ||||
|     repositories { | ||||
|         google() | ||||
|         mavenCentral() | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| buildCache { | ||||
|     remote<HttpBuildCache> { | ||||
|         url = uri("http://18.0.0.7:3071/cache/") | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| object SqlDelight { | ||||
|     const val runtime = "com.squareup.sqldelight:runtime:1.5.3" | ||||
|     const val android = "com.squareup.sqldelight:android-driver:1.5.3" | ||||
|     const val native = "com.squareup.sqldelight:native-driver:1.5.3" | ||||
|     const val runtime = "com.squareup.sqldelight:runtime:1.5.4" | ||||
|     const val android = "com.squareup.sqldelight:android-driver:1.5.4" | ||||
|     const val native = "com.squareup.sqldelight:native-driver:1.5.4" | ||||
|  | ||||
| } | ||||
|  | ||||
| @@ -18,7 +18,7 @@ kotlin { | ||||
|     listOf( | ||||
|         iosX64(), | ||||
|         iosArm64(), | ||||
|         //iosSimulatorArm64() sure all ios dependencies support this target | ||||
|         // iosSimulatorArm64() | ||||
|     ).forEach { | ||||
|         it.binaries.framework { | ||||
|             baseName = "shared" | ||||
| @@ -28,26 +28,23 @@ kotlin { | ||||
|     sourceSets { | ||||
|         val commonMain by getting { | ||||
|             dependencies { | ||||
|                 implementation("io.ktor:ktor-client-core:2.0.1") | ||||
|                 implementation("io.ktor:ktor-client-content-negotiation:2.0.1") | ||||
|                 implementation("io.ktor:ktor-serialization-kotlinx-json:2.0.1") | ||||
|                 implementation("io.ktor:ktor-client-logging:2.0.1") | ||||
|                 implementation("io.ktor:ktor-client-core:2.1.1") | ||||
|                 implementation("io.ktor:ktor-client-content-negotiation:2.1.1") | ||||
|                 implementation("io.ktor:ktor-serialization-kotlinx-json:2.1.1") | ||||
|                 implementation("io.ktor:ktor-client-logging:2.1.1") | ||||
|                 implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.0") | ||||
|                 implementation("io.ktor:ktor-client-auth:2.0.1") | ||||
|                 implementation("io.ktor:ktor-client-auth:2.1.1") | ||||
|                 implementation("org.jsoup:jsoup:1.14.3") | ||||
|  | ||||
|                 //Dependency Injection | ||||
|                 implementation("org.kodein.di:kodein-di:7.12.0") | ||||
|  | ||||
|                 //Settings | ||||
|                 implementation("com.russhwolf:multiplatform-settings-no-arg:0.9") | ||||
|                 implementation("com.russhwolf:multiplatform-settings-no-arg:1.0.0-RC") | ||||
|  | ||||
|                 //Logging | ||||
|                 implementation("io.github.aakira:napier:2.6.1") | ||||
|  | ||||
|                 // Network information | ||||
|                 implementation("com.github.ln-12:multiplatform-connectivity-status:1.3.0") | ||||
|  | ||||
|                 // Sql | ||||
|                 implementation(SqlDelight.runtime) | ||||
|             } | ||||
| @@ -60,7 +57,8 @@ kotlin { | ||||
|         } | ||||
|         val androidMain by getting { | ||||
|             dependencies { | ||||
|                 implementation("io.ktor:ktor-client-android:2.0.1") | ||||
|                 implementation("io.ktor:ktor-client-okhttp:2.1.1") | ||||
|                 implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") | ||||
|  | ||||
|                 // Sql | ||||
|                 implementation(SqlDelight.android) | ||||
| @@ -74,50 +72,49 @@ kotlin { | ||||
|         } | ||||
|         val iosX64Main by getting | ||||
|         val iosArm64Main by getting | ||||
|         //val iosSimulatorArm64Main by getting | ||||
|         // val iosSimulatorArm64Main by getting | ||||
|         val iosMain by creating { | ||||
|             dependsOn(commonMain) | ||||
|             iosX64Main.dependsOn(this) | ||||
|             iosArm64Main.dependsOn(this) | ||||
|             //iosSimulatorArm64Main.dependsOn(this) | ||||
|             // iosSimulatorArm64Main.dependsOn(this) | ||||
|  | ||||
|             // Sql | ||||
|             dependencies { | ||||
|                 implementation(SqlDelight.native) | ||||
|                 implementation("io.ktor:ktor-client-ios:2.1.1") | ||||
|             } | ||||
|         } | ||||
|         val iosX64Test by getting | ||||
|         val iosArm64Test by getting | ||||
|         //val iosSimulatorArm64Test by getting | ||||
|         // val iosSimulatorArm64Test by getting | ||||
|         val iosTest by creating { | ||||
|             dependsOn(commonTest) | ||||
|             iosX64Test.dependsOn(this) | ||||
|             iosArm64Test.dependsOn(this) | ||||
|             dependencies { | ||||
|                 implementation("io.ktor:ktor-client-ios:2.0.1") | ||||
|             } | ||||
|             //iosSimulatorArm64Test.dependsOn(this) | ||||
|             // iosSimulatorArm64Test.dependsOn(this) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| android { | ||||
|     compileSdk = 31 | ||||
|     compileSdk = 32 | ||||
|     sourceSets["main"].manifest.srcFile("src/androidMain/AndroidManifest.xml") | ||||
|     defaultConfig { | ||||
|         minSdk = 21 | ||||
|         targetSdk = 31 | ||||
|         targetSdk = 32 | ||||
|     } | ||||
|     compileOptions { | ||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
|         targetCompatibility = JavaVersion.VERSION_1_8 | ||||
|     } | ||||
|     namespace = "bou.amine.apps.readerforselfossv2" | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| sqldelight { | ||||
|     database("ReaderForSelfossDB") { | ||||
|         packageName = "bou.amine.apps.readerforselfossv2.dao" | ||||
|         sourceFolders = listOf("sqldelight") | ||||
|     } | ||||
| } | ||||
|  | ||||
| } | ||||
| @@ -1,2 +1,2 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <manifest package="bou.amine.apps.readerforselfossv2" /> | ||||
| <manifest /> | ||||
| @@ -1,37 +1,29 @@ | ||||
| package bou.amine.apps.readerforselfossv2.utils | ||||
|  | ||||
| import android.text.format.DateUtils | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import java.time.Instant | ||||
| import java.time.LocalDateTime | ||||
| import java.time.OffsetDateTime | ||||
| import java.time.ZoneOffset | ||||
| import java.time.format.DateTimeFormatter | ||||
| import kotlinx.datetime.* | ||||
|  | ||||
| actual class DateUtils actual constructor(appSettingsService: AppSettingsService) { | ||||
|     val ads: AppSettingsService = appSettingsService // TODO: why is this needed now ? | ||||
|  | ||||
|     actual fun parseDate(dateString: String): Long { | ||||
| actual class DateUtils { | ||||
|     actual companion object { | ||||
|         actual fun parseDate(dateString: String): Long { | ||||
|             return try { | ||||
|                 Instant.parse(dateString).toEpochMilliseconds() | ||||
|             } catch (e: Exception) { | ||||
|                 LocalDateTime.parse(dateString.replace(" ", "T")).toInstant(TimeZone.currentSystemDefault()).toEpochMilliseconds() | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val FORMATTERV1 = "yyyy-MM-dd HH:mm:ss" | ||||
|         actual fun parseRelativeDate(dateString: String): String { | ||||
|  | ||||
|         return if (ads.getApiVersion() >= 4) { | ||||
|             OffsetDateTime.parse(dateString).toInstant().toEpochMilli() | ||||
|         } else { | ||||
|             LocalDateTime.parse(dateString, DateTimeFormatter.ofPattern(FORMATTERV1)).toInstant( | ||||
|                 ZoneOffset.UTC).toEpochMilli() | ||||
|             val date = parseDate(dateString) | ||||
|  | ||||
|             return " " + DateUtils.getRelativeTimeSpanString( | ||||
|                 date, | ||||
|                 Clock.System.now().toEpochMilliseconds(), | ||||
|                 DateUtils.MINUTE_IN_MILLIS, | ||||
|                 DateUtils.FORMAT_ABBREV_RELATIVE | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     actual fun parseRelativeDate(dateString: String): String { | ||||
|  | ||||
|         val date = parseDate(dateString) | ||||
|  | ||||
|         return " " + DateUtils.getRelativeTimeSpanString( | ||||
|             date, | ||||
|             Instant.now().toEpochMilli(), | ||||
|             DateUtils.MINUTE_IN_MILLIS, | ||||
|             DateUtils.FORMAT_ABBREV_RELATIVE | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| @@ -1,5 +1,6 @@ | ||||
| package bou.amine.apps.readerforselfossv2.DI | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.rest.MercuryApi | ||||
| import bou.amine.apps.readerforselfossv2.rest.SelfossApi | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import org.kodein.di.DI | ||||
| @@ -10,4 +11,5 @@ import org.kodein.di.singleton | ||||
| val networkModule by DI.Module { | ||||
|     bind<AppSettingsService>() with singleton { AppSettingsService() } | ||||
|     bind<SelfossApi>() with singleton { SelfossApi(instance()) } | ||||
|     bind<MercuryApi>() with singleton { MercuryApi() } | ||||
| } | ||||
| @@ -0,0 +1,157 @@ | ||||
| package bou.amine.apps.readerforselfossv2.model | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.utils.DateUtils | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import kotlinx.serialization.KSerializer | ||||
| import kotlinx.serialization.Serializable | ||||
| import kotlinx.serialization.descriptors.PrimitiveKind | ||||
| import kotlinx.serialization.descriptors.PrimitiveSerialDescriptor | ||||
| import kotlinx.serialization.descriptors.SerialDescriptor | ||||
| import kotlinx.serialization.encoding.Decoder | ||||
| import kotlinx.serialization.encoding.Encoder | ||||
| import kotlinx.serialization.json.* | ||||
|  | ||||
| class MercuryModel { | ||||
|  | ||||
|     @Serializable | ||||
|     class ParsedContent( | ||||
|         val title: String, | ||||
|         val content: String?, | ||||
|         val lead_image_url: String?, | ||||
|         val url: String | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     data class Tag( | ||||
|         val tag: String, | ||||
|         val color: String, | ||||
|         val unread: Int | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     class Stats( | ||||
|         val total: Int, | ||||
|         val unread: Int, | ||||
|         val starred: Int | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     data class Spout( | ||||
|         val name: String, | ||||
|         val description: String | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     data class ApiVersion( | ||||
|         val version: String?, | ||||
|         val apiversion: String? | ||||
|     ) { | ||||
|         fun getApiMajorVersion() : Int { | ||||
|             var versionNumber = 0 | ||||
|             if (apiversion != null) { | ||||
|                 versionNumber = apiversion.substringBefore(".").toInt() | ||||
|             } | ||||
|             return versionNumber | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Serializable | ||||
|     data class Source( | ||||
|         val id: Int, | ||||
|         val title: String, | ||||
|         @Serializable(with = TagsListSerializer::class) | ||||
|         val tags: List<String>, | ||||
|         val spout: String, | ||||
|         val error: String, | ||||
|         val icon: String? | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     data class Item( | ||||
|         val id: Int, | ||||
|         val datetime: String, | ||||
|         val title: String, | ||||
|         val content: String, | ||||
|         @Serializable(with = BooleanSerializer::class) | ||||
|         var unread: Boolean, | ||||
|         @Serializable(with = BooleanSerializer::class) | ||||
|         var starred: Boolean, | ||||
|         val thumbnail: String?, | ||||
|         val icon: String?, | ||||
|         val link: String, | ||||
|         val sourcetitle: String, | ||||
|         @Serializable(with = TagsListSerializer::class) | ||||
|         val tags: List<String> | ||||
|     ) { | ||||
|         // TODO: maybe find a better way to handle these kind of urls | ||||
|         fun getLinkDecoded(): String { | ||||
|             var stringUrl: String | ||||
|             stringUrl = | ||||
|                 if (link.startsWith("http://news.google.com/news/") || link.startsWith("https://news.google.com/news/")) { | ||||
|                     if (link.contains("&url=")) { | ||||
|                         link.substringAfter("&url=") | ||||
|                     } else { | ||||
|                         this.link.replace("&", "&") | ||||
|                     } | ||||
|                 } else { | ||||
|                     this.link.replace("&", "&") | ||||
|                 } | ||||
|  | ||||
|             // handle :443 => https | ||||
|             if (stringUrl.contains(":443")) { | ||||
|                 stringUrl = stringUrl.replace(":443", "").replace("http://", "https://") | ||||
|             } | ||||
|  | ||||
|             // handle url not starting with http | ||||
|             if (stringUrl.startsWith("//")) { | ||||
|                 stringUrl = "http:$stringUrl" | ||||
|             } | ||||
|  | ||||
|             return stringUrl | ||||
|         } | ||||
|  | ||||
|         fun sourceAndDateText(): String = | ||||
|             this.sourcetitle.getHtmlDecoded() + DateUtils.parseRelativeDate(this.datetime) | ||||
|  | ||||
|         fun toggleStar(): Item { | ||||
|             this.starred = !this.starred | ||||
|             return this | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // TODO: this seems to be super slow. | ||||
|     object TagsListSerializer : KSerializer<List<String>> { | ||||
|         override fun deserialize(decoder: Decoder): List<String> { | ||||
|             return when(val json = ((decoder as JsonDecoder).decodeJsonElement())) { | ||||
|                 is JsonArray -> json.toList().map { it.toString() } | ||||
|                 else -> json.toString().split(",") | ||||
|             } | ||||
|  | ||||
|         } | ||||
|  | ||||
|         override val descriptor: SerialDescriptor | ||||
|             get() = PrimitiveSerialDescriptor("tags", PrimitiveKind.STRING) | ||||
|  | ||||
|         override fun serialize(encoder: Encoder, value: List<String>) { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     object BooleanSerializer : KSerializer<Boolean> { | ||||
|         override fun deserialize(decoder: Decoder): Boolean { | ||||
|             val json = ((decoder as JsonDecoder).decodeJsonElement()).jsonPrimitive | ||||
|             return if (json.booleanOrNull != null) { | ||||
|                 json.boolean | ||||
|             } else { | ||||
|                 json.int == 1 | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         override val descriptor: SerialDescriptor | ||||
|             get() = PrimitiveSerialDescriptor("b", PrimitiveKind.BOOLEAN) | ||||
|  | ||||
|         override fun serialize(encoder: Encoder, value: Boolean) { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,40 @@ | ||||
| package bou.amine.apps.readerforselfossv2.model | ||||
|  | ||||
| import io.ktor.client.call.* | ||||
| import io.ktor.client.statement.* | ||||
| import io.ktor.http.* | ||||
| import kotlinx.serialization.Serializable | ||||
|  | ||||
| @Serializable | ||||
| class SuccessResponse(val success: Boolean) { | ||||
|     val isSuccess: Boolean | ||||
|         get() = success | ||||
| } | ||||
|  | ||||
| class StatusAndData<T>(val success: Boolean, val data: T? = null) { | ||||
|     companion object { | ||||
|         fun <T> succes(d: T): StatusAndData<T> { | ||||
|             return StatusAndData(true, d) | ||||
|         } | ||||
|  | ||||
|         fun <T> error(): StatusAndData<T> { | ||||
|             return StatusAndData(false) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| suspend fun maybeResponse(r: HttpResponse): SuccessResponse { | ||||
|     return if (r.status.isSuccess()) { | ||||
|         r.body() | ||||
|     } else { | ||||
|         SuccessResponse(false) | ||||
|     } | ||||
| } | ||||
|  | ||||
| suspend inline fun <reified T> bodyOrFailure(r: HttpResponse): StatusAndData<T> { | ||||
|     return if (r.status.isSuccess()) { | ||||
|         StatusAndData.succes(r.body()) | ||||
|     } else { | ||||
|         StatusAndData.error() | ||||
|     } | ||||
| } | ||||
| @@ -2,7 +2,14 @@ package bou.amine.apps.readerforselfossv2.model | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.utils.DateUtils | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import kotlinx.serialization.KSerializer | ||||
| import kotlinx.serialization.Serializable | ||||
| import kotlinx.serialization.descriptors.PrimitiveKind | ||||
| import kotlinx.serialization.descriptors.PrimitiveSerialDescriptor | ||||
| import kotlinx.serialization.descriptors.SerialDescriptor | ||||
| import kotlinx.serialization.encoding.Decoder | ||||
| import kotlinx.serialization.encoding.Encoder | ||||
| import kotlinx.serialization.json.* | ||||
|  | ||||
| class SelfossModel { | ||||
|  | ||||
| @@ -13,12 +20,6 @@ class SelfossModel { | ||||
|         val unread: Int | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     class SuccessResponse(val success: Boolean) { | ||||
|         val isSuccess: Boolean | ||||
|             get() = success | ||||
|     } | ||||
|  | ||||
|     @Serializable | ||||
|     class Stats( | ||||
|         val total: Int, | ||||
| @@ -50,6 +51,7 @@ class SelfossModel { | ||||
|     data class Source( | ||||
|         val id: Int, | ||||
|         val title: String, | ||||
|         @Serializable(with = TagsListSerializer::class) | ||||
|         val tags: List<String>, | ||||
|         val spout: String, | ||||
|         val error: String, | ||||
| @@ -62,12 +64,15 @@ class SelfossModel { | ||||
|         val datetime: String, | ||||
|         val title: String, | ||||
|         val content: String, | ||||
|         @Serializable(with = BooleanSerializer::class) | ||||
|         var unread: Boolean, | ||||
|         @Serializable(with = BooleanSerializer::class) | ||||
|         var starred: Boolean, | ||||
|         val thumbnail: String?, | ||||
|         val icon: String?, | ||||
|         val link: String, | ||||
|         val sourcetitle: String, | ||||
|         @Serializable(with = TagsListSerializer::class) | ||||
|         val tags: List<String> | ||||
|     ) { | ||||
|         // TODO: maybe find a better way to handle these kind of urls | ||||
| @@ -97,12 +102,48 @@ class SelfossModel { | ||||
|             return stringUrl | ||||
|         } | ||||
|  | ||||
|         fun sourceAndDateText(dateUtils: DateUtils): String = | ||||
|             this.sourcetitle.getHtmlDecoded() + dateUtils.parseRelativeDate(this.datetime) | ||||
|         fun sourceAndDateText(): String = | ||||
|             this.sourcetitle.getHtmlDecoded() + DateUtils.parseRelativeDate(this.datetime) | ||||
|  | ||||
|         fun toggleStar(): Item { | ||||
|             this.starred = !this.starred | ||||
|             return this | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
|     // TODO: this seems to be super slow. | ||||
|     object TagsListSerializer : KSerializer<List<String>> { | ||||
|         override fun deserialize(decoder: Decoder): List<String> { | ||||
|             return when(val json = ((decoder as JsonDecoder).decodeJsonElement())) { | ||||
|                 is JsonArray -> json.toList().map { it.toString() } | ||||
|                 else -> json.toString().split(",") | ||||
|             } | ||||
|  | ||||
|         } | ||||
|  | ||||
|         override val descriptor: SerialDescriptor | ||||
|             get() = PrimitiveSerialDescriptor("tags", PrimitiveKind.STRING) | ||||
|  | ||||
|         override fun serialize(encoder: Encoder, value: List<String>) { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     object BooleanSerializer : KSerializer<Boolean> { | ||||
|         override fun deserialize(decoder: Decoder): Boolean { | ||||
|             val json = ((decoder as JsonDecoder).decodeJsonElement()).jsonPrimitive | ||||
|             return if (json.booleanOrNull != null) { | ||||
|                 json.boolean | ||||
|             } else { | ||||
|                 json.int == 1 | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         override val descriptor: SerialDescriptor | ||||
|             get() = PrimitiveSerialDescriptor("b", PrimitiveKind.BOOLEAN) | ||||
|  | ||||
|         override fun serialize(encoder: Encoder, value: Boolean) { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -3,23 +3,23 @@ package bou.amine.apps.readerforselfossv2.repository | ||||
| import bou.amine.apps.readerforselfossv2.dao.* | ||||
| import bou.amine.apps.readerforselfossv2.model.NetworkUnavailableException | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.model.StatusAndData | ||||
| import bou.amine.apps.readerforselfossv2.rest.SelfossApi | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.* | ||||
| import com.github.ln_12.library.ConnectivityStatus | ||||
| import io.github.aakira.napier.Napier | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.flow.MutableStateFlow | ||||
| import kotlinx.coroutines.flow.asStateFlow | ||||
| import kotlinx.coroutines.launch | ||||
|  | ||||
| class Repository(private val api: SelfossApi, private val appSettingsService: AppSettingsService, connectivityStatus: ConnectivityStatus, private val db: ReaderForSelfossDB) { | ||||
| class Repository(private val api: SelfossApi, private val appSettingsService: AppSettingsService, val isConnectionAvailable: MutableStateFlow<Boolean>, private val db: ReaderForSelfossDB) { | ||||
|  | ||||
|     var items = ArrayList<SelfossModel.Item>() | ||||
|     val isConnectionAvailable = connectivityStatus.isNetworkConnected | ||||
|     var connectionMonitored = false | ||||
|  | ||||
|     var baseUrl = appSettingsService.getBaseUrl() | ||||
|     lateinit var dateUtils: DateUtils | ||||
|  | ||||
|     var displayedItems = ItemType.UNREAD | ||||
|  | ||||
| @@ -29,25 +29,20 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|  | ||||
|     var offlineOverride = false | ||||
|  | ||||
|     var badgeUnread = 0 | ||||
|     set(value) {field = if (value < 0) { 0 } else { value } } | ||||
|     var badgeAll = 0 | ||||
|     set(value) {field = if (value < 0) { 0 } else { value } } | ||||
|     var badgeStarred = 0 | ||||
|     set(value) {field = if (value < 0) { 0 } else { value } } | ||||
|     private val _badgeUnread = MutableStateFlow(0) | ||||
|     val badgeUnread = _badgeUnread.asStateFlow() | ||||
|     private val _badgeAll = MutableStateFlow(0) | ||||
|     val badgeAll = _badgeAll.asStateFlow() | ||||
|     private val _badgeStarred = MutableStateFlow(0) | ||||
|     val badgeStarred = _badgeStarred.asStateFlow() | ||||
|  | ||||
|     init { | ||||
|         // TODO: Dispatchers.IO not available in KMM, an alternative solution should be found | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             updateApiVersion() | ||||
|             dateUtils = DateUtils(appSettingsService) | ||||
|             reloadBadges() | ||||
|         } | ||||
|     } | ||||
|     private var fetchedSources = false | ||||
|     private var fetchedTags = false | ||||
|  | ||||
|     suspend fun getNewerItems(): ArrayList<SelfossModel.Item> { | ||||
|         // TODO: Use the updatedSince parameter | ||||
|         var fetchedItems: List<SelfossModel.Item>? = null | ||||
|         var fetchedItems: StatusAndData<List<SelfossModel.Item>> = StatusAndData.error() | ||||
|         var fromDB = false | ||||
|         if (isNetworkAvailable()) { | ||||
|             fetchedItems = api.getItems( | ||||
|                 displayedItems.type, | ||||
| @@ -59,23 +54,35 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|             ) | ||||
|         } else { | ||||
|             if (appSettingsService.isItemCachingEnabled()) { | ||||
|                 fetchedItems = getDBItems().filter { | ||||
|                 fromDB = true | ||||
|                 var dbItems = getDBItems().filter { | ||||
|                     displayedItems == ItemType.ALL || | ||||
|                             (it.unread && displayedItems == ItemType.UNREAD) || | ||||
|                             (it.starred && displayedItems == ItemType.STARRED) | ||||
|                 }.map { it.toView() } | ||||
|                 } | ||||
|                 if (tagFilter != null) { | ||||
|                     dbItems = dbItems.filter { it.tags.split(',').contains(tagFilter!!.tag) } | ||||
|                 } | ||||
|                 if (sourceFilter != null) { | ||||
|                     dbItems = dbItems.filter { it.sourcetitle == sourceFilter!!.title } | ||||
|                 } | ||||
|                 fetchedItems = StatusAndData.succes( | ||||
|                     dbItems.map { it.toView() } | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (fetchedItems != null) { | ||||
|             items = ArrayList(fetchedItems) | ||||
|             sortItems() | ||||
|         if (fetchedItems.success && fetchedItems.data != null) { | ||||
|             items = ArrayList(fetchedItems.data!!) | ||||
|             if (fromDB) { | ||||
|                 items.sortByDescending { DateUtils.parseDate(it.datetime) } | ||||
|             } | ||||
|         } | ||||
|         return items | ||||
|     } | ||||
|  | ||||
|     suspend fun getOlderItems(): ArrayList<SelfossModel.Item> { | ||||
|         var fetchedItems: List<SelfossModel.Item>? = null | ||||
|         var fetchedItems: StatusAndData<List<SelfossModel.Item>> = StatusAndData.error() | ||||
|         if (isNetworkAvailable()) { | ||||
|             val offset = items.size | ||||
|             fetchedItems = api.getItems( | ||||
| @@ -88,78 +95,105 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|             ) | ||||
|         } // When using the db cache, we load everything the first time, so there should be nothing more to load. | ||||
|  | ||||
|         if (fetchedItems != null) { | ||||
|             items.addAll(fetchedItems) | ||||
|             sortItems() | ||||
|         if (fetchedItems.success && fetchedItems.data != null) { | ||||
|             items.addAll(fetchedItems.data!!) | ||||
|         } | ||||
|         return items | ||||
|     } | ||||
|  | ||||
|     private suspend fun getMaxItemsForBackground(itemType: ItemType): List<SelfossModel.Item>? { | ||||
|     private suspend fun getMaxItemsForBackground(itemType: ItemType): List<SelfossModel.Item> { | ||||
|         return if (isNetworkAvailable()) { | ||||
|             api.getItems( | ||||
|             val items = api.getItems( | ||||
|                 itemType.type, | ||||
|                 0, | ||||
|                 tagFilter?.tag, | ||||
|                 sourceFilter?.id?.toLong(), | ||||
|                 searchFilter, | ||||
|                 null, | ||||
|                 null, | ||||
|                 null, | ||||
|                 null, | ||||
|                 200 | ||||
|             ) | ||||
|             return if (items.success && items.data != null) { | ||||
|                 items.data | ||||
|             } else { | ||||
|                 emptyList() | ||||
|             } | ||||
|         } else { | ||||
|             emptyList() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun sortItems() { | ||||
|         items.sortByDescending { dateUtils.parseDate(it.datetime) } | ||||
|     } | ||||
|  | ||||
|     suspend fun reloadBadges(): Boolean { | ||||
|         var success = false | ||||
|         if (isNetworkAvailable()) { | ||||
|             val response = api.stats() | ||||
|             if (response != null) { | ||||
|                 badgeUnread = response.unread | ||||
|                 badgeAll = response.total | ||||
|                 badgeStarred = response.starred | ||||
|             if (response.success && response.data != null) { | ||||
|                 _badgeUnread.value = response.data.unread | ||||
|                 _badgeAll.value = response.data.total | ||||
|                 _badgeStarred.value = response.data.starred | ||||
|                 success = true | ||||
|             } | ||||
|         } else { | ||||
|         } else if (appSettingsService.isItemCachingEnabled()) { | ||||
|             // TODO: do this differently, because it's not efficient | ||||
|             val dbItems = getDBItems() | ||||
|             badgeUnread = dbItems.filter { item -> item.unread }.size | ||||
|             badgeStarred = dbItems.filter { item -> item.starred }.size | ||||
|             badgeAll = items.size | ||||
|             _badgeUnread.value = dbItems.filter { item -> item.unread }.size | ||||
|             _badgeStarred.value = dbItems.filter { item -> item.starred }.size | ||||
|             _badgeAll.value = dbItems.size | ||||
|             success = true | ||||
|         } | ||||
|         return success | ||||
|     } | ||||
|  | ||||
|     suspend fun getTags(): List<SelfossModel.Tag>? { | ||||
|         return if (isNetworkAvailable()) { | ||||
|             api.tags() | ||||
|         } else { | ||||
|     suspend fun getTags(): List<SelfossModel.Tag> { | ||||
|         val isDatabaseEnabled = appSettingsService.isItemCachingEnabled() || !appSettingsService.isUpdateSourcesEnabled() | ||||
|         return if (isNetworkAvailable() && !fetchedTags) { | ||||
|             val apiTags = api.tags() | ||||
|             if (apiTags.success && apiTags.data != null && isDatabaseEnabled) { | ||||
|                 resetDBTagsWithData(apiTags.data) | ||||
|                 if (!appSettingsService.isUpdateSourcesEnabled()) { | ||||
|                     fetchedTags = true | ||||
|                 } | ||||
|             } | ||||
|             apiTags.data ?: emptyList() | ||||
|         } else if (isDatabaseEnabled) { | ||||
|             getDBTags().map { it.toView() } | ||||
|         } else { | ||||
|             emptyList() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     suspend fun getSpouts(): Map<String, SelfossModel.Spout>? { | ||||
|     // TODO: Add tests | ||||
|     suspend fun getSpouts(): Map<String, SelfossModel.Spout> { | ||||
|         return if (isNetworkAvailable()) { | ||||
|             api.spouts() | ||||
|             val spouts = api.spouts() | ||||
|             if (spouts.success && spouts.data != null) { | ||||
|                 spouts.data | ||||
|             } else { | ||||
|                 emptyMap() // TODO: do something here | ||||
|             } | ||||
|         } else { | ||||
|             throw NetworkUnavailableException() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     suspend fun getSources(): ArrayList<SelfossModel.Source>? { | ||||
|  | ||||
|         return if (isNetworkAvailable()) { | ||||
|             api.sources() | ||||
|         } else { | ||||
|     suspend fun getSources(): ArrayList<SelfossModel.Source> { | ||||
|         val isDatabaseEnabled = appSettingsService.isItemCachingEnabled() || !appSettingsService.isUpdateSourcesEnabled() | ||||
|         return if (isNetworkAvailable() && !fetchedSources) { | ||||
|             val apiSources = api.sources() | ||||
|             if (apiSources.success && apiSources.data != null && isDatabaseEnabled) { | ||||
|                 resetDBSourcesWithData(apiSources.data) | ||||
|                 if (!appSettingsService.isUpdateSourcesEnabled()) { | ||||
|                     fetchedSources = true | ||||
|                 } | ||||
|             } | ||||
|             apiSources.data ?: ArrayList() | ||||
|         } else if (isDatabaseEnabled) { | ||||
|             ArrayList(getDBSources().map { it.toView() }) | ||||
|         } else { | ||||
|             ArrayList() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // TODO: Add tests | ||||
|     suspend fun markAsRead(item: SelfossModel.Item): Boolean { | ||||
|         val success = markAsReadById(item.id) | ||||
|  | ||||
| @@ -171,14 +205,14 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|  | ||||
|     private suspend fun markAsReadById(id: Int): Boolean { | ||||
|         return if (isNetworkAvailable()) { | ||||
|             api.markAsRead(id.toString())?.isSuccess == true | ||||
|             api.markAsRead(id.toString()).isSuccess | ||||
|         } else { | ||||
|             insertDBAction(id.toString(), read = true) | ||||
|             true | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     // TODO: Add tests | ||||
|     suspend fun unmarkAsRead(item: SelfossModel.Item): Boolean { | ||||
|         val success = unmarkAsReadById(item.id) | ||||
|  | ||||
| @@ -190,13 +224,14 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|  | ||||
|     private suspend fun unmarkAsReadById(id: Int): Boolean { | ||||
|         return if (isNetworkAvailable()) { | ||||
|             api.unmarkAsRead(id.toString())?.isSuccess == true | ||||
|             api.unmarkAsRead(id.toString()).isSuccess | ||||
|         } else { | ||||
|             insertDBAction(id.toString(), unread = true) | ||||
|             true | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // TODO: Add tests | ||||
|     suspend fun starr(item: SelfossModel.Item): Boolean { | ||||
|         val success = starrById(item.id) | ||||
|  | ||||
| @@ -208,13 +243,14 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|  | ||||
|     private suspend fun starrById(id: Int): Boolean { | ||||
|         return if (isNetworkAvailable()) { | ||||
|             api.starr(id.toString())?.isSuccess == true | ||||
|             api.starr(id.toString()).isSuccess | ||||
|         } else { | ||||
|             insertDBAction(id.toString(), starred = true) | ||||
|             true | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // TODO: Add tests | ||||
|     suspend fun unstarr(item: SelfossModel.Item): Boolean { | ||||
|         val success = unstarrById(item.id) | ||||
|  | ||||
| @@ -226,17 +262,18 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|  | ||||
|     private suspend fun unstarrById(id: Int): Boolean { | ||||
|         return if (isNetworkAvailable()) { | ||||
|             api.unstarr(id.toString())?.isSuccess == true | ||||
|             api.unstarr(id.toString()).isSuccess | ||||
|         } else { | ||||
|             insertDBAction(id.toString(), starred = true) | ||||
|             true | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // TODO: Add tests | ||||
|     suspend fun markAllAsRead(items: ArrayList<SelfossModel.Item>): Boolean { | ||||
|         var success = false | ||||
|  | ||||
|         if (isNetworkAvailable() && api.markAllAsRead(items.map { it.id.toString() })?.isSuccess == true) { | ||||
|         if (isNetworkAvailable() && api.markAllAsRead(items.map { it.id.toString() }).isSuccess) { | ||||
|             success = true | ||||
|             for (item in items) { | ||||
|                 markAsReadLocally(item) | ||||
| @@ -248,7 +285,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun markAsReadLocally(item: SelfossModel.Item) { | ||||
|         if (item.unread) { | ||||
|             item.unread = false | ||||
|             badgeUnread -= 1 | ||||
|             _badgeUnread.value -= 1 | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -259,7 +296,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun unmarkAsReadLocally(item: SelfossModel.Item) { | ||||
|         if (!item.unread) { | ||||
|             item.unread = true | ||||
|             badgeUnread += 1 | ||||
|             _badgeUnread.value += 1 | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -270,7 +307,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun starrLocally(item: SelfossModel.Item) { | ||||
|         if (!item.starred) { | ||||
|             item.starred = true | ||||
|             badgeStarred += 1 | ||||
|             _badgeStarred.value += 1 | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -281,7 +318,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun unstarrLocally(item: SelfossModel.Item) { | ||||
|         if (item.starred) { | ||||
|             item.starred = false | ||||
|             badgeStarred -= 1 | ||||
|             _badgeStarred.value -= 1 | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -305,7 +342,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|                 tags, | ||||
|                 filter, | ||||
|                 appSettingsService.getApiVersion() | ||||
|             )?.isSuccess == true | ||||
|             ).isSuccess == true | ||||
|         } | ||||
|  | ||||
|         return response | ||||
| @@ -315,9 +352,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|         var success = false | ||||
|         if (isNetworkAvailable()) { | ||||
|             val response = api.deleteSource(id) | ||||
|             if (response != null) { | ||||
|                 success = response.isSuccess | ||||
|             } | ||||
|             success = response.isSuccess | ||||
|         } | ||||
|  | ||||
|         return success | ||||
| @@ -325,7 +360,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|  | ||||
|     suspend fun updateRemote(): Boolean { | ||||
|         return if (isNetworkAvailable()) { | ||||
|             api.update()?.equals("finished") ?: false | ||||
|             api.update().data.equals("finished") | ||||
|         } else { | ||||
|             false | ||||
|         } | ||||
| @@ -336,10 +371,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|         if (isNetworkAvailable()) { | ||||
|             try { | ||||
|                 val response = api.login() | ||||
|                 result = response?.isSuccess == true | ||||
|                 if (result) { | ||||
|                     updateApiVersion() | ||||
|                 } | ||||
|                 result = response.isSuccess == true | ||||
|             } catch (cause: Throwable) { | ||||
|                 Napier.e(cause.stackTraceToString(), tag = "RepositoryImpl.updateRemote") | ||||
|             } | ||||
| @@ -358,8 +390,8 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|  | ||||
|         if (isNetworkAvailable()) { | ||||
|             val fetchedVersion = api.version() | ||||
|             if (fetchedVersion != null && fetchedVersion.getApiMajorVersion() != apiMajorVersion) { | ||||
|                 appSettingsService.updateApiVersion(fetchedVersion.getApiMajorVersion()) | ||||
|             if (fetchedVersion.success && fetchedVersion.data != null && fetchedVersion.data.getApiMajorVersion() != apiMajorVersion) { | ||||
|                 appSettingsService.updateApiVersion(fetchedVersion.data.getApiMajorVersion()) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| @@ -372,11 +404,11 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun deleteDBAction(action: ACTION) = | ||||
|         db.actionsQueries.deleteAction(action.id) | ||||
|  | ||||
|     fun getDBTags(): List<TAG> = db.tagsQueries.tags().executeAsList() | ||||
|     private fun getDBTags(): List<TAG> = db.tagsQueries.tags().executeAsList() | ||||
|  | ||||
|     fun getDBSources(): List<SOURCE> = db.sourcesQueries.sources().executeAsList() | ||||
|     private fun getDBSources(): List<SOURCE> = db.sourcesQueries.sources().executeAsList() | ||||
|  | ||||
|     fun resetDBTagsWithData(tagEntities: List<SelfossModel.Tag>) { | ||||
|     private fun resetDBTagsWithData(tagEntities: List<SelfossModel.Tag>) { | ||||
|         db.tagsQueries.deleteAllTags() | ||||
|  | ||||
|         db.tagsQueries.transaction { | ||||
| @@ -386,7 +418,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun resetDBSourcesWithData(sources: List<SelfossModel.Source>) { | ||||
|     private fun resetDBSourcesWithData(sources: List<SelfossModel.Source>) { | ||||
|         db.sourcesQueries.deleteAllSources() | ||||
|  | ||||
|         db.sourcesQueries.transaction { | ||||
| @@ -412,13 +444,12 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun updateDBItem(item: SelfossModel.Item) = | ||||
|         db.itemsQueries.updateItem(item.datetime, item.title.getHtmlDecoded(), item.content, item.unread, item.starred, item.thumbnail, item.icon, item.link, item.sourcetitle, item.tags.joinToString(","), item.id.toString()) | ||||
|  | ||||
|  | ||||
|     suspend fun tryToCacheItemsAndGetNewOnes(): List<SelfossModel.Item>? { | ||||
|     suspend fun tryToCacheItemsAndGetNewOnes(): List<SelfossModel.Item> { | ||||
|         try { | ||||
|             val newItems = getMaxItemsForBackground(ItemType.UNREAD) | ||||
|             val allItems = getMaxItemsForBackground(ItemType.ALL) | ||||
|             val starredItems = getMaxItemsForBackground(ItemType.STARRED) | ||||
|             insertDBItems(newItems.orEmpty() + allItems.orEmpty() + starredItems.orEmpty()) | ||||
|             insertDBItems(newItems + allItems + starredItems) | ||||
|             return newItems | ||||
|         } catch (e: Throwable) { | ||||
|             // We do nothing | ||||
| @@ -426,6 +457,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|         return emptyList() | ||||
|     } | ||||
|  | ||||
|     // TODO: Add tests | ||||
|     suspend fun handleDBActions() { | ||||
|  | ||||
|         val actions: List<ACTION> = getDBActions() | ||||
|   | ||||
| @@ -0,0 +1,43 @@ | ||||
| package bou.amine.apps.readerforselfossv2.rest | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.model.* | ||||
| import io.github.aakira.napier.Napier | ||||
| import io.ktor.client.* | ||||
| import io.ktor.client.plugins.cache.* | ||||
| import io.ktor.client.plugins.contentnegotiation.* | ||||
| import io.ktor.client.plugins.logging.* | ||||
| import io.ktor.client.request.* | ||||
| import io.ktor.serialization.kotlinx.json.* | ||||
| import kotlinx.serialization.json.Json | ||||
|  | ||||
| class MercuryApi() { | ||||
|  | ||||
|     var client = createHttpClient() | ||||
|  | ||||
|     private fun createHttpClient(): HttpClient { | ||||
|         return HttpClient { | ||||
|             install(ContentNegotiation) { | ||||
|                 install(HttpCache) | ||||
|                 json(Json { | ||||
|                     prettyPrint = true | ||||
|                     isLenient = true | ||||
|                     ignoreUnknownKeys = true | ||||
|                 }) | ||||
|             } | ||||
|             install(Logging) { | ||||
|                 logger = object : Logger { | ||||
|                     override fun log(message: String) { | ||||
|                         Napier.d(message, tag = "LogMercuryCalls") | ||||
|                     } | ||||
|                 } | ||||
|                 level = LogLevel.INFO | ||||
|             } | ||||
|             expectSuccess = false | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     suspend fun query(url: String): StatusAndData<MercuryModel.ParsedContent> = | ||||
|         bodyOrFailure(client.get("https://amine-louveau.fr/parser.php") { | ||||
|             parameter("link", url) | ||||
|         }) | ||||
| } | ||||
| @@ -1,6 +1,6 @@ | ||||
| package bou.amine.apps.readerforselfossv2.rest | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.model.* | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import io.ktor.client.* | ||||
| import io.ktor.client.call.* | ||||
| @@ -10,6 +10,7 @@ import io.ktor.client.plugins.contentnegotiation.* | ||||
| import io.ktor.client.plugins.logging.* | ||||
| import io.ktor.client.request.* | ||||
| import io.ktor.client.request.forms.* | ||||
| import io.ktor.client.statement.* | ||||
| import io.ktor.http.* | ||||
| import io.ktor.serialization.kotlinx.json.* | ||||
| import kotlinx.serialization.json.Json | ||||
| @@ -34,7 +35,7 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|                         appSettingsService.logApiCalls(message) | ||||
|                     } | ||||
|                 } | ||||
|                 level = LogLevel.ALL | ||||
|                 level = LogLevel.INFO | ||||
|             } | ||||
|             install(HttpTimeout) { | ||||
|                 requestTimeoutMillis = appSettingsService.getApiTimeout() | ||||
| @@ -65,11 +66,11 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         client = createHttpClient() | ||||
|     } | ||||
|  | ||||
|     suspend fun login(): SelfossModel.SuccessResponse? = | ||||
|         client.get(url("/login")) { | ||||
|     suspend fun login(): SuccessResponse = | ||||
|         maybeResponse(client.get(url("/login")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|         }.body() | ||||
|         }) | ||||
|  | ||||
|     suspend fun getItems( | ||||
|         type: String, | ||||
| @@ -79,8 +80,8 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         search: String?, | ||||
|         updatedSince: String?, | ||||
|         items: Int? = null | ||||
|     ): List<SelfossModel.Item>? = | ||||
|         client.get(url("/items")) { | ||||
|     ): StatusAndData<List<SelfossModel.Item>> = | ||||
|         bodyOrFailure(client.get(url("/items")) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|                 parameter("type", type) | ||||
| @@ -90,74 +91,74 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|                 parameter("updatedsince", updatedSince) | ||||
|                 parameter("items", items ?: appSettingsService.getItemsNumber()) | ||||
|                 parameter("offset", offset) | ||||
|             }.body() | ||||
|             }) | ||||
|  | ||||
|     suspend fun stats(): SelfossModel.Stats? = | ||||
|         client.get(url("/stats")) { | ||||
|     suspend fun stats(): StatusAndData<SelfossModel.Stats> = | ||||
|         bodyOrFailure(client.get(url("/stats")) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }.body() | ||||
|             }) | ||||
|  | ||||
|     suspend fun tags(): List<SelfossModel.Tag>? = | ||||
|         client.get(url("/tags")) { | ||||
|     suspend fun tags(): StatusAndData<List<SelfossModel.Tag>> = | ||||
|         bodyOrFailure(client.get(url("/tags")) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }.body() | ||||
|             }) | ||||
|  | ||||
|     suspend fun update(): String? = | ||||
|         client.get(url("/update")) { | ||||
|     suspend fun update(): StatusAndData<String> = | ||||
|         bodyOrFailure(client.get(url("/update")) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }.body() | ||||
|             }) | ||||
|  | ||||
|     suspend fun spouts(): Map<String, SelfossModel.Spout>? = | ||||
|         client.get(url("/sources/spouts")) { | ||||
|     suspend fun spouts(): StatusAndData<Map<String, SelfossModel.Spout>> = | ||||
|         bodyOrFailure(client.get(url("/sources/spouts")) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }.body() | ||||
|             }) | ||||
|  | ||||
|     suspend fun sources(): ArrayList<SelfossModel.Source>? = | ||||
|         client.get(url("/sources/list")) { | ||||
|     suspend fun sources(): StatusAndData<ArrayList<SelfossModel.Source>> = | ||||
|         bodyOrFailure(client.get(url("/sources/list")) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }.body() | ||||
|             }) | ||||
|  | ||||
|     suspend fun version(): SelfossModel.ApiVersion? = | ||||
|         client.get(url("/api/about")).body() | ||||
|     suspend fun version(): StatusAndData<SelfossModel.ApiVersion> = | ||||
|         bodyOrFailure(client.get(url("/api/about"))) | ||||
|  | ||||
|     suspend fun markAsRead(id: String): SelfossModel.SuccessResponse? = | ||||
|         client.post(url("/mark/$id")) { | ||||
|     suspend fun markAsRead(id: String): SuccessResponse = | ||||
|         maybeResponse(client.post(url("/mark/$id")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|         }.body() | ||||
|         }) | ||||
|  | ||||
|     suspend fun unmarkAsRead(id: String): SelfossModel.SuccessResponse? = | ||||
|         client.post(url("/unmark/$id")) { | ||||
|     suspend fun unmarkAsRead(id: String): SuccessResponse = | ||||
|         maybeResponse(client.post(url("/unmark/$id")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|         }.body() | ||||
|         }) | ||||
|  | ||||
|     suspend fun starr(id: String): SelfossModel.SuccessResponse? = | ||||
|         client.post(url("/starr/$id")) { | ||||
|     suspend fun starr(id: String): SuccessResponse = | ||||
|         maybeResponse(client.post(url("/starr/$id")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|         }.body() | ||||
|         }) | ||||
|  | ||||
|     suspend fun unstarr(id: String): SelfossModel.SuccessResponse? = | ||||
|         client.post(url("/unstarr/$id")) { | ||||
|     suspend fun unstarr(id: String): SuccessResponse = | ||||
|         maybeResponse(client.post(url("/unstarr/$id")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|         }.body() | ||||
|         }) | ||||
|  | ||||
|     suspend fun markAllAsRead(ids: List<String>): SelfossModel.SuccessResponse? = | ||||
|         client.submitForm( | ||||
|     suspend fun markAllAsRead(ids: List<String>): SuccessResponse = | ||||
|         maybeResponse(client.submitForm( | ||||
|             url = url("/mark"), | ||||
|             formParameters = Parameters.build { | ||||
|                 append("username", appSettingsService.getUserName()) | ||||
|                 append("password", appSettingsService.getPassword()) | ||||
|                 ids.map { append("ids[]", it) } | ||||
|             } | ||||
|         ).body() | ||||
|         )) | ||||
|  | ||||
|     suspend fun createSourceForVersion( | ||||
|         title: String, | ||||
| @@ -166,12 +167,14 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         tags: String, | ||||
|         filter: String, | ||||
|         version: Int | ||||
|     ): SelfossModel.SuccessResponse? = | ||||
|         if (version > 1) { | ||||
|             createSource2(title, url, spout, tags, filter) | ||||
|         } else { | ||||
|             createSource(title, url, spout, tags, filter) | ||||
|         } | ||||
|     ): SuccessResponse = | ||||
|         maybeResponse( | ||||
|             if (version > 1) { | ||||
|                 createSource2(title, url, spout, tags, filter) | ||||
|             } else { | ||||
|                 createSource(title, url, spout, tags, filter) | ||||
|             } | ||||
|         ) | ||||
|  | ||||
|     suspend fun createSource( | ||||
|         title: String, | ||||
| @@ -179,7 +182,7 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         spout: String, | ||||
|         tags: String, | ||||
|         filter: String | ||||
|     ): SelfossModel.SuccessResponse? = | ||||
|     ): HttpResponse = | ||||
|         client.submitForm( | ||||
|             url = url("/source?username=${appSettingsService.getUserName()}&password=${appSettingsService.getPassword()}"), | ||||
|             formParameters = Parameters.build { | ||||
| @@ -189,7 +192,7 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|                 append("tags", tags) | ||||
|                 append("filter", filter) | ||||
|             } | ||||
|         ).body() | ||||
|         ) | ||||
|  | ||||
|     suspend fun createSource2( | ||||
|         title: String, | ||||
| @@ -197,7 +200,7 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         spout: String, | ||||
|         tags: String, | ||||
|         filter: String | ||||
|     ): SelfossModel.SuccessResponse? = | ||||
|     ): HttpResponse = | ||||
|         client.submitForm( | ||||
|             url = url("/source?username=${appSettingsService.getUserName()}&password=${appSettingsService.getPassword()}"), | ||||
|             formParameters = Parameters.build { | ||||
| @@ -207,11 +210,11 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|                 append("tags[]", tags) | ||||
|                 append("filter", filter) | ||||
|             } | ||||
|         ).body() | ||||
|         ) | ||||
|  | ||||
|     suspend fun deleteSource(id: Int): SelfossModel.SuccessResponse? = | ||||
|         client.delete(url("/source/$id")) { | ||||
|     suspend fun deleteSource(id: Int): SuccessResponse = | ||||
|         maybeResponse(client.delete(url("/source/$id")) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }.body() | ||||
|             }) | ||||
| } | ||||
| @@ -15,7 +15,6 @@ class AppSettingsService { | ||||
|  | ||||
|     // User settings related | ||||
|     private var _itemsCaching: Boolean? = null | ||||
|     private var _internalBrowser: Boolean? = null | ||||
|     private var _articleViewer: Boolean? = null | ||||
|     private var _shouldBeCardView: Boolean? = null | ||||
|     private var _displayUnreadCount: Boolean? = null | ||||
| @@ -25,7 +24,6 @@ class AppSettingsService { | ||||
|     private var _periodicRefresh: Boolean? = null | ||||
|     private var _refreshWhenChargingOnly: Boolean? = null | ||||
|     private var _infiniteLoading: Boolean? = null | ||||
|     private var _displayAccountHeader: Boolean? = null | ||||
|     private var _notifyNewItems: Boolean? = null | ||||
|     private var _itemsNumber: Int? = null | ||||
|     private var _apiTimeout: Long? = null | ||||
| @@ -34,9 +32,10 @@ class AppSettingsService { | ||||
|     private var _markOnScroll: Boolean? = null | ||||
|     private var _activeAlignment: Int? = null | ||||
|  | ||||
|     private var _fontSize: Int? = null // settings.getString("reader_font_size", "16").toInt() | ||||
|     private var _staticBar: Boolean? = null // settings.getBoolean("reader_static_bar", false) | ||||
|     private var _font: String = "" // settings.getString("reader_font", "") | ||||
|     private var _fontSize: Int? = null | ||||
|     private var _staticBar: Boolean? = null | ||||
|     private var _font: String = "" | ||||
|     private var _theme: Int? = null | ||||
|  | ||||
|  | ||||
|     init { | ||||
| @@ -56,8 +55,8 @@ class AppSettingsService { | ||||
|         return _apiVersion | ||||
|     } | ||||
|  | ||||
|     fun refreshApiVersion() { | ||||
|         _apiVersion = settings.getInt("apiVersionMajor", -1) | ||||
|     private fun refreshApiVersion() { | ||||
|         _apiVersion = settings.getInt(API_VERSION_MAJOR, -1) | ||||
|     } | ||||
|  | ||||
|     fun getBaseUrl(): String { | ||||
| @@ -69,7 +68,7 @@ class AppSettingsService { | ||||
|  | ||||
|     fun getUserName(): String { | ||||
|         if (_userName.isEmpty()) { | ||||
|             refrershUsername() | ||||
|             refreshUsername() | ||||
|         } | ||||
|         return _userName | ||||
|     } | ||||
| @@ -88,8 +87,8 @@ class AppSettingsService { | ||||
|         return _itemsNumber!! | ||||
|     } | ||||
|  | ||||
|     fun refreshItemsNumber() { | ||||
|         _itemsNumber = settings.getString("prefer_api_items_number", "200").toInt() | ||||
|     private fun refreshItemsNumber() { | ||||
|         _itemsNumber = settings.getString(API_ITEMS_NUMBER, "20").toInt() | ||||
|     } | ||||
|  | ||||
|     fun getApiTimeout(): Long { | ||||
| @@ -100,35 +99,24 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshApiTimeout() { | ||||
|         val settingsTimeout = settings.getLong("api_timeout", HttpTimeout.INFINITE_TIMEOUT_MS) | ||||
|         val settingsTimeout = settings.getLong(API_TIMEOUT, HttpTimeout.INFINITE_TIMEOUT_MS) | ||||
|         _apiTimeout = if (settingsTimeout > 0) settingsTimeout else HttpTimeout.INFINITE_TIMEOUT_MS | ||||
|     } | ||||
|  | ||||
|     private fun refreshBaseUrl() { | ||||
|         _baseUrl = settings.getString("url", "") | ||||
|         _baseUrl = settings.getString(BASE_URL, "") | ||||
|     } | ||||
|  | ||||
|     private fun refrershUsername() { | ||||
|         _userName = settings.getString("login", "") | ||||
|     private fun refreshUsername() { | ||||
|         _userName = settings.getString(LOGIN, "") | ||||
|     } | ||||
|  | ||||
|     private fun refreshPassword() { | ||||
|         _password = settings.getString("password", "") | ||||
|     } | ||||
|  | ||||
|     private fun refreshInternalBrowserEnabled() { | ||||
|         _internalBrowser = settings.getBoolean("prefer_internal_browser", true) | ||||
|     } | ||||
|  | ||||
|     fun isInternalBrowserEnabled(): Boolean { | ||||
|         if (_internalBrowser != null) { | ||||
|             refreshInternalBrowserEnabled() | ||||
|         } | ||||
|         return _internalBrowser == true | ||||
|         _password = settings.getString(PASSWORD, "") | ||||
|     } | ||||
|  | ||||
|     private fun refreshArticleViewerEnabled() { | ||||
|         _articleViewer = settings.getBoolean("prefer_article_viewer", true) | ||||
|         _articleViewer = settings.getBoolean(PREFER_ARTICLE_VIEWER, true) | ||||
|     } | ||||
|  | ||||
|     fun isArticleViewerEnabled(): Boolean { | ||||
| @@ -138,7 +126,7 @@ class AppSettingsService { | ||||
|         return _articleViewer == true | ||||
|     } | ||||
|     private fun refreshShouldBeCardViewEnabled() { | ||||
|         _shouldBeCardView = settings.getBoolean("card_view_active", false) | ||||
|         _shouldBeCardView = settings.getBoolean(CARD_VIEW_ACTIVE, false) | ||||
|     } | ||||
|  | ||||
|     fun isCardViewEnabled(): Boolean { | ||||
| @@ -148,7 +136,7 @@ class AppSettingsService { | ||||
|         return _shouldBeCardView == true | ||||
|     } | ||||
|     private fun refreshDisplayUnreadCountEnabled() { | ||||
|         _displayUnreadCount = settings.getBoolean("display_unread_count", true) | ||||
|         _displayUnreadCount = settings.getBoolean(DISPLAY_UNREAD_COUNT, true) | ||||
|     } | ||||
|  | ||||
|     fun isDisplayUnreadCountEnabled(): Boolean { | ||||
| @@ -158,7 +146,7 @@ class AppSettingsService { | ||||
|         return _displayUnreadCount == true | ||||
|     } | ||||
|     private fun refreshDisplayAllCountEnabled() { | ||||
|         _displayAllCount = settings.getBoolean("display_other_count", false) | ||||
|         _displayAllCount = settings.getBoolean(DISPLAY_OTHER_COUNT, false) | ||||
|     } | ||||
|  | ||||
|     fun isDisplayAllCountEnabled(): Boolean { | ||||
| @@ -168,7 +156,7 @@ class AppSettingsService { | ||||
|         return _displayAllCount == true | ||||
|     } | ||||
|     private fun refreshFullHeightCardsEnabled() { | ||||
|         _fullHeightCards = settings.getBoolean("full_height_cards", false) | ||||
|         _fullHeightCards = settings.getBoolean(FULL_HEIGHT_CARDS, false) | ||||
|     } | ||||
|  | ||||
|     fun isFullHeightCardsEnabled(): Boolean { | ||||
| @@ -178,7 +166,7 @@ class AppSettingsService { | ||||
|         return _fullHeightCards == true | ||||
|     } | ||||
|     private fun refreshUpdateSourcesEnabled() { | ||||
|         _updateSources = settings.getBoolean("update_sources", true) | ||||
|         _updateSources = settings.getBoolean(UPDATE_SOURCES, true) | ||||
|     } | ||||
|  | ||||
|     fun isUpdateSourcesEnabled(): Boolean { | ||||
| @@ -188,7 +176,7 @@ class AppSettingsService { | ||||
|         return _updateSources == true | ||||
|     } | ||||
|     private fun refreshPeriodicRefreshEnabled() { | ||||
|         _periodicRefresh = settings.getBoolean("periodic_refresh", false) | ||||
|         _periodicRefresh = settings.getBoolean(PERIODIC_REFRESH, false) | ||||
|     } | ||||
|  | ||||
|     fun isPeriodicRefreshEnabled(): Boolean { | ||||
| @@ -199,7 +187,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshRefreshWhenChargingOnlyEnabled() { | ||||
|         _refreshWhenChargingOnly = settings.getBoolean("refresh_when_charging", false) | ||||
|         _refreshWhenChargingOnly = settings.getBoolean(REFRESH_WHEN_CHARGING, false) | ||||
|     } | ||||
|  | ||||
|     fun isRefreshWhenChargingOnlyEnabled(): Boolean { | ||||
| @@ -210,22 +198,22 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshRefreshMinutes() { | ||||
|         _refreshMinutes = settings.getString("periodic_refresh_minutes", "360").toLong() | ||||
|         _refreshMinutes = settings.getString(PERIODIC_REFRESH_MINUTES, "360").toLong() | ||||
|         if (_refreshMinutes <= 15) { | ||||
|             _refreshMinutes = 15 | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun getRefreshMinutes(): Long { | ||||
|         if (_refreshMinutes != null) { | ||||
|         if (_refreshMinutes != 360L) { | ||||
|             refreshRefreshMinutes() | ||||
|         } | ||||
|         return _refreshMinutes | ||||
|     } | ||||
|  | ||||
|     private fun refreshHiddenTags() { | ||||
|         if (settings.getString("hidden_tags", "").isNotEmpty()) { | ||||
|             _hiddenTags = settings.getString("hidden_tags", "").replace("\\s".toRegex(), "").split(",") | ||||
|         if (settings.getString(HIDDEN_TAGS, "").isNotEmpty()) { | ||||
|             _hiddenTags = settings.getString(HIDDEN_TAGS, "").replace("\\s".toRegex(), "").split(",") | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -237,7 +225,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshInfiniteLoadingEnabled() { | ||||
|         _infiniteLoading = settings.getBoolean("infinite_loading", false) | ||||
|         _infiniteLoading = settings.getBoolean(INFINITE_LOADING, false) | ||||
|     } | ||||
|  | ||||
|     fun isInfiniteLoadingEnabled(): Boolean { | ||||
| @@ -247,19 +235,8 @@ class AppSettingsService { | ||||
|         return _infiniteLoading == true | ||||
|     } | ||||
|  | ||||
|     private fun refreshDisplayAccountHeaderEnabled() { | ||||
|         _displayAccountHeader = settings.getBoolean("account_header_displaying", false) | ||||
|     } | ||||
|  | ||||
|     fun isDisplayAccountHeaderEnabled(): Boolean { | ||||
|         if (_displayAccountHeader != null) { | ||||
|             refreshDisplayAccountHeaderEnabled() | ||||
|         } | ||||
|         return _displayAccountHeader == true | ||||
|     } | ||||
|  | ||||
|     private fun refreshItemCachingEnabled() { | ||||
|         _itemsCaching = settings.getBoolean("items_caching", false) | ||||
|         _itemsCaching = settings.getBoolean(ITEMS_CACHING, false) | ||||
|     } | ||||
|  | ||||
|     fun isItemCachingEnabled(): Boolean { | ||||
| @@ -270,7 +247,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshNotifyNewItemsEnabled() { | ||||
|         _notifyNewItems = settings.getBoolean("notify_new_items", false) | ||||
|         _notifyNewItems = settings.getBoolean(NOTIFY_NEW_ITEMS, false) | ||||
|     } | ||||
|  | ||||
|     fun isNotifyNewItemsEnabled(): Boolean { | ||||
| @@ -282,7 +259,7 @@ class AppSettingsService { | ||||
|  | ||||
|  | ||||
|     private fun refreshMarkOnScrollEnabled() { | ||||
|         _markOnScroll = settings.getBoolean("mark_on_scroll", false) | ||||
|         _markOnScroll = settings.getBoolean(MARK_ON_SCROLL, false) | ||||
|     } | ||||
|  | ||||
|     fun isMarkOnScrollEnabled(): Boolean { | ||||
| @@ -294,7 +271,7 @@ class AppSettingsService { | ||||
|  | ||||
|  | ||||
|     private fun refreshActiveAllignment() { | ||||
|         _activeAlignment = settings.getInt("text_align", JUSTIFY) | ||||
|         _activeAlignment = settings.getInt(TEXT_ALIGN, JUSTIFY) | ||||
|     } | ||||
|  | ||||
|     fun getActiveAllignment(): Int { | ||||
| @@ -305,12 +282,12 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     fun changeAllignment(allignment: Int) { | ||||
|         settings.putInt("text_align", allignment) | ||||
|         settings.putInt(TEXT_ALIGN, allignment) | ||||
|         _activeAlignment = allignment | ||||
|     } | ||||
|  | ||||
|     private fun refreshFontSize() { | ||||
|         _fontSize = settings.getString("reader_font_size", "16").toInt() | ||||
|         _fontSize = settings.getString(READER_FONT_SIZE, "16").toInt() | ||||
|     } | ||||
|  | ||||
|     fun getFontSize(): Int { | ||||
| @@ -321,7 +298,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshStaticBarEnabled() { | ||||
|         _staticBar = settings.getBoolean("reader_static_bar", false) | ||||
|         _staticBar = settings.getBoolean(READER_STATIC_BAR, false) | ||||
|     } | ||||
|  | ||||
|     fun isStaticBarEnabled(): Boolean { | ||||
| @@ -332,19 +309,30 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshFont() { | ||||
|         _font = settings.getString("reader_font", "") | ||||
|         _font = settings.getString(READER_FONT, "") | ||||
|     } | ||||
|  | ||||
|     fun getFont(): String { | ||||
|         if (_font != null) { | ||||
|         if (_font.isEmpty()) { | ||||
|             refreshFont() | ||||
|         } | ||||
|         return _font | ||||
|     } | ||||
|  | ||||
|     private fun refreshCurrentTheme() { | ||||
|         _theme = settings.getString(CURRENT_THEME, "-1").toInt() | ||||
|     } | ||||
|  | ||||
|     fun getCurrentTheme(): Int { | ||||
|         if (_theme == null) { | ||||
|             refreshCurrentTheme() | ||||
|         } | ||||
|         return _theme ?: -1 | ||||
|     } | ||||
|  | ||||
|     fun refreshApiSettings() { | ||||
|         refreshPassword() | ||||
|         refrershUsername() | ||||
|         refreshUsername() | ||||
|         refreshBaseUrl() | ||||
|         refreshApiVersion() | ||||
|     } | ||||
| @@ -352,7 +340,6 @@ class AppSettingsService { | ||||
|     fun refreshUserSettings() { | ||||
|         refreshItemsNumber() | ||||
|         refreshApiTimeout() | ||||
|         refreshInternalBrowserEnabled() | ||||
|         refreshArticleViewerEnabled() | ||||
|         refreshShouldBeCardViewEnabled() | ||||
|         refreshDisplayUnreadCountEnabled() | ||||
| @@ -364,7 +351,6 @@ class AppSettingsService { | ||||
|         refreshRefreshMinutes() | ||||
|         refreshHiddenTags() | ||||
|         refreshInfiniteLoadingEnabled() | ||||
|         refreshDisplayAccountHeaderEnabled() | ||||
|         refreshItemCachingEnabled() | ||||
|         refreshNotifyNewItemsEnabled() | ||||
|         refreshMarkOnScrollEnabled() | ||||
| @@ -372,6 +358,7 @@ class AppSettingsService { | ||||
|         refreshFontSize() | ||||
|         refreshFont() | ||||
|         refreshStaticBarEnabled() | ||||
|         refreshCurrentTheme() | ||||
|     } | ||||
|  | ||||
|     fun refreshLoginInformation( | ||||
| @@ -379,21 +366,21 @@ class AppSettingsService { | ||||
|         login: String, | ||||
|         password: String | ||||
|     ) { | ||||
|         settings.putString("url", url) | ||||
|         settings.putString("login", login) | ||||
|         settings.putString("password", password) | ||||
|         settings.putString(BASE_URL, url) | ||||
|         settings.putString(LOGIN, login) | ||||
|         settings.putString(PASSWORD, password) | ||||
|         refreshApiSettings() | ||||
|     } | ||||
|  | ||||
|     fun resetLoginInformation() { | ||||
|         settings.remove("url") | ||||
|         settings.remove("login") | ||||
|         settings.remove("password") | ||||
|         settings.remove(BASE_URL) | ||||
|         settings.remove(LOGIN) | ||||
|         settings.remove(PASSWORD) | ||||
|         refreshApiSettings() | ||||
|     } | ||||
|  | ||||
|     fun updateApiVersion(apiMajorVersion: Int) { | ||||
|         settings.putInt("apiVersionMajor", apiMajorVersion) | ||||
|         settings.putInt(API_VERSION_MAJOR, apiMajorVersion) | ||||
|         refreshApiVersion() | ||||
|     } | ||||
|  | ||||
| @@ -404,7 +391,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     fun disableArticleViewer() { | ||||
|         settings.putBoolean("prefer_article_viewer", false) | ||||
|         settings.putBoolean(PREFER_ARTICLE_VIEWER, false) | ||||
|         refreshArticleViewerEnabled() | ||||
|     } | ||||
|  | ||||
| @@ -422,5 +409,55 @@ class AppSettingsService { | ||||
|         const val JUSTIFY = 1 | ||||
|  | ||||
|         const val ALIGN_LEFT = 2 | ||||
|  | ||||
|         const val API_VERSION_MAJOR = "apiVersionMajor" | ||||
|  | ||||
|         const val API_ITEMS_NUMBER = "prefer_api_items_number" | ||||
|  | ||||
|         const val API_TIMEOUT = "api_timeout" | ||||
|  | ||||
|         const val BASE_URL = "url" | ||||
|  | ||||
|         const val LOGIN = "login" | ||||
|  | ||||
|         const val PASSWORD = "password" | ||||
|  | ||||
|         const val PREFER_ARTICLE_VIEWER = "prefer_article_viewer" | ||||
|  | ||||
|         const val CARD_VIEW_ACTIVE = "card_view_active" | ||||
|  | ||||
|         const val DISPLAY_UNREAD_COUNT = "display_unread_count" | ||||
|  | ||||
|         const val DISPLAY_OTHER_COUNT = "display_other_count" | ||||
|  | ||||
|         const val FULL_HEIGHT_CARDS = "full_height_cards" | ||||
|  | ||||
|         const val UPDATE_SOURCES = "update_sources" | ||||
|  | ||||
|         const val PERIODIC_REFRESH = "periodic_refresh" | ||||
|  | ||||
|         const val REFRESH_WHEN_CHARGING = "refresh_when_charging" | ||||
|  | ||||
|         const val READER_FONT = "reader_font" | ||||
|  | ||||
|         const val READER_STATIC_BAR = "reader_static_bar" | ||||
|  | ||||
|         const val READER_FONT_SIZE = "reader_font_size" | ||||
|  | ||||
|         const val TEXT_ALIGN = "text_align" | ||||
|  | ||||
|         const val MARK_ON_SCROLL = "mark_on_scroll" | ||||
|  | ||||
|         const val NOTIFY_NEW_ITEMS = "notify_new_items" | ||||
|  | ||||
|         const val PERIODIC_REFRESH_MINUTES = "periodic_refresh_minutes" | ||||
|  | ||||
|         const val HIDDEN_TAGS = "hidden_tags" | ||||
|  | ||||
|         const val INFINITE_LOADING = "infinite_loading" | ||||
|  | ||||
|         const val ITEMS_CACHING = "items_caching" | ||||
|  | ||||
|         const val CURRENT_THEME = "currentMode" | ||||
|     } | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user