Compare commits
	
		
			107 Commits
		
	
	
		
			023a30c008
			...
			v122123611
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 27eafe4ff4 | ||
|  | 8c83a9408b | ||
|  | fe2410f719 | ||
|  | a5e86bfb77 | ||
|  | 23be633798 | ||
|  | 813e0707d8 | ||
|  | 9ed9bf07fc | ||
|  | 47265c10d0 | ||
|  | 5cc633246a | ||
|  | 1f40385786 | ||
|  | eb2876324a | ||
|  | 633b817d76 | ||
|  | 2cfaa9b285 | ||
|  | f42ae97326 | ||
|  | 3b0028164b | ||
|  | 7420adeb5c | ||
|  | 316027ca3b | ||
|  | 9d58fba5c9 | ||
|  | 284c19ef89 | ||
|  | 7cfd17231a | ||
|  | 527830a5ae | ||
|  | c4ed30f594 | ||
|  | 156c1681cf | ||
|  | 3593fbca78 | ||
|  | 430fc8e8cb | ||
|  | 4fce19bad4 | ||
|  | 49f5848e7b | ||
|  | 90452100a4 | ||
|  | bf1196dd0f | ||
|  | 4316dc6516 | ||
|  | 9833a66a64 | ||
|  | 797bf06a9c | ||
|  | d98b00533d | ||
|  | bf8f7d8667 | ||
|  | 89c570f34f | ||
|  | d6a562863a | ||
|  | a02f06fe2e | ||
|  | 7b088d7bb4 | ||
|  | 477883ed39 | ||
|  | 748ed41096 | ||
|  | 86c50d4881 | ||
|  | c4c92e6dd9 | ||
|  | 7f0ba193ec | ||
|  | 87ed5b0fa8 | ||
|  | 6947743ac0 | ||
|  | 07e3710d44 | ||
|  | e68da7764f | ||
|  | c3ff894027 | ||
|  | f09f731d30 | ||
|  | 956c4341c7 | ||
|  | 7b68264dd7 | ||
|  | cfcf030bf8 | ||
|  | 0e7d7a5835 | ||
|  | 0856ebb889 | ||
|  | 25bf68cf0c | ||
|  | afc6f392c6 | ||
|  | a0b5e2052b | ||
|  | 87d1ef2bce | ||
|  | 537a6d3a0b | ||
| dbe97f564e | |||
|  | 3a3bf03114 | ||
| c09a32e9ad | |||
| b02a588dff | |||
|  | 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 | 
							
								
								
									
										37
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										37
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -3,24 +3,35 @@ type: docker | |||||||
| name: test | name: test | ||||||
|  |  | ||||||
| steps: | steps: | ||||||
|   - name: AnylyseBuildTest |   - name: BuildAndTest | ||||||
|     image: mingc/android-build-box:latest |     image: mingc/android-build-box:latest | ||||||
|     commands: |     commands: | ||||||
|       - echo "---------------------------------------------------------" |       - echo "---------------------------------------------------------" | ||||||
|       - echo "Analysing..." |       - echo "Configure gradle..." | ||||||
|       - ./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\"" |       - mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=true\npushCache=false\nsystemProp.org.gradle.internal.http.connectionTimeout=180000\nsystemProp.org.gradle.internal.http.socketTimeout=180000" >> ~/.gradle/gradle.properties | ||||||
|       - echo "---------------------------------------------------------" |       - echo "---------------------------------------------------------" | ||||||
|       - echo "Building..." |       - echo "Building..." | ||||||
|       - ./gradlew :androidApp:build -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" -P pushCache=false |       - ./gradlew build -x test | ||||||
|       - echo "---------------------------------------------------------" |       - echo "---------------------------------------------------------" | ||||||
|       - echo "Testing..." |       - echo "Testing..." | ||||||
|       - echo "---------------------------------------------------------" |       - echo "---------------------------------------------------------" | ||||||
|       - ./gradlew test -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" -P pushCache=false |       - ./gradlew koverMergedXmlReport | ||||||
|     environment: |     environment: | ||||||
|  |       TZ: Europe/Paris | ||||||
|       SONAR_HOST_URL: |       SONAR_HOST_URL: | ||||||
|         from_secret: sonarScannerHostUrl |         from_secret: sonarScannerHostUrl | ||||||
|       SONAR_LOGIN: |       SONAR_LOGIN: | ||||||
|         from_secret: sonarScannerLogin |         from_secret: sonarScannerLogin | ||||||
|  |   - name: Analyse | ||||||
|  |     image: kytay/sonar-node-plugin | ||||||
|  |     settings: | ||||||
|  |       sonar_host: | ||||||
|  |         from_secret: sonarScannerHostUrl | ||||||
|  |       sonar_token: | ||||||
|  |         from_secret: sonarScannerLogin | ||||||
|  |       use_node_version: 16.18.1 | ||||||
|  |       sonar_debug: true | ||||||
|  |       sonar_project_settings: ./sonar-project.properties | ||||||
| trigger: | trigger: | ||||||
|   event: |   event: | ||||||
|     - push |     - push | ||||||
| @@ -40,6 +51,7 @@ steps: | |||||||
|       - git remote add pushing https://$GITEA_USR:$GITEA_PASS@gitea.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform.git |       - git remote add pushing https://$GITEA_USR:$GITEA_PASS@gitea.amine-louveau.fr/Louvorg/ReaderForSelfoss-multiplatform.git | ||||||
|       - git push pushing --tags |       - git push pushing --tags | ||||||
|     environment: |     environment: | ||||||
|  |       TZ: Europe/Paris | ||||||
|       GITEA_USR: |       GITEA_USR: | ||||||
|         from_secret: giteaUsr |         from_secret: giteaUsr | ||||||
|       GITEA_PASS: |       GITEA_PASS: | ||||||
| @@ -65,10 +77,7 @@ steps: | |||||||
|         from_secret: privateKey |         from_secret: privateKey | ||||||
|       command_timeout: 2m |       command_timeout: 2m | ||||||
|       script: |       script: | ||||||
|         - cd /home/ubuntu |         - 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/ | ||||||
|         - sudo rm -rf /var/www/amine/version.txt |  | ||||||
|         - sudo chown www-data:www-data ./version.txt |  | ||||||
|         - sudo mv version.txt /var/www/amine/ |  | ||||||
|  |  | ||||||
| trigger: | trigger: | ||||||
|   event: |   event: | ||||||
| @@ -85,8 +94,15 @@ steps: | |||||||
|   - name: build |   - name: build | ||||||
|     image: mingc/android-build-box:latest |     image: mingc/android-build-box:latest | ||||||
|     commands: |     commands: | ||||||
|  |       - echo "---------------------------------------------------------" | ||||||
|  |       - echo "Fetch tags..." | ||||||
|  |       - git fetch --tags | ||||||
|  |       - echo "---------------------------------------------------------" | ||||||
|  |       - echo "Configure gradle..." | ||||||
|  |       - mkdir -p ~/.gradle && echo "org.gradle.daemon=false\nignoreGitVersion=false\npushCache=false\nsystemProp.org.gradle.internal.http.connectionTimeout=180000\nsystemProp.org.gradle.internal.http.socketTimeout=180000" >> ~/.gradle/gradle.properties | ||||||
|  |       - echo "---------------------------------------------------------" | ||||||
|       - echo "Generate APK" |       - echo "Generate APK" | ||||||
|       - ./gradlew :androidApp:assembleGithubConfigRelease -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" -P pushCache=false |       - ./gradlew :androidApp:assembleGithubConfigRelease  -P pushCache=false | ||||||
|       - echo "---------------------------------------------------------" |       - echo "---------------------------------------------------------" | ||||||
|       - echo "Get Key" |       - echo "Get Key" | ||||||
|       - wget https://amine-louveau.fr/key |       - wget https://amine-louveau.fr/key | ||||||
| @@ -100,6 +116,7 @@ steps: | |||||||
|       - echo "Verify" |       - echo "Verify" | ||||||
|       - $ANDROID_HOME/build-tools/31.0.0/apksigner verify signed.apk |       - $ANDROID_HOME/build-tools/31.0.0/apksigner verify signed.apk | ||||||
|     environment: |     environment: | ||||||
|  |       TZ: Europe/Paris | ||||||
|       YOUR_KEYSTORE_PASSWORD: |       YOUR_KEYSTORE_PASSWORD: | ||||||
|         from_secret: keyPass |         from_secret: keyPass | ||||||
|       YOUR_KEY_ALIAS: |       YOUR_KEY_ALIAS: | ||||||
|   | |||||||
							
								
								
									
										25
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										25
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							| @@ -46,28 +46,3 @@ Always check if the web version of your instance is working. | |||||||
| I won't provide any selfoss instance url. If you want to help, but to not have one, you'll have to install one, and use it. | I won't provide any selfoss instance url. If you want to help, but to not have one, you'll have to install one, and use it. | ||||||
|  |  | ||||||
| All the details to need are [here](https://selfoss.aditu.de/). | All the details to need are [here](https://selfoss.aditu.de/). | ||||||
|  |  | ||||||
| # Build the project |  | ||||||
|  |  | ||||||
| You can directly import this project into IntellIJ/Android Studio. |  | ||||||
|  |  | ||||||
| You'll have to: |  | ||||||
|  |  | ||||||
| - Define some parameters either in `~/.gradle/gradle.properties` or as gradle parameters (see the examples) |  | ||||||
|  |  | ||||||
|     - appLoginUrl, appLoginUsername and appLoginPassword: url, username and password of a selfoss instance. **These are only used for tests. They can be empty if you don't test API calls.** |  | ||||||
|  |  | ||||||
| ### Examples: |  | ||||||
| #### Inside ~/.gradle/gradle.properties |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| appLoginUrl="URL" # It can be empty. |  | ||||||
| appLoginUsername="LOGIN" # It can be empty. |  | ||||||
| appLoginPassword="PASS" # It can be empty. |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| #### As gradle parameters |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| ./gradlew .... -P appLoginUrl="URL" -P appLoginUsername="LOGIN" -P appLoginPassword="PASS" |  | ||||||
| ``` |  | ||||||
|   | |||||||
| @@ -1,11 +1,14 @@ | |||||||
| import java.io.ByteArrayOutputStream | import java.io.ByteArrayOutputStream | ||||||
|  |  | ||||||
| val ignoreGitVersion: String by project | val ignoreGitVersion: String by project | ||||||
|  | val acraVersion = "5.9.7" | ||||||
|  |  | ||||||
| plugins { | plugins { | ||||||
|     id("com.android.application") |     id("com.android.application") | ||||||
|     kotlin("android") |     kotlin("android") | ||||||
|     kotlin("kapt") |     kotlin("kapt") | ||||||
|  |     id("com.mikepenz.aboutlibraries.plugin") | ||||||
|  |     id("org.jetbrains.kotlinx.kover") version "0.6.1" | ||||||
| } | } | ||||||
|  |  | ||||||
| fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | ||||||
| @@ -54,11 +57,15 @@ fun versionNameFromGit(): String { | |||||||
| android { | android { | ||||||
|     compileOptions { |     compileOptions { | ||||||
|         // Flag to enable support for the new language APIs |         // Flag to enable support for the new language APIs | ||||||
|         isCoreLibraryDesugaringEnabled = true |         sourceCompatibility = JavaVersion.VERSION_11 | ||||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 |         targetCompatibility = JavaVersion.VERSION_11 | ||||||
|         targetCompatibility = JavaVersion.VERSION_1_8 |  | ||||||
|     } |     } | ||||||
|     compileSdk = 31 |  | ||||||
|  |     // For Kotlin projects | ||||||
|  |     kotlinOptions { | ||||||
|  |         jvmTarget = "11" | ||||||
|  |     } | ||||||
|  |     compileSdk = 33 | ||||||
|     buildToolsVersion = "31.0.0" |     buildToolsVersion = "31.0.0" | ||||||
|     buildFeatures { |     buildFeatures { | ||||||
|         viewBinding = true |         viewBinding = true | ||||||
| @@ -66,7 +73,7 @@ android { | |||||||
|     defaultConfig { |     defaultConfig { | ||||||
|         applicationId = "bou.amine.apps.readerforselfossv2.android" |         applicationId = "bou.amine.apps.readerforselfossv2.android" | ||||||
|         minSdk = 21 |         minSdk = 21 | ||||||
|         targetSdk = 31 |         targetSdk = 33 | ||||||
|         versionCode = versionCodeFromGit() |         versionCode = versionCodeFromGit() | ||||||
|         versionName = versionNameFromGit() |         versionName = versionNameFromGit() | ||||||
|  |  | ||||||
| @@ -79,6 +86,11 @@ android { | |||||||
|         // tests |         // tests | ||||||
|         testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" |         testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" | ||||||
|     } |     } | ||||||
|  |     packagingOptions { | ||||||
|  |         resources { | ||||||
|  |             excludes += "/META-INF/{AL2.0,LGPL2.1}" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|     buildTypes { |     buildTypes { | ||||||
|         getByName("release") { |         getByName("release") { | ||||||
|             isMinifyEnabled = true |             isMinifyEnabled = true | ||||||
| @@ -86,9 +98,6 @@ android { | |||||||
|             proguardFiles(getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro") |             proguardFiles(getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro") | ||||||
|         } |         } | ||||||
|         getByName("debug") { |         getByName("debug") { | ||||||
|             buildConfigField("String", "LOGIN_URL", properties["appLoginUrl"] as String) |  | ||||||
|             buildConfigField("String", "LOGIN_PASSWORD", properties["appLoginPassword"] as String) |  | ||||||
|             buildConfigField("String", "LOGIN_USERNAME", properties["appLoginUsername"] as String) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     flavorDimensions.add("build") |     flavorDimensions.add("build") | ||||||
| @@ -98,9 +107,6 @@ android { | |||||||
|             dimension = "build" |             dimension = "build" | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     kotlinOptions { |  | ||||||
|         jvmTarget = "1.8" |  | ||||||
|     } |  | ||||||
|     namespace = "bou.amine.apps.readerforselfossv2.android" |     namespace = "bou.amine.apps.readerforselfossv2.android" | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -114,13 +120,6 @@ dependencies { | |||||||
|  |  | ||||||
|     implementation("androidx.preference:preference-ktx:1.1.1") |     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"))) |     implementation(fileTree(mapOf("include" to listOf("*.jar"), "dir" to "libs"))) | ||||||
|  |  | ||||||
|     // Android Support |     // Android Support | ||||||
| @@ -135,35 +134,20 @@ dependencies { | |||||||
|     implementation("androidx.constraintlayout:constraintlayout:2.1.3") |     implementation("androidx.constraintlayout:constraintlayout:2.1.3") | ||||||
|     implementation("org.jsoup:jsoup:1.14.3") |     implementation("org.jsoup:jsoup:1.14.3") | ||||||
|  |  | ||||||
|     coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:1.1.5") |  | ||||||
|  |  | ||||||
|     //multidex |     //multidex | ||||||
|     implementation("androidx.multidex:multidex:2.0.1") |     implementation("androidx.multidex:multidex:2.0.1") | ||||||
|  |  | ||||||
|     // About |     // About | ||||||
|     implementation("com.mikepenz:aboutlibraries-core:8.9.4") |     implementation("com.mikepenz:aboutlibraries-core:10.5.1") | ||||||
|     implementation("com.mikepenz:aboutlibraries:8.9.4") |     implementation("com.mikepenz:aboutlibraries:10.5.1") | ||||||
|     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") |  | ||||||
|  |  | ||||||
|     // Material-ish things |     // Material-ish things | ||||||
|     implementation("com.ashokvarma.android:bottom-navigation-bar:2.2.0") |     implementation("com.ashokvarma.android:bottom-navigation-bar:2.2.0") | ||||||
|     implementation("com.amulyakhare:com.amulyakhare.textdrawable:1.0.1") |     implementation("com.amulyakhare:com.amulyakhare.textdrawable:1.0.1") | ||||||
|  |  | ||||||
|     // glide |     // glide | ||||||
|     kapt("com.github.bumptech.glide:compiler:4.11.0") |     kapt("com.github.bumptech.glide:compiler:4.14.2") | ||||||
|     implementation("com.github.bumptech.glide:okhttp3-integration:4.1.1") |     implementation("com.github.bumptech.glide:okhttp3-integration:4.14.2") | ||||||
|  |  | ||||||
|     // Drawer |  | ||||||
|     implementation("com.mikepenz:materialdrawer:8.4.5") |  | ||||||
|  |  | ||||||
|     // Themes |     // Themes | ||||||
|     implementation("com.github.rubensousa:floatingtoolbar:1.5.1") |     implementation("com.github.rubensousa:floatingtoolbar:1.5.1") | ||||||
| @@ -188,14 +172,44 @@ dependencies { | |||||||
|  |  | ||||||
|     implementation("androidx.core:core-ktx:1.8.0") |     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") |     implementation("androidx.lifecycle:lifecycle-extensions:2.2.0") | ||||||
|  |  | ||||||
|     // Network information |     // Network information | ||||||
|      implementation("com.github.ln-12:multiplatform-connectivity-status:1.3.0") |      implementation("com.github.ln-12:multiplatform-connectivity-status:1.3.0") | ||||||
|  |  | ||||||
|     // SQLDELIGHT |     // 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") | ||||||
|  |  | ||||||
|  |     implementation("ch.acra:acra-http:$acraVersion") | ||||||
|  |     implementation("ch.acra:acra-toast:$acraVersion") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | 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 | ||||||
| } | } | ||||||
							
								
								
									
										9
									
								
								androidApp/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								androidApp/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							| @@ -30,15 +30,8 @@ | |||||||
|     <fields>; |     <fields>; | ||||||
| } | } | ||||||
|  |  | ||||||
| -dontwarn okio.** |  | ||||||
| -dontwarn retrofit2.Platform$Java8 |  | ||||||
| -keep class retrofit.** { *; } |  | ||||||
| -keepclasseswithmembers class * { |  | ||||||
|     @retrofit.http.* <methods>; |  | ||||||
| } |  | ||||||
| -keepattributes *Annotation*,Signature | -keepattributes *Annotation*,Signature | ||||||
| -keepattributes Exceptions | -keepattributes Exceptions | ||||||
| -dontwarn okio.** |  | ||||||
| -dontwarn javax.annotation.Nullable | -dontwarn javax.annotation.Nullable | ||||||
| -dontwarn javax.annotation.ParametersAreNonnullByDefault | -dontwarn javax.annotation.ParametersAreNonnullByDefault | ||||||
|  |  | ||||||
| @@ -90,3 +83,5 @@ | |||||||
| # @Serializable and @Polymorphic are used at runtime for polymorphic serialization. | # @Serializable and @Polymorphic are used at runtime for polymorphic serialization. | ||||||
| -keepattributes RuntimeVisibleAnnotations,AnnotationDefault | -keepattributes RuntimeVisibleAnnotations,AnnotationDefault | ||||||
|  |  | ||||||
|  | -dontwarn io.mockk.** | ||||||
|  | -keep class io.mockk.** { *; } | ||||||
|   | |||||||
| @@ -15,7 +15,8 @@ | |||||||
|         android:supportsRtl="true" |         android:supportsRtl="true" | ||||||
|         android:networkSecurityConfig="@xml/network_security_config" |         android:networkSecurityConfig="@xml/network_security_config" | ||||||
|         android:theme="@style/NoBar" |         android:theme="@style/NoBar" | ||||||
|         android:dataExtractionRules="@xml/data_extraction_rules"> |         android:dataExtractionRules="@xml/data_extraction_rules" | ||||||
|  |         android:configChanges="uiMode"> | ||||||
|         <activity |         <activity | ||||||
|             android:name=".MainActivity" |             android:name=".MainActivity" | ||||||
|             android:theme="@style/SplashTheme" |             android:theme="@style/SplashTheme" | ||||||
| @@ -78,8 +79,5 @@ | |||||||
|             android:value="true" /> |             android:value="true" /> | ||||||
|  |  | ||||||
|         <meta-data android:name="android.max_aspect" android:value="2.1" /> |         <meta-data android:name="android.max_aspect" android:value="2.1" /> | ||||||
|         <meta-data |  | ||||||
|             android:name="preloaded_fonts" |  | ||||||
|             android:resource="@array/preloaded_fonts" /> |  | ||||||
|     </application> |     </application> | ||||||
| </manifest> | </manifest> | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
|  | import org.acra.ACRA | ||||||
|  | import org.acra.ktx.sendSilentlyWithAcra | ||||||
|  |  | ||||||
|  | fun Throwable.sendSilentlyWithAcraWithName(name: String) { | ||||||
|  |     ACRA.errorReporter.putCustomData("error_source", name) | ||||||
|  |     this.sendSilentlyWithAcra() | ||||||
|  | } | ||||||
| @@ -94,7 +94,7 @@ class AddSourceActivity : AppCompatActivity(), DIAware { | |||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
|             try { |             try { | ||||||
|                 val items = repository.getSpouts() |                 val items = repository.getSpouts() | ||||||
|                 if (items != null) { |                 if (items.isNotEmpty()) { | ||||||
|                     val itemsStrings = items.map { it.value.name } |                     val itemsStrings = items.map { it.value.name } | ||||||
|                     for ((key, value) in items) { |                     for ((key, value) in items) { | ||||||
|                         spoutsKV[value.name] = key |                         spoutsKV[value.name] = key | ||||||
|   | |||||||
| @@ -1,23 +1,17 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.android | package bou.amine.apps.readerforselfossv2.android | ||||||
|  |  | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.graphics.Color |  | ||||||
| import android.graphics.drawable.Drawable |  | ||||||
| import android.graphics.drawable.GradientDrawable |  | ||||||
| import android.net.Uri |  | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.view.Menu | import android.view.Menu | ||||||
| import android.view.MenuItem | import android.view.MenuItem | ||||||
| import android.view.View | import android.view.View | ||||||
| import android.widget.ImageView |  | ||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import androidx.activity.result.contract.ActivityResultContracts | import androidx.activity.result.contract.ActivityResultContracts | ||||||
| import androidx.appcompat.app.ActionBarDrawerToggle |  | ||||||
| import androidx.appcompat.app.AlertDialog | import androidx.appcompat.app.AlertDialog | ||||||
| import androidx.appcompat.app.AppCompatActivity | import androidx.appcompat.app.AppCompatActivity | ||||||
| import androidx.appcompat.widget.SearchView | import androidx.appcompat.widget.SearchView | ||||||
| import androidx.core.view.doOnNextLayout | import androidx.core.view.doOnNextLayout | ||||||
| import androidx.drawerlayout.widget.DrawerLayout | import androidx.lifecycle.lifecycleScope | ||||||
| import androidx.recyclerview.widget.* | import androidx.recyclerview.widget.* | ||||||
| import androidx.work.Constraints | import androidx.work.Constraints | ||||||
| import androidx.work.ExistingPeriodicWorkPolicy | import androidx.work.ExistingPeriodicWorkPolicy | ||||||
| @@ -28,46 +22,29 @@ import bou.amine.apps.readerforselfossv2.android.adapters.ItemListAdapter | |||||||
| import bou.amine.apps.readerforselfossv2.android.adapters.ItemsAdapter | import bou.amine.apps.readerforselfossv2.android.adapters.ItemsAdapter | ||||||
| import bou.amine.apps.readerforselfossv2.android.background.LoadingWorker | import bou.amine.apps.readerforselfossv2.android.background.LoadingWorker | ||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivityHomeBinding | import bou.amine.apps.readerforselfossv2.android.databinding.ActivityHomeBinding | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.fragments.FilterSheetFragment | ||||||
| import bou.amine.apps.readerforselfossv2.android.settings.SettingsActivity | import bou.amine.apps.readerforselfossv2.android.settings.SettingsActivity | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.bottombar.maybeShow | 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.bottombar.removeBadge | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
| import bou.amine.apps.readerforselfossv2.utils.* | import bou.amine.apps.readerforselfossv2.utils.ItemType | ||||||
| import com.ashokvarma.bottomnavigation.BottomNavigationBar | import com.ashokvarma.bottomnavigation.BottomNavigationBar | ||||||
| import com.ashokvarma.bottomnavigation.BottomNavigationItem | import com.ashokvarma.bottomnavigation.BottomNavigationItem | ||||||
| import com.ashokvarma.bottomnavigation.TextBadgeItem | import com.ashokvarma.bottomnavigation.TextBadgeItem | ||||||
| import com.bumptech.glide.Glide |  | ||||||
| import com.bumptech.glide.request.RequestOptions |  | ||||||
| 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.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 kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| import kotlinx.coroutines.Dispatchers | import kotlinx.coroutines.Dispatchers | ||||||
| import kotlinx.coroutines.launch | import kotlinx.coroutines.launch | ||||||
| import org.kodein.di.DIAware | import org.kodein.di.DIAware | ||||||
| import org.kodein.di.android.closestDI | import org.kodein.di.android.closestDI | ||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
|  | import java.security.MessageDigest | ||||||
| import java.util.concurrent.TimeUnit | import java.util.concurrent.TimeUnit | ||||||
|  |  | ||||||
|  |  | ||||||
| class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAware { | class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAware { | ||||||
|  |  | ||||||
|     private val DRAWER_ID_TAGS = 100101L |  | ||||||
|     private val DRAWER_ID_HIDDEN_TAGS = 101100L |  | ||||||
|     private val DRAWER_ID_SOURCES = 100110L |  | ||||||
|     private val DRAWER_ID_FILTERS = 100111L |  | ||||||
|  |  | ||||||
|     private var items: ArrayList<SelfossModel.Item> = ArrayList() |     private var items: ArrayList<SelfossModel.Item> = ArrayList() | ||||||
|  |  | ||||||
|     private var elementsShown: ItemType = ItemType.UNREAD |     private var elementsShown: ItemType = ItemType.UNREAD | ||||||
| @@ -85,8 +62,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|     private var fromTabShortcut: Boolean = false |     private var fromTabShortcut: Boolean = false | ||||||
|  |  | ||||||
|     private lateinit var tagsBadge: Map<Long, Int> |  | ||||||
|  |  | ||||||
|     private val settingsLauncher = registerForActivityResult(ActivityResultContracts.StartActivityForResult()) { |     private val settingsLauncher = registerForActivityResult(ActivityResultContracts.StartActivityForResult()) { | ||||||
|         appSettingsService.refreshUserSettings() |         appSettingsService.refreshUserSettings() | ||||||
|     } |     } | ||||||
| @@ -95,6 +70,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|     private val repository : Repository by instance() |     private val repository : Repository by instance() | ||||||
|     private val appSettingsService : AppSettingsService by instance() |     private val appSettingsService : AppSettingsService by instance() | ||||||
|  |  | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|         binding = ActivityHomeBinding.inflate(layoutInflater) |         binding = ActivityHomeBinding.inflate(layoutInflater) | ||||||
| @@ -110,14 +86,8 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         setContentView(view) |         setContentView(view) | ||||||
|  |  | ||||||
|         setSupportActionBar(binding.toolBar) |         setSupportActionBar(binding.toolBar) | ||||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) |  | ||||||
|         supportActionBar?.setHomeButtonEnabled(true) |  | ||||||
|         val mDrawerToggle = ActionBarDrawerToggle(this, binding.drawerContainer, binding.toolBar, R.string.material_drawer_open, R.string.material_drawer_close) |  | ||||||
|         binding.drawerContainer.addDrawerListener(mDrawerToggle) |  | ||||||
|         mDrawerToggle.syncState() |  | ||||||
|  |  | ||||||
|         handleBottomBar() |         handleBottomBar() | ||||||
|         initDrawer() |  | ||||||
|  |  | ||||||
|         handleSwipeRefreshLayout() |         handleSwipeRefreshLayout() | ||||||
|  |  | ||||||
| @@ -138,7 +108,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         binding.swipeRefreshLayout.setOnRefreshListener { |         binding.swipeRefreshLayout.setOnRefreshListener { | ||||||
|             repository.offlineOverride = false |             repository.offlineOverride = false | ||||||
|             lastFetchDone = false |             lastFetchDone = false | ||||||
|             handleDrawerItems() |  | ||||||
|             CoroutineScope(Dispatchers.Main).launch { |             CoroutineScope(Dispatchers.Main).launch { | ||||||
|                 getElementsAccordingToTab() |                 getElementsAccordingToTab() | ||||||
|                 binding.swipeRefreshLayout.isRefreshing = false |                 binding.swipeRefreshLayout.isRefreshing = false | ||||||
| @@ -178,18 +147,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|                         adapter.handleItemAtIndex(position) |                         adapter.handleItemAtIndex(position) | ||||||
|  |  | ||||||
|                         reloadBadgeContent() |  | ||||||
|  |  | ||||||
|                         val tagHashes = i.tags.map { it.longHash() } |  | ||||||
|                         tagsBadge = tagsBadge.map { |  | ||||||
|                             if (tagHashes.contains(it.key)) { |  | ||||||
|                                 (it.key to (it.value - 1)) |  | ||||||
|                             } else { |  | ||||||
|                                 (it.key to it.value) |  | ||||||
|                             } |  | ||||||
|                         }.toMap() |  | ||||||
|                         reloadTagsBadges() |  | ||||||
|  |  | ||||||
|                         // Just load everythin |                         // Just load everythin | ||||||
|                         if (items.size <= 0) { |                         if (items.size <= 0) { | ||||||
|                             getElementsAccordingToTab() |                             getElementsAccordingToTab() | ||||||
| @@ -207,6 +164,16 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         ItemTouchHelper(simpleItemTouchCallback).attachToRecyclerView(binding.recyclerView) |         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() { |     private fun handleBottomBar() { | ||||||
|  |  | ||||||
|         tabNewBadge = TextBadgeItem() |         tabNewBadge = TextBadgeItem() | ||||||
| @@ -219,6 +186,28 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|             .setText("") |             .setText("") | ||||||
|             .setHideOnSelect(false).hide(false) |             .setHideOnSelect(false).hide(false) | ||||||
|  |  | ||||||
|  |         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 = |         val tabNew = | ||||||
|             BottomNavigationItem( |             BottomNavigationItem( | ||||||
|                 R.drawable.ic_tab_fiber_new_black_24dp, |                 R.drawable.ic_tab_fiber_new_black_24dp, | ||||||
| @@ -254,8 +243,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|     override fun onResume() { |     override fun onResume() { | ||||||
|         super.onResume() |         super.onResume() | ||||||
|  |  | ||||||
|         handleDrawerItems() |  | ||||||
|  |  | ||||||
|         reloadLayoutManager() |         reloadLayoutManager() | ||||||
|  |  | ||||||
|         if (appSettingsService.isInfiniteLoadingEnabled()) { |         if (appSettingsService.isInfiniteLoadingEnabled()) { | ||||||
| @@ -266,6 +253,8 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|         handleBottomBarActions() |         handleBottomBarActions() | ||||||
|  |  | ||||||
|  |         handleGDPRDialog(appSettingsService.settings.getBoolean("GDPR_shown", false)) | ||||||
|  |  | ||||||
|         handleRecurringTask() |         handleRecurringTask() | ||||||
|  |  | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
| @@ -275,245 +264,22 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         getElementsAccordingToTab() |         getElementsAccordingToTab() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     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) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             override fun cancel(imageView: ImageView) { |     private fun handleGDPRDialog(GDPRShown: Boolean) { | ||||||
|                 Glide.with(this@HomeActivity).clear(imageView) |         val messageDigest: MessageDigest = MessageDigest.getInstance("SHA-256") | ||||||
|  |         messageDigest.update(appSettingsService.getBaseUrl().toByteArray()) | ||||||
|  |         if (!GDPRShown) { | ||||||
|  |             val alertDialog = AlertDialog.Builder(this).create() | ||||||
|  |             alertDialog.setTitle(getString(R.string.gdpr_dialog_title)) | ||||||
|  |             alertDialog.setMessage(getString(R.string.gdpr_dialog_message)) | ||||||
|  |             alertDialog.setButton( | ||||||
|  |                 AlertDialog.BUTTON_NEUTRAL, | ||||||
|  |                 "OK" | ||||||
|  |             ) { dialog, _ -> | ||||||
|  |                 appSettingsService.settings.putBoolean("GDPR_shown", true) | ||||||
|  |                 dialog.dismiss() | ||||||
|             } |             } | ||||||
|         }) |             alertDialog.show() | ||||||
|  |  | ||||||
|         val drawerListener = object : DrawerLayout.DrawerListener { |  | ||||||
|             override fun onDrawerSlide(drawerView: View, slideOffset: Float) { |  | ||||||
|                 // We do nothing |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             override fun onDrawerOpened(drawerView: View) { |  | ||||||
|                 binding.bottomBar.hide() |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             override fun onDrawerClosed(drawerView: View) { |  | ||||||
|                 binding.bottomBar.show() |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             override fun onDrawerStateChanged(newState: Int) { |  | ||||||
|                 // We do nothing |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         binding.drawerContainer.addDrawerListener(drawerListener) |  | ||||||
|  |  | ||||||
|         // 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 |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     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() |  | ||||||
|         binding.mainDrawer.itemAdapter.add( |  | ||||||
|             PrimaryDrawerItem().apply { |  | ||||||
|                 nameRes = R.string.drawer_loading |  | ||||||
|                 isSelectable = false |  | ||||||
|             } |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         CoroutineScope(Dispatchers.IO).launch { |  | ||||||
|             val tags = repository.getTags() |  | ||||||
|             val sources = repository.getSources() |  | ||||||
|             runOnUiThread { |  | ||||||
|                 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) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -598,7 +364,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|         }) |         }) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun fetchOnEmptyList() { |     fun fetchOnEmptyList() { | ||||||
|         binding.recyclerView.doOnNextLayout { |         binding.recyclerView.doOnNextLayout { | ||||||
|             // TODO: do if last element (or is empty ?) |             // TODO: do if last element (or is empty ?) | ||||||
|             getElementsAccordingToTab(true) |             getElementsAccordingToTab(true) | ||||||
| @@ -639,7 +405,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|             binding.emptyText.visibility = View.GONE |             binding.emptyText.visibility = View.GONE | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     private fun getElementsAccordingToTab( |     fun getElementsAccordingToTab( | ||||||
|         appendResults: Boolean = false |         appendResults: Boolean = false | ||||||
|     ) { |     ) { | ||||||
|         offset = if (appendResults && items.size > 0) { |         offset = if (appendResults && items.size > 0) { | ||||||
| @@ -714,36 +480,12 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|     private fun reloadBadges() { |     private fun reloadBadges() { | ||||||
|         if (appSettingsService.isDisplayUnreadCountEnabled() || appSettingsService.isDisplayAllCountEnabled()) { |         if (appSettingsService.isDisplayUnreadCountEnabled() || appSettingsService.isDisplayAllCountEnabled()) { | ||||||
|             CoroutineScope(Dispatchers.Main).launch { |             CoroutineScope(Dispatchers.IO).launch { | ||||||
|                 repository.reloadBadges() |                 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())) |  | ||||||
|         } |  | ||||||
|         binding.mainDrawer.resetDrawerContent() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private fun calculateNoOfColumns(): Int { |     private fun calculateNoOfColumns(): Int { | ||||||
|         val displayMetrics = resources.displayMetrics |         val displayMetrics = resources.displayMetrics | ||||||
|         val dpWidth = displayMetrics.widthPixels / displayMetrics.density |         val dpWidth = displayMetrics.widthPixels / displayMetrics.density | ||||||
| @@ -789,6 +531,11 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|         when (item.itemId) { |         when (item.itemId) { | ||||||
|  |             R.id.action_filter -> { | ||||||
|  |                 val filterSheetFragment = FilterSheetFragment() | ||||||
|  |                 filterSheetFragment.show(supportFragmentManager, FilterSheetFragment.TAG) | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|             R.id.refresh -> { |             R.id.refresh -> { | ||||||
|                 needsConfirmation(R.string.menu_home_refresh, R.string.refresh_dialog_message) { |                 needsConfirmation(R.string.menu_home_refresh, R.string.refresh_dialog_message) { | ||||||
|                     Toast.makeText(this, R.string.refresh_in_progress, Toast.LENGTH_SHORT).show() |                     Toast.makeText(this, R.string.refresh_in_progress, Toast.LENGTH_SHORT).show() | ||||||
| @@ -828,8 +575,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|                                 ).show() |                                 ).show() | ||||||
|                                 tabNewBadge.removeBadge() |                                 tabNewBadge.removeBadge() | ||||||
|  |  | ||||||
|                                 handleDrawerItems() |  | ||||||
|  |  | ||||||
|                                 getElementsAccordingToTab() |                                 getElementsAccordingToTab() | ||||||
|                             } else { |                             } else { | ||||||
|                                 Toast.makeText( |                                 Toast.makeText( | ||||||
| @@ -846,10 +591,16 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|             R.id.action_disconnect -> { |             R.id.action_disconnect -> { | ||||||
|                 appSettingsService.clearAll() |                 CoroutineScope(Dispatchers.Main).launch { | ||||||
|  |                     repository.logout() | ||||||
|  |                 } | ||||||
|  |                 this@HomeActivity.finish() | ||||||
|                 val intent = Intent(this, LoginActivity::class.java) |                 val intent = Intent(this, LoginActivity::class.java) | ||||||
|                 this.startActivity(intent) |                 this.startActivity(intent) | ||||||
|                 this@HomeActivity.finish() |                 return true | ||||||
|  |             } | ||||||
|  |             R.id.action_settings -> { | ||||||
|  |                 settingsLauncher.launch(Intent(this, SettingsActivity::class.java)) | ||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|             else -> return super.onOptionsItemSelected(item) |             else -> return super.onOptionsItemSelected(item) | ||||||
| @@ -858,10 +609,10 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | |||||||
|  |  | ||||||
|     private fun maxItemNumber(): Int = |     private fun maxItemNumber(): Int = | ||||||
|         when (elementsShown) { |         when (elementsShown) { | ||||||
|             ItemType.UNREAD -> repository.badgeUnread |             ItemType.UNREAD -> repository.badgeUnread.value | ||||||
|             ItemType.ALL -> repository.badgeAll |             ItemType.ALL -> repository.badgeAll.value | ||||||
|             ItemType.STARRED -> repository.badgeStarred |             ItemType.STARRED -> repository.badgeStarred.value | ||||||
|             else -> repository.badgeUnread // if !elementsShown then unread are fetched. |             else -> repository.badgeUnread.value // if !elementsShown then unread are fetched. | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     private fun updateItems(adapterItems: ArrayList<SelfossModel.Item>) { |     private fun updateItems(adapterItems: ArrayList<SelfossModel.Item>) { | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ package bou.amine.apps.readerforselfossv2.android | |||||||
|  |  | ||||||
| import android.animation.Animator | import android.animation.Animator | ||||||
| import android.animation.AnimatorListenerAdapter | import android.animation.AnimatorListenerAdapter | ||||||
|  | import android.annotation.SuppressLint | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.text.TextUtils | import android.text.TextUtils | ||||||
| @@ -10,6 +11,7 @@ import android.view.MenuItem | |||||||
| import android.view.View | import android.view.View | ||||||
| import android.view.inputmethod.EditorInfo | import android.view.inputmethod.EditorInfo | ||||||
| import android.widget.TextView | import android.widget.TextView | ||||||
|  | import android.widget.Toast | ||||||
| import androidx.appcompat.app.AlertDialog | import androidx.appcompat.app.AlertDialog | ||||||
| import androidx.appcompat.app.AppCompatActivity | import androidx.appcompat.app.AppCompatActivity | ||||||
| import androidx.appcompat.app.AppCompatDelegate | import androidx.appcompat.app.AppCompatDelegate | ||||||
| @@ -21,9 +23,12 @@ import com.mikepenz.aboutlibraries.LibsBuilder | |||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| import kotlinx.coroutines.Dispatchers | import kotlinx.coroutines.Dispatchers | ||||||
| import kotlinx.coroutines.launch | import kotlinx.coroutines.launch | ||||||
|  | import org.acra.ACRA | ||||||
| import org.kodein.di.DIAware | import org.kodein.di.DIAware | ||||||
| import org.kodein.di.android.closestDI | import org.kodein.di.android.closestDI | ||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
|  | import java.security.MessageDigest | ||||||
|  |  | ||||||
|  |  | ||||||
| class LoginActivity : AppCompatActivity(), DIAware { | class LoginActivity : AppCompatActivity(), DIAware { | ||||||
|  |  | ||||||
| @@ -36,9 +41,11 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|     private val repository: Repository by instance() |     private val repository: Repository by instance() | ||||||
|     private val appSettingsService: AppSettingsService by instance() |     private val appSettingsService: AppSettingsService by instance() | ||||||
|  |  | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|         AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_FOLLOW_SYSTEM) |  | ||||||
|  |         handleTheme() | ||||||
|  |  | ||||||
|         binding = ActivityLoginBinding.inflate(layoutInflater) |         binding = ActivityLoginBinding.inflate(layoutInflater) | ||||||
|         val view = binding.root |         val view = binding.root | ||||||
| @@ -50,12 +57,40 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|         handleBaseUrlFail() |         handleBaseUrlFail() | ||||||
|  |  | ||||||
|         if (appSettingsService.getBaseUrl().isNotEmpty()) { |         if (appSettingsService.getBaseUrl().isNotEmpty()) { | ||||||
|  |             showProgress(true) | ||||||
|  |             // This should be reverted when "old" users connected with a non-selfoss rss | ||||||
|  |             // are handled. Revert to "simple" way. | ||||||
|  |             CoroutineScope(Dispatchers.Main).launch { | ||||||
|  |                 try { | ||||||
|  |                     val (errorFetching, displaySelfossOnly) = repository.shouldBeSelfossInstance() | ||||||
|  |                     if (!errorFetching && !displaySelfossOnly) { | ||||||
|                         goToMain() |                         goToMain() | ||||||
|  |                     } else { | ||||||
|  |                         showProgress(false) | ||||||
|  |                         if (displaySelfossOnly) { | ||||||
|  |                             Toast.makeText( | ||||||
|  |                                 applicationContext, | ||||||
|  |                                 R.string.application_selfoss_only, | ||||||
|  |                                 Toast.LENGTH_LONG | ||||||
|  |                             ).show() | ||||||
|  |                         } | ||||||
|  |                         repository.logout() | ||||||
|  |                     } | ||||||
|  |                 } catch (e: Throwable) { | ||||||
|  |                     repository.logout() | ||||||
|  |                     showProgress(false) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         handleActions() |         handleActions() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @SuppressLint("WrongConstant") // Constant is fetched from the settings | ||||||
|  |     private fun handleTheme() { | ||||||
|  |         AppCompatDelegate.setDefaultNightMode(appSettingsService.getCurrentTheme()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private fun handleActions() { |     private fun handleActions() { | ||||||
|  |  | ||||||
|         binding.passwordView.setOnEditorActionListener( |         binding.passwordView.setOnEditorActionListener( | ||||||
| @@ -93,12 +128,16 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun goToMain() { |     private fun goToMain() { | ||||||
|  |         CoroutineScope(Dispatchers.Main).launch { | ||||||
|  |             repository.updateApiVersion() | ||||||
|  |             ACRA.errorReporter.putCustomData("SELFOSS_API_VERSION", appSettingsService.getApiVersion().toString()) | ||||||
|  |         } | ||||||
|         val intent = Intent(this, HomeActivity::class.java) |         val intent = Intent(this, HomeActivity::class.java) | ||||||
|         startActivity(intent) |         startActivity(intent) | ||||||
|         finish() |         finish() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun preferenceError(t: Throwable) { |     private fun preferenceError() { | ||||||
|         appSettingsService.resetLoginInformation() |         appSettingsService.resetLoginInformation() | ||||||
|  |  | ||||||
|         binding.urlView.error = getString(R.string.wrong_infos) |         binding.urlView.error = getString(R.string.wrong_infos) | ||||||
| @@ -160,19 +199,29 @@ class LoginActivity : AppCompatActivity(), DIAware { | |||||||
|  |  | ||||||
|             repository.refreshLoginInformation(url, login, password) |             repository.refreshLoginInformation(url, login, password) | ||||||
|  |  | ||||||
|             CoroutineScope(Dispatchers.IO).launch { |             CoroutineScope(Dispatchers.Main).launch { | ||||||
|                 val result = repository.login() |                 val result = repository.login() | ||||||
|                 if (result) { |                 if (result) { | ||||||
|  |                     val (errorFetching, displaySelfossOnly) = repository.shouldBeSelfossInstance() | ||||||
|  |                     if (!errorFetching && !displaySelfossOnly) { | ||||||
|                         goToMain() |                         goToMain() | ||||||
|                     } else { |                     } else { | ||||||
|                     CoroutineScope(Dispatchers.Main).launch { |                         if (displaySelfossOnly) { | ||||||
|                         preferenceError(Exception("Not success")) |                             Toast.makeText( | ||||||
|  |                                 applicationContext, | ||||||
|  |                                 R.string.application_selfoss_only, | ||||||
|  |                                 Toast.LENGTH_LONG | ||||||
|  |                             ).show() | ||||||
|                         } |                         } | ||||||
|  |                         preferenceError() | ||||||
|                     } |                     } | ||||||
|  |                 } else { | ||||||
|  |                     preferenceError() | ||||||
|                 } |                 } | ||||||
|                 showProgress(false) |                 showProgress(false) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private fun showProgress(show: Boolean) { |     private fun showProgress(show: Boolean) { | ||||||
|         val shortAnimTime = resources.getInteger(android.R.integer.config_shortAnimTime) |         val shortAnimTime = resources.getInteger(android.R.integer.config_shortAnimTime) | ||||||
|   | |||||||
| @@ -12,7 +12,6 @@ import androidx.lifecycle.DefaultLifecycleObserver | |||||||
| import androidx.lifecycle.LifecycleOwner | import androidx.lifecycle.LifecycleOwner | ||||||
| import androidx.lifecycle.ProcessLifecycleOwner | import androidx.lifecycle.ProcessLifecycleOwner | ||||||
| import androidx.multidex.MultiDexApplication | import androidx.multidex.MultiDexApplication | ||||||
| import androidx.preference.PreferenceManager |  | ||||||
| import bou.amine.apps.readerforselfossv2.DI.networkModule | import bou.amine.apps.readerforselfossv2.DI.networkModule | ||||||
| import bou.amine.apps.readerforselfossv2.android.viewmodel.AppViewModel | import bou.amine.apps.readerforselfossv2.android.viewmodel.AppViewModel | ||||||
| import bou.amine.apps.readerforselfossv2.dao.DriverFactory | import bou.amine.apps.readerforselfossv2.dao.DriverFactory | ||||||
| @@ -22,13 +21,19 @@ import bou.amine.apps.readerforselfossv2.service.AppSettingsService | |||||||
| import com.bumptech.glide.Glide | import com.bumptech.glide.Glide | ||||||
| import com.bumptech.glide.request.RequestOptions | import com.bumptech.glide.request.RequestOptions | ||||||
| import com.github.ln_12.library.ConnectivityStatus | import com.github.ln_12.library.ConnectivityStatus | ||||||
| import com.mikepenz.materialdrawer.util.AbstractDrawerImageLoader |  | ||||||
| import com.mikepenz.materialdrawer.util.DrawerImageLoader |  | ||||||
| import io.github.aakira.napier.DebugAntilog | import io.github.aakira.napier.DebugAntilog | ||||||
| import io.github.aakira.napier.Napier | import io.github.aakira.napier.Napier | ||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| import kotlinx.coroutines.Dispatchers | import kotlinx.coroutines.Dispatchers | ||||||
|  | import kotlinx.coroutines.flow.MutableStateFlow | ||||||
| import kotlinx.coroutines.launch | import kotlinx.coroutines.launch | ||||||
|  | import org.acra.ACRA | ||||||
|  | import org.acra.ReportField | ||||||
|  | import org.acra.config.httpSender | ||||||
|  | import org.acra.config.toast | ||||||
|  | import org.acra.data.StringFormat | ||||||
|  | import org.acra.ktx.initAcra | ||||||
|  | import org.acra.sender.HttpSender | ||||||
| import org.kodein.di.* | import org.kodein.di.* | ||||||
|  |  | ||||||
| class MyApp : MultiDexApplication(), DIAware { | class MyApp : MultiDexApplication(), DIAware { | ||||||
| @@ -37,7 +42,7 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|         import(networkModule) |         import(networkModule) | ||||||
|         bind<DriverFactory>() with singleton { DriverFactory(applicationContext) } |         bind<DriverFactory>() with singleton { DriverFactory(applicationContext) } | ||||||
|         bind<ReaderForSelfossDB>() with singleton { ReaderForSelfossDB(driverFactory.createDriver()) } |         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<ConnectivityStatus>() with singleton { ConnectivityStatus(applicationContext) } | ||||||
|         bind<AppViewModel>() with singleton { AppViewModel(repository = instance()) } |         bind<AppViewModel>() with singleton { AppViewModel(repository = instance()) } | ||||||
|     } |     } | ||||||
| @@ -47,12 +52,14 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|     private val connectivityStatus: ConnectivityStatus by instance() |     private val connectivityStatus: ConnectivityStatus by instance() | ||||||
|     private val driverFactory: DriverFactory by instance() |     private val driverFactory: DriverFactory by instance() | ||||||
|  |  | ||||||
|  |     // TODO: handle with the "previous" way | ||||||
|  |     private val isConnectionAvailable: MutableStateFlow<Boolean> = MutableStateFlow(true) | ||||||
|  |  | ||||||
|     override fun onCreate() { |     override fun onCreate() { | ||||||
|         super.onCreate() |         super.onCreate() | ||||||
|         Napier.base(DebugAntilog()) |         Napier.base(DebugAntilog()) | ||||||
|  |  | ||||||
|         initDrawerImageLoader() |         if (!ACRA.isACRASenderServiceProcess()) { | ||||||
|  |  | ||||||
|             tryToHandleBug() |             tryToHandleBug() | ||||||
|  |  | ||||||
|             handleNotificationChannels() |             handleNotificationChannels() | ||||||
| @@ -76,6 +83,34 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun attachBaseContext(base: Context?) { | ||||||
|  |         super.attachBaseContext(base) | ||||||
|  |  | ||||||
|  |         initAcra { | ||||||
|  |             reportFormat = StringFormat.JSON | ||||||
|  |             reportContent = listOf( | ||||||
|  |                 ReportField.REPORT_ID, ReportField.INSTALLATION_ID, | ||||||
|  |                 ReportField.APP_VERSION_CODE, ReportField.APP_VERSION_NAME, | ||||||
|  |                 ReportField.BUILD, ReportField.ANDROID_VERSION, ReportField.BRAND, ReportField.PHONE_MODEL, | ||||||
|  |                 ReportField.AVAILABLE_MEM_SIZE, ReportField.TOTAL_MEM_SIZE, | ||||||
|  |                 ReportField.STACK_TRACE, ReportField.APPLICATION_LOG, ReportField.LOGCAT, | ||||||
|  |                 ReportField.INITIAL_CONFIGURATION, ReportField.CRASH_CONFIGURATION, ReportField.IS_SILENT, | ||||||
|  |                 ReportField.USER_APP_START_DATE, ReportField.USER_COMMENT, ReportField.USER_CRASH_DATE, ReportField.USER_EMAIL, ReportField.CUSTOM_DATA) | ||||||
|  |             toast { | ||||||
|  |                 //required | ||||||
|  |                 text = getString(R.string.crash_toast_text) | ||||||
|  |                 length = Toast.LENGTH_SHORT | ||||||
|  |             } | ||||||
|  |             httpSender { | ||||||
|  |                 uri = "https://bugs.amine-louveau.fr/report" /*best guess, you may need to adjust this*/ | ||||||
|  |                 basicAuthLogin = "LMTlLZuazADohTCm" | ||||||
|  |                 basicAuthPassword = "he6ghHp83F0PYPfh" | ||||||
|  |                 httpMethod = HttpSender.Method.POST | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private fun handleNotificationChannels() { |     private fun handleNotificationChannels() { | ||||||
|         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { |         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { | ||||||
| @@ -94,33 +129,13 @@ class MyApp : MultiDexApplication(), DIAware { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun initDrawerImageLoader() { |  | ||||||
|         DrawerImageLoader.init(object : AbstractDrawerImageLoader() { |  | ||||||
|             override fun set(imageView: ImageView, uri: Uri, placeholder: Drawable, tag: String?) { |  | ||||||
|                 Glide.with(imageView.context) |  | ||||||
|                     .load(uri.toString()) |  | ||||||
|                     .apply(RequestOptions.fitCenterTransform().placeholder(placeholder)) |  | ||||||
|                     .into(imageView) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             override fun cancel(imageView: ImageView) { |  | ||||||
|                 Glide.with(imageView.context).clear(imageView) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             override fun placeholder(ctx: Context, tag: String?): Drawable { |  | ||||||
|                 return baseContext.resources.getDrawable(R.mipmap.ic_launcher) |  | ||||||
|             } |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private fun tryToHandleBug() { |     private fun tryToHandleBug() { | ||||||
|         val oldHandler = Thread.getDefaultUncaughtExceptionHandler() |         val oldHandler = Thread.getDefaultUncaughtExceptionHandler() | ||||||
|  |  | ||||||
|         Thread.setDefaultUncaughtExceptionHandler { thread, e -> |         Thread.setDefaultUncaughtExceptionHandler { thread, e -> | ||||||
|             if (e is java.lang.NoClassDefFoundError && e.stackTrace.asList().any { |             if (e is NoClassDefFoundError && e.stackTrace.asList().any { | ||||||
|                     it.toString().contains("android.view.ViewDebug") |                     it.toString().contains("android.view.ViewDebug") | ||||||
|                 }) { |                 }) { | ||||||
|                 Unit |  | ||||||
|             } else { |             } else { | ||||||
|                 oldHandler.uncaughtException(thread, e) |                 oldHandler.uncaughtException(thread, e) | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -30,15 +30,17 @@ class ReaderActivity : AppCompatActivity(), DIAware { | |||||||
|  |  | ||||||
|     private lateinit var binding: ActivityReaderBinding |     private lateinit var binding: ActivityReaderBinding | ||||||
|  |  | ||||||
|  |     private var allItems: ArrayList<SelfossModel.Item> = ArrayList() | ||||||
|  |  | ||||||
|     override val di by closestDI() |     override val di by closestDI() | ||||||
|     private val repository: Repository by instance() |     private val repository: Repository by instance() | ||||||
|     private val appSettingsService: AppSettingsService by instance() |     private val appSettingsService: AppSettingsService by instance() | ||||||
|  |  | ||||||
|     private fun showMenuItem(willAddToFavorite: Boolean) { |     private fun showMenuItem(willAddToFavorite: Boolean) { | ||||||
|         if (willAddToFavorite) { |         if (willAddToFavorite) { | ||||||
|             toolbarMenu.findItem(R.id.star).icon.setTint(Color.WHITE) |             toolbarMenu.findItem(R.id.star).icon?.setTint(Color.WHITE) | ||||||
|         } else { |         } else { | ||||||
|             toolbarMenu.findItem(R.id.star).icon.setTint(Color.RED) |             toolbarMenu.findItem(R.id.star).icon?.setTint(Color.RED) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -61,12 +63,14 @@ class ReaderActivity : AppCompatActivity(), DIAware { | |||||||
|         supportActionBar?.setDisplayHomeAsUpEnabled(true) |         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||||
|         supportActionBar?.setDisplayShowHomeEnabled(true) |         supportActionBar?.setDisplayShowHomeEnabled(true) | ||||||
|  |  | ||||||
|         if (allItems.isEmpty()) { |         currentItem = intent.getIntExtra("currentItem", 0) | ||||||
|  |  | ||||||
|  |         allItems = repository.getReaderItems() | ||||||
|  |  | ||||||
|  |         if (allItems.isEmpty() || currentItem > allItems.size) { | ||||||
|             finish() |             finish() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         currentItem = intent.getIntExtra("currentItem", 0) |  | ||||||
|  |  | ||||||
|         readItem(allItems[currentItem]) |         readItem(allItems[currentItem]) | ||||||
|  |  | ||||||
|         binding.pager.adapter = ScreenSlidePagerAdapter(this) |         binding.pager.adapter = ScreenSlidePagerAdapter(this) | ||||||
| @@ -214,8 +218,4 @@ class ReaderActivity : AppCompatActivity(), DIAware { | |||||||
|         startActivity(intent) |         startActivity(intent) | ||||||
|         overridePendingTransition(0, 0) |         overridePendingTransition(0, 0) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     companion object { |  | ||||||
|         var allItems: ArrayList<SelfossModel.Item> = ArrayList() |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -56,20 +56,13 @@ class SourcesActivity : AppCompatActivity(), DIAware { | |||||||
|  |  | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
|             val response = repository.getSources() |             val response = repository.getSources() | ||||||
|             if (response != null) { |             if (response.isNotEmpty()) { | ||||||
|                 items = response |                 items = response | ||||||
|                 val mAdapter = SourcesListAdapter( |                 val mAdapter = SourcesListAdapter( | ||||||
|                     this@SourcesActivity, items |                     this@SourcesActivity, items | ||||||
|                 ) |                 ) | ||||||
|                 binding.recyclerView.adapter = mAdapter |                 binding.recyclerView.adapter = mAdapter | ||||||
|                 mAdapter.notifyDataSetChanged() |                 mAdapter.notifyDataSetChanged() | ||||||
|                 if (items.isEmpty()) { |  | ||||||
|                     Toast.makeText( |  | ||||||
|                         this@SourcesActivity, |  | ||||||
|                         R.string.nothing_here, |  | ||||||
|                         Toast.LENGTH_SHORT |  | ||||||
|                     ).show() |  | ||||||
|                 } |  | ||||||
|             } else { |             } else { | ||||||
|                 Toast.makeText( |                 Toast.makeText( | ||||||
|                     this@SourcesActivity, |                     this@SourcesActivity, | ||||||
|   | |||||||
| @@ -10,9 +10,12 @@ import androidx.recyclerview.widget.RecyclerView | |||||||
| import bou.amine.apps.readerforselfossv2.android.R | import bou.amine.apps.readerforselfossv2.android.R | ||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.CardItemBinding | import bou.amine.apps.readerforselfossv2.android.databinding.CardItemBinding | ||||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.* | 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.bitmapCenterCrop | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.circularBitmapDrawable | 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.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
| @@ -59,7 +62,7 @@ class ItemCardAdapter( | |||||||
|  |  | ||||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) |             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||||
|  |  | ||||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText(repository.dateUtils) |             binding.sourceTitleAndDate.text = itm.sourceAuthorAndDate() | ||||||
|  |  | ||||||
|             if (!appSettingsService.isFullHeightCardsEnabled()) { |             if (!appSettingsService.isFullHeightCardsEnabled()) { | ||||||
|                 binding.itemImage.maxHeight = imageMaxHeight |                 binding.itemImage.maxHeight = imageMaxHeight | ||||||
| @@ -108,13 +111,11 @@ class ItemCardAdapter( | |||||||
|                     CoroutineScope(Dispatchers.IO).launch { |                     CoroutineScope(Dispatchers.IO).launch { | ||||||
|                         repository.unstarr(item) |                         repository.unstarr(item) | ||||||
|                     } |                     } | ||||||
|                     item.starred = false |  | ||||||
|                     binding.favButton.isSelected = false |                     binding.favButton.isSelected = false | ||||||
|                 } else { |                 } else { | ||||||
|                     CoroutineScope(Dispatchers.IO).launch { |                     CoroutineScope(Dispatchers.IO).launch { | ||||||
|                         repository.starr(item) |                         repository.starr(item) | ||||||
|                     } |                     } | ||||||
|                     item.starred = true |  | ||||||
|                     binding.favButton.isSelected = true |                     binding.favButton.isSelected = true | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -131,8 +132,8 @@ class ItemCardAdapter( | |||||||
|  |  | ||||||
|         private fun handleLinkOpening() { |         private fun handleLinkOpening() { | ||||||
|             binding.root.setOnClickListener { |             binding.root.setOnClickListener { | ||||||
|  |                 repository.setReaderItems(items) | ||||||
|                 c.openItemUrl( |                 c.openItemUrl( | ||||||
|                     items, |  | ||||||
|                     bindingAdapterPosition, |                     bindingAdapterPosition, | ||||||
|                     items[bindingAdapterPosition].getLinkDecoded(), |                     items[bindingAdapterPosition].getLinkDecoded(), | ||||||
|                     appSettingsService.isArticleViewerEnabled(), |                     appSettingsService.isArticleViewerEnabled(), | ||||||
|   | |||||||
| @@ -51,7 +51,7 @@ class ItemListAdapter( | |||||||
|  |  | ||||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) |             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||||
|  |  | ||||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText(repository.dateUtils) |             binding.sourceTitleAndDate.text = itm.sourceAuthorAndDate() | ||||||
|  |  | ||||||
|             if (itm.getThumbnail(repository.baseUrl).isEmpty()) { |             if (itm.getThumbnail(repository.baseUrl).isEmpty()) { | ||||||
|  |  | ||||||
| @@ -84,8 +84,8 @@ class ItemListAdapter( | |||||||
|  |  | ||||||
|         private fun handleLinkOpening() { |         private fun handleLinkOpening() { | ||||||
|             binding.root.setOnClickListener { |             binding.root.setOnClickListener { | ||||||
|  |                 repository.setReaderItems(items) | ||||||
|                 c.openItemUrl( |                 c.openItemUrl( | ||||||
|                     items, |  | ||||||
|                     bindingAdapterPosition, |                     bindingAdapterPosition, | ||||||
|                     items[bindingAdapterPosition].getLinkDecoded(), |                     items[bindingAdapterPosition].getLinkDecoded(), | ||||||
|                     appSettingsService.isArticleViewerEnabled(), |                     appSettingsService.isArticleViewerEnabled(), | ||||||
|   | |||||||
| @@ -36,10 +36,8 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|                 Snackbar.LENGTH_LONG |                 Snackbar.LENGTH_LONG | ||||||
|             ) |             ) | ||||||
|             .setAction(R.string.undo_string) { |             .setAction(R.string.undo_string) { | ||||||
|                 CoroutineScope(Dispatchers.IO).launch { |  | ||||||
|                 unreadItemAtIndex(item, position, false) |                 unreadItemAtIndex(item, position, false) | ||||||
|             } |             } | ||||||
|             } |  | ||||||
|  |  | ||||||
|         val view = s.view |         val view = s.view | ||||||
|         val tv: TextView = view.findViewById(com.google.android.material.R.id.snackbar_text) |         val tv: TextView = view.findViewById(com.google.android.material.R.id.snackbar_text) | ||||||
|   | |||||||
| @@ -61,9 +61,13 @@ class SourcesListAdapter( | |||||||
|         binding.sourceTitle.text = itm.title.getHtmlDecoded() |         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 |     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 { |         init { | ||||||
|             handleClickListeners() |             handleClickListeners() | ||||||
| @@ -74,13 +78,13 @@ class SourcesListAdapter( | |||||||
|             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) |             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) | ||||||
|  |  | ||||||
|             deleteBtn.setOnClickListener { |             deleteBtn.setOnClickListener { | ||||||
|                 val (id) = items[adapterPosition] |                 val (id, title) = items[bindingAdapterPosition] | ||||||
|                 CoroutineScope(Dispatchers.IO).launch { |                 CoroutineScope(Dispatchers.IO).launch { | ||||||
|                     val successfullyDeletedSource = repository.deleteSource(id) |                     val successfullyDeletedSource = repository.deleteSource(id, title) | ||||||
|                     if (successfullyDeletedSource) { |                     if (successfullyDeletedSource) { | ||||||
|                         items.removeAt(adapterPosition) |                         items.removeAt(bindingAdapterPosition) | ||||||
|                         notifyItemRemoved(adapterPosition) |                         notifyItemRemoved(bindingAdapterPosition) | ||||||
|                         notifyItemRangeChanged(adapterPosition, itemCount) |                         notifyItemRangeChanged(bindingAdapterPosition, itemCount) | ||||||
|                     } else { |                     } else { | ||||||
|                         Toast.makeText( |                         Toast.makeText( | ||||||
|                             app, |                             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> |  | ||||||
| } |  | ||||||
| @@ -16,22 +16,21 @@ import android.webkit.WebView | |||||||
| import android.webkit.WebViewClient | import android.webkit.WebViewClient | ||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import androidx.appcompat.app.AlertDialog | import androidx.appcompat.app.AlertDialog | ||||||
| import androidx.core.content.res.ResourcesCompat |  | ||||||
| import androidx.core.widget.NestedScrollView | import androidx.core.widget.NestedScrollView | ||||||
| import androidx.fragment.app.Fragment | import androidx.fragment.app.Fragment | ||||||
| import bou.amine.apps.readerforselfossv2.android.ImageActivity | import bou.amine.apps.readerforselfossv2.android.ImageActivity | ||||||
| import bou.amine.apps.readerforselfossv2.android.R | 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.databinding.FragmentArticleBinding | ||||||
| import bou.amine.apps.readerforselfossv2.android.model.ParecelableItem | import bou.amine.apps.readerforselfossv2.android.model.ParecelableItem | ||||||
| import bou.amine.apps.readerforselfossv2.android.model.toModel | import bou.amine.apps.readerforselfossv2.android.model.toModel | ||||||
| import bou.amine.apps.readerforselfossv2.android.model.toParcelable | import bou.amine.apps.readerforselfossv2.android.model.toParcelable | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.getBitmapInputStream | import bou.amine.apps.readerforselfossv2.android.utils.glide.getBitmapInputStream | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | ||||||
| import bou.amine.apps.readerforselfossv2.android.utils.shareLink | import bou.amine.apps.readerforselfossv2.android.utils.shareLink | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | 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.service.AppSettingsService | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getImages | import bou.amine.apps.readerforselfossv2.utils.getImages | ||||||
| @@ -49,10 +48,8 @@ import org.kodein.di.DI | |||||||
| import org.kodein.di.DIAware | import org.kodein.di.DIAware | ||||||
| import org.kodein.di.android.x.closestDI | import org.kodein.di.android.x.closestDI | ||||||
| import org.kodein.di.instance | import org.kodein.di.instance | ||||||
| import retrofit2.Call |  | ||||||
| import retrofit2.Callback |  | ||||||
| import retrofit2.Response |  | ||||||
| import java.net.MalformedURLException | import java.net.MalformedURLException | ||||||
|  | import java.net.SocketTimeoutException | ||||||
| import java.net.URL | import java.net.URL | ||||||
| import java.util.* | import java.util.* | ||||||
| import java.util.concurrent.ExecutionException | import java.util.concurrent.ExecutionException | ||||||
| @@ -70,7 +67,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|     private lateinit var fab: FloatingActionButton |     private lateinit var fab: FloatingActionButton | ||||||
|     private lateinit var textAlignment: String |     private lateinit var textAlignment: String | ||||||
|     private var _binding: FragmentArticleBinding? = null |     private var _binding: FragmentArticleBinding? = null | ||||||
|     private val binding get() = _binding!! |     private val binding get() = _binding | ||||||
|  |  | ||||||
|     override val di : DI by closestDI() |     override val di : DI by closestDI() | ||||||
|     private val repository: Repository by instance() |     private val repository: Repository by instance() | ||||||
| @@ -81,6 +78,9 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|     private var font = "" |     private var font = "" | ||||||
|     private var staticBar = false |     private var staticBar = false | ||||||
|  |  | ||||||
|  |     private val mercuryApi : MercuryApi by instance() | ||||||
|  |  | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
|  |  | ||||||
| @@ -101,32 +101,22 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|             contentText = item.content |             contentText = item.content | ||||||
|             contentTitle = item.title.getHtmlDecoded() |             contentTitle = item.title.getHtmlDecoded() | ||||||
|             contentImage = item.getThumbnail(repository.baseUrl) |             contentImage = item.getThumbnail(repository.baseUrl) | ||||||
|             contentSource = item.sourceAndDateText(repository.dateUtils) |             contentSource = item.sourceAuthorAndDate() | ||||||
|             allImages = item.getImages() |             allImages = item.getImages() | ||||||
|  |  | ||||||
|             fontSize = appSettingsService.getFontSize() |             fontSize = appSettingsService.getFontSize() | ||||||
|             staticBar = appSettingsService.isStaticBarEnabled() |             staticBar = appSettingsService.isStaticBarEnabled() | ||||||
|             font = appSettingsService.getFont() |             font = appSettingsService.getFont() | ||||||
|  |  | ||||||
|             if (font.isNotEmpty()) { |  | ||||||
|                 resId = requireContext().resources.getIdentifier(font, "font", requireContext().packageName) |  | ||||||
|                 typeface = try { |  | ||||||
|                     ResourcesCompat.getFont(requireContext(), resId)!! |  | ||||||
|                 } catch (e: java.lang.Exception) { |  | ||||||
|                     // Just to be sure |  | ||||||
|                     null |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             refreshAlignment() |             refreshAlignment() | ||||||
|  |  | ||||||
|             fab = binding.fab |             fab = binding!!.fab | ||||||
|  |  | ||||||
|             fab.backgroundTintList = ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) |             fab.backgroundTintList = ColorStateList.valueOf(resources.getColor(R.color.colorAccent)) | ||||||
|  |  | ||||||
|             fab.rippleColor = resources.getColor(R.color.colorAccentDark) |             fab.rippleColor = resources.getColor(R.color.colorAccentDark) | ||||||
|  |  | ||||||
|             val floatingToolbar: FloatingToolbar = binding.floatingToolbar |             val floatingToolbar: FloatingToolbar = binding!!.floatingToolbar | ||||||
|             floatingToolbar.attachFab(fab) |             floatingToolbar.attachFab(fab) | ||||||
|  |  | ||||||
|             floatingToolbar.background = ColorDrawable(resources.getColor(R.color.colorAccent)) |             floatingToolbar.background = ColorDrawable(resources.getColor(R.color.colorAccent)) | ||||||
| @@ -174,35 +164,35 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                 floatingToolbar.show() |                 floatingToolbar.show() | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             binding.source.text = contentSource |             binding!!.source.text = contentSource | ||||||
|             if (typeface != null) { |             if (typeface != null) { | ||||||
|                 binding.source.typeface = typeface |                 binding!!.source.typeface = typeface | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if (contentText.isEmptyOrNullOrNullString()) { |             if (contentText.isEmptyOrNullOrNullString()) { | ||||||
|                 getContentFromMercury() |                 getContentFromMercury() | ||||||
|             } else { |             } else { | ||||||
|                 binding.titleView.text = contentTitle |                 binding!!.titleView.text = contentTitle | ||||||
|                 if (typeface != null) { |                 if (typeface != null) { | ||||||
|                     binding.titleView.typeface = typeface |                     binding!!.titleView.typeface = typeface | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 htmlToWebview() |                 htmlToWebview() | ||||||
|  |  | ||||||
|                 if (!contentImage.isEmptyOrNullOrNullString() && context != null) { |                 if (!contentImage.isEmptyOrNullOrNullString() && context != null) { | ||||||
|                     binding.imageView.visibility = View.VISIBLE |                     binding!!.imageView.visibility = View.VISIBLE | ||||||
|                     Glide |                     Glide | ||||||
|                         .with(requireContext()) |                         .with(requireContext()) | ||||||
|                         .asBitmap() |                         .asBitmap() | ||||||
|                         .load(contentImage) |                         .load(contentImage) | ||||||
|                         .apply(RequestOptions.fitCenterTransform()) |                         .apply(RequestOptions.fitCenterTransform()) | ||||||
|                         .into(binding.imageView) |                         .into(binding!!.imageView) | ||||||
|                 } else { |                 } else { | ||||||
|                     binding.imageView.visibility = View.GONE |                     binding!!.imageView.visibility = View.GONE | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             binding.nestedScrollView.setOnScrollChangeListener( |             binding!!.nestedScrollView.setOnScrollChangeListener( | ||||||
|                 NestedScrollView.OnScrollChangeListener { _, _, scrollY, _, oldScrollY -> |                 NestedScrollView.OnScrollChangeListener { _, _, scrollY, _, oldScrollY -> | ||||||
|                     if (scrollY > oldScrollY) { |                     if (scrollY > oldScrollY) { | ||||||
|                         floatingToolbar.hide() |                         floatingToolbar.hide() | ||||||
| @@ -218,6 +208,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         } catch (e: InflateException) { |         } catch (e: InflateException) { | ||||||
|  |             e.sendSilentlyWithAcraWithName("webview not available") | ||||||
|             AlertDialog.Builder(requireContext()) |             AlertDialog.Builder(requireContext()) | ||||||
|                 .setMessage(requireContext().getString(R.string.webview_dialog_issue_message)) |                 .setMessage(requireContext().getString(R.string.webview_dialog_issue_message)) | ||||||
|                 .setTitle(requireContext().getString(R.string.webview_dialog_issue_title)) |                 .setTitle(requireContext().getString(R.string.webview_dialog_issue_title)) | ||||||
| @@ -230,7 +221,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|                 .show() |                 .show() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return binding.root |         return binding!!.root | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onDestroyView() { |     override fun onDestroyView() { | ||||||
| @@ -248,90 +239,76 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|  |  | ||||||
|     private fun getContentFromMercury() { |     private fun getContentFromMercury() { | ||||||
|         if (repository.isNetworkAvailable()) { |         if (repository.isNetworkAvailable()) { | ||||||
|             binding.progressBar.visibility = View.VISIBLE |             binding!!.progressBar.visibility = View.VISIBLE | ||||||
|             val parser = MercuryApi() |  | ||||||
|  |  | ||||||
|             parser.parseUrl(url).enqueue( |             CoroutineScope(Dispatchers.Main).launch { | ||||||
|                 object : Callback<ParsedContent> { |  | ||||||
|                     override fun onResponse( |  | ||||||
|                         call: Call<ParsedContent>, |  | ||||||
|                         response: Response<ParsedContent> |  | ||||||
|                     ) { |  | ||||||
|                         // TODO: clean all the following after finding the mercury content issue |  | ||||||
|                 try { |                 try { | ||||||
|                             if (response.body() != null && response.body()!!.content != null && !response.body()!!.content.isNullOrEmpty()) { |                     val response = mercuryApi.query(url) | ||||||
|  |                     if (response.success && response.data != null && !response.data?.content.isNullOrEmpty()) { | ||||||
|  |                         binding!!.titleView.text = response.data!!.title.orEmpty() | ||||||
|                         try { |                         try { | ||||||
|                                     binding.titleView.text = response.body()!!.title |  | ||||||
|                             if (typeface != null) { |                             if (typeface != null) { | ||||||
|                                         binding.titleView.typeface = typeface |                                 binding!!.titleView.typeface = typeface | ||||||
|                             } |                             } | ||||||
|  |                         } catch (e: Exception) { | ||||||
|  |                             e.sendSilentlyWithAcraWithName("getContentFromMercury > typeface") | ||||||
|  |                         } | ||||||
|  |  | ||||||
|                         try { |                         try { | ||||||
|                             // Note: Mercury may return relative urls... If it does the url val will not be changed. |                             // Note: Mercury may return relative urls... If it does the url val will not be changed. | ||||||
|                                         URL(response.body()!!.url) |                             URL(response.data!!.url) | ||||||
|                                         url = response.body()!!.url |                             url = response.data!!.url | ||||||
|                         } catch (e: MalformedURLException) { |                         } catch (e: MalformedURLException) { | ||||||
|                                         // Mercury returned a relative url. We do nothing. |                             // Mercury returned a relative url | ||||||
|                                     } |                             e.sendSilentlyWithAcraWithName("getContentFromMercury > malformedurlexception") | ||||||
|                                 } catch (e: Exception) { |  | ||||||
|                         } |                         } | ||||||
|  |  | ||||||
|                         try { |                         try { | ||||||
|                                     contentText = response.body()!!.content.orEmpty() |                             contentText = response.data!!.content.orEmpty() | ||||||
|                             htmlToWebview() |                             htmlToWebview() | ||||||
|                         } catch (e: Exception) { |                         } catch (e: Exception) { | ||||||
|  |                             e.sendSilentlyWithAcraWithName("getContentFromMercury > contenttext or html") | ||||||
|                         } |                         } | ||||||
|  |  | ||||||
|  |                         if (!response.data?.lead_image_url.isNullOrEmpty() && context != null) { | ||||||
|                             try { |                             try { | ||||||
|                                     if (response.body()!!.lead_image_url != null && !response.body()!!.lead_image_url.isNullOrEmpty() && context != null) { |                                 binding!!.imageView.visibility = View.VISIBLE | ||||||
|                                         binding.imageView.visibility = View.VISIBLE |  | ||||||
|                                 try { |                                 try { | ||||||
|                                     Glide |                                     Glide | ||||||
|                                         .with(requireContext()) |                                         .with(requireContext()) | ||||||
|                                         .asBitmap() |                                         .asBitmap() | ||||||
|                                         .load( |                                         .load( | ||||||
|                                                     response.body()!!.lead_image_url.orEmpty() |                                             response.data!!.lead_image_url.orEmpty() | ||||||
|                                         ) |                                         ) | ||||||
|                                         .apply(RequestOptions.fitCenterTransform()) |                                         .apply(RequestOptions.fitCenterTransform()) | ||||||
|                                                 .into(binding.imageView) |                                         .into(binding!!.imageView) | ||||||
|                                 } catch (e: Exception) { |                                 } catch (e: Exception) { | ||||||
|  |                                     e.sendSilentlyWithAcraWithName("getContentFromMercury > glide lead image") | ||||||
|  |                                 } | ||||||
|  |                             } catch (e: Exception) { | ||||||
|  |                                 e.sendSilentlyWithAcraWithName("getContentFromMercury > outside glide lead image") | ||||||
|                             } |                             } | ||||||
|                         } else { |                         } else { | ||||||
|                                         binding.imageView.visibility = View.GONE |                             binding!!.imageView.visibility = View.GONE | ||||||
|                                     } |  | ||||||
|                                 } catch (e: Exception) { |  | ||||||
|                                     if (context != null) { |  | ||||||
|                                     } |  | ||||||
|                         } |                         } | ||||||
|  |  | ||||||
|                         try { |                         try { | ||||||
|                                     binding.nestedScrollView.scrollTo(0, 0) |                             binding!!.nestedScrollView.scrollTo(0, 0) | ||||||
|  |                             binding!!.progressBar.visibility = View.GONE | ||||||
|                                     binding.progressBar.visibility = View.GONE |  | ||||||
|                         } catch (e: Exception) { |                         } catch (e: Exception) { | ||||||
|                                     if (context != null) { |                             e.sendSilentlyWithAcraWithName("getContentFromMercury > scrollview") | ||||||
|                                     } |  | ||||||
|                         } |                         } | ||||||
|                     } else { |                     } else { | ||||||
|                                 try { |                         openInBrowserAfterFailing() | ||||||
|  |                     } | ||||||
|  |                 } catch (e: SocketTimeoutException) { | ||||||
|                     openInBrowserAfterFailing() |                     openInBrowserAfterFailing() | ||||||
|                 } catch (e: Exception) { |                 } catch (e: Exception) { | ||||||
|                                     if (context != null) { |                     e.sendSilentlyWithAcraWithName("getContentFromMercury > whole thing") | ||||||
|  |                     openInBrowserAfterFailing() | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|                         } catch (e: Exception) { |  | ||||||
|                             if (context != null) { |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     override fun onFailure( |  | ||||||
|                         call: Call<ParsedContent>, |  | ||||||
|                         t: Throwable |  | ||||||
|                     ) = openInBrowserAfterFailing() |  | ||||||
|                 } |  | ||||||
|             ) |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun htmlToWebview() { |     private fun htmlToWebview() { | ||||||
| @@ -340,8 +317,8 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|         val a: TypedArray = requireContext().obtainStyledAttributes(resId, attrs) |         val a: TypedArray = requireContext().obtainStyledAttributes(resId, attrs) | ||||||
|  |  | ||||||
|  |  | ||||||
|         binding.webcontent.settings.standardFontFamily = a.getString(0) |         binding!!.webcontent.settings.standardFontFamily = a.getString(0) | ||||||
|         binding.webcontent.visibility = View.VISIBLE |         binding!!.webcontent.visibility = View.VISIBLE | ||||||
|  |  | ||||||
|         val colorOnSurface = TypedValue() |         val colorOnSurface = TypedValue() | ||||||
|         requireContext().theme.resolveAttribute(R.attr.colorOnSurface, colorOnSurface, true) |         requireContext().theme.resolveAttribute(R.attr.colorOnSurface, colorOnSurface, true) | ||||||
| @@ -349,37 +326,45 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|         val colorSurface = TypedValue() |         val colorSurface = TypedValue() | ||||||
|         requireContext().theme.resolveAttribute(R.attr.colorSurface, colorSurface, true) |         requireContext().theme.resolveAttribute(R.attr.colorSurface, colorSurface, true) | ||||||
|  |  | ||||||
|         binding.webcontent.settings.useWideViewPort = true |         binding!!.webcontent.settings.useWideViewPort = true | ||||||
|         binding.webcontent.settings.loadWithOverviewMode = true |         binding!!.webcontent.settings.loadWithOverviewMode = true | ||||||
|         binding.webcontent.settings.javaScriptEnabled = false |         binding!!.webcontent.settings.javaScriptEnabled = false | ||||||
|  |  | ||||||
|         binding.webcontent.webViewClient = object : WebViewClient() { |         binding!!.webcontent.webViewClient = object : WebViewClient() { | ||||||
|  |             @Deprecated("Deprecated in Java") | ||||||
|             override fun shouldOverrideUrlLoading(view: WebView?, url : String): Boolean { |             override fun shouldOverrideUrlLoading(view: WebView?, url : String): Boolean { | ||||||
|                 if (binding.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { |                 if (binding!!.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||||
|                     requireContext().startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) |                     requireContext().startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) | ||||||
|                 } |                 } | ||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             override fun shouldInterceptRequest(view: WebView?, url: String): WebResourceResponse? { |             @Deprecated("Deprecated in Java") | ||||||
|  |             override fun shouldInterceptRequest(view: WebView, url: String): WebResourceResponse? { | ||||||
|                 val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) |                 val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||||
|                 if (url.lowercase(Locale.US).contains(".jpg") || url.lowercase(Locale.US).contains(".jpeg")) { |                 if (url.lowercase(Locale.US).contains(".jpg") || url.lowercase(Locale.US).contains(".jpeg")) { | ||||||
|                     try { |                     try { | ||||||
|                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() |                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||||
|                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.JPEG)) |                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.JPEG)) | ||||||
|                     }catch ( e : ExecutionException) {} |                     } catch ( e : ExecutionException) { | ||||||
|  |                         e.sendSilentlyWithAcraWithName("shouldInterceptRequest > jpeg > $url") | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|                 else if (url.lowercase(Locale.US).contains(".png")) { |                 else if (url.lowercase(Locale.US).contains(".png")) { | ||||||
|                     try { |                     try { | ||||||
|                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() |                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||||
|                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.PNG)) |                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.PNG)) | ||||||
|                     }catch ( e : ExecutionException) {} |                     } catch ( e : ExecutionException) { | ||||||
|  |                         e.sendSilentlyWithAcraWithName("shouldInterceptRequest > png > $url") | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|                 else if (url.lowercase(Locale.US).contains(".webp")) { |                 else if (url.lowercase(Locale.US).contains(".webp")) { | ||||||
|                     try { |                     try { | ||||||
|                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() |                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||||
|                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.WEBP)) |                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.WEBP)) | ||||||
|                     }catch ( e : ExecutionException) {} |                     } catch ( e : ExecutionException) { | ||||||
|  |                         e.sendSilentlyWithAcraWithName("shouldInterceptRequest > webp > $url") | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 return super.shouldInterceptRequest(view, url) |                 return super.shouldInterceptRequest(view, url) | ||||||
| @@ -387,14 +372,14 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         val gestureDetector = GestureDetector(activity, object : GestureDetector.SimpleOnGestureListener() { |         val gestureDetector = GestureDetector(activity, object : GestureDetector.SimpleOnGestureListener() { | ||||||
|             override fun onSingleTapUp(e: MotionEvent?): Boolean { |             override fun onSingleTapUp(e: MotionEvent): Boolean { | ||||||
|                 return performClick() |                 return performClick() | ||||||
|             } |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|         binding.webcontent.setOnTouchListener { _, event -> gestureDetector.onTouchEvent(event)} |         binding!!.webcontent.setOnTouchListener { _, event -> gestureDetector.onTouchEvent(event)} | ||||||
|  |  | ||||||
|         binding.webcontent.settings.layoutAlgorithm = |         binding!!.webcontent.settings.layoutAlgorithm = | ||||||
|                 WebSettings.LayoutAlgorithm.TEXT_AUTOSIZING |                 WebSettings.LayoutAlgorithm.TEXT_AUTOSIZING | ||||||
|  |  | ||||||
|         var baseUrl: String? = null |         var baseUrl: String? = null | ||||||
| @@ -403,11 +388,13 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|             val itemUrl = URL(url) |             val itemUrl = URL(url) | ||||||
|             baseUrl = itemUrl.protocol + "://" + itemUrl.host |             baseUrl = itemUrl.protocol + "://" + itemUrl.host | ||||||
|         } catch (e: MalformedURLException) { |         } catch (e: MalformedURLException) { | ||||||
|  |             e.sendSilentlyWithAcraWithName("htmlToWebview > item url") | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         val fontName =  when (font) { |         val fontName =  when (font) { | ||||||
|             getString(R.string.open_sans_font_id) -> "Open Sans" |             getString(R.string.open_sans_font_id) -> "Open Sans" | ||||||
|             getString(R.string.roboto_font_id) -> "Roboto" |             getString(R.string.roboto_font_id) -> "Roboto" | ||||||
|  |             getString(R.string.source_code_pro_font_id) -> "Source Code Pro" | ||||||
|             else -> "" |             else -> "" | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -423,7 +410,7 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|             "" |             "" | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         binding.webcontent.loadDataWithBaseURL( |         binding!!.webcontent.loadDataWithBaseURL( | ||||||
|             baseUrl, |             baseUrl, | ||||||
|             """<html> |             """<html> | ||||||
|                 |<head> |                 |<head> | ||||||
| @@ -476,17 +463,17 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun scrollDown() { |     fun scrollDown() { | ||||||
|         val height = binding.nestedScrollView.measuredHeight |         val height = binding!!.nestedScrollView.measuredHeight | ||||||
|         binding.nestedScrollView.smoothScrollBy(0, height/2) |         binding!!.nestedScrollView.smoothScrollBy(0, height/2) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun scrollUp() { |     fun scrollUp() { | ||||||
|         val height = binding.nestedScrollView.measuredHeight |         val height = binding!!.nestedScrollView.measuredHeight | ||||||
|         binding.nestedScrollView.smoothScrollBy(0, -height/2) |         binding!!.nestedScrollView.smoothScrollBy(0, -height/2) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun openInBrowserAfterFailing() { |     private fun openInBrowserAfterFailing() { | ||||||
|         binding.progressBar.visibility = View.GONE |         binding!!.progressBar.visibility = View.GONE | ||||||
|         requireActivity().openInBrowserAsNewTask(this@ArticleFragment.item) |         requireActivity().openInBrowserAsNewTask(this@ArticleFragment.item) | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -505,10 +492,10 @@ class ArticleFragment : Fragment(), DIAware { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun performClick(): Boolean { |     fun performClick(): Boolean { | ||||||
|         if (binding.webcontent.hitTestResult.type == WebView.HitTestResult.IMAGE_TYPE || |         if (binding!!.webcontent.hitTestResult.type == WebView.HitTestResult.IMAGE_TYPE || | ||||||
|                 binding.webcontent.hitTestResult.type == WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { |                 binding!!.webcontent.hitTestResult.type == WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||||
|  |  | ||||||
|             val position : Int = allImages.indexOf(binding.webcontent.hitTestResult.extra) |             val position : Int = allImages.indexOf(binding!!.webcontent.hitTestResult.extra) | ||||||
|  |  | ||||||
|             val intent = Intent(activity, ImageActivity::class.java) |             val intent = Intent(activity, ImageActivity::class.java) | ||||||
|             intent.putExtra("allImages", allImages) |             intent.putExtra("allImages", allImages) | ||||||
|   | |||||||
| @@ -0,0 +1,181 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.android.fragments | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import android.graphics.Color | ||||||
|  | import android.graphics.drawable.Drawable | ||||||
|  | import android.graphics.drawable.GradientDrawable | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.LayoutInflater | ||||||
|  | import android.view.View | ||||||
|  | import android.view.View.GONE | ||||||
|  | import android.view.View.VISIBLE | ||||||
|  | import android.view.ViewGroup | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.HomeActivity | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.R | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||||
|  | import bou.amine.apps.readerforselfossv2.repository.Repository | ||||||
|  | import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||||
|  | import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||||
|  | import com.bumptech.glide.Glide | ||||||
|  | import com.bumptech.glide.load.DataSource | ||||||
|  | import com.bumptech.glide.load.engine.GlideException | ||||||
|  | import com.bumptech.glide.request.RequestListener | ||||||
|  | import com.bumptech.glide.request.target.Target | ||||||
|  | import com.google.android.material.bottomsheet.BottomSheetDialogFragment | ||||||
|  | import com.google.android.material.chip.Chip | ||||||
|  | import kotlinx.coroutines.CoroutineScope | ||||||
|  | import kotlinx.coroutines.Dispatchers | ||||||
|  | import kotlinx.coroutines.launch | ||||||
|  | import org.kodein.di.DI | ||||||
|  | import org.kodein.di.DIAware | ||||||
|  | import org.kodein.di.android.x.closestDI | ||||||
|  | import org.kodein.di.instance | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FilterSheetFragment : BottomSheetDialogFragment(), DIAware { | ||||||
|  |  | ||||||
|  |     override val di: DI by closestDI() | ||||||
|  |     private val repository: Repository by instance() | ||||||
|  |  | ||||||
|  |     private var selectedChip: Chip? = null | ||||||
|  |  | ||||||
|  |     override fun onCreateView( | ||||||
|  |         inflater: LayoutInflater, | ||||||
|  |         container: ViewGroup?, | ||||||
|  |         savedInstanceState: Bundle? | ||||||
|  |     ): View { | ||||||
|  |         val binding = | ||||||
|  |             bou.amine.apps.readerforselfossv2.android.databinding.FilterFragmentBinding.inflate( | ||||||
|  |                 inflater, | ||||||
|  |                 container, | ||||||
|  |                 false | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |         val context: Context? = context | ||||||
|  |  | ||||||
|  |         val tagGroup = binding.tagsGroup | ||||||
|  |         val sourceGroup = binding.sourcesGroup | ||||||
|  |  | ||||||
|  |         if (context == null) { | ||||||
|  |             dismiss() | ||||||
|  |             Exception("FilterSheetFragment context is null").sendSilentlyWithAcraWithName("FilterSheetFragment > onCreateView") | ||||||
|  |         } else { | ||||||
|  |             CoroutineScope(Dispatchers.Main).launch { | ||||||
|  |                 val tags = repository.getTags() | ||||||
|  |  | ||||||
|  |                 tags.forEach { tag -> | ||||||
|  |                     val c = Chip(context) | ||||||
|  |                     c.text = tag.tag | ||||||
|  |  | ||||||
|  |                     val gd = GradientDrawable() | ||||||
|  |                     val gdColor = try { | ||||||
|  |                         Color.parseColor(tag.color) | ||||||
|  |                     } catch (e: IllegalArgumentException) { | ||||||
|  |                         e.sendSilentlyWithAcraWithName("color issue " + tag.color) | ||||||
|  |                         resources.getColor(R.color.colorPrimary) | ||||||
|  |                     } | ||||||
|  |                     gd.setColor(gdColor) | ||||||
|  |                     gd.shape = GradientDrawable.RECTANGLE | ||||||
|  |                     gd.setSize(30, 30) | ||||||
|  |                     gd.cornerRadius = 30F | ||||||
|  |                     c.chipIcon = gd | ||||||
|  |  | ||||||
|  |                     c.setOnCloseIconClickListener { | ||||||
|  |                         (it as Chip).isCloseIconVisible = false | ||||||
|  |                         selectedChip = null | ||||||
|  |                         repository.setTagFilter(null) | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     c.setOnClickListener { | ||||||
|  |                         if (selectedChip != null) { | ||||||
|  |                             selectedChip!!.isCloseIconVisible = false | ||||||
|  |                         } | ||||||
|  |                         (it as Chip).isCloseIconVisible = true | ||||||
|  |                         selectedChip = it | ||||||
|  |                         repository.setTagFilter(tag) | ||||||
|  |  | ||||||
|  |                         repository.setSourceFilter(null) | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     if (repository.tagFilter.value?.equals(tag) == true) { | ||||||
|  |                         c.isCloseIconVisible = true | ||||||
|  |                         selectedChip = c | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     tagGroup.addView(c) | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 repository.getSources().forEach { source -> | ||||||
|  |                     val c = Chip(context) | ||||||
|  |  | ||||||
|  |                     Glide.with(context) | ||||||
|  |                         .load(source.getIcon(repository.baseUrl)) | ||||||
|  |                         .listener(object : RequestListener<Drawable?> { | ||||||
|  |                             override fun onLoadFailed( | ||||||
|  |                                 e: GlideException?, | ||||||
|  |                                 model: Any?, | ||||||
|  |                                 target: Target<Drawable?>?, | ||||||
|  |                                 isFirstResource: Boolean | ||||||
|  |                             ): Boolean { | ||||||
|  |                                 return false | ||||||
|  |                             } | ||||||
|  |  | ||||||
|  |                             override fun onResourceReady( | ||||||
|  |                                 resource: Drawable?, | ||||||
|  |                                 model: Any?, | ||||||
|  |                                 target: Target<Drawable?>?, | ||||||
|  |                                 dataSource: DataSource?, | ||||||
|  |                                 isFirstResource: Boolean | ||||||
|  |                             ): Boolean { | ||||||
|  |                                 c.chipIcon = resource | ||||||
|  |                                 return false | ||||||
|  |                             } | ||||||
|  |                         }).preload() | ||||||
|  |  | ||||||
|  |                     c.text = source.title.getHtmlDecoded() | ||||||
|  |  | ||||||
|  |                     c.setOnCloseIconClickListener { | ||||||
|  |                         (it as Chip).isCloseIconVisible = false | ||||||
|  |                         selectedChip = null | ||||||
|  |                         repository.setSourceFilter(null) | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     c.setOnClickListener { | ||||||
|  |                         if (selectedChip != null) { | ||||||
|  |                             selectedChip!!.isCloseIconVisible = false | ||||||
|  |                         } | ||||||
|  |                         (it as Chip).isCloseIconVisible = true | ||||||
|  |                         selectedChip = it | ||||||
|  |                         repository.setSourceFilter(source) | ||||||
|  |  | ||||||
|  |                         repository.setTagFilter(null) | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |                     if (repository.sourceFilter.value?.equals(source) == true) { | ||||||
|  |                         c.isCloseIconVisible = true | ||||||
|  |                         selectedChip = c | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     sourceGroup.addView(c) | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 binding.progressBar2.visibility = GONE | ||||||
|  |                 binding.filterView.visibility = VISIBLE | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         binding.floatingActionButton2.setOnClickListener { | ||||||
|  |             (activity as HomeActivity).getElementsAccordingToTab() | ||||||
|  |             (activity as HomeActivity).fetchOnEmptyList() | ||||||
|  |             dismiss() | ||||||
|  |         } | ||||||
|  |         return binding.root | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     companion object { | ||||||
|  |         const val TAG = "FilterModalBottomSheet" | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -28,7 +28,7 @@ class ImageFragment : Fragment() { | |||||||
|         val view = binding?.root |         val view = binding?.root | ||||||
|  |  | ||||||
|         binding!!.photoView.visibility = View.VISIBLE |         binding!!.photoView.visibility = View.VISIBLE | ||||||
|         Glide.with(activity) |         Glide.with(requireActivity()) | ||||||
|                 .asBitmap() |                 .asBitmap() | ||||||
|                 .apply(glideOptions) |                 .apply(glideOptions) | ||||||
|                 .load(imageUrl) |                 .load(imageUrl) | ||||||
|   | |||||||
| @@ -2,11 +2,13 @@ package bou.amine.apps.readerforselfossv2.android.model | |||||||
|  |  | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.webkit.URLUtil | import android.webkit.URLUtil | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import bou.amine.apps.readerforselfossv2.utils.getImages | import bou.amine.apps.readerforselfossv2.utils.getImages | ||||||
| import com.bumptech.glide.Glide | import com.bumptech.glide.Glide | ||||||
| import com.bumptech.glide.load.engine.DiskCacheStrategy | import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||||
| import com.bumptech.glide.request.RequestOptions | import com.bumptech.glide.request.RequestOptions | ||||||
|  | import org.acra.ktx.sendSilentlyWithAcra | ||||||
|  |  | ||||||
| fun SelfossModel.Item.preloadImages(context: Context) : Boolean { | fun SelfossModel.Item.preloadImages(context: Context) : Boolean { | ||||||
|     val imageUrls = this.getImages() |     val imageUrls = this.getImages() | ||||||
| @@ -23,6 +25,7 @@ fun SelfossModel.Item.preloadImages(context: Context) : Boolean { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } catch (e : Error) { |     } catch (e : Error) { | ||||||
|  |         e.sendSilentlyWithAcraWithName("preloadImages") | ||||||
|         return false |         return false | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -35,7 +38,7 @@ fun String.toTextDrawableString(): String { | |||||||
|         try { |         try { | ||||||
|             textDrawable.append(s[0]) |             textDrawable.append(s[0]) | ||||||
|         } catch (e: StringIndexOutOfBoundsException) { |         } catch (e: StringIndexOutOfBoundsException) { | ||||||
|             // We do nothing |             e.sendSilentlyWithAcraWithName("toTextDrawableString") | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     return textDrawable.toString() |     return textDrawable.toString() | ||||||
|   | |||||||
| @@ -3,7 +3,6 @@ package bou.amine.apps.readerforselfossv2.android.model | |||||||
| import android.os.Parcel | import android.os.Parcel | ||||||
| import android.os.Parcelable | import android.os.Parcelable | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
| import com.google.gson.annotations.SerializedName |  | ||||||
|  |  | ||||||
| fun SelfossModel.Item.toParcelable() : ParecelableItem = | fun SelfossModel.Item.toParcelable() : ParecelableItem = | ||||||
|     ParecelableItem( |     ParecelableItem( | ||||||
| @@ -17,7 +16,8 @@ fun SelfossModel.Item.toParcelable() : ParecelableItem = | |||||||
|         this.icon, |         this.icon, | ||||||
|         this.link, |         this.link, | ||||||
|         this.sourcetitle, |         this.sourcetitle, | ||||||
|         this.tags.joinToString(",") |         this.tags.joinToString(","), | ||||||
|  |         this.author | ||||||
|     ) |     ) | ||||||
| fun ParecelableItem.toModel() : SelfossModel.Item = | fun ParecelableItem.toModel() : SelfossModel.Item = | ||||||
|     SelfossModel.Item( |     SelfossModel.Item( | ||||||
| @@ -31,20 +31,22 @@ fun ParecelableItem.toModel() : SelfossModel.Item = | |||||||
|         this.icon, |         this.icon, | ||||||
|         this.link, |         this.link, | ||||||
|         this.sourcetitle, |         this.sourcetitle, | ||||||
|         this.tags.split(",") |         this.tags.split(","), | ||||||
|  |         this.author | ||||||
|     ) |     ) | ||||||
| data class ParecelableItem( | data class ParecelableItem( | ||||||
|     @SerializedName("id") val id: Int, |     val id: Int, | ||||||
|     @SerializedName("datetime") val datetime: String, |     val datetime: String, | ||||||
|     @SerializedName("title") val title: String, |     val title: String, | ||||||
|     @SerializedName("content") val content: String, |     val content: String, | ||||||
|     @SerializedName("unread") var unread: Boolean, |     var unread: Boolean, | ||||||
|     @SerializedName("starred") var starred: Boolean, |     var starred: Boolean, | ||||||
|     @SerializedName("thumbnail") val thumbnail: String?, |     val thumbnail: String?, | ||||||
|     @SerializedName("icon") val icon: String?, |     val icon: String?, | ||||||
|     @SerializedName("link") val link: String, |     val link: String, | ||||||
|     @SerializedName("sourcetitle") val sourcetitle: String, |     val sourcetitle: String, | ||||||
|     @SerializedName("tags") val tags: String |     val tags: String, | ||||||
|  |     val author: String | ||||||
| ) : Parcelable { | ) : Parcelable { | ||||||
|  |  | ||||||
|     companion object { |     companion object { | ||||||
| @@ -66,7 +68,8 @@ data class ParecelableItem( | |||||||
|         icon = source.readString(), |         icon = source.readString(), | ||||||
|         link = source.readString().orEmpty(), |         link = source.readString().orEmpty(), | ||||||
|         sourcetitle = source.readString().orEmpty(), |         sourcetitle = source.readString().orEmpty(), | ||||||
|         tags = source.readString().orEmpty() |         tags = source.readString().orEmpty(), | ||||||
|  |         author = source.readString().orEmpty() | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     override fun describeContents() = 0 |     override fun describeContents() = 0 | ||||||
| @@ -83,5 +86,6 @@ data class ParecelableItem( | |||||||
|         dest.writeString(link) |         dest.writeString(link) | ||||||
|         dest.writeString(sourcetitle) |         dest.writeString(sourcetitle) | ||||||
|         dest.writeString(tags) |         dest.writeString(tags) | ||||||
|  |         dest.writeString(author) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -16,12 +16,18 @@ import androidx.preference.Preference | |||||||
| import androidx.preference.PreferenceFragmentCompat | import androidx.preference.PreferenceFragmentCompat | ||||||
| import bou.amine.apps.readerforselfossv2.android.R | import bou.amine.apps.readerforselfossv2.android.R | ||||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySettingsBinding | import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySettingsBinding | ||||||
|  | import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
|  | import com.mikepenz.aboutlibraries.LibsBuilder | ||||||
|  | import org.kodein.di.DIAware | ||||||
|  | import org.kodein.di.android.closestDI | ||||||
|  | import org.kodein.di.instance | ||||||
|  |  | ||||||
| private const val TITLE_TAG = "settingsActivityTitle" | private const val TITLE_TAG = "settingsActivityTitle" | ||||||
|  |  | ||||||
| class SettingsActivity : AppCompatActivity(), | class SettingsActivity : AppCompatActivity(), | ||||||
|         PreferenceFragmentCompat.OnPreferenceStartFragmentCallback { |         PreferenceFragmentCompat.OnPreferenceStartFragmentCallback, DIAware { | ||||||
|  |     override val di by closestDI() | ||||||
|  |  | ||||||
|     override fun onCreate(savedInstanceState: Bundle?) { |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|         super.onCreate(savedInstanceState) |         super.onCreate(savedInstanceState) | ||||||
| @@ -91,6 +97,21 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|     class MainPreferenceFragment : PreferenceFragmentCompat() { |     class MainPreferenceFragment : PreferenceFragmentCompat() { | ||||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { |         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||||
|             setPreferencesFromResource(R.xml.pref_main, rootKey) |             setPreferencesFromResource(R.xml.pref_main, rootKey) | ||||||
|  |  | ||||||
|  |             preferenceManager.findPreference<Preference>("currentMode")?.onPreferenceChangeListener = Preference.OnPreferenceChangeListener { _, newValue -> | ||||||
|  |                 AppCompatDelegate.setDefaultNightMode(newValue.toString().toInt()) // ListPreference Only takes string-arrays ¯\_(ツ)_/¯ | ||||||
|  |                 true | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             preferenceManager.findPreference<Preference>("action_about")?.onPreferenceClickListener = Preference.OnPreferenceClickListener { _ -> | ||||||
|  |                 context?.let { | ||||||
|  |                     LibsBuilder() | ||||||
|  |                         .withAboutIconShown(true) | ||||||
|  |                         .withAboutVersionShown(true) | ||||||
|  |                         .start(it) | ||||||
|  |                 } | ||||||
|  |                 true | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -107,6 +128,7 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|                                 val input: Int = (dest.toString() + source.toString()).toInt() |                                 val input: Int = (dest.toString() + source.toString()).toInt() | ||||||
|                                 if (input in 1..200) return@InputFilter null |                                 if (input in 1..200) return@InputFilter null | ||||||
|                             } catch (nfe: NumberFormatException) { |                             } catch (nfe: NumberFormatException) { | ||||||
|  |                                 nfe.sendSilentlyWithAcraWithName("GeneralPreferenceFragment") | ||||||
|                                 Toast.makeText(activity, R.string.items_number_should_be_number, Toast.LENGTH_LONG).show() |                                 Toast.makeText(activity, R.string.items_number_should_be_number, Toast.LENGTH_LONG).show() | ||||||
|                             } |                             } | ||||||
|                             "" |                             "" | ||||||
| @@ -130,6 +152,7 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|                         try { |                         try { | ||||||
|                             editText.textSize = editable.toString().toInt().toFloat() |                             editText.textSize = editable.toString().toInt().toFloat() | ||||||
|                         } catch (e: NumberFormatException) { |                         } catch (e: NumberFormatException) { | ||||||
|  |                             e.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > afterTextChanged") | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 } } |                 } } | ||||||
| @@ -139,6 +162,7 @@ class SettingsActivity : AppCompatActivity(), | |||||||
|                                 val input = (dest.toString() + source.toString()).toInt() |                                 val input = (dest.toString() + source.toString()).toInt() | ||||||
|                                 if (input > 0) return@InputFilter null |                                 if (input > 0) return@InputFilter null | ||||||
|                             } catch (nfe: NumberFormatException) { |                             } catch (nfe: NumberFormatException) { | ||||||
|  |                                 nfe.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > filters") | ||||||
|                             } |                             } | ||||||
|                             "" |                             "" | ||||||
|                         } |                         } | ||||||
|   | |||||||
| @@ -1,13 +1,9 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.android.utils | package bou.amine.apps.readerforselfossv2.android.utils | ||||||
|  |  | ||||||
| import android.app.Activity | import android.app.Activity | ||||||
| import android.app.PendingIntent |  | ||||||
| import android.content.ActivityNotFoundException |  | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.graphics.BitmapFactory |  | ||||||
| import android.net.Uri | import android.net.Uri | ||||||
| import android.os.Build |  | ||||||
| import android.text.Spannable | import android.text.Spannable | ||||||
| import android.text.style.ClickableSpan | import android.text.style.ClickableSpan | ||||||
| import android.util.Patterns | import android.util.Patterns | ||||||
| @@ -22,7 +18,6 @@ import bou.amine.apps.readerforselfossv2.utils.toStringUriWithHttp | |||||||
| import okhttp3.HttpUrl.Companion.toHttpUrlOrNull | import okhttp3.HttpUrl.Companion.toHttpUrlOrNull | ||||||
|  |  | ||||||
| fun Context.openItemUrl( | fun Context.openItemUrl( | ||||||
|     allItems: ArrayList<SelfossModel.Item>, |  | ||||||
|     currentItem: Int, |     currentItem: Int, | ||||||
|     linkDecoded: String, |     linkDecoded: String, | ||||||
|     articleViewer: Boolean, |     articleViewer: Boolean, | ||||||
| @@ -37,7 +32,6 @@ fun Context.openItemUrl( | |||||||
|         ).show() |         ).show() | ||||||
|     } else { |     } else { | ||||||
|         if (articleViewer) { |         if (articleViewer) { | ||||||
|             ReaderActivity.allItems = allItems |  | ||||||
|             val intent = Intent(this, ReaderActivity::class.java) |             val intent = Intent(this, ReaderActivity::class.java) | ||||||
|             intent.putExtra("currentItem", currentItem) |             intent.putExtra("currentItem", currentItem) | ||||||
|             app.startActivity(intent) |             app.startActivity(intent) | ||||||
|   | |||||||
| @@ -1,14 +0,0 @@ | |||||||
| /* From https://github.com/mikepenz/MaterialDrawer/blob/develop/app/src/main/java/com/mikepenz/materialdrawer/app/drawerItems/CustomBaseViewHolder.java */ |  | ||||||
| package bou.amine.apps.readerforselfossv2.android.utils.drawer |  | ||||||
|  |  | ||||||
| import android.view.View |  | ||||||
| import android.widget.ImageView |  | ||||||
| import android.widget.TextView |  | ||||||
| import androidx.recyclerview.widget.RecyclerView |  | ||||||
| import bou.amine.apps.readerforselfossv2.android.R |  | ||||||
|  |  | ||||||
| open class CustomBaseViewHolder(var view: View) : RecyclerView.ViewHolder(view) { |  | ||||||
|     var icon: ImageView = view.findViewById(R.id.material_drawer_icon) |  | ||||||
|     var name: TextView = view.findViewById(R.id.material_drawer_name) |  | ||||||
|     var description: TextView = view.findViewById(R.id.material_drawer_description) |  | ||||||
| } |  | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <!-- Copyright 2015 Google Inc. All Rights Reserved. |  | ||||||
|      Licensed under the Apache License, Version 2.0 (the "License"); |  | ||||||
|      you may not use this file except in compliance with the License. |  | ||||||
|      You may obtain a copy of the License at |  | ||||||
|          http://www.apache.org/licenses/LICENSE-2.0 |  | ||||||
|      Unless required by applicable law or agreed to in writing, software |  | ||||||
|      distributed under the License is distributed on an "AS IS" BASIS, |  | ||||||
|      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |  | ||||||
|      See the License for the specific language governing permissions and |  | ||||||
|      limitations under the License. |  | ||||||
| --> |  | ||||||
| <set xmlns:android="http://schemas.android.com/apk/res/android"> |  | ||||||
|     <translate android:fromXDelta="100%p" android:toXDelta="0" |  | ||||||
|                android:duration="@android:integer/config_mediumAnimTime"/> |  | ||||||
| </set> |  | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <!-- Copyright 2015 Google Inc. All Rights Reserved. |  | ||||||
|      Licensed under the Apache License, Version 2.0 (the "License"); |  | ||||||
|      you may not use this file except in compliance with the License. |  | ||||||
|      You may obtain a copy of the License at |  | ||||||
|          http://www.apache.org/licenses/LICENSE-2.0 |  | ||||||
|      Unless required by applicable law or agreed to in writing, software |  | ||||||
|      distributed under the License is distributed on an "AS IS" BASIS, |  | ||||||
|      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |  | ||||||
|      See the License for the specific language governing permissions and |  | ||||||
|      limitations under the License. |  | ||||||
| --> |  | ||||||
| <set xmlns:android="http://schemas.android.com/apk/res/android"> |  | ||||||
|     <translate android:fromXDelta="0" android:toXDelta="-100%p" |  | ||||||
|                android:duration="@android:integer/config_mediumAnimTime"/> |  | ||||||
| </set> |  | ||||||
							
								
								
									
										5
									
								
								androidApp/src/main/res/drawable/checkerboard.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								androidApp/src/main/res/drawable/checkerboard.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | <bitmap | ||||||
|  |     xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:dither="true" | ||||||
|  |     android:src="@drawable/checktile" | ||||||
|  |     android:tileMode="repeat"/> | ||||||
							
								
								
									
										
											BIN
										
									
								
								androidApp/src/main/res/drawable/checktile.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								androidApp/src/main/res/drawable/checktile.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 235 B | 
| @@ -0,0 +1,5 @@ | |||||||
|  | <vector android:height="24dp" android:tint="#000000" | ||||||
|  |     android:viewportHeight="24" android:viewportWidth="24" | ||||||
|  |     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|  |     <path android:fillColor="@android:color/white" android:pathData="M20,8h-2.81c-0.45,-0.78 -1.07,-1.45 -1.82,-1.96L17,4.41 15.59,3l-2.17,2.17C12.96,5.06 12.49,5 12,5c-0.49,0 -0.96,0.06 -1.41,0.17L8.41,3 7,4.41l1.62,1.63C7.88,6.55 7.26,7.22 6.81,8L4,8v2h2.09c-0.05,0.33 -0.09,0.66 -0.09,1v1L4,12v2h2v1c0,0.34 0.04,0.67 0.09,1L4,16v2h2.81c1.04,1.79 2.97,3 5.19,3s4.15,-1.21 5.19,-3L20,18v-2h-2.09c0.05,-0.33 0.09,-0.66 0.09,-1v-1h2v-2h-2v-1c0,-0.34 -0.04,-0.67 -0.09,-1L20,10L20,8zM14,16h-4v-2h4v2zM14,12h-4v-2h4v2z"/> | ||||||
|  | </vector> | ||||||
| @@ -0,0 +1,5 @@ | |||||||
|  | <vector android:height="24dp" android:tint="#FFFFFF" | ||||||
|  |     android:viewportHeight="24" android:viewportWidth="24" | ||||||
|  |     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|  |     <path android:fillColor="@android:color/white" android:pathData="M4.25,5.61C6.27,8.2 10,13 10,13v6c0,0.55 0.45,1 1,1h2c0.55,0 1,-0.45 1,-1v-6c0,0 3.72,-4.8 5.74,-7.39C20.25,4.95 19.78,4 18.95,4H5.04C4.21,4 3.74,4.95 4.25,5.61z"/> | ||||||
|  | </vector> | ||||||
| @@ -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="#FF000000" |  | ||||||
|         android:pathData="M20,8h-2.81c-0.45,-0.78 -1.07,-1.45 -1.82,-1.96L17,4.41 15.59,3l-2.17,2.17C12.96,5.06 12.49,5 12,5c-0.49,0 -0.96,0.06 -1.41,0.17L8.41,3 7,4.41l1.62,1.63C7.88,6.55 7.26,7.22 6.81,8L4,8v2h2.09c-0.05,0.33 -0.09,0.66 -0.09,1v1L4,12v2h2v1c0,0.34 0.04,0.67 0.09,1L4,16v2h2.81c1.04,1.79 2.97,3 5.19,3s4.15,-1.21 5.19,-3L20,18v-2h-2.09c0.05,-0.33 0.09,-0.66 0.09,-1v-1h2v-2h-2v-1c0,-0.34 -0.04,-0.67 -0.09,-1L20,10L20,8zM14,16h-4v-2h4v2zM14,12h-4v-2h4v2z"/> |  | ||||||
| </vector> |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| <vector android:height="24dp" android:tint="#FFFFFF" |  | ||||||
|     android:viewportHeight="24.0" android:viewportWidth="24.0" |  | ||||||
|     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> |  | ||||||
|     <path android:fillColor="#FF000000" android:pathData="M19,6.41L17.59,5 12,10.59 6.41,5 5,6.41 10.59,12 5,17.59 6.41,19 12,13.41 17.59,19 19,17.59 13.41,12z"/> |  | ||||||
| </vector> |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| <vector android:height="24dp" android:tint="#FFFFFF" |  | ||||||
|     android:viewportHeight="24.0" android:viewportWidth="24.0" |  | ||||||
|     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> |  | ||||||
|     <path android:fillColor="#FF000000" android:pathData="M13,3c-4.97,0 -9,4.03 -9,9L1,12l3.89,3.89 0.07,0.14L9,12L6,12c0,-3.87 3.13,-7 7,-7s7,3.13 7,7 -3.13,7 -7,7c-1.93,0 -3.68,-0.79 -4.94,-2.06l-1.42,1.42C8.27,19.99 10.51,21 13,21c4.97,0 9,-4.03 9,-9s-4.03,-9 -9,-9zM12,8v5l4.28,2.54 0.72,-1.21 -3.5,-2.08L13.5,8L12,8z"/> |  | ||||||
| </vector> |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| <vector android:height="24dp" android:tint="#FFFFFF" |  | ||||||
|     android:viewportHeight="24.0" android:viewportWidth="24.0" |  | ||||||
|     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> |  | ||||||
|     <path android:fillColor="#FF000000" android:pathData="M17.65,6.35C16.2,4.9 14.21,4 12,4c-4.42,0 -7.99,3.58 -7.99,8s3.57,8 7.99,8c3.73,0 6.84,-2.55 7.73,-6h-2.08c-0.82,2.33 -3.04,4 -5.65,4 -3.31,0 -6,-2.69 -6,-6s2.69,-6 6,-6c1.66,0 3.14,0.69 4.22,1.78L13,11h7V4l-2.35,2.35z"/> |  | ||||||
| </vector> |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| <?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="Open Sans" |  | ||||||
|         app:fontProviderCerts="@array/com_google_android_gms_fonts_certs"> |  | ||||||
| </font-family> |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| <?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="Roboto" |  | ||||||
|         app:fontProviderCerts="@array/com_google_android_gms_fonts_certs"> |  | ||||||
| </font-family> |  | ||||||
| @@ -85,11 +85,4 @@ | |||||||
|             app:bnbActiveColor="@color/colorAccent" |             app:bnbActiveColor="@color/colorAccent" | ||||||
|             app:bnbBackgroundColor="?attr/bottomBarBackground" /> |             app:bnbBackgroundColor="?attr/bottomBarBackground" /> | ||||||
|     </androidx.coordinatorlayout.widget.CoordinatorLayout> |     </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||||
|  |  | ||||||
|     <com.mikepenz.materialdrawer.widget.MaterialDrawerSliderView |  | ||||||
|         android:id="@+id/mainDrawer" |  | ||||||
|         android:layout_width="wrap_content" |  | ||||||
|         android:layout_height="match_parent" |  | ||||||
|         android:layout_gravity="start" |  | ||||||
|         android:fitsSystemWindows="true" /> |  | ||||||
| </androidx.drawerlayout.widget.DrawerLayout> | </androidx.drawerlayout.widget.DrawerLayout> | ||||||
							
								
								
									
										89
									
								
								androidApp/src/main/res/layout/filter_fragment.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								androidApp/src/main/res/layout/filter_fragment.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,89 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <androidx.constraintlayout.widget.ConstraintLayout | ||||||
|  |     xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|  |     xmlns:tools="http://schemas.android.com/tools" | ||||||
|  |     android:layout_width="match_parent" | ||||||
|  |     android:layout_height="match_parent"> | ||||||
|  |  | ||||||
|  |     <ProgressBar | ||||||
|  |         android:id="@+id/progressBar2" | ||||||
|  |         style="?android:attr/progressBarStyle" | ||||||
|  |         android:layout_width="48dp" | ||||||
|  |         android:layout_height="48dp" | ||||||
|  |         app:layout_constraintBottom_toBottomOf="parent" | ||||||
|  |         app:layout_constraintEnd_toEndOf="parent" | ||||||
|  |         app:layout_constraintStart_toStartOf="parent" | ||||||
|  |         app:layout_constraintTop_toTopOf="parent" | ||||||
|  |         tools:visibility="gone" /> | ||||||
|  |  | ||||||
|  |     <androidx.constraintlayout.widget.ConstraintLayout | ||||||
|  |         android:id="@+id/filterView" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:visibility="gone" | ||||||
|  |         tools:visibility="visible"> | ||||||
|  |  | ||||||
|  |         <com.google.android.material.floatingactionbutton.FloatingActionButton | ||||||
|  |             android:id="@+id/floatingActionButton2" | ||||||
|  |             android:layout_width="wrap_content" | ||||||
|  |             android:layout_height="wrap_content" | ||||||
|  |             android:layout_marginEnd="16dp" | ||||||
|  |  | ||||||
|  |             android:layout_marginTop="8dp" | ||||||
|  |             android:clickable="true" | ||||||
|  |             app:backgroundTint="@color/colorAccent" | ||||||
|  |             app:layout_constraintEnd_toEndOf="parent" | ||||||
|  |             app:layout_constraintTop_toTopOf="parent" | ||||||
|  |             app:rippleColor="@color/colorAccentDark" | ||||||
|  |             app:srcCompat="@drawable/ic_menu_search_white_24dp" /> | ||||||
|  |  | ||||||
|  |         <TextView | ||||||
|  |             android:id="@+id/filterTagsTitle" | ||||||
|  |             style="@style/MaterialAlertDialog.MaterialComponents.Title.Text" | ||||||
|  |             android:layout_width="wrap_content" | ||||||
|  |             android:layout_height="wrap_content" | ||||||
|  |             android:layout_marginStart="24dp" | ||||||
|  |             android:layout_marginTop="16dp" | ||||||
|  |             android:text="@string/filter_item_tags" | ||||||
|  |             app:layout_constraintStart_toStartOf="parent" | ||||||
|  |             app:layout_constraintTop_toTopOf="parent" /> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         <com.google.android.material.chip.ChipGroup | ||||||
|  |             android:id="@+id/tagsGroup" | ||||||
|  |             android:layout_width="match_parent" | ||||||
|  |             android:layout_height="wrap_content" | ||||||
|  |             android:layout_marginStart="16dp" | ||||||
|  |             android:layout_marginTop="24dp" | ||||||
|  |             app:layout_constraintStart_toStartOf="parent" | ||||||
|  |             app:layout_constraintTop_toBottomOf="@+id/filterTagsTitle" | ||||||
|  |             app:singleSelection="true"> | ||||||
|  |  | ||||||
|  |         </com.google.android.material.chip.ChipGroup> | ||||||
|  |  | ||||||
|  |         <TextView | ||||||
|  |             android:id="@+id/filterSourcesTitle" | ||||||
|  |             style="@style/MaterialAlertDialog.MaterialComponents.Title.Text" | ||||||
|  |             android:layout_width="wrap_content" | ||||||
|  |             android:layout_height="wrap_content" | ||||||
|  |             android:layout_marginStart="24dp" | ||||||
|  |             android:layout_marginTop="24dp" | ||||||
|  |             android:text="@string/filter_item_sources" | ||||||
|  |             app:layout_constraintStart_toStartOf="parent" | ||||||
|  |             app:layout_constraintTop_toBottomOf="@+id/tagsGroup" /> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         <com.google.android.material.chip.ChipGroup | ||||||
|  |             android:id="@+id/sourcesGroup" | ||||||
|  |             android:layout_width="match_parent" | ||||||
|  |             android:layout_height="wrap_content" | ||||||
|  |             android:layout_marginStart="16dp" | ||||||
|  |             android:layout_marginTop="24dp" | ||||||
|  |             app:layout_constraintStart_toStartOf="parent" | ||||||
|  |             app:layout_constraintTop_toBottomOf="@+id/filterSourcesTitle"> | ||||||
|  |  | ||||||
|  |         </com.google.android.material.chip.ChipGroup> | ||||||
|  |  | ||||||
|  |     </androidx.constraintlayout.widget.ConstraintLayout> | ||||||
|  | </androidx.constraintlayout.widget.ConstraintLayout> | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android" | <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|     android:layout_width="match_parent" |     android:layout_width="match_parent" | ||||||
|     android:layout_height="match_parent"> |     android:layout_height="match_parent"> | ||||||
| @@ -9,8 +9,8 @@ | |||||||
|         android:layout_width="match_parent" |         android:layout_width="match_parent" | ||||||
|         android:layout_height="match_parent" |         android:layout_height="match_parent" | ||||||
|         android:layout_centerVertical="true" |         android:layout_centerVertical="true" | ||||||
|         android:layout_centerHorizontal="true" |         android:adjustViewBounds="true" | ||||||
|         android:background="@android:color/black" |         android:background="@drawable/checkerboard" | ||||||
|         app:srcCompat="@android:drawable/screen_background_dark" /> |         app:srcCompat="@android:drawable/screen_background_dark" /> | ||||||
|  |  | ||||||
| </androidx.coordinatorlayout.widget.CoordinatorLayout> | </RelativeLayout> | ||||||
| @@ -8,16 +8,27 @@ | |||||||
|         app:showAsAction="ifRoom|collapseActionView" |         app:showAsAction="ifRoom|collapseActionView" | ||||||
|         app:actionViewClass="androidx.appcompat.widget.SearchView" /> |         app:actionViewClass="androidx.appcompat.widget.SearchView" /> | ||||||
|  |  | ||||||
|     <item android:id="@+id/readAll" |     <item android:id="@+id/action_filter" | ||||||
|           android:icon="@drawable/ic_menu_done_all_white_24dp" |         android:title="@string/menu_home_filter" | ||||||
|           android:title="@string/readAll" |         android:icon="@drawable/ic_baseline_filter_alt_24" | ||||||
|         android:orderInCategory="1" |         android:orderInCategory="1" | ||||||
|         app:showAsAction="always" /> |         app:showAsAction="always" /> | ||||||
|  |  | ||||||
|  |     <item android:id="@+id/readAll" | ||||||
|  |           android:icon="@drawable/ic_menu_done_all_white_24dp" | ||||||
|  |           android:title="@string/readAll" | ||||||
|  |           android:orderInCategory="2" | ||||||
|  |           app:showAsAction="ifRoom"/> | ||||||
|  |  | ||||||
|  |     <item android:id="@+id/action_settings" | ||||||
|  |         android:title="@string/title_activity_settings" | ||||||
|  |         android:orderInCategory="98" | ||||||
|  |         app:showAsAction="never"/> | ||||||
|  |  | ||||||
|     <item |     <item | ||||||
|         android:id="@+id/refresh" |         android:id="@+id/refresh" | ||||||
|         android:icon="@drawable/ic_menu_refresh_white_24dp" |         app:showAsAction="never" | ||||||
|         android:orderInCategory="99" |         android:orderInCategory="101" | ||||||
|         android:title="@string/menu_home_refresh" /> |         android:title="@string/menu_home_refresh" /> | ||||||
|  |  | ||||||
|     <item android:id="@+id/action_disconnect" |     <item android:id="@+id/action_disconnect" | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"Autenticació (si és necessària)"</string> |     <string name="withLoginSwitch">"Autenticació (si és necessària)"</string> | ||||||
|     <string name="login_url_problem">"Pot ser que falti una \"/\" al final de l'url."</string> |     <string name="login_url_problem">"Pot ser que falti una \"/\" al final de l'url."</string> | ||||||
|     <string name="prompt_login">"Nom d'usuari"</string> |     <string name="prompt_login">"Nom d'usuari"</string> | ||||||
|     <string name="label_share">"Comparteix"</string> |  | ||||||
|     <string name="readAll">"Llegeix-ho tot"</string> |     <string name="readAll">"Llegeix-ho tot"</string> | ||||||
|     <string name="action_disconnect">"Desconnecta't"</string> |     <string name="action_disconnect">"Desconnecta't"</string> | ||||||
|     <string name="title_activity_settings">"Configuració"</string> |     <string name="title_activity_settings">"Configuració"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">L\'alçada de les targetes s\'ajustarà al seu contingut</string> |     <string name="card_height_on">L\'alçada de les targetes s\'ajustarà al seu contingut</string> | ||||||
|     <string name="card_height_off">L\'alçada de les targetes serà fixa</string> |     <string name="card_height_off">L\'alçada de les targetes serà fixa</string> | ||||||
|     <string name="source_code">Codi font</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="filter_item_tags">Etiquetes</string> | ||||||
|     <string name="drawer_item_filters">Filtres</string> |     <string name="filter_item_sources">Fonts</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="drawer_loading">S\'està carregant…</string> |  | ||||||
|     <string name="menu_home_search">Cerca</string> |     <string name="menu_home_search">Cerca</string> | ||||||
|     <string name="can_delete_source">No es pot suprimir la font</string> |     <string name="can_delete_source">No es pot suprimir la font</string> | ||||||
|     <string name="base_url_error">S\'ha produït un error en comunicar-se amb la instància de Selfoss. Si el problema persisteix, posa\'t en contacte amb mi.</string> |     <string name="base_url_error">S\'ha produït un error en comunicar-se amb la instància de Selfoss. Si el problema persisteix, posa\'t en contacte amb mi.</string> | ||||||
|     <string name="pref_header_theme">Temes</string> |     <string name="pref_header_theme">Temes</string> | ||||||
|     <string name="default_theme">Predeterminat</string> |  | ||||||
|     <string name="default_dark_theme">Predeterminat/Fosc</string> |  | ||||||
|     <string name="pref_selfoss_category">API de Selfoss</string> |     <string name="pref_selfoss_category">API de Selfoss</string> | ||||||
|     <string name="pref_api_items_number_title">Nombre d\'elements carregats</string> |     <string name="pref_api_items_number_title">Nombre d\'elements carregats</string> | ||||||
|     <string name="pref_hidden_tags">Etiquetes ocultes</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Carrega articles en desplaçar</string> |     <string name="pref_general_infinite_loading_title">Carrega articles en desplaçar</string> | ||||||
|     <string name="translation">Traducció</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> |     <string name="cant_open_invalid_url">L\'element URL no és vàlid. Estic intentant solucionar aquest problema perquè l\'aplicació no falli.</string> | ||||||
|     <string name="drawer_report_bug">Informa d\'un error</string> |  | ||||||
|     <string name="items_number_should_be_number">El nombre d\'elements ha de ser enter.</string> |     <string name="items_number_should_be_number">El nombre d\'elements ha de ser enter.</string> | ||||||
|     <string name="reader_action_more">Més informació</string> |  | ||||||
|     <string name="reader_action_open">Obre al navegador</string> |     <string name="reader_action_open">Obre al navegador</string> | ||||||
|     <string name="reader_action_share">Comparteix</string> |     <string name="reader_action_share">Comparteix</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Es marcaran els articles com a llegits en lliscar el dit d\'un article a l\'altre.</string> |     <string name="pref_switch_actions_pager_scroll_on">Es marcaran els articles com a llegits en lliscar el dit d\'un article a l\'altre.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">Aquesta acció marcarà els elements com a llegits.</string> |     <string name="markall_dialog_message">Aquesta acció marcarà els elements com a llegits.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Marca com a llegit en lliscar el dit</string> |     <string name="pref_switch_actions_pager_scroll">Marca com a llegit en lliscar el dit</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">No es marcaran els articles com a llegits en lliscar el dit d\'un article a l\'altre.</string> |     <string name="pref_switch_actions_pager_scroll_off">No es marcaran els articles com a llegits en lliscar el dit d\'un article a l\'altre.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Etiquetes ocultes</string> |  | ||||||
|     <string name="unmark">Marca com no llegit</string> |     <string name="unmark">Marca com no llegit</string> | ||||||
|     <string name="pref_header_offline">Sense connexió i memòria clau</string> |     <string name="pref_header_offline">Sense connexió i memòria clau</string> | ||||||
|     <string name="pref_switch_items_caching_off">Els articles no es guardaran a la memòria del dispositiu i l\'aplicació no es podrà utilitzar sense connexió.</string> |     <string name="pref_switch_items_caching_off">Els articles no es guardaran a la memòria del dispositiu i l\'aplicació no es podrà utilitzar sense connexió.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Els articles se sincronitzaran periòdicament</string> |     <string name="pref_switch_periodic_refresh_on">Els articles se sincronitzaran periòdicament</string> | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Interval de sincronització ( >= 15 minuts)]]></string> |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Interval de sincronització ( >= 15 minuts)]]></string> | ||||||
|     <string name="pref_switch_refresh_when_charging">Sincronitza només quan el telèfon s\'està carregant</string> |     <string name="pref_switch_refresh_when_charging">Sincronitza només quan el telèfon s\'està carregant</string> | ||||||
|     <string name="loading_notification_title">S\'està carregant...</string> |     <string name="loading_notification_title">S\'està carregant…</string> | ||||||
|     <string name="loading_notification_text">Selfoss està sincronitzant els articles</string> |     <string name="loading_notification_text">Selfoss està sincronitzant els articles</string> | ||||||
|     <string name="notification_channel_sync">Notificació de sincronització</string> |     <string name="notification_channel_sync">Notificació de sincronització</string> | ||||||
|     <string name="new_items_channel_sync">Notificació d\'elements nous</string> |     <string name="new_items_channel_sync">Notificació d\'elements nous</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"Anmeldung erforderlich?"</string> |     <string name="withLoginSwitch">"Anmeldung erforderlich?"</string> | ||||||
|     <string name="login_url_problem">"Ups. Du musst eventuell ein \"/\" am Ende der URL anhängen."</string> |     <string name="login_url_problem">"Ups. Du musst eventuell ein \"/\" am Ende der URL anhängen."</string> | ||||||
|     <string name="prompt_login">"Benutzername"</string> |     <string name="prompt_login">"Benutzername"</string> | ||||||
|     <string name="label_share">"Teilen"</string> |  | ||||||
|     <string name="readAll">"Alle gelesen"</string> |     <string name="readAll">"Alle gelesen"</string> | ||||||
|     <string name="action_disconnect">"Verbindung trennen"</string> |     <string name="action_disconnect">"Verbindung trennen"</string> | ||||||
|     <string name="title_activity_settings">"Einstellungen"</string> |     <string name="title_activity_settings">"Einstellungen"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Kartenhöhe passt sich Inhalt an</string> |     <string name="card_height_on">Kartenhöhe passt sich Inhalt an</string> | ||||||
|     <string name="card_height_off">Kartenhöhe ist fix</string> |     <string name="card_height_off">Kartenhöhe ist fix</string> | ||||||
|     <string name="source_code">Quellcode</string> |     <string name="source_code">Quellcode</string> | ||||||
|     <string name="drawer_error_loading_tags">Fehler beim Laden der Tags…</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="drawer_item_filters">Filter</string> |     <string name="filter_item_sources">Quellen</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="drawer_loading">Lade…</string> |  | ||||||
|     <string name="menu_home_search">Suche</string> |     <string name="menu_home_search">Suche</string> | ||||||
|     <string name="can_delete_source">Can\'t delete the source…</string> |     <string name="can_delete_source">Can\'t delete the source…</string> | ||||||
|     <string name="base_url_error">Beim Versuch deine Selfoss-Instanz zu erreichen ist ein Fehler aufgetreten. Solltet dieser Fehler bestehen bleiben, trete bitte mit mir in Kontakt.</string> |     <string name="base_url_error">Beim Versuch deine Selfoss-Instanz zu erreichen ist ein Fehler aufgetreten. Solltet dieser Fehler bestehen bleiben, trete bitte mit mir in Kontakt.</string> | ||||||
|     <string name="pref_header_theme">Designs</string> |     <string name="pref_header_theme">Designs</string> | ||||||
|     <string name="default_theme">Standard</string> |  | ||||||
|     <string name="default_dark_theme">Standard (Dunkel)</string> |  | ||||||
|     <string name="pref_selfoss_category">selfoss API</string> |     <string name="pref_selfoss_category">selfoss API</string> | ||||||
|     <string name="pref_api_items_number_title">Loaded items number</string> |     <string name="pref_api_items_number_title">Loaded items number</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> |     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||||
|     <string name="translation">Übersetzung</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Melde einen Fehler</string> |  | ||||||
|     <string name="items_number_should_be_number">The items number should be an integer.</string> |     <string name="items_number_should_be_number">The items number should be an integer.</string> | ||||||
|     <string name="reader_action_more">Read more</string> |  | ||||||
|     <string name="reader_action_open">Im Browser öffnen</string> |     <string name="reader_action_open">Im Browser öffnen</string> | ||||||
|     <string name="reader_action_share">Teilen</string> |     <string name="reader_action_share">Teilen</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">Dies wird alle Elemente als gelesen markieren.</string> |     <string name="markall_dialog_message">Dies wird alle Elemente als gelesen markieren.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Beim Wischen als gelesen markieren</string> |     <string name="pref_switch_actions_pager_scroll">Beim Wischen als gelesen markieren</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Eintrag als ungelesen markieren</string> |     <string name="unmark">Eintrag als ungelesen markieren</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Die Artikel werden regelmäßig synchronisiert</string> |     <string name="pref_switch_periodic_refresh_on">Die Artikel werden regelmäßig synchronisiert</string> | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|     <string name="pref_switch_refresh_when_charging">Nur aktualisieren, wenn das Telefon aufgeladen wird</string> |     <string name="pref_switch_refresh_when_charging">Nur aktualisieren, wenn das Telefon aufgeladen wird</string> | ||||||
|     <string name="loading_notification_title">Lädt...</string> |     <string name="loading_notification_title">Lädt…</string> | ||||||
|     <string name="loading_notification_text">Selfoss synchronisiert Ihre Artikel</string> |     <string name="loading_notification_text">Selfoss synchronisiert Ihre Artikel</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"Inicio de sesión requerido ?"</string> |     <string name="withLoginSwitch">"Inicio de sesión requerido ?"</string> | ||||||
|     <string name="login_url_problem">"Oops. Puede que necesite añadir un \"/\" al final de la url."</string> |     <string name="login_url_problem">"Oops. Puede que necesite añadir un \"/\" al final de la url."</string> | ||||||
|     <string name="prompt_login">"Nombre de usuario"</string> |     <string name="prompt_login">"Nombre de usuario"</string> | ||||||
|     <string name="label_share">"Compartir"</string> |  | ||||||
|     <string name="readAll">"Leer todo"</string> |     <string name="readAll">"Leer todo"</string> | ||||||
|     <string name="action_disconnect">"Desconectar"</string> |     <string name="action_disconnect">"Desconectar"</string> | ||||||
|     <string name="title_activity_settings">"Configuración"</string> |     <string name="title_activity_settings">"Configuración"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Altura de tarjetas se ajustará a su contenido</string> |     <string name="card_height_on">Altura de tarjetas se ajustará a su contenido</string> | ||||||
|     <string name="card_height_off">Se fijará la altura de la tarjeta</string> |     <string name="card_height_off">Se fijará la altura de la tarjeta</string> | ||||||
|     <string name="source_code">Código fuente</string> |     <string name="source_code">Código fuente</string> | ||||||
|     <string name="drawer_error_loading_tags">Error al cargar etiquetas…</string> |     <string name="filter_item_tags">Etiquetas</string> | ||||||
|     <string name="drawer_item_filters">Filtros</string> |     <string name="filter_item_sources">Fuentes</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="drawer_loading">Cargando…</string> |  | ||||||
|     <string name="menu_home_search">Buscar</string> |     <string name="menu_home_search">Buscar</string> | ||||||
|     <string name="can_delete_source">No se puede eliminar la fuente…</string> |     <string name="can_delete_source">No se puede eliminar la fuente…</string> | ||||||
|     <string name="base_url_error">Hubo un problema al intentar comunicarse con su instancia de Selfoss. Si el problema persiste, póngase en contacto conmigo.</string> |     <string name="base_url_error">Hubo un problema al intentar comunicarse con su instancia de Selfoss. Si el problema persiste, póngase en contacto conmigo.</string> | ||||||
|     <string name="pref_header_theme">Temas</string> |     <string name="pref_header_theme">Temas</string> | ||||||
|     <string name="default_theme">Predeterminado</string> |  | ||||||
|     <string name="default_dark_theme">Predeterminado/Oscuro</string> |  | ||||||
|     <string name="pref_selfoss_category">Api de Selfoss</string> |     <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_api_items_number_title">Número de artículos cargados</string> | ||||||
|     <string name="pref_hidden_tags">Etiquetas ocultas</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Cargar más artículos en desplazamiento</string> |     <string name="pref_general_infinite_loading_title">Cargar más artículos en desplazamiento</string> | ||||||
|     <string name="translation">Traducción</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Reportar un error</string> |  | ||||||
|     <string name="items_number_should_be_number">El número de artículos debe ser un número entero.</string> |     <string name="items_number_should_be_number">El número de artículos debe ser un número entero.</string> | ||||||
|     <string name="reader_action_more">Leer más</string> |  | ||||||
|     <string name="reader_action_open">Abrir en el navegador</string> |     <string name="reader_action_open">Abrir en el navegador</string> | ||||||
|     <string name="reader_action_share">Compartir</string> |     <string name="reader_action_share">Compartir</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Marcar artículos como leidos al desplazarse entre ellos.</string> |     <string name="pref_switch_actions_pager_scroll_on">Marcar artículos como leidos al desplazarse entre ellos.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">Esto marcará todos los artículos como leídos.</string> |     <string name="markall_dialog_message">Esto marcará todos los artículos como leídos.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Marcar artículos como leídos al deslizar con el dedo hacia los lados</string> |     <string name="pref_switch_actions_pager_scroll">Marcar artículos como leídos al deslizar con el dedo hacia los lados</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">No marcar artículos como leídos al deslizar con el dedo hacia los lados.</string> |     <string name="pref_switch_actions_pager_scroll_off">No marcar artículos como leídos al deslizar con el dedo hacia los lados.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Etiquetas ocultas</string> |  | ||||||
|     <string name="unmark">Marcar artículo como no leído</string> |     <string name="unmark">Marcar artículo como no leído</string> | ||||||
|     <string name="pref_header_offline">Sin conexión y caché</string> |     <string name="pref_header_offline">Sin conexión y caché</string> | ||||||
|     <string name="pref_switch_items_caching_off">Los artículos no se guardarán en la memoria del dispositivo y la aplicación no se podrá utilizar sin conexión.</string> |     <string name="pref_switch_items_caching_off">Los artículos no se guardarán en la memoria del dispositivo y la aplicación no se podrá utilizar sin conexión.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Los artículos se sincronizarán periódicamente</string> |     <string name="pref_switch_periodic_refresh_on">Los artículos se sincronizarán periódicamente</string> | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Intervalo de sincronización (>= 15 minutos)]]></string> |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Intervalo de sincronización (>= 15 minutos)]]></string> | ||||||
|     <string name="pref_switch_refresh_when_charging">Sólo refrescar cuando el teléfono está cargando</string> |     <string name="pref_switch_refresh_when_charging">Sólo refrescar cuando el teléfono está cargando</string> | ||||||
|     <string name="loading_notification_title">Cargando...</string> |     <string name="loading_notification_title">Cargando…</string> | ||||||
|     <string name="loading_notification_text">Selfoss está sincronizando tus artículos</string> |     <string name="loading_notification_text">Selfoss está sincronizando tus artículos</string> | ||||||
|     <string name="notification_channel_sync">Notificación de sincronización</string> |     <string name="notification_channel_sync">Notificación de sincronización</string> | ||||||
|     <string name="new_items_channel_sync">Notificación de elementos nuevos</string> |     <string name="new_items_channel_sync">Notificación de elementos nuevos</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"Login required ?"</string> |     <string name="withLoginSwitch">"Login required ?"</string> | ||||||
|     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> |     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> | ||||||
|     <string name="prompt_login">"Username"</string> |     <string name="prompt_login">"Username"</string> | ||||||
|     <string name="label_share">"Share"</string> |  | ||||||
|     <string name="readAll">"Read all"</string> |     <string name="readAll">"Read all"</string> | ||||||
|     <string name="action_disconnect">"Disconnect"</string> |     <string name="action_disconnect">"Disconnect"</string> | ||||||
|     <string name="title_activity_settings">"Settings"</string> |     <string name="title_activity_settings">"Settings"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Cards height will adjust to its content</string> |     <string name="card_height_on">Cards height will adjust to its content</string> | ||||||
|     <string name="card_height_off">Card height will be fixed</string> |     <string name="card_height_off">Card height will be fixed</string> | ||||||
|     <string name="source_code">Source code</string> |     <string name="source_code">Source code</string> | ||||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="drawer_item_filters">Filters</string> |     <string name="filter_item_sources">Sources</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="drawer_loading">Loading …</string> |  | ||||||
|     <string name="menu_home_search">Search</string> |     <string name="menu_home_search">Search</string> | ||||||
|     <string name="can_delete_source">Can\'t delete the source…</string> |     <string name="can_delete_source">Can\'t delete the source…</string> | ||||||
|     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> |     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||||
|     <string name="pref_header_theme">Themes</string> |     <string name="pref_header_theme">Themes</string> | ||||||
|     <string name="default_theme">Default</string> |  | ||||||
|     <string name="default_dark_theme">Default/Dark</string> |  | ||||||
|     <string name="pref_selfoss_category">Selfoss Api</string> |     <string name="pref_selfoss_category">Selfoss Api</string> | ||||||
|     <string name="pref_api_items_number_title">Loaded items number</string> |     <string name="pref_api_items_number_title">Loaded items number</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> |     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||||
|     <string name="translation">Translation</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Report a bug</string> |  | ||||||
|     <string name="items_number_should_be_number">The items number should be an integer.</string> |     <string name="items_number_should_be_number">The items number should be an integer.</string> | ||||||
|     <string name="reader_action_more">Read more</string> |  | ||||||
|     <string name="reader_action_open">Open in browser</string> |     <string name="reader_action_open">Open in browser</string> | ||||||
|     <string name="reader_action_share">Share</string> |     <string name="reader_action_share">Share</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"Avec login ?"</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="prompt_login">"Utilisateur"</string> | ||||||
|     <string name="label_share">"Partager"</string> |  | ||||||
|     <string name="readAll">"Tout lire"</string> |     <string name="readAll">"Tout lire"</string> | ||||||
|     <string name="action_disconnect">"Déconnecter"</string> |     <string name="action_disconnect">"Déconnecter"</string> | ||||||
|     <string name="title_activity_settings">"Paramètres"</string> |     <string name="title_activity_settings">"Paramètres"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">La taille de la carte s\'adaptera au contenu</string> |     <string name="card_height_on">La taille de la carte s\'adaptera au contenu</string> | ||||||
|     <string name="card_height_off">La taille de la carte sera fixe</string> |     <string name="card_height_off">La taille de la carte sera fixe</string> | ||||||
|     <string name="source_code">Code source</string> |     <string name="source_code">Code source</string> | ||||||
|     <string name="drawer_error_loading_tags">Erreur lors du chargement des tags…</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="drawer_item_filters">Filtres</string> |     <string name="filter_item_sources">Sources</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="drawer_loading">Chargement …</string> |  | ||||||
|     <string name="menu_home_search">Rechercher</string> |     <string name="menu_home_search">Rechercher</string> | ||||||
|     <string name="can_delete_source">Impossible de supprimer la source…</string> |     <string name="can_delete_source">Impossible de supprimer la source…</string> | ||||||
|     <string name="base_url_error">Il y a eu un souci lors de la communication avec votre instance Selfoss. Si le problèmes persiste, contactez-moi pour trouver une solution.</string> |     <string name="base_url_error">Il y a eu un souci lors de la communication avec votre instance Selfoss. Si le problèmes persiste, contactez-moi pour trouver une solution.</string> | ||||||
|     <string name="pref_header_theme">Thèmes</string> |     <string name="pref_header_theme">Thèmes</string> | ||||||
|     <string name="default_theme">Par défaut</string> |  | ||||||
|     <string name="default_dark_theme">Par défaut/Foncé</string> |  | ||||||
|     <string name="pref_selfoss_category">Api Selfoss</string> |     <string name="pref_selfoss_category">Api Selfoss</string> | ||||||
|     <string name="pref_api_items_number_title">Nombre d\'articles chargés</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="pref_general_infinite_loading_title">Charger plus d\'articles au scroll</string> |     <string name="pref_general_infinite_loading_title">Charger plus d\'articles au scroll</string> | ||||||
|     <string name="translation">Traduction</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Signaler un bug</string> |  | ||||||
|     <string name="items_number_should_be_number">Le nombre d\'articles doit être un entier.</string> |     <string name="items_number_should_be_number">Le nombre d\'articles doit être un entier.</string> | ||||||
|     <string name="reader_action_more">Lire plus</string> |  | ||||||
|     <string name="reader_action_open">Ouvrir</string> |     <string name="reader_action_open">Ouvrir</string> | ||||||
|     <string name="reader_action_share">Partager</string> |     <string name="reader_action_share">Partager</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Marquer les articles comme lus à la navigation dans le lecteur d\'article.</string> |     <string name="pref_switch_actions_pager_scroll_on">Marquer les articles comme lus à la navigation dans le lecteur d\'article.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">Marquer tous les éléments comme lus ?</string> |     <string name="markall_dialog_message">Marquer tous les éléments comme lus ?</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Marquer comme lu à la navigation.</string> |     <string name="pref_switch_actions_pager_scroll">Marquer comme lu à la navigation.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Ne pas marquer les articles comme lus à la navigation.</string> |     <string name="pref_switch_actions_pager_scroll_off">Ne pas marquer les articles comme lus à la navigation.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Tags Cachés</string> |  | ||||||
|     <string name="unmark">Marquer l\'article comme non lu</string> |     <string name="unmark">Marquer l\'article comme non lu</string> | ||||||
|     <string name="pref_header_offline">Hors ligne et cache</string> |     <string name="pref_header_offline">Hors ligne et cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Les articles ne seront pas enregistrés et l\'application ne sera pas utilisable hors ligne.</string> |     <string name="pref_switch_items_caching_off">Les articles ne seront pas enregistrés et l\'application ne sera pas utilisable hors ligne.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles seront périodiquement synchronisées</string> |     <string name="pref_switch_periodic_refresh_on">Articles seront périodiquement synchronisées</string> | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Interval de synchronisation ( >= 15 minutes)]]></string> |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Interval de synchronisation ( >= 15 minutes)]]></string> | ||||||
|     <string name="pref_switch_refresh_when_charging">Synchroniser uniquement lorsque le téléphone est en charge</string> |     <string name="pref_switch_refresh_when_charging">Synchroniser uniquement lorsque le téléphone est en charge</string> | ||||||
|     <string name="loading_notification_title">Chargement ...</string> |     <string name="loading_notification_title">Chargement …</string> | ||||||
|     <string name="loading_notification_text">Selfoss synchronise vos articles</string> |     <string name="loading_notification_text">Selfoss synchronise vos articles</string> | ||||||
|     <string name="notification_channel_sync">Notification de synchronisation</string> |     <string name="notification_channel_sync">Notification de synchronisation</string> | ||||||
|     <string name="new_items_channel_sync">Notification de nouveaux articles</string> |     <string name="new_items_channel_sync">Notification de nouveaux articles</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Thème sombre</string> |     <string name="mode_dark">Thème sombre</string> | ||||||
|     <string name="mode_system">Utiliser les paramètres système</string> |     <string name="mode_system">Utiliser les paramètres système</string> | ||||||
|     <string name="mode_light">Thème clair</string> |     <string name="mode_light">Thème clair</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"É preciso iniciar sesión?"</string> |     <string name="withLoginSwitch">"É preciso iniciar sesión?"</string> | ||||||
|     <string name="login_url_problem">"Ups! Pode que precises engadir un \"/\" o final da URL."</string> |     <string name="login_url_problem">"Ups! Pode que precises engadir un \"/\" o final da URL."</string> | ||||||
|     <string name="prompt_login">"Nome de usuario"</string> |     <string name="prompt_login">"Nome de usuario"</string> | ||||||
|     <string name="label_share">"Compartir"</string> |  | ||||||
|     <string name="readAll">"Ler todos"</string> |     <string name="readAll">"Ler todos"</string> | ||||||
|     <string name="action_disconnect">"Desconectar"</string> |     <string name="action_disconnect">"Desconectar"</string> | ||||||
|     <string name="title_activity_settings">"Axustes"</string> |     <string name="title_activity_settings">"Axustes"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">A altura das tarxetas axustarase ao seu contido</string> |     <string name="card_height_on">A altura das tarxetas axustarase ao seu contido</string> | ||||||
|     <string name="card_height_off">A altura das tarxetas será fixa</string> |     <string name="card_height_off">A altura das tarxetas será fixa</string> | ||||||
|     <string name="source_code">Código fonte</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="filter_item_tags">Etiquetas</string> | ||||||
|     <string name="drawer_item_filters">Filtros</string> |     <string name="filter_item_sources">Fontes</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="drawer_loading">Cargando…</string> |  | ||||||
|     <string name="menu_home_search">Procurar</string> |     <string name="menu_home_search">Procurar</string> | ||||||
|     <string name="can_delete_source">Non se puido eliminar a fonte…</string> |     <string name="can_delete_source">Non se puido eliminar a fonte…</string> | ||||||
|     <string name="base_url_error">Houno unha incidencia ao tratar de comunicarse coa túa instancia de Selfoss. Se o problema persiste, prégolle que se poña en contacto conmigo.</string> |     <string name="base_url_error">Houno unha incidencia ao tratar de comunicarse coa túa instancia de Selfoss. Se o problema persiste, prégolle que se poña en contacto conmigo.</string> | ||||||
|     <string name="pref_header_theme">Temas</string> |     <string name="pref_header_theme">Temas</string> | ||||||
|     <string name="default_theme">Predeterminado</string> |  | ||||||
|     <string name="default_dark_theme">Predeterminado/Escuro</string> |  | ||||||
|     <string name="pref_selfoss_category">API de Selfoss</string> |     <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_api_items_number_title">Número de elementos cargados</string> | ||||||
|     <string name="pref_hidden_tags">Etiquetas ocultas</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Cargar máis artigos ao desprazarse</string> |     <string name="pref_general_infinite_loading_title">Cargar máis artigos ao desprazarse</string> | ||||||
|     <string name="translation">Traducción</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Informar dun erro</string> |  | ||||||
|     <string name="items_number_should_be_number">O número de elementos debería ser un enteiro.</string> |     <string name="items_number_should_be_number">O número de elementos debería ser un enteiro.</string> | ||||||
|     <string name="reader_action_more">Ler máis</string> |  | ||||||
|     <string name="reader_action_open">Abrir no navegador</string> |     <string name="reader_action_open">Abrir no navegador</string> | ||||||
|     <string name="reader_action_share">Compartir</string> |     <string name="reader_action_share">Compartir</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Marcar artigos como lidos cando se desliza o dedo dun a outro.</string> |     <string name="pref_switch_actions_pager_scroll_on">Marcar artigos como lidos cando se desliza o dedo dun a outro.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">Isto marcara todos os elementos como lidos.</string> |     <string name="markall_dialog_message">Isto marcara todos os elementos como lidos.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Marcar artigos como lidos ao deslizar co dedo cara os lados</string> |     <string name="pref_switch_actions_pager_scroll">Marcar artigos como lidos ao deslizar co dedo cara os lados</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Non marcar artigos como lidos ao deslizar co dedo cara os lados.</string> |     <string name="pref_switch_actions_pager_scroll_off">Non marcar artigos como lidos ao deslizar co dedo cara os lados.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Etiquetas ocultas</string> |  | ||||||
|     <string name="unmark">Marcar artículo como non lido</string> |     <string name="unmark">Marcar artículo como non lido</string> | ||||||
|     <string name="pref_header_offline">Sen conexión e caché</string> |     <string name="pref_header_offline">Sen conexión e caché</string> | ||||||
|     <string name="pref_switch_items_caching_off">Os artigos non se gardaran na memoria do dispositivo e non se poderá utilizar a aplicación sen conexión.</string> |     <string name="pref_switch_items_caching_off">Os artigos non se gardaran na memoria do dispositivo e non se poderá utilizar a aplicación sen conexión.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Os artigos sincronizaranse periódicamente</string> |     <string name="pref_switch_periodic_refresh_on">Os artigos sincronizaranse periódicamente</string> | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Intervalo de sincronización (>= 15 minutos)]]></string> |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Intervalo de sincronización (>= 15 minutos)]]></string> | ||||||
|     <string name="pref_switch_refresh_when_charging">Só refrescar cando o teléfono se está a cargar</string> |     <string name="pref_switch_refresh_when_charging">Só refrescar cando o teléfono se está a cargar</string> | ||||||
|     <string name="loading_notification_title">Cargando...</string> |     <string name="loading_notification_title">Cargando…</string> | ||||||
|     <string name="loading_notification_text">Selfoss está sincronizando os teus ar tigos</string> |     <string name="loading_notification_text">Selfoss está sincronizando os teus ar tigos</string> | ||||||
|     <string name="notification_channel_sync">Notificación de sincronización</string> |     <string name="notification_channel_sync">Notificación de sincronización</string> | ||||||
|     <string name="new_items_channel_sync">Notificación de actualizacións</string> |     <string name="new_items_channel_sync">Notificación de actualizacións</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"Harus masuk?"</string> |     <string name="withLoginSwitch">"Harus masuk?"</string> | ||||||
|     <string name="login_url_problem">"Ups. Anda mungkin harus menambahkan \"/\" di akhir url."</string> |     <string name="login_url_problem">"Ups. Anda mungkin harus menambahkan \"/\" di akhir url."</string> | ||||||
|     <string name="prompt_login">"Nama pengguna"</string> |     <string name="prompt_login">"Nama pengguna"</string> | ||||||
|     <string name="label_share">"Bagikan"</string> |  | ||||||
|     <string name="readAll">"Baca semua"</string> |     <string name="readAll">"Baca semua"</string> | ||||||
|     <string name="action_disconnect">"Putuskan sambungan"</string> |     <string name="action_disconnect">"Putuskan sambungan"</string> | ||||||
|     <string name="title_activity_settings">"Pengaturan"</string> |     <string name="title_activity_settings">"Pengaturan"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Tinggi kartu akan disesuaikan dengan konten</string> |     <string name="card_height_on">Tinggi kartu akan disesuaikan dengan konten</string> | ||||||
|     <string name="card_height_off">Ukuran kartu akan tetap</string> |     <string name="card_height_off">Ukuran kartu akan tetap</string> | ||||||
|     <string name="source_code">Kode sumber</string> |     <string name="source_code">Kode sumber</string> | ||||||
|     <string name="drawer_error_loading_tags">Kesalahan saat memuat tag…</string> |     <string name="filter_item_tags">Tag</string> | ||||||
|     <string name="drawer_item_filters">Filter</string> |     <string name="filter_item_sources">Sumber</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="drawer_loading">Memuat …</string> |  | ||||||
|     <string name="menu_home_search">Cari</string> |     <string name="menu_home_search">Cari</string> | ||||||
|     <string name="can_delete_source">Tidak dapat menghapus sumber…</string> |     <string name="can_delete_source">Tidak dapat menghapus sumber…</string> | ||||||
|     <string name="base_url_error">Ada masalah saat berkomunikasi dengan Selfoss Anda. Jika masalah berlanjut, tolong hubungi saya.</string> |     <string name="base_url_error">Ada masalah saat berkomunikasi dengan Selfoss Anda. Jika masalah berlanjut, tolong hubungi saya.</string> | ||||||
|     <string name="pref_header_theme">Tema</string> |     <string name="pref_header_theme">Tema</string> | ||||||
|     <string name="default_theme">Bawaan</string> |  | ||||||
|     <string name="default_dark_theme">Bawaan/Gelap</string> |  | ||||||
|     <string name="pref_selfoss_category">Selfoss Api</string> |     <string name="pref_selfoss_category">Selfoss Api</string> | ||||||
|     <string name="pref_api_items_number_title">Item nomor dimuat</string> |     <string name="pref_api_items_number_title">Item nomor dimuat</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Muat lebih banyak artikel saat membalik halaman</string> |     <string name="pref_general_infinite_loading_title">Muat lebih banyak artikel saat membalik halaman</string> | ||||||
|     <string name="translation">Terjemahan</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> |     <string name="cant_open_invalid_url">Alamat tautan proyek tidak valid. Saya mencoba memecahkan masalah ini untuk menghindari aplikasi berhenti.</string> | ||||||
|     <string name="drawer_report_bug">Laporkan bug</string> |  | ||||||
|     <string name="items_number_should_be_number">Jumlah item harus berupa bilangan bulat.</string> |     <string name="items_number_should_be_number">Jumlah item harus berupa bilangan bulat.</string> | ||||||
|     <string name="reader_action_more">Baca lebih lanjut</string> |  | ||||||
|     <string name="reader_action_open">Buka di peramban</string> |     <string name="reader_action_open">Buka di peramban</string> | ||||||
|     <string name="reader_action_share">Bagikan</string> |     <string name="reader_action_share">Bagikan</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"È richiesto l'accesso?"</string> |     <string name="withLoginSwitch">"È richiesto l'accesso?"</string> | ||||||
|     <string name="login_url_problem">"Oops. Potrebbe essere necessario aggiungere un \"/\" alla fine dell'url."</string> |     <string name="login_url_problem">"Oops. Potrebbe essere necessario aggiungere un \"/\" alla fine dell'url."</string> | ||||||
|     <string name="prompt_login">"Nome utente"</string> |     <string name="prompt_login">"Nome utente"</string> | ||||||
|     <string name="label_share">"Condividi"</string> |  | ||||||
|     <string name="readAll">"Segna tutte come lette"</string> |     <string name="readAll">"Segna tutte come lette"</string> | ||||||
|     <string name="action_disconnect">"Scollegati"</string> |     <string name="action_disconnect">"Scollegati"</string> | ||||||
|     <string name="title_activity_settings">"Impostazioni"</string> |     <string name="title_activity_settings">"Impostazioni"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Cards height will adjust to its content</string> |     <string name="card_height_on">Cards height will adjust to its content</string> | ||||||
|     <string name="card_height_off">Card height will be fixed</string> |     <string name="card_height_off">Card height will be fixed</string> | ||||||
|     <string name="source_code">Codice sorgente</string> |     <string name="source_code">Codice sorgente</string> | ||||||
|     <string name="drawer_error_loading_tags">Errore nel caricamento dei tag…</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="drawer_item_filters">Filtri</string> |     <string name="filter_item_sources">Fonti</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="drawer_loading">Caricamento…</string> |  | ||||||
|     <string name="menu_home_search">Cerca</string> |     <string name="menu_home_search">Cerca</string> | ||||||
|     <string name="can_delete_source">Non è possibile eliminare la fonte…</string> |     <string name="can_delete_source">Non è possibile eliminare la fonte…</string> | ||||||
|     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> |     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||||
|     <string name="pref_header_theme">Temi</string> |     <string name="pref_header_theme">Temi</string> | ||||||
|     <string name="default_theme">Predefinito</string> |  | ||||||
|     <string name="default_dark_theme">Predefinito (Scuro)</string> |  | ||||||
|     <string name="pref_selfoss_category">Api di Selfoss</string> |     <string name="pref_selfoss_category">Api di Selfoss</string> | ||||||
|     <string name="pref_api_items_number_title">Numero di elementi caricati</string> |     <string name="pref_api_items_number_title">Numero di elementi caricati</string> | ||||||
|     <string name="pref_hidden_tags">Tag nascosti</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> |     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||||
|     <string name="translation">Traduzioni</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Segnala un bug</string> |  | ||||||
|     <string name="items_number_should_be_number">The items number should be an integer.</string> |     <string name="items_number_should_be_number">The items number should be an integer.</string> | ||||||
|     <string name="reader_action_more">Read more</string> |  | ||||||
|     <string name="reader_action_open">Open in browser</string> |     <string name="reader_action_open">Open in browser</string> | ||||||
|     <string name="reader_action_share">Share</string> |     <string name="reader_action_share">Share</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Segna come non letto</string> |     <string name="unmark">Segna come non letto</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"로그인이 필요합니까?"</string> |     <string name="withLoginSwitch">"로그인이 필요합니까?"</string> | ||||||
|     <string name="login_url_problem">"죄송합니다. Url의 끝에 \"/\"를 추가할 필요가 있습니다."</string> |     <string name="login_url_problem">"죄송합니다. Url의 끝에 \"/\"를 추가할 필요가 있습니다."</string> | ||||||
|     <string name="prompt_login">"사용자 이름"</string> |     <string name="prompt_login">"사용자 이름"</string> | ||||||
|     <string name="label_share">"공유"</string> |  | ||||||
|     <string name="readAll">"모두 읽기"</string> |     <string name="readAll">"모두 읽기"</string> | ||||||
|     <string name="action_disconnect">"연결 해제"</string> |     <string name="action_disconnect">"연결 해제"</string> | ||||||
|     <string name="title_activity_settings">"설정"</string> |     <string name="title_activity_settings">"설정"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Cards height will adjust to its content</string> |     <string name="card_height_on">Cards height will adjust to its content</string> | ||||||
|     <string name="card_height_off">Card height will be fixed</string> |     <string name="card_height_off">Card height will be fixed</string> | ||||||
|     <string name="source_code">Source code</string> |     <string name="source_code">Source code</string> | ||||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="drawer_item_filters">Filters</string> |     <string name="filter_item_sources">Sources</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="drawer_loading">Loading …</string> |  | ||||||
|     <string name="menu_home_search">Search</string> |     <string name="menu_home_search">Search</string> | ||||||
|     <string name="can_delete_source">Can\'t delete the source…</string> |     <string name="can_delete_source">Can\'t delete the source…</string> | ||||||
|     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> |     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||||
|     <string name="pref_header_theme">Themes</string> |     <string name="pref_header_theme">Themes</string> | ||||||
|     <string name="default_theme">Default</string> |  | ||||||
|     <string name="default_dark_theme">Default/Dark</string> |  | ||||||
|     <string name="pref_selfoss_category">Selfoss Api</string> |     <string name="pref_selfoss_category">Selfoss Api</string> | ||||||
|     <string name="pref_api_items_number_title">Loaded items number</string> |     <string name="pref_api_items_number_title">Loaded items number</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> |     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||||
|     <string name="translation">Translation</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Report a bug</string> |  | ||||||
|     <string name="items_number_should_be_number">The items number should be an integer.</string> |     <string name="items_number_should_be_number">The items number should be an integer.</string> | ||||||
|     <string name="reader_action_more">Read more</string> |  | ||||||
|     <string name="reader_action_open">Open in browser</string> |     <string name="reader_action_open">Open in browser</string> | ||||||
|     <string name="reader_action_share">Share</string> |     <string name="reader_action_share">Share</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								androidApp/src/main/res/values-night/strings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								androidApp/src/main/res/values-night/strings.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <resources> | ||||||
|  |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
|  | </resources> | ||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"Authenticatie vereist?"</string> |     <string name="withLoginSwitch">"Authenticatie vereist?"</string> | ||||||
|     <string name="login_url_problem">"Oeps, ben je soms de \"/\" vergeten aan het eind?"</string> |     <string name="login_url_problem">"Oeps, ben je soms de \"/\" vergeten aan het eind?"</string> | ||||||
|     <string name="prompt_login">"Gebruikersnaam"</string> |     <string name="prompt_login">"Gebruikersnaam"</string> | ||||||
|     <string name="label_share">"Delen"</string> |  | ||||||
|     <string name="readAll">"Alles lezen"</string> |     <string name="readAll">"Alles lezen"</string> | ||||||
|     <string name="action_disconnect">"Verbinding verbreken"</string> |     <string name="action_disconnect">"Verbinding verbreken"</string> | ||||||
|     <string name="title_activity_settings">"Instellingen"</string> |     <string name="title_activity_settings">"Instellingen"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Hoogte aanpassen aan de hand van kaartinhoud</string> |     <string name="card_height_on">Hoogte aanpassen aan de hand van kaartinhoud</string> | ||||||
|     <string name="card_height_off">Vaste hoogte</string> |     <string name="card_height_off">Vaste hoogte</string> | ||||||
|     <string name="source_code">Broncode</string> |     <string name="source_code">Broncode</string> | ||||||
|     <string name="drawer_error_loading_tags">Fout bij het laden van tags…</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="drawer_item_filters">Filters</string> |     <string name="filter_item_sources">Bronnen</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="drawer_loading">Bezig met laden …</string> |  | ||||||
|     <string name="menu_home_search">Zoeken</string> |     <string name="menu_home_search">Zoeken</string> | ||||||
|     <string name="can_delete_source">Kan de bron niet verwijderen…</string> |     <string name="can_delete_source">Kan de bron niet verwijderen…</string> | ||||||
|     <string name="base_url_error">Er was een probleem bij het communiceren met uw Selfoss Instance. Als het probleem blijft, neem dan contact met mij op.</string> |     <string name="base_url_error">Er was een probleem bij het communiceren met uw Selfoss Instance. Als het probleem blijft, neem dan contact met mij op.</string> | ||||||
|     <string name="pref_header_theme">Thema \'s</string> |     <string name="pref_header_theme">Thema \'s</string> | ||||||
|     <string name="default_theme">Standaard</string> |  | ||||||
|     <string name="default_dark_theme">Standaard/Donker</string> |  | ||||||
|     <string name="pref_selfoss_category">Selfoss Api</string> |     <string name="pref_selfoss_category">Selfoss Api</string> | ||||||
|     <string name="pref_api_items_number_title">Geladen items nummer</string> |     <string name="pref_api_items_number_title">Geladen items nummer</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Laad meer artikelen door te bladeren</string> |     <string name="pref_general_infinite_loading_title">Laad meer artikelen door te bladeren</string> | ||||||
|     <string name="translation">Vertaling</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> |     <string name="cant_open_invalid_url">De URL is ongeldig. Ik probeer dit probleem op te lossen, zodat de toepassing niet wordt afgesloten.</string> | ||||||
|     <string name="drawer_report_bug">Een fout melden</string> |  | ||||||
|     <string name="items_number_should_be_number">Het aantal items moet een geheel getal zijn.</string> |     <string name="items_number_should_be_number">Het aantal items moet een geheel getal zijn.</string> | ||||||
|     <string name="reader_action_more">Lees meer</string> |  | ||||||
|     <string name="reader_action_open">Openen in browser</string> |     <string name="reader_action_open">Openen in browser</string> | ||||||
|     <string name="reader_action_share">Delen</string> |     <string name="reader_action_share">Delen</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"É necessário o login ?"</string> |     <string name="withLoginSwitch">"É necessário o login ?"</string> | ||||||
|     <string name="login_url_problem">"Oops. Talvez você precise adicionar uma \"/\" no final da url."</string> |     <string name="login_url_problem">"Oops. Talvez você precise adicionar uma \"/\" no final da url."</string> | ||||||
|     <string name="prompt_login">"Usuário"</string> |     <string name="prompt_login">"Usuário"</string> | ||||||
|     <string name="label_share">"Compartilhar"</string> |  | ||||||
|     <string name="readAll">"Ler todos"</string> |     <string name="readAll">"Ler todos"</string> | ||||||
|     <string name="action_disconnect">"Desconectar"</string> |     <string name="action_disconnect">"Desconectar"</string> | ||||||
|     <string name="title_activity_settings">"Configurações"</string> |     <string name="title_activity_settings">"Configurações"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Cards com altura ajustáveis de acordo com o conteúdo</string> |     <string name="card_height_on">Cards com altura ajustáveis de acordo com o conteúdo</string> | ||||||
|     <string name="card_height_off">Cards com altura de tamanho fixo</string> |     <string name="card_height_off">Cards com altura de tamanho fixo</string> | ||||||
|     <string name="source_code">Código fonte</string> |     <string name="source_code">Código fonte</string> | ||||||
|     <string name="drawer_error_loading_tags">Erro ao carregar as tags…</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="drawer_item_filters">Filtros</string> |     <string name="filter_item_sources">Fontes</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="drawer_loading">Carregando …</string> |  | ||||||
|     <string name="menu_home_search">Procurar</string> |     <string name="menu_home_search">Procurar</string> | ||||||
|     <string name="can_delete_source">Não foi possível apagar a fonte…</string> |     <string name="can_delete_source">Não foi possível apagar a fonte…</string> | ||||||
|     <string name="base_url_error">Houve um problema ao tentar se comunicar com o seu Selfoss. Se o problema persistir, entre em contato comigo.</string> |     <string name="base_url_error">Houve um problema ao tentar se comunicar com o seu Selfoss. Se o problema persistir, entre em contato comigo.</string> | ||||||
|     <string name="pref_header_theme">Temas</string> |     <string name="pref_header_theme">Temas</string> | ||||||
|     <string name="default_theme">Padrão</string> |  | ||||||
|     <string name="default_dark_theme">Padrão/Escuro</string> |  | ||||||
|     <string name="pref_selfoss_category">Selfoss Api</string> |     <string name="pref_selfoss_category">Selfoss Api</string> | ||||||
|     <string name="pref_api_items_number_title">Quantidade de itens carregados</string> |     <string name="pref_api_items_number_title">Quantidade de itens carregados</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Carregar mais artigos ao realizar o scroll</string> |     <string name="pref_general_infinite_loading_title">Carregar mais artigos ao realizar o scroll</string> | ||||||
|     <string name="translation">Traduções</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> |     <string name="cant_open_invalid_url">A url está inválida. Estou tentando resolver esse problema para que o aplicativo não encerre.</string> | ||||||
|     <string name="drawer_report_bug">Reportar erro</string> |  | ||||||
|     <string name="items_number_should_be_number">O número dos itens deve ser um número inteiro.</string> |     <string name="items_number_should_be_number">O número dos itens deve ser um número inteiro.</string> | ||||||
|     <string name="reader_action_more">Leia mais</string> |  | ||||||
|     <string name="reader_action_open">Abrir no navegador</string> |     <string name="reader_action_open">Abrir no navegador</string> | ||||||
|     <string name="reader_action_share">Compartilhar</string> |     <string name="reader_action_share">Compartilhar</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Se esta configuração estiver ativada, os artigos serão marcados como lidos ao deslizar para a esquerda e para a direita no leitor do artigo.</string> |     <string name="pref_switch_actions_pager_scroll_on">Se esta configuração estiver ativada, os artigos serão marcados como lidos ao deslizar para a esquerda e para a direita no leitor do artigo.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">Isso marcará todos os itens como lidos.</string> |     <string name="markall_dialog_message">Isso marcará todos os itens como lidos.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Marcar Como Lida ao Abrir</string> |     <string name="pref_switch_actions_pager_scroll">Marcar Como Lida ao Abrir</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Não marca artigos como lido quando abrir.</string> |     <string name="pref_switch_actions_pager_scroll_off">Não marca artigos como lido quando abrir.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"É necessário fazer login?"</string> |     <string name="withLoginSwitch">"É necessário fazer login?"</string> | ||||||
|     <string name="login_url_problem">"Uups. Você pode precisar adicionar uma \"/\" no final da url."</string> |     <string name="login_url_problem">"Uups. Você pode precisar adicionar uma \"/\" no final da url."</string> | ||||||
|     <string name="prompt_login">"Nome do usuário"</string> |     <string name="prompt_login">"Nome do usuário"</string> | ||||||
|     <string name="label_share">"Compartilhar"</string> |  | ||||||
|     <string name="readAll">"Ler tudo"</string> |     <string name="readAll">"Ler tudo"</string> | ||||||
|     <string name="action_disconnect">"Desligar"</string> |     <string name="action_disconnect">"Desligar"</string> | ||||||
|     <string name="title_activity_settings">"Configurações"</string> |     <string name="title_activity_settings">"Configurações"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Altura de cartas irá ajustar ao seu conteúdo</string> |     <string name="card_height_on">Altura de cartas irá ajustar ao seu conteúdo</string> | ||||||
|     <string name="card_height_off">Altura do cartão será corrigida</string> |     <string name="card_height_off">Altura do cartão será corrigida</string> | ||||||
|     <string name="source_code">Código fonte</string> |     <string name="source_code">Código fonte</string> | ||||||
|     <string name="drawer_error_loading_tags">Erro ao carregar etiquetas…</string> |     <string name="filter_item_tags">Etiquetas</string> | ||||||
|     <string name="drawer_item_filters">Filtros</string> |     <string name="filter_item_sources">Fontes</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="drawer_loading">A carregar…</string> |  | ||||||
|     <string name="menu_home_search">Buscar</string> |     <string name="menu_home_search">Buscar</string> | ||||||
|     <string name="can_delete_source">Não é possível excluir a fonte…</string> |     <string name="can_delete_source">Não é possível excluir a fonte…</string> | ||||||
|     <string name="base_url_error">Houve um problema ao tentar se comunicar com sua instância de Selfoss. Se o problema persistir, por favor entre em contato comigo.</string> |     <string name="base_url_error">Houve um problema ao tentar se comunicar com sua instância de Selfoss. Se o problema persistir, por favor entre em contato comigo.</string> | ||||||
|     <string name="pref_header_theme">Temas</string> |     <string name="pref_header_theme">Temas</string> | ||||||
|     <string name="default_theme">Predefinição</string> |  | ||||||
|     <string name="default_dark_theme">Padrão/escuro</string> |  | ||||||
|     <string name="pref_selfoss_category">Api de Selfoss</string> |     <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_api_items_number_title">Número de itens carregados</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Carregar mais artigos no pergaminho</string> |     <string name="pref_general_infinite_loading_title">Carregar mais artigos no pergaminho</string> | ||||||
|     <string name="translation">Tradução</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Reportar falha</string> |  | ||||||
|     <string name="items_number_should_be_number">O número de itens deve ser um número inteiro.</string> |     <string name="items_number_should_be_number">O número de itens deve ser um número inteiro.</string> | ||||||
|     <string name="reader_action_more">Ler mais</string> |  | ||||||
|     <string name="reader_action_open">Abrir no browser</string> |     <string name="reader_action_open">Abrir no browser</string> | ||||||
|     <string name="reader_action_share">Compartilhar</string> |     <string name="reader_action_share">Compartilhar</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Artigos de marca como lida quando passar entre artigos.</string> |     <string name="pref_switch_actions_pager_scroll_on">Artigos de marca como lida quando passar entre artigos.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"Login required ?"</string> |     <string name="withLoginSwitch">"Login required ?"</string> | ||||||
|     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> |     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> | ||||||
|     <string name="prompt_login">"පරිශීලක නාමය"</string> |     <string name="prompt_login">"පරිශීලක නාමය"</string> | ||||||
|     <string name="label_share">"Share"</string> |  | ||||||
|     <string name="readAll">"Read all"</string> |     <string name="readAll">"Read all"</string> | ||||||
|     <string name="action_disconnect">"Disconnect"</string> |     <string name="action_disconnect">"Disconnect"</string> | ||||||
|     <string name="title_activity_settings">"සැකසුම්"</string> |     <string name="title_activity_settings">"සැකසුම්"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Cards height will adjust to its content</string> |     <string name="card_height_on">Cards height will adjust to its content</string> | ||||||
|     <string name="card_height_off">Card height will be fixed</string> |     <string name="card_height_off">Card height will be fixed</string> | ||||||
|     <string name="source_code">Source code</string> |     <string name="source_code">Source code</string> | ||||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="drawer_item_filters">Filters</string> |     <string name="filter_item_sources">Sources</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="drawer_loading">Loading …</string> |  | ||||||
|     <string name="menu_home_search">Search</string> |     <string name="menu_home_search">Search</string> | ||||||
|     <string name="can_delete_source">Can\'t delete the source…</string> |     <string name="can_delete_source">Can\'t delete the source…</string> | ||||||
|     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> |     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||||
|     <string name="pref_header_theme">Themes</string> |     <string name="pref_header_theme">Themes</string> | ||||||
|     <string name="default_theme">Default</string> |  | ||||||
|     <string name="default_dark_theme">Default/Dark</string> |  | ||||||
|     <string name="pref_selfoss_category">Selfoss Api</string> |     <string name="pref_selfoss_category">Selfoss Api</string> | ||||||
|     <string name="pref_api_items_number_title">Loaded items number</string> |     <string name="pref_api_items_number_title">Loaded items number</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> |     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||||
|     <string name="translation">Translation</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Report a bug</string> |  | ||||||
|     <string name="items_number_should_be_number">The items number should be an integer.</string> |     <string name="items_number_should_be_number">The items number should be an integer.</string> | ||||||
|     <string name="reader_action_more">Read more</string> |  | ||||||
|     <string name="reader_action_open">Open in browser</string> |     <string name="reader_action_open">Open in browser</string> | ||||||
|     <string name="reader_action_share">Share</string> |     <string name="reader_action_share">Share</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"Kullanıcı Girişi Gerekli?"</string> |     <string name="withLoginSwitch">"Kullanıcı Girişi Gerekli?"</string> | ||||||
|     <string name="login_url_problem">"Oops. Url'nin sonuna \"/\" eklemek gerekebilir."</string> |     <string name="login_url_problem">"Oops. Url'nin sonuna \"/\" eklemek gerekebilir."</string> | ||||||
|     <string name="prompt_login">"Kullanıcı adı"</string> |     <string name="prompt_login">"Kullanıcı adı"</string> | ||||||
|     <string name="label_share">"Paylaş"</string> |  | ||||||
|     <string name="readAll">"Tümünü oku"</string> |     <string name="readAll">"Tümünü oku"</string> | ||||||
|     <string name="action_disconnect">"Bağlantıyı kes"</string> |     <string name="action_disconnect">"Bağlantıyı kes"</string> | ||||||
|     <string name="title_activity_settings">"Ayarlar"</string> |     <string name="title_activity_settings">"Ayarlar"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">Kartların yüksekliği içeriğine göre ayarlanır</string> |     <string name="card_height_on">Kartların yüksekliği içeriğine göre ayarlanır</string> | ||||||
|     <string name="card_height_off">Kart yüksekliği sabit olacak</string> |     <string name="card_height_off">Kart yüksekliği sabit olacak</string> | ||||||
|     <string name="source_code">Kaynak kodu</string> |     <string name="source_code">Kaynak kodu</string> | ||||||
|     <string name="drawer_error_loading_tags">Etiketler yükleme hatası…</string> |     <string name="filter_item_tags">Etiketler</string> | ||||||
|     <string name="drawer_item_filters">Filtreler</string> |     <string name="filter_item_sources">Kaynaklar</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="drawer_loading">Yükleniyor…</string> |  | ||||||
|     <string name="menu_home_search">Ara</string> |     <string name="menu_home_search">Ara</string> | ||||||
|     <string name="can_delete_source">Kaynak silinemiyor…</string> |     <string name="can_delete_source">Kaynak silinemiyor…</string> | ||||||
|     <string name="base_url_error">Selfoss Örneğinizle iletişim kurmaya çalışırken bir sorun oluştu. Sorun devam ederse, lütfen benimle iletişime geçin.</string> |     <string name="base_url_error">Selfoss Örneğinizle iletişim kurmaya çalışırken bir sorun oluştu. Sorun devam ederse, lütfen benimle iletişime geçin.</string> | ||||||
|     <string name="pref_header_theme">Temalar</string> |     <string name="pref_header_theme">Temalar</string> | ||||||
|     <string name="default_theme">Varsayılan</string> |  | ||||||
|     <string name="default_dark_theme">Varsayılan/koyu</string> |  | ||||||
|     <string name="pref_selfoss_category">Selfoss Uygulaması</string> |     <string name="pref_selfoss_category">Selfoss Uygulaması</string> | ||||||
|     <string name="pref_api_items_number_title">Yüklenen öğe numarası</string> |     <string name="pref_api_items_number_title">Yüklenen öğe numarası</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Kaydırma üzerine daha fazla makale yükleyin</string> |     <string name="pref_general_infinite_loading_title">Kaydırma üzerine daha fazla makale yükleyin</string> | ||||||
|     <string name="translation">Çeviri</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> |     <string name="cant_open_invalid_url">Öğe url geçersiz. Uygulama çökmeyeceği için bu sorunu çözmeye çalışıyorum.</string> | ||||||
|     <string name="drawer_report_bug">Hata bildir</string> |  | ||||||
|     <string name="items_number_should_be_number">Öğe sayısı bir tamsayı olmalıdır.</string> |     <string name="items_number_should_be_number">Öğe sayısı bir tamsayı olmalıdır.</string> | ||||||
|     <string name="reader_action_more">Daha fazlasını görüntüle</string> |  | ||||||
|     <string name="reader_action_open">Tarayıcıda aç</string> |     <string name="reader_action_open">Tarayıcıda aç</string> | ||||||
|     <string name="reader_action_share">Paylaş</string> |     <string name="reader_action_share">Paylaş</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"需要登录?"</string> |     <string name="withLoginSwitch">"需要登录?"</string> | ||||||
|     <string name="login_url_problem">"哎呀。您可能需要在网址的末尾添加一个 \"/\"。"</string> |     <string name="login_url_problem">"哎呀。您可能需要在网址的末尾添加一个 \"/\"。"</string> | ||||||
|     <string name="prompt_login">"用户名"</string> |     <string name="prompt_login">"用户名"</string> | ||||||
|     <string name="label_share">"分享"</string> |  | ||||||
|     <string name="readAll">"全部阅读"</string> |     <string name="readAll">"全部阅读"</string> | ||||||
|     <string name="action_disconnect">"断开连接"</string> |     <string name="action_disconnect">"断开连接"</string> | ||||||
|     <string name="title_activity_settings">"设置"</string> |     <string name="title_activity_settings">"设置"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">卡片高度将根据内容调整</string> |     <string name="card_height_on">卡片高度将根据内容调整</string> | ||||||
|     <string name="card_height_off">卡片高度将被固定</string> |     <string name="card_height_off">卡片高度将被固定</string> | ||||||
|     <string name="source_code">源代码</string> |     <string name="source_code">源代码</string> | ||||||
|     <string name="drawer_error_loading_tags">加载标记时出错..。</string> |     <string name="filter_item_tags">标签</string> | ||||||
|     <string name="drawer_item_filters">搜索条件</string> |     <string name="filter_item_sources">来源</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="drawer_loading">正在载入…</string> |  | ||||||
|     <string name="menu_home_search">搜索</string> |     <string name="menu_home_search">搜索</string> | ||||||
|     <string name="can_delete_source">无法删除数据源…</string> |     <string name="can_delete_source">无法删除数据源…</string> | ||||||
|     <string name="base_url_error">与您的 Selfoss 通信时出现问题。如果问题一直存在,请与我联系。</string> |     <string name="base_url_error">与您的 Selfoss 通信时出现问题。如果问题一直存在,请与我联系。</string> | ||||||
|     <string name="pref_header_theme">主题</string> |     <string name="pref_header_theme">主题</string> | ||||||
|     <string name="default_theme">默认</string> |  | ||||||
|     <string name="default_dark_theme">默认值/暗</string> |  | ||||||
|     <string name="pref_selfoss_category">塞尔福斯 Api</string> |     <string name="pref_selfoss_category">塞尔福斯 Api</string> | ||||||
|     <string name="pref_api_items_number_title">已加载项目编号</string> |     <string name="pref_api_items_number_title">已加载项目编号</string> | ||||||
|     <string name="pref_hidden_tags">隐藏标签</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">翻页时载入更多文章</string> |     <string name="pref_general_infinite_loading_title">翻页时载入更多文章</string> | ||||||
|     <string name="translation">翻译</string> |     <string name="translation">翻译</string> | ||||||
|     <string name="cant_open_invalid_url">项目链接地址无效。我正在设法解决这个问题,以避免应用程序崩溃。</string> |     <string name="cant_open_invalid_url">项目链接地址无效。我正在设法解决这个问题,以避免应用程序崩溃。</string> | ||||||
|     <string name="drawer_report_bug">报告错误</string> |  | ||||||
|     <string name="items_number_should_be_number">项目数应为整数。</string> |     <string name="items_number_should_be_number">项目数应为整数。</string> | ||||||
|     <string name="reader_action_more">阅读更多</string> |  | ||||||
|     <string name="reader_action_open">在浏览器中打开</string> |     <string name="reader_action_open">在浏览器中打开</string> | ||||||
|     <string name="reader_action_share">分享</string> |     <string name="reader_action_share">分享</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">切换文章时将文章标记为已读。</string> |     <string name="pref_switch_actions_pager_scroll_on">切换文章时将文章标记为已读。</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">这将标记所有项目为已读。</string> |     <string name="markall_dialog_message">这将标记所有项目为已读。</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">滑动时标为已读</string> |     <string name="pref_switch_actions_pager_scroll">滑动时标为已读</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">滑动时不标记文章为已读</string> |     <string name="pref_switch_actions_pager_scroll_off">滑动时不标记文章为已读</string> | ||||||
|     <string name="drawer_item_hidden_tags">隐藏标签</string> |  | ||||||
|     <string name="unmark">标记条目为未读</string> |     <string name="unmark">标记条目为未读</string> | ||||||
|     <string name="pref_header_offline">离线和缓存</string> |     <string name="pref_header_offline">离线和缓存</string> | ||||||
|     <string name="pref_switch_items_caching_off">文章不会被保存到设备内存,应用程序在离线时将无法阅读它们</string> |     <string name="pref_switch_items_caching_off">文章不会被保存到设备内存,应用程序在离线时将无法阅读它们</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">将定期同步文章</string> |     <string name="pref_switch_periodic_refresh_on">将定期同步文章</string> | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[同步间隔 (>= 15分钟)]]></string> |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[同步间隔 (>= 15分钟)]]></string> | ||||||
|     <string name="pref_switch_refresh_when_charging">仅在手机充电时刷新</string> |     <string name="pref_switch_refresh_when_charging">仅在手机充电时刷新</string> | ||||||
|     <string name="loading_notification_title">加载中...</string> |     <string name="loading_notification_title">加载中…</string> | ||||||
|     <string name="loading_notification_text">Selfoss 正在同步您的文章</string> |     <string name="loading_notification_text">Selfoss 正在同步您的文章</string> | ||||||
|     <string name="notification_channel_sync">同步通知</string> |     <string name="notification_channel_sync">同步通知</string> | ||||||
|     <string name="new_items_channel_sync">新条目通知</string> |     <string name="new_items_channel_sync">新条目通知</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">深色模式</string> |     <string name="mode_dark">深色模式</string> | ||||||
|     <string name="mode_system">遵循系统设置</string> |     <string name="mode_system">遵循系统设置</string> | ||||||
|     <string name="mode_light">浅色模式</string> |     <string name="mode_light">浅色模式</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">该应用不分享任何关于您的个人数据。</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[崩溃报告发送现已启用。 可以从设置页面禁用它。 请记住,崩溃报告对于应用程序开发是必需的。]]></string> | ||||||
|  |     <string name="crash_toast_text">发生崩溃。请将细节发送给开发人员。</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"禁用自动错误报告 "</string> | ||||||
|  |     <string name="menu_home_filter">筛选器</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ | |||||||
|     <string name="withLoginSwitch">"需要登入?"</string> |     <string name="withLoginSwitch">"需要登入?"</string> | ||||||
|     <string name="login_url_problem">"哎呀。您可能需要在网址的末尾添加一个 \"/\"。"</string> |     <string name="login_url_problem">"哎呀。您可能需要在网址的末尾添加一个 \"/\"。"</string> | ||||||
|     <string name="prompt_login">"使用者名稱"</string> |     <string name="prompt_login">"使用者名稱"</string> | ||||||
|     <string name="label_share">"分享"</string> |  | ||||||
|     <string name="readAll">"全部阅读"</string> |     <string name="readAll">"全部阅读"</string> | ||||||
|     <string name="action_disconnect">"断开连接"</string> |     <string name="action_disconnect">"断开连接"</string> | ||||||
|     <string name="title_activity_settings">"设置"</string> |     <string name="title_activity_settings">"设置"</string> | ||||||
| @@ -62,28 +61,18 @@ | |||||||
|     <string name="card_height_on">卡片高度将根据内容调整</string> |     <string name="card_height_on">卡片高度将根据内容调整</string> | ||||||
|     <string name="card_height_off">卡片高度将被固定</string> |     <string name="card_height_off">卡片高度将被固定</string> | ||||||
|     <string name="source_code">源代码</string> |     <string name="source_code">源代码</string> | ||||||
|     <string name="drawer_error_loading_tags">加载标记时出错..。</string> |     <string name="filter_item_tags">标签</string> | ||||||
|     <string name="drawer_item_filters">搜索条件</string> |     <string name="filter_item_sources">来源</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="drawer_loading">正在载入…</string> |  | ||||||
|     <string name="menu_home_search">搜索</string> |     <string name="menu_home_search">搜索</string> | ||||||
|     <string name="can_delete_source">无法删除数据源…</string> |     <string name="can_delete_source">无法删除数据源…</string> | ||||||
|     <string name="base_url_error">与您的 Selfoss 通信时出现问题。如果问题一直存在,请与我联系。</string> |     <string name="base_url_error">与您的 Selfoss 通信时出现问题。如果问题一直存在,请与我联系。</string> | ||||||
|     <string name="pref_header_theme">主题</string> |     <string name="pref_header_theme">主题</string> | ||||||
|     <string name="default_theme">默认</string> |  | ||||||
|     <string name="default_dark_theme">默认值/暗</string> |  | ||||||
|     <string name="pref_selfoss_category">塞尔福斯 Api</string> |     <string name="pref_selfoss_category">塞尔福斯 Api</string> | ||||||
|     <string name="pref_api_items_number_title">已加载项目编号</string> |     <string name="pref_api_items_number_title">已加载项目编号</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">翻页时载入更多文章</string> |     <string name="pref_general_infinite_loading_title">翻页时载入更多文章</string> | ||||||
|     <string name="translation">翻译</string> |     <string name="translation">翻译</string> | ||||||
|     <string name="cant_open_invalid_url">项目链接地址无效。我正在设法解决这个问题,以避免应用程序崩溃。</string> |     <string name="cant_open_invalid_url">项目链接地址无效。我正在设法解决这个问题,以避免应用程序崩溃。</string> | ||||||
|     <string name="drawer_report_bug">报告错误</string> |  | ||||||
|     <string name="items_number_should_be_number">项目数应为整数。</string> |     <string name="items_number_should_be_number">项目数应为整数。</string> | ||||||
|     <string name="reader_action_more">阅读更多</string> |  | ||||||
|     <string name="reader_action_open">在浏览器中打开</string> |     <string name="reader_action_open">在浏览器中打开</string> | ||||||
|     <string name="reader_action_share">分享</string> |     <string name="reader_action_share">分享</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -94,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -109,7 +97,7 @@ | |||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <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_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="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="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|     <string name="notification_channel_sync">Sync notification</string> |     <string name="notification_channel_sync">Sync notification</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">New items notification</string> | ||||||
| @@ -132,5 +120,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -1,17 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <resources> |  | ||||||
|     <array name="com_google_android_gms_fonts_certs"> |  | ||||||
|         <item>@array/com_google_android_gms_fonts_certs_dev</item> |  | ||||||
|         <item>@array/com_google_android_gms_fonts_certs_prod</item> |  | ||||||
|     </array> |  | ||||||
|     <string-array name="com_google_android_gms_fonts_certs_dev"> |  | ||||||
|         <item> |  | ||||||
|             MIIEqDCCA5CgAwIBAgIJANWFuGx90071MA0GCSqGSIb3DQEBBAUAMIGUMQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTEWMBQGA1UEBxMNTW91bnRhaW4gVmlldzEQMA4GA1UEChMHQW5kcm9pZDEQMA4GA1UECxMHQW5kcm9pZDEQMA4GA1UEAxMHQW5kcm9pZDEiMCAGCSqGSIb3DQEJARYTYW5kcm9pZEBhbmRyb2lkLmNvbTAeFw0wODA0MTUyMzM2NTZaFw0zNTA5MDEyMzM2NTZaMIGUMQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTEWMBQGA1UEBxMNTW91bnRhaW4gVmlldzEQMA4GA1UEChMHQW5kcm9pZDEQMA4GA1UECxMHQW5kcm9pZDEQMA4GA1UEAxMHQW5kcm9pZDEiMCAGCSqGSIb3DQEJARYTYW5kcm9pZEBhbmRyb2lkLmNvbTCCASAwDQYJKoZIhvcNAQEBBQADggENADCCAQgCggEBANbOLggKv+IxTdGNs8/TGFy0PTP6DHThvbbR24kT9ixcOd9W+EaBPWW+wPPKQmsHxajtWjmQwWfna8mZuSeJS48LIgAZlKkpFeVyxW0qMBujb8X8ETrWy550NaFtI6t9+u7hZeTfHwqNvacKhp1RbE6dBRGWynwMVX8XW8N1+UjFaq6GCJukT4qmpN2afb8sCjUigq0GuMwYXrFVee74bQgLHWGJwPmvmLHC69EH6kWr22ijx4OKXlSIx2xT1AsSHee70w5iDBiK4aph27yH3TxkXy9V89TDdexAcKk/cVHYNnDBapcavl7y0RiQ4biu8ymM8Ga/nmzhRKya6G0cGw8CAQOjgfwwgfkwHQYDVR0OBBYEFI0cxb6VTEM8YYY6FbBMvAPyT+CyMIHJBgNVHSMEgcEwgb6AFI0cxb6VTEM8YYY6FbBMvAPyT+CyoYGapIGXMIGUMQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTEWMBQGA1UEBxMNTW91bnRhaW4gVmlldzEQMA4GA1UEChMHQW5kcm9pZDEQMA4GA1UECxMHQW5kcm9pZDEQMA4GA1UEAxMHQW5kcm9pZDEiMCAGCSqGSIb3DQEJARYTYW5kcm9pZEBhbmRyb2lkLmNvbYIJANWFuGx90071MAwGA1UdEwQFMAMBAf8wDQYJKoZIhvcNAQEEBQADggEBABnTDPEF+3iSP0wNfdIjIz1AlnrPzgAIHVvXxunW7SBrDhEglQZBbKJEk5kT0mtKoOD1JMrSu1xuTKEBahWRbqHsXclaXjoBADb0kkjVEJu/Lh5hgYZnOjvlba8Ld7HCKePCVePoTJBdI4fvugnL8TsgK05aIskyY0hKI9L8KfqfGTl1lzOv2KoWD0KWwtAWPoGChZxmQ+nBli+gwYMzM1vAkP+aayLe0a1EQimlOalO762r0GXO0ks+UeXde2Z4e+8S/pf7pITEI/tP+MxJTALw9QUWEv9lKTk+jkbqxbsh8nfBUapfKqYn0eidpwq2AzVp3juYl7//fKnaPhJD9gs= |  | ||||||
|         </item> |  | ||||||
|     </string-array> |  | ||||||
|     <string-array name="com_google_android_gms_fonts_certs_prod"> |  | ||||||
|         <item> |  | ||||||
|             MIIEQzCCAyugAwIBAgIJAMLgh0ZkSjCNMA0GCSqGSIb3DQEBBAUAMHQxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpDYWxpZm9ybmlhMRYwFAYDVQQHEw1Nb3VudGFpbiBWaWV3MRQwEgYDVQQKEwtHb29nbGUgSW5jLjEQMA4GA1UECxMHQW5kcm9pZDEQMA4GA1UEAxMHQW5kcm9pZDAeFw0wODA4MjEyMzEzMzRaFw0zNjAxMDcyMzEzMzRaMHQxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpDYWxpZm9ybmlhMRYwFAYDVQQHEw1Nb3VudGFpbiBWaWV3MRQwEgYDVQQKEwtHb29nbGUgSW5jLjEQMA4GA1UECxMHQW5kcm9pZDEQMA4GA1UEAxMHQW5kcm9pZDCCASAwDQYJKoZIhvcNAQEBBQADggENADCCAQgCggEBAKtWLgDYO6IIrgqWbxJOKdoR8qtW0I9Y4sypEwPpt1TTcvZApxsdyxMJZ2JORland2qSGT2y5b+3JKkedxiLDmpHpDsz2WCbdxgxRczfey5YZnTJ4VZbH0xqWVW/8lGmPav5xVwnIiJS6HXk+BVKZF+JcWjAsb/GEuq/eFdpuzSqeYTcfi6idkyugwfYwXFU1+5fZKUaRKYCwkkFQVfcAs1fXA5V+++FGfvjJ/CxURaSxaBvGdGDhfXE28LWuT9ozCl5xw4Yq5OGazvV24mZVSoOO0yZ31j7kYvtwYK6NeADwbSxDdJEqO4k//0zOHKrUiGYXtqw/A0LFFtqoZKFjnkCAQOjgdkwgdYwHQYDVR0OBBYEFMd9jMIhF1Ylmn/Tgt9r45jk14alMIGmBgNVHSMEgZ4wgZuAFMd9jMIhF1Ylmn/Tgt9r45jk14aloXikdjB0MQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTEWMBQGA1UEBxMNTW91bnRhaW4gVmlldzEUMBIGA1UEChMLR29vZ2xlIEluYy4xEDAOBgNVBAsTB0FuZHJvaWQxEDAOBgNVBAMTB0FuZHJvaWSCCQDC4IdGZEowjTAMBgNVHRMEBTADAQH/MA0GCSqGSIb3DQEBBAUAA4IBAQBt0lLO74UwLDYKqs6Tm8/yzKkEu116FmH4rkaymUIE0P9KaMftGlMexFlaYjzmB2OxZyl6euNXEsQH8gjwyxCUKRJNexBiGcCEyj6z+a1fuHHvkiaai+KL8W1EyNmgjmyy8AW7P+LLlkR+ho5zEHatRbM/YAnqGcFh5iZBqpknHf1SKMXFh4dd239FJ1jWYfbMDMy3NS5CTMQ2XFI1MvcyUTdZPErjQfTbQe3aDQsQcafEQPD+nqActifKZ0Np0IS9L9kR/wbNvyz6ENwPiTrjV2KRkEjH78ZMcUQXg0L3BYHJ3lc69Vs5Ddf9uUGGMYldX3WfMBEmh/9iFBDAaTCK |  | ||||||
|         </item> |  | ||||||
|     </string-array> |  | ||||||
| </resources> |  | ||||||
| @@ -9,24 +9,6 @@ | |||||||
|     <string-array name="ModeValues"> |     <string-array name="ModeValues"> | ||||||
|         <item>1</item> <!--MODE_NIGHT_NO--> |         <item>1</item> <!--MODE_NIGHT_NO--> | ||||||
|         <item>2</item> <!--MODE_NIGHT_YES--> |         <item>2</item> <!--MODE_NIGHT_YES--> | ||||||
|         <item>0</item> <!--MODE_NIGHT_AUTO_TIME--> |         <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> |     </string-array> | ||||||
| </resources> | </resources> | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> |  | ||||||
| <resources> |  | ||||||
|     <array name="preloaded_fonts" translatable="false"> |  | ||||||
|         <item>@font/open_sans</item> |  | ||||||
|         <item>@font/roboto</item> |  | ||||||
|     </array> |  | ||||||
| </resources> |  | ||||||
| @@ -4,5 +4,6 @@ | |||||||
|         <item></item> |         <item></item> | ||||||
|         <item>@string/open_sans_font_id</item> |         <item>@string/open_sans_font_id</item> | ||||||
|         <item>@string/roboto_font_id</item> |         <item>@string/roboto_font_id</item> | ||||||
|  |         <item>@string/source_code_pro_font_id</item> | ||||||
|     </array> |     </array> | ||||||
| </resources> | </resources> | ||||||
| @@ -4,5 +4,6 @@ | |||||||
|         <item>Systems</item> |         <item>Systems</item> | ||||||
|         <item>Open Sans</item> |         <item>Open Sans</item> | ||||||
|         <item>Roboto</item> |         <item>Roboto</item> | ||||||
|  |         <item>Source Code Pro</item> | ||||||
|     </array> |     </array> | ||||||
| </resources> | </resources> | ||||||
| @@ -9,7 +9,6 @@ | |||||||
|     <string name="withLoginSwitch">"Login required ?"</string> |     <string name="withLoginSwitch">"Login required ?"</string> | ||||||
|     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> |     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> | ||||||
|     <string name="prompt_login">"Username"</string> |     <string name="prompt_login">"Username"</string> | ||||||
|     <string name="label_share">"Share"</string> |  | ||||||
|     <string name="readAll">"Read all"</string> |     <string name="readAll">"Read all"</string> | ||||||
|     <string name="action_disconnect">"Disconnect"</string> |     <string name="action_disconnect">"Disconnect"</string> | ||||||
|     <string name="title_activity_settings">"Settings"</string> |     <string name="title_activity_settings">"Settings"</string> | ||||||
| @@ -62,29 +61,18 @@ | |||||||
|     <string name="card_height_on">Cards height will adjust to its content</string> |     <string name="card_height_on">Cards height will adjust to its content</string> | ||||||
|     <string name="card_height_off">Card height will be fixed</string> |     <string name="card_height_off">Card height will be fixed</string> | ||||||
|     <string name="source_code">Source code</string> |     <string name="source_code">Source code</string> | ||||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> |     <string name="filter_item_tags">Tags</string> | ||||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> |     <string name="filter_item_sources">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="drawer_loading">Loading …</string> |  | ||||||
|     <string name="menu_home_search">Search</string> |     <string name="menu_home_search">Search</string> | ||||||
|     <string name="can_delete_source">Can\'t delete the source…</string> |     <string name="can_delete_source">Can\'t delete the source…</string> | ||||||
|     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> |     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||||
|     <string name="pref_header_theme">Themes</string> |     <string name="pref_header_theme">Themes</string> | ||||||
|     <string name="default_theme">Default</string> |  | ||||||
|     <string name="default_dark_theme">Default/Dark</string> |  | ||||||
|     <string name="pref_selfoss_category">Selfoss Api</string> |     <string name="pref_selfoss_category">Selfoss Api</string> | ||||||
|     <string name="pref_api_items_number_title">Loaded items number</string> |     <string name="pref_api_items_number_title">Loaded items number</string> | ||||||
|     <string name="pref_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> |     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||||
|     <string name="translation">Translation</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> |     <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> | ||||||
|     <string name="drawer_report_bug">Report a bug</string> |  | ||||||
|     <string name="items_number_should_be_number">The items number should be an integer.</string> |     <string name="items_number_should_be_number">The items number should be an integer.</string> | ||||||
|     <string name="reader_action_more">Read more</string> |  | ||||||
|     <string name="reader_action_open">Open in browser</string> |     <string name="reader_action_open">Open in browser</string> | ||||||
|     <string name="reader_action_share">Share</string> |     <string name="reader_action_share">Share</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
| @@ -95,7 +83,6 @@ | |||||||
|     <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">This will mark all the items as read.</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |  | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">Mark item as unread</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
| @@ -118,13 +105,14 @@ | |||||||
|     <string name="new_items_notification_text">%1$d new items loaded.</string> |     <string name="new_items_notification_text">%1$d new items loaded.</string> | ||||||
|     <string name="pref_switch_notify_new_items">Notify on new items synced.</string> |     <string name="pref_switch_notify_new_items">Notify on new items synced.</string> | ||||||
|     <string name="shortcut_offline">Offline</string> |     <string name="shortcut_offline">Offline</string> | ||||||
|     <string name="pref_api_timeout">Api Timeout</string> |     <string name="pref_api_timeout">Api Timeout (seconds)</string> | ||||||
|     <string name="pref_header_experimental">Experimental</string> |     <string name="pref_header_experimental">Experimental</string> | ||||||
|     <string name="webview_dialog_issue_message">Webview not available. Disabling the article viewer to avoid any future crashes. Will load articles inside of your browser from now on.</string> |     <string name="webview_dialog_issue_message">Webview not available. Disabling the article viewer to avoid any future crashes. Will load articles inside of your browser from now on.</string> | ||||||
|     <string name="webview_dialog_issue_title">Webview issue</string> |     <string name="webview_dialog_issue_title">Webview issue</string> | ||||||
|     <string name="reader_text_align_left">Align left</string> |     <string name="reader_text_align_left">Align left</string> | ||||||
|     <string name="reader_text_align_justify">Justify</string> |     <string name="reader_text_align_justify">Justify</string> | ||||||
|     <string name="settings_reader_font">Reader font</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="open_sans_font_id" translatable="false">open_sans</string> | ||||||
|     <string name="roboto_font_id" translatable="false">roboto</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_title">Static bottom bar in the article viewer</string> | ||||||
| @@ -135,4 +123,10 @@ | |||||||
|     <string name="mode_dark">Dark mode</string> |     <string name="mode_dark">Dark mode</string> | ||||||
|     <string name="mode_system">Follow the system setting</string> |     <string name="mode_system">Follow the system setting</string> | ||||||
|     <string name="mode_light">Light mode</string> |     <string name="mode_light">Light mode</string> | ||||||
|  |     <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> | ||||||
|  |     <string name="gdpr_dialog_message"><![CDATA[Crash reports sending is now enabled. It can be disabled from the settings page. Keep in mind that crash reports are essential for the app development.]]></string> | ||||||
|  |     <string name="crash_toast_text">A crash occured. Sending the details to the developper.</string> | ||||||
|  |     <string name="pref_switch_disable_acra">"Disable automatic bug reporting. "</string> | ||||||
|  |     <string name="menu_home_filter">Filters</string> | ||||||
|  |     <string name="application_selfoss_only">This app only works with a Selfoss instance, and no other RSS feed.</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -25,7 +25,5 @@ | |||||||
|         <item name="android:textColorPrimary">@color/white</item> |         <item name="android:textColorPrimary">@color/white</item> | ||||||
|         <item name="android:textColorSecondary">@color/white</item> |         <item name="android:textColorSecondary">@color/white</item> | ||||||
|         <item name="actionMenuTextColor">@color/white</item> |         <item name="actionMenuTextColor">@color/white</item> | ||||||
|         <!--<item name="actionOverflowButtonStyle">@style/ActionButtonOverflowStyle</item> |  | ||||||
|         <item name="drawerArrowStyle">@style/DrawerArrowStyle</item>--> |  | ||||||
|     </style> |     </style> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" | <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|     xmlns:app="http://schemas.android.com/apk/res-auto"> |     xmlns:app="http://schemas.android.com/apk/res-auto"> | ||||||
|  |  | ||||||
|     <EditTextPreference |     <EditTextPreference | ||||||
|         android:inputType="number" |         android:inputType="number" | ||||||
|         android:key="api_timeout" |         android:key="api_timeout" | ||||||
|   | |||||||
| @@ -14,15 +14,6 @@ | |||||||
|         android:title="@string/pref_api_items_number_title" |         android:title="@string/pref_api_items_number_title" | ||||||
|         app:iconSpaceReserved="false"/> |         app:iconSpaceReserved="false"/> | ||||||
|  |  | ||||||
|     <EditTextPreference |  | ||||||
|         android:defaultValue="" |  | ||||||
|         android:hint="@string/add_source_hint_tags" |  | ||||||
|         android:key="hidden_tags" |  | ||||||
|         android:selectAllOnFocus="true" |  | ||||||
|         android:singleLine="true" |  | ||||||
|         android:title="@string/pref_hidden_tags" |  | ||||||
|         app:iconSpaceReserved="false"/> |  | ||||||
|  |  | ||||||
|     <SwitchPreference |     <SwitchPreference | ||||||
|         android:defaultValue="false" |         android:defaultValue="false" | ||||||
|         android:key="infinite_loading" |         android:key="infinite_loading" | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" | <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|     android:title="@string/title_activity_settings"> |     android:title="@string/title_activity_settings"> | ||||||
|  |  | ||||||
|     <Preference |     <Preference | ||||||
| @@ -17,9 +18,13 @@ | |||||||
|         android:title="@string/pref_header_offline" |         android:title="@string/pref_header_offline" | ||||||
|         android:icon="@drawable/ic_signal_wifi_off_black_24dp" /> |         android:icon="@drawable/ic_signal_wifi_off_black_24dp" /> | ||||||
|  |  | ||||||
|     <Preference |     <ListPreference | ||||||
|         android:fragment="bou.amine.apps.readerforselfossv2.android.settings.SettingsActivity$ThemePreferenceFragment" |         android:defaultValue="0" | ||||||
|  |         android:entries="@array/ModeTitles" | ||||||
|  |         android:entryValues="@array/ModeValues" | ||||||
|  |         android:key="currentMode" | ||||||
|         android:title="@string/pref_header_theme" |         android:title="@string/pref_header_theme" | ||||||
|  |         app:useSimpleSummaryProvider="false" | ||||||
|         android:icon="@drawable/ic_color_lens_black_24dp" /> |         android:icon="@drawable/ic_color_lens_black_24dp" /> | ||||||
|  |  | ||||||
|     <Preference |     <Preference | ||||||
| @@ -32,4 +37,18 @@ | |||||||
|         android:title="@string/pref_header_experimental" |         android:title="@string/pref_header_experimental" | ||||||
|         android:icon="@drawable/ic_widgets_black_24dp" /> |         android:icon="@drawable/ic_widgets_black_24dp" /> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     <SwitchPreference | ||||||
|  |         android:defaultValue="false" | ||||||
|  |         android:key="acra.disable" | ||||||
|  |         android:title="@string/pref_switch_disable_acra" | ||||||
|  |         android:icon="@drawable/ic_baseline_bug_report_24"/> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     <Preference | ||||||
|  |         android:key="action_about" | ||||||
|  |         android:title="@string/action_about" | ||||||
|  |         android:icon="@drawable/ic_info_outline_white_24dp" /> | ||||||
|  |  | ||||||
| </PreferenceScreen> | </PreferenceScreen> | ||||||
							
								
								
									
										46
									
								
								androidApp/src/test/kotlin/DatesTest.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								androidApp/src/test/kotlin/DatesTest.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | 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" | ||||||
|  |     private val bug1Date = "2022-12-24T17:00:08+00" | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun v3_date_should_be_parsed() { | ||||||
|  |         val date = DateUtils.parseDate(v3Date) | ||||||
|  |         val expected = | ||||||
|  |             LocalDateTime(2013, 4, 7, 14, 43, 0, 0).toInstant(TimeZone.currentSystemDefault()) | ||||||
|  |                 .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) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     fun bug1_date_should_be_parsed() { | ||||||
|  |         val date = DateUtils.parseDate(bug1Date) | ||||||
|  |         val expected = | ||||||
|  |             LocalDateTime(2022, 12, 24, 18, 0, 8, 0).toInstant(TimeZone.currentSystemDefault()) | ||||||
|  |                 .toEpochMilliseconds() | ||||||
|  |  | ||||||
|  |         assertEquals(date, expected) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										1042
									
								
								androidApp/src/test/kotlin/RepositoryTest.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1042
									
								
								androidApp/src/test/kotlin/RepositoryTest.kt
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										60
									
								
								androidApp/src/test/kotlin/TestUtils.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								androidApp/src/test/kotlin/TestUtils.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | |||||||
|  | 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, | ||||||
|  |             author = item.author | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | 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(','), | ||||||
|  |             author = item.author | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | 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" | ||||||
|  |     var author = "Someone important" | ||||||
|  | } | ||||||
| @@ -1,32 +1,36 @@ | |||||||
| buildscript { | buildscript { | ||||||
|     repositories { |  | ||||||
|         gradlePluginPortal() |  | ||||||
|         google() |  | ||||||
|         mavenCentral() |  | ||||||
|     } |  | ||||||
|     dependencies { |     dependencies { | ||||||
|         classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.6.10") |  | ||||||
|         classpath("com.android.tools.build:gradle:7.3.0") |  | ||||||
|  |  | ||||||
|         // sonarquve |  | ||||||
|         classpath("org.sonarsource.scanner.gradle:sonarqube-gradle-plugin:3.4.0.2513") |  | ||||||
|  |  | ||||||
|         // SqlDelight |         // SqlDelight | ||||||
|         classpath("com.squareup.sqldelight:gradle-plugin:1.5.3") |         classpath("com.squareup.sqldelight:gradle-plugin:1.5.4") | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| apply(plugin = "org.sonarqube") | 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("com.mikepenz.aboutlibraries.plugin").version("10.5.1").apply(false) | ||||||
|  |     id("org.jetbrains.kotlinx.kover") version "0.6.1" | ||||||
|  | } | ||||||
|  |  | ||||||
| allprojects { | allprojects { | ||||||
|     repositories { |     repositories { | ||||||
|         google() |         maven { url = uri("https://nexus.amine-louveau.fr/repository/maven-public/")} | ||||||
|         mavenCentral() |         // IMPORTANT : Add back when new library added | ||||||
|         jcenter() |         // google() | ||||||
|         maven { url = uri("https://www.jitpack.io") } |         // mavenCentral() | ||||||
|  |         // jcenter() | ||||||
|  |         // maven { url = uri("https://www.jitpack.io") } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| tasks.register("clean", Delete::class) { | tasks.register("clean", Delete::class) { | ||||||
|     delete(rootProject.buildDir) |     delete(rootProject.buildDir) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | koverMerged { | ||||||
|  |     enable() | ||||||
|  | } | ||||||
| @@ -11,14 +11,26 @@ | |||||||
| # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects | ||||||
| # org.gradle.parallel=true | # org.gradle.parallel=true | ||||||
| #Tue Mar 22 16:50:00 CET 2022 | #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.code.style=official | ||||||
| kotlin.mpp.enableCInteropCommonization=true |  | ||||||
| org.gradle.jvmargs=-Xmx2048M -Dkotlin.daemon.jvm.options\="-Xmx2048M" | #Android | ||||||
| kotlin.native.enableDependencyPropagation=false |  | ||||||
| android.useAndroidX=true | android.useAndroidX=true | ||||||
|  | kotlin.native.enableDependencyPropagation=false | ||||||
|  | #android.nonTransitiveRClass=true | ||||||
| android.enableJetifier=true | android.enableJetifier=true | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #MPP | ||||||
|  | kotlin.mpp.enableCInteropCommonization=true | ||||||
| kotlin.mpp.enableGranularSourceSetsMetadata=true | kotlin.mpp.enableGranularSourceSetsMetadata=true | ||||||
|  |  | ||||||
|  |  | ||||||
| org.gradle.parallel=true | org.gradle.parallel=true | ||||||
| org.gradle.caching=true | org.gradle.caching=true | ||||||
| ignoreGitVersion=false | ignoreGitVersion=false | ||||||
|  | kotlin.native.cacheKind.iosX64=none | ||||||
| pushCache=true | pushCache=true | ||||||
|   | |||||||
| @@ -2,12 +2,9 @@ import SwiftUI | |||||||
| import shared | import shared | ||||||
|  |  | ||||||
| struct ContentView: View { | struct ContentView: View { | ||||||
|     let greet = Greeting().greeting() |  | ||||||
|  |  | ||||||
|     let toto = SelfossApi().getItems() |  | ||||||
|  |  | ||||||
| 	var body: some View { | 	var body: some View { | ||||||
| 		Text(greet) | 		Text("ototot") | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,12 +2,24 @@ val pushCache: String by settings | |||||||
|  |  | ||||||
| pluginManagement { | pluginManagement { | ||||||
|     repositories { |     repositories { | ||||||
|         google() |         maven { url = uri("https://nexus.amine-louveau.fr/repository/maven-public/")} | ||||||
|         gradlePluginPortal() |         // IMPORTANT : Add back when new plugin added | ||||||
|         mavenCentral() |         // google() | ||||||
|  |         // gradlePluginPortal() | ||||||
|  |         // mavenCentral() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | dependencyResolutionManagement { | ||||||
|  |     repositories { | ||||||
|  |         maven { url = uri("https://nexus.amine-louveau.fr/repository/maven-public/")} | ||||||
|  |         // IMPORTANT : Add back when new library added | ||||||
|  |         // google() | ||||||
|  |         // mavenCentral() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| buildCache { | buildCache { | ||||||
|     remote<HttpBuildCache> { |     remote<HttpBuildCache> { | ||||||
|         url = uri("http://18.0.0.7:3071/cache/") |         url = uri("http://18.0.0.7:3071/cache/") | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| object SqlDelight { | object SqlDelight { | ||||||
|     const val runtime = "com.squareup.sqldelight:runtime:1.5.3" |     const val runtime = "com.squareup.sqldelight:runtime:1.5.4" | ||||||
|     const val android = "com.squareup.sqldelight:android-driver:1.5.3" |     const val android = "com.squareup.sqldelight:android-driver:1.5.4" | ||||||
|     const val native = "com.squareup.sqldelight:native-driver:1.5.3" |     const val native = "com.squareup.sqldelight:native-driver:1.5.4" | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -10,6 +10,7 @@ plugins { | |||||||
|     id("com.android.library") |     id("com.android.library") | ||||||
|     id("com.squareup.sqldelight") |     id("com.squareup.sqldelight") | ||||||
|     kotlin("plugin.serialization") version "1.4.10" |     kotlin("plugin.serialization") version "1.4.10" | ||||||
|  |     id("org.jetbrains.kotlinx.kover") version "0.6.1" | ||||||
| } | } | ||||||
|  |  | ||||||
| kotlin { | kotlin { | ||||||
| @@ -18,7 +19,7 @@ kotlin { | |||||||
|     listOf( |     listOf( | ||||||
|         iosX64(), |         iosX64(), | ||||||
|         iosArm64(), |         iosArm64(), | ||||||
|         //iosSimulatorArm64() sure all ios dependencies support this target |         // iosSimulatorArm64() | ||||||
|     ).forEach { |     ).forEach { | ||||||
|         it.binaries.framework { |         it.binaries.framework { | ||||||
|             baseName = "shared" |             baseName = "shared" | ||||||
| @@ -40,14 +41,11 @@ kotlin { | |||||||
|                 implementation("org.kodein.di:kodein-di:7.12.0") |                 implementation("org.kodein.di:kodein-di:7.12.0") | ||||||
|  |  | ||||||
|                 //Settings |                 //Settings | ||||||
|                 implementation("com.russhwolf:multiplatform-settings-no-arg:0.9") |                 implementation("com.russhwolf:multiplatform-settings-no-arg:1.0.0-RC") | ||||||
|  |  | ||||||
|                 //Logging |                 //Logging | ||||||
|                 implementation("io.github.aakira:napier:2.6.1") |                 implementation("io.github.aakira:napier:2.6.1") | ||||||
|  |  | ||||||
|                 // Network information |  | ||||||
|                 implementation("com.github.ln-12:multiplatform-connectivity-status:1.3.0") |  | ||||||
|  |  | ||||||
|                 // Sql |                 // Sql | ||||||
|                 implementation(SqlDelight.runtime) |                 implementation(SqlDelight.runtime) | ||||||
|             } |             } | ||||||
| @@ -56,13 +54,12 @@ kotlin { | |||||||
|             dependencies { |             dependencies { | ||||||
|                 implementation(kotlin("test-common")) |                 implementation(kotlin("test-common")) | ||||||
|                 implementation(kotlin("test-annotations-common")) |                 implementation(kotlin("test-annotations-common")) | ||||||
|                 implementation("io.mockk:mockk:1.12.0") |  | ||||||
|                 implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.0") |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         val androidMain by getting { |         val androidMain by getting { | ||||||
|             dependencies { |             dependencies { | ||||||
|                 implementation("io.ktor:ktor-client-okhttp:2.1.1") |                 implementation("io.ktor:ktor-client-okhttp:2.1.1") | ||||||
|  |                 implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") | ||||||
|  |  | ||||||
|                 // Sql |                 // Sql | ||||||
|                 implementation(SqlDelight.android) |                 implementation(SqlDelight.android) | ||||||
| @@ -83,9 +80,9 @@ kotlin { | |||||||
|             iosArm64Main.dependsOn(this) |             iosArm64Main.dependsOn(this) | ||||||
|             // iosSimulatorArm64Main.dependsOn(this) |             // iosSimulatorArm64Main.dependsOn(this) | ||||||
|  |  | ||||||
|             // Sql |  | ||||||
|             dependencies { |             dependencies { | ||||||
|                 implementation(SqlDelight.native) |                 implementation(SqlDelight.native) | ||||||
|  |                 implementation("io.ktor:ktor-client-ios:2.1.1") | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         val iosX64Test by getting |         val iosX64Test by getting | ||||||
| @@ -95,20 +92,17 @@ kotlin { | |||||||
|             dependsOn(commonTest) |             dependsOn(commonTest) | ||||||
|             iosX64Test.dependsOn(this) |             iosX64Test.dependsOn(this) | ||||||
|             iosArm64Test.dependsOn(this) |             iosArm64Test.dependsOn(this) | ||||||
|             dependencies { |  | ||||||
|                 implementation("io.ktor:ktor-client-ios:2.1.1") |  | ||||||
|             } |  | ||||||
|             // iosSimulatorArm64Test.dependsOn(this) |             // iosSimulatorArm64Test.dependsOn(this) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| android { | android { | ||||||
|     compileSdk = 31 |     compileSdk = 32 | ||||||
|     sourceSets["main"].manifest.srcFile("src/androidMain/AndroidManifest.xml") |     sourceSets["main"].manifest.srcFile("src/androidMain/AndroidManifest.xml") | ||||||
|     defaultConfig { |     defaultConfig { | ||||||
|         minSdk = 21 |         minSdk = 21 | ||||||
|         targetSdk = 31 |         targetSdk = 32 | ||||||
|     } |     } | ||||||
|     compileOptions { |     compileOptions { | ||||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 |         sourceCompatibility = JavaVersion.VERSION_1_8 | ||||||
| @@ -117,10 +111,11 @@ android { | |||||||
|     namespace = "bou.amine.apps.readerforselfossv2" |     namespace = "bou.amine.apps.readerforselfossv2" | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| sqldelight { | sqldelight { | ||||||
|     database("ReaderForSelfossDB") { |     database("ReaderForSelfossDB") { | ||||||
|         packageName = "bou.amine.apps.readerforselfossv2.dao" |         packageName = "bou.amine.apps.readerforselfossv2.dao" | ||||||
|         sourceFolders = listOf("sqldelight") |         sourceFolders = listOf("sqldelight") | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,24 +1,20 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.utils | package bou.amine.apps.readerforselfossv2.utils | ||||||
|  |  | ||||||
| import android.text.format.DateUtils | import android.text.format.DateUtils | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import kotlinx.datetime.* | ||||||
| import java.time.Instant |  | ||||||
| import java.time.LocalDateTime |  | ||||||
| import java.time.OffsetDateTime |  | ||||||
| import java.time.ZoneOffset |  | ||||||
| import java.time.format.DateTimeFormatter |  | ||||||
|  |  | ||||||
| actual class DateUtils actual constructor(actual val appSettingsService: AppSettingsService) { |  | ||||||
|  |  | ||||||
|  | actual class DateUtils { | ||||||
|  |     actual companion object { | ||||||
|         actual fun parseDate(dateString: String): Long { |         actual fun parseDate(dateString: String): Long { | ||||||
|  |             return try { | ||||||
|         val FORMATTERV1 = "yyyy-MM-dd HH:mm:ss" |                 Instant.parse(dateString).toEpochMilliseconds() | ||||||
|  |             } catch (e: Exception) { | ||||||
|         return if (appSettingsService.getApiVersion() >= 4) { |                 var str = dateString.replace(" ", "T") | ||||||
|             OffsetDateTime.parse(dateString).toInstant().toEpochMilli() |                 if (str.matches("\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\+\\d{2}".toRegex())) { | ||||||
|         } else { |                     str = str.split("+")[0] | ||||||
|             LocalDateTime.parse(dateString, DateTimeFormatter.ofPattern(FORMATTERV1)).toInstant( |                 } | ||||||
|                 ZoneOffset.UTC).toEpochMilli() |                 LocalDateTime.parse(str).toInstant(TimeZone.currentSystemDefault()).toEpochMilliseconds() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -28,9 +24,10 @@ actual class DateUtils actual constructor(actual val appSettingsService: AppSett | |||||||
|  |  | ||||||
|             return " " + DateUtils.getRelativeTimeSpanString( |             return " " + DateUtils.getRelativeTimeSpanString( | ||||||
|                 date, |                 date, | ||||||
|             Instant.now().toEpochMilli(), |                 Clock.System.now().toEpochMilliseconds(), | ||||||
|                 DateUtils.MINUTE_IN_MILLIS, |                 DateUtils.MINUTE_IN_MILLIS, | ||||||
|                 DateUtils.FORMAT_ABBREV_RELATIVE |                 DateUtils.FORMAT_ABBREV_RELATIVE | ||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | } | ||||||
| @@ -1,5 +1,6 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.DI | 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.rest.SelfossApi | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
| import org.kodein.di.DI | import org.kodein.di.DI | ||||||
| @@ -10,4 +11,5 @@ import org.kodein.di.singleton | |||||||
| val networkModule by DI.Module { | val networkModule by DI.Module { | ||||||
|     bind<AppSettingsService>() with singleton { AppSettingsService() } |     bind<AppSettingsService>() with singleton { AppSettingsService() } | ||||||
|     bind<SelfossApi>() with singleton { SelfossApi(instance()) } |     bind<SelfossApi>() with singleton { SelfossApi(instance()) } | ||||||
|  |     bind<MercuryApi>() with singleton { MercuryApi() } | ||||||
| } | } | ||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.model | ||||||
|  |  | ||||||
|  | import kotlinx.serialization.Serializable | ||||||
|  |  | ||||||
|  | class MercuryModel { | ||||||
|  |  | ||||||
|  |     @Serializable | ||||||
|  |     class ParsedContent( | ||||||
|  |         val title: String, | ||||||
|  |         val content: String?, | ||||||
|  |         val lead_image_url: String?, | ||||||
|  |         val url: String | ||||||
|  |     ) | ||||||
|  | } | ||||||
| @@ -0,0 +1,21 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.model | ||||||
|  |  | ||||||
|  | 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) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -20,12 +20,6 @@ class SelfossModel { | |||||||
|         val unread: Int |         val unread: Int | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     @Serializable |  | ||||||
|     class SuccessResponse(val success: Boolean) { |  | ||||||
|         val isSuccess: Boolean |  | ||||||
|             get() = success |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Serializable |     @Serializable | ||||||
|     class Stats( |     class Stats( | ||||||
|         val total: Int, |         val total: Int, | ||||||
| @@ -63,7 +57,6 @@ class SelfossModel { | |||||||
|         val error: String, |         val error: String, | ||||||
|         val icon: String? |         val icon: String? | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     @Serializable |     @Serializable | ||||||
|     data class Item( |     data class Item( | ||||||
|         val id: Int, |         val id: Int, | ||||||
| @@ -79,7 +72,8 @@ class SelfossModel { | |||||||
|         val link: String, |         val link: String, | ||||||
|         val sourcetitle: String, |         val sourcetitle: String, | ||||||
|         @Serializable(with = TagsListSerializer::class) |         @Serializable(with = TagsListSerializer::class) | ||||||
|         val tags: List<String> |         val tags: List<String>, | ||||||
|  |         val author: String | ||||||
|     ) { |     ) { | ||||||
|         // TODO: maybe find a better way to handle these kind of urls |         // TODO: maybe find a better way to handle these kind of urls | ||||||
|         fun getLinkDecoded(): String { |         fun getLinkDecoded(): String { | ||||||
| @@ -108,8 +102,14 @@ class SelfossModel { | |||||||
|             return stringUrl |             return stringUrl | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         fun sourceAndDateText(dateUtils: DateUtils): String = |         fun sourceAuthorAndDate(): String { | ||||||
|             this.sourcetitle.getHtmlDecoded() + dateUtils.parseRelativeDate(this.datetime) |             var txt = this.sourcetitle.getHtmlDecoded() | ||||||
|  |             if (this.author.isNotEmpty()) { | ||||||
|  |                 txt += " (by ${this.author}) " | ||||||
|  |             } | ||||||
|  |             txt += DateUtils.parseRelativeDate(this.datetime) | ||||||
|  |             return txt | ||||||
|  |         } | ||||||
|  |  | ||||||
|         fun toggleStar(): Item { |         fun toggleStar(): Item { | ||||||
|             this.starred = !this.starred |             this.starred = !this.starred | ||||||
| @@ -117,6 +117,7 @@ class SelfossModel { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     // TODO: this seems to be super slow. |     // TODO: this seems to be super slow. | ||||||
|     object TagsListSerializer : KSerializer<List<String>> { |     object TagsListSerializer : KSerializer<List<String>> { | ||||||
|         override fun deserialize(decoder: Decoder): List<String> { |         override fun deserialize(decoder: Decoder): List<String> { | ||||||
| @@ -152,16 +153,4 @@ class SelfossModel { | |||||||
|             TODO("Not yet implemented") |             TODO("Not yet implemented") | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     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) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,75 +3,81 @@ package bou.amine.apps.readerforselfossv2.repository | |||||||
| import bou.amine.apps.readerforselfossv2.dao.* | import bou.amine.apps.readerforselfossv2.dao.* | ||||||
| import bou.amine.apps.readerforselfossv2.model.NetworkUnavailableException | import bou.amine.apps.readerforselfossv2.model.NetworkUnavailableException | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | 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.rest.SelfossApi | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
| import bou.amine.apps.readerforselfossv2.utils.* | import bou.amine.apps.readerforselfossv2.utils.* | ||||||
| import com.github.ln_12.library.ConnectivityStatus |  | ||||||
| import io.github.aakira.napier.Napier | import io.github.aakira.napier.Napier | ||||||
|  | import io.ktor.client.call.* | ||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| import kotlinx.coroutines.Dispatchers | import kotlinx.coroutines.Dispatchers | ||||||
|  | import kotlinx.coroutines.flow.MutableStateFlow | ||||||
|  | import kotlinx.coroutines.flow.asStateFlow | ||||||
| import kotlinx.coroutines.launch | import kotlinx.coroutines.launch | ||||||
| import kotlinx.coroutines.runBlocking |  | ||||||
|  |  | ||||||
| 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>() |     var items = ArrayList<SelfossModel.Item>() | ||||||
|     val isConnectionAvailable = connectivityStatus.isNetworkConnected |  | ||||||
|     var connectionMonitored = false |     var connectionMonitored = false | ||||||
|  |  | ||||||
|     var baseUrl = appSettingsService.getBaseUrl() |     var baseUrl = appSettingsService.getBaseUrl() | ||||||
|     lateinit var dateUtils: DateUtils |  | ||||||
|  |  | ||||||
|     var displayedItems = ItemType.UNREAD |     var displayedItems = ItemType.UNREAD | ||||||
|  |  | ||||||
|     var tagFilter: SelfossModel.Tag? = null |     private var _tagFilter = MutableStateFlow<SelfossModel.Tag?>(null) | ||||||
|     var sourceFilter: SelfossModel.Source? = null |     var tagFilter = _tagFilter.asStateFlow() | ||||||
|  |     private var _sourceFilter = MutableStateFlow<SelfossModel.Source?>(null) | ||||||
|  |     var sourceFilter = _sourceFilter.asStateFlow() | ||||||
|     var searchFilter: String? = null |     var searchFilter: String? = null | ||||||
|  |  | ||||||
|     var offlineOverride = false |     var offlineOverride = false | ||||||
|  |  | ||||||
|     var badgeUnread = 0 |     private val _badgeUnread = MutableStateFlow(0) | ||||||
|     set(value) {field = if (value < 0) { 0 } else { value } } |     val badgeUnread = _badgeUnread.asStateFlow() | ||||||
|     var badgeAll = 0 |     private val _badgeAll = MutableStateFlow(0) | ||||||
|     set(value) {field = if (value < 0) { 0 } else { value } } |     val badgeAll = _badgeAll.asStateFlow() | ||||||
|     var badgeStarred = 0 |     private val _badgeStarred = MutableStateFlow(0) | ||||||
|     set(value) {field = if (value < 0) { 0 } else { value } } |     val badgeStarred = _badgeStarred.asStateFlow() | ||||||
|  |  | ||||||
|     private var fetchedSources = false |     private var fetchedSources = false | ||||||
|     private var fetchedTags = false |     private var fetchedTags = false | ||||||
|  |  | ||||||
|     init { |     private var _readerItems = ArrayList<SelfossModel.Item>() | ||||||
|         // TODO: Dispatchers.IO not available in KMM, an alternative solution should be found |  | ||||||
|         connectivityStatus.start() |  | ||||||
|         runBlocking { |  | ||||||
|             updateApiVersion() |  | ||||||
|             dateUtils = DateUtils(appSettingsService) |  | ||||||
|             reloadBadges() |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     suspend fun getNewerItems(): ArrayList<SelfossModel.Item> { |     suspend fun getNewerItems(): ArrayList<SelfossModel.Item> { | ||||||
|         // TODO: Use the updatedSince parameter |         // TODO: Use the updatedSince parameter | ||||||
|         var fetchedItems: SelfossModel.StatusAndData<List<SelfossModel.Item>> = SelfossModel.StatusAndData.error() |         var fetchedItems: StatusAndData<List<SelfossModel.Item>> = StatusAndData.error() | ||||||
|         var fromDB = false |         var fromDB = false | ||||||
|         if (isNetworkAvailable()) { |         if (isNetworkAvailable()) { | ||||||
|             fetchedItems = api.getItems( |             fetchedItems = api.getItems( | ||||||
|                 displayedItems.type, |                 displayedItems.type, | ||||||
|                 offset = 0, |                 offset = 0, | ||||||
|                 tagFilter?.tag, |                 tagFilter.value?.tag, | ||||||
|                 sourceFilter?.id?.toLong(), |                 sourceFilter.value?.id?.toLong(), | ||||||
|                 searchFilter, |                 searchFilter, | ||||||
|                 null |                 null | ||||||
|             ) |             ) | ||||||
|         } else { |         } else { | ||||||
|             if (appSettingsService.isItemCachingEnabled()) { |             if (appSettingsService.isItemCachingEnabled()) { | ||||||
|                 fromDB = true |                 fromDB = true | ||||||
|                 fetchedItems = SelfossModel.StatusAndData.succes( |                 var dbItems = getDBItems().filter { | ||||||
|                     getDBItems().filter { |  | ||||||
|                     displayedItems == ItemType.ALL || |                     displayedItems == ItemType.ALL || | ||||||
|                             (it.unread && displayedItems == ItemType.UNREAD) || |                             (it.unread && displayedItems == ItemType.UNREAD) || | ||||||
|                             (it.starred && displayedItems == ItemType.STARRED) |                             (it.starred && displayedItems == ItemType.STARRED) | ||||||
|                     }.map { it.toView() } |                 } | ||||||
|  |                 if (tagFilter.value != null) { | ||||||
|  |                     dbItems = dbItems.filter { it.tags.split(',').contains(tagFilter.value!!.tag) } | ||||||
|  |                 } | ||||||
|  |                 if (sourceFilter.value != null) { | ||||||
|  |                     dbItems = dbItems.filter { it.sourcetitle == sourceFilter.value!!.title } | ||||||
|  |                 } | ||||||
|  |                 fetchedItems = StatusAndData.succes( | ||||||
|  |                     dbItems.map { it.toView() } | ||||||
|                 ) |                 ) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -79,21 +85,21 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|         if (fetchedItems.success && fetchedItems.data != null) { |         if (fetchedItems.success && fetchedItems.data != null) { | ||||||
|             items = ArrayList(fetchedItems.data!!) |             items = ArrayList(fetchedItems.data!!) | ||||||
|             if (fromDB) { |             if (fromDB) { | ||||||
|                 items.sortByDescending { dateUtils.parseDate(it.datetime) } |                 items.sortByDescending { DateUtils.parseDate(it.datetime) } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return items |         return items | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     suspend fun getOlderItems(): ArrayList<SelfossModel.Item> { |     suspend fun getOlderItems(): ArrayList<SelfossModel.Item> { | ||||||
|         var fetchedItems: SelfossModel.StatusAndData<List<SelfossModel.Item>> = SelfossModel.StatusAndData.error() |         var fetchedItems: StatusAndData<List<SelfossModel.Item>> = StatusAndData.error() | ||||||
|         if (isNetworkAvailable()) { |         if (isNetworkAvailable()) { | ||||||
|             val offset = items.size |             val offset = items.size | ||||||
|             fetchedItems = api.getItems( |             fetchedItems = api.getItems( | ||||||
|                 displayedItems.type, |                 displayedItems.type, | ||||||
|                 offset, |                 offset, | ||||||
|                 tagFilter?.tag, |                 tagFilter.value?.tag, | ||||||
|                 sourceFilter?.id?.toLong(), |                 sourceFilter.value?.id?.toLong(), | ||||||
|                 searchFilter, |                 searchFilter, | ||||||
|                 null |                 null | ||||||
|             ) |             ) | ||||||
| @@ -131,24 +137,25 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|         if (isNetworkAvailable()) { |         if (isNetworkAvailable()) { | ||||||
|             val response = api.stats() |             val response = api.stats() | ||||||
|             if (response.success && response.data != null) { |             if (response.success && response.data != null) { | ||||||
|                 badgeUnread = response.data.unread |                 _badgeUnread.value = response.data.unread | ||||||
|                 badgeAll = response.data.total |                 _badgeAll.value = response.data.total | ||||||
|                 badgeStarred = response.data.starred |                 _badgeStarred.value = response.data.starred | ||||||
|                 success = true |                 success = true | ||||||
|             } |             } | ||||||
|         } else if (appSettingsService.isItemCachingEnabled()) { |         } else if (appSettingsService.isItemCachingEnabled()) { | ||||||
|             // TODO: do this differently, because it's not efficient |             // TODO: do this differently, because it's not efficient | ||||||
|             val dbItems = getDBItems() |             val dbItems = getDBItems() | ||||||
|             badgeUnread = dbItems.filter { item -> item.unread }.size |             _badgeUnread.value = dbItems.filter { item -> item.unread }.size | ||||||
|             badgeStarred = dbItems.filter { item -> item.starred }.size |             _badgeStarred.value = dbItems.filter { item -> item.starred }.size | ||||||
|             badgeAll = dbItems.size |             _badgeAll.value = dbItems.size | ||||||
|             success = true |             success = true | ||||||
|         } |         } | ||||||
|         return success |         return success | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     suspend fun getTags(): List<SelfossModel.Tag> { |     suspend fun getTags(): List<SelfossModel.Tag> { | ||||||
|         val isDatabaseEnabled = appSettingsService.isItemCachingEnabled() || !appSettingsService.isUpdateSourcesEnabled() |         val isDatabaseEnabled = | ||||||
|  |             appSettingsService.isItemCachingEnabled() || !appSettingsService.isUpdateSourcesEnabled() | ||||||
|         return if (isNetworkAvailable() && !fetchedTags) { |         return if (isNetworkAvailable() && !fetchedTags) { | ||||||
|             val apiTags = api.tags() |             val apiTags = api.tags() | ||||||
|             if (apiTags.success && apiTags.data != null && isDatabaseEnabled) { |             if (apiTags.success && apiTags.data != null && isDatabaseEnabled) { | ||||||
| @@ -180,7 +187,8 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     suspend fun getSources(): ArrayList<SelfossModel.Source> { |     suspend fun getSources(): ArrayList<SelfossModel.Source> { | ||||||
|         val isDatabaseEnabled = appSettingsService.isItemCachingEnabled() || !appSettingsService.isUpdateSourcesEnabled() |         val isDatabaseEnabled = | ||||||
|  |             appSettingsService.isItemCachingEnabled() || !appSettingsService.isUpdateSourcesEnabled() | ||||||
|         return if (isNetworkAvailable() && !fetchedSources) { |         return if (isNetworkAvailable() && !fetchedSources) { | ||||||
|             val apiSources = api.sources() |             val apiSources = api.sources() | ||||||
|             if (apiSources.success && apiSources.data != null && isDatabaseEnabled) { |             if (apiSources.success && apiSources.data != null && isDatabaseEnabled) { | ||||||
| @@ -289,7 +297,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|     private fun markAsReadLocally(item: SelfossModel.Item) { |     private fun markAsReadLocally(item: SelfossModel.Item) { | ||||||
|         if (item.unread) { |         if (item.unread) { | ||||||
|             item.unread = false |             item.unread = false | ||||||
|             badgeUnread -= 1 |             _badgeUnread.value -= 1 | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
| @@ -300,7 +308,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|     private fun unmarkAsReadLocally(item: SelfossModel.Item) { |     private fun unmarkAsReadLocally(item: SelfossModel.Item) { | ||||||
|         if (!item.unread) { |         if (!item.unread) { | ||||||
|             item.unread = true |             item.unread = true | ||||||
|             badgeUnread += 1 |             _badgeUnread.value += 1 | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
| @@ -311,7 +319,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|     private fun starrLocally(item: SelfossModel.Item) { |     private fun starrLocally(item: SelfossModel.Item) { | ||||||
|         if (!item.starred) { |         if (!item.starred) { | ||||||
|             item.starred = true |             item.starred = true | ||||||
|             badgeStarred += 1 |             _badgeStarred.value += 1 | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
| @@ -322,7 +330,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|     private fun unstarrLocally(item: SelfossModel.Item) { |     private fun unstarrLocally(item: SelfossModel.Item) { | ||||||
|         if (item.starred) { |         if (item.starred) { | ||||||
|             item.starred = false |             item.starred = false | ||||||
|             badgeStarred -= 1 |             _badgeStarred.value -= 1 | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         CoroutineScope(Dispatchers.Main).launch { |         CoroutineScope(Dispatchers.Main).launch { | ||||||
| @@ -344,21 +352,27 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|                 url, |                 url, | ||||||
|                 spout, |                 spout, | ||||||
|                 tags, |                 tags, | ||||||
|                 filter, |                 filter | ||||||
|                 appSettingsService.getApiVersion() |  | ||||||
|             ).isSuccess == true |             ).isSuccess == true | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return response |         return response | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     suspend fun deleteSource(id: Int): Boolean { |     suspend fun deleteSource(id: Int, title: String): Boolean { | ||||||
|         var success = false |         var success = false | ||||||
|         if (isNetworkAvailable()) { |         if (isNetworkAvailable()) { | ||||||
|             val response = api.deleteSource(id) |             val response = api.deleteSource(id) | ||||||
|             success = response.isSuccess |             success = response.isSuccess | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         // We filter on success or if the network isn't available | ||||||
|  |         if (success || !isNetworkAvailable()) { | ||||||
|  |             items = ArrayList(items.filter { it.sourcetitle != title }) | ||||||
|  |             setReaderItems(items) | ||||||
|  |             db.itemsQueries.deleteItemsWhereSource(title) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return success |         return success | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -376,23 +390,54 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|             try { |             try { | ||||||
|                 val response = api.login() |                 val response = api.login() | ||||||
|                 result = response.isSuccess == true |                 result = response.isSuccess == true | ||||||
|                 if (result) { |  | ||||||
|                     updateApiVersion() |  | ||||||
|                 } |  | ||||||
|             } catch (cause: Throwable) { |             } catch (cause: Throwable) { | ||||||
|                 Napier.e(cause.stackTraceToString(), tag = "RepositoryImpl.updateRemote") |                 Napier.e(cause.stackTraceToString(), tag = "RepositoryImpl.login") | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return result |         return result | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     suspend fun shouldBeSelfossInstance(): Pair<Boolean, Boolean> { | ||||||
|  |         var fetchFailed = true | ||||||
|  |         var showSelfossOnlyModal = false | ||||||
|  |         if (isNetworkAvailable()) { | ||||||
|  |             try { | ||||||
|  |                 // Trying to fetch one item, and check someone is trying to use the app with | ||||||
|  |                 // a random rss feed, that would throw a NoTransformationFoundException | ||||||
|  |                 fetchFailed = !api.getItemsWithoutCatch().success | ||||||
|  |             } catch (e: NoTransformationFoundException) { | ||||||
|  |                 showSelfossOnlyModal = true | ||||||
|  |             } catch (e: Throwable) { | ||||||
|  |                 Napier.e(e.stackTraceToString(), tag = "RepositoryImpl.shouldBeSelfossInstance") | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return Pair(fetchFailed, showSelfossOnlyModal) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     suspend fun logout() { | ||||||
|  |         if (isNetworkAvailable()) { | ||||||
|  |             try { | ||||||
|  |                 val response = api.logout() | ||||||
|  |                 if (!response.isSuccess) { | ||||||
|  |                     Napier.e("Couldn't logout.", tag = "RepositoryImpl.logout") | ||||||
|  |                 } | ||||||
|  |             } catch (cause: Throwable) { | ||||||
|  |                 Napier.e(cause.stackTraceToString(), tag = "RepositoryImpl.logout") | ||||||
|  |             } | ||||||
|  |             appSettingsService.clearAll() | ||||||
|  |         } else { | ||||||
|  |             appSettingsService.clearAll() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     fun refreshLoginInformation(url: String, login: String, password: String) { |     fun refreshLoginInformation(url: String, login: String, password: String) { | ||||||
|         appSettingsService.refreshLoginInformation(url, login, password) |         appSettingsService.refreshLoginInformation(url, login, password) | ||||||
|         baseUrl = url |         baseUrl = url | ||||||
|         api.refreshLoginInformation() |         api.refreshLoginInformation() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private suspend fun updateApiVersion() { |     suspend fun updateApiVersion() { | ||||||
|         val apiMajorVersion = appSettingsService.getApiVersion() |         val apiMajorVersion = appSettingsService.getApiVersion() | ||||||
|  |  | ||||||
|         if (isNetworkAvailable()) { |         if (isNetworkAvailable()) { | ||||||
| @@ -445,11 +490,29 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|  |  | ||||||
|     private fun getDBItems(): List<ITEM> = db.itemsQueries.items().executeAsList() |     private fun getDBItems(): List<ITEM> = db.itemsQueries.items().executeAsList() | ||||||
|  |  | ||||||
|     private fun insertDBAction(articleid: String, read: Boolean = false, unread: Boolean = false, starred: Boolean = false, unstarred: Boolean = false) = |     private fun insertDBAction( | ||||||
|  |         articleid: String, | ||||||
|  |         read: Boolean = false, | ||||||
|  |         unread: Boolean = false, | ||||||
|  |         starred: Boolean = false, | ||||||
|  |         unstarred: Boolean = false | ||||||
|  |     ) = | ||||||
|         db.actionsQueries.insertAction(articleid, read, unread, starred, unstarred) |         db.actionsQueries.insertAction(articleid, read, unread, starred, unstarred) | ||||||
|  |  | ||||||
|     private fun updateDBItem(item: SelfossModel.Item) = |     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()) |         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 { |         try { | ||||||
| @@ -496,4 +559,20 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | |||||||
|             deleteDBAction(action) |             deleteDBAction(action) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     fun setTagFilter(tag: SelfossModel.Tag?) { | ||||||
|  |         _tagFilter.value = tag | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun setSourceFilter(source: SelfossModel.Source?) { | ||||||
|  |         _sourceFilter.value = source | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun setReaderItems(readerItems: ArrayList<SelfossModel.Item>) { | ||||||
|  |         _readerItems = readerItems | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getReaderItems(): ArrayList<SelfossModel.Item> { | ||||||
|  |         return _readerItems | ||||||
|  |     } | ||||||
| } | } | ||||||
| @@ -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) | ||||||
|  |         }) | ||||||
|  | } | ||||||
| @@ -0,0 +1,79 @@ | |||||||
|  | package bou.amine.apps.readerforselfossv2.rest | ||||||
|  |  | ||||||
|  | import bou.amine.apps.readerforselfossv2.model.StatusAndData | ||||||
|  | import bou.amine.apps.readerforselfossv2.model.SuccessResponse | ||||||
|  | import io.github.aakira.napier.Napier | ||||||
|  | import io.ktor.client.* | ||||||
|  | import io.ktor.client.call.* | ||||||
|  | import io.ktor.client.request.* | ||||||
|  | import io.ktor.client.request.forms.* | ||||||
|  | import io.ktor.client.statement.* | ||||||
|  | import io.ktor.http.* | ||||||
|  |  | ||||||
|  |  | ||||||
|  | suspend fun responseOrSuccessIf404(r: HttpResponse?): SuccessResponse { | ||||||
|  |     return if (r != null && r.status === HttpStatusCode.NotFound) { | ||||||
|  |         SuccessResponse(true) | ||||||
|  |     } else { | ||||||
|  |         maybeResponse(r) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | suspend fun maybeResponse(r: HttpResponse?): SuccessResponse { | ||||||
|  |     return if (r != null && r.status.isSuccess()) { | ||||||
|  |         r.body() | ||||||
|  |     } else { | ||||||
|  |         SuccessResponse(false) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | suspend inline fun <reified T> bodyOrFailure(r: HttpResponse?): StatusAndData<T> { | ||||||
|  |     return if (r != null && r.status.isSuccess()) { | ||||||
|  |         StatusAndData.succes(r.body()) | ||||||
|  |     } else { | ||||||
|  |         StatusAndData.error() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | inline fun tryToRequest( | ||||||
|  |     requestType: String, | ||||||
|  |     fn: () -> HttpResponse | ||||||
|  | ): HttpResponse? { | ||||||
|  |     var response: HttpResponse? = null | ||||||
|  |     try { | ||||||
|  |         response = fn() | ||||||
|  |     } catch (ex: Exception) { | ||||||
|  |         Napier.e("Couldn't execute $requestType request", ex, "tryTo$requestType") | ||||||
|  |     } | ||||||
|  |     return response | ||||||
|  | } | ||||||
|  |  | ||||||
|  | suspend inline fun HttpClient.tryToGet( | ||||||
|  |     urlString: String, | ||||||
|  |     crossinline block: HttpRequestBuilder.() -> Unit = {} | ||||||
|  | ): HttpResponse? = tryToRequest("Get") { return this.get { url(urlString); block() } } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | suspend inline fun HttpClient.tryToPost( | ||||||
|  |     urlString: String, | ||||||
|  |     block: HttpRequestBuilder.() -> Unit = {} | ||||||
|  | ): HttpResponse? = tryToRequest("Post") { return this.post { url(urlString); block() } } | ||||||
|  |  | ||||||
|  | suspend inline fun HttpClient.tryToDelete( | ||||||
|  |     urlString: String, | ||||||
|  |     block: HttpRequestBuilder.() -> Unit = {} | ||||||
|  | ): HttpResponse? = tryToRequest("Delete") { return this.delete { url(urlString); block() } } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | suspend fun HttpClient.tryToSubmitForm( | ||||||
|  |     url: String, | ||||||
|  |     formParameters: Parameters = Parameters.Empty, | ||||||
|  |     encodeInQuery: Boolean = false, | ||||||
|  |     block: HttpRequestBuilder.() -> Unit = {} | ||||||
|  | ): HttpResponse? = | ||||||
|  |     tryToRequest("SubmitForm") { | ||||||
|  |         return this.submitForm(formParameters, encodeInQuery) { | ||||||
|  |             url(url) | ||||||
|  |             block() | ||||||
|  |         } | ||||||
|  |     } | ||||||
| @@ -1,18 +1,23 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.rest | package bou.amine.apps.readerforselfossv2.rest | ||||||
|  |  | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||||
|  | import bou.amine.apps.readerforselfossv2.model.StatusAndData | ||||||
|  | import bou.amine.apps.readerforselfossv2.model.SuccessResponse | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
|  | import io.github.aakira.napier.Napier | ||||||
| import io.ktor.client.* | import io.ktor.client.* | ||||||
| import io.ktor.client.call.* |  | ||||||
| import io.ktor.client.plugins.* | import io.ktor.client.plugins.* | ||||||
| import io.ktor.client.plugins.cache.* | import io.ktor.client.plugins.cache.* | ||||||
| import io.ktor.client.plugins.contentnegotiation.* | import io.ktor.client.plugins.contentnegotiation.* | ||||||
|  | import io.ktor.client.plugins.cookies.* | ||||||
| import io.ktor.client.plugins.logging.* | import io.ktor.client.plugins.logging.* | ||||||
| import io.ktor.client.request.* | import io.ktor.client.request.* | ||||||
| import io.ktor.client.request.forms.* |  | ||||||
| import io.ktor.client.statement.* | import io.ktor.client.statement.* | ||||||
| import io.ktor.http.* | import io.ktor.http.* | ||||||
| import io.ktor.serialization.kotlinx.json.* | import io.ktor.serialization.kotlinx.json.* | ||||||
|  | import kotlinx.coroutines.CoroutineScope | ||||||
|  | import kotlinx.coroutines.Dispatchers | ||||||
|  | import kotlinx.coroutines.launch | ||||||
| import kotlinx.serialization.json.Json | import kotlinx.serialization.json.Json | ||||||
|  |  | ||||||
| class SelfossApi(private val appSettingsService: AppSettingsService) { | class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||||
| @@ -20,7 +25,7 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | |||||||
|     var client = createHttpClient() |     var client = createHttpClient() | ||||||
|  |  | ||||||
|     private fun createHttpClient(): HttpClient { |     private fun createHttpClient(): HttpClient { | ||||||
|         return HttpClient { |         val client = HttpClient { | ||||||
|             install(ContentNegotiation) { |             install(ContentNegotiation) { | ||||||
|                 install(HttpCache) |                 install(HttpCache) | ||||||
|                 json(Json { |                 json(Json { | ||||||
| @@ -32,7 +37,7 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | |||||||
|             install(Logging) { |             install(Logging) { | ||||||
|                 logger = object : Logger { |                 logger = object : Logger { | ||||||
|                     override fun log(message: String) { |                     override fun log(message: String) { | ||||||
|                         appSettingsService.logApiCalls(message) |                         Napier.d(message, tag = "LogApiCalls") | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 level = LogLevel.INFO |                 level = LogLevel.INFO | ||||||
| @@ -40,22 +45,25 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | |||||||
|             install(HttpTimeout) { |             install(HttpTimeout) { | ||||||
|                 requestTimeoutMillis = appSettingsService.getApiTimeout() |                 requestTimeoutMillis = appSettingsService.getApiTimeout() | ||||||
|             } |             } | ||||||
|             /* TODO: Auth as basic |             install(HttpCookies) | ||||||
|             if (apiDetailsService.getUserName().isNotEmpty() && apiDetailsService.getPassword().isNotEmpty()) { |             install(HttpRequestRetry) { | ||||||
|  |                 maxRetries = 2 | ||||||
|                 install(Auth) { |                 retryIf { _, response -> | ||||||
|                     basic { |                     response.status == HttpStatusCode.Forbidden && shouldHavePostLogin() && hasLoginInfo() | ||||||
|                         credentials { |  | ||||||
|                             BasicAuthCredentials(username = apiDetailsService.getUserName(), password = apiDetailsService.getPassword()) |  | ||||||
|                 } |                 } | ||||||
|                         sendWithoutRequest { |                 modifyRequest { | ||||||
|                             true |                     Napier.i("Will modify", tag = "HttpSend") | ||||||
|  |                     CoroutineScope(Dispatchers.Main).launch { | ||||||
|  |                         Napier.i("Will login", tag = "HttpSend") | ||||||
|  |                         this@SelfossApi.login() | ||||||
|  |                         Napier.i("Did login", tag = "HttpSend") | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             }*/ |  | ||||||
|             expectSuccess = false |             expectSuccess = false | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         return client | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun url(path: String) = |     fun url(path: String) = | ||||||
| @@ -66,12 +74,43 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | |||||||
|         client = createHttpClient() |         client = createHttpClient() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     suspend fun login(): SelfossModel.SuccessResponse = |     // Api version was introduces after the POST login, so when there is a version, it should be available | ||||||
|         maybeResponse(client.get(url("/login")) { |     private fun shouldHavePostLogin() = appSettingsService.getApiVersion() != -1 | ||||||
|  |     private fun hasLoginInfo() = appSettingsService.getUserName().isNotEmpty() && appSettingsService.getPassword().isNotEmpty() | ||||||
|  |  | ||||||
|  |     suspend fun login(): SuccessResponse = | ||||||
|  |         if (appSettingsService.getUserName().isNotEmpty() && appSettingsService.getPassword().isNotEmpty()) { | ||||||
|  |             if (shouldHavePostLogin()) { | ||||||
|  |                 postLogin() | ||||||
|  |             } else { | ||||||
|  |                 getLogin() | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             SuccessResponse(true) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     private suspend fun getLogin() = maybeResponse(client.tryToGet(url("/login")) { | ||||||
|         parameter("username", appSettingsService.getUserName()) |         parameter("username", appSettingsService.getUserName()) | ||||||
|         parameter("password", appSettingsService.getPassword()) |         parameter("password", appSettingsService.getPassword()) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|  |     private suspend fun postLogin() = maybeResponse(client.tryToPost(url("/login")) { | ||||||
|  |         parameter("username", appSettingsService.getUserName()) | ||||||
|  |         parameter("password", appSettingsService.getPassword()) | ||||||
|  |     }) | ||||||
|  |  | ||||||
|  |     private fun shouldHaveNewLogout() = appSettingsService.getApiVersion() >= 5 // We are missing 4.1.0 | ||||||
|  |     suspend fun logout(): SuccessResponse = | ||||||
|  |         if (shouldHaveNewLogout()) { | ||||||
|  |             doLogout() | ||||||
|  |         } else { | ||||||
|  |             maybeLogoutIfAvailable() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     private suspend fun maybeLogoutIfAvailable() = responseOrSuccessIf404(client.tryToGet(url("/logout"))) | ||||||
|  |  | ||||||
|  |     private suspend fun doLogout() = maybeResponse(client.tryToDelete(url("/api/session/current"))) | ||||||
|  |  | ||||||
|     suspend fun getItems( |     suspend fun getItems( | ||||||
|         type: String, |         type: String, | ||||||
|         offset: Int, |         offset: Int, | ||||||
| @@ -80,10 +119,12 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | |||||||
|         search: String?, |         search: String?, | ||||||
|         updatedSince: String?, |         updatedSince: String?, | ||||||
|         items: Int? = null |         items: Int? = null | ||||||
|     ): SelfossModel.StatusAndData<List<SelfossModel.Item>> = |     ): StatusAndData<List<SelfossModel.Item>> = | ||||||
|         bodyOrFailure(client.get(url("/items")) { |         bodyOrFailure(client.tryToGet(url("/items")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|             parameter("type", type) |             parameter("type", type) | ||||||
|             parameter("tag", tag) |             parameter("tag", tag) | ||||||
|             parameter("source", source) |             parameter("source", source) | ||||||
| @@ -93,69 +134,99 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | |||||||
|             parameter("offset", offset) |             parameter("offset", offset) | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun stats(): SelfossModel.StatusAndData<SelfossModel.Stats> = |     suspend fun getItemsWithoutCatch(): StatusAndData<List<SelfossModel.Item>> = | ||||||
|         bodyOrFailure(client.get(url("/stats")) { |         bodyOrFailure(client.get(url("/items")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|  |             parameter("type", "all") | ||||||
|  |             parameter("items", 1) | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun tags(): SelfossModel.StatusAndData<List<SelfossModel.Tag>> = |     suspend fun stats(): StatusAndData<SelfossModel.Stats> = | ||||||
|         bodyOrFailure(client.get(url("/tags")) { |         bodyOrFailure(client.tryToGet(url("/stats")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun update(): SelfossModel.StatusAndData<String> = |     suspend fun tags(): StatusAndData<List<SelfossModel.Tag>> = | ||||||
|         bodyOrFailure(client.get(url("/update")) { |         bodyOrFailure(client.tryToGet(url("/tags")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun spouts(): SelfossModel.StatusAndData<Map<String, SelfossModel.Spout>> = |     suspend fun update(): StatusAndData<String> = | ||||||
|         bodyOrFailure(client.get(url("/sources/spouts")) { |         bodyOrFailure(client.tryToGet(url("/update")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun sources(): SelfossModel.StatusAndData<ArrayList<SelfossModel.Source>> = |     suspend fun spouts(): StatusAndData<Map<String, SelfossModel.Spout>> = | ||||||
|         bodyOrFailure(client.get(url("/sources/list")) { |         bodyOrFailure(client.tryToGet(url("/sources/spouts")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun version(): SelfossModel.StatusAndData<SelfossModel.ApiVersion> = |     suspend fun sources(): StatusAndData<ArrayList<SelfossModel.Source>> = | ||||||
|         bodyOrFailure(client.get(url("/api/about"))) |         bodyOrFailure(client.tryToGet(url("/sources/list")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|     suspend fun markAsRead(id: String): SelfossModel.SuccessResponse = |  | ||||||
|         maybeResponse(client.post(url("/mark/$id")) { |  | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun unmarkAsRead(id: String): SelfossModel.SuccessResponse = |     suspend fun version(): StatusAndData<SelfossModel.ApiVersion> = | ||||||
|         maybeResponse(client.post(url("/unmark/$id")) { |         bodyOrFailure(client.tryToGet(url("/api/about"))) | ||||||
|  |  | ||||||
|  |     suspend fun markAsRead(id: String): SuccessResponse = | ||||||
|  |         maybeResponse(client.tryToPost(url("/mark/$id")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun starr(id: String): SelfossModel.SuccessResponse = |     suspend fun unmarkAsRead(id: String): SuccessResponse = | ||||||
|         maybeResponse(client.post(url("/starr/$id")) { |         maybeResponse(client.tryToPost(url("/unmark/$id")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun unstarr(id: String): SelfossModel.SuccessResponse = |     suspend fun starr(id: String): SuccessResponse = | ||||||
|         maybeResponse(client.post(url("/unstarr/$id")) { |         maybeResponse(client.tryToPost(url("/starr/$id")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun markAllAsRead(ids: List<String>): SelfossModel.SuccessResponse = |     suspend fun unstarr(id: String): SuccessResponse = | ||||||
|         maybeResponse(client.submitForm( |         maybeResponse(client.tryToPost(url("/unstarr/$id")) { | ||||||
|  |             if (!shouldHavePostLogin()) { | ||||||
|  |                 parameter("username", appSettingsService.getUserName()) | ||||||
|  |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |     suspend fun markAllAsRead(ids: List<String>): SuccessResponse = | ||||||
|  |         maybeResponse(client.tryToSubmitForm( | ||||||
|             url = url("/mark"), |             url = url("/mark"), | ||||||
|             formParameters = Parameters.build { |             formParameters = Parameters.build { | ||||||
|  |                 if (!shouldHavePostLogin()) { | ||||||
|                     append("username", appSettingsService.getUserName()) |                     append("username", appSettingsService.getUserName()) | ||||||
|                     append("password", appSettingsService.getPassword()) |                     append("password", appSettingsService.getPassword()) | ||||||
|  |                 } | ||||||
|                 ids.map { append("ids[]", it) } |                 ids.map { append("ids[]", it) } | ||||||
|             } |             } | ||||||
|         )) |         )) | ||||||
| @@ -165,72 +236,44 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | |||||||
|         url: String, |         url: String, | ||||||
|         spout: String, |         spout: String, | ||||||
|         tags: String, |         tags: String, | ||||||
|         filter: String, |         filter: String | ||||||
|         version: Int |     ): SuccessResponse = | ||||||
|     ): SelfossModel.SuccessResponse = |  | ||||||
|         maybeResponse( |         maybeResponse( | ||||||
|             if (version > 1) { |             if (appSettingsService.getApiVersion() > 1) { | ||||||
|                 createSource2(title, url, spout, tags, filter) |                 createSource("tags[]", title, url, spout, tags, filter) | ||||||
|             } else { |             } else { | ||||||
|                 createSource(title, url, spout, tags, filter) |                 createSource("tags", title, url, spout, tags, filter) | ||||||
|             } |             } | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     suspend fun createSource( |     private suspend fun createSource( | ||||||
|  |         tagsParamName: String, | ||||||
|         title: String, |         title: String, | ||||||
|         url: String, |         url: String, | ||||||
|         spout: String, |         spout: String, | ||||||
|         tags: String, |         tags: String, | ||||||
|         filter: String |         filter: String | ||||||
|     ): HttpResponse = |     ): HttpResponse? = | ||||||
|         client.submitForm( |         client.tryToSubmitForm( | ||||||
|             url = url("/source?username=${appSettingsService.getUserName()}&password=${appSettingsService.getPassword()}"), |             url = url("/source"), | ||||||
|             formParameters = Parameters.build { |             formParameters = Parameters.build { | ||||||
|  |                 if (!shouldHavePostLogin()) { | ||||||
|  |                     append("username", appSettingsService.getUserName()) | ||||||
|  |                     append("password", appSettingsService.getPassword()) | ||||||
|  |                 } | ||||||
|                 append("title", title) |                 append("title", title) | ||||||
|                 append("url", url) |                 append("url", url) | ||||||
|                 append("spout", spout) |                 append("spout", spout) | ||||||
|                 append("tags", tags) |                 append(tagsParamName, tags) | ||||||
|                 append("filter", filter) |                 append("filter", filter) | ||||||
|             } |             } | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     suspend fun createSource2( |     suspend fun deleteSource(id: Int): SuccessResponse = | ||||||
|         title: String, |         maybeResponse(client.tryToDelete(url("/source/$id")) { | ||||||
|         url: String, |             if (!shouldHavePostLogin()) { | ||||||
|         spout: String, |  | ||||||
|         tags: String, |  | ||||||
|         filter: String |  | ||||||
|     ): HttpResponse = |  | ||||||
|         client.submitForm( |  | ||||||
|             url = url("/source?username=${appSettingsService.getUserName()}&password=${appSettingsService.getPassword()}"), |  | ||||||
|             formParameters = Parameters.build { |  | ||||||
|                 append("title", title) |  | ||||||
|                 append("url", url) |  | ||||||
|                 append("spout", spout) |  | ||||||
|                 append("tags[]", tags) |  | ||||||
|                 append("filter", filter) |  | ||||||
|             } |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     suspend fun deleteSource(id: Int): SelfossModel.SuccessResponse = |  | ||||||
|         maybeResponse(client.delete(url("/source/$id")) { |  | ||||||
|                 parameter("username", appSettingsService.getUserName()) |                 parameter("username", appSettingsService.getUserName()) | ||||||
|                 parameter("password", appSettingsService.getPassword()) |                 parameter("password", appSettingsService.getPassword()) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     suspend fun maybeResponse(r: HttpResponse): SelfossModel.SuccessResponse { |  | ||||||
|         return if (r.status.isSuccess()) { |  | ||||||
|             r.body() |  | ||||||
|         } else { |  | ||||||
|             SelfossModel.SuccessResponse(false) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     suspend inline fun <reified T> bodyOrFailure(r: HttpResponse): SelfossModel.StatusAndData<T> { |  | ||||||
|         return if (r.status.isSuccess()) { |  | ||||||
|             SelfossModel.StatusAndData.succes(r.body()) |  | ||||||
|         } else { |  | ||||||
|             SelfossModel.StatusAndData.error() |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } | } | ||||||
| @@ -27,7 +27,6 @@ class AppSettingsService { | |||||||
|     private var _notifyNewItems: Boolean? = null |     private var _notifyNewItems: Boolean? = null | ||||||
|     private var _itemsNumber: Int? = null |     private var _itemsNumber: Int? = null | ||||||
|     private var _apiTimeout: Long? = null |     private var _apiTimeout: Long? = null | ||||||
|     private var _hiddenTags: List<String>? = null |  | ||||||
|     private var _refreshMinutes: Long = 360 |     private var _refreshMinutes: Long = 360 | ||||||
|     private var _markOnScroll: Boolean? = null |     private var _markOnScroll: Boolean? = null | ||||||
|     private var _activeAlignment: Int? = null |     private var _activeAlignment: Int? = null | ||||||
| @@ -35,6 +34,7 @@ class AppSettingsService { | |||||||
|     private var _fontSize: Int? = null |     private var _fontSize: Int? = null | ||||||
|     private var _staticBar: Boolean? = null |     private var _staticBar: Boolean? = null | ||||||
|     private var _font: String = "" |     private var _font: String = "" | ||||||
|  |     private var _theme: Int? = null | ||||||
|  |  | ||||||
|  |  | ||||||
|     init { |     init { | ||||||
| @@ -42,10 +42,6 @@ class AppSettingsService { | |||||||
|         refreshUserSettings() |         refreshUserSettings() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun logApiCalls(message: String) { |  | ||||||
|         Napier.d(message, tag = "LogApiCalls") |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fun getApiVersion(): Int { |     fun getApiVersion(): Int { | ||||||
|         if (_apiVersion == -1) { |         if (_apiVersion == -1) { | ||||||
|             refreshApiVersion() |             refreshApiVersion() | ||||||
| @@ -54,8 +50,8 @@ class AppSettingsService { | |||||||
|         return _apiVersion |         return _apiVersion | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun refreshApiVersion() { |     private fun refreshApiVersion() { | ||||||
|         _apiVersion = settings.getInt("apiVersionMajor", -1) |         _apiVersion = settings.getInt(API_VERSION_MAJOR, -1) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getBaseUrl(): String { |     fun getBaseUrl(): String { | ||||||
| @@ -86,8 +82,14 @@ class AppSettingsService { | |||||||
|         return _itemsNumber!! |         return _itemsNumber!! | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun refreshItemsNumber() { |     private fun refreshItemsNumber() { | ||||||
|         _itemsNumber = settings.getString("prefer_api_items_number", "20").toInt() |         _itemsNumber = try { | ||||||
|  |             settings.getString(API_ITEMS_NUMBER, "20").toInt() | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             settings.remove(API_ITEMS_NUMBER) | ||||||
|  |             20 | ||||||
|  |         } | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getApiTimeout(): Long { |     fun getApiTimeout(): Long { | ||||||
| @@ -97,25 +99,37 @@ class AppSettingsService { | |||||||
|         return _apiTimeout!! |         return _apiTimeout!! | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private fun secToMs(n: Long) = n * 1000 | ||||||
|  |  | ||||||
|     private fun refreshApiTimeout() { |     private fun refreshApiTimeout() { | ||||||
|         val settingsTimeout = settings.getLong("api_timeout", HttpTimeout.INFINITE_TIMEOUT_MS) |         _apiTimeout = secToMs(try { | ||||||
|         _apiTimeout = if (settingsTimeout > 0) settingsTimeout else HttpTimeout.INFINITE_TIMEOUT_MS |             val settingsTimeout = settings.getString(API_TIMEOUT, "60") | ||||||
|  |             if (settingsTimeout.toLong() > 0) { | ||||||
|  |                 settingsTimeout.toLong() | ||||||
|  |             } else { | ||||||
|  |                 settings.remove(API_TIMEOUT) | ||||||
|  |                 60 | ||||||
|  |             } | ||||||
|  |         } catch (e: Exception) { | ||||||
|  |             settings.remove(API_TIMEOUT) | ||||||
|  |             60 | ||||||
|  |         }) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshBaseUrl() { |     private fun refreshBaseUrl() { | ||||||
|         _baseUrl = settings.getString("url", "") |         _baseUrl = settings.getString(BASE_URL, "") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshUsername() { |     private fun refreshUsername() { | ||||||
|         _userName = settings.getString("login", "") |         _userName = settings.getString(LOGIN, "") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshPassword() { |     private fun refreshPassword() { | ||||||
|         _password = settings.getString("password", "") |         _password = settings.getString(PASSWORD, "") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshArticleViewerEnabled() { |     private fun refreshArticleViewerEnabled() { | ||||||
|         _articleViewer = settings.getBoolean("prefer_article_viewer", true) |         _articleViewer = settings.getBoolean(PREFER_ARTICLE_VIEWER, true) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isArticleViewerEnabled(): Boolean { |     fun isArticleViewerEnabled(): Boolean { | ||||||
| @@ -125,7 +139,7 @@ class AppSettingsService { | |||||||
|         return _articleViewer == true |         return _articleViewer == true | ||||||
|     } |     } | ||||||
|     private fun refreshShouldBeCardViewEnabled() { |     private fun refreshShouldBeCardViewEnabled() { | ||||||
|         _shouldBeCardView = settings.getBoolean("card_view_active", false) |         _shouldBeCardView = settings.getBoolean(CARD_VIEW_ACTIVE, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isCardViewEnabled(): Boolean { |     fun isCardViewEnabled(): Boolean { | ||||||
| @@ -135,7 +149,7 @@ class AppSettingsService { | |||||||
|         return _shouldBeCardView == true |         return _shouldBeCardView == true | ||||||
|     } |     } | ||||||
|     private fun refreshDisplayUnreadCountEnabled() { |     private fun refreshDisplayUnreadCountEnabled() { | ||||||
|         _displayUnreadCount = settings.getBoolean("display_unread_count", true) |         _displayUnreadCount = settings.getBoolean(DISPLAY_UNREAD_COUNT, true) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isDisplayUnreadCountEnabled(): Boolean { |     fun isDisplayUnreadCountEnabled(): Boolean { | ||||||
| @@ -145,7 +159,7 @@ class AppSettingsService { | |||||||
|         return _displayUnreadCount == true |         return _displayUnreadCount == true | ||||||
|     } |     } | ||||||
|     private fun refreshDisplayAllCountEnabled() { |     private fun refreshDisplayAllCountEnabled() { | ||||||
|         _displayAllCount = settings.getBoolean("display_other_count", false) |         _displayAllCount = settings.getBoolean(DISPLAY_OTHER_COUNT, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isDisplayAllCountEnabled(): Boolean { |     fun isDisplayAllCountEnabled(): Boolean { | ||||||
| @@ -155,7 +169,7 @@ class AppSettingsService { | |||||||
|         return _displayAllCount == true |         return _displayAllCount == true | ||||||
|     } |     } | ||||||
|     private fun refreshFullHeightCardsEnabled() { |     private fun refreshFullHeightCardsEnabled() { | ||||||
|         _fullHeightCards = settings.getBoolean("full_height_cards", false) |         _fullHeightCards = settings.getBoolean(FULL_HEIGHT_CARDS, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isFullHeightCardsEnabled(): Boolean { |     fun isFullHeightCardsEnabled(): Boolean { | ||||||
| @@ -165,7 +179,7 @@ class AppSettingsService { | |||||||
|         return _fullHeightCards == true |         return _fullHeightCards == true | ||||||
|     } |     } | ||||||
|     private fun refreshUpdateSourcesEnabled() { |     private fun refreshUpdateSourcesEnabled() { | ||||||
|         _updateSources = settings.getBoolean("update_sources", true) |         _updateSources = settings.getBoolean(UPDATE_SOURCES, true) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isUpdateSourcesEnabled(): Boolean { |     fun isUpdateSourcesEnabled(): Boolean { | ||||||
| @@ -175,7 +189,7 @@ class AppSettingsService { | |||||||
|         return _updateSources == true |         return _updateSources == true | ||||||
|     } |     } | ||||||
|     private fun refreshPeriodicRefreshEnabled() { |     private fun refreshPeriodicRefreshEnabled() { | ||||||
|         _periodicRefresh = settings.getBoolean("periodic_refresh", false) |         _periodicRefresh = settings.getBoolean(PERIODIC_REFRESH, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isPeriodicRefreshEnabled(): Boolean { |     fun isPeriodicRefreshEnabled(): Boolean { | ||||||
| @@ -186,7 +200,7 @@ class AppSettingsService { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshRefreshWhenChargingOnlyEnabled() { |     private fun refreshRefreshWhenChargingOnlyEnabled() { | ||||||
|         _refreshWhenChargingOnly = settings.getBoolean("refresh_when_charging", false) |         _refreshWhenChargingOnly = settings.getBoolean(REFRESH_WHEN_CHARGING, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isRefreshWhenChargingOnlyEnabled(): Boolean { |     fun isRefreshWhenChargingOnlyEnabled(): Boolean { | ||||||
| @@ -197,34 +211,21 @@ class AppSettingsService { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshRefreshMinutes() { |     private fun refreshRefreshMinutes() { | ||||||
|         _refreshMinutes = settings.getString("periodic_refresh_minutes", "360").toLong() |         _refreshMinutes = settings.getString(PERIODIC_REFRESH_MINUTES, "360").toLong() | ||||||
|         if (_refreshMinutes <= 15) { |         if (_refreshMinutes <= 15) { | ||||||
|             _refreshMinutes = 15 |             _refreshMinutes = 15 | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getRefreshMinutes(): Long { |     fun getRefreshMinutes(): Long { | ||||||
|         if (_refreshMinutes != null) { |         if (_refreshMinutes != 360L) { | ||||||
|             refreshRefreshMinutes() |             refreshRefreshMinutes() | ||||||
|         } |         } | ||||||
|         return _refreshMinutes |         return _refreshMinutes | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshHiddenTags() { |  | ||||||
|         if (settings.getString("hidden_tags", "").isNotEmpty()) { |  | ||||||
|             _hiddenTags = settings.getString("hidden_tags", "").replace("\\s".toRegex(), "").split(",") |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fun getHiddenTags(): List<String> { |  | ||||||
|         if (_hiddenTags != null) { |  | ||||||
|             refreshHiddenTags() |  | ||||||
|         } |  | ||||||
|         return _hiddenTags.orEmpty() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private fun refreshInfiniteLoadingEnabled() { |     private fun refreshInfiniteLoadingEnabled() { | ||||||
|         _infiniteLoading = settings.getBoolean("infinite_loading", false) |         _infiniteLoading = settings.getBoolean(INFINITE_LOADING, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isInfiniteLoadingEnabled(): Boolean { |     fun isInfiniteLoadingEnabled(): Boolean { | ||||||
| @@ -235,7 +236,7 @@ class AppSettingsService { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshItemCachingEnabled() { |     private fun refreshItemCachingEnabled() { | ||||||
|         _itemsCaching = settings.getBoolean("items_caching", false) |         _itemsCaching = settings.getBoolean(ITEMS_CACHING, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isItemCachingEnabled(): Boolean { |     fun isItemCachingEnabled(): Boolean { | ||||||
| @@ -246,7 +247,7 @@ class AppSettingsService { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshNotifyNewItemsEnabled() { |     private fun refreshNotifyNewItemsEnabled() { | ||||||
|         _notifyNewItems = settings.getBoolean("notify_new_items", false) |         _notifyNewItems = settings.getBoolean(NOTIFY_NEW_ITEMS, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isNotifyNewItemsEnabled(): Boolean { |     fun isNotifyNewItemsEnabled(): Boolean { | ||||||
| @@ -258,7 +259,7 @@ class AppSettingsService { | |||||||
|  |  | ||||||
|  |  | ||||||
|     private fun refreshMarkOnScrollEnabled() { |     private fun refreshMarkOnScrollEnabled() { | ||||||
|         _markOnScroll = settings.getBoolean("mark_on_scroll", false) |         _markOnScroll = settings.getBoolean(MARK_ON_SCROLL, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isMarkOnScrollEnabled(): Boolean { |     fun isMarkOnScrollEnabled(): Boolean { | ||||||
| @@ -270,7 +271,7 @@ class AppSettingsService { | |||||||
|  |  | ||||||
|  |  | ||||||
|     private fun refreshActiveAllignment() { |     private fun refreshActiveAllignment() { | ||||||
|         _activeAlignment = settings.getInt("text_align", JUSTIFY) |         _activeAlignment = settings.getInt(TEXT_ALIGN, JUSTIFY) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getActiveAllignment(): Int { |     fun getActiveAllignment(): Int { | ||||||
| @@ -281,12 +282,12 @@ class AppSettingsService { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun changeAllignment(allignment: Int) { |     fun changeAllignment(allignment: Int) { | ||||||
|         settings.putInt("text_align", allignment) |         settings.putInt(TEXT_ALIGN, allignment) | ||||||
|         _activeAlignment = allignment |         _activeAlignment = allignment | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshFontSize() { |     private fun refreshFontSize() { | ||||||
|         _fontSize = settings.getString("reader_font_size", "16").toInt() |         _fontSize = settings.getString(READER_FONT_SIZE, "16").toInt() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getFontSize(): Int { |     fun getFontSize(): Int { | ||||||
| @@ -297,7 +298,7 @@ class AppSettingsService { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshStaticBarEnabled() { |     private fun refreshStaticBarEnabled() { | ||||||
|         _staticBar = settings.getBoolean("reader_static_bar", false) |         _staticBar = settings.getBoolean(READER_STATIC_BAR, false) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun isStaticBarEnabled(): Boolean { |     fun isStaticBarEnabled(): Boolean { | ||||||
| @@ -308,16 +309,27 @@ class AppSettingsService { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun refreshFont() { |     private fun refreshFont() { | ||||||
|         _font = settings.getString("reader_font", "") |         _font = settings.getString(READER_FONT, "") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getFont(): String { |     fun getFont(): String { | ||||||
|         if (_font != null) { |         if (_font.isEmpty()) { | ||||||
|             refreshFont() |             refreshFont() | ||||||
|         } |         } | ||||||
|         return _font |         return _font | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private fun refreshCurrentTheme() { | ||||||
|  |         _theme = settings.getString(CURRENT_THEME, "-1").toInt() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getCurrentTheme(): Int { | ||||||
|  |         if (_theme == null) { | ||||||
|  |             refreshCurrentTheme() | ||||||
|  |         } | ||||||
|  |         return _theme ?: -1 | ||||||
|  |     } | ||||||
|  |  | ||||||
|     fun refreshApiSettings() { |     fun refreshApiSettings() { | ||||||
|         refreshPassword() |         refreshPassword() | ||||||
|         refreshUsername() |         refreshUsername() | ||||||
| @@ -337,7 +349,6 @@ class AppSettingsService { | |||||||
|         refreshPeriodicRefreshEnabled() |         refreshPeriodicRefreshEnabled() | ||||||
|         refreshRefreshWhenChargingOnlyEnabled() |         refreshRefreshWhenChargingOnlyEnabled() | ||||||
|         refreshRefreshMinutes() |         refreshRefreshMinutes() | ||||||
|         refreshHiddenTags() |  | ||||||
|         refreshInfiniteLoadingEnabled() |         refreshInfiniteLoadingEnabled() | ||||||
|         refreshItemCachingEnabled() |         refreshItemCachingEnabled() | ||||||
|         refreshNotifyNewItemsEnabled() |         refreshNotifyNewItemsEnabled() | ||||||
| @@ -346,6 +357,7 @@ class AppSettingsService { | |||||||
|         refreshFontSize() |         refreshFontSize() | ||||||
|         refreshFont() |         refreshFont() | ||||||
|         refreshStaticBarEnabled() |         refreshStaticBarEnabled() | ||||||
|  |         refreshCurrentTheme() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun refreshLoginInformation( |     fun refreshLoginInformation( | ||||||
| @@ -353,21 +365,21 @@ class AppSettingsService { | |||||||
|         login: String, |         login: String, | ||||||
|         password: String |         password: String | ||||||
|     ) { |     ) { | ||||||
|         settings.putString("url", url) |         settings.putString(BASE_URL, url) | ||||||
|         settings.putString("login", login) |         settings.putString(LOGIN, login) | ||||||
|         settings.putString("password", password) |         settings.putString(PASSWORD, password) | ||||||
|         refreshApiSettings() |         refreshApiSettings() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun resetLoginInformation() { |     fun resetLoginInformation() { | ||||||
|         settings.remove("url") |         settings.remove(BASE_URL) | ||||||
|         settings.remove("login") |         settings.remove(LOGIN) | ||||||
|         settings.remove("password") |         settings.remove(PASSWORD) | ||||||
|         refreshApiSettings() |         refreshApiSettings() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun updateApiVersion(apiMajorVersion: Int) { |     fun updateApiVersion(apiMajorVersion: Int) { | ||||||
|         settings.putInt("apiVersionMajor", apiMajorVersion) |         settings.putInt(API_VERSION_MAJOR, apiMajorVersion) | ||||||
|         refreshApiVersion() |         refreshApiVersion() | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -378,7 +390,7 @@ class AppSettingsService { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun disableArticleViewer() { |     fun disableArticleViewer() { | ||||||
|         settings.putBoolean("prefer_article_viewer", false) |         settings.putBoolean(PREFER_ARTICLE_VIEWER, false) | ||||||
|         refreshArticleViewerEnabled() |         refreshArticleViewerEnabled() | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -396,5 +408,55 @@ class AppSettingsService { | |||||||
|         const val JUSTIFY = 1 |         const val JUSTIFY = 1 | ||||||
|  |  | ||||||
|         const val ALIGN_LEFT = 2 |         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 INFINITE_LOADING = "infinite_loading" | ||||||
|  |  | ||||||
|  |         const val ITEMS_CACHING = "items_caching" | ||||||
|  |  | ||||||
|  |         const val CURRENT_THEME = "currentMode" | ||||||
|  |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,27 +0,0 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.service |  | ||||||
|  |  | ||||||
| import bou.amine.apps.readerforselfossv2.utils.DateUtils |  | ||||||
|  |  | ||||||
| class SearchService(val dateUtils: DateUtils) { |  | ||||||
|     var displayedItems: String = "unread" |  | ||||||
|         set(value) { |  | ||||||
|             field = when (value) { |  | ||||||
|                 "all" -> "all" |  | ||||||
|                 "unread" -> "unread" |  | ||||||
|                 "read" -> "read" |  | ||||||
|                 "starred" -> "starred" |  | ||||||
|                 else -> "all" |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|     var position = 0 |  | ||||||
|     var searchFilter: String? = null |  | ||||||
|     var sourceIDFilter: Long? = null |  | ||||||
|     var sourceFilter: String? = null |  | ||||||
|     var tagFilter: String? = null |  | ||||||
|     var itemsCaching = false |  | ||||||
|  |  | ||||||
|     var badgeUnread = -1 |  | ||||||
|     var badgeAll = -1 |  | ||||||
|     var badgeStarred = -1 |  | ||||||
| } |  | ||||||
| @@ -1,16 +1,9 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.utils | package bou.amine.apps.readerforselfossv2.utils | ||||||
|  |  | ||||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | expect class DateUtils() { | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService |     companion object { | ||||||
|  |  | ||||||
|  |  | ||||||
| fun SelfossModel.Item.parseDate(dateUtils: DateUtils): Long = |  | ||||||
|     dateUtils.parseDate(this.datetime) |  | ||||||
|  |  | ||||||
| expect class DateUtils constructor(appSettingsService: AppSettingsService) { |  | ||||||
|     val appSettingsService: AppSettingsService // This is needed because of https://stackoverflow.com/a/65249085 |  | ||||||
|  |  | ||||||
|         fun parseDate(dateString: String): Long |         fun parseDate(dateString: String): Long | ||||||
|  |  | ||||||
|         fun parseRelativeDate(dateString: String): String |         fun parseRelativeDate(dateString: String): String | ||||||
|     } |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -51,7 +51,8 @@ fun ITEM.toView(): SelfossModel.Item = | |||||||
|         this.icon, |         this.icon, | ||||||
|         this.link, |         this.link, | ||||||
|         this.sourcetitle, |         this.sourcetitle, | ||||||
|         this.tags.split(",") |         this.tags.split(","), | ||||||
|  |         this.author | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
| fun SelfossModel.Item.toEntity(): ITEM = | fun SelfossModel.Item.toEntity(): ITEM = | ||||||
| @@ -65,6 +66,7 @@ fun SelfossModel.Item.toEntity(): ITEM = | |||||||
|         this.thumbnail, |         this.thumbnail, | ||||||
|         this.icon, |         this.icon, | ||||||
|         this.link, |         this.link, | ||||||
|         this.title.getHtmlDecoded(), |         this.sourcetitle.getHtmlDecoded(), | ||||||
|         this.tags.joinToString(",") |         this.tags.joinToString(","), | ||||||
|  |         this.author | ||||||
|     ) |     ) | ||||||
| @@ -0,0 +1 @@ | |||||||
|  | ALTER TABLE ITEM ADD COLUMN `author` TEXT NOT NULL; | ||||||
| @@ -10,6 +10,7 @@ CREATE TABLE ITEM ( | |||||||
|     `link` TEXT NOT NULL, |     `link` TEXT NOT NULL, | ||||||
|     `sourcetitle` TEXT NOT NULL, |     `sourcetitle` TEXT NOT NULL, | ||||||
|     `tags` TEXT NOT NULL, |     `tags` TEXT NOT NULL, | ||||||
|  |     `author` TEXT NOT NULL, | ||||||
|     PRIMARY KEY(`id`) |     PRIMARY KEY(`id`) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -26,5 +27,8 @@ INSERT OR REPLACE INTO ITEM VALUES ?; | |||||||
| deleteItem: | deleteItem: | ||||||
| DELETE FROM ITEM WHERE `id` = ?; | DELETE FROM ITEM WHERE `id` = ?; | ||||||
|  |  | ||||||
|  | deleteItemsWhereSource: | ||||||
|  | DELETE FROM ITEM WHERE `sourcetitle` = ?; | ||||||
|  |  | ||||||
| updateItem: | updateItem: | ||||||
| UPDATE ITEM SET `datetime` = ?, `title` = ?, `content` = ?, `unread` = ?, `starred` = ?, `thumbnail` = ?, `icon` = ?, `link` = ?, `sourcetitle` = ?, `tags` = ? WHERE `id` = ?; | UPDATE ITEM SET `datetime` = ?, `title` = ?, `content` = ?, `unread` = ?, `starred` = ?, `thumbnail` = ?, `icon` = ?, `link` = ?, `sourcetitle` = ?, `tags` = ? WHERE `id` = ?; | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,8 +1,7 @@ | |||||||
| package bou.amine.apps.readerforselfossv2.utils | package bou.amine.apps.readerforselfossv2.utils | ||||||
|  |  | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | actual class DateUtils { | ||||||
|  |     actual companion object { | ||||||
| actual class DateUtils actual constructor(actual val appSettingsService: AppSettingsService) { |  | ||||||
|         actual fun parseDate(dateString: String): Long { |         actual fun parseDate(dateString: String): Long { | ||||||
|             TODO("Not yet implemented") |             TODO("Not yet implemented") | ||||||
|         } |         } | ||||||
| @@ -10,5 +9,5 @@ actual class DateUtils actual constructor(actual val appSettingsService: AppSett | |||||||
|         actual fun parseRelativeDate(dateString: String): String { |         actual fun parseRelativeDate(dateString: String): String { | ||||||
|             TODO("Not yet implemented") |             TODO("Not yet implemented") | ||||||
|         } |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| package bou.amine.apps.readerforselfossv2 |  | ||||||
|  |  | ||||||
| import kotlin.test.Test |  | ||||||
| import kotlin.test.assertTrue |  | ||||||
|  |  | ||||||
| class IosGreetingTest { |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     fun testExample() { |  | ||||||
|         assertTrue(Greeting().greeting().contains("iOS"), "Check iOS is mentioned") |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -2,7 +2,8 @@ package bou.amine.apps.readerforselfossv2.utils | |||||||
|  |  | ||||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||||
|  |  | ||||||
| actual class DateUtils actual constructor(actual val appSettingsService: AppSettingsService) { | actual class DateUtils { | ||||||
|  |     actual companion object { | ||||||
|         actual fun parseDate(dateString: String): Long { |         actual fun parseDate(dateString: String): Long { | ||||||
|             TODO("Not yet implemented") |             TODO("Not yet implemented") | ||||||
|         } |         } | ||||||
| @@ -10,5 +11,6 @@ actual class DateUtils actual constructor(actual val appSettingsService: AppSett | |||||||
|         actual fun parseRelativeDate(dateString: String): String { |         actual fun parseRelativeDate(dateString: String): String { | ||||||
|             TODO("Not yet implemented") |             TODO("Not yet implemented") | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
							
								
								
									
										5
									
								
								sonar-project.properties
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								sonar-project.properties
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | sonar.projectKey=RFS2 | ||||||
|  | sonar.coverage.jacoco.xmlReportPaths=build/reports/kover/merged/xml/report.xml | ||||||
|  | sonar.sourceEncoding=UTF-8 | ||||||
|  | sonar.sources=. | ||||||
|  | sonar.exclusions=shared/src/iosArm64Main/**, shared/src/iosX64Main/**, docs/**  | ||||||
		Reference in New Issue
	
	Block a user