Compare commits
	
		
			26 Commits
		
	
	
		
			v171810290
			...
			v171811309
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 7e520e9bed | ||
|  | 32e2d05014 | ||
|  | 40d9c97f73 | ||
|  | 1aa68d3449 | ||
|  | aeeac8cccd | ||
|  | 7292edf997 | ||
|  | f49256c72f | ||
|  | d02b28b81f | ||
|  | 08117043dd | ||
|  | 63496c993e | ||
|  | 00ef542e49 | ||
|  | a78c6e6b33 | ||
|  | 363eaf9bf9 | ||
|  | fec6683701 | ||
|  | 1549edb647 | ||
|  | 3de48ba162 | ||
|  | a2a3d6f1a7 | ||
|  | ccab2c7648 | ||
|  | 880dd1db5c | ||
|  | ed18fea356 | ||
|  | 9816b20bf6 | ||
|  | 0bb2195bff | ||
|  | ab2d0c4036 | ||
|  | 99fc417109 | ||
|  | dc304ef8c1 | ||
|  | c5511880bc | 
| @@ -1,9 +1,17 @@ | |||||||
| **1.7.x** | **1.7.x** | ||||||
|  |  | ||||||
|  | - Closes #216. Issue with selfoss version 2.19. | ||||||
|  |  | ||||||
|  | - Closes #179. Sync of read/unread/star/unstar items on background task or on app reload with network available. | ||||||
|  |  | ||||||
|  | - Closes #33. Background sync with settings. | ||||||
|  |  | ||||||
| - Closing #1. Initial article caching. | - Closing #1. Initial article caching. | ||||||
|  |  | ||||||
| - Closing #228 by removing the list action bar. Action buttons are exclusively on the card view from now on. | - Closing #228 by removing the list action bar. Action buttons are exclusively on the card view from now on. | ||||||
|  |  | ||||||
|  | - Closing #38. Only doing api calls on network available. | ||||||
|  |  | ||||||
| **1.6.x** | **1.6.x** | ||||||
|  |  | ||||||
| - Handling hidden tags. | - Handling hidden tags. | ||||||
|   | |||||||
| @@ -152,14 +152,16 @@ dependencies { | |||||||
|     implementation 'androidx.core:core-ktx:1.0.0' |     implementation 'androidx.core:core-ktx:1.0.0' | ||||||
|  |  | ||||||
|     // Crash |     // Crash | ||||||
|     implementation 'ch.acra:acra-http:5.1.3' |     implementation 'ch.acra:acra-http:5.2.1' | ||||||
|     implementation 'ch.acra:acra-dialog:5.1.3' |     implementation 'ch.acra:acra-dialog:5.2.1' | ||||||
|  |  | ||||||
|     implementation "androidx.lifecycle:lifecycle-livedata:$lifecycle_version" |     implementation "androidx.lifecycle:lifecycle-livedata:$lifecycle_version" | ||||||
|     implementation "androidx.lifecycle:lifecycle-common-java8:$lifecycle_version" |     implementation "androidx.lifecycle:lifecycle-common-java8:$lifecycle_version" | ||||||
|  |  | ||||||
|     implementation "androidx.room:room-runtime:$room_version" |     implementation "androidx.room:room-runtime:$room_version" | ||||||
|     kapt "androidx.room:room-compiler:$room_version" |     kapt "androidx.room:room-compiler:$room_version" | ||||||
|  |  | ||||||
|  |     implementation "android.arch.work:work-runtime-ktx:$work_version" | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										19
									
								
								app/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								app/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							| @@ -30,22 +30,6 @@ | |||||||
|     <fields>; |     <fields>; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| ##Retrofit |  | ||||||
| #-keep class com.google.gson.** { *; } |  | ||||||
| #-keep class com.google.inject.** { *; } |  | ||||||
| #-keep class org.apache.http.** { *; } |  | ||||||
| #-keep class org.apache.james.mime4j.** { *; } |  | ||||||
| #-keep class javax.inject.** { *; } |  | ||||||
| #-keep class retrofit.** { *; } |  | ||||||
| #-keepclassmembernames interface * { |  | ||||||
| #    @retrofit.http.* <methods>; |  | ||||||
| #} |  | ||||||
| #-keep class retrofit.** { *; } |  | ||||||
| #-keep class apps.amine.bou.readerforselfoss.api.selfoss.model.** { *; } |  | ||||||
| #-keepclassmembernames interface * { |  | ||||||
| #    @retrofit.http.* <methods>; |  | ||||||
| #} |  | ||||||
| -dontwarn okio.** | -dontwarn okio.** | ||||||
| -dontwarn retrofit2.Platform$Java8 | -dontwarn retrofit2.Platform$Java8 | ||||||
| -keep class retrofit.** { *; } | -keep class retrofit.** { *; } | ||||||
| @@ -76,3 +60,6 @@ | |||||||
| -dontwarn javax.annotation.** | -dontwarn javax.annotation.** | ||||||
|  |  | ||||||
| -keep class android.support.v7.widget.SearchView { *; } | -keep class android.support.v7.widget.SearchView { *; } | ||||||
|  |  | ||||||
|  | # maybe remove later ? | ||||||
|  | -keep class * extends androidx.fragment.app.Fragment | ||||||
|   | |||||||
| @@ -0,0 +1,226 @@ | |||||||
|  | { | ||||||
|  |   "formatVersion": 1, | ||||||
|  |   "database": { | ||||||
|  |     "version": 3, | ||||||
|  |     "identityHash": "7ad9c4961992c13b670128485ebb3efc", | ||||||
|  |     "entities": [ | ||||||
|  |       { | ||||||
|  |         "tableName": "tags", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`tag` TEXT NOT NULL, `color` TEXT NOT NULL, `unread` INTEGER NOT NULL, PRIMARY KEY(`tag`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tag", | ||||||
|  |             "columnName": "tag", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "color", | ||||||
|  |             "columnName": "color", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "tag" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "sources", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `title` TEXT NOT NULL, `tags` TEXT NOT NULL, `spout` TEXT NOT NULL, `error` TEXT NOT NULL, `icon` TEXT NOT NULL, PRIMARY KEY(`id`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "title", | ||||||
|  |             "columnName": "title", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tags", | ||||||
|  |             "columnName": "tags", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "spout", | ||||||
|  |             "columnName": "spout", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "error", | ||||||
|  |             "columnName": "error", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "items", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `datetime` TEXT NOT NULL, `title` TEXT NOT NULL, `content` TEXT NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `thumbnail` TEXT NOT NULL, `icon` TEXT NOT NULL, `link` TEXT NOT NULL, `sourcetitle` TEXT NOT NULL, `tags` TEXT NOT NULL, PRIMARY KEY(`id`))", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "datetime", | ||||||
|  |             "columnName": "datetime", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "title", | ||||||
|  |             "columnName": "title", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "content", | ||||||
|  |             "columnName": "content", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "starred", | ||||||
|  |             "columnName": "starred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "thumbnail", | ||||||
|  |             "columnName": "thumbnail", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "link", | ||||||
|  |             "columnName": "link", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "sourcetitle", | ||||||
|  |             "columnName": "sourcetitle", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "tags", | ||||||
|  |             "columnName": "tags", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": false | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "tableName": "actions", | ||||||
|  |         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `articleid` TEXT NOT NULL, `read` INTEGER NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `unstarred` INTEGER NOT NULL)", | ||||||
|  |         "fields": [ | ||||||
|  |           { | ||||||
|  |             "fieldPath": "id", | ||||||
|  |             "columnName": "id", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "articleId", | ||||||
|  |             "columnName": "articleid", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "read", | ||||||
|  |             "columnName": "read", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unread", | ||||||
|  |             "columnName": "unread", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "starred", | ||||||
|  |             "columnName": "starred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "unstarred", | ||||||
|  |             "columnName": "unstarred", | ||||||
|  |             "affinity": "INTEGER", | ||||||
|  |             "notNull": true | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "primaryKey": { | ||||||
|  |           "columnNames": [ | ||||||
|  |             "id" | ||||||
|  |           ], | ||||||
|  |           "autoGenerate": true | ||||||
|  |         }, | ||||||
|  |         "indices": [], | ||||||
|  |         "foreignKeys": [] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "views": [], | ||||||
|  |     "setupQueries": [ | ||||||
|  |       "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", | ||||||
|  |       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, \"7ad9c4961992c13b670128485ebb3efc\")" | ||||||
|  |     ] | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -3,6 +3,7 @@ | |||||||
|     package="apps.amine.bou.readerforselfoss" |     package="apps.amine.bou.readerforselfoss" | ||||||
|     xmlns:tools="http://schemas.android.com/tools"> |     xmlns:tools="http://schemas.android.com/tools"> | ||||||
|  |  | ||||||
|  |     <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> | ||||||
|     <uses-permission android:name="android.permission.INTERNET" /> |     <uses-permission android:name="android.permission.INTERNET" /> | ||||||
|  |  | ||||||
|     <application |     <application | ||||||
|   | |||||||
| @@ -25,6 +25,12 @@ import android.view.View | |||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import androidx.room.Room | import androidx.room.Room | ||||||
| import androidx.room.RoomDatabase | import androidx.room.RoomDatabase | ||||||
|  | import androidx.work.Constraints | ||||||
|  | import androidx.work.ExistingPeriodicWorkPolicy | ||||||
|  | import androidx.work.NetworkType | ||||||
|  | import androidx.work.OneTimeWorkRequestBuilder | ||||||
|  | import androidx.work.PeriodicWorkRequestBuilder | ||||||
|  | import androidx.work.WorkManager | ||||||
| import apps.amine.bou.readerforselfoss.adapters.ItemCardAdapter | import apps.amine.bou.readerforselfoss.adapters.ItemCardAdapter | ||||||
| import apps.amine.bou.readerforselfoss.adapters.ItemListAdapter | import apps.amine.bou.readerforselfoss.adapters.ItemListAdapter | ||||||
| import apps.amine.bou.readerforselfoss.adapters.ItemsAdapter | import apps.amine.bou.readerforselfoss.adapters.ItemsAdapter | ||||||
| @@ -34,8 +40,11 @@ import apps.amine.bou.readerforselfoss.api.selfoss.Source | |||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Stats | import apps.amine.bou.readerforselfoss.api.selfoss.Stats | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Tag | import apps.amine.bou.readerforselfoss.api.selfoss.Tag | ||||||
|  | import apps.amine.bou.readerforselfoss.background.LoadingWorker | ||||||
| import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
| import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_1_2 | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_1_2 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_2_3 | ||||||
| import apps.amine.bou.readerforselfoss.settings.SettingsActivity | import apps.amine.bou.readerforselfoss.settings.SettingsActivity | ||||||
| import apps.amine.bou.readerforselfoss.themes.AppColors | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
| import apps.amine.bou.readerforselfoss.themes.Toppings | import apps.amine.bou.readerforselfoss.themes.Toppings | ||||||
| @@ -47,6 +56,7 @@ import apps.amine.bou.readerforselfoss.utils.drawer.CustomUrlPrimaryDrawerItem | |||||||
| import apps.amine.bou.readerforselfoss.utils.flattenTags | import apps.amine.bou.readerforselfoss.utils.flattenTags | ||||||
| import apps.amine.bou.readerforselfoss.utils.longHash | import apps.amine.bou.readerforselfoss.utils.longHash | ||||||
| import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | ||||||
| import apps.amine.bou.readerforselfoss.utils.persistence.toView | import apps.amine.bou.readerforselfoss.utils.persistence.toView | ||||||
| import co.zsmb.materialdrawerkt.builders.accountHeader | import co.zsmb.materialdrawerkt.builders.accountHeader | ||||||
| @@ -74,6 +84,7 @@ import org.acra.ACRA | |||||||
| import retrofit2.Call | import retrofit2.Call | ||||||
| import retrofit2.Callback | import retrofit2.Callback | ||||||
| import retrofit2.Response | import retrofit2.Response | ||||||
|  | import java.util.concurrent.TimeUnit | ||||||
| import kotlin.concurrent.thread | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
| class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | ||||||
| @@ -110,6 +121,10 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|     private var itemsCaching: Boolean = false |     private var itemsCaching: Boolean = false | ||||||
|     private var hiddenTags: List<String> = emptyList() |     private var hiddenTags: List<String> = emptyList() | ||||||
|  |  | ||||||
|  |     private var periodicRefresh = false | ||||||
|  |     private var refreshMinutes: Long = 360L | ||||||
|  |     private var refreshWhenChargingOnly = false | ||||||
|  |  | ||||||
|     private lateinit var tabNewBadge: TextBadgeItem |     private lateinit var tabNewBadge: TextBadgeItem | ||||||
|     private lateinit var tabArchiveBadge: TextBadgeItem |     private lateinit var tabArchiveBadge: TextBadgeItem | ||||||
|     private lateinit var tabStarredBadge: TextBadgeItem |     private lateinit var tabStarredBadge: TextBadgeItem | ||||||
| @@ -158,7 +173,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|         db = Room.databaseBuilder( |         db = Room.databaseBuilder( | ||||||
|             applicationContext, |             applicationContext, | ||||||
|             AppDatabase::class.java, "selfoss-database" |             AppDatabase::class.java, "selfoss-database" | ||||||
|         ).addMigrations(MIGRATION_1_2).build() |         ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).build() | ||||||
|  |  | ||||||
|  |  | ||||||
|         customTabActivityHelper = CustomTabActivityHelper() |         customTabActivityHelper = CustomTabActivityHelper() | ||||||
| @@ -233,7 +248,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                         badgeNew-- |                         badgeNew-- | ||||||
|                         reloadBadgeContent() |                         reloadBadgeContent() | ||||||
|  |  | ||||||
|                         val tagHashes = i.tags.split(",").map { it.longHash() } |                         val tagHashes = i.tags.tags.split(",").map { it.longHash() } | ||||||
|                         tagsBadge = tagsBadge.map { |                         tagsBadge = tagsBadge.map { | ||||||
|                             if (tagHashes.contains(it.key)) { |                             if (tagHashes.contains(it.key)) { | ||||||
|                                 (it.key to (it.value - 1)) |                                 (it.key to (it.value - 1)) | ||||||
| @@ -333,8 +348,11 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|  |  | ||||||
|         getElementsAccordingToTab() |         getElementsAccordingToTab() | ||||||
|  |  | ||||||
|  |  | ||||||
|         handleGDPRDialog(sharedPref.getBoolean("GDPR_shown", false)) |         handleGDPRDialog(sharedPref.getBoolean("GDPR_shown", false)) | ||||||
|  |  | ||||||
|  |         handleRecurringTask() | ||||||
|  |  | ||||||
|  |         handleOfflineActions() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun getAndStoreAllItems() { |     private fun getAndStoreAllItems() { | ||||||
| @@ -349,7 +367,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                 thread { |                 thread { | ||||||
|                     if (response.body() != null) { |                     if (response.body() != null) { | ||||||
|                         val apiItems = (response.body() as ArrayList<Item>).filter { |                         val apiItems = (response.body() as ArrayList<Item>).filter { | ||||||
|                             maybeTagFilter != null || filter(it.tags) |                             maybeTagFilter != null || filter(it.tags.tags) | ||||||
|                         } as ArrayList<Item> |                         } as ArrayList<Item> | ||||||
|                         db.itemsDao().deleteAllItems() |                         db.itemsDao().deleteAllItems() | ||||||
|                         db.itemsDao() |                         db.itemsDao() | ||||||
| @@ -384,6 +402,13 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|         } else { |         } else { | ||||||
|             emptyList() |             emptyList() | ||||||
|         } |         } | ||||||
|  |         periodicRefresh = sharedPref.getBoolean("periodic_refresh", false) | ||||||
|  |         refreshWhenChargingOnly = sharedPref.getBoolean("refresh_when_charging", false) | ||||||
|  |         refreshMinutes = sharedPref.getString("periodic_refresh_minutes", "360").toLong() | ||||||
|  |  | ||||||
|  |         if (refreshMinutes <= 15) { | ||||||
|  |             refreshMinutes = 15 | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun handleThemeBinding() { |     private fun handleThemeBinding() { | ||||||
| @@ -696,6 +721,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|             var sources: List<Source>? |             var sources: List<Source>? | ||||||
|  |  | ||||||
|             fun sourcesApiCall() { |             fun sourcesApiCall() { | ||||||
|  |                 if (this@HomeActivity.isNetworkAccessible(null)) { | ||||||
|                     api.sources.enqueue(object : Callback<List<Source>> { |                     api.sources.enqueue(object : Callback<List<Source>> { | ||||||
|                         override fun onResponse( |                         override fun onResponse( | ||||||
|                             call: Call<List<Source>>?, |                             call: Call<List<Source>>?, | ||||||
| @@ -709,10 +735,16 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                         } |                         } | ||||||
|  |  | ||||||
|                         override fun onFailure(call: Call<List<Source>>?, t: Throwable?) { |                         override fun onFailure(call: Call<List<Source>>?, t: Throwable?) { | ||||||
|  |                             val apiDrawerData = DrawerData(tags, null) | ||||||
|  |                             if ((maybeDrawerData != null && maybeDrawerData != apiDrawerData) || maybeDrawerData == null) { | ||||||
|  |                                 handleDrawerData(apiDrawerData) | ||||||
|  |                             } | ||||||
|                         } |                         } | ||||||
|                     }) |                     }) | ||||||
|                 } |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (this@HomeActivity.isNetworkAccessible(null)) { | ||||||
|                 api.tags.enqueue(object : Callback<List<Tag>> { |                 api.tags.enqueue(object : Callback<List<Tag>> { | ||||||
|                     override fun onResponse( |                     override fun onResponse( | ||||||
|                         call: Call<List<Tag>>, |                         call: Call<List<Tag>>, | ||||||
| @@ -727,6 +759,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                     } |                     } | ||||||
|                 }) |                 }) | ||||||
|             } |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|         drawer.addItem( |         drawer.addItem( | ||||||
|             PrimaryDrawerItem().withName(getString(R.string.drawer_loading)).withSelectable( |             PrimaryDrawerItem().withName(getString(R.string.drawer_loading)).withSelectable( | ||||||
| @@ -817,12 +850,52 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|             override fun onTabSelected(position: Int) { |             override fun onTabSelected(position: Int) { | ||||||
|                 offset = 0 |                 offset = 0 | ||||||
|                 lastFetchDone = false |                 lastFetchDone = false | ||||||
|  |  | ||||||
|  |                 if (itemsCaching) { | ||||||
|  |  | ||||||
|  |                     if (!swipeRefreshLayout.isRefreshing) { | ||||||
|  |                         swipeRefreshLayout.post { swipeRefreshLayout.isRefreshing = true } | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     thread { | ||||||
|  |                         val dbItems = db.itemsDao().items().map { it.toView() } | ||||||
|  |                         runOnUiThread { | ||||||
|  |                             if (dbItems.isNotEmpty()) { | ||||||
|  |                                 items = when (position) { | ||||||
|  |                                     0 -> ArrayList(dbItems.filter { it.unread }) | ||||||
|  |                                     1 -> ArrayList(dbItems.filter { !it.unread }) | ||||||
|  |                                     2 -> ArrayList(dbItems.filter { it.starred }) | ||||||
|  |                                     else -> ArrayList(dbItems.filter { it.unread }) | ||||||
|  |                                 } | ||||||
|  |                                 handleListResult() | ||||||
|                                 when (position) { |                                 when (position) { | ||||||
|                                     0 -> getUnRead() |                                     0 -> getUnRead() | ||||||
|                                     1 -> getRead() |                                     1 -> getRead() | ||||||
|                                     2 -> getStarred() |                                     2 -> getStarred() | ||||||
|                                     else -> Unit |                                     else -> Unit | ||||||
|                                 } |                                 } | ||||||
|  |                             } else { | ||||||
|  |                                 if (this@HomeActivity.isNetworkAccessible(this@HomeActivity.findViewById(R.id.coordLayout))) { | ||||||
|  |                                     when (position) { | ||||||
|  |                                         0 -> getUnRead() | ||||||
|  |                                         1 -> getRead() | ||||||
|  |                                         2 -> getStarred() | ||||||
|  |                                         else -> Unit | ||||||
|  |                                     } | ||||||
|  |                                     getAndStoreAllItems() | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                 } else { | ||||||
|  |                     when (position) { | ||||||
|  |                         0 -> getUnRead() | ||||||
|  |                         1 -> getRead() | ||||||
|  |                         2 -> getStarred() | ||||||
|  |                         else -> Unit | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         }) |         }) | ||||||
|     } |     } | ||||||
| @@ -899,11 +972,13 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                         handleListResult() |                         handleListResult() | ||||||
|                         doGetAccordingToTab() |                         doGetAccordingToTab() | ||||||
|                     } else { |                     } else { | ||||||
|  |                         if (this@HomeActivity.isNetworkAccessible(this@HomeActivity.findViewById(R.id.coordLayout))) { | ||||||
|                             doGetAccordingToTab() |                             doGetAccordingToTab() | ||||||
|                             getAndStoreAllItems() |                             getAndStoreAllItems() | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|         } else { |         } else { | ||||||
|             doGetAccordingToTab() |             doGetAccordingToTab() | ||||||
| @@ -928,7 +1003,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                     getAndStoreAllItems() |                     getAndStoreAllItems() | ||||||
|                     items = response.body() as ArrayList<Item> |                     items = response.body() as ArrayList<Item> | ||||||
|                     items = items.filter { |                     items = items.filter { | ||||||
|                         maybeTagFilter != null || filter(it.tags) |                         maybeTagFilter != null || filter(it.tags.tags) | ||||||
|                     } as ArrayList<Item> |                     } as ArrayList<Item> | ||||||
|  |  | ||||||
|                     if (allItems.isEmpty()) { |                     if (allItems.isEmpty()) { | ||||||
| @@ -956,6 +1031,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|             swipeRefreshLayout.post { swipeRefreshLayout.isRefreshing = true } |             swipeRefreshLayout.post { swipeRefreshLayout.isRefreshing = true } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (this@HomeActivity.isNetworkAccessible(this@HomeActivity.findViewById(R.id.coordLayout))) { | ||||||
|             call(maybeTagFilter?.tag, maybeSourceFilter?.id?.toLong(), maybeSearchFilter) |             call(maybeTagFilter?.tag, maybeSourceFilter?.id?.toLong(), maybeSearchFilter) | ||||||
|                 .enqueue(object : Callback<List<Item>> { |                 .enqueue(object : Callback<List<Item>> { | ||||||
|                     override fun onResponse( |                     override fun onResponse( | ||||||
| @@ -974,6 +1050,9 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                         ).show() |                         ).show() | ||||||
|                     } |                     } | ||||||
|                 }) |                 }) | ||||||
|  |         } else { | ||||||
|  |             swipeRefreshLayout.post { swipeRefreshLayout.isRefreshing = false } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun getUnRead(appendResults: Boolean = false) { |     private fun getUnRead(appendResults: Boolean = false) { | ||||||
| @@ -1082,7 +1161,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun reloadBadges() { |     private fun reloadBadges() { | ||||||
|         if (displayUnreadCount || displayAllCount) { |         if (this@HomeActivity.isNetworkAccessible(null) && (displayUnreadCount || displayAllCount)) { | ||||||
|             api.stats.enqueue(object : Callback<Stats> { |             api.stats.enqueue(object : Callback<Stats> { | ||||||
|                 override fun onResponse(call: Call<Stats>, response: Response<Stats>) { |                 override fun onResponse(call: Call<Stats>, response: Response<Stats>) { | ||||||
|                     if (response.body() != null) { |                     if (response.body() != null) { | ||||||
| @@ -1188,6 +1267,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|         when (item.itemId) { |         when (item.itemId) { | ||||||
|             R.id.refresh -> { |             R.id.refresh -> { | ||||||
|  |                 if (this@HomeActivity.isNetworkAccessible(null)) { | ||||||
|                     needsConfirmation(R.string.menu_home_refresh, R.string.refresh_dialog_message) { |                     needsConfirmation(R.string.menu_home_refresh, R.string.refresh_dialog_message) { | ||||||
|                         api.update().enqueue(object : Callback<String> { |                         api.update().enqueue(object : Callback<String> { | ||||||
|                             override fun onResponse( |                             override fun onResponse( | ||||||
| @@ -1212,6 +1292,9 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                         Toast.makeText(this, R.string.refresh_in_progress, Toast.LENGTH_SHORT).show() |                         Toast.makeText(this, R.string.refresh_in_progress, Toast.LENGTH_SHORT).show() | ||||||
|                     } |                     } | ||||||
|                     return true |                     return true | ||||||
|  |                 } else { | ||||||
|  |                     return false | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             R.id.readAll -> { |             R.id.readAll -> { | ||||||
|                 if (elementsShown == UNREAD_SHOWN) { |                 if (elementsShown == UNREAD_SHOWN) { | ||||||
| @@ -1220,7 +1303,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                         val ids = allItems.map { it.id } |                         val ids = allItems.map { it.id } | ||||||
|                         val itemsByTag: Map<Long, Int> = |                         val itemsByTag: Map<Long, Int> = | ||||||
|                             allItems.flattenTags() |                             allItems.flattenTags() | ||||||
|                                 .groupBy { it.tags.longHash() } |                                 .groupBy { it.tags.tags.longHash() } | ||||||
|                                 .map { it.key to it.value.size } |                                 .map { it.key to it.value.size } | ||||||
|                                 .toMap() |                                 .toMap() | ||||||
|  |  | ||||||
| @@ -1228,7 +1311,7 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|                             ACRA.getErrorReporter().maybeHandleSilentException(e, this@HomeActivity) |                             ACRA.getErrorReporter().maybeHandleSilentException(e, this@HomeActivity) | ||||||
|                         } |                         } | ||||||
|  |  | ||||||
|                         if (ids.isNotEmpty()) { |                         if (ids.isNotEmpty() && this@HomeActivity.isNetworkAccessible(null)) { | ||||||
|                             api.readAll(ids).enqueue(object : Callback<SuccessResponse> { |                             api.readAll(ids).enqueue(object : Callback<SuccessResponse> { | ||||||
|                                 override fun onResponse( |                                 override fun onResponse( | ||||||
|                                     call: Call<SuccessResponse>, |                                     call: Call<SuccessResponse>, | ||||||
| @@ -1334,5 +1417,58 @@ class HomeActivity : AppCompatActivity(), SearchView.OnQueryTextListener { | |||||||
|             alertDialog.show() |             alertDialog.show() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private fun handleRecurringTask() { | ||||||
|  |         if (periodicRefresh) { | ||||||
|  |             val myConstraints = Constraints.Builder() | ||||||
|  |                 .setRequiresBatteryNotLow(true) | ||||||
|  |                 .setRequiresCharging(refreshWhenChargingOnly) | ||||||
|  |                 .setRequiresStorageNotLow(true) | ||||||
|  |                 .build() | ||||||
|  |  | ||||||
|  |             val backgroundWork = | ||||||
|  |                 PeriodicWorkRequestBuilder<LoadingWorker>(refreshMinutes, TimeUnit.MINUTES) | ||||||
|  |                     .setConstraints(myConstraints) | ||||||
|  |                     .addTag("selfoss-loading") | ||||||
|  |                     .build() | ||||||
|  |  | ||||||
|  |  | ||||||
|  |             WorkManager.getInstance().enqueueUniquePeriodicWork("selfoss-loading", ExistingPeriodicWorkPolicy.KEEP, backgroundWork) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun handleOfflineActions() { | ||||||
|  |         fun <T>doAndReportOnFail(call: Call<T>, action: ActionEntity) { | ||||||
|  |            call.enqueue(object: Callback<T> { | ||||||
|  |                override fun onResponse( | ||||||
|  |                    call: Call<T>, | ||||||
|  |                    response: Response<T> | ||||||
|  |                ) { | ||||||
|  |                    thread { | ||||||
|  |                        db.actionsDao().delete(action) | ||||||
|  |                    } | ||||||
|  |                } | ||||||
|  |  | ||||||
|  |                override fun onFailure(call: Call<T>, t: Throwable) { | ||||||
|  |                    ACRA.getErrorReporter().maybeHandleSilentException(t, this@HomeActivity) | ||||||
|  |                } | ||||||
|  |            }) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (this@HomeActivity.isNetworkAccessible(null)) { | ||||||
|  |             thread { | ||||||
|  |                 val actions = db.actionsDao().actions() | ||||||
|  |  | ||||||
|  |                 actions.forEach { action -> | ||||||
|  |                     when { | ||||||
|  |                         action.read -> doAndReportOnFail(api.markItem(action.articleId), action) | ||||||
|  |                         action.unread -> doAndReportOnFail(api.unmarkItem(action.articleId), action) | ||||||
|  |                         action.starred -> doAndReportOnFail(api.starrItem(action.articleId), action) | ||||||
|  |                         action.unstarred -> doAndReportOnFail(api.unstarrItem(action.articleId), action) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -21,6 +21,7 @@ import apps.amine.bou.readerforselfoss.themes.AppColors | |||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import apps.amine.bou.readerforselfoss.utils.isBaseUrlValid | import apps.amine.bou.readerforselfoss.utils.isBaseUrlValid | ||||||
| import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import com.mikepenz.aboutlibraries.Libs | import com.mikepenz.aboutlibraries.Libs | ||||||
| import com.mikepenz.aboutlibraries.LibsBuilder | import com.mikepenz.aboutlibraries.LibsBuilder | ||||||
| import kotlinx.android.synthetic.main.activity_login.* | import kotlinx.android.synthetic.main.activity_login.* | ||||||
| @@ -196,6 +197,8 @@ class LoginActivity : AppCompatActivity() { | |||||||
|                 isWithSelfSignedCert, |                 isWithSelfSignedCert, | ||||||
|                 isWithSelfSignedCert |                 isWithSelfSignedCert | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|  |             if (this@LoginActivity.isNetworkAccessible(this@LoginActivity.findViewById(R.id.loginForm))) { | ||||||
|                 api.login().enqueue(object : Callback<SuccessResponse> { |                 api.login().enqueue(object : Callback<SuccessResponse> { | ||||||
|                     private fun preferenceError(t: Throwable) { |                     private fun preferenceError(t: Throwable) { | ||||||
|                         editor.remove("url") |                         editor.remove("url") | ||||||
| @@ -235,6 +238,9 @@ class LoginActivity : AppCompatActivity() { | |||||||
|                         preferenceError(t) |                         preferenceError(t) | ||||||
|                     } |                     } | ||||||
|                 }) |                 }) | ||||||
|  |             } else { | ||||||
|  |                 showProgress(false) | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,8 +1,11 @@ | |||||||
| package apps.amine.bou.readerforselfoss | package apps.amine.bou.readerforselfoss | ||||||
|  |  | ||||||
|  | import android.app.NotificationChannel | ||||||
|  | import android.app.NotificationManager | ||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.graphics.drawable.Drawable | import android.graphics.drawable.Drawable | ||||||
| import android.net.Uri | import android.net.Uri | ||||||
|  | import android.os.Build | ||||||
| import android.preference.PreferenceManager | import android.preference.PreferenceManager | ||||||
| import androidx.multidex.MultiDexApplication | import androidx.multidex.MultiDexApplication | ||||||
| import android.widget.ImageView | import android.widget.ImageView | ||||||
| @@ -59,6 +62,18 @@ class MyApp : MultiDexApplication() { | |||||||
|         initTheme() |         initTheme() | ||||||
|  |  | ||||||
|         tryToHandleBug() |         tryToHandleBug() | ||||||
|  |  | ||||||
|  |         handleNotificationChannels() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun handleNotificationChannels() { | ||||||
|  |         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { | ||||||
|  |             val name = getString(R.string.notification_channel_sync) | ||||||
|  |             val importance = NotificationManager.IMPORTANCE_LOW | ||||||
|  |             val mChannel = NotificationChannel(Config.syncChannelId, name, importance) | ||||||
|  |             val notificationManager = getSystemService(NOTIFICATION_SERVICE) as NotificationManager | ||||||
|  |             notificationManager.createNotificationChannel(mChannel) | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun attachBaseContext(base: Context?) { |     override fun attachBaseContext(base: Context?) { | ||||||
|   | |||||||
| @@ -19,11 +19,14 @@ import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | |||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
| import apps.amine.bou.readerforselfoss.fragments.ArticleFragment | import apps.amine.bou.readerforselfoss.fragments.ArticleFragment | ||||||
| import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
| import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_1_2 | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_1_2 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_2_3 | ||||||
| import apps.amine.bou.readerforselfoss.themes.AppColors | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
| import apps.amine.bou.readerforselfoss.themes.Toppings | import apps.amine.bou.readerforselfoss.themes.Toppings | ||||||
| import apps.amine.bou.readerforselfoss.transformers.DepthPageTransformer | import apps.amine.bou.readerforselfoss.transformers.DepthPageTransformer | ||||||
| import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | ||||||
| import apps.amine.bou.readerforselfoss.utils.succeeded | import apps.amine.bou.readerforselfoss.utils.succeeded | ||||||
| import apps.amine.bou.readerforselfoss.utils.toggleStar | import apps.amine.bou.readerforselfoss.utils.toggleStar | ||||||
| @@ -70,7 +73,7 @@ class ReaderActivity : AppCompatActivity() { | |||||||
|         db = Room.databaseBuilder( |         db = Room.databaseBuilder( | ||||||
|             applicationContext, |             applicationContext, | ||||||
|             AppDatabase::class.java, "selfoss-database" |             AppDatabase::class.java, "selfoss-database" | ||||||
|         ).addMigrations(MIGRATION_1_2).build() |         ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).build() | ||||||
|  |  | ||||||
|         val scoop = Scoop.getInstance() |         val scoop = Scoop.getInstance() | ||||||
|         scoop.bind(this, Toppings.PRIMARY.value, toolBar) |         scoop.bind(this, Toppings.PRIMARY.value, toolBar) | ||||||
| @@ -103,7 +106,8 @@ class ReaderActivity : AppCompatActivity() { | |||||||
|  |  | ||||||
|         readItem(allItems[currentItem]) |         readItem(allItems[currentItem]) | ||||||
|  |  | ||||||
|         pager.adapter = ScreenSlidePagerAdapter(supportFragmentManager, AppColors(this@ReaderActivity)) |         pager.adapter = | ||||||
|  |                 ScreenSlidePagerAdapter(supportFragmentManager, AppColors(this@ReaderActivity)) | ||||||
|         pager.currentItem = currentItem |         pager.currentItem = currentItem | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -136,6 +140,7 @@ class ReaderActivity : AppCompatActivity() { | |||||||
|             thread { |             thread { | ||||||
|                 db.itemsDao().delete(item.toEntity()) |                 db.itemsDao().delete(item.toEntity()) | ||||||
|             } |             } | ||||||
|  |             if (this@ReaderActivity.isNetworkAccessible(this@ReaderActivity.findViewById(R.id.reader_activity_view))) { | ||||||
|                 api.markItem(item.id).enqueue( |                 api.markItem(item.id).enqueue( | ||||||
|                     object : Callback<SuccessResponse> { |                     object : Callback<SuccessResponse> { | ||||||
|                         override fun onResponse( |                         override fun onResponse( | ||||||
| @@ -164,11 +169,17 @@ class ReaderActivity : AppCompatActivity() { | |||||||
|                                 db.itemsDao().insertAllItems(item.toEntity()) |                                 db.itemsDao().insertAllItems(item.toEntity()) | ||||||
|                             } |                             } | ||||||
|                             if (debugReadingItems) { |                             if (debugReadingItems) { | ||||||
|                             ACRA.getErrorReporter().maybeHandleSilentException(t, this@ReaderActivity) |                                 ACRA.getErrorReporter() | ||||||
|  |                                     .maybeHandleSilentException(t, this@ReaderActivity) | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 ) |                 ) | ||||||
|  |             } else { | ||||||
|  |                 thread { | ||||||
|  |                     db.actionsDao().insertAllActions(ActionEntity(item.id, true, false, false, false)) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -191,7 +202,6 @@ class ReaderActivity : AppCompatActivity() { | |||||||
|     private inner class ScreenSlidePagerAdapter(fm: FragmentManager, val appColors: AppColors) : |     private inner class ScreenSlidePagerAdapter(fm: FragmentManager, val appColors: AppColors) : | ||||||
|         FragmentStatePagerAdapter(fm) { |         FragmentStatePagerAdapter(fm) { | ||||||
|  |  | ||||||
|  |  | ||||||
|         override fun getCount(): Int { |         override fun getCount(): Int { | ||||||
|             return allItems.size |             return allItems.size | ||||||
|         } |         } | ||||||
| @@ -203,7 +213,12 @@ class ReaderActivity : AppCompatActivity() { | |||||||
|         override fun startUpdate(container: ViewGroup) { |         override fun startUpdate(container: ViewGroup) { | ||||||
|             super.startUpdate(container) |             super.startUpdate(container) | ||||||
|  |  | ||||||
|             container.background = ColorDrawable(ContextCompat.getColor(this@ReaderActivity, appColors.colorBackground)) |             container.background = ColorDrawable( | ||||||
|  |                 ContextCompat.getColor( | ||||||
|  |                     this@ReaderActivity, | ||||||
|  |                     appColors.colorBackground | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -222,21 +237,33 @@ class ReaderActivity : AppCompatActivity() { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onOptionsItemSelected(item: MenuItem): Boolean { |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|  |         fun afterSave() { | ||||||
|  |             allItems[pager.currentItem] = | ||||||
|  |                     allItems[pager.currentItem].toggleStar() | ||||||
|  |             notifyAdapter() | ||||||
|  |             canRemoveFromFavorite() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         fun afterUnsave() { | ||||||
|  |             allItems[pager.currentItem] = allItems[pager.currentItem].toggleStar() | ||||||
|  |             notifyAdapter() | ||||||
|  |             canFavorite() | ||||||
|  |         } | ||||||
|  |  | ||||||
|         when (item.itemId) { |         when (item.itemId) { | ||||||
|             android.R.id.home -> { |             android.R.id.home -> { | ||||||
|                 onBackPressed() |                 onBackPressed() | ||||||
|                 return true |                 return true | ||||||
|             } |             } | ||||||
|             R.id.save -> { |             R.id.save -> { | ||||||
|  |                 if (this@ReaderActivity.isNetworkAccessible(null)) { | ||||||
|                     api.starrItem(allItems[pager.currentItem].id) |                     api.starrItem(allItems[pager.currentItem].id) | ||||||
|                         .enqueue(object : Callback<SuccessResponse> { |                         .enqueue(object : Callback<SuccessResponse> { | ||||||
|                             override fun onResponse( |                             override fun onResponse( | ||||||
|                                 call: Call<SuccessResponse>, |                                 call: Call<SuccessResponse>, | ||||||
|                                 response: Response<SuccessResponse> |                                 response: Response<SuccessResponse> | ||||||
|                             ) { |                             ) { | ||||||
|                             allItems[pager.currentItem] = allItems[pager.currentItem].toggleStar() |                                 afterSave() | ||||||
|                             notifyAdapter() |  | ||||||
|                             canRemoveFromFavorite() |  | ||||||
|                             } |                             } | ||||||
|  |  | ||||||
|                             override fun onFailure( |                             override fun onFailure( | ||||||
| @@ -250,17 +277,22 @@ class ReaderActivity : AppCompatActivity() { | |||||||
|                                 ).show() |                                 ).show() | ||||||
|                             } |                             } | ||||||
|                         }) |                         }) | ||||||
|  |                 } else { | ||||||
|  |                     thread { | ||||||
|  |                         db.actionsDao().insertAllActions(ActionEntity(allItems[pager.currentItem].id, false, false, true, false)) | ||||||
|  |                         afterSave() | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             R.id.unsave -> { |             R.id.unsave -> { | ||||||
|  |                 if (this@ReaderActivity.isNetworkAccessible(null)) { | ||||||
|                     api.unstarrItem(allItems[pager.currentItem].id) |                     api.unstarrItem(allItems[pager.currentItem].id) | ||||||
|                         .enqueue(object : Callback<SuccessResponse> { |                         .enqueue(object : Callback<SuccessResponse> { | ||||||
|                             override fun onResponse( |                             override fun onResponse( | ||||||
|                                 call: Call<SuccessResponse>, |                                 call: Call<SuccessResponse>, | ||||||
|                                 response: Response<SuccessResponse> |                                 response: Response<SuccessResponse> | ||||||
|                             ) { |                             ) { | ||||||
|                             allItems[pager.currentItem] = allItems[pager.currentItem].toggleStar() |                                 afterUnsave() | ||||||
|                             notifyAdapter() |  | ||||||
|                             canFavorite() |  | ||||||
|                             } |                             } | ||||||
|  |  | ||||||
|                             override fun onFailure( |                             override fun onFailure( | ||||||
| @@ -274,6 +306,12 @@ class ReaderActivity : AppCompatActivity() { | |||||||
|                                 ).show() |                                 ).show() | ||||||
|                             } |                             } | ||||||
|                         }) |                         }) | ||||||
|  |                 } else { | ||||||
|  |                     thread { | ||||||
|  |                         db.actionsDao().insertAllActions(ActionEntity(allItems[pager.currentItem].id, false, false, false, true)) | ||||||
|  |                         afterUnsave() | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return super.onOptionsItemSelected(item) |         return super.onOptionsItemSelected(item) | ||||||
|   | |||||||
| @@ -13,6 +13,7 @@ import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | |||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Source | import apps.amine.bou.readerforselfoss.api.selfoss.Source | ||||||
| import apps.amine.bou.readerforselfoss.themes.AppColors | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
| import apps.amine.bou.readerforselfoss.themes.Toppings | import apps.amine.bou.readerforselfoss.themes.Toppings | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import com.ftinc.scoop.Scoop | import com.ftinc.scoop.Scoop | ||||||
| import kotlinx.android.synthetic.main.activity_sources.* | import kotlinx.android.synthetic.main.activity_sources.* | ||||||
| import retrofit2.Call | import retrofit2.Call | ||||||
| @@ -66,6 +67,7 @@ class SourcesActivity : AppCompatActivity() { | |||||||
|         recyclerView.setHasFixedSize(true) |         recyclerView.setHasFixedSize(true) | ||||||
|         recyclerView.layoutManager = mLayoutManager |         recyclerView.layoutManager = mLayoutManager | ||||||
|  |  | ||||||
|  |         if (this@SourcesActivity.isNetworkAccessible(this@SourcesActivity.findViewById(R.id.recyclerView))) { | ||||||
|             api.sources.enqueue(object : Callback<List<Source>> { |             api.sources.enqueue(object : Callback<List<Source>> { | ||||||
|                 override fun onResponse( |                 override fun onResponse( | ||||||
|                     call: Call<List<Source>>, |                     call: Call<List<Source>>, | ||||||
| @@ -94,6 +96,7 @@ class SourcesActivity : AppCompatActivity() { | |||||||
|                     ).show() |                     ).show() | ||||||
|                 } |                 } | ||||||
|             }) |             }) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         fab.setOnClickListener { |         fab.setOnClickListener { | ||||||
|             startActivity(Intent(this@SourcesActivity, AddSourceActivity::class.java)) |             startActivity(Intent(this@SourcesActivity, AddSourceActivity::class.java)) | ||||||
|   | |||||||
| @@ -15,11 +15,14 @@ import apps.amine.bou.readerforselfoss.api.selfoss.Item | |||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
| import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
| import apps.amine.bou.readerforselfoss.themes.AppColors | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.LinkOnTouchListener | ||||||
| import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | ||||||
| import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | ||||||
| import apps.amine.bou.readerforselfoss.utils.glide.bitmapCenterCrop | import apps.amine.bou.readerforselfoss.utils.glide.bitmapCenterCrop | ||||||
| import apps.amine.bou.readerforselfoss.utils.glide.circularBitmapDrawable | import apps.amine.bou.readerforselfoss.utils.glide.circularBitmapDrawable | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.openInBrowserAsNewTask | import apps.amine.bou.readerforselfoss.utils.openInBrowserAsNewTask | ||||||
| import apps.amine.bou.readerforselfoss.utils.openItemUrl | import apps.amine.bou.readerforselfoss.utils.openItemUrl | ||||||
| import apps.amine.bou.readerforselfoss.utils.shareLink | import apps.amine.bou.readerforselfoss.utils.shareLink | ||||||
| @@ -34,6 +37,7 @@ import kotlinx.android.synthetic.main.card_item.view.* | |||||||
| import retrofit2.Call | import retrofit2.Call | ||||||
| import retrofit2.Callback | import retrofit2.Callback | ||||||
| import retrofit2.Response | import retrofit2.Response | ||||||
|  | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
| class ItemCardAdapter( | class ItemCardAdapter( | ||||||
|     override val app: Activity, |     override val app: Activity, | ||||||
| @@ -65,6 +69,7 @@ class ItemCardAdapter( | |||||||
|  |  | ||||||
|         holder.mView.favButton.isLiked = itm.starred |         holder.mView.favButton.isLiked = itm.starred | ||||||
|         holder.mView.title.text = Html.fromHtml(itm.title) |         holder.mView.title.text = Html.fromHtml(itm.title) | ||||||
|  |         holder.mView.title.setOnTouchListener(LinkOnTouchListener()) | ||||||
|  |  | ||||||
|         holder.mView.title.setLinkTextColor(appColors.colorAccent) |         holder.mView.title.setLinkTextColor(appColors.colorAccent) | ||||||
|  |  | ||||||
| @@ -116,6 +121,7 @@ class ItemCardAdapter( | |||||||
|             mView.favButton.setOnLikeListener(object : OnLikeListener { |             mView.favButton.setOnLikeListener(object : OnLikeListener { | ||||||
|                 override fun liked(likeButton: LikeButton) { |                 override fun liked(likeButton: LikeButton) { | ||||||
|                     val (id) = items[adapterPosition] |                     val (id) = items[adapterPosition] | ||||||
|  |                     if (c.isNetworkAccessible(null)) { | ||||||
|                         api.starrItem(id).enqueue(object : Callback<SuccessResponse> { |                         api.starrItem(id).enqueue(object : Callback<SuccessResponse> { | ||||||
|                             override fun onResponse( |                             override fun onResponse( | ||||||
|                                 call: Call<SuccessResponse>, |                                 call: Call<SuccessResponse>, | ||||||
| @@ -135,10 +141,16 @@ class ItemCardAdapter( | |||||||
|                                 ).show() |                                 ).show() | ||||||
|                             } |                             } | ||||||
|                         }) |                         }) | ||||||
|  |                     } else { | ||||||
|  |                         thread { | ||||||
|  |                             db.actionsDao().insertAllActions(ActionEntity(id, false, false, true, false)) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 override fun unLiked(likeButton: LikeButton) { |                 override fun unLiked(likeButton: LikeButton) { | ||||||
|                     val (id) = items[adapterPosition] |                     val (id) = items[adapterPosition] | ||||||
|  |                     if (c.isNetworkAccessible(null)) { | ||||||
|                         api.unstarrItem(id).enqueue(object : Callback<SuccessResponse> { |                         api.unstarrItem(id).enqueue(object : Callback<SuccessResponse> { | ||||||
|                             override fun onResponse( |                             override fun onResponse( | ||||||
|                                 call: Call<SuccessResponse>, |                                 call: Call<SuccessResponse>, | ||||||
| @@ -158,6 +170,11 @@ class ItemCardAdapter( | |||||||
|                                 ).show() |                                 ).show() | ||||||
|                             } |                             } | ||||||
|                         }) |                         }) | ||||||
|  |                     } else { | ||||||
|  |                         thread { | ||||||
|  |                             db.actionsDao().insertAllActions(ActionEntity(id, false, false, false, true)) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             }) |             }) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,10 +5,14 @@ import android.content.Context | |||||||
| import androidx.constraintlayout.widget.ConstraintLayout | import androidx.constraintlayout.widget.ConstraintLayout | ||||||
| import androidx.recyclerview.widget.RecyclerView | import androidx.recyclerview.widget.RecyclerView | ||||||
| import android.text.Html | import android.text.Html | ||||||
|  | import android.text.Spannable | ||||||
|  | import android.text.style.ClickableSpan | ||||||
| import android.util.TypedValue | import android.util.TypedValue | ||||||
| import android.view.LayoutInflater | import android.view.LayoutInflater | ||||||
|  | import android.view.MotionEvent | ||||||
| import android.view.View | import android.view.View | ||||||
| import android.view.ViewGroup | import android.view.ViewGroup | ||||||
|  | import android.widget.TextView | ||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import apps.amine.bou.readerforselfoss.R | import apps.amine.bou.readerforselfoss.R | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Item | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
| @@ -16,6 +20,7 @@ import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | |||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
| import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
| import apps.amine.bou.readerforselfoss.themes.AppColors | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.LinkOnTouchListener | ||||||
| import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | ||||||
| import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | ||||||
| import apps.amine.bou.readerforselfoss.utils.glide.bitmapCenterCrop | import apps.amine.bou.readerforselfoss.utils.glide.bitmapCenterCrop | ||||||
| @@ -67,6 +72,8 @@ class ItemListAdapter( | |||||||
|  |  | ||||||
|         holder.mView.title.text = Html.fromHtml(itm.title) |         holder.mView.title.text = Html.fromHtml(itm.title) | ||||||
|  |  | ||||||
|  |         holder.mView.title.setOnTouchListener(LinkOnTouchListener()) | ||||||
|  |  | ||||||
|         holder.mView.title.setLinkTextColor(appColors.colorAccent) |         holder.mView.title.setLinkTextColor(appColors.colorAccent) | ||||||
|  |  | ||||||
|         holder.mView.sourceTitleAndDate.text = itm.sourceAndDateText() |         holder.mView.sourceTitleAndDate.text = itm.sourceAndDateText() | ||||||
|   | |||||||
| @@ -11,8 +11,10 @@ import apps.amine.bou.readerforselfoss.api.selfoss.Item | |||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
| import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
| import apps.amine.bou.readerforselfoss.themes.AppColors | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
| import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | ||||||
| import apps.amine.bou.readerforselfoss.utils.succeeded | import apps.amine.bou.readerforselfoss.utils.succeeded | ||||||
| import org.acra.ACRA | import org.acra.ACRA | ||||||
| @@ -52,6 +54,7 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|                 notifyItemInserted(position) |                 notifyItemInserted(position) | ||||||
|                 updateItems(items) |                 updateItems(items) | ||||||
|  |  | ||||||
|  |                 if (app.isNetworkAccessible(null)) { | ||||||
|                     api.unmarkItem(i.id).enqueue(object : Callback<SuccessResponse> { |                     api.unmarkItem(i.id).enqueue(object : Callback<SuccessResponse> { | ||||||
|                         override fun onResponse( |                         override fun onResponse( | ||||||
|                             call: Call<SuccessResponse>, |                             call: Call<SuccessResponse>, | ||||||
| @@ -69,6 +72,11 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|                             doUnmark(i, position) |                             doUnmark(i, position) | ||||||
|                         } |                         } | ||||||
|                     }) |                     }) | ||||||
|  |                 } else { | ||||||
|  |                     thread { | ||||||
|  |                         db.actionsDao().deleteReadActionForArticle(i.id) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|         val view = s.view |         val view = s.view | ||||||
| @@ -78,20 +86,16 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun removeItemAtIndex(position: Int) { |     fun removeItemAtIndex(position: Int) { | ||||||
|  |  | ||||||
|         val i = items[position] |         val i = items[position] | ||||||
|  |  | ||||||
|         items.remove(i) |         items.remove(i) | ||||||
|         notifyItemRemoved(position) |         notifyItemRemoved(position) | ||||||
|         updateItems(items) |         updateItems(items) | ||||||
|  |  | ||||||
|         // TODO: Handle network status. |  | ||||||
|         // IF offline, delete from cached articles, and add to some table that will replay the calls on network activation. |  | ||||||
|  |  | ||||||
|         thread { |         thread { | ||||||
|             db.itemsDao().delete(i.toEntity()) |             db.itemsDao().delete(i.toEntity()) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (app.isNetworkAccessible(null)) { | ||||||
|             api.markItem(i.id).enqueue(object : Callback<SuccessResponse> { |             api.markItem(i.id).enqueue(object : Callback<SuccessResponse> { | ||||||
|                 override fun onResponse( |                 override fun onResponse( | ||||||
|                     call: Call<SuccessResponse>, |                     call: Call<SuccessResponse>, | ||||||
| @@ -132,6 +136,12 @@ abstract class ItemsAdapter<VH : RecyclerView.ViewHolder?> : RecyclerView.Adapte | |||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             }) |             }) | ||||||
|  |         } else { | ||||||
|  |             thread { | ||||||
|  |                 db.actionsDao().insertAllActions(ActionEntity(i.id, true, false, false, false)) | ||||||
|  |                 doUnmark(i, position) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun addItemAtIndex(item: Item, position: Int) { |     fun addItemAtIndex(item: Item, position: Int) { | ||||||
|   | |||||||
| @@ -13,6 +13,7 @@ import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | |||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Source | import apps.amine.bou.readerforselfoss.api.selfoss.Source | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
| import apps.amine.bou.readerforselfoss.utils.glide.circularBitmapDrawable | import apps.amine.bou.readerforselfoss.utils.glide.circularBitmapDrawable | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.toTextDrawableString | import apps.amine.bou.readerforselfoss.utils.toTextDrawableString | ||||||
| import com.amulyakhare.textdrawable.TextDrawable | import com.amulyakhare.textdrawable.TextDrawable | ||||||
| import com.amulyakhare.textdrawable.util.ColorGenerator | import com.amulyakhare.textdrawable.util.ColorGenerator | ||||||
| @@ -70,6 +71,7 @@ class SourcesListAdapter( | |||||||
|             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) |             val deleteBtn: Button = mView.findViewById(R.id.deleteBtn) | ||||||
|  |  | ||||||
|             deleteBtn.setOnClickListener { |             deleteBtn.setOnClickListener { | ||||||
|  |                 if (c.isNetworkAccessible(null)) { | ||||||
|                     val (id) = items[adapterPosition] |                     val (id) = items[adapterPosition] | ||||||
|                     api.deleteSource(id).enqueue(object : Callback<SuccessResponse> { |                     api.deleteSource(id).enqueue(object : Callback<SuccessResponse> { | ||||||
|                         override fun onResponse( |                         override fun onResponse( | ||||||
| @@ -101,3 +103,4 @@ class SourcesListAdapter( | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ import java.util.concurrent.ConcurrentHashMap | |||||||
|  |  | ||||||
| class SelfossApi( | class SelfossApi( | ||||||
|     c: Context, |     c: Context, | ||||||
|     callingActivity: Activity, |     callingActivity: Activity?, | ||||||
|     isWithSelfSignedCert: Boolean, |     isWithSelfSignedCert: Boolean, | ||||||
|     shouldLog: Boolean |     shouldLog: Boolean | ||||||
| ) { | ) { | ||||||
| @@ -66,6 +66,7 @@ class SelfossApi( | |||||||
|         val gson = |         val gson = | ||||||
|             GsonBuilder() |             GsonBuilder() | ||||||
|                 .registerTypeAdapter(Boolean::class.javaPrimitiveType, BooleanTypeAdapter()) |                 .registerTypeAdapter(Boolean::class.javaPrimitiveType, BooleanTypeAdapter()) | ||||||
|  |                 .registerTypeAdapter(SelfossTagType::class.java, SelfossTagTypeTypeAdapter()) | ||||||
|                 .setLenient() |                 .setLenient() | ||||||
|                 .create() |                 .create() | ||||||
|  |  | ||||||
| @@ -91,9 +92,11 @@ class SelfossApi( | |||||||
|                     .build() |                     .build() | ||||||
|             service = retrofit.create(SelfossService::class.java) |             service = retrofit.create(SelfossService::class.java) | ||||||
|         } catch (e: IllegalArgumentException) { |         } catch (e: IllegalArgumentException) { | ||||||
|  |             if (callingActivity != null) { | ||||||
|                 Config.logoutAndRedirect(c, callingActivity, config.settings.edit(), baseUrlFail = true) |                 Config.logoutAndRedirect(c, callingActivity, config.settings.edit(), baseUrlFail = true) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     fun login(): Call<SuccessResponse> = |     fun login(): Call<SuccessResponse> = | ||||||
|         service.loginToSelfoss(config.userLogin, config.userPassword) |         service.loginToSelfoss(config.userLogin, config.userPassword) | ||||||
|   | |||||||
| @@ -45,7 +45,7 @@ data class Spout( | |||||||
| data class Source( | data class Source( | ||||||
|     @SerializedName("id") val id: String, |     @SerializedName("id") val id: String, | ||||||
|     @SerializedName("title") val title: String, |     @SerializedName("title") val title: String, | ||||||
|     @SerializedName("tags") val tags: String, |     @SerializedName("tags") val tags: SelfossTagType, | ||||||
|     @SerializedName("spout") val spout: String, |     @SerializedName("spout") val spout: String, | ||||||
|     @SerializedName("error") val error: String, |     @SerializedName("error") val error: String, | ||||||
|     @SerializedName("icon") val icon: String |     @SerializedName("icon") val icon: String | ||||||
| @@ -71,7 +71,7 @@ data class Item( | |||||||
|     @SerializedName("icon") val icon: String, |     @SerializedName("icon") val icon: String, | ||||||
|     @SerializedName("link") val link: String, |     @SerializedName("link") val link: String, | ||||||
|     @SerializedName("sourcetitle") val sourcetitle: String, |     @SerializedName("sourcetitle") val sourcetitle: String, | ||||||
|     @SerializedName("tags") val tags: String |     @SerializedName("tags") val tags: SelfossTagType | ||||||
| ) : Parcelable { | ) : Parcelable { | ||||||
|  |  | ||||||
|     var config: Config? = null |     var config: Config? = null | ||||||
| @@ -94,7 +94,7 @@ data class Item( | |||||||
|         icon = source.readString(), |         icon = source.readString(), | ||||||
|         link = source.readString(), |         link = source.readString(), | ||||||
|         sourcetitle = source.readString(), |         sourcetitle = source.readString(), | ||||||
|         tags = source.readString() |         tags = source.readParcelable(ClassLoader.getSystemClassLoader()) | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     override fun describeContents() = 0 |     override fun describeContents() = 0 | ||||||
| @@ -110,7 +110,7 @@ data class Item( | |||||||
|         dest.writeString(icon) |         dest.writeString(icon) | ||||||
|         dest.writeString(link) |         dest.writeString(link) | ||||||
|         dest.writeString(sourcetitle) |         dest.writeString(sourcetitle) | ||||||
|         dest.writeString(tags) |         dest.writeParcelable(tags, flags) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun getIcon(app: Context): String { |     fun getIcon(app: Context): String { | ||||||
| @@ -154,3 +154,26 @@ data class Item( | |||||||
|         return stringUrl |         return stringUrl | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | data class SelfossTagType(val tags: String) : Parcelable { | ||||||
|  |  | ||||||
|  |     companion object { | ||||||
|  |         @JvmField val CREATOR: Parcelable.Creator<SelfossTagType> = | ||||||
|  |             object : Parcelable.Creator<SelfossTagType> { | ||||||
|  |                 override fun createFromParcel(source: Parcel): SelfossTagType = | ||||||
|  |                     SelfossTagType(source) | ||||||
|  |  | ||||||
|  |                 override fun newArray(size: Int): Array<SelfossTagType?> = arrayOfNulls(size) | ||||||
|  |             } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     constructor(source: Parcel) : this( | ||||||
|  |         tags = source.readString() | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     override fun describeContents() = 0 | ||||||
|  |  | ||||||
|  |     override fun writeToParcel(dest: Parcel, flags: Int) { | ||||||
|  |         dest.writeString(tags) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,22 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.api.selfoss | ||||||
|  |  | ||||||
|  | import com.google.gson.JsonDeserializationContext | ||||||
|  | import com.google.gson.JsonDeserializer | ||||||
|  | import com.google.gson.JsonElement | ||||||
|  | import com.google.gson.JsonParseException | ||||||
|  | import java.lang.reflect.Type | ||||||
|  |  | ||||||
|  | internal class SelfossTagTypeTypeAdapter : JsonDeserializer<SelfossTagType> { | ||||||
|  |  | ||||||
|  |     @Throws(JsonParseException::class) | ||||||
|  |     override fun deserialize( | ||||||
|  |         json: JsonElement, | ||||||
|  |         typeOfT: Type, | ||||||
|  |         context: JsonDeserializationContext | ||||||
|  |     ): SelfossTagType? = | ||||||
|  |         if (json.isJsonArray) { | ||||||
|  |             SelfossTagType(json.asJsonArray.joinToString(",") { it.toString() }) | ||||||
|  |         } else { | ||||||
|  |             SelfossTagType(json.toString()) | ||||||
|  |         } | ||||||
|  | } | ||||||
| @@ -0,0 +1,126 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.background | ||||||
|  |  | ||||||
|  | import android.app.NotificationManager | ||||||
|  | import android.content.Context | ||||||
|  | import android.os.Handler | ||||||
|  | import android.preference.PreferenceManager | ||||||
|  | import androidx.core.app.NotificationCompat | ||||||
|  | import androidx.core.app.NotificationCompat.PRIORITY_LOW | ||||||
|  | import androidx.room.Room | ||||||
|  | import androidx.work.Worker | ||||||
|  | import androidx.work.WorkerParameters | ||||||
|  | import apps.amine.bou.readerforselfoss.R | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_1_2 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_2_3 | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | ||||||
|  | import org.acra.ACRA | ||||||
|  | import retrofit2.Call | ||||||
|  | import retrofit2.Callback | ||||||
|  | import retrofit2.Response | ||||||
|  | import java.util.* | ||||||
|  | import kotlin.concurrent.schedule | ||||||
|  | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
|  | class LoadingWorker(val context: Context, params: WorkerParameters) : Worker(context, params) { | ||||||
|  |     lateinit var db: AppDatabase | ||||||
|  |  | ||||||
|  |     override fun doWork(): Result { | ||||||
|  |         if (context.isNetworkAccessible(null)) { | ||||||
|  |  | ||||||
|  |             val notificationManager = | ||||||
|  |                 applicationContext.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager | ||||||
|  |  | ||||||
|  |             val notification = NotificationCompat.Builder(applicationContext, Config.syncChannelId) | ||||||
|  |                 .setContentTitle(context.getString(R.string.loading_notification_title)) | ||||||
|  |                 .setContentText(context.getString(R.string.loading_notification_text)) | ||||||
|  |                 .setOngoing(true) | ||||||
|  |                 .setPriority(PRIORITY_LOW) | ||||||
|  |                 .setChannelId(Config.syncChannelId) | ||||||
|  |                 .setSmallIcon(R.drawable.ic_cloud_download) | ||||||
|  |  | ||||||
|  |             notificationManager.notify(1, notification.build()) | ||||||
|  |  | ||||||
|  |             val settings = | ||||||
|  |                 this.context.getSharedPreferences(Config.settingsName, Context.MODE_PRIVATE) | ||||||
|  |             val sharedPref = PreferenceManager.getDefaultSharedPreferences(this.context) | ||||||
|  |             val shouldLogEverything = sharedPref.getBoolean("should_log_everything", false) | ||||||
|  |  | ||||||
|  |             db = Room.databaseBuilder( | ||||||
|  |                 applicationContext, | ||||||
|  |                 AppDatabase::class.java, "selfoss-database" | ||||||
|  |             ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).build() | ||||||
|  |  | ||||||
|  |             val api = SelfossApi( | ||||||
|  |                 this.context, | ||||||
|  |                 null, | ||||||
|  |                 settings.getBoolean("isSelfSignedCert", false), | ||||||
|  |                 shouldLogEverything | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |             api.allItems().enqueue(object : Callback<List<Item>> { | ||||||
|  |                 override fun onFailure(call: Call<List<Item>>, t: Throwable) { | ||||||
|  |                     Timer("", false).schedule(4000) { | ||||||
|  |                         notificationManager.cancel(1) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 override fun onResponse( | ||||||
|  |                     call: Call<List<Item>>, | ||||||
|  |                     response: Response<List<Item>> | ||||||
|  |                 ) { | ||||||
|  |                     thread { | ||||||
|  |                         if (response.body() != null) { | ||||||
|  |                             val apiItems = (response.body() as ArrayList<Item>) | ||||||
|  |                             db.itemsDao().deleteAllItems() | ||||||
|  |                             db.itemsDao() | ||||||
|  |                                 .insertAllItems(*(apiItems.map { it.toEntity() }).toTypedArray()) | ||||||
|  |                         } | ||||||
|  |                         Timer("", false).schedule(4000) { | ||||||
|  |                             notificationManager.cancel(1) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             }) | ||||||
|  |             thread { | ||||||
|  |                 val actions = db.actionsDao().actions() | ||||||
|  |  | ||||||
|  |                 actions.forEach { action -> | ||||||
|  |                     when { | ||||||
|  |                         action.read -> doAndReportOnFail(api.markItem(action.articleId), action) | ||||||
|  |                         action.unread -> doAndReportOnFail(api.unmarkItem(action.articleId), action) | ||||||
|  |                         action.starred -> doAndReportOnFail(api.starrItem(action.articleId), action) | ||||||
|  |                         action.unstarred -> doAndReportOnFail( | ||||||
|  |                             api.unstarrItem(action.articleId), | ||||||
|  |                             action | ||||||
|  |                         ) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return Result.SUCCESS | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private fun <T> doAndReportOnFail(call: Call<T>, action: ActionEntity) { | ||||||
|  |         call.enqueue(object : Callback<T> { | ||||||
|  |             override fun onResponse( | ||||||
|  |                 call: Call<T>, | ||||||
|  |                 response: Response<T> | ||||||
|  |             ) { | ||||||
|  |                 thread { | ||||||
|  |                     db.actionsDao().delete(action) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             override fun onFailure(call: Call<T>, t: Throwable) { | ||||||
|  |                 ACRA.getErrorReporter().maybeHandleSilentException(t, context) | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -17,18 +17,24 @@ import android.view.MenuItem | |||||||
| import android.view.View | import android.view.View | ||||||
| import android.view.ViewGroup | import android.view.ViewGroup | ||||||
| import android.webkit.WebSettings | import android.webkit.WebSettings | ||||||
|  | import androidx.room.Room | ||||||
| import apps.amine.bou.readerforselfoss.R | import apps.amine.bou.readerforselfoss.R | ||||||
| import apps.amine.bou.readerforselfoss.api.mercury.MercuryApi | import apps.amine.bou.readerforselfoss.api.mercury.MercuryApi | ||||||
| import apps.amine.bou.readerforselfoss.api.mercury.ParsedContent | import apps.amine.bou.readerforselfoss.api.mercury.ParsedContent | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Item | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossApi | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | import apps.amine.bou.readerforselfoss.api.selfoss.SuccessResponse | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_1_2 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_2_3 | ||||||
| import apps.amine.bou.readerforselfoss.themes.AppColors | import apps.amine.bou.readerforselfoss.themes.AppColors | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | import apps.amine.bou.readerforselfoss.utils.buildCustomTabsIntent | ||||||
| import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | import apps.amine.bou.readerforselfoss.utils.customtabs.CustomTabActivityHelper | ||||||
| import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | ||||||
| import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | import apps.amine.bou.readerforselfoss.utils.maybeHandleSilentException | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.openItemUrl | import apps.amine.bou.readerforselfoss.utils.openItemUrl | ||||||
| import apps.amine.bou.readerforselfoss.utils.shareLink | import apps.amine.bou.readerforselfoss.utils.shareLink | ||||||
| import apps.amine.bou.readerforselfoss.utils.sourceAndDateText | import apps.amine.bou.readerforselfoss.utils.sourceAndDateText | ||||||
| @@ -44,6 +50,7 @@ import retrofit2.Callback | |||||||
| import retrofit2.Response | import retrofit2.Response | ||||||
| import java.net.MalformedURLException | import java.net.MalformedURLException | ||||||
| import java.net.URL | import java.net.URL | ||||||
|  | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
| class ArticleFragment : Fragment() { | class ArticleFragment : Fragment() { | ||||||
|     private lateinit var pageNumber: Number |     private lateinit var pageNumber: Number | ||||||
| @@ -58,6 +65,7 @@ class ArticleFragment : Fragment() { | |||||||
|     private lateinit var editor: SharedPreferences.Editor |     private lateinit var editor: SharedPreferences.Editor | ||||||
|     private lateinit var fab: FloatingActionButton |     private lateinit var fab: FloatingActionButton | ||||||
|     private lateinit var appColors: AppColors |     private lateinit var appColors: AppColors | ||||||
|  |     private lateinit var db: AppDatabase | ||||||
|  |  | ||||||
|     override fun onStop() { |     override fun onStop() { | ||||||
|         super.onStop() |         super.onStop() | ||||||
| @@ -71,6 +79,11 @@ class ArticleFragment : Fragment() { | |||||||
|  |  | ||||||
|         pageNumber = arguments!!.getInt(ARG_POSITION) |         pageNumber = arguments!!.getInt(ARG_POSITION) | ||||||
|         allItems = arguments!!.getParcelableArrayList(ARG_ITEMS) |         allItems = arguments!!.getParcelableArrayList(ARG_ITEMS) | ||||||
|  |  | ||||||
|  |         db = Room.databaseBuilder( | ||||||
|  |             context!!, | ||||||
|  |             AppDatabase::class.java, "selfoss-database" | ||||||
|  |         ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).build() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private lateinit var rootView: ViewGroup |     private lateinit var rootView: ViewGroup | ||||||
| @@ -135,7 +148,8 @@ class ArticleFragment : Fragment() { | |||||||
|                             false, |                             false, | ||||||
|                             activity!! |                             activity!! | ||||||
|                         ) |                         ) | ||||||
|                         R.id.unread_action -> api.unmarkItem(allItems[pageNumber.toInt()].id).enqueue( |                         R.id.unread_action -> if ((context != null && context!!.isNetworkAccessible(null)) || context == null) { | ||||||
|  |                             api.unmarkItem(allItems[pageNumber.toInt()].id).enqueue( | ||||||
|                                 object : Callback<SuccessResponse> { |                                 object : Callback<SuccessResponse> { | ||||||
|                                     override fun onResponse( |                                     override fun onResponse( | ||||||
|                                         call: Call<SuccessResponse>, |                                         call: Call<SuccessResponse>, | ||||||
| @@ -164,6 +178,11 @@ class ArticleFragment : Fragment() { | |||||||
|                                     } |                                     } | ||||||
|                                 } |                                 } | ||||||
|                             ) |                             ) | ||||||
|  |                         } else { | ||||||
|  |                             thread { | ||||||
|  |                                 db.actionsDao().insertAllActions(ActionEntity(allItems[pageNumber.toInt()].id, false, true, false, false)) | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|                         else -> Unit |                         else -> Unit | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
| @@ -212,6 +231,7 @@ class ArticleFragment : Fragment() { | |||||||
|         customTabsIntent: CustomTabsIntent, |         customTabsIntent: CustomTabsIntent, | ||||||
|         prefs: SharedPreferences |         prefs: SharedPreferences | ||||||
|     ) { |     ) { | ||||||
|  |         if ((context != null && context!!.isNetworkAccessible(null)) || context == null) { | ||||||
|             rootView.progressBar.visibility = View.VISIBLE |             rootView.progressBar.visibility = View.VISIBLE | ||||||
|             val parser = MercuryApi( |             val parser = MercuryApi( | ||||||
|                 prefs.getBoolean("should_log_everything", false) |                 prefs.getBoolean("should_log_everything", false) | ||||||
| @@ -303,6 +323,7 @@ class ArticleFragment : Fragment() { | |||||||
|                 } |                 } | ||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private fun htmlToWebview(c: String, prefs: SharedPreferences) { |     private fun htmlToWebview(c: String, prefs: SharedPreferences) { | ||||||
|         val stringColor = String.format("#%06X", 0xFFFFFF and appColors.colorAccent) |         val stringColor = String.format("#%06X", 0xFFFFFF and appColors.colorAccent) | ||||||
|   | |||||||
| @@ -0,0 +1,23 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.dao | ||||||
|  |  | ||||||
|  | import androidx.room.Dao | ||||||
|  | import androidx.room.Delete | ||||||
|  | import androidx.room.Insert | ||||||
|  | import androidx.room.OnConflictStrategy | ||||||
|  | import androidx.room.Query | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
|  |  | ||||||
|  | @Dao | ||||||
|  | interface ActionsDao { | ||||||
|  |     @Query("SELECT * FROM actions order by id asc") | ||||||
|  |     fun actions(): List<ActionEntity> | ||||||
|  |  | ||||||
|  |     @Insert(onConflict = OnConflictStrategy.REPLACE) | ||||||
|  |     fun insertAllActions(vararg actions: ActionEntity) | ||||||
|  |  | ||||||
|  |     @Query("DELETE FROM actions WHERE articleid = :article_id AND read = 1") | ||||||
|  |     fun deleteReadActionForArticle(article_id: String) | ||||||
|  |  | ||||||
|  |     @Delete | ||||||
|  |     fun delete(action: ActionEntity) | ||||||
|  | } | ||||||
| @@ -2,15 +2,19 @@ package apps.amine.bou.readerforselfoss.persistence.database | |||||||
|  |  | ||||||
| import androidx.room.RoomDatabase | import androidx.room.RoomDatabase | ||||||
| import androidx.room.Database | import androidx.room.Database | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.dao.ActionsDao | ||||||
| import apps.amine.bou.readerforselfoss.persistence.dao.DrawerDataDao | import apps.amine.bou.readerforselfoss.persistence.dao.DrawerDataDao | ||||||
| import apps.amine.bou.readerforselfoss.persistence.dao.ItemsDao | import apps.amine.bou.readerforselfoss.persistence.dao.ItemsDao | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | ||||||
| import apps.amine.bou.readerforselfoss.persistence.entities.ItemEntity | import apps.amine.bou.readerforselfoss.persistence.entities.ItemEntity | ||||||
| import apps.amine.bou.readerforselfoss.persistence.entities.SourceEntity | import apps.amine.bou.readerforselfoss.persistence.entities.SourceEntity | ||||||
| import apps.amine.bou.readerforselfoss.persistence.entities.TagEntity | import apps.amine.bou.readerforselfoss.persistence.entities.TagEntity | ||||||
|  |  | ||||||
| @Database(entities = [TagEntity::class, SourceEntity::class, ItemEntity::class], version = 2) | @Database(entities = [TagEntity::class, SourceEntity::class, ItemEntity::class, ActionEntity::class], version = 3) | ||||||
| abstract class AppDatabase : RoomDatabase() { | abstract class AppDatabase : RoomDatabase() { | ||||||
|     abstract fun drawerDataDao(): DrawerDataDao |     abstract fun drawerDataDao(): DrawerDataDao | ||||||
|  |  | ||||||
|     abstract fun itemsDao(): ItemsDao |     abstract fun itemsDao(): ItemsDao | ||||||
|  |  | ||||||
|  |     abstract fun actionsDao(): ActionsDao | ||||||
| } | } | ||||||
| @@ -0,0 +1,22 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.persistence.entities | ||||||
|  |  | ||||||
|  | import androidx.room.ColumnInfo | ||||||
|  | import androidx.room.Entity | ||||||
|  | import androidx.room.PrimaryKey | ||||||
|  |  | ||||||
|  | @Entity(tableName = "actions") | ||||||
|  | data class ActionEntity( | ||||||
|  |     @ColumnInfo(name = "articleid") | ||||||
|  |     val articleId: String, | ||||||
|  |     @ColumnInfo(name = "read") | ||||||
|  |     val read: Boolean, | ||||||
|  |     @ColumnInfo(name = "unread") | ||||||
|  |     val unread: Boolean, | ||||||
|  |     @ColumnInfo(name = "starred") | ||||||
|  |     var starred: Boolean, | ||||||
|  |     @ColumnInfo(name = "unstarred") | ||||||
|  |     var unstarred: Boolean | ||||||
|  | ) { | ||||||
|  |     @PrimaryKey(autoGenerate = true) | ||||||
|  |     var id: Int = 0 | ||||||
|  | } | ||||||
| @@ -8,3 +8,9 @@ val MIGRATION_1_2: Migration = object : Migration(1, 2) { | |||||||
|         database.execSQL("CREATE TABLE IF NOT EXISTS `items` (`id` TEXT NOT NULL, `datetime` TEXT NOT NULL, `title` TEXT NOT NULL, `content` TEXT NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `thumbnail` TEXT NOT NULL, `icon` TEXT NOT NULL, `link` TEXT NOT NULL, `sourcetitle` TEXT NOT NULL, `tags` TEXT NOT NULL, PRIMARY KEY(`id`))") |         database.execSQL("CREATE TABLE IF NOT EXISTS `items` (`id` TEXT NOT NULL, `datetime` TEXT NOT NULL, `title` TEXT NOT NULL, `content` TEXT NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `thumbnail` TEXT NOT NULL, `icon` TEXT NOT NULL, `link` TEXT NOT NULL, `sourcetitle` TEXT NOT NULL, `tags` TEXT NOT NULL, PRIMARY KEY(`id`))") | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | val MIGRATION_2_3: Migration = object : Migration(2, 3) { | ||||||
|  |     override fun migrate(database: SupportSQLiteDatabase) { | ||||||
|  |         database.execSQL("CREATE TABLE IF NOT EXISTS `actions` (`id` INTEGER NOT NULL, `articleid` TEXT NOT NULL, `read` INTEGER NOT NULL, `unread` INTEGER NOT NULL, `starred` INTEGER NOT NULL, `unstarred` INTEGER NOT NULL, PRIMARY KEY(`id`))") | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -36,6 +36,8 @@ class Config(c: Context) { | |||||||
|  |  | ||||||
|         const val trackerUrl = "https://github.com/aminecmi/ReaderforSelfoss/issues" |         const val trackerUrl = "https://github.com/aminecmi/ReaderforSelfoss/issues" | ||||||
|  |  | ||||||
|  |         const val syncChannelId = "sync-channel-id" | ||||||
|  |  | ||||||
|         fun logoutAndRedirect( |         fun logoutAndRedirect( | ||||||
|             c: Context, |             c: Context, | ||||||
|             callingActivity: Activity, |             callingActivity: Activity, | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ package apps.amine.bou.readerforselfoss.utils | |||||||
| import android.content.Context | import android.content.Context | ||||||
| import android.text.format.DateUtils | import android.text.format.DateUtils | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Item | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossTagType | ||||||
| import org.acra.ACRA | import org.acra.ACRA | ||||||
| import java.text.ParseException | import java.text.ParseException | ||||||
| import java.text.SimpleDateFormat | import java.text.SimpleDateFormat | ||||||
| @@ -44,8 +45,8 @@ fun Item.toggleStar(): Item { | |||||||
| fun List<Item>.flattenTags(): List<Item> = | fun List<Item>.flattenTags(): List<Item> = | ||||||
|     this.flatMap { |     this.flatMap { | ||||||
|         val item = it |         val item = it | ||||||
|         val tags: List<String> = it.tags.split(",") |         val tags: List<String> = it.tags.tags.split(",") | ||||||
|         tags.map { |         tags.map { t -> | ||||||
|             item.copy(tags = it.trim()) |             item.copy(tags = SelfossTagType(t.trim())) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -6,8 +6,13 @@ import android.content.Context | |||||||
| import android.content.Intent | import android.content.Intent | ||||||
| import android.graphics.BitmapFactory | import android.graphics.BitmapFactory | ||||||
| import android.net.Uri | import android.net.Uri | ||||||
|  | import android.text.Spannable | ||||||
|  | import android.text.style.ClickableSpan | ||||||
| import androidx.browser.customtabs.CustomTabsIntent | import androidx.browser.customtabs.CustomTabsIntent | ||||||
| import android.util.Patterns | import android.util.Patterns | ||||||
|  | import android.view.MotionEvent | ||||||
|  | import android.view.View | ||||||
|  | import android.widget.TextView | ||||||
| import android.widget.Toast | import android.widget.Toast | ||||||
| import apps.amine.bou.readerforselfoss.R | import apps.amine.bou.readerforselfoss.R | ||||||
| import apps.amine.bou.readerforselfoss.ReaderActivity | import apps.amine.bou.readerforselfoss.ReaderActivity | ||||||
| @@ -146,3 +151,40 @@ fun Context.openInBrowserAsNewTask(i: Item) { | |||||||
|     intent.data = Uri.parse(i.getLinkDecoded().toStringUriWithHttp()) |     intent.data = Uri.parse(i.getLinkDecoded().toStringUriWithHttp()) | ||||||
|     startActivity(intent) |     startActivity(intent) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | class LinkOnTouchListener: View.OnTouchListener { | ||||||
|  |     override fun onTouch(v: View?, event: MotionEvent?): Boolean { | ||||||
|  |         var ret = false | ||||||
|  |         val widget: TextView = v as TextView | ||||||
|  |         val text: CharSequence = widget.text | ||||||
|  |         val stext = Spannable.Factory.getInstance().newSpannable(text) | ||||||
|  |  | ||||||
|  |         val action = event!!.action | ||||||
|  |  | ||||||
|  |         if (action == MotionEvent.ACTION_UP || | ||||||
|  |             action == MotionEvent.ACTION_DOWN) { | ||||||
|  |             var x: Float = event.x | ||||||
|  |             var y: Float = event.y | ||||||
|  |  | ||||||
|  |             x -= widget.totalPaddingLeft | ||||||
|  |             y -= widget.totalPaddingTop | ||||||
|  |  | ||||||
|  |             x += widget.scrollX | ||||||
|  |             y += widget.scrollY | ||||||
|  |  | ||||||
|  |             val layout = widget.layout | ||||||
|  |             val line = layout.getLineForVertical(y.toInt()) | ||||||
|  |             val off = layout.getOffsetForHorizontal(line, x) | ||||||
|  |  | ||||||
|  |             val link = stext.getSpans(off, off, ClickableSpan::class.java) | ||||||
|  |  | ||||||
|  |             if (link.isNotEmpty()) { | ||||||
|  |                 if (action == MotionEvent.ACTION_UP) { | ||||||
|  |                     link[0].onClick(widget) | ||||||
|  |                 } | ||||||
|  |                 ret = true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return ret | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -0,0 +1,41 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.utils.network | ||||||
|  |  | ||||||
|  | import android.content.Context | ||||||
|  | import android.graphics.Color | ||||||
|  | import android.net.ConnectivityManager | ||||||
|  | import android.net.NetworkInfo | ||||||
|  | import android.view.View | ||||||
|  | import android.widget.TextView | ||||||
|  | import apps.amine.bou.readerforselfoss.R | ||||||
|  | import com.google.android.material.snackbar.Snackbar | ||||||
|  |  | ||||||
|  | var snackBarShown = false | ||||||
|  | var view: View? = null | ||||||
|  |  | ||||||
|  | fun Context.isNetworkAccessible(v: View?): Boolean { | ||||||
|  |     val cm = this.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager | ||||||
|  |     val activeNetwork: NetworkInfo? = cm.activeNetworkInfo | ||||||
|  |     val networkIsAccessible = activeNetwork != null && activeNetwork.isConnectedOrConnecting | ||||||
|  |  | ||||||
|  |     if (v != null && !networkIsAccessible && (!snackBarShown || v != view)) { | ||||||
|  |         view = v | ||||||
|  |         val s = Snackbar | ||||||
|  |             .make( | ||||||
|  |                 v, | ||||||
|  |                 R.string.no_network_connectivity, | ||||||
|  |                 Snackbar.LENGTH_INDEFINITE | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |         s.setAction(android.R.string.ok) { | ||||||
|  |             snackBarShown = false | ||||||
|  |             s.dismiss() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val view = s.view | ||||||
|  |         val tv: TextView = view.findViewById(com.google.android.material.R.id.snackbar_text) | ||||||
|  |         tv.setTextColor(Color.WHITE) | ||||||
|  |         s.show() | ||||||
|  |         snackBarShown = true | ||||||
|  |     } | ||||||
|  |     return networkIsAccessible | ||||||
|  | } | ||||||
| @@ -1,6 +1,7 @@ | |||||||
| package apps.amine.bou.readerforselfoss.utils.persistence | package apps.amine.bou.readerforselfoss.utils.persistence | ||||||
|  |  | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Item | import apps.amine.bou.readerforselfoss.api.selfoss.Item | ||||||
|  | import apps.amine.bou.readerforselfoss.api.selfoss.SelfossTagType | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Source | import apps.amine.bou.readerforselfoss.api.selfoss.Source | ||||||
| import apps.amine.bou.readerforselfoss.api.selfoss.Tag | import apps.amine.bou.readerforselfoss.api.selfoss.Tag | ||||||
| import apps.amine.bou.readerforselfoss.persistence.entities.ItemEntity | import apps.amine.bou.readerforselfoss.persistence.entities.ItemEntity | ||||||
| @@ -18,7 +19,7 @@ fun SourceEntity.toView(): Source = | |||||||
|         Source( |         Source( | ||||||
|             this.id, |             this.id, | ||||||
|             this.title, |             this.title, | ||||||
|             this.tags, |             SelfossTagType(this.tags), | ||||||
|             this.spout, |             this.spout, | ||||||
|             this.error, |             this.error, | ||||||
|             this.icon |             this.icon | ||||||
| @@ -28,7 +29,7 @@ fun Source.toEntity(): SourceEntity = | |||||||
|         SourceEntity( |         SourceEntity( | ||||||
|             this.id, |             this.id, | ||||||
|             this.title, |             this.title, | ||||||
|             this.tags, |             this.tags.tags, | ||||||
|             this.spout, |             this.spout, | ||||||
|             this.error, |             this.error, | ||||||
|             this.icon.orEmpty() |             this.icon.orEmpty() | ||||||
| @@ -53,7 +54,7 @@ fun ItemEntity.toView(): Item = | |||||||
|             this.icon, |             this.icon, | ||||||
|             this.link, |             this.link, | ||||||
|             this.sourcetitle, |             this.sourcetitle, | ||||||
|             this.tags |             SelfossTagType(this.tags) | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
| fun Item.toEntity(): ItemEntity = | fun Item.toEntity(): ItemEntity = | ||||||
| @@ -68,5 +69,5 @@ fun Item.toEntity(): ItemEntity = | |||||||
|         this.icon, |         this.icon, | ||||||
|         this.link, |         this.link, | ||||||
|         this.sourcetitle, |         this.sourcetitle, | ||||||
|         this.tags |         this.tags.tags | ||||||
|     ) |     ) | ||||||
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-hdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								app/src/main/res/drawable-hdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 334 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-mdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								app/src/main/res/drawable-mdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 228 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xhdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xhdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 380 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xxhdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xxhdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 547 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xxxhdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xxxhdpi/ic_cloud_download.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 678 B | 
| @@ -2,6 +2,7 @@ | |||||||
| <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" | <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|     android:layout_width="match_parent" |     android:layout_width="match_parent" | ||||||
|  |     android:id="@+id/reader_activity_view" | ||||||
|     android:layout_height="match_parent"> |     android:layout_height="match_parent"> | ||||||
|  |  | ||||||
|     <com.google.android.material.appbar.AppBarLayout |     <com.google.android.material.appbar.AppBarLayout | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Los artículos no se guardarán en la memoria del dispositivo y la aplicación no se podrá utilizar sin conexión.</string> |     <string name="pref_switch_items_caching_off">Los artículos no se guardarán en la memoria del dispositivo y la aplicación no se podrá utilizar sin conexión.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Los artículos se guardarán en la memoria del dispositivo y se utilizarán para el uso sin conexión.</string> |     <string name="pref_switch_items_caching_on">Los artículos se guardarán en la memoria del dispositivo y se utilizarán para el uso sin conexión.</string> | ||||||
|     <string name="pref_switch_items_caching">Guardar elementos para uso sin conexión</string> |     <string name="pref_switch_items_caching">Guardar elementos para uso sin conexión</string> | ||||||
|  |     <string name="no_network_connectivity">Sin conexión!</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sincronizar artículos</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Los artículos no se sincronizarán en segundo plano</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Los artículos se sincronizarán periódicamente</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Intervalo de sincronización (>= 15 minutos)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Sólo refrescar cuando el teléfono está cargando</string> | ||||||
|  |     <string name="loading_notification_title">Cargando...</string> | ||||||
|  |     <string name="loading_notification_text">Selfoss está sincronizando tus artículos</string> | ||||||
|  |     <string name="notification_channel_sync">Notificación de sincronización</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -150,6 +150,15 @@ | |||||||
|     <string name="unmark">Marquer l\'article comme non lu</string> |     <string name="unmark">Marquer l\'article comme non lu</string> | ||||||
|     <string name="pref_header_offline">Hors ligne et cache</string> |     <string name="pref_header_offline">Hors ligne et cache</string> | ||||||
|     <string name="pref_switch_items_caching_off">Les articles ne seront pas enregistrés et l\'application ne sera pas utilisable hors ligne.</string> |     <string name="pref_switch_items_caching_off">Les articles ne seront pas enregistrés et l\'application ne sera pas utilisable hors ligne.</string> | ||||||
|   <string name="pref_switch_items_caching_on">Les articles seront enregistrés et l\'application ne sera utilisable hors ligne.</string> |     <string name="pref_switch_items_caching_on">Les articles seront enregistrés et l\'application sera utilisable hors ligne.</string> | ||||||
|     <string name="pref_switch_items_caching">Sauvegarder les articles pour une utilisation hors ligne</string> |     <string name="pref_switch_items_caching">Sauvegarder les articles pour une utilisation hors ligne</string> | ||||||
|  |     <string name="no_network_connectivity">Hors connexion !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Synchroniser les articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Les articles ne seront pas synchronisés en arrière plan</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles seront périodiquement synchronisées</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Interval de synchronisation ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Synchroniser uniquement lorsque le téléphone est en charge</string> | ||||||
|  |     <string name="loading_notification_title">Chargement ...</string> | ||||||
|  |     <string name="loading_notification_text">Selfoss synchronise vos articles</string> | ||||||
|  |     <string name="notification_channel_sync">Notification de synchronisation</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Os artigos non se gardaran na memoria do dispositivo e non se poderá utilizar a aplicación sen conexión.</string> |     <string name="pref_switch_items_caching_off">Os artigos non se gardaran na memoria do dispositivo e non se poderá utilizar a aplicación sen conexión.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Os artigos gardaranse na memoria do dispositivo e estarán dispoñibles sen conexión.</string> |     <string name="pref_switch_items_caching_on">Os artigos gardaranse na memoria do dispositivo e estarán dispoñibles sen conexión.</string> | ||||||
|     <string name="pref_switch_items_caching">Gardar elementos para uso sen conexión</string> |     <string name="pref_switch_items_caching">Gardar elementos para uso sen conexión</string> | ||||||
|  |     <string name="no_network_connectivity">Non conectado!</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sincronizar artigos</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Os artigos non se sincronizarán coa aplicación de fondo</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Os artigos sincronizaranse periódicamente</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Intervalo de sincronización (>= 15 minutos)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Só refrescar cando o teléfono se está a cargar</string> | ||||||
|  |     <string name="loading_notification_title">Cargando...</string> | ||||||
|  |     <string name="loading_notification_text">Selfoss está sincronizando os teus ar tigos</string> | ||||||
|  |     <string name="notification_channel_sync">Notificación de sincronización</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -108,7 +108,7 @@ | |||||||
|     <string name="self_signed_cert_warning">出于安全考虑, 默认情况下不支持自签名证书。如果激活此项, 您遇到的任何安全问题我将概不负责。</string> |     <string name="self_signed_cert_warning">出于安全考虑, 默认情况下不支持自签名证书。如果激活此项, 您遇到的任何安全问题我将概不负责。</string> | ||||||
|     <string name="pref_selfoss_category">塞尔福斯 Api</string> |     <string name="pref_selfoss_category">塞尔福斯 Api</string> | ||||||
|     <string name="pref_api_items_number_title">已加载项目编号</string> |     <string name="pref_api_items_number_title">已加载项目编号</string> | ||||||
|   <string name="pref_hidden_tags">Hidden Tags</string> |     <string name="pref_hidden_tags">隐藏段落</string> | ||||||
|     <string name="read_debug_title">已读文章显示为未读?</string> |     <string name="read_debug_title">已读文章显示为未读?</string> | ||||||
|     <string name="read_debug_off">将项目标记为已读时没有记录</string> |     <string name="read_debug_off">将项目标记为已读时没有记录</string> | ||||||
|     <string name="read_debug_on">将项目标记为已读时将记录 Api 调用</string> |     <string name="read_debug_on">将项目标记为已读时将记录 Api 调用</string> | ||||||
| @@ -133,17 +133,17 @@ | |||||||
|     <string name="pref_content_reader_font_size">Article reader content font size</string> |     <string name="pref_content_reader_font_size">Article reader content font size</string> | ||||||
|     <string name="pref_header_viewer">Article viewer</string> |     <string name="pref_header_viewer">Article viewer</string> | ||||||
|     <string name="refresh_dialog_message">This will refresh your Selfoss instance.</string> |     <string name="refresh_dialog_message">This will refresh your Selfoss instance.</string> | ||||||
|   <string name="markall_dialog_message">This will mark all the items as read.</string> |     <string name="markall_dialog_message">這會使全部項目標示為已讀</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> |     <string name="pref_switch_actions_pager_scroll">Mark as read on swipe</string> | ||||||
|     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> |     <string name="pref_switch_actions_pager_scroll_off">Don\'t mark articles as read when swiping.</string> | ||||||
|     <string name="gdpr_dialog_message">The app does not collect any personal data. Every analytics tools were removed. Crash reports sending is now optional, as is the debug logging. Keep in mind that debugging and crash reports are essential for the app development (You can configure everything in Settings > Debug).</string> |     <string name="gdpr_dialog_message">The app does not collect any personal data. Every analytics tools were removed. Crash reports sending is now optional, as is the debug logging. Keep in mind that debugging and crash reports are essential for the app development (You can configure everything in Settings > Debug).</string> | ||||||
|   <string name="gdpr_dialog_title">The app does not share any personal data about you.</string> |     <string name="gdpr_dialog_title">這應用程式不會分享你的任何個人資訊</string> | ||||||
|     <string name="crash_dialog_text">Something went wrong. Please send the report to the developer.</string> |     <string name="crash_dialog_text">Something went wrong. Please send the report to the developer.</string> | ||||||
|     <string name="crash_dialog_comment">You can add any helpful details in the comment bellow. Don\'t include any personal data in your comment. You could send me and email with your debug id, and I\'ll keep you posted when the issue is resolved.</string> |     <string name="crash_dialog_comment">You can add any helpful details in the comment bellow. Don\'t include any personal data in your comment. You could send me and email with your debug id, and I\'ll keep you posted when the issue is resolved.</string> | ||||||
|   <string name="pref_acra_alwaysaccept">Automatically send crash reports</string> |     <string name="pref_acra_alwaysaccept">自动发送錯誤报告</string> | ||||||
|     <string name="pref_acra_alwaysaccept_enabled">Will send crash reports automatically</string> |     <string name="pref_acra_alwaysaccept_enabled">Will send crash reports automatically</string> | ||||||
|     <string name="pref_acra_alwaysaccept_disabled">Will ask everytime when sending crash reports.</string> |     <string name="pref_acra_alwaysaccept_disabled">Will ask everytime when sending crash reports.</string> | ||||||
|   <string name="pref_debug_crash_reports">Crash reports</string> |     <string name="pref_debug_crash_reports">错误报告</string> | ||||||
|     <string name="pref_debug_debug_logs">Debug logging (these will be sent without a dialog)</string> |     <string name="pref_debug_debug_logs">Debug logging (these will be sent without a dialog)</string> | ||||||
|     <string name="acra_login">Enable logging</string> |     <string name="acra_login">Enable logging</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |     <string name="drawer_item_hidden_tags">Hidden Tags</string> | ||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">未连接</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> | ||||||
|  |     <string name="loading_notification_title">加载中...</string> | ||||||
|  |     <string name="loading_notification_text">Selfoss is syncing your articles</string> | ||||||
|  |     <string name="notification_channel_sync">同步通知</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -152,4 +152,13 @@ | |||||||
|     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> |     <string name="pref_switch_items_caching_on">Articles will be saved to the device memory and will be used for offline use.</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">Save items for offline use</string> | ||||||
|  |     <string name="no_network_connectivity">Not connected !</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh">Sync articles</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> | ||||||
|  |     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> | ||||||
|  |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> | ||||||
|  |     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</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> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -5,4 +5,27 @@ | |||||||
|         android:summaryOff="@string/pref_switch_items_caching_off" |         android:summaryOff="@string/pref_switch_items_caching_off" | ||||||
|         android:summaryOn="@string/pref_switch_items_caching_on" |         android:summaryOn="@string/pref_switch_items_caching_on" | ||||||
|         android:title="@string/pref_switch_items_caching" /> |         android:title="@string/pref_switch_items_caching" /> | ||||||
|  |  | ||||||
|  |     <SwitchPreference | ||||||
|  |         android:defaultValue="false" | ||||||
|  |         android:key="periodic_refresh" | ||||||
|  |         android:dependency="items_caching" | ||||||
|  |         android:summaryOff="@string/pref_switch_periodic_refresh_off" | ||||||
|  |         android:summaryOn="@string/pref_switch_periodic_refresh_on" | ||||||
|  |         android:title="@string/pref_switch_periodic_refresh" /> | ||||||
|  |  | ||||||
|  |     <EditTextPreference | ||||||
|  |         android:dependency="periodic_refresh" | ||||||
|  |         android:defaultValue="360" | ||||||
|  |         android:inputType="number" | ||||||
|  |         android:key="periodic_refresh_minutes" | ||||||
|  |         android:selectAllOnFocus="true" | ||||||
|  |         android:singleLine="true" | ||||||
|  |         android:title="@string/pref_periodic_refresh_minutes_title" /> | ||||||
|  |  | ||||||
|  |     <SwitchPreference | ||||||
|  |         android:defaultValue="false" | ||||||
|  |         android:key="refresh_when_charging" | ||||||
|  |         android:dependency="periodic_refresh" | ||||||
|  |         android:title="@string/pref_switch_refresh_when_charging" /> | ||||||
| </PreferenceScreen> | </PreferenceScreen> | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ buildscript { | |||||||
|         android_version = '1.0.0' |         android_version = '1.0.0' | ||||||
|         lifecycle_version = '2.0.0' |         lifecycle_version = '2.0.0' | ||||||
|         room_version = '2.1.0-alpha01' |         room_version = '2.1.0-alpha01' | ||||||
|  |         work_version = "1.0.0-alpha10" | ||||||
|     } |     } | ||||||
|     repositories { |     repositories { | ||||||
|         google() |         google() | ||||||
| @@ -15,7 +16,7 @@ buildscript { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     dependencies { |     dependencies { | ||||||
|         classpath 'com.android.tools.build:gradle:3.2.0' |         classpath 'com.android.tools.build:gradle:3.2.1' | ||||||
|         classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" |         classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user