Compare commits
	
		
			74 Commits
		
	
	
		
			054e936657
			...
			v122123371
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 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 | |||
|  | 023a30c008 | ||
|  | a2862a2587 | ||
| 1d2e5069b8 | |||
| a147646743 | |||
| c15bf44032 | |||
| 0bcd55bd4e | |||
| ebef0b3511 | |||
| 713ceb05bf | |||
| dc8381b661 | |||
| b5b820c64b | |||
| f7055626d9 | 
							
								
								
									
										15
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -6,16 +6,19 @@ steps: | ||||
|   - name: AnylyseBuildTest | ||||
|     image: mingc/android-build-box:latest | ||||
|     commands: | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Configure gradle..." | ||||
|       - 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 "Analysing..." | ||||
|       - ./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\"" | ||||
|       - ./gradlew sonarqube -Dsonar.projectKey=RFS2 -Dsonar.host.url=$SONAR_HOST_URL -Dsonar.login=$SONAR_LOGIN | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Building..." | ||||
|       - ./gradlew :androidApp:build -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" -P pushCache=false | ||||
|       - ./gradlew build | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Testing..." | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - ./gradlew test -PignoreGitVersion=true -P appLoginUrl="\"URL\"" -P appLoginUsername="\"LOGIN\"" -P appLoginPassword="\"PASS\"" -P pushCache=false | ||||
|       - ./gradlew test | ||||
|     environment: | ||||
|       SONAR_HOST_URL: | ||||
|         from_secret: sonarScannerHostUrl | ||||
| @@ -85,8 +88,12 @@ steps: | ||||
|   - name: build | ||||
|     image: mingc/android-build-box:latest | ||||
|     commands: | ||||
|       - echo "---------------------------------------------------------" | ||||
|       - echo "Configure gradle..." | ||||
|       - 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 "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 "Get Key" | ||||
|       - wget https://amine-louveau.fr/key | ||||
|   | ||||
							
								
								
									
										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. | ||||
|  | ||||
| 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,13 @@ | ||||
| import java.io.ByteArrayOutputStream | ||||
|  | ||||
| val ignoreGitVersion: String by project | ||||
| val acraVersion = "5.9.7" | ||||
|  | ||||
| plugins { | ||||
|     id("com.android.application") | ||||
|     kotlin("android") | ||||
|     kotlin("kapt") | ||||
|     id("com.mikepenz.aboutlibraries.plugin") | ||||
| } | ||||
|  | ||||
| fun Project.execWithOutput(cmd: String, ignore: Boolean = false): String { | ||||
| @@ -54,11 +56,15 @@ fun versionNameFromGit(): String { | ||||
| android { | ||||
|     compileOptions { | ||||
|         // Flag to enable support for the new language APIs | ||||
|         isCoreLibraryDesugaringEnabled = true | ||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
|         targetCompatibility = JavaVersion.VERSION_1_8 | ||||
|         sourceCompatibility = JavaVersion.VERSION_11 | ||||
|         targetCompatibility = JavaVersion.VERSION_11 | ||||
|     } | ||||
|     compileSdk = 31 | ||||
|  | ||||
|     // For Kotlin projects | ||||
|     kotlinOptions { | ||||
|         jvmTarget = "11" | ||||
|     } | ||||
|     compileSdk = 33 | ||||
|     buildToolsVersion = "31.0.0" | ||||
|     buildFeatures { | ||||
|         viewBinding = true | ||||
| @@ -66,7 +72,7 @@ android { | ||||
|     defaultConfig { | ||||
|         applicationId = "bou.amine.apps.readerforselfossv2.android" | ||||
|         minSdk = 21 | ||||
|         targetSdk = 31 | ||||
|         targetSdk = 33 | ||||
|         versionCode = versionCodeFromGit() | ||||
|         versionName = versionNameFromGit() | ||||
|  | ||||
| @@ -79,6 +85,11 @@ android { | ||||
|         // tests | ||||
|         testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" | ||||
|     } | ||||
|     packagingOptions { | ||||
|         resources { | ||||
|             excludes += "/META-INF/{AL2.0,LGPL2.1}" | ||||
|         } | ||||
|     } | ||||
|     buildTypes { | ||||
|         getByName("release") { | ||||
|             isMinifyEnabled = true | ||||
| @@ -86,9 +97,6 @@ android { | ||||
|             proguardFiles(getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro") | ||||
|         } | ||||
|         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") | ||||
| @@ -98,9 +106,6 @@ android { | ||||
|             dimension = "build" | ||||
|         } | ||||
|     } | ||||
|     kotlinOptions { | ||||
|         jvmTarget = "1.8" | ||||
|     } | ||||
|     namespace = "bou.amine.apps.readerforselfossv2.android" | ||||
|  | ||||
| } | ||||
| @@ -114,13 +119,6 @@ dependencies { | ||||
|  | ||||
|     implementation("androidx.preference:preference-ktx:1.1.1") | ||||
|  | ||||
|     // Testing | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-core:3.4.0-alpha02") | ||||
|     androidTestImplementation("androidx.test:runner:1.3.1-alpha02") | ||||
|     // Espresso-contrib for DatePicker, RecyclerView, Drawer actions, Accessibility checks, CountingIdlingResource | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-contrib:3.4.0-alpha02") | ||||
|     // Espresso-intents for validation and stubbing of Intents | ||||
|     androidTestImplementation("androidx.test.espresso:espresso-intents:3.4.0-alpha02") | ||||
|     implementation(fileTree(mapOf("include" to listOf("*.jar"), "dir" to "libs"))) | ||||
|  | ||||
|     // Android Support | ||||
| @@ -135,24 +133,12 @@ dependencies { | ||||
|     implementation("androidx.constraintlayout:constraintlayout:2.1.3") | ||||
|     implementation("org.jsoup:jsoup:1.14.3") | ||||
|  | ||||
|     coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:1.1.5") | ||||
|  | ||||
|     //multidex | ||||
|     implementation("androidx.multidex:multidex:2.0.1") | ||||
|  | ||||
|     // About | ||||
|     implementation("com.mikepenz:aboutlibraries-core:8.9.4") | ||||
|     implementation("com.mikepenz:aboutlibraries:8.9.4") | ||||
|     implementation("com.mikepenz:aboutlibraries-definitions:8.9.4") | ||||
|  | ||||
|     // Async | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.6.0") | ||||
|  | ||||
|     // Retrofit + http logging + okhttp | ||||
|     implementation("com.squareup.retrofit2:retrofit:2.9.0") | ||||
|     implementation("com.squareup.okhttp3:logging-interceptor:5.0.0-alpha.3") | ||||
|     implementation("com.squareup.retrofit2:converter-gson:2.9.0") | ||||
|     implementation("com.burgstaller:okhttp-digest:2.5") | ||||
|     implementation("com.mikepenz:aboutlibraries-core:10.5.1") | ||||
|     implementation("com.mikepenz:aboutlibraries:10.5.1") | ||||
|  | ||||
|     // Material-ish things | ||||
|     implementation("com.ashokvarma.android:bottom-navigation-bar:2.2.0") | ||||
| @@ -188,14 +174,47 @@ dependencies { | ||||
|  | ||||
|     implementation("androidx.core:core-ktx:1.8.0") | ||||
|  | ||||
|     // implementation("androidx.lifecycle:lifecycle-livedata-ktx:2.5.1") | ||||
|     // implementation("androidx.lifecycle:lifecycle-common-java8:2.5.1") | ||||
|     // implementation("androidx.lifecycle:lifecycle-runtime:2.5.1") | ||||
|     implementation("androidx.lifecycle:lifecycle-extensions:2.2.0") | ||||
|  | ||||
|     // Network information | ||||
|      implementation("com.github.ln-12:multiplatform-connectivity-status:1.3.0") | ||||
|  | ||||
|     // SQLDELIGHT | ||||
|     implementation("com.squareup.sqldelight:android-driver:1.5.3") | ||||
|     implementation("com.squareup.sqldelight:android-driver:1.5.4") | ||||
|  | ||||
|     //test | ||||
|     testImplementation("junit:junit:4.13.2") | ||||
|     testImplementation("io.mockk:mockk:1.12.0") | ||||
|     testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.0") | ||||
|     implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") | ||||
|  | ||||
|     implementation("ch.acra:acra-http:$acraVersion") | ||||
|     implementation("ch.acra:acra-toast:$acraVersion") | ||||
|  | ||||
|     // Matomo | ||||
|     implementation("com.github.matomo-org:matomo-sdk-android:4.1.4") | ||||
| } | ||||
|  | ||||
| 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>; | ||||
| } | ||||
|  | ||||
| -dontwarn okio.** | ||||
| -dontwarn retrofit2.Platform$Java8 | ||||
| -keep class retrofit.** { *; } | ||||
| -keepclasseswithmembers class * { | ||||
|     @retrofit.http.* <methods>; | ||||
| } | ||||
| -keepattributes *Annotation*,Signature | ||||
| -keepattributes Exceptions | ||||
| -dontwarn okio.** | ||||
| -dontwarn javax.annotation.Nullable | ||||
| -dontwarn javax.annotation.ParametersAreNonnullByDefault | ||||
|  | ||||
| @@ -90,3 +83,5 @@ | ||||
| # @Serializable and @Polymorphic are used at runtime for polymorphic serialization. | ||||
| -keepattributes RuntimeVisibleAnnotations,AnnotationDefault | ||||
|  | ||||
| -dontwarn io.mockk.** | ||||
| -keep class io.mockk.** { *; } | ||||
|   | ||||
| @@ -15,7 +15,8 @@ | ||||
|         android:supportsRtl="true" | ||||
|         android:networkSecurityConfig="@xml/network_security_config" | ||||
|         android:theme="@style/NoBar" | ||||
|         android:dataExtractionRules="@xml/data_extraction_rules"> | ||||
|         android:dataExtractionRules="@xml/data_extraction_rules" | ||||
|         android:configChanges="uiMode"> | ||||
|         <activity | ||||
|             android:name=".MainActivity" | ||||
|             android:theme="@style/SplashTheme" | ||||
|   | ||||
| @@ -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 { | ||||
|             try { | ||||
|                 val items = repository.getSpouts() | ||||
|                 if (items != null) { | ||||
|                 if (items.isNotEmpty()) { | ||||
|                     val itemsStrings = items.map { it.value.name } | ||||
|                     for ((key, value) in items) { | ||||
|                         spoutsKV[value.name] = key | ||||
|   | ||||
| @@ -15,12 +15,10 @@ import androidx.activity.result.contract.ActivityResultContracts | ||||
| import androidx.appcompat.app.ActionBarDrawerToggle | ||||
| import androidx.appcompat.app.AlertDialog | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.appcompat.app.AppCompatDelegate | ||||
| import androidx.appcompat.app.AppCompatDelegate.MODE_NIGHT_NO | ||||
| import androidx.appcompat.app.AppCompatDelegate.MODE_NIGHT_YES | ||||
| import androidx.appcompat.widget.SearchView | ||||
| import androidx.core.view.doOnNextLayout | ||||
| import androidx.drawerlayout.widget.DrawerLayout | ||||
| import androidx.lifecycle.lifecycleScope | ||||
| import androidx.recyclerview.widget.* | ||||
| import androidx.work.Constraints | ||||
| import androidx.work.ExistingPeriodicWorkPolicy | ||||
| @@ -37,7 +35,10 @@ import bou.amine.apps.readerforselfossv2.android.utils.bottombar.removeBadge | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.* | ||||
| import bou.amine.apps.readerforselfossv2.utils.ItemType | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import bou.amine.apps.readerforselfossv2.utils.getIcon | ||||
| import bou.amine.apps.readerforselfossv2.utils.longHash | ||||
| import com.ashokvarma.bottomnavigation.BottomNavigationBar | ||||
| import com.ashokvarma.bottomnavigation.BottomNavigationItem | ||||
| import com.ashokvarma.bottomnavigation.TextBadgeItem | ||||
| @@ -58,9 +59,14 @@ import com.mikepenz.materialdrawer.util.updateBadge | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| import org.acra.ACRA | ||||
| import org.acra.ktx.sendSilentlyWithAcra | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
| import org.matomo.sdk.Tracker | ||||
| import org.matomo.sdk.extra.TrackHelper | ||||
| import java.security.MessageDigest | ||||
| import java.util.concurrent.TimeUnit | ||||
|  | ||||
|  | ||||
| @@ -97,14 +103,16 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|     override val di by closestDI() | ||||
|     private val repository : Repository by instance() | ||||
|     private val appSettingsService : AppSettingsService by instance() | ||||
|     private val tracker : Tracker by instance() | ||||
|  | ||||
|     data class DrawerData(val tags: List<SelfossModel.Tag>?, val sources: List<SelfossModel.Source>?) | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         binding = ActivityHomeBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
|  | ||||
|         TrackHelper.track().screen("/home").with(tracker) | ||||
|  | ||||
|         fromTabShortcut =  intent.getIntExtra("shortcutTab", -1) != -1 | ||||
|         repository.offlineOverride =  intent.getBooleanExtra("startOffline", false) | ||||
|  | ||||
| @@ -183,8 +191,6 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|                         adapter.handleItemAtIndex(position) | ||||
|  | ||||
|                         reloadBadgeContent() | ||||
|  | ||||
|                         val tagHashes = i.tags.map { it.longHash() } | ||||
|                         tagsBadge = tagsBadge.map { | ||||
|                             if (tagHashes.contains(it.key)) { | ||||
| @@ -212,6 +218,16 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|         ItemTouchHelper(simpleItemTouchCallback).attachToRecyclerView(binding.recyclerView) | ||||
|     } | ||||
|  | ||||
|     private fun updateBottomBarBadgeCount(badge: TextBadgeItem, count: Int) { | ||||
|         if (count > 0) { | ||||
|             badge | ||||
|                 .setText(count.toString()) | ||||
|                 .maybeShow() | ||||
|         } else { | ||||
|             badge.removeBadge() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleBottomBar() { | ||||
|  | ||||
|         tabNewBadge = TextBadgeItem() | ||||
| @@ -224,6 +240,28 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|             .setText("") | ||||
|             .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 = | ||||
|             BottomNavigationItem( | ||||
|                 R.drawable.ic_tab_fiber_new_black_24dp, | ||||
| @@ -271,6 +309,8 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|         handleBottomBarActions() | ||||
|  | ||||
|         handleGDPRDialog(appSettingsService.settings.getBoolean("GDPR_shown", false)) | ||||
|  | ||||
|         handleRecurringTask() | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -280,6 +320,26 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|         getElementsAccordingToTab() | ||||
|     } | ||||
|  | ||||
|  | ||||
|     private fun handleGDPRDialog(GDPRShown: Boolean) { | ||||
|         val messageDigest: MessageDigest = MessageDigest.getInstance("SHA-256") | ||||
|         messageDigest.update(appSettingsService.getBaseUrl().toByteArray()) | ||||
|         ACRA.errorReporter.putCustomData("unique_id", String(messageDigest.digest())) | ||||
|         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() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun initDrawer() { | ||||
|         DrawerImageLoader.init(object : AbstractDrawerImageLoader() { | ||||
|             override fun set(imageView: ImageView, uri: Uri, placeholder: Drawable, tag: String?) { | ||||
| @@ -352,28 +412,15 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|         ) | ||||
|  | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             val drawerData = DrawerData(repository.getDBTags().map { it.toView() }, | ||||
|                                         repository.getDBSources().map { it.toView() }) | ||||
|             val tags = repository.getTags() | ||||
|             val sources = repository.getSources() | ||||
|             runOnUiThread { | ||||
|                 // TODO: All this logic should be handled by the repository, simplify and remove direct DB access | ||||
|                 // Only refresh if there is no data in the DB, or if the `UpdateSources` setting is enabled | ||||
|                 if (drawerData.sources?.isEmpty() == true || appSettingsService.isUpdateSourcesEnabled()) { | ||||
|                     drawerApiCalls(drawerData) | ||||
|                 } else { | ||||
|                     handleDrawerData(drawerData, loadedFromCache = true) | ||||
|                 } | ||||
|                 handleDrawerData(tags, sources) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun drawerApiCalls(drawerData: DrawerData) { | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             val apiDrawerData = DrawerData(repository.getTags(), repository.getSources()) | ||||
|             handleDrawerData(if (drawerData != apiDrawerData) apiDrawerData else drawerData) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleDrawerData(drawerData: DrawerData, loadedFromCache: Boolean = false) { | ||||
|     private fun handleDrawerData(tags: List<SelfossModel.Tag>, sources: List<SelfossModel.Source>) { | ||||
|         binding.mainDrawer.itemAdapter.clear() | ||||
|  | ||||
|         // Filters title with clear action | ||||
| @@ -387,24 +434,24 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|         } | ||||
|  | ||||
|         // Hidden tags | ||||
|         if (drawerData.tags != null && drawerData.tags.isNotEmpty() && appSettingsService.getHiddenTags().isNotEmpty()) { | ||||
|         if (tags.isNotEmpty() && appSettingsService.getHiddenTags().isNotEmpty()) { | ||||
|             secondaryItem( | ||||
|                 withDivider = true, | ||||
|                 R.string.drawer_item_hidden_tags, | ||||
|                 DRAWER_ID_HIDDEN_TAGS | ||||
|             ) | ||||
|             handleHiddenTags(drawerData.tags) | ||||
|             handleHiddenTags(tags) | ||||
|         } | ||||
|  | ||||
|         // Tags | ||||
|         secondaryItem(withDivider = true, R.string.drawer_item_tags, DRAWER_ID_TAGS) | ||||
|         if (drawerData.tags == null && !loadedFromCache) { | ||||
|         if (tags.isEmpty()) { | ||||
|             binding.mainDrawer.itemAdapter.add( | ||||
|                 SecondaryDrawerItem() | ||||
|                     .apply { nameRes = R.string.drawer_error_loading_tags; isSelectable = false } | ||||
|             ) | ||||
|         } else { | ||||
|             handleTags(drawerData.tags!!) | ||||
|             handleTags(tags) | ||||
|         } | ||||
|  | ||||
|         // Sources | ||||
| @@ -412,15 +459,15 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|             startActivity(Intent(v!!.context, SourcesActivity::class.java)) | ||||
|             false | ||||
|         } | ||||
|         if (drawerData.sources == null && !loadedFromCache) { | ||||
|         if (sources.isEmpty()) { | ||||
|             binding.mainDrawer.itemAdapter.add( | ||||
|                 SecondaryDrawerItem().apply { | ||||
|                     nameRes = R.string.drawer_error_loading_tags | ||||
|                     nameRes = R.string.drawer_error_loading_sources | ||||
|                     isSelectable = false | ||||
|                 } | ||||
|             ) | ||||
|         } else { | ||||
|             handleSources(drawerData.sources!!) | ||||
|             handleSources(sources) | ||||
|         } | ||||
|  | ||||
|         // About action | ||||
| @@ -465,6 +512,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|         val gdColor = try { | ||||
|             Color.parseColor(it.color) | ||||
|         } catch (e: IllegalArgumentException) { | ||||
|             e.sendSilentlyWithAcraWithName("color issue " + it.color) | ||||
|             resources.getColor(R.color.colorPrimary) | ||||
|         } | ||||
|         gd.setColor(gdColor) | ||||
| @@ -732,29 +780,12 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|     private fun reloadBadges() { | ||||
|         if (appSettingsService.isDisplayUnreadCountEnabled() || appSettingsService.isDisplayAllCountEnabled()) { | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|             CoroutineScope(Dispatchers.IO).launch { | ||||
|                 repository.reloadBadges() | ||||
|                 reloadBadgeContent() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun reloadBadgeContent() { | ||||
|         if (appSettingsService.isDisplayUnreadCountEnabled()) { | ||||
|             tabNewBadge | ||||
|                 .setText(repository.badgeUnread.toString()) | ||||
|                 .maybeShow() | ||||
|         } | ||||
|         if (appSettingsService.isDisplayAllCountEnabled()) { | ||||
|             tabArchiveBadge | ||||
|                 .setText(repository.badgeAll.toString()) | ||||
|                 .maybeShow() | ||||
|             tabStarredBadge | ||||
|                 .setText(repository.badgeStarred.toString()) | ||||
|                 .maybeShow() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun reloadTagsBadges() { | ||||
|         tagsBadge.forEach { | ||||
|             binding.mainDrawer.updateBadge(it.key, StringHolder(it.value.toString())) | ||||
| @@ -864,7 +895,9 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|                 return true | ||||
|             } | ||||
|             R.id.action_disconnect -> { | ||||
|                 appSettingsService.clearAll() | ||||
|                 CoroutineScope(Dispatchers.Main).launch { | ||||
|                     repository.logout() | ||||
|                 } | ||||
|                 val intent = Intent(this, LoginActivity::class.java) | ||||
|                 this.startActivity(intent) | ||||
|                 this@HomeActivity.finish() | ||||
| @@ -876,10 +909,10 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener, DIAwar | ||||
|  | ||||
|     private fun maxItemNumber(): Int = | ||||
|         when (elementsShown) { | ||||
|             ItemType.UNREAD -> repository.badgeUnread | ||||
|             ItemType.ALL -> repository.badgeAll | ||||
|             ItemType.STARRED -> repository.badgeStarred | ||||
|             else -> repository.badgeUnread // if !elementsShown then unread are fetched. | ||||
|             ItemType.UNREAD -> repository.badgeUnread.value | ||||
|             ItemType.ALL -> repository.badgeAll.value | ||||
|             ItemType.STARRED -> repository.badgeStarred.value | ||||
|             else -> repository.badgeUnread.value // if !elementsShown then unread are fetched. | ||||
|         } | ||||
|  | ||||
|     private fun updateItems(adapterItems: ArrayList<SelfossModel.Item>) { | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package bou.amine.apps.readerforselfossv2.android | ||||
|  | ||||
| import android.animation.Animator | ||||
| import android.animation.AnimatorListenerAdapter | ||||
| import android.annotation.SuppressLint | ||||
| import android.content.Intent | ||||
| import android.os.Bundle | ||||
| import android.text.TextUtils | ||||
| @@ -24,6 +25,12 @@ import kotlinx.coroutines.launch | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
| import org.matomo.sdk.Tracker | ||||
| import org.matomo.sdk.extra.DimensionQueue | ||||
| import org.matomo.sdk.extra.DownloadTracker | ||||
| import org.matomo.sdk.extra.TrackHelper | ||||
| import java.security.MessageDigest | ||||
|  | ||||
|  | ||||
| class LoginActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
| @@ -35,10 +42,17 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|     override val di by closestDI() | ||||
|     private val repository : Repository by instance() | ||||
|     private val appSettingsService : AppSettingsService by instance() | ||||
|     private val tracker : Tracker by instance() | ||||
|  | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_FOLLOW_SYSTEM) | ||||
|  | ||||
|         TrackHelper.track().download().identifier(DownloadTracker.Extra.ApkChecksum(applicationContext)) | ||||
|             .with(tracker) | ||||
|         TrackHelper.track().screen("/login").with(tracker) | ||||
|  | ||||
|         handleTheme() | ||||
|  | ||||
|         binding = ActivityLoginBinding.inflate(layoutInflater) | ||||
|         val view = binding.root | ||||
| @@ -56,6 +70,11 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|         handleActions() | ||||
|     } | ||||
|  | ||||
|     @SuppressLint("WrongConstant") // Constant is fetched from the settings | ||||
|     private fun handleTheme() { | ||||
|         AppCompatDelegate.setDefaultNightMode(appSettingsService.getCurrentTheme()) | ||||
|     } | ||||
|  | ||||
|     private fun handleActions() { | ||||
|  | ||||
|         binding.passwordView.setOnEditorActionListener( | ||||
| @@ -93,12 +112,24 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|     } | ||||
|  | ||||
|     private fun goToMain() { | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             repository.updateApiVersion() | ||||
|  | ||||
|             val messageDigest: MessageDigest = MessageDigest.getInstance("SHA-256") | ||||
|             messageDigest.update(appSettingsService.getBaseUrl().toByteArray()) | ||||
|             tracker.userId = String(messageDigest.digest()) | ||||
|  | ||||
|             val mDimensionQueue = DimensionQueue(tracker) | ||||
|             mDimensionQueue.add(1, appSettingsService.getApiVersion().toString()) | ||||
|  | ||||
|             tracker.isOptOut = !appSettingsService.isAnalyticsEnabled() | ||||
|         } | ||||
|         val intent = Intent(this, HomeActivity::class.java) | ||||
|         startActivity(intent) | ||||
|         finish() | ||||
|     } | ||||
|  | ||||
|     private fun preferenceError(t: Throwable) { | ||||
|     private fun preferenceError() { | ||||
|         appSettingsService.resetLoginInformation() | ||||
|  | ||||
|         binding.urlView.error = getString(R.string.wrong_infos) | ||||
| @@ -166,7 +197,7 @@ class LoginActivity : AppCompatActivity(), DIAware { | ||||
|                     goToMain() | ||||
|                 } else { | ||||
|                     CoroutineScope(Dispatchers.Main).launch { | ||||
|                         preferenceError(Exception("Not success")) | ||||
|                         preferenceError() | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|   | ||||
| @@ -12,7 +12,6 @@ import androidx.lifecycle.DefaultLifecycleObserver | ||||
| import androidx.lifecycle.LifecycleOwner | ||||
| import androidx.lifecycle.ProcessLifecycleOwner | ||||
| import androidx.multidex.MultiDexApplication | ||||
| import androidx.preference.PreferenceManager | ||||
| import bou.amine.apps.readerforselfossv2.DI.networkModule | ||||
| import bou.amine.apps.readerforselfossv2.android.viewmodel.AppViewModel | ||||
| import bou.amine.apps.readerforselfossv2.dao.DriverFactory | ||||
| @@ -28,8 +27,19 @@ import io.github.aakira.napier.DebugAntilog | ||||
| import io.github.aakira.napier.Napier | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.flow.MutableStateFlow | ||||
| import kotlinx.coroutines.launch | ||||
| import org.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.matomo.sdk.Matomo | ||||
| import org.matomo.sdk.Tracker | ||||
| import org.matomo.sdk.TrackerBuilder | ||||
|  | ||||
| class MyApp : MultiDexApplication(), DIAware { | ||||
|  | ||||
| @@ -37,9 +47,11 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|         import(networkModule) | ||||
|         bind<DriverFactory>() with singleton { DriverFactory(applicationContext) } | ||||
|         bind<ReaderForSelfossDB>() with singleton { ReaderForSelfossDB(driverFactory.createDriver()) } | ||||
|         bind<Repository>() with singleton { Repository(instance(), instance(), connectivityStatus, instance()) } | ||||
|         bind<Repository>() with singleton { Repository(instance(), instance(), isConnectionAvailable, instance()) } | ||||
|         bind<ConnectivityStatus>() with singleton { ConnectivityStatus(applicationContext) } | ||||
|         bind<AppViewModel>() with singleton { AppViewModel(repository = instance()) } | ||||
|         bind<Tracker>() with singleton { TrackerBuilder.createDefault("https://matomo.amine-louveau.fr/matomo.php", if (BuildConfig.DEBUG) 4 else 5).build( | ||||
|             Matomo.getInstance(applicationContext)) } | ||||
|     } | ||||
|  | ||||
|     private val repository: Repository by instance() | ||||
| @@ -47,35 +59,66 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|     private val connectivityStatus: ConnectivityStatus by instance() | ||||
|     private val driverFactory: DriverFactory by instance() | ||||
|  | ||||
|     // TODO: handle with the "previous" way | ||||
|     private val isConnectionAvailable: MutableStateFlow<Boolean> = MutableStateFlow(true) | ||||
|  | ||||
|     override fun onCreate() { | ||||
|         super.onCreate() | ||||
|         Napier.base(DebugAntilog()) | ||||
|  | ||||
|         initDrawerImageLoader() | ||||
|         if (!ACRA.isACRASenderServiceProcess()) { | ||||
|             initDrawerImageLoader() | ||||
|  | ||||
|         tryToHandleBug() | ||||
|             tryToHandleBug() | ||||
|  | ||||
|         handleNotificationChannels() | ||||
|             handleNotificationChannels() | ||||
|  | ||||
|         ProcessLifecycleOwner.get().lifecycle.addObserver(AppLifeCycleObserver(connectivityStatus, repository)) | ||||
|             ProcessLifecycleOwner.get().lifecycle.addObserver(AppLifeCycleObserver(connectivityStatus, repository)) | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             viewModel.networkAvailableProvider.collect { networkAvailable -> | ||||
|                 val toastMessage = if (networkAvailable) { | ||||
|                     repository.handleDBActions() | ||||
|                     R.string.network_connectivity_retrieved | ||||
|                 } else { | ||||
|                     R.string.network_connectivity_lost | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|                 viewModel.networkAvailableProvider.collect { networkAvailable -> | ||||
|                     val toastMessage = if (networkAvailable) { | ||||
|                         repository.handleDBActions() | ||||
|                         R.string.network_connectivity_retrieved | ||||
|                     } else { | ||||
|                         R.string.network_connectivity_lost | ||||
|                     } | ||||
|  | ||||
|                     Toast.makeText( | ||||
|                         applicationContext, | ||||
|                         toastMessage, | ||||
|                         Toast.LENGTH_SHORT | ||||
|                     ).show() | ||||
|                 } | ||||
|  | ||||
|                 Toast.makeText( | ||||
|                     applicationContext, | ||||
|                     toastMessage, | ||||
|                     Toast.LENGTH_SHORT | ||||
|                 ).show() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     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() { | ||||
| @@ -118,10 +161,9 @@ class MyApp : MultiDexApplication(), DIAware { | ||||
|         val oldHandler = Thread.getDefaultUncaughtExceptionHandler() | ||||
|  | ||||
|         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") | ||||
|                 }) { | ||||
|                 Unit | ||||
|             } else { | ||||
|                 oldHandler.uncaughtException(thread, e) | ||||
|             } | ||||
|   | ||||
| @@ -36,9 +36,9 @@ class ReaderActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|     private fun showMenuItem(willAddToFavorite: Boolean) { | ||||
|         if (willAddToFavorite) { | ||||
|             toolbarMenu.findItem(R.id.star).icon.setTint(Color.WHITE) | ||||
|             toolbarMenu.findItem(R.id.star).icon?.setTint(Color.WHITE) | ||||
|         } else { | ||||
|             toolbarMenu.findItem(R.id.star).icon.setTint(Color.RED) | ||||
|             toolbarMenu.findItem(R.id.star).icon?.setTint(Color.RED) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -56,20 +56,13 @@ class SourcesActivity : AppCompatActivity(), DIAware { | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
|             val response = repository.getSources() | ||||
|             if (response != null) { | ||||
|             if (response.isNotEmpty()) { | ||||
|                 items = response | ||||
|                 val mAdapter = SourcesListAdapter( | ||||
|                     this@SourcesActivity, items | ||||
|                 ) | ||||
|                 binding.recyclerView.adapter = mAdapter | ||||
|                 mAdapter.notifyDataSetChanged() | ||||
|                 if (items.isEmpty()) { | ||||
|                     Toast.makeText( | ||||
|                         this@SourcesActivity, | ||||
|                         R.string.nothing_here, | ||||
|                         Toast.LENGTH_SHORT | ||||
|                     ).show() | ||||
|                 } | ||||
|             } else { | ||||
|                 Toast.makeText( | ||||
|                     this@SourcesActivity, | ||||
|   | ||||
| @@ -10,9 +10,12 @@ import androidx.recyclerview.widget.RecyclerView | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.CardItemBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toTextDrawableString | ||||
| import bou.amine.apps.readerforselfossv2.android.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.circularBitmapDrawable | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openItemUrl | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.shareLink | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| @@ -59,7 +62,7 @@ class ItemCardAdapter( | ||||
|  | ||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText(repository.dateUtils) | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText() | ||||
|  | ||||
|             if (!appSettingsService.isFullHeightCardsEnabled()) { | ||||
|                 binding.itemImage.maxHeight = imageMaxHeight | ||||
| @@ -108,13 +111,11 @@ class ItemCardAdapter( | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.unstarr(item) | ||||
|                     } | ||||
|                     item.starred = false | ||||
|                     binding.favButton.isSelected = false | ||||
|                 } else { | ||||
|                     CoroutineScope(Dispatchers.IO).launch { | ||||
|                         repository.starr(item) | ||||
|                     } | ||||
|                     item.starred = true | ||||
|                     binding.favButton.isSelected = true | ||||
|                 } | ||||
|             } | ||||
|   | ||||
| @@ -51,7 +51,7 @@ class ItemListAdapter( | ||||
|  | ||||
|             binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent)) | ||||
|  | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText(repository.dateUtils) | ||||
|             binding.sourceTitleAndDate.text = itm.sourceAndDateText() | ||||
|  | ||||
|             if (itm.getThumbnail(repository.baseUrl).isEmpty()) { | ||||
|  | ||||
|   | ||||
| @@ -28,7 +28,7 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|         updateItems(this.items) | ||||
|     } | ||||
|  | ||||
|     private fun unmarkSnackbar(position: Int) { | ||||
|     private fun unmarkSnackbar(item: SelfossModel.Item, position: Int) { | ||||
|         val s = Snackbar | ||||
|             .make( | ||||
|                 app.findViewById(R.id.coordLayout), | ||||
| @@ -37,7 +37,7 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|             ) | ||||
|             .setAction(R.string.undo_string) { | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     unreadItemAtIndex(position, false) | ||||
|                     unreadItemAtIndex(item, position, false) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
| @@ -47,7 +47,7 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|         s.show() | ||||
|     } | ||||
|  | ||||
|     private fun markSnackbar(position: Int) { | ||||
|     private fun markSnackbar(item: SelfossModel.Item, position: Int) { | ||||
|         val s = Snackbar | ||||
|             .make( | ||||
|                 app.findViewById(R.id.coordLayout), | ||||
| @@ -55,7 +55,7 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|                 Snackbar.LENGTH_LONG | ||||
|             ) | ||||
|             .setAction(R.string.undo_string) { | ||||
|                 readItemAtIndex(position) | ||||
|                 readItemAtIndex(item, position, false) | ||||
|             } | ||||
|  | ||||
|         val view = s.view | ||||
| @@ -66,37 +66,36 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | ||||
|  | ||||
|     fun handleItemAtIndex(position: Int) { | ||||
|         if (items[position].unread) { | ||||
|             readItemAtIndex(position) | ||||
|             readItemAtIndex(items[position], position) | ||||
|         } else { | ||||
|             unreadItemAtIndex(position) | ||||
|             unreadItemAtIndex(items[position], position) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun readItemAtIndex(position: Int, showSnackbar: Boolean = true) { | ||||
|         val i = items[position] | ||||
|     private fun readItemAtIndex(item: SelfossModel.Item, position: Int, showSnackbar: Boolean = true) { | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             repository.markAsRead(i) | ||||
|             repository.markAsRead(item) | ||||
|         } | ||||
|         if (repository.displayedItems == ItemType.UNREAD) { | ||||
|             items.remove(i) | ||||
|             items.remove(item) | ||||
|             notifyItemRemoved(position) | ||||
|             updateItems(items) | ||||
|         } else { | ||||
|             notifyItemChanged(position) | ||||
|         } | ||||
|         if (showSnackbar) { | ||||
|             unmarkSnackbar(position) | ||||
|             unmarkSnackbar(item, position) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun unreadItemAtIndex(position: Int, showSnackbar: Boolean = true) { | ||||
|     private fun unreadItemAtIndex(item: SelfossModel.Item, position: Int, showSnackbar: Boolean = true) { | ||||
|         CoroutineScope(Dispatchers.IO).launch { | ||||
|             repository.unmarkAsRead(items[position]) | ||||
|             repository.unmarkAsRead(item) | ||||
|  | ||||
|         } | ||||
|         notifyItemChanged(position) | ||||
|         if (showSnackbar) { | ||||
|             markSnackbar(position) | ||||
|             markSnackbar(item, position) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -61,9 +61,13 @@ class SourcesListAdapter( | ||||
|         binding.sourceTitle.text = itm.title.getHtmlDecoded() | ||||
|     } | ||||
|  | ||||
|     override fun getItemId(position: Int) = position.toLong() | ||||
|  | ||||
|     override fun getItemViewType(position: Int) = position | ||||
|  | ||||
|     override fun getItemCount(): Int = items.size | ||||
|  | ||||
|     inner class ViewHolder(internal val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) { | ||||
|     inner class ViewHolder(private val mView: ConstraintLayout) : RecyclerView.ViewHolder(mView) { | ||||
|  | ||||
|         init { | ||||
|             handleClickListeners() | ||||
| @@ -74,13 +78,13 @@ class SourcesListAdapter( | ||||
|             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) | ||||
|  | ||||
|             deleteBtn.setOnClickListener { | ||||
|                 val (id) = items[adapterPosition] | ||||
|                 val (id) = items[bindingAdapterPosition] | ||||
|                 CoroutineScope(Dispatchers.IO).launch { | ||||
|                     val successfullyDeletedSource = repository.deleteSource(id) | ||||
|                     if (successfullyDeletedSource) { | ||||
|                         items.removeAt(adapterPosition) | ||||
|                         notifyItemRemoved(adapterPosition) | ||||
|                         notifyItemRangeChanged(adapterPosition, itemCount) | ||||
|                         items.removeAt(bindingAdapterPosition) | ||||
|                         notifyItemRemoved(bindingAdapterPosition) | ||||
|                         notifyItemRangeChanged(bindingAdapterPosition, itemCount) | ||||
|                     } else { | ||||
|                         Toast.makeText( | ||||
|                             app, | ||||
|   | ||||
| @@ -1,35 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.api.mercury | ||||
|  | ||||
| import com.google.gson.GsonBuilder | ||||
| import okhttp3.OkHttpClient | ||||
| import okhttp3.logging.HttpLoggingInterceptor | ||||
| import retrofit2.Call | ||||
| import retrofit2.Retrofit | ||||
| import retrofit2.converter.gson.GsonConverterFactory | ||||
|  | ||||
| class MercuryApi() { | ||||
|     private val service: MercuryService | ||||
|  | ||||
|     init { | ||||
|  | ||||
|         val interceptor = HttpLoggingInterceptor() | ||||
|         interceptor.level = HttpLoggingInterceptor.Level.NONE | ||||
|         val client = OkHttpClient.Builder().addInterceptor(interceptor).build() | ||||
|  | ||||
|         val gson = GsonBuilder() | ||||
|             .setLenient() | ||||
|             .create() | ||||
|         val retrofit = | ||||
|             Retrofit | ||||
|                 .Builder() | ||||
|                 .baseUrl("https://www.amine-louveau.fr") | ||||
|                 .client(client) | ||||
|                 .addConverterFactory(GsonConverterFactory.create(gson)) | ||||
|                 .build() | ||||
|         service = retrofit.create(MercuryService::class.java) | ||||
|     } | ||||
|  | ||||
|     fun parseUrl(url: String): Call<ParsedContent> { | ||||
|         return service.parseUrl(url) | ||||
|     } | ||||
| } | ||||
| @@ -1,59 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.api.mercury | ||||
|  | ||||
| import android.os.Parcel | ||||
| import android.os.Parcelable | ||||
| import com.google.gson.annotations.SerializedName | ||||
|  | ||||
| class ParsedContent( | ||||
|     @SerializedName("title") val title: String, | ||||
|     @SerializedName("content") val content: String?, | ||||
|     @SerializedName("date_published") val date_published: String, | ||||
|     @SerializedName("lead_image_url") val lead_image_url: String?, | ||||
|     @SerializedName("dek") val dek: String, | ||||
|     @SerializedName("url") val url: String, | ||||
|     @SerializedName("domain") val domain: String, | ||||
|     @SerializedName("excerpt") val excerpt: String, | ||||
|     @SerializedName("total_pages") val total_pages: Int, | ||||
|     @SerializedName("rendered_pages") val rendered_pages: Int, | ||||
|     @SerializedName("next_page_url") val next_page_url: String | ||||
| ) : Parcelable { | ||||
|  | ||||
|     companion object { | ||||
|         @JvmField | ||||
|         val CREATOR: Parcelable.Creator<ParsedContent> = | ||||
|             object : Parcelable.Creator<ParsedContent> { | ||||
|                 override fun createFromParcel(source: Parcel): ParsedContent = ParsedContent(source) | ||||
|                 override fun newArray(size: Int): Array<ParsedContent?> = arrayOfNulls(size) | ||||
|             } | ||||
|     } | ||||
|  | ||||
|     constructor(source: Parcel) : this( | ||||
|         title = source.readString().orEmpty(), | ||||
|         content = source.readString(), | ||||
|         date_published = source.readString().orEmpty(), | ||||
|         lead_image_url = source.readString(), | ||||
|         dek = source.readString().orEmpty(), | ||||
|         url = source.readString().orEmpty(), | ||||
|         domain = source.readString().orEmpty(), | ||||
|         excerpt = source.readString().orEmpty(), | ||||
|         total_pages = source.readInt(), | ||||
|         rendered_pages = source.readInt(), | ||||
|         next_page_url = source.readString().orEmpty() | ||||
|     ) | ||||
|  | ||||
|     override fun describeContents() = 0 | ||||
|  | ||||
|     override fun writeToParcel(dest: Parcel, flags: Int) { | ||||
|         dest.writeString(title) | ||||
|         dest.writeString(content) | ||||
|         dest.writeString(date_published) | ||||
|         dest.writeString(lead_image_url) | ||||
|         dest.writeString(dek) | ||||
|         dest.writeString(url) | ||||
|         dest.writeString(domain) | ||||
|         dest.writeString(excerpt) | ||||
|         dest.writeInt(total_pages) | ||||
|         dest.writeInt(rendered_pages) | ||||
|         dest.writeString(next_page_url) | ||||
|     } | ||||
| } | ||||
| @@ -1,10 +0,0 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.api.mercury | ||||
|  | ||||
| import retrofit2.Call | ||||
| import retrofit2.http.GET | ||||
| import retrofit2.http.Query | ||||
|  | ||||
| interface MercuryService { | ||||
|     @GET("parser.php") | ||||
|     fun parseUrl(@Query("link") link: String): Call<ParsedContent> | ||||
| } | ||||
| @@ -21,17 +21,17 @@ import androidx.core.widget.NestedScrollView | ||||
| import androidx.fragment.app.Fragment | ||||
| import bou.amine.apps.readerforselfossv2.android.ImageActivity | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.api.mercury.MercuryApi | ||||
| import bou.amine.apps.readerforselfossv2.android.api.mercury.ParsedContent | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.FragmentArticleBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.model.ParecelableItem | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toModel | ||||
| import bou.amine.apps.readerforselfossv2.android.model.toParcelable | ||||
| import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.glide.getBitmapInputStream | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.openInBrowserAsNewTask | ||||
| import bou.amine.apps.readerforselfossv2.android.utils.shareLink | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.repository.Repository | ||||
| import bou.amine.apps.readerforselfossv2.rest.MercuryApi | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import bou.amine.apps.readerforselfossv2.utils.getImages | ||||
| @@ -45,13 +45,12 @@ import com.google.android.material.floatingactionbutton.FloatingActionButton | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.launch | ||||
| import org.acra.ktx.sendSilentlyWithAcra | ||||
| import org.acra.ktx.sendWithAcra | ||||
| import org.kodein.di.DI | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.x.closestDI | ||||
| import org.kodein.di.instance | ||||
| import retrofit2.Call | ||||
| import retrofit2.Callback | ||||
| import retrofit2.Response | ||||
| import java.net.MalformedURLException | ||||
| import java.net.URL | ||||
| import java.util.* | ||||
| @@ -81,6 +80,9 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|     private var font = "" | ||||
|     private var staticBar = false | ||||
|  | ||||
|     private val mercuryApi : MercuryApi by instance() | ||||
|  | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|  | ||||
| @@ -101,7 +103,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|             contentText = item.content | ||||
|             contentTitle = item.title.getHtmlDecoded() | ||||
|             contentImage = item.getThumbnail(repository.baseUrl) | ||||
|             contentSource = item.sourceAndDateText(repository.dateUtils) | ||||
|             contentSource = item.sourceAndDateText() | ||||
|             allImages = item.getImages() | ||||
|  | ||||
|             fontSize = appSettingsService.getFontSize() | ||||
| @@ -113,6 +115,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 typeface = try { | ||||
|                     ResourcesCompat.getFont(requireContext(), resId)!! | ||||
|                 } catch (e: java.lang.Exception) { | ||||
|                     e.sendSilentlyWithAcraWithName("typeface") | ||||
|                     // Just to be sure | ||||
|                     null | ||||
|                 } | ||||
| @@ -218,6 +221,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|             ) | ||||
|  | ||||
|         } catch (e: InflateException) { | ||||
|             e.sendSilentlyWithAcraWithName("webview not available") | ||||
|             AlertDialog.Builder(requireContext()) | ||||
|                 .setMessage(requireContext().getString(R.string.webview_dialog_issue_message)) | ||||
|                 .setTitle(requireContext().getString(R.string.webview_dialog_issue_title)) | ||||
| @@ -249,88 +253,79 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|     private fun getContentFromMercury() { | ||||
|         if (repository.isNetworkAvailable()) { | ||||
|             binding.progressBar.visibility = View.VISIBLE | ||||
|             val parser = MercuryApi() | ||||
|  | ||||
|             parser.parseUrl(url).enqueue( | ||||
|                 object : Callback<ParsedContent> { | ||||
|                     override fun onResponse( | ||||
|                         call: Call<ParsedContent>, | ||||
|                         response: Response<ParsedContent> | ||||
|                     ) { | ||||
|                         // TODO: clean all the following after finding the mercury content issue | ||||
|                         try { | ||||
|                             if (response.body() != null && response.body()!!.content != null && !response.body()!!.content.isNullOrEmpty()) { | ||||
|                                 try { | ||||
|                                     binding.titleView.text = response.body()!!.title | ||||
|                                     if (typeface != null) { | ||||
|                                         binding.titleView.typeface = typeface | ||||
|                                     } | ||||
|                                     try { | ||||
|                                         // Note: Mercury may return relative urls... If it does the url val will not be changed. | ||||
|                                         URL(response.body()!!.url) | ||||
|                                         url = response.body()!!.url | ||||
|                                     } catch (e: MalformedURLException) { | ||||
|                                         // Mercury returned a relative url. We do nothing. | ||||
|                                     } | ||||
|                                 } catch (e: Exception) { | ||||
|             CoroutineScope(Dispatchers.Main).launch { | ||||
|                  val response = mercuryApi.query(url) | ||||
|                 if (response.success) { | ||||
|                     try { | ||||
|                         if (response.data != null && response.data!!.content != null && !response.data!!.content.isNullOrEmpty()) { | ||||
|                             try { | ||||
|                                 binding.titleView.text = response.data!!.title | ||||
|                                 if (typeface != null) { | ||||
|                                     binding.titleView.typeface = typeface | ||||
|                                 } | ||||
|  | ||||
|                                 try { | ||||
|                                     contentText = response.body()!!.content.orEmpty() | ||||
|                                     htmlToWebview() | ||||
|                                 } catch (e: Exception) { | ||||
|                                 } | ||||
|  | ||||
|                                 try { | ||||
|                                     if (response.body()!!.lead_image_url != null && !response.body()!!.lead_image_url.isNullOrEmpty() && context != null) { | ||||
|                                         binding.imageView.visibility = View.VISIBLE | ||||
|                                         try { | ||||
|                                             Glide | ||||
|                                                 .with(requireContext()) | ||||
|                                                 .asBitmap() | ||||
|                                                 .load( | ||||
|                                                     response.body()!!.lead_image_url.orEmpty() | ||||
|                                                 ) | ||||
|                                                 .apply(RequestOptions.fitCenterTransform()) | ||||
|                                                 .into(binding.imageView) | ||||
|                                         } catch (e: Exception) { | ||||
|                                         } | ||||
|                                     } else { | ||||
|                                         binding.imageView.visibility = View.GONE | ||||
|                                     } | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|                                 } | ||||
|  | ||||
|                                 try { | ||||
|                                     binding.nestedScrollView.scrollTo(0, 0) | ||||
|  | ||||
|                                     binding.progressBar.visibility = View.GONE | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|                                 } | ||||
|                             } else { | ||||
|                                 try { | ||||
|                                     openInBrowserAfterFailing() | ||||
|                                 } catch (e: Exception) { | ||||
|                                     if (context != null) { | ||||
|                                     } | ||||
|                                     // Note: Mercury may return relative urls... If it does the url val will not be changed. | ||||
|                                     URL(response.data!!.url) | ||||
|                                     url = response.data!!.url | ||||
|                                 } catch (e: MalformedURLException) { | ||||
|                                     // Mercury returned a relative url | ||||
|                                     e.sendSilentlyWithAcraWithName("getContentFromMercury > malformedurlexception") | ||||
|                                 } | ||||
|                             } catch (e: Exception) { | ||||
|                                 e.sendSilentlyWithAcraWithName("getContentFromMercury > title or typeface") | ||||
|                             } | ||||
|                         } catch (e: Exception) { | ||||
|                             if (context != null) { | ||||
|  | ||||
|                             try { | ||||
|                                 contentText = response.data!!.content.orEmpty() | ||||
|                                 htmlToWebview() | ||||
|                             } catch (e: Exception) { | ||||
|                                 e.sendSilentlyWithAcraWithName("getContentFromMercury > contenttext or html") | ||||
|                             } | ||||
|  | ||||
|                             try { | ||||
|                                 if (response.data!!.lead_image_url != null && !response.data!!.lead_image_url.isNullOrEmpty() && context != null) { | ||||
|                                     binding.imageView.visibility = View.VISIBLE | ||||
|                                     try { | ||||
|                                         Glide | ||||
|                                             .with(requireContext()) | ||||
|                                             .asBitmap() | ||||
|                                             .load( | ||||
|                                                 response.data!!.lead_image_url.orEmpty() | ||||
|                                             ) | ||||
|                                             .apply(RequestOptions.fitCenterTransform()) | ||||
|                                             .into(binding.imageView) | ||||
|                                     } catch (e: Exception) { | ||||
|                                         e.sendSilentlyWithAcraWithName("getContentFromMercury > glide lead image") | ||||
|                                     } | ||||
|                                 } else { | ||||
|                                     binding.imageView.visibility = View.GONE | ||||
|                                 } | ||||
|                             } catch (e: Exception) { | ||||
|                                 e.sendSilentlyWithAcraWithName("getContentFromMercury > outside glide lead image") | ||||
|                             } | ||||
|  | ||||
|                             try { | ||||
|                                 binding.nestedScrollView.scrollTo(0, 0) | ||||
|  | ||||
|                                 binding.progressBar.visibility = View.GONE | ||||
|                             } catch (e: Exception) { | ||||
|                                 e.sendSilentlyWithAcraWithName("getContentFromMercury > scrollview") | ||||
|                             } | ||||
|                         } else { | ||||
|                             try { | ||||
|                                 openInBrowserAfterFailing() | ||||
|                             } catch (e: Exception) { | ||||
|                                 e.sendSilentlyWithAcraWithName("getContentFromMercury > open after fail") | ||||
|                             } | ||||
|                         } | ||||
|                     } catch (e: Exception) { | ||||
|                         e.sendSilentlyWithAcraWithName("getContentFromMercury > whole thing") | ||||
|                     } | ||||
|  | ||||
|                     override fun onFailure( | ||||
|                         call: Call<ParsedContent>, | ||||
|                         t: Throwable | ||||
|                     ) = openInBrowserAfterFailing() | ||||
|                 } else { | ||||
|                     openInBrowserAfterFailing() | ||||
|                 } | ||||
|             ) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -354,6 +349,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         binding.webcontent.settings.javaScriptEnabled = false | ||||
|  | ||||
|         binding.webcontent.webViewClient = object : WebViewClient() { | ||||
|             @Deprecated("Deprecated in Java") | ||||
|             override fun shouldOverrideUrlLoading(view: WebView?, url : String): Boolean { | ||||
|                 if (binding.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||
|                     requireContext().startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) | ||||
| @@ -361,25 +357,32 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|                 return true | ||||
|             } | ||||
|  | ||||
|             @Deprecated("Deprecated in Java") | ||||
|             override fun shouldInterceptRequest(view: WebView?, url: String): WebResourceResponse? { | ||||
|                 val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||
|                 if (url.lowercase(Locale.US).contains(".jpg") || url.lowercase(Locale.US).contains(".jpeg")) { | ||||
|                     try { | ||||
|                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||
|                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.JPEG)) | ||||
|                     }catch ( e : ExecutionException) {} | ||||
|                     } catch ( e : ExecutionException) { | ||||
|                         e.sendSilentlyWithAcraWithName("shouldInterceptRequest > jpeg") | ||||
|                     } | ||||
|                 } | ||||
|                 else if (url.lowercase(Locale.US).contains(".png")) { | ||||
|                     try { | ||||
|                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||
|                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.PNG)) | ||||
|                     }catch ( e : ExecutionException) {} | ||||
|                     } catch ( e : ExecutionException) { | ||||
|                         e.sendSilentlyWithAcraWithName("shouldInterceptRequest > png") | ||||
|                     } | ||||
|                 } | ||||
|                 else if (url.lowercase(Locale.US).contains(".webp")) { | ||||
|                     try { | ||||
|                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||
|                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.WEBP)) | ||||
|                     }catch ( e : ExecutionException) {} | ||||
|                     } catch ( e : ExecutionException) { | ||||
|                         e.sendSilentlyWithAcraWithName("shouldInterceptRequest > webp") | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 return super.shouldInterceptRequest(view, url) | ||||
| @@ -387,7 +390,7 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|         } | ||||
|  | ||||
|         val gestureDetector = GestureDetector(activity, object : GestureDetector.SimpleOnGestureListener() { | ||||
|             override fun onSingleTapUp(e: MotionEvent?): Boolean { | ||||
|             override fun onSingleTapUp(e: MotionEvent): Boolean { | ||||
|                 return performClick() | ||||
|             } | ||||
|         }) | ||||
| @@ -403,11 +406,13 @@ class ArticleFragment : Fragment(), DIAware { | ||||
|             val itemUrl = URL(url) | ||||
|             baseUrl = itemUrl.protocol + "://" + itemUrl.host | ||||
|         } catch (e: MalformedURLException) { | ||||
|             e.sendSilentlyWithAcraWithName("htmlToWebview > item url") | ||||
|         } | ||||
|  | ||||
|         val fontName =  when (font) { | ||||
|             getString(R.string.open_sans_font_id) -> "Open Sans" | ||||
|             getString(R.string.roboto_font_id) -> "Roboto" | ||||
|             getString(R.string.source_code_pro_font_id) -> "Source Code Pro" | ||||
|             else -> "" | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -2,11 +2,13 @@ package bou.amine.apps.readerforselfossv2.android.model | ||||
|  | ||||
| import android.content.Context | ||||
| import android.webkit.URLUtil | ||||
| import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.utils.getImages | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||
| import com.bumptech.glide.request.RequestOptions | ||||
| import org.acra.ktx.sendSilentlyWithAcra | ||||
|  | ||||
| fun SelfossModel.Item.preloadImages(context: Context) : Boolean { | ||||
|     val imageUrls = this.getImages() | ||||
| @@ -23,6 +25,7 @@ fun SelfossModel.Item.preloadImages(context: Context) : Boolean { | ||||
|             } | ||||
|         } | ||||
|     } catch (e : Error) { | ||||
|         e.sendSilentlyWithAcraWithName("preloadImages") | ||||
|         return false | ||||
|     } | ||||
|  | ||||
| @@ -35,7 +38,7 @@ fun String.toTextDrawableString(): String { | ||||
|         try { | ||||
|             textDrawable.append(s[0]) | ||||
|         } catch (e: StringIndexOutOfBoundsException) { | ||||
|             // We do nothing | ||||
|             e.sendSilentlyWithAcraWithName("toTextDrawableString") | ||||
|         } | ||||
|     } | ||||
|     return textDrawable.toString() | ||||
|   | ||||
| @@ -3,7 +3,6 @@ package bou.amine.apps.readerforselfossv2.android.model | ||||
| import android.os.Parcel | ||||
| import android.os.Parcelable | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import com.google.gson.annotations.SerializedName | ||||
|  | ||||
| fun SelfossModel.Item.toParcelable() : ParecelableItem = | ||||
|     ParecelableItem( | ||||
| @@ -34,17 +33,17 @@ fun ParecelableItem.toModel() : SelfossModel.Item = | ||||
|         this.tags.split(",") | ||||
|     ) | ||||
| data class ParecelableItem( | ||||
|     @SerializedName("id") val id: Int, | ||||
|     @SerializedName("datetime") val datetime: String, | ||||
|     @SerializedName("title") val title: String, | ||||
|     @SerializedName("content") val content: String, | ||||
|     @SerializedName("unread") var unread: Boolean, | ||||
|     @SerializedName("starred") var starred: Boolean, | ||||
|     @SerializedName("thumbnail") val thumbnail: String?, | ||||
|     @SerializedName("icon") val icon: String?, | ||||
|     @SerializedName("link") val link: String, | ||||
|     @SerializedName("sourcetitle") val sourcetitle: String, | ||||
|     @SerializedName("tags") val tags: String | ||||
|     val id: Int, | ||||
|     val datetime: String, | ||||
|     val title: String, | ||||
|     val content: String, | ||||
|     var unread: Boolean, | ||||
|     var starred: Boolean, | ||||
|     val thumbnail: String?, | ||||
|     val icon: String?, | ||||
|     val link: String, | ||||
|     val sourcetitle: String, | ||||
|     val tags: String | ||||
| ) : Parcelable { | ||||
|  | ||||
|     companion object { | ||||
|   | ||||
| @@ -16,17 +16,30 @@ import androidx.preference.Preference | ||||
| import androidx.preference.PreferenceFragmentCompat | ||||
| import bou.amine.apps.readerforselfossv2.android.R | ||||
| import bou.amine.apps.readerforselfossv2.android.databinding.ActivitySettingsBinding | ||||
| import bou.amine.apps.readerforselfossv2.android.sendSilentlyWithAcraWithName | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import org.acra.ktx.sendSilentlyWithAcra | ||||
| import org.acra.ktx.sendWithAcra | ||||
| import org.kodein.di.DIAware | ||||
| import org.kodein.di.android.closestDI | ||||
| import org.kodein.di.instance | ||||
| import org.matomo.sdk.Tracker | ||||
| import org.matomo.sdk.extra.TrackHelper | ||||
|  | ||||
| private const val TITLE_TAG = "settingsActivityTitle" | ||||
|  | ||||
| class SettingsActivity : AppCompatActivity(), | ||||
|         PreferenceFragmentCompat.OnPreferenceStartFragmentCallback { | ||||
|         PreferenceFragmentCompat.OnPreferenceStartFragmentCallback, DIAware { | ||||
|     override val di by closestDI() | ||||
|  | ||||
|     private val tracker : Tracker by instance() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         val binding = ActivitySettingsBinding.inflate(layoutInflater) | ||||
|  | ||||
|         TrackHelper.track().screen("/settings").with(tracker) | ||||
|  | ||||
|         setContentView(binding.root) | ||||
|         if (savedInstanceState == null) { | ||||
|             supportFragmentManager | ||||
| @@ -91,6 +104,11 @@ class SettingsActivity : AppCompatActivity(), | ||||
|     class MainPreferenceFragment : PreferenceFragmentCompat() { | ||||
|         override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { | ||||
|             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 | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -107,6 +125,7 @@ class SettingsActivity : AppCompatActivity(), | ||||
|                                 val input: Int = (dest.toString() + source.toString()).toInt() | ||||
|                                 if (input in 1..200) return@InputFilter null | ||||
|                             } catch (nfe: NumberFormatException) { | ||||
|                                 nfe.sendSilentlyWithAcraWithName("GeneralPreferenceFragment") | ||||
|                                 Toast.makeText(activity, R.string.items_number_should_be_number, Toast.LENGTH_LONG).show() | ||||
|                             } | ||||
|                             "" | ||||
| @@ -130,6 +149,7 @@ class SettingsActivity : AppCompatActivity(), | ||||
|                         try { | ||||
|                             editText.textSize = editable.toString().toInt().toFloat() | ||||
|                         } catch (e: NumberFormatException) { | ||||
|                             e.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > afterTextChanged") | ||||
|                         } | ||||
|                     } | ||||
|                 } } | ||||
| @@ -139,6 +159,7 @@ class SettingsActivity : AppCompatActivity(), | ||||
|                                 val input = (dest.toString() + source.toString()).toInt() | ||||
|                                 if (input > 0) return@InputFilter null | ||||
|                             } catch (nfe: NumberFormatException) { | ||||
|                                 nfe.sendSilentlyWithAcraWithName("ArticleViewerPreferenceFragment > filters") | ||||
|                             } | ||||
|                             "" | ||||
|                         } | ||||
|   | ||||
| @@ -1,13 +1,9 @@ | ||||
| package bou.amine.apps.readerforselfossv2.android.utils | ||||
|  | ||||
| import android.app.Activity | ||||
| import android.app.PendingIntent | ||||
| import android.content.ActivityNotFoundException | ||||
| import android.content.Context | ||||
| import android.content.Intent | ||||
| import android.graphics.BitmapFactory | ||||
| import android.net.Uri | ||||
| import android.os.Build | ||||
| import android.text.Spannable | ||||
| import android.text.style.ClickableSpan | ||||
| import android.util.Patterns | ||||
|   | ||||
							
								
								
									
										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,7 @@ | ||||
| <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="M21,8c-1.45,0 -2.26,1.44 -1.93,2.51l-3.55,3.56c-0.3,-0.09 -0.74,-0.09 -1.04,0l-2.55,-2.55C12.27,10.45 11.46,9 10,9c-1.45,0 -2.27,1.44 -1.93,2.52l-4.56,4.55C2.44,15.74 1,16.55 1,18c0,1.1 0.9,2 2,2c1.45,0 2.26,-1.44 1.93,-2.51l4.55,-4.56c0.3,0.09 0.74,0.09 1.04,0l2.55,2.55C12.73,16.55 13.54,18 15,18c1.45,0 2.27,-1.44 1.93,-2.52l3.56,-3.55C21.56,12.26 23,11.45 23,10C23,8.9 22.1,8 21,8z"/> | ||||
|     <path android:fillColor="@android:color/white" android:pathData="M15,9l0.94,-2.07l2.06,-0.93l-2.06,-0.93l-0.94,-2.07l-0.92,2.07l-2.08,0.93l2.08,0.93z"/> | ||||
|     <path android:fillColor="@android:color/white" android:pathData="M3.5,11l0.5,-2l2,-0.5l-2,-0.5l-0.5,-2l-0.5,2l-2,0.5l2,0.5z"/> | ||||
| </vector> | ||||
							
								
								
									
										7
									
								
								androidApp/src/main/res/font/source_code_pro_medium.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								androidApp/src/main/res/font/source_code_pro_medium.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <font-family xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|         app:fontProviderAuthority="com.google.android.gms.fonts" | ||||
|         app:fontProviderPackage="com.google.android.gms" | ||||
|         app:fontProviderQuery="name=Source Code Pro&weight=500" | ||||
|         app:fontProviderCerts="@array/com_google_android_gms_fonts_certs"> | ||||
| </font-family> | ||||
| @@ -1,5 +1,5 @@ | ||||
| <?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" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent"> | ||||
| @@ -9,8 +9,8 @@ | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:layout_centerVertical="true" | ||||
|         android:layout_centerHorizontal="true" | ||||
|         android:background="@android:color/black" | ||||
|         android:adjustViewBounds="true" | ||||
|         android:background="@drawable/checkerboard" | ||||
|         app:srcCompat="@android:drawable/screen_background_dark" /> | ||||
|  | ||||
| </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||
| </RelativeLayout> | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Thème sombre</string> | ||||
|     <string name="mode_system">Utiliser les paramètres système</string> | ||||
|     <string name="mode_light">Thème clair</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
							
								
								
									
										8
									
								
								androidApp/src/main/res/values-night/strings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								androidApp/src/main/res/values-night/strings.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">深色模式</string> | ||||
|     <string name="mode_system">遵循系统设置</string> | ||||
|     <string name="mode_light">浅色模式</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -132,4 +132,10 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
|     <string-array name="ModeValues"> | ||||
|         <item>1</item> <!--MODE_NIGHT_NO--> | ||||
|         <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"> | ||||
|   | ||||
| @@ -3,5 +3,6 @@ | ||||
|     <array name="preloaded_fonts" translatable="false"> | ||||
|         <item>@font/open_sans</item> | ||||
|         <item>@font/roboto</item> | ||||
|         <item>@font/source_code_pro_medium</item> | ||||
|     </array> | ||||
| </resources> | ||||
|   | ||||
| @@ -4,5 +4,6 @@ | ||||
|         <item></item> | ||||
|         <item>@string/open_sans_font_id</item> | ||||
|         <item>@string/roboto_font_id</item> | ||||
|         <item>@string/source_code_pro_font_id</item> | ||||
|     </array> | ||||
| </resources> | ||||
| @@ -4,5 +4,6 @@ | ||||
|         <item>Systems</item> | ||||
|         <item>Open Sans</item> | ||||
|         <item>Roboto</item> | ||||
|         <item>Source Code Pro</item> | ||||
|     </array> | ||||
| </resources> | ||||
| @@ -63,6 +63,7 @@ | ||||
|     <string name="card_height_off">Card height will be fixed</string> | ||||
|     <string name="source_code">Source code</string> | ||||
|     <string name="drawer_error_loading_tags">Error loading tags…</string> | ||||
|     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||
|     <string name="drawer_item_filters">Filters</string> | ||||
|     <string name="drawer_action_clear">clear</string> | ||||
|     <string name="drawer_item_tags">Tags</string> | ||||
| @@ -109,7 +110,7 @@ | ||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> | ||||
|     <string name="loading_notification_title">Loading ...</string> | ||||
|     <string name="loading_notification_title">Loading …</string> | ||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</string> | ||||
|     <string name="notification_channel_sync">Sync notification</string> | ||||
|     <string name="new_items_channel_sync">New items notification</string> | ||||
| @@ -124,6 +125,7 @@ | ||||
|     <string name="reader_text_align_left">Align left</string> | ||||
|     <string name="reader_text_align_justify">Justify</string> | ||||
|     <string name="settings_reader_font">Reader font</string> | ||||
|     <string name="source_code_pro_font_id" translatable="false">source_code_pro_medium</string> | ||||
|     <string name="open_sans_font_id" translatable="false">open_sans</string> | ||||
|     <string name="roboto_font_id" translatable="false">roboto</string> | ||||
|     <string name="reader_static_bar_title">Static bottom bar in the article viewer</string> | ||||
| @@ -134,4 +136,9 @@ | ||||
|     <string name="mode_dark">Dark mode</string> | ||||
|     <string name="mode_system">Follow the system setting</string> | ||||
|     <string name="mode_light">Light mode</string> | ||||
|     <string name="pref_switch_enable_analytics">Enable analytics</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> | ||||
| </resources> | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto"> | ||||
|  | ||||
|     <EditTextPreference | ||||
|         android:inputType="number" | ||||
|         android:key="api_timeout" | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     android:title="@string/title_activity_settings"> | ||||
|  | ||||
|     <Preference | ||||
| @@ -17,9 +18,13 @@ | ||||
|         android:title="@string/pref_header_offline" | ||||
|         android:icon="@drawable/ic_signal_wifi_off_black_24dp" /> | ||||
|  | ||||
|     <Preference | ||||
|         android:fragment="bou.amine.apps.readerforselfossv2.android.settings.SettingsActivity$ThemePreferenceFragment" | ||||
|     <ListPreference | ||||
|         android:defaultValue="0" | ||||
|         android:entries="@array/ModeTitles" | ||||
|         android:entryValues="@array/ModeValues" | ||||
|         android:key="currentMode" | ||||
|         android:title="@string/pref_header_theme" | ||||
|         app:useSimpleSummaryProvider="false" | ||||
|         android:icon="@drawable/ic_color_lens_black_24dp" /> | ||||
|  | ||||
|     <Preference | ||||
| @@ -32,4 +37,18 @@ | ||||
|         android:title="@string/pref_header_experimental" | ||||
|         android:icon="@drawable/ic_widgets_black_24dp" /> | ||||
|  | ||||
|  | ||||
|     <SwitchPreference | ||||
|         android:defaultValue="false" | ||||
|         android:key="enable_analytics" | ||||
|         android:title="@string/pref_switch_enable_analytics" | ||||
|         android:icon="@drawable/ic_baseline_insights_24"/> | ||||
|  | ||||
|  | ||||
|     <SwitchPreference | ||||
|         android:defaultValue="false" | ||||
|         android:key="acra.disable" | ||||
|         android:title="@string/pref_switch_disable_acra" | ||||
|         android:icon="@drawable/ic_baseline_bug_report_24"/> | ||||
|  | ||||
| </PreferenceScreen> | ||||
							
								
								
									
										33
									
								
								androidApp/src/test/kotlin/DatesTest.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								androidApp/src/test/kotlin/DatesTest.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| package bou.amine.apps.readerforselfossv2.repository | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.utils.DateUtils | ||||
| import junit.framework.TestCase.assertEquals | ||||
| import kotlinx.datetime.LocalDateTime | ||||
| import kotlinx.datetime.TimeZone | ||||
| import kotlinx.datetime.toInstant | ||||
| import org.junit.Test | ||||
|  | ||||
| class DatesTest { | ||||
|  | ||||
|     private val v3Date = "2013-04-07T13:43:00+01:00" | ||||
|     private val v4Date = "2013-04-07 13:43:00" | ||||
|  | ||||
|     @Test | ||||
|     fun v3_date_should_be_parsed() { | ||||
|         val date = DateUtils.parseDate(v3Date) | ||||
|         val expected = LocalDateTime(2013, 4, 7, 13, 43, 0, 0).toInstant(TimeZone.of("UTC+1")) .toEpochMilliseconds() | ||||
|  | ||||
|         assertEquals(date, expected) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun v4_date_should_be_parsed() { | ||||
|         val date = DateUtils.parseDate(v4Date) | ||||
|         val expected = | ||||
|             LocalDateTime(2013, 4, 7, 13, 43, 0, 0).toInstant(TimeZone.currentSystemDefault()) | ||||
|                 .toEpochMilliseconds() | ||||
|  | ||||
|         assertEquals(date, expected) | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										1367
									
								
								androidApp/src/test/kotlin/RepositoryTest.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1367
									
								
								androidApp/src/test/kotlin/RepositoryTest.kt
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										57
									
								
								androidApp/src/test/kotlin/TestUtils.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								androidApp/src/test/kotlin/TestUtils.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | ||||
| package bou.amine.apps.readerforselfossv2.repository | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.dao.ITEM | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
|  | ||||
|  | ||||
| fun generateTestDBItems(item: FakeItemParameters = FakeItemParameters()): List<ITEM> { | ||||
|     return listOf( | ||||
|         ITEM( | ||||
|             id = item.id, | ||||
|             datetime = item.datetime, | ||||
|             title = item.title, | ||||
|             content = item.content, | ||||
|             unread = item.unread, | ||||
|             starred = item.starred, | ||||
|             thumbnail = item.thumbnail, | ||||
|             icon = item.icon, | ||||
|             link = item.link, | ||||
|             sourcetitle = item.sourcetitle, | ||||
|             tags = item.tags | ||||
|         ) | ||||
|     ) | ||||
| } | ||||
|  | ||||
| fun generateTestApiItem(item: FakeItemParameters = FakeItemParameters()): List<SelfossModel.Item> { | ||||
|     return listOf( | ||||
|         SelfossModel.Item( | ||||
|             id = item.id.toInt(), | ||||
|             datetime = item.datetime, | ||||
|             title = item.title, | ||||
|             content = item.content, | ||||
|             unread = item.unread, | ||||
|             starred = item.starred, | ||||
|             thumbnail = item.thumbnail, | ||||
|             icon = item.icon, | ||||
|             link = item.link, | ||||
|             sourcetitle = item.sourcetitle, | ||||
|             tags = item.tags.split(',') | ||||
|         ) | ||||
|     ) | ||||
| } | ||||
|  | ||||
| class FakeItemParameters { | ||||
|     var id = "20" | ||||
|     var datetime = "2022-09-09T03:32:01-04:00" | ||||
|     val title = "Etica della ricerca sotto i riflettori." | ||||
|     val content = | ||||
|         "<p><strong>Luigi Campanella, già Presidente SCI</strong></p>\n<p>L’etica della scienza è di certo ambito di cui continuiamo a scoprire nuovi aspetti e risvolti.</p>\n<p>L’ultimo è quello delle intelligenze artificiali capaci di creare opere complesse basate su immagini e parole memorizzate con il rischio di fake news e di contenuti disturbanti.</p>\n<p>Per evitare che ciò accada si sta procedendo filtrando secondo criteri di autocensura i dati da cui l’intelligenza artificiale parte.</p>\n<p>Comincia ad intravedersi un futuro prossimo di competizione fra autori umani ed artificiali nel quale sarà importante, quando i loro prodotti saranno indistinguibili, dichiararne l’origine.</p>\n<p>Come si comprende, si conferma che gli aspetti etici dell’innovazione e della ricerca si diversificato sempre di più.</p>\n<p>La biologia molecolare e la genetica già in passato hanno posto all’attenzione comune aspetti di etica della scienza che hanno indotto a nuove riflessioni circa i limiti delle ricerche.</p>\n<p>L’argomento, sempre attuale, torna sulle prime pagine a seguito della pubblicazione di una ricerca della Università di Cambridge che ha sviluppato una struttura cellulare di un topo con un cuore che batte regolarmente.</p>\n<img src=\"https://ilblogdellasci.files.wordpress.com/2022/09/image002-1.png?w=481\" alt=\"\" width=\"697\" height=\"430\" /><img src=\"https://ilblogdellasci.files.wordpress.com/2022/09/image003-1.png?w=906\" alt=\"\" /><p>Magdalena Zernicka-Goetz</p>\n<img src=\"https://ilblogdellasci.files.wordpress.com/2022/09/image004.jpg?w=474\" alt=\"\" width=\"622\" height=\"465\" /><p>Gianluca Amadei</p>\n<p>Del gruppo fa parte anche uno scienziato italiano Gianluca Amadei,che dinnanzi alle obiezioni di natura etica sulla realizzazione della vita artificiale si è affrettato a sostenere che non è creare nuove vite il fine primario della ricerca, ma quello di salvare quelle esistenti, di dare contributi essenziali alla medicina citando il caso del fallimento tuttora non interpretato di alcune gravidanze e di superare la sperimentazione animale, così contribuendo positivamente alla soluzione di un altro dilemma etico.</p>\n<p>L’embrione sintetico ha ovviamente come primo traguardo il contributo ai trapianti oggi drammaticamente carenti nell’offerta rispetto alla domanda, con attese fino a 4 anni per i trapianti di cuore ed a 2 anni per quelli di fegato. Il lavoro dovrebbe adesso continuare presso l’Ateneo di Padova per creare nuovi organi e nuovi farmaci.</p>" | ||||
|     var unread = true | ||||
|     var starred = true | ||||
|     val thumbnail = null | ||||
|     val icon = "ba79e238383ce83c23a169929c8906ef.png" | ||||
|     val link = | ||||
|         "https://ilblogdellasci.wordpress.com/2022/09/09/etica-della-ricerca-sotto-i-riflettori/" | ||||
|     var sourcetitle = "La Chimica e la Società" | ||||
|     var tags = "Chimica, Testing" | ||||
| } | ||||
| @@ -1,21 +1,20 @@ | ||||
| buildscript { | ||||
|     repositories { | ||||
|         gradlePluginPortal() | ||||
|         google() | ||||
|         mavenCentral() | ||||
|     } | ||||
|     dependencies { | ||||
|         classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.6.10") | ||||
|         classpath("com.android.tools.build:gradle:7.3.0") | ||||
|  | ||||
|         // sonarquve | ||||
|         classpath("org.sonarsource.scanner.gradle:sonarqube-gradle-plugin:3.4.0.2513") | ||||
|  | ||||
|         // SqlDelight | ||||
|         classpath("com.squareup.sqldelight:gradle-plugin:1.5.3") | ||||
|         classpath("com.squareup.sqldelight:gradle-plugin:1.5.4") | ||||
|     } | ||||
| } | ||||
|  | ||||
| plugins { | ||||
|     //trick: for the same plugin versions in all sub-modules | ||||
|     id("com.android.application").version("7.3.1").apply(false) | ||||
|     id("com.android.library").version("7.3.1").apply(false) | ||||
|     kotlin("android").version("1.7.20").apply(false) | ||||
|     kotlin("multiplatform").version("1.7.20").apply(false) | ||||
|     id("org.sonarqube").version("3.4.0.2513").apply(false) | ||||
|     id("com.mikepenz.aboutlibraries.plugin").version("10.5.1").apply(false) | ||||
| } | ||||
|  | ||||
| apply(plugin = "org.sonarqube") | ||||
|  | ||||
| allprojects { | ||||
| @@ -27,6 +26,7 @@ allprojects { | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| tasks.register("clean", Delete::class) { | ||||
|     delete(rootProject.buildDir) | ||||
| } | ||||
| } | ||||
|   | ||||
| @@ -11,14 +11,26 @@ | ||||
| # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects | ||||
| # org.gradle.parallel=true | ||||
| #Tue Mar 22 16:50:00 CET 2022 | ||||
| #Gradle | ||||
| org.gradle.jvmargs=-Xmx2048M -Dfile.encoding=UTF-8 -Dkotlin.daemon.jvm.options\="-Xmx2048M" | ||||
|  | ||||
| #Kotlin | ||||
| kotlin.code.style=official | ||||
| kotlin.mpp.enableCInteropCommonization=true | ||||
| org.gradle.jvmargs=-Xmx2048M -Dkotlin.daemon.jvm.options\="-Xmx2048M" | ||||
| kotlin.native.enableDependencyPropagation=false | ||||
|  | ||||
| #Android | ||||
| android.useAndroidX=true | ||||
| kotlin.native.enableDependencyPropagation=false | ||||
| #android.nonTransitiveRClass=true | ||||
| android.enableJetifier=true | ||||
|  | ||||
|  | ||||
| #MPP | ||||
| kotlin.mpp.enableCInteropCommonization=true | ||||
| kotlin.mpp.enableGranularSourceSetsMetadata=true | ||||
|  | ||||
|  | ||||
| org.gradle.parallel=true | ||||
| org.gradle.caching=true | ||||
| ignoreGitVersion=false | ||||
| kotlin.native.cacheKind.iosX64=none | ||||
| pushCache=true | ||||
|   | ||||
| @@ -1,383 +1,383 @@ | ||||
|     // !$*UTF8*$! | ||||
|     { | ||||
|     	archiveVersion = 1; | ||||
|     	classes = { | ||||
|     	}; | ||||
|     	objectVersion = 50; | ||||
|     	objects = { | ||||
| // !$*UTF8*$! | ||||
| { | ||||
| 	archiveVersion = 1; | ||||
| 	classes = { | ||||
| 	}; | ||||
| 	objectVersion = 50; | ||||
| 	objects = { | ||||
|  | ||||
|     /* Begin PBXBuildFile section */ | ||||
| 058557BB273AAA24004C7B11 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 058557BA273AAA24004C7B11 /* Assets.xcassets */; }; | ||||
| 058557D9273AAEEB004C7B11 /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */; }; | ||||
|     		2152FB042600AC8F00CF470E /* iOSApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2152FB032600AC8F00CF470E /* iOSApp.swift */; }; | ||||
|     		7555FF83242A565900829871 /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7555FF82242A565900829871 /* ContentView.swift */; }; | ||||
|     /* End PBXBuildFile section */ | ||||
| /* Begin PBXBuildFile section */ | ||||
| 		058557BB273AAA24004C7B11 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 058557BA273AAA24004C7B11 /* Assets.xcassets */; }; | ||||
| 		058557D9273AAEEB004C7B11 /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */; }; | ||||
| 		2152FB042600AC8F00CF470E /* iOSApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2152FB032600AC8F00CF470E /* iOSApp.swift */; }; | ||||
| 		7555FF83242A565900829871 /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7555FF82242A565900829871 /* ContentView.swift */; }; | ||||
| /* End PBXBuildFile section */ | ||||
|  | ||||
|     /* Begin PBXCopyFilesBuildPhase section */ | ||||
|     		7555FFB4242A642300829871 /* Embed Frameworks */ = { | ||||
|     			isa = PBXCopyFilesBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			dstPath = ""; | ||||
|     			dstSubfolderSpec = 10; | ||||
|     			files = ( | ||||
|     			); | ||||
|     			name = "Embed Frameworks"; | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     		}; | ||||
|     /* End PBXCopyFilesBuildPhase section */ | ||||
| /* Begin PBXCopyFilesBuildPhase section */ | ||||
| 		7555FFB4242A642300829871 /* Embed Frameworks */ = { | ||||
| 			isa = PBXCopyFilesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			dstPath = ""; | ||||
| 			dstSubfolderSpec = 10; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			name = "Embed Frameworks"; | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| /* End PBXCopyFilesBuildPhase section */ | ||||
|  | ||||
|     /* Begin PBXFileReference section */ | ||||
|     		058557BA273AAA24004C7B11 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; }; | ||||
| 058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = "Preview Assets.xcassets"; sourceTree = "<group>"; }; | ||||
|     		2152FB032600AC8F00CF470E /* iOSApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = iOSApp.swift; sourceTree = "<group>"; }; | ||||
|     		7555FF7B242A565900829871 /* iosApp.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = iosApp.app; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
|     		7555FF82242A565900829871 /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = "<group>"; }; | ||||
|     		7555FF8C242A565B00829871 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; }; | ||||
|     /* End PBXFileReference section */ | ||||
| /* Begin PBXFileReference section */ | ||||
| 		058557BA273AAA24004C7B11 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; }; | ||||
| 		058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = "Preview Assets.xcassets"; sourceTree = "<group>"; }; | ||||
| 		2152FB032600AC8F00CF470E /* iOSApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = iOSApp.swift; sourceTree = "<group>"; }; | ||||
| 		7555FF7B242A565900829871 /* iosApp.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = iosApp.app; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
| 		7555FF82242A565900829871 /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = "<group>"; }; | ||||
| 		7555FF8C242A565B00829871 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; }; | ||||
| /* End PBXFileReference section */ | ||||
|  | ||||
|     /* Begin PBXFrameworksBuildPhase section */ | ||||
|     		7555FF78242A565900829871 /* Frameworks */ = { | ||||
|     			isa = PBXFrameworksBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			files = ( | ||||
|     			); | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     		}; | ||||
|     /* End PBXFrameworksBuildPhase section */ | ||||
| /* Begin PBXFrameworksBuildPhase section */ | ||||
| 		7555FF78242A565900829871 /* Frameworks */ = { | ||||
| 			isa = PBXFrameworksBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| /* End PBXFrameworksBuildPhase section */ | ||||
|  | ||||
|     /* Begin PBXGroup section */ | ||||
|     		058557D7273AAEEB004C7B11 /* Preview Content */ = { | ||||
| 	isa = PBXGroup; | ||||
| 	children = ( | ||||
| 		058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */, | ||||
| 	); | ||||
| 	path = "Preview Content"; | ||||
| 	sourceTree = "<group>"; | ||||
| }; | ||||
|     		7555FF72242A565900829871 = { | ||||
|     			isa = PBXGroup; | ||||
|     			children = ( | ||||
|     				7555FF7D242A565900829871 /* iosApp */, | ||||
|     				7555FF7C242A565900829871 /* Products */, | ||||
|     				7555FFB0242A642200829871 /* Frameworks */, | ||||
|     			); | ||||
|     			sourceTree = "<group>"; | ||||
|     		}; | ||||
|     		7555FF7C242A565900829871 /* Products */ = { | ||||
|     			isa = PBXGroup; | ||||
|     			children = ( | ||||
|     				7555FF7B242A565900829871 /* iosApp.app */, | ||||
|     			); | ||||
|     			name = Products; | ||||
|     			sourceTree = "<group>"; | ||||
|     		}; | ||||
|     		7555FF7D242A565900829871 /* iosApp */ = { | ||||
|     			isa = PBXGroup; | ||||
|     			children = ( | ||||
|     				058557BA273AAA24004C7B11 /* Assets.xcassets */, | ||||
|     				7555FF82242A565900829871 /* ContentView.swift */, | ||||
|     				7555FF8C242A565B00829871 /* Info.plist */, | ||||
|     				2152FB032600AC8F00CF470E /* iOSApp.swift */, | ||||
| 		058557D7273AAEEB004C7B11 /* Preview Content */, | ||||
|     			); | ||||
|     			path = iosApp; | ||||
|     			sourceTree = "<group>"; | ||||
|     		}; | ||||
|     		7555FFB0242A642200829871 /* Frameworks */ = { | ||||
|     			isa = PBXGroup; | ||||
|     			children = ( | ||||
|     			); | ||||
|     			name = Frameworks; | ||||
|     			sourceTree = "<group>"; | ||||
|     		}; | ||||
|     /* End PBXGroup section */ | ||||
| /* Begin PBXGroup section */ | ||||
| 		058557D7273AAEEB004C7B11 /* Preview Content */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				058557D8273AAEEB004C7B11 /* Preview Assets.xcassets */, | ||||
| 			); | ||||
| 			path = "Preview Content"; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		7555FF72242A565900829871 = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				7555FF7D242A565900829871 /* iosApp */, | ||||
| 				7555FF7C242A565900829871 /* Products */, | ||||
| 				7555FFB0242A642200829871 /* Frameworks */, | ||||
| 			); | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		7555FF7C242A565900829871 /* Products */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				7555FF7B242A565900829871 /* iosApp.app */, | ||||
| 			); | ||||
| 			name = Products; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		7555FF7D242A565900829871 /* iosApp */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				058557BA273AAA24004C7B11 /* Assets.xcassets */, | ||||
| 				7555FF82242A565900829871 /* ContentView.swift */, | ||||
| 				7555FF8C242A565B00829871 /* Info.plist */, | ||||
| 				2152FB032600AC8F00CF470E /* iOSApp.swift */, | ||||
| 				058557D7273AAEEB004C7B11 /* Preview Content */, | ||||
| 			); | ||||
| 			path = iosApp; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		7555FFB0242A642200829871 /* Frameworks */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 			); | ||||
| 			name = Frameworks; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| /* End PBXGroup section */ | ||||
|  | ||||
|     /* Begin PBXNativeTarget section */ | ||||
|     		7555FF7A242A565900829871 /* iosApp */ = { | ||||
|     			isa = PBXNativeTarget; | ||||
|     			buildConfigurationList = 7555FFA5242A565B00829871 /* Build configuration list for PBXNativeTarget "iosApp" */; | ||||
|     			buildPhases = ( | ||||
|     				7555FFB5242A651A00829871 /* ShellScript */, | ||||
|     				7555FF77242A565900829871 /* Sources */, | ||||
|     				7555FF78242A565900829871 /* Frameworks */, | ||||
|     				7555FF79242A565900829871 /* Resources */, | ||||
|     				7555FFB4242A642300829871 /* Embed Frameworks */, | ||||
|     			); | ||||
|     			buildRules = ( | ||||
|     			); | ||||
|     			dependencies = ( | ||||
|     			); | ||||
|     			name = iosApp; | ||||
|     			productName = iosApp; | ||||
|     			productReference = 7555FF7B242A565900829871 /* iosApp.app */; | ||||
|     			productType = "com.apple.product-type.application"; | ||||
|     		}; | ||||
|     /* End PBXNativeTarget section */ | ||||
| /* Begin PBXNativeTarget section */ | ||||
| 		7555FF7A242A565900829871 /* iosApp */ = { | ||||
| 			isa = PBXNativeTarget; | ||||
| 			buildConfigurationList = 7555FFA5242A565B00829871 /* Build configuration list for PBXNativeTarget "iosApp" */; | ||||
| 			buildPhases = ( | ||||
| 				7555FFB5242A651A00829871 /* ShellScript */, | ||||
| 				7555FF77242A565900829871 /* Sources */, | ||||
| 				7555FF78242A565900829871 /* Frameworks */, | ||||
| 				7555FF79242A565900829871 /* Resources */, | ||||
| 				7555FFB4242A642300829871 /* Embed Frameworks */, | ||||
| 			); | ||||
| 			buildRules = ( | ||||
| 			); | ||||
| 			dependencies = ( | ||||
| 			); | ||||
| 			name = iosApp; | ||||
| 			productName = iosApp; | ||||
| 			productReference = 7555FF7B242A565900829871 /* iosApp.app */; | ||||
| 			productType = "com.apple.product-type.application"; | ||||
| 		}; | ||||
| /* End PBXNativeTarget section */ | ||||
|  | ||||
|     /* Begin PBXProject section */ | ||||
|     		7555FF73242A565900829871 /* Project object */ = { | ||||
|     			isa = PBXProject; | ||||
|     			attributes = { | ||||
|     				LastSwiftUpdateCheck = 1130; | ||||
|     				LastUpgradeCheck = 1130; | ||||
|     				ORGANIZATIONNAME = orgName; | ||||
|     				TargetAttributes = { | ||||
|     					7555FF7A242A565900829871 = { | ||||
|     						CreatedOnToolsVersion = 11.3.1; | ||||
|     					}; | ||||
|     				}; | ||||
|     			}; | ||||
|     			buildConfigurationList = 7555FF76242A565900829871 /* Build configuration list for PBXProject "iosApp" */; | ||||
|     			compatibilityVersion = "Xcode 9.3"; | ||||
|     			developmentRegion = en; | ||||
|     			hasScannedForEncodings = 0; | ||||
|     			knownRegions = ( | ||||
|     				en, | ||||
|     				Base, | ||||
|     			); | ||||
|     			mainGroup = 7555FF72242A565900829871; | ||||
|     			productRefGroup = 7555FF7C242A565900829871 /* Products */; | ||||
|     			projectDirPath = ""; | ||||
|     			projectRoot = ""; | ||||
|     			targets = ( | ||||
|     				7555FF7A242A565900829871 /* iosApp */, | ||||
|     			); | ||||
|     		}; | ||||
|     /* End PBXProject section */ | ||||
| /* Begin PBXProject section */ | ||||
| 		7555FF73242A565900829871 /* Project object */ = { | ||||
| 			isa = PBXProject; | ||||
| 			attributes = { | ||||
| 				LastSwiftUpdateCheck = 1130; | ||||
| 				LastUpgradeCheck = 1130; | ||||
| 				ORGANIZATIONNAME = orgName; | ||||
| 				TargetAttributes = { | ||||
| 					7555FF7A242A565900829871 = { | ||||
| 						CreatedOnToolsVersion = 11.3.1; | ||||
| 					}; | ||||
| 				}; | ||||
| 			}; | ||||
| 			buildConfigurationList = 7555FF76242A565900829871 /* Build configuration list for PBXProject "iosApp" */; | ||||
| 			compatibilityVersion = "Xcode 9.3"; | ||||
| 			developmentRegion = en; | ||||
| 			hasScannedForEncodings = 0; | ||||
| 			knownRegions = ( | ||||
| 				en, | ||||
| 				Base, | ||||
| 			); | ||||
| 			mainGroup = 7555FF72242A565900829871; | ||||
| 			productRefGroup = 7555FF7C242A565900829871 /* Products */; | ||||
| 			projectDirPath = ""; | ||||
| 			projectRoot = ""; | ||||
| 			targets = ( | ||||
| 				7555FF7A242A565900829871 /* iosApp */, | ||||
| 			); | ||||
| 		}; | ||||
| /* End PBXProject section */ | ||||
|  | ||||
|     /* Begin PBXResourcesBuildPhase section */ | ||||
|     		7555FF79242A565900829871 /* Resources */ = { | ||||
|     			isa = PBXResourcesBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			files = ( | ||||
| 		058557D9273AAEEB004C7B11 /* Preview Assets.xcassets in Resources */, | ||||
| 		058557BB273AAA24004C7B11 /* Assets.xcassets in Resources */, | ||||
|     			); | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     		}; | ||||
|     /* End PBXResourcesBuildPhase section */ | ||||
| /* Begin PBXResourcesBuildPhase section */ | ||||
| 		7555FF79242A565900829871 /* Resources */ = { | ||||
| 			isa = PBXResourcesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 				058557D9273AAEEB004C7B11 /* Preview Assets.xcassets in Resources */, | ||||
| 				058557BB273AAA24004C7B11 /* Assets.xcassets in Resources */, | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| /* End PBXResourcesBuildPhase section */ | ||||
|  | ||||
|     /* Begin PBXShellScriptBuildPhase section */ | ||||
|     		7555FFB5242A651A00829871 /* ShellScript */ = { | ||||
|     			isa = PBXShellScriptBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			files = ( | ||||
|     			); | ||||
|     			inputFileListPaths = ( | ||||
|     			); | ||||
|     			inputPaths = ( | ||||
|     			); | ||||
|     			outputFileListPaths = ( | ||||
|     			); | ||||
|     			outputPaths = ( | ||||
|     			); | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     			shellPath = /bin/sh; | ||||
|     			shellScript = "cd \"$SRCROOT/..\"\n./gradlew :shared:embedAndSignAppleFrameworkForXcode\n"; | ||||
|     		}; | ||||
|     /* End PBXShellScriptBuildPhase section */ | ||||
| /* Begin PBXShellScriptBuildPhase section */ | ||||
| 		7555FFB5242A651A00829871 /* ShellScript */ = { | ||||
| 			isa = PBXShellScriptBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			inputFileListPaths = ( | ||||
| 			); | ||||
| 			inputPaths = ( | ||||
| 			); | ||||
| 			outputFileListPaths = ( | ||||
| 			); | ||||
| 			outputPaths = ( | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 			shellPath = /bin/sh; | ||||
| 			shellScript = "cd \"$SRCROOT/..\"\n./gradlew :shared:embedAndSignAppleFrameworkForXcode\n"; | ||||
| 		}; | ||||
| /* End PBXShellScriptBuildPhase section */ | ||||
|  | ||||
|     /* Begin PBXSourcesBuildPhase section */ | ||||
|     		7555FF77242A565900829871 /* Sources */ = { | ||||
|     			isa = PBXSourcesBuildPhase; | ||||
|     			buildActionMask = 2147483647; | ||||
|     			files = ( | ||||
|     				2152FB042600AC8F00CF470E /* iOSApp.swift in Sources */, | ||||
|     				7555FF83242A565900829871 /* ContentView.swift in Sources */, | ||||
|     			); | ||||
|     			runOnlyForDeploymentPostprocessing = 0; | ||||
|     		}; | ||||
|     /* End PBXSourcesBuildPhase section */ | ||||
| /* Begin PBXSourcesBuildPhase section */ | ||||
| 		7555FF77242A565900829871 /* Sources */ = { | ||||
| 			isa = PBXSourcesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 				2152FB042600AC8F00CF470E /* iOSApp.swift in Sources */, | ||||
| 				7555FF83242A565900829871 /* ContentView.swift in Sources */, | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| /* End PBXSourcesBuildPhase section */ | ||||
|  | ||||
|     /* Begin XCBuildConfiguration section */ | ||||
|     		7555FFA3242A565B00829871 /* Debug */ = { | ||||
|     			isa = XCBuildConfiguration; | ||||
|     			buildSettings = { | ||||
|     				ALWAYS_SEARCH_USER_PATHS = NO; | ||||
|     				CLANG_ANALYZER_NONNULL = YES; | ||||
|     				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
|     				CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; | ||||
|     				CLANG_CXX_LIBRARY = "libc++"; | ||||
|     				CLANG_ENABLE_MODULES = YES; | ||||
|     				CLANG_ENABLE_OBJC_ARC = YES; | ||||
|     				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
|     				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; | ||||
|     				CLANG_WARN_BOOL_CONVERSION = YES; | ||||
|     				CLANG_WARN_COMMA = YES; | ||||
|     				CLANG_WARN_CONSTANT_CONVERSION = YES; | ||||
|     				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; | ||||
|     				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; | ||||
|     				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
|     				CLANG_WARN_EMPTY_BODY = YES; | ||||
|     				CLANG_WARN_ENUM_CONVERSION = YES; | ||||
|     				CLANG_WARN_INFINITE_RECURSION = YES; | ||||
|     				CLANG_WARN_INT_CONVERSION = YES; | ||||
|     				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; | ||||
|     				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
|     				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
|     				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
|     				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
|     				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
|     				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
|     				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
|     				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
|     				CLANG_WARN_UNREACHABLE_CODE = YES; | ||||
|     				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; | ||||
|     				COPY_PHASE_STRIP = NO; | ||||
|     				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; | ||||
|     				ENABLE_STRICT_OBJC_MSGSEND = YES; | ||||
|     				ENABLE_TESTABILITY = YES; | ||||
|     				GCC_C_LANGUAGE_STANDARD = gnu11; | ||||
|     				GCC_DYNAMIC_NO_PIC = NO; | ||||
|     				GCC_NO_COMMON_BLOCKS = YES; | ||||
|     				GCC_OPTIMIZATION_LEVEL = 0; | ||||
|     				GCC_PREPROCESSOR_DEFINITIONS = ( | ||||
|     					"DEBUG=1", | ||||
|     					"$(inherited)", | ||||
|     				); | ||||
|     				GCC_WARN_64_TO_32_BIT_CONVERSION = YES; | ||||
|     				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; | ||||
|     				GCC_WARN_UNDECLARED_SELECTOR = YES; | ||||
|     				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||
|     				GCC_WARN_UNUSED_FUNCTION = YES; | ||||
|     				GCC_WARN_UNUSED_VARIABLE = YES; | ||||
|     				IPHONEOS_DEPLOYMENT_TARGET = 14.1; | ||||
|     				MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; | ||||
|     				MTL_FAST_MATH = YES; | ||||
|     				ONLY_ACTIVE_ARCH = YES; | ||||
|     				SDKROOT = iphoneos; | ||||
|     				SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; | ||||
|     				SWIFT_OPTIMIZATION_LEVEL = "-Onone"; | ||||
|     			}; | ||||
|     			name = Debug; | ||||
|     		}; | ||||
|     		7555FFA4242A565B00829871 /* Release */ = { | ||||
|     			isa = XCBuildConfiguration; | ||||
|     			buildSettings = { | ||||
|     				ALWAYS_SEARCH_USER_PATHS = NO; | ||||
|     				CLANG_ANALYZER_NONNULL = YES; | ||||
|     				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
|     				CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; | ||||
|     				CLANG_CXX_LIBRARY = "libc++"; | ||||
|     				CLANG_ENABLE_MODULES = YES; | ||||
|     				CLANG_ENABLE_OBJC_ARC = YES; | ||||
|     				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
|     				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; | ||||
|     				CLANG_WARN_BOOL_CONVERSION = YES; | ||||
|     				CLANG_WARN_COMMA = YES; | ||||
|     				CLANG_WARN_CONSTANT_CONVERSION = YES; | ||||
|     				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; | ||||
|     				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; | ||||
|     				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
|     				CLANG_WARN_EMPTY_BODY = YES; | ||||
|     				CLANG_WARN_ENUM_CONVERSION = YES; | ||||
|     				CLANG_WARN_INFINITE_RECURSION = YES; | ||||
|     				CLANG_WARN_INT_CONVERSION = YES; | ||||
|     				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; | ||||
|     				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
|     				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
|     				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
|     				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
|     				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
|     				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
|     				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
|     				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
|     				CLANG_WARN_UNREACHABLE_CODE = YES; | ||||
|     				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; | ||||
|     				COPY_PHASE_STRIP = NO; | ||||
|     				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; | ||||
|     				ENABLE_NS_ASSERTIONS = NO; | ||||
|     				ENABLE_STRICT_OBJC_MSGSEND = YES; | ||||
|     				GCC_C_LANGUAGE_STANDARD = gnu11; | ||||
|     				GCC_NO_COMMON_BLOCKS = YES; | ||||
|     				GCC_WARN_64_TO_32_BIT_CONVERSION = YES; | ||||
|     				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; | ||||
|     				GCC_WARN_UNDECLARED_SELECTOR = YES; | ||||
|     				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||
|     				GCC_WARN_UNUSED_FUNCTION = YES; | ||||
|     				GCC_WARN_UNUSED_VARIABLE = YES; | ||||
|     				IPHONEOS_DEPLOYMENT_TARGET = 14.1; | ||||
|     				MTL_ENABLE_DEBUG_INFO = NO; | ||||
|     				MTL_FAST_MATH = YES; | ||||
|     				SDKROOT = iphoneos; | ||||
|     				SWIFT_COMPILATION_MODE = wholemodule; | ||||
|     				SWIFT_OPTIMIZATION_LEVEL = "-O"; | ||||
|     				VALIDATE_PRODUCT = YES; | ||||
|     			}; | ||||
|     			name = Release; | ||||
|     		}; | ||||
|     		7555FFA6242A565B00829871 /* Debug */ = { | ||||
|     			isa = XCBuildConfiguration; | ||||
|     			buildSettings = { | ||||
|     				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
|     				CODE_SIGN_STYLE = Automatic; | ||||
| 		DEVELOPMENT_ASSET_PATHS = "\"iosApp/Preview Content\""; | ||||
|     				ENABLE_PREVIEWS = YES; | ||||
|     				FRAMEWORK_SEARCH_PATHS = "$(SRCROOT)/../shared/build/xcode-frameworks/$(CONFIGURATION)/$(SDK_NAME)"; | ||||
|     				INFOPLIST_FILE = iosApp/Info.plist; | ||||
|     				LD_RUNPATH_SEARCH_PATHS = ( | ||||
|     					"$(inherited)", | ||||
|     					"@executable_path/Frameworks", | ||||
|     				); | ||||
|     				OTHER_LDFLAGS = ( | ||||
|     					"$(inherited)", | ||||
|     					"-framework", | ||||
|     					shared, | ||||
|     				); | ||||
|     				PRODUCT_BUNDLE_IDENTIFIER = orgIdentifier.iosApp; | ||||
|     				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
|     				SWIFT_VERSION = 5.0; | ||||
|     				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
|     			}; | ||||
|     			name = Debug; | ||||
|     		}; | ||||
|     		7555FFA7242A565B00829871 /* Release */ = { | ||||
|     			isa = XCBuildConfiguration; | ||||
|     			buildSettings = { | ||||
|     				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
|     				CODE_SIGN_STYLE = Automatic; | ||||
|     				DEVELOPMENT_ASSET_PATHS = "\"iosApp/Preview Content\""; | ||||
|     				ENABLE_PREVIEWS = YES; | ||||
|     				FRAMEWORK_SEARCH_PATHS = "$(SRCROOT)/../shared/build/xcode-frameworks/$(CONFIGURATION)/$(SDK_NAME)"; | ||||
|     				INFOPLIST_FILE = iosApp/Info.plist; | ||||
|     				LD_RUNPATH_SEARCH_PATHS = ( | ||||
|     					"$(inherited)", | ||||
|     					"@executable_path/Frameworks", | ||||
|     				); | ||||
|     				OTHER_LDFLAGS = ( | ||||
|     					"$(inherited)", | ||||
|     					"-framework", | ||||
|     					shared, | ||||
|     				); | ||||
|     				PRODUCT_BUNDLE_IDENTIFIER = orgIdentifier.iosApp; | ||||
|     				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
|     				SWIFT_VERSION = 5.0; | ||||
|     				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
|     			}; | ||||
|     			name = Release; | ||||
|     		}; | ||||
|     /* End XCBuildConfiguration section */ | ||||
| /* Begin XCBuildConfiguration section */ | ||||
| 		7555FFA3242A565B00829871 /* Debug */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ALWAYS_SEARCH_USER_PATHS = NO; | ||||
| 				CLANG_ANALYZER_NONNULL = YES; | ||||
| 				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
| 				CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; | ||||
| 				CLANG_CXX_LIBRARY = "libc++"; | ||||
| 				CLANG_ENABLE_MODULES = YES; | ||||
| 				CLANG_ENABLE_OBJC_ARC = YES; | ||||
| 				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
| 				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; | ||||
| 				CLANG_WARN_BOOL_CONVERSION = YES; | ||||
| 				CLANG_WARN_COMMA = YES; | ||||
| 				CLANG_WARN_CONSTANT_CONVERSION = YES; | ||||
| 				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; | ||||
| 				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; | ||||
| 				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
| 				CLANG_WARN_EMPTY_BODY = YES; | ||||
| 				CLANG_WARN_ENUM_CONVERSION = YES; | ||||
| 				CLANG_WARN_INFINITE_RECURSION = YES; | ||||
| 				CLANG_WARN_INT_CONVERSION = YES; | ||||
| 				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
| 				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
| 				CLANG_WARN_UNREACHABLE_CODE = YES; | ||||
| 				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; | ||||
| 				COPY_PHASE_STRIP = NO; | ||||
| 				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; | ||||
| 				ENABLE_STRICT_OBJC_MSGSEND = YES; | ||||
| 				ENABLE_TESTABILITY = YES; | ||||
| 				GCC_C_LANGUAGE_STANDARD = gnu11; | ||||
| 				GCC_DYNAMIC_NO_PIC = NO; | ||||
| 				GCC_NO_COMMON_BLOCKS = YES; | ||||
| 				GCC_OPTIMIZATION_LEVEL = 0; | ||||
| 				GCC_PREPROCESSOR_DEFINITIONS = ( | ||||
| 					"DEBUG=1", | ||||
| 					"$(inherited)", | ||||
| 				); | ||||
| 				GCC_WARN_64_TO_32_BIT_CONVERSION = YES; | ||||
| 				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; | ||||
| 				GCC_WARN_UNDECLARED_SELECTOR = YES; | ||||
| 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||
| 				GCC_WARN_UNUSED_FUNCTION = YES; | ||||
| 				GCC_WARN_UNUSED_VARIABLE = YES; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 14.1; | ||||
| 				MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; | ||||
| 				MTL_FAST_MATH = YES; | ||||
| 				ONLY_ACTIVE_ARCH = YES; | ||||
| 				SDKROOT = iphoneos; | ||||
| 				SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; | ||||
| 				SWIFT_OPTIMIZATION_LEVEL = "-Onone"; | ||||
| 			}; | ||||
| 			name = Debug; | ||||
| 		}; | ||||
| 		7555FFA4242A565B00829871 /* Release */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ALWAYS_SEARCH_USER_PATHS = NO; | ||||
| 				CLANG_ANALYZER_NONNULL = YES; | ||||
| 				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
| 				CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; | ||||
| 				CLANG_CXX_LIBRARY = "libc++"; | ||||
| 				CLANG_ENABLE_MODULES = YES; | ||||
| 				CLANG_ENABLE_OBJC_ARC = YES; | ||||
| 				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
| 				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; | ||||
| 				CLANG_WARN_BOOL_CONVERSION = YES; | ||||
| 				CLANG_WARN_COMMA = YES; | ||||
| 				CLANG_WARN_CONSTANT_CONVERSION = YES; | ||||
| 				CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; | ||||
| 				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; | ||||
| 				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
| 				CLANG_WARN_EMPTY_BODY = YES; | ||||
| 				CLANG_WARN_ENUM_CONVERSION = YES; | ||||
| 				CLANG_WARN_INFINITE_RECURSION = YES; | ||||
| 				CLANG_WARN_INT_CONVERSION = YES; | ||||
| 				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
| 				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
| 				CLANG_WARN_UNREACHABLE_CODE = YES; | ||||
| 				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; | ||||
| 				COPY_PHASE_STRIP = NO; | ||||
| 				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; | ||||
| 				ENABLE_NS_ASSERTIONS = NO; | ||||
| 				ENABLE_STRICT_OBJC_MSGSEND = YES; | ||||
| 				GCC_C_LANGUAGE_STANDARD = gnu11; | ||||
| 				GCC_NO_COMMON_BLOCKS = YES; | ||||
| 				GCC_WARN_64_TO_32_BIT_CONVERSION = YES; | ||||
| 				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; | ||||
| 				GCC_WARN_UNDECLARED_SELECTOR = YES; | ||||
| 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||
| 				GCC_WARN_UNUSED_FUNCTION = YES; | ||||
| 				GCC_WARN_UNUSED_VARIABLE = YES; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 14.1; | ||||
| 				MTL_ENABLE_DEBUG_INFO = NO; | ||||
| 				MTL_FAST_MATH = YES; | ||||
| 				SDKROOT = iphoneos; | ||||
| 				SWIFT_COMPILATION_MODE = wholemodule; | ||||
| 				SWIFT_OPTIMIZATION_LEVEL = "-O"; | ||||
| 				VALIDATE_PRODUCT = YES; | ||||
| 			}; | ||||
| 			name = Release; | ||||
| 		}; | ||||
| 		7555FFA6242A565B00829871 /* Debug */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				DEVELOPMENT_ASSET_PATHS = "\"iosApp/Preview Content\""; | ||||
| 				ENABLE_PREVIEWS = YES; | ||||
| 				FRAMEWORK_SEARCH_PATHS = "$(SRCROOT)/../shared/build/xcode-frameworks/$(CONFIGURATION)/$(SDK_NAME)"; | ||||
| 				INFOPLIST_FILE = iosApp/Info.plist; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 				); | ||||
| 				OTHER_LDFLAGS = ( | ||||
| 					"$(inherited)", | ||||
| 					"-framework", | ||||
| 					shared, | ||||
| 				); | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = orgIdentifier.iosApp; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 			}; | ||||
| 			name = Debug; | ||||
| 		}; | ||||
| 		7555FFA7242A565B00829871 /* Release */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				DEVELOPMENT_ASSET_PATHS = "\"iosApp/Preview Content\""; | ||||
| 				ENABLE_PREVIEWS = YES; | ||||
| 				FRAMEWORK_SEARCH_PATHS = "$(SRCROOT)/../shared/build/xcode-frameworks/$(CONFIGURATION)/$(SDK_NAME)"; | ||||
| 				INFOPLIST_FILE = iosApp/Info.plist; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 				); | ||||
| 				OTHER_LDFLAGS = ( | ||||
| 					"$(inherited)", | ||||
| 					"-framework", | ||||
| 					shared, | ||||
| 				); | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = orgIdentifier.iosApp; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 			}; | ||||
| 			name = Release; | ||||
| 		}; | ||||
| /* End XCBuildConfiguration section */ | ||||
|  | ||||
|     /* Begin XCConfigurationList section */ | ||||
|     		7555FF76242A565900829871 /* Build configuration list for PBXProject "iosApp" */ = { | ||||
|     			isa = XCConfigurationList; | ||||
|     			buildConfigurations = ( | ||||
|     				7555FFA3242A565B00829871 /* Debug */, | ||||
|     				7555FFA4242A565B00829871 /* Release */, | ||||
|     			); | ||||
|     			defaultConfigurationIsVisible = 0; | ||||
|     			defaultConfigurationName = Release; | ||||
|     		}; | ||||
|     		7555FFA5242A565B00829871 /* Build configuration list for PBXNativeTarget "iosApp" */ = { | ||||
|     			isa = XCConfigurationList; | ||||
|     			buildConfigurations = ( | ||||
|     				7555FFA6242A565B00829871 /* Debug */, | ||||
|     				7555FFA7242A565B00829871 /* Release */, | ||||
|     			); | ||||
|     			defaultConfigurationIsVisible = 0; | ||||
|     			defaultConfigurationName = Release; | ||||
|     		}; | ||||
|     /* End XCConfigurationList section */ | ||||
|     	}; | ||||
|     	rootObject = 7555FF73242A565900829871 /* Project object */; | ||||
|     } | ||||
| /* Begin XCConfigurationList section */ | ||||
| 		7555FF76242A565900829871 /* Build configuration list for PBXProject "iosApp" */ = { | ||||
| 			isa = XCConfigurationList; | ||||
| 			buildConfigurations = ( | ||||
| 				7555FFA3242A565B00829871 /* Debug */, | ||||
| 				7555FFA4242A565B00829871 /* Release */, | ||||
| 			); | ||||
| 			defaultConfigurationIsVisible = 0; | ||||
| 			defaultConfigurationName = Release; | ||||
| 		}; | ||||
| 		7555FFA5242A565B00829871 /* Build configuration list for PBXNativeTarget "iosApp" */ = { | ||||
| 			isa = XCConfigurationList; | ||||
| 			buildConfigurations = ( | ||||
| 				7555FFA6242A565B00829871 /* Debug */, | ||||
| 				7555FFA7242A565B00829871 /* Release */, | ||||
| 			); | ||||
| 			defaultConfigurationIsVisible = 0; | ||||
| 			defaultConfigurationName = Release; | ||||
| 		}; | ||||
| /* End XCConfigurationList section */ | ||||
| 	}; | ||||
| 	rootObject = 7555FF73242A565900829871 /* Project object */; | ||||
| } | ||||
|   | ||||
| @@ -2,12 +2,9 @@ import SwiftUI | ||||
| import shared | ||||
|  | ||||
| struct ContentView: View { | ||||
|     let greet = Greeting().greeting() | ||||
|  | ||||
|     let toto = SelfossApi().getItems() | ||||
|  | ||||
| 	var body: some View { | ||||
| 		Text(greet) | ||||
| 		Text("ototot") | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -8,6 +8,14 @@ pluginManagement { | ||||
|     } | ||||
| } | ||||
|  | ||||
| dependencyResolutionManagement { | ||||
|     repositories { | ||||
|         google() | ||||
|         mavenCentral() | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| buildCache { | ||||
|     remote<HttpBuildCache> { | ||||
|         url = uri("http://18.0.0.7:3071/cache/") | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| object SqlDelight { | ||||
|     const val runtime = "com.squareup.sqldelight:runtime:1.5.3" | ||||
|     const val android = "com.squareup.sqldelight:android-driver:1.5.3" | ||||
|     const val native = "com.squareup.sqldelight:native-driver:1.5.3" | ||||
|     const val runtime = "com.squareup.sqldelight:runtime:1.5.4" | ||||
|     const val android = "com.squareup.sqldelight:android-driver:1.5.4" | ||||
|     const val native = "com.squareup.sqldelight:native-driver:1.5.4" | ||||
|  | ||||
| } | ||||
|  | ||||
| @@ -18,7 +18,7 @@ kotlin { | ||||
|     listOf( | ||||
|         iosX64(), | ||||
|         iosArm64(), | ||||
|         //iosSimulatorArm64() sure all ios dependencies support this target | ||||
|         // iosSimulatorArm64() | ||||
|     ).forEach { | ||||
|         it.binaries.framework { | ||||
|             baseName = "shared" | ||||
| @@ -40,14 +40,11 @@ kotlin { | ||||
|                 implementation("org.kodein.di:kodein-di:7.12.0") | ||||
|  | ||||
|                 //Settings | ||||
|                 implementation("com.russhwolf:multiplatform-settings-no-arg:0.9") | ||||
|                 implementation("com.russhwolf:multiplatform-settings-no-arg:1.0.0-RC") | ||||
|  | ||||
|                 //Logging | ||||
|                 implementation("io.github.aakira:napier:2.6.1") | ||||
|  | ||||
|                 // Network information | ||||
|                 implementation("com.github.ln-12:multiplatform-connectivity-status:1.3.0") | ||||
|  | ||||
|                 // Sql | ||||
|                 implementation(SqlDelight.runtime) | ||||
|             } | ||||
| @@ -56,13 +53,12 @@ kotlin { | ||||
|             dependencies { | ||||
|                 implementation(kotlin("test-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 { | ||||
|             dependencies { | ||||
|                 implementation("io.ktor:ktor-client-okhttp:2.1.1") | ||||
|                 implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.4.0") | ||||
|  | ||||
|                 // Sql | ||||
|                 implementation(SqlDelight.android) | ||||
| @@ -76,39 +72,36 @@ kotlin { | ||||
|         } | ||||
|         val iosX64Main by getting | ||||
|         val iosArm64Main by getting | ||||
|         //val iosSimulatorArm64Main by getting | ||||
|         // val iosSimulatorArm64Main by getting | ||||
|         val iosMain by creating { | ||||
|             dependsOn(commonMain) | ||||
|             iosX64Main.dependsOn(this) | ||||
|             iosArm64Main.dependsOn(this) | ||||
|             //iosSimulatorArm64Main.dependsOn(this) | ||||
|             // iosSimulatorArm64Main.dependsOn(this) | ||||
|  | ||||
|             // Sql | ||||
|             dependencies { | ||||
|                 implementation(SqlDelight.native) | ||||
|                 implementation("io.ktor:ktor-client-ios:2.1.1") | ||||
|             } | ||||
|         } | ||||
|         val iosX64Test by getting | ||||
|         val iosArm64Test by getting | ||||
|         //val iosSimulatorArm64Test by getting | ||||
|         // val iosSimulatorArm64Test by getting | ||||
|         val iosTest by creating { | ||||
|             dependsOn(commonTest) | ||||
|             iosX64Test.dependsOn(this) | ||||
|             iosArm64Test.dependsOn(this) | ||||
|             dependencies { | ||||
|                 implementation("io.ktor:ktor-client-ios:2.1.1") | ||||
|             } | ||||
|             //iosSimulatorArm64Test.dependsOn(this) | ||||
|             // iosSimulatorArm64Test.dependsOn(this) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| android { | ||||
|     compileSdk = 31 | ||||
|     compileSdk = 32 | ||||
|     sourceSets["main"].manifest.srcFile("src/androidMain/AndroidManifest.xml") | ||||
|     defaultConfig { | ||||
|         minSdk = 21 | ||||
|         targetSdk = 31 | ||||
|         targetSdk = 32 | ||||
|     } | ||||
|     compileOptions { | ||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
| @@ -117,10 +110,11 @@ android { | ||||
|     namespace = "bou.amine.apps.readerforselfossv2" | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| sqldelight { | ||||
|     database("ReaderForSelfossDB") { | ||||
|         packageName = "bou.amine.apps.readerforselfossv2.dao" | ||||
|         sourceFolders = listOf("sqldelight") | ||||
|     } | ||||
| } | ||||
|  | ||||
| } | ||||
| @@ -1,36 +1,29 @@ | ||||
| package bou.amine.apps.readerforselfossv2.utils | ||||
|  | ||||
| import android.text.format.DateUtils | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import java.time.Instant | ||||
| import java.time.LocalDateTime | ||||
| import java.time.OffsetDateTime | ||||
| import java.time.ZoneOffset | ||||
| import java.time.format.DateTimeFormatter | ||||
| import kotlinx.datetime.* | ||||
|  | ||||
| actual class DateUtils actual constructor(actual val appSettingsService: AppSettingsService) { | ||||
|  | ||||
|     actual fun parseDate(dateString: String): Long { | ||||
| actual class DateUtils { | ||||
|     actual companion object { | ||||
|         actual fun parseDate(dateString: String): Long { | ||||
|             return try { | ||||
|                 Instant.parse(dateString).toEpochMilliseconds() | ||||
|             } catch (e: Exception) { | ||||
|                 LocalDateTime.parse(dateString.replace(" ", "T")).toInstant(TimeZone.currentSystemDefault()).toEpochMilliseconds() | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val FORMATTERV1 = "yyyy-MM-dd HH:mm:ss" | ||||
|         actual fun parseRelativeDate(dateString: String): String { | ||||
|  | ||||
|         return if (appSettingsService.getApiVersion() >= 4) { | ||||
|             OffsetDateTime.parse(dateString).toInstant().toEpochMilli() | ||||
|         } else { | ||||
|             LocalDateTime.parse(dateString, DateTimeFormatter.ofPattern(FORMATTERV1)).toInstant( | ||||
|                 ZoneOffset.UTC).toEpochMilli() | ||||
|             val date = parseDate(dateString) | ||||
|  | ||||
|             return " " + DateUtils.getRelativeTimeSpanString( | ||||
|                 date, | ||||
|                 Clock.System.now().toEpochMilliseconds(), | ||||
|                 DateUtils.MINUTE_IN_MILLIS, | ||||
|                 DateUtils.FORMAT_ABBREV_RELATIVE | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     actual fun parseRelativeDate(dateString: String): String { | ||||
|  | ||||
|         val date = parseDate(dateString) | ||||
|  | ||||
|         return " " + DateUtils.getRelativeTimeSpanString( | ||||
|             date, | ||||
|             Instant.now().toEpochMilli(), | ||||
|             DateUtils.MINUTE_IN_MILLIS, | ||||
|             DateUtils.FORMAT_ABBREV_RELATIVE | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| @@ -1,5 +1,6 @@ | ||||
| package bou.amine.apps.readerforselfossv2.DI | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.rest.MercuryApi | ||||
| import bou.amine.apps.readerforselfossv2.rest.SelfossApi | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import org.kodein.di.DI | ||||
| @@ -10,4 +11,5 @@ import org.kodein.di.singleton | ||||
| val networkModule by DI.Module { | ||||
|     bind<AppSettingsService>() with singleton { AppSettingsService() } | ||||
|     bind<SelfossApi>() with singleton { SelfossApi(instance()) } | ||||
|     bind<MercuryApi>() with singleton { MercuryApi() } | ||||
| } | ||||
| @@ -0,0 +1,157 @@ | ||||
| package bou.amine.apps.readerforselfossv2.model | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.utils.DateUtils | ||||
| import bou.amine.apps.readerforselfossv2.utils.getHtmlDecoded | ||||
| import kotlinx.serialization.KSerializer | ||||
| import kotlinx.serialization.Serializable | ||||
| import kotlinx.serialization.descriptors.PrimitiveKind | ||||
| import kotlinx.serialization.descriptors.PrimitiveSerialDescriptor | ||||
| import kotlinx.serialization.descriptors.SerialDescriptor | ||||
| import kotlinx.serialization.encoding.Decoder | ||||
| import kotlinx.serialization.encoding.Encoder | ||||
| import kotlinx.serialization.json.* | ||||
|  | ||||
| class MercuryModel { | ||||
|  | ||||
|     @Serializable | ||||
|     class ParsedContent( | ||||
|         val title: String, | ||||
|         val content: String?, | ||||
|         val lead_image_url: String?, | ||||
|         val url: String | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     data class Tag( | ||||
|         val tag: String, | ||||
|         val color: String, | ||||
|         val unread: Int | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     class Stats( | ||||
|         val total: Int, | ||||
|         val unread: Int, | ||||
|         val starred: Int | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     data class Spout( | ||||
|         val name: String, | ||||
|         val description: String | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     data class ApiVersion( | ||||
|         val version: String?, | ||||
|         val apiversion: String? | ||||
|     ) { | ||||
|         fun getApiMajorVersion() : Int { | ||||
|             var versionNumber = 0 | ||||
|             if (apiversion != null) { | ||||
|                 versionNumber = apiversion.substringBefore(".").toInt() | ||||
|             } | ||||
|             return versionNumber | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Serializable | ||||
|     data class Source( | ||||
|         val id: Int, | ||||
|         val title: String, | ||||
|         @Serializable(with = TagsListSerializer::class) | ||||
|         val tags: List<String>, | ||||
|         val spout: String, | ||||
|         val error: String, | ||||
|         val icon: String? | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     data class Item( | ||||
|         val id: Int, | ||||
|         val datetime: String, | ||||
|         val title: String, | ||||
|         val content: String, | ||||
|         @Serializable(with = BooleanSerializer::class) | ||||
|         var unread: Boolean, | ||||
|         @Serializable(with = BooleanSerializer::class) | ||||
|         var starred: Boolean, | ||||
|         val thumbnail: String?, | ||||
|         val icon: String?, | ||||
|         val link: String, | ||||
|         val sourcetitle: String, | ||||
|         @Serializable(with = TagsListSerializer::class) | ||||
|         val tags: List<String> | ||||
|     ) { | ||||
|         // TODO: maybe find a better way to handle these kind of urls | ||||
|         fun getLinkDecoded(): String { | ||||
|             var stringUrl: String | ||||
|             stringUrl = | ||||
|                 if (link.startsWith("http://news.google.com/news/") || link.startsWith("https://news.google.com/news/")) { | ||||
|                     if (link.contains("&url=")) { | ||||
|                         link.substringAfter("&url=") | ||||
|                     } else { | ||||
|                         this.link.replace("&", "&") | ||||
|                     } | ||||
|                 } else { | ||||
|                     this.link.replace("&", "&") | ||||
|                 } | ||||
|  | ||||
|             // handle :443 => https | ||||
|             if (stringUrl.contains(":443")) { | ||||
|                 stringUrl = stringUrl.replace(":443", "").replace("http://", "https://") | ||||
|             } | ||||
|  | ||||
|             // handle url not starting with http | ||||
|             if (stringUrl.startsWith("//")) { | ||||
|                 stringUrl = "http:$stringUrl" | ||||
|             } | ||||
|  | ||||
|             return stringUrl | ||||
|         } | ||||
|  | ||||
|         fun sourceAndDateText(): String = | ||||
|             this.sourcetitle.getHtmlDecoded() + DateUtils.parseRelativeDate(this.datetime) | ||||
|  | ||||
|         fun toggleStar(): Item { | ||||
|             this.starred = !this.starred | ||||
|             return this | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // TODO: this seems to be super slow. | ||||
|     object TagsListSerializer : KSerializer<List<String>> { | ||||
|         override fun deserialize(decoder: Decoder): List<String> { | ||||
|             return when(val json = ((decoder as JsonDecoder).decodeJsonElement())) { | ||||
|                 is JsonArray -> json.toList().map { it.toString() } | ||||
|                 else -> json.toString().split(",") | ||||
|             } | ||||
|  | ||||
|         } | ||||
|  | ||||
|         override val descriptor: SerialDescriptor | ||||
|             get() = PrimitiveSerialDescriptor("tags", PrimitiveKind.STRING) | ||||
|  | ||||
|         override fun serialize(encoder: Encoder, value: List<String>) { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     object BooleanSerializer : KSerializer<Boolean> { | ||||
|         override fun deserialize(decoder: Decoder): Boolean { | ||||
|             val json = ((decoder as JsonDecoder).decodeJsonElement()).jsonPrimitive | ||||
|             return if (json.booleanOrNull != null) { | ||||
|                 json.boolean | ||||
|             } else { | ||||
|                 json.int == 1 | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         override val descriptor: SerialDescriptor | ||||
|             get() = PrimitiveSerialDescriptor("b", PrimitiveKind.BOOLEAN) | ||||
|  | ||||
|         override fun serialize(encoder: Encoder, value: Boolean) { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,48 @@ | ||||
| package bou.amine.apps.readerforselfossv2.model | ||||
|  | ||||
| import io.ktor.client.call.* | ||||
| import io.ktor.client.statement.* | ||||
| import io.ktor.http.* | ||||
| import kotlinx.serialization.Serializable | ||||
|  | ||||
| @Serializable | ||||
| class SuccessResponse(val success: Boolean) { | ||||
|     val isSuccess: Boolean | ||||
|         get() = success | ||||
| } | ||||
|  | ||||
| class StatusAndData<T>(val success: Boolean, val data: T? = null) { | ||||
|     companion object { | ||||
|         fun <T> succes(d: T): StatusAndData<T> { | ||||
|             return StatusAndData(true, d) | ||||
|         } | ||||
|  | ||||
|         fun <T> error(): StatusAndData<T> { | ||||
|             return StatusAndData(false) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| suspend fun responseOrSuccessIf404(r: HttpResponse): SuccessResponse { | ||||
|     return if (r.status === HttpStatusCode.NotFound) { | ||||
|         SuccessResponse(true) | ||||
|     } else { | ||||
|         maybeResponse(r) | ||||
|     } | ||||
| } | ||||
|  | ||||
| suspend fun maybeResponse(r: HttpResponse): SuccessResponse { | ||||
|     return if (r.status.isSuccess()) { | ||||
|         r.body() | ||||
|     } else { | ||||
|         SuccessResponse(false) | ||||
|     } | ||||
| } | ||||
|  | ||||
| suspend inline fun <reified T> bodyOrFailure(r: HttpResponse): StatusAndData<T> { | ||||
|     return if (r.status.isSuccess()) { | ||||
|         StatusAndData.succes(r.body()) | ||||
|     } else { | ||||
|         StatusAndData.error() | ||||
|     } | ||||
| } | ||||
| @@ -20,12 +20,6 @@ class SelfossModel { | ||||
|         val unread: Int | ||||
|     ) | ||||
|  | ||||
|     @Serializable | ||||
|     class SuccessResponse(val success: Boolean) { | ||||
|         val isSuccess: Boolean | ||||
|             get() = success | ||||
|     } | ||||
|  | ||||
|     @Serializable | ||||
|     class Stats( | ||||
|         val total: Int, | ||||
| @@ -108,8 +102,8 @@ class SelfossModel { | ||||
|             return stringUrl | ||||
|         } | ||||
|  | ||||
|         fun sourceAndDateText(dateUtils: DateUtils): String = | ||||
|             this.sourcetitle.getHtmlDecoded() + dateUtils.parseRelativeDate(this.datetime) | ||||
|         fun sourceAndDateText(): String = | ||||
|             this.sourcetitle.getHtmlDecoded() + DateUtils.parseRelativeDate(this.datetime) | ||||
|  | ||||
|         fun toggleStar(): Item { | ||||
|             this.starred = !this.starred | ||||
| @@ -152,16 +146,4 @@ class SelfossModel { | ||||
|             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,24 +3,23 @@ package bou.amine.apps.readerforselfossv2.repository | ||||
| import bou.amine.apps.readerforselfossv2.dao.* | ||||
| import bou.amine.apps.readerforselfossv2.model.NetworkUnavailableException | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.model.StatusAndData | ||||
| import bou.amine.apps.readerforselfossv2.rest.SelfossApi | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import bou.amine.apps.readerforselfossv2.utils.* | ||||
| import com.github.ln_12.library.ConnectivityStatus | ||||
| import io.github.aakira.napier.Napier | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.flow.MutableStateFlow | ||||
| import kotlinx.coroutines.flow.asStateFlow | ||||
| import kotlinx.coroutines.launch | ||||
| 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>() | ||||
|     val isConnectionAvailable = connectivityStatus.isNetworkConnected | ||||
|     var connectionMonitored = false | ||||
|  | ||||
|     var baseUrl = appSettingsService.getBaseUrl() | ||||
|     lateinit var dateUtils: DateUtils | ||||
|  | ||||
|     var displayedItems = ItemType.UNREAD | ||||
|  | ||||
| @@ -30,28 +29,19 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|  | ||||
|     var offlineOverride = false | ||||
|  | ||||
|     var badgeUnread = 0 | ||||
|     set(value) {field = if (value < 0) { 0 } else { value } } | ||||
|     var badgeAll = 0 | ||||
|     set(value) {field = if (value < 0) { 0 } else { value } } | ||||
|     var badgeStarred = 0 | ||||
|     set(value) {field = if (value < 0) { 0 } else { value } } | ||||
|     private val _badgeUnread = MutableStateFlow(0) | ||||
|     val badgeUnread = _badgeUnread.asStateFlow() | ||||
|     private val _badgeAll = MutableStateFlow(0) | ||||
|     val badgeAll = _badgeAll.asStateFlow() | ||||
|     private val _badgeStarred = MutableStateFlow(0) | ||||
|     val badgeStarred = _badgeStarred.asStateFlow() | ||||
|  | ||||
|     private var fetchedSources = false | ||||
|     private var fetchedTags = false | ||||
|  | ||||
|     init { | ||||
|         // TODO: Dispatchers.IO not available in KMM, an alternative solution should be found | ||||
|         runBlocking { | ||||
|             updateApiVersion() | ||||
|             dateUtils = DateUtils(appSettingsService) | ||||
|             reloadBadges() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     suspend fun getNewerItems(): ArrayList<SelfossModel.Item> { | ||||
|         // 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 | ||||
|         if (isNetworkAvailable()) { | ||||
|             fetchedItems = api.getItems( | ||||
| @@ -65,12 +55,19 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|         } else { | ||||
|             if (appSettingsService.isItemCachingEnabled()) { | ||||
|                 fromDB = true | ||||
|                 fetchedItems = SelfossModel.StatusAndData.succes( | ||||
|                     getDBItems().filter { | ||||
|                         displayedItems == ItemType.ALL || | ||||
|                                 (it.unread && displayedItems == ItemType.UNREAD) || | ||||
|                                 (it.starred && displayedItems == ItemType.STARRED) | ||||
|                     }.map { it.toView() } | ||||
|                 var dbItems = getDBItems().filter { | ||||
|                     displayedItems == ItemType.ALL || | ||||
|                             (it.unread && displayedItems == ItemType.UNREAD) || | ||||
|                             (it.starred && displayedItems == ItemType.STARRED) | ||||
|                 } | ||||
|                 if (tagFilter != null) { | ||||
|                     dbItems = dbItems.filter { it.tags.split(',').contains(tagFilter!!.tag) } | ||||
|                 } | ||||
|                 if (sourceFilter != null) { | ||||
|                     dbItems = dbItems.filter { it.sourcetitle == sourceFilter!!.title } | ||||
|                 } | ||||
|                 fetchedItems = StatusAndData.succes( | ||||
|                     dbItems.map { it.toView() } | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
| @@ -78,14 +75,14 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|         if (fetchedItems.success && fetchedItems.data != null) { | ||||
|             items = ArrayList(fetchedItems.data!!) | ||||
|             if (fromDB) { | ||||
|                 items.sortByDescending { dateUtils.parseDate(it.datetime) } | ||||
|                 items.sortByDescending { DateUtils.parseDate(it.datetime) } | ||||
|             } | ||||
|         } | ||||
|         return items | ||||
|     } | ||||
|  | ||||
|     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()) { | ||||
|             val offset = items.size | ||||
|             fetchedItems = api.getItems( | ||||
| @@ -130,17 +127,17 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|         if (isNetworkAvailable()) { | ||||
|             val response = api.stats() | ||||
|             if (response.success && response.data != null) { | ||||
|                 badgeUnread = response.data.unread | ||||
|                 badgeAll = response.data.total | ||||
|                 badgeStarred = response.data.starred | ||||
|                 _badgeUnread.value = response.data.unread | ||||
|                 _badgeAll.value = response.data.total | ||||
|                 _badgeStarred.value = response.data.starred | ||||
|                 success = true | ||||
|             } | ||||
|         } else if (appSettingsService.isItemCachingEnabled()) { | ||||
|             // TODO: do this differently, because it's not efficient | ||||
|             val dbItems = getDBItems() | ||||
|             badgeUnread = dbItems.filter { item -> item.unread }.size | ||||
|             badgeStarred = dbItems.filter { item -> item.starred }.size | ||||
|             badgeAll = dbItems.size | ||||
|             _badgeUnread.value = dbItems.filter { item -> item.unread }.size | ||||
|             _badgeStarred.value = dbItems.filter { item -> item.starred }.size | ||||
|             _badgeAll.value = dbItems.size | ||||
|             success = true | ||||
|         } | ||||
|         return success | ||||
| @@ -288,7 +285,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun markAsReadLocally(item: SelfossModel.Item) { | ||||
|         if (item.unread) { | ||||
|             item.unread = false | ||||
|             badgeUnread -= 1 | ||||
|             _badgeUnread.value -= 1 | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -299,7 +296,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun unmarkAsReadLocally(item: SelfossModel.Item) { | ||||
|         if (!item.unread) { | ||||
|             item.unread = true | ||||
|             badgeUnread += 1 | ||||
|             _badgeUnread.value += 1 | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -310,7 +307,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun starrLocally(item: SelfossModel.Item) { | ||||
|         if (!item.starred) { | ||||
|             item.starred = true | ||||
|             badgeStarred += 1 | ||||
|             _badgeStarred.value += 1 | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -321,7 +318,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun unstarrLocally(item: SelfossModel.Item) { | ||||
|         if (item.starred) { | ||||
|             item.starred = false | ||||
|             badgeStarred -= 1 | ||||
|             _badgeStarred.value -= 1 | ||||
|         } | ||||
|  | ||||
|         CoroutineScope(Dispatchers.Main).launch { | ||||
| @@ -343,8 +340,7 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|                 url, | ||||
|                 spout, | ||||
|                 tags, | ||||
|                 filter, | ||||
|                 appSettingsService.getApiVersion() | ||||
|                 filter | ||||
|             ).isSuccess == true | ||||
|         } | ||||
|  | ||||
| @@ -375,23 +371,37 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|             try { | ||||
|                 val response = api.login() | ||||
|                 result = response.isSuccess == true | ||||
|                 if (result) { | ||||
|                     updateApiVersion() | ||||
|                 } | ||||
|             } catch (cause: Throwable) { | ||||
|                 Napier.e(cause.stackTraceToString(), tag = "RepositoryImpl.updateRemote") | ||||
|                 Napier.e(cause.stackTraceToString(), tag = "RepositoryImpl.login") | ||||
|             } | ||||
|         } | ||||
|         return result | ||||
|     } | ||||
|  | ||||
|     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") | ||||
|             } finally { | ||||
|                 appSettingsService.clearAll() | ||||
|             } | ||||
|         } else { | ||||
|             appSettingsService.clearAll() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun refreshLoginInformation(url: String, login: String, password: String) { | ||||
|         appSettingsService.refreshLoginInformation(url, login, password) | ||||
|         baseUrl = url | ||||
|         api.refreshLoginInformation() | ||||
|     } | ||||
|  | ||||
|     private suspend fun updateApiVersion() { | ||||
|     suspend fun updateApiVersion() { | ||||
|         val apiMajorVersion = appSettingsService.getApiVersion() | ||||
|  | ||||
|         if (isNetworkAvailable()) { | ||||
| @@ -410,11 +420,9 @@ class Repository(private val api: SelfossApi, private val appSettingsService: Ap | ||||
|     private fun deleteDBAction(action: ACTION) = | ||||
|         db.actionsQueries.deleteAction(action.id) | ||||
|  | ||||
|     // TODO: This function should be private | ||||
|     fun getDBTags(): List<TAG> = db.tagsQueries.tags().executeAsList() | ||||
|     private fun getDBTags(): List<TAG> = db.tagsQueries.tags().executeAsList() | ||||
|  | ||||
|     // TODO: This function should be private | ||||
|     fun getDBSources(): List<SOURCE> = db.sourcesQueries.sources().executeAsList() | ||||
|     private fun getDBSources(): List<SOURCE> = db.sourcesQueries.sources().executeAsList() | ||||
|  | ||||
|     private fun resetDBTagsWithData(tagEntities: List<SelfossModel.Tag>) { | ||||
|         db.tagsQueries.deleteAllTags() | ||||
|   | ||||
| @@ -0,0 +1,43 @@ | ||||
| package bou.amine.apps.readerforselfossv2.rest | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.model.* | ||||
| import io.github.aakira.napier.Napier | ||||
| import io.ktor.client.* | ||||
| import io.ktor.client.plugins.cache.* | ||||
| import io.ktor.client.plugins.contentnegotiation.* | ||||
| import io.ktor.client.plugins.logging.* | ||||
| import io.ktor.client.request.* | ||||
| import io.ktor.serialization.kotlinx.json.* | ||||
| import kotlinx.serialization.json.Json | ||||
|  | ||||
| class MercuryApi() { | ||||
|  | ||||
|     var client = createHttpClient() | ||||
|  | ||||
|     private fun createHttpClient(): HttpClient { | ||||
|         return HttpClient { | ||||
|             install(ContentNegotiation) { | ||||
|                 install(HttpCache) | ||||
|                 json(Json { | ||||
|                     prettyPrint = true | ||||
|                     isLenient = true | ||||
|                     ignoreUnknownKeys = true | ||||
|                 }) | ||||
|             } | ||||
|             install(Logging) { | ||||
|                 logger = object : Logger { | ||||
|                     override fun log(message: String) { | ||||
|                         Napier.d(message, tag = "LogMercuryCalls") | ||||
|                     } | ||||
|                 } | ||||
|                 level = LogLevel.INFO | ||||
|             } | ||||
|             expectSuccess = false | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     suspend fun query(url: String): StatusAndData<MercuryModel.ParsedContent> = | ||||
|         bodyOrFailure(client.get("https://amine-louveau.fr/parser.php") { | ||||
|             parameter("link", url) | ||||
|         }) | ||||
| } | ||||
| @@ -1,12 +1,13 @@ | ||||
| package bou.amine.apps.readerforselfossv2.rest | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.model.* | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| import io.github.aakira.napier.Napier | ||||
| import io.ktor.client.* | ||||
| import io.ktor.client.call.* | ||||
| import io.ktor.client.plugins.* | ||||
| import io.ktor.client.plugins.cache.* | ||||
| import io.ktor.client.plugins.contentnegotiation.* | ||||
| import io.ktor.client.plugins.cookies.* | ||||
| import io.ktor.client.plugins.logging.* | ||||
| import io.ktor.client.request.* | ||||
| import io.ktor.client.request.forms.* | ||||
| @@ -20,7 +21,7 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|     var client = createHttpClient() | ||||
|  | ||||
|     private fun createHttpClient(): HttpClient { | ||||
|         return HttpClient { | ||||
|         val client = HttpClient { | ||||
|             install(ContentNegotiation) { | ||||
|                 install(HttpCache) | ||||
|                 json(Json { | ||||
| @@ -32,7 +33,7 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|             install(Logging) { | ||||
|                 logger = object : Logger { | ||||
|                     override fun log(message: String) { | ||||
|                         appSettingsService.logApiCalls(message) | ||||
|                         Napier.d(message, tag = "LogApiCalls") | ||||
|                     } | ||||
|                 } | ||||
|                 level = LogLevel.INFO | ||||
| @@ -40,22 +41,26 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|             install(HttpTimeout) { | ||||
|                 requestTimeoutMillis = appSettingsService.getApiTimeout() | ||||
|             } | ||||
|             /* TODO: Auth as basic | ||||
|             if (apiDetailsService.getUserName().isNotEmpty() && apiDetailsService.getPassword().isNotEmpty()) { | ||||
|  | ||||
|                 install(Auth) { | ||||
|                     basic { | ||||
|                         credentials { | ||||
|                             BasicAuthCredentials(username = apiDetailsService.getUserName(), password = apiDetailsService.getPassword()) | ||||
|                         } | ||||
|                         sendWithoutRequest { | ||||
|                             true | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             }*/ | ||||
|             install(HttpCookies) | ||||
|             expectSuccess = false | ||||
|         } | ||||
|  | ||||
|         client.plugin(HttpSend).intercept { request -> | ||||
|             val originalCall = execute(request) | ||||
|             if (originalCall.response.status == HttpStatusCode.Forbidden && shouldHavePostLogin() && hasLoginInfo()) { | ||||
|                 Napier.i("Forbidden action, will try to login and retry", tag = "HttpSend") | ||||
|  | ||||
|                 if (login().isSuccess) { | ||||
|                     Napier.i("Logged in worked", tag = "HttpSend") | ||||
|                     execute(request) | ||||
|                 } | ||||
|                 originalCall | ||||
|             } else { | ||||
|                 originalCall | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return client | ||||
|     } | ||||
|  | ||||
|     fun url(path: String) = | ||||
| @@ -66,11 +71,38 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         client = createHttpClient() | ||||
|     } | ||||
|  | ||||
|     suspend fun login(): SelfossModel.SuccessResponse = | ||||
|         maybeResponse(client.get(url("/login")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|         }) | ||||
|     // Api version was introduces after the POST login, so when there is a version, it should be available | ||||
|     private fun shouldHavePostLogin() = appSettingsService.getApiVersion() != -1 | ||||
|     private fun hasLoginInfo() = appSettingsService.getUserName() != null && appSettingsService.getPassword() != null | ||||
|  | ||||
|     suspend fun login(): SuccessResponse = | ||||
|         if (shouldHavePostLogin()) { | ||||
|             postLogin() | ||||
|         } else { | ||||
|             getLogin() | ||||
|         } | ||||
|  | ||||
|     private suspend fun getLogin() = maybeResponse(client.get(url("/login")) { | ||||
|         parameter("username", appSettingsService.getUserName()) | ||||
|         parameter("password", appSettingsService.getPassword()) | ||||
|     }) | ||||
|  | ||||
|     private suspend fun postLogin() = maybeResponse(client.post(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.get(url("/logout"))) | ||||
|  | ||||
|     private suspend fun doLogout() = maybeResponse(client.delete(url("/api/session/current"))) | ||||
|  | ||||
|     suspend fun getItems( | ||||
|         type: String, | ||||
| @@ -80,82 +112,104 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         search: String?, | ||||
|         updatedSince: String?, | ||||
|         items: Int? = null | ||||
|     ): SelfossModel.StatusAndData<List<SelfossModel.Item>> = | ||||
|     ): StatusAndData<List<SelfossModel.Item>> = | ||||
|         bodyOrFailure(client.get(url("/items")) { | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|                 parameter("type", type) | ||||
|                 parameter("tag", tag) | ||||
|                 parameter("source", source) | ||||
|                 parameter("search", search) | ||||
|                 parameter("updatedsince", updatedSince) | ||||
|                 parameter("items", items ?: appSettingsService.getItemsNumber()) | ||||
|                 parameter("offset", offset) | ||||
|             }) | ||||
|             } | ||||
|             parameter("type", type) | ||||
|             parameter("tag", tag) | ||||
|             parameter("source", source) | ||||
|             parameter("search", search) | ||||
|             parameter("updatedsince", updatedSince) | ||||
|             parameter("items", items ?: appSettingsService.getItemsNumber()) | ||||
|             parameter("offset", offset) | ||||
|         }) | ||||
|  | ||||
|     suspend fun stats(): SelfossModel.StatusAndData<SelfossModel.Stats> = | ||||
|     suspend fun stats(): StatusAndData<SelfossModel.Stats> = | ||||
|         bodyOrFailure(client.get(url("/stats")) { | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }) | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|     suspend fun tags(): SelfossModel.StatusAndData<List<SelfossModel.Tag>> = | ||||
|     suspend fun tags(): StatusAndData<List<SelfossModel.Tag>> = | ||||
|         bodyOrFailure(client.get(url("/tags")) { | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }) | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|     suspend fun update(): SelfossModel.StatusAndData<String> = | ||||
|     suspend fun update(): StatusAndData<String> = | ||||
|         bodyOrFailure(client.get(url("/update")) { | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }) | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|     suspend fun spouts(): SelfossModel.StatusAndData<Map<String, SelfossModel.Spout>> = | ||||
|     suspend fun spouts(): StatusAndData<Map<String, SelfossModel.Spout>> = | ||||
|         bodyOrFailure(client.get(url("/sources/spouts")) { | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }) | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|     suspend fun sources(): SelfossModel.StatusAndData<ArrayList<SelfossModel.Source>> = | ||||
|     suspend fun sources(): StatusAndData<ArrayList<SelfossModel.Source>> = | ||||
|         bodyOrFailure(client.get(url("/sources/list")) { | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             }) | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|     suspend fun version(): SelfossModel.StatusAndData<SelfossModel.ApiVersion> = | ||||
|     suspend fun version(): StatusAndData<SelfossModel.ApiVersion> = | ||||
|         bodyOrFailure(client.get(url("/api/about"))) | ||||
|  | ||||
|     suspend fun markAsRead(id: String): SelfossModel.SuccessResponse = | ||||
|     suspend fun markAsRead(id: String): SuccessResponse = | ||||
|         maybeResponse(client.post(url("/mark/$id")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|     suspend fun unmarkAsRead(id: String): SelfossModel.SuccessResponse = | ||||
|     suspend fun unmarkAsRead(id: String): SuccessResponse = | ||||
|         maybeResponse(client.post(url("/unmark/$id")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|     suspend fun starr(id: String): SelfossModel.SuccessResponse = | ||||
|     suspend fun starr(id: String): SuccessResponse = | ||||
|         maybeResponse(client.post(url("/starr/$id")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|     suspend fun unstarr(id: String): SelfossModel.SuccessResponse = | ||||
|     suspend fun unstarr(id: String): SuccessResponse = | ||||
|         maybeResponse(client.post(url("/unstarr/$id")) { | ||||
|             parameter("username", appSettingsService.getUserName()) | ||||
|             parameter("password", appSettingsService.getPassword()) | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 parameter("password", appSettingsService.getPassword()) | ||||
|             } | ||||
|         }) | ||||
|  | ||||
|     suspend fun markAllAsRead(ids: List<String>): SelfossModel.SuccessResponse = | ||||
|     suspend fun markAllAsRead(ids: List<String>): SuccessResponse = | ||||
|         maybeResponse(client.submitForm( | ||||
|             url = url("/mark"), | ||||
|             formParameters = Parameters.build { | ||||
|                 append("username", appSettingsService.getUserName()) | ||||
|                 append("password", appSettingsService.getPassword()) | ||||
|                 if (!shouldHavePostLogin()) { | ||||
|                     append("username", appSettingsService.getUserName()) | ||||
|                     append("password", appSettingsService.getPassword()) | ||||
|                 } | ||||
|                 ids.map { append("ids[]", it) } | ||||
|             } | ||||
|         )) | ||||
| @@ -165,18 +219,17 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         url: String, | ||||
|         spout: String, | ||||
|         tags: String, | ||||
|         filter: String, | ||||
|         version: Int | ||||
|     ): SelfossModel.SuccessResponse = | ||||
|         filter: String | ||||
|     ): SuccessResponse = | ||||
|         maybeResponse( | ||||
|             if (version > 1) { | ||||
|             if (appSettingsService.getApiVersion() > 1) { | ||||
|                 createSource2(title, url, spout, tags, filter) | ||||
|             } else { | ||||
|                 createSource(title, url, spout, tags, filter) | ||||
|             } | ||||
|         ) | ||||
|  | ||||
|     suspend fun createSource( | ||||
|     private suspend fun createSource( | ||||
|         title: String, | ||||
|         url: String, | ||||
|         spout: String, | ||||
| @@ -184,8 +237,13 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         filter: String | ||||
|     ): HttpResponse = | ||||
|         client.submitForm( | ||||
|             url = url("/source?username=${appSettingsService.getUserName()}&password=${appSettingsService.getPassword()}"), | ||||
|             url = url("/source"), | ||||
|             formParameters = Parameters.build { | ||||
|                 // TODO: test this | ||||
|                 if (!shouldHavePostLogin()) { | ||||
|                     append("username", appSettingsService.getUserName()) | ||||
|                     append("password", appSettingsService.getPassword()) | ||||
|                 } | ||||
|                 append("title", title) | ||||
|                 append("url", url) | ||||
|                 append("spout", spout) | ||||
| @@ -194,7 +252,7 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|             } | ||||
|         ) | ||||
|  | ||||
|     suspend fun createSource2( | ||||
|     private suspend fun createSource2( | ||||
|         title: String, | ||||
|         url: String, | ||||
|         spout: String, | ||||
| @@ -202,8 +260,12 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|         filter: String | ||||
|     ): HttpResponse = | ||||
|         client.submitForm( | ||||
|             url = url("/source?username=${appSettingsService.getUserName()}&password=${appSettingsService.getPassword()}"), | ||||
|             url = url("/source"), | ||||
|             formParameters = Parameters.build { | ||||
|                 if (!shouldHavePostLogin()) { | ||||
|                     append("username", appSettingsService.getUserName()) | ||||
|                     append("password", appSettingsService.getPassword()) | ||||
|                 } | ||||
|                 append("title", title) | ||||
|                 append("url", url) | ||||
|                 append("spout", spout) | ||||
| @@ -212,25 +274,11 @@ class SelfossApi(private val appSettingsService: AppSettingsService) { | ||||
|             } | ||||
|         ) | ||||
|  | ||||
|     suspend fun deleteSource(id: Int): SelfossModel.SuccessResponse = | ||||
|     suspend fun deleteSource(id: Int): SuccessResponse = | ||||
|         maybeResponse(client.delete(url("/source/$id")) { | ||||
|             if (!shouldHavePostLogin()) { | ||||
|                 parameter("username", appSettingsService.getUserName()) | ||||
|                 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() | ||||
|         } | ||||
|     } | ||||
|             } | ||||
|         }) | ||||
| } | ||||
| @@ -35,6 +35,8 @@ class AppSettingsService { | ||||
|     private var _fontSize: Int? = null | ||||
|     private var _staticBar: Boolean? = null | ||||
|     private var _font: String = "" | ||||
|     private var _theme: Int? = null | ||||
|     private var _enableAnalytics: Boolean? = null | ||||
|  | ||||
|  | ||||
|     init { | ||||
| @@ -42,10 +44,6 @@ class AppSettingsService { | ||||
|         refreshUserSettings() | ||||
|     } | ||||
|  | ||||
|     fun logApiCalls(message: String) { | ||||
|         Napier.d(message, tag = "LogApiCalls") | ||||
|     } | ||||
|  | ||||
|     fun getApiVersion(): Int { | ||||
|         if (_apiVersion == -1) { | ||||
|             refreshApiVersion() | ||||
| @@ -54,8 +52,8 @@ class AppSettingsService { | ||||
|         return _apiVersion | ||||
|     } | ||||
|  | ||||
|     fun refreshApiVersion() { | ||||
|         _apiVersion = settings.getInt("apiVersionMajor", -1) | ||||
|     private fun refreshApiVersion() { | ||||
|         _apiVersion = settings.getInt(API_VERSION_MAJOR, -1) | ||||
|     } | ||||
|  | ||||
|     fun getBaseUrl(): String { | ||||
| @@ -86,8 +84,8 @@ class AppSettingsService { | ||||
|         return _itemsNumber!! | ||||
|     } | ||||
|  | ||||
|     fun refreshItemsNumber() { | ||||
|         _itemsNumber = settings.getString("prefer_api_items_number", "20").toInt() | ||||
|     private fun refreshItemsNumber() { | ||||
|         _itemsNumber = settings.getString(API_ITEMS_NUMBER, "20").toInt() | ||||
|     } | ||||
|  | ||||
|     fun getApiTimeout(): Long { | ||||
| @@ -98,24 +96,24 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshApiTimeout() { | ||||
|         val settingsTimeout = settings.getLong("api_timeout", HttpTimeout.INFINITE_TIMEOUT_MS) | ||||
|         val settingsTimeout = settings.getLong(API_TIMEOUT, HttpTimeout.INFINITE_TIMEOUT_MS) | ||||
|         _apiTimeout = if (settingsTimeout > 0) settingsTimeout else HttpTimeout.INFINITE_TIMEOUT_MS | ||||
|     } | ||||
|  | ||||
|     private fun refreshBaseUrl() { | ||||
|         _baseUrl = settings.getString("url", "") | ||||
|         _baseUrl = settings.getString(BASE_URL, "") | ||||
|     } | ||||
|  | ||||
|     private fun refreshUsername() { | ||||
|         _userName = settings.getString("login", "") | ||||
|         _userName = settings.getString(LOGIN, "") | ||||
|     } | ||||
|  | ||||
|     private fun refreshPassword() { | ||||
|         _password = settings.getString("password", "") | ||||
|         _password = settings.getString(PASSWORD, "") | ||||
|     } | ||||
|  | ||||
|     private fun refreshArticleViewerEnabled() { | ||||
|         _articleViewer = settings.getBoolean("prefer_article_viewer", true) | ||||
|         _articleViewer = settings.getBoolean(PREFER_ARTICLE_VIEWER, true) | ||||
|     } | ||||
|  | ||||
|     fun isArticleViewerEnabled(): Boolean { | ||||
| @@ -125,7 +123,7 @@ class AppSettingsService { | ||||
|         return _articleViewer == true | ||||
|     } | ||||
|     private fun refreshShouldBeCardViewEnabled() { | ||||
|         _shouldBeCardView = settings.getBoolean("card_view_active", false) | ||||
|         _shouldBeCardView = settings.getBoolean(CARD_VIEW_ACTIVE, false) | ||||
|     } | ||||
|  | ||||
|     fun isCardViewEnabled(): Boolean { | ||||
| @@ -135,7 +133,7 @@ class AppSettingsService { | ||||
|         return _shouldBeCardView == true | ||||
|     } | ||||
|     private fun refreshDisplayUnreadCountEnabled() { | ||||
|         _displayUnreadCount = settings.getBoolean("display_unread_count", true) | ||||
|         _displayUnreadCount = settings.getBoolean(DISPLAY_UNREAD_COUNT, true) | ||||
|     } | ||||
|  | ||||
|     fun isDisplayUnreadCountEnabled(): Boolean { | ||||
| @@ -145,7 +143,7 @@ class AppSettingsService { | ||||
|         return _displayUnreadCount == true | ||||
|     } | ||||
|     private fun refreshDisplayAllCountEnabled() { | ||||
|         _displayAllCount = settings.getBoolean("display_other_count", false) | ||||
|         _displayAllCount = settings.getBoolean(DISPLAY_OTHER_COUNT, false) | ||||
|     } | ||||
|  | ||||
|     fun isDisplayAllCountEnabled(): Boolean { | ||||
| @@ -155,7 +153,7 @@ class AppSettingsService { | ||||
|         return _displayAllCount == true | ||||
|     } | ||||
|     private fun refreshFullHeightCardsEnabled() { | ||||
|         _fullHeightCards = settings.getBoolean("full_height_cards", false) | ||||
|         _fullHeightCards = settings.getBoolean(FULL_HEIGHT_CARDS, false) | ||||
|     } | ||||
|  | ||||
|     fun isFullHeightCardsEnabled(): Boolean { | ||||
| @@ -165,7 +163,7 @@ class AppSettingsService { | ||||
|         return _fullHeightCards == true | ||||
|     } | ||||
|     private fun refreshUpdateSourcesEnabled() { | ||||
|         _updateSources = settings.getBoolean("update_sources", true) | ||||
|         _updateSources = settings.getBoolean(UPDATE_SOURCES, true) | ||||
|     } | ||||
|  | ||||
|     fun isUpdateSourcesEnabled(): Boolean { | ||||
| @@ -175,7 +173,7 @@ class AppSettingsService { | ||||
|         return _updateSources == true | ||||
|     } | ||||
|     private fun refreshPeriodicRefreshEnabled() { | ||||
|         _periodicRefresh = settings.getBoolean("periodic_refresh", false) | ||||
|         _periodicRefresh = settings.getBoolean(PERIODIC_REFRESH, false) | ||||
|     } | ||||
|  | ||||
|     fun isPeriodicRefreshEnabled(): Boolean { | ||||
| @@ -186,7 +184,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshRefreshWhenChargingOnlyEnabled() { | ||||
|         _refreshWhenChargingOnly = settings.getBoolean("refresh_when_charging", false) | ||||
|         _refreshWhenChargingOnly = settings.getBoolean(REFRESH_WHEN_CHARGING, false) | ||||
|     } | ||||
|  | ||||
|     fun isRefreshWhenChargingOnlyEnabled(): Boolean { | ||||
| @@ -197,22 +195,22 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshRefreshMinutes() { | ||||
|         _refreshMinutes = settings.getString("periodic_refresh_minutes", "360").toLong() | ||||
|         _refreshMinutes = settings.getString(PERIODIC_REFRESH_MINUTES, "360").toLong() | ||||
|         if (_refreshMinutes <= 15) { | ||||
|             _refreshMinutes = 15 | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun getRefreshMinutes(): Long { | ||||
|         if (_refreshMinutes != null) { | ||||
|         if (_refreshMinutes != 360L) { | ||||
|             refreshRefreshMinutes() | ||||
|         } | ||||
|         return _refreshMinutes | ||||
|     } | ||||
|  | ||||
|     private fun refreshHiddenTags() { | ||||
|         if (settings.getString("hidden_tags", "").isNotEmpty()) { | ||||
|             _hiddenTags = settings.getString("hidden_tags", "").replace("\\s".toRegex(), "").split(",") | ||||
|         if (settings.getString(HIDDEN_TAGS, "").isNotEmpty()) { | ||||
|             _hiddenTags = settings.getString(HIDDEN_TAGS, "").replace("\\s".toRegex(), "").split(",") | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -224,7 +222,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshInfiniteLoadingEnabled() { | ||||
|         _infiniteLoading = settings.getBoolean("infinite_loading", false) | ||||
|         _infiniteLoading = settings.getBoolean(INFINITE_LOADING, false) | ||||
|     } | ||||
|  | ||||
|     fun isInfiniteLoadingEnabled(): Boolean { | ||||
| @@ -235,7 +233,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshItemCachingEnabled() { | ||||
|         _itemsCaching = settings.getBoolean("items_caching", false) | ||||
|         _itemsCaching = settings.getBoolean(ITEMS_CACHING, false) | ||||
|     } | ||||
|  | ||||
|     fun isItemCachingEnabled(): Boolean { | ||||
| @@ -246,7 +244,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshNotifyNewItemsEnabled() { | ||||
|         _notifyNewItems = settings.getBoolean("notify_new_items", false) | ||||
|         _notifyNewItems = settings.getBoolean(NOTIFY_NEW_ITEMS, false) | ||||
|     } | ||||
|  | ||||
|     fun isNotifyNewItemsEnabled(): Boolean { | ||||
| @@ -258,7 +256,7 @@ class AppSettingsService { | ||||
|  | ||||
|  | ||||
|     private fun refreshMarkOnScrollEnabled() { | ||||
|         _markOnScroll = settings.getBoolean("mark_on_scroll", false) | ||||
|         _markOnScroll = settings.getBoolean(MARK_ON_SCROLL, false) | ||||
|     } | ||||
|  | ||||
|     fun isMarkOnScrollEnabled(): Boolean { | ||||
| @@ -270,7 +268,7 @@ class AppSettingsService { | ||||
|  | ||||
|  | ||||
|     private fun refreshActiveAllignment() { | ||||
|         _activeAlignment = settings.getInt("text_align", JUSTIFY) | ||||
|         _activeAlignment = settings.getInt(TEXT_ALIGN, JUSTIFY) | ||||
|     } | ||||
|  | ||||
|     fun getActiveAllignment(): Int { | ||||
| @@ -281,12 +279,12 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     fun changeAllignment(allignment: Int) { | ||||
|         settings.putInt("text_align", allignment) | ||||
|         settings.putInt(TEXT_ALIGN, allignment) | ||||
|         _activeAlignment = allignment | ||||
|     } | ||||
|  | ||||
|     private fun refreshFontSize() { | ||||
|         _fontSize = settings.getString("reader_font_size", "16").toInt() | ||||
|         _fontSize = settings.getString(READER_FONT_SIZE, "16").toInt() | ||||
|     } | ||||
|  | ||||
|     fun getFontSize(): Int { | ||||
| @@ -297,7 +295,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     private fun refreshStaticBarEnabled() { | ||||
|         _staticBar = settings.getBoolean("reader_static_bar", false) | ||||
|         _staticBar = settings.getBoolean(READER_STATIC_BAR, false) | ||||
|     } | ||||
|  | ||||
|     fun isStaticBarEnabled(): Boolean { | ||||
| @@ -307,17 +305,39 @@ class AppSettingsService { | ||||
|         return _staticBar == true | ||||
|     } | ||||
|  | ||||
|     private fun refreshAnalyticsEnabled() { | ||||
|         _enableAnalytics = settings.getBoolean(ENABLE_ANALYTICS, true) | ||||
|     } | ||||
|  | ||||
|     fun isAnalyticsEnabled(): Boolean { | ||||
|         if (_enableAnalytics != null) { | ||||
|             refreshAnalyticsEnabled() | ||||
|         } | ||||
|         return _enableAnalytics == true | ||||
|     } | ||||
|  | ||||
|     private fun refreshFont() { | ||||
|         _font = settings.getString("reader_font", "") | ||||
|         _font = settings.getString(READER_FONT, "") | ||||
|     } | ||||
|  | ||||
|     fun getFont(): String { | ||||
|         if (_font != null) { | ||||
|         if (_font.isEmpty()) { | ||||
|             refreshFont() | ||||
|         } | ||||
|         return _font | ||||
|     } | ||||
|  | ||||
|     private fun refreshCurrentTheme() { | ||||
|         _theme = settings.getString(CURRENT_THEME, "-1").toInt() | ||||
|     } | ||||
|  | ||||
|     fun getCurrentTheme(): Int { | ||||
|         if (_theme == null) { | ||||
|             refreshCurrentTheme() | ||||
|         } | ||||
|         return _theme ?: -1 | ||||
|     } | ||||
|  | ||||
|     fun refreshApiSettings() { | ||||
|         refreshPassword() | ||||
|         refreshUsername() | ||||
| @@ -346,6 +366,8 @@ class AppSettingsService { | ||||
|         refreshFontSize() | ||||
|         refreshFont() | ||||
|         refreshStaticBarEnabled() | ||||
|         refreshCurrentTheme() | ||||
|         refreshAnalyticsEnabled() | ||||
|     } | ||||
|  | ||||
|     fun refreshLoginInformation( | ||||
| @@ -353,21 +375,21 @@ class AppSettingsService { | ||||
|         login: String, | ||||
|         password: String | ||||
|     ) { | ||||
|         settings.putString("url", url) | ||||
|         settings.putString("login", login) | ||||
|         settings.putString("password", password) | ||||
|         settings.putString(BASE_URL, url) | ||||
|         settings.putString(LOGIN, login) | ||||
|         settings.putString(PASSWORD, password) | ||||
|         refreshApiSettings() | ||||
|     } | ||||
|  | ||||
|     fun resetLoginInformation() { | ||||
|         settings.remove("url") | ||||
|         settings.remove("login") | ||||
|         settings.remove("password") | ||||
|         settings.remove(BASE_URL) | ||||
|         settings.remove(LOGIN) | ||||
|         settings.remove(PASSWORD) | ||||
|         refreshApiSettings() | ||||
|     } | ||||
|  | ||||
|     fun updateApiVersion(apiMajorVersion: Int) { | ||||
|         settings.putInt("apiVersionMajor", apiMajorVersion) | ||||
|         settings.putInt(API_VERSION_MAJOR, apiMajorVersion) | ||||
|         refreshApiVersion() | ||||
|     } | ||||
|  | ||||
| @@ -378,7 +400,7 @@ class AppSettingsService { | ||||
|     } | ||||
|  | ||||
|     fun disableArticleViewer() { | ||||
|         settings.putBoolean("prefer_article_viewer", false) | ||||
|         settings.putBoolean(PREFER_ARTICLE_VIEWER, false) | ||||
|         refreshArticleViewerEnabled() | ||||
|     } | ||||
|  | ||||
| @@ -396,5 +418,57 @@ class AppSettingsService { | ||||
|         const val JUSTIFY = 1 | ||||
|  | ||||
|         const val ALIGN_LEFT = 2 | ||||
|  | ||||
|         const val API_VERSION_MAJOR = "apiVersionMajor" | ||||
|  | ||||
|         const val API_ITEMS_NUMBER = "prefer_api_items_number" | ||||
|  | ||||
|         const val API_TIMEOUT = "api_timeout" | ||||
|  | ||||
|         const val BASE_URL = "url" | ||||
|  | ||||
|         const val LOGIN = "login" | ||||
|  | ||||
|         const val PASSWORD = "password" | ||||
|  | ||||
|         const val PREFER_ARTICLE_VIEWER = "prefer_article_viewer" | ||||
|  | ||||
|         const val CARD_VIEW_ACTIVE = "card_view_active" | ||||
|  | ||||
|         const val DISPLAY_UNREAD_COUNT = "display_unread_count" | ||||
|  | ||||
|         const val DISPLAY_OTHER_COUNT = "display_other_count" | ||||
|  | ||||
|         const val FULL_HEIGHT_CARDS = "full_height_cards" | ||||
|  | ||||
|         const val UPDATE_SOURCES = "update_sources" | ||||
|  | ||||
|         const val PERIODIC_REFRESH = "periodic_refresh" | ||||
|  | ||||
|         const val REFRESH_WHEN_CHARGING = "refresh_when_charging" | ||||
|  | ||||
|         const val READER_FONT = "reader_font" | ||||
|  | ||||
|         const val READER_STATIC_BAR = "reader_static_bar" | ||||
|  | ||||
|         const val READER_FONT_SIZE = "reader_font_size" | ||||
|  | ||||
|         const val TEXT_ALIGN = "text_align" | ||||
|  | ||||
|         const val MARK_ON_SCROLL = "mark_on_scroll" | ||||
|  | ||||
|         const val NOTIFY_NEW_ITEMS = "notify_new_items" | ||||
|  | ||||
|         const val PERIODIC_REFRESH_MINUTES = "periodic_refresh_minutes" | ||||
|  | ||||
|         const val HIDDEN_TAGS = "hidden_tags" | ||||
|  | ||||
|         const val INFINITE_LOADING = "infinite_loading" | ||||
|  | ||||
|         const val ITEMS_CACHING = "items_caching" | ||||
|  | ||||
|         const val CURRENT_THEME = "currentMode" | ||||
|  | ||||
|         const val ENABLE_ANALYTICS = "enable_analytics" | ||||
|     } | ||||
| } | ||||
| @@ -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 | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.model.SelfossModel | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| expect class DateUtils() { | ||||
|     companion object { | ||||
|         fun parseDate(dateString: String): Long | ||||
|  | ||||
|  | ||||
| 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 parseRelativeDate(dateString: String): String | ||||
|         fun parseRelativeDate(dateString: String): String | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -65,6 +65,6 @@ fun SelfossModel.Item.toEntity(): ITEM = | ||||
|         this.thumbnail, | ||||
|         this.icon, | ||||
|         this.link, | ||||
|         this.title.getHtmlDecoded(), | ||||
|         this.sourcetitle.getHtmlDecoded(), | ||||
|         this.tags.joinToString(",") | ||||
|     ) | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,14 +1,13 @@ | ||||
| package bou.amine.apps.readerforselfossv2.utils | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
| actual class DateUtils { | ||||
|     actual companion object { | ||||
|         actual fun parseDate(dateString: String): Long { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|  | ||||
| actual class DateUtils actual constructor(actual val appSettingsService: AppSettingsService) { | ||||
|     actual fun parseDate(dateString: String): Long { | ||||
|         TODO("Not yet implemented") | ||||
|         actual fun parseRelativeDate(dateString: String): String { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     actual fun parseRelativeDate(dateString: String): String { | ||||
|         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,13 +2,15 @@ package bou.amine.apps.readerforselfossv2.utils | ||||
|  | ||||
| import bou.amine.apps.readerforselfossv2.service.AppSettingsService | ||||
|  | ||||
| actual class DateUtils actual constructor(actual val appSettingsService: AppSettingsService) { | ||||
|     actual fun parseDate(dateString: String): Long { | ||||
|         TODO("Not yet implemented") | ||||
|     } | ||||
| actual class DateUtils { | ||||
|     actual companion object { | ||||
|         actual fun parseDate(dateString: String): Long { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|  | ||||
|     actual fun parseRelativeDate(dateString: String): String { | ||||
|         TODO("Not yet implemented") | ||||
|         actual fun parseRelativeDate(dateString: String): String { | ||||
|             TODO("Not yet implemented") | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
		Reference in New Issue
	
	Block a user