Compare commits
	
		
			14 Commits
		
	
	
		
			v172002032
			...
			v172012358
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 953940690d | ||
| 918661be2d | |||
|  | 7b8a5c9a56 | ||
| 2d5ab7bf0c | |||
| 9ba281befb | |||
| 00c8eed034 | |||
| a1e4f89cd1 | |||
| 36a43b3861 | |||
|  | aa6d470f40 | ||
|  | 0046a8a477 | ||
|  | 43ff9d186a | ||
|  | 73dae304be | ||
|  | 66103a451b | ||
|  | 600c62316d | 
							
								
								
									
										16
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.md
									
									
									
									
									
								
							| @@ -4,8 +4,24 @@ | |||||||
|  |  | ||||||
| It's an RSS Reader for Android, that **only** works with [Selfoss](https://selfoss.aditu.de/) | It's an RSS Reader for Android, that **only** works with [Selfoss](https://selfoss.aditu.de/) | ||||||
|  |  | ||||||
|  | **The project is not dead at all.**  | ||||||
|  |  | ||||||
|  | I still want to work on it, but for the last few months, I didn't have that much time to do so.  | ||||||
|  |  | ||||||
|  | If you are a developer, don't hesitate to help with PRs. | ||||||
|  |  | ||||||
|  | If you are a user, you can still create new issues. I'll fix them when I can. | ||||||
|  |  | ||||||
| <a href="https://f-droid.org/packages/apps.amine.bou.readerforselfoss"><img src="https://f-droid.org/badge/get-it-on.png" alt="Get it on F-Droid" height="100"></a> | <a href="https://f-droid.org/packages/apps.amine.bou.readerforselfoss"><img src="https://f-droid.org/badge/get-it-on.png" alt="Get it on F-Droid" height="100"></a> | ||||||
|  |  | ||||||
|  | ## Screen captures | ||||||
|  |  | ||||||
|  | <img src="res//fr-card.png?raw=true" alt="card view" width="400"/> <img src="res//fr-list.png?raw=true" alt="list view" width="400"/> | ||||||
|  |  | ||||||
|  | ## Like my app ? | ||||||
|  |  | ||||||
|  | <a href="https://www.buymeacoffee.com/aminecmi" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/lato-orange.png" alt="Buy Me A Coffee" style="height: 51px !important;width: 217px !important;" ></a> | ||||||
|  |  | ||||||
| ## Want to help ? | ## Want to help ? | ||||||
|  |  | ||||||
| 1. **You'll have to have a Selfoss instance running.** You'll find everything you need to install it [here](https://selfoss.aditu.de/). | 1. **You'll have to have a Selfoss instance running.** You'll find everything you need to install it [here](https://selfoss.aditu.de/). | ||||||
|   | |||||||
| @@ -101,7 +101,8 @@ dependencies { | |||||||
|     implementation "androidx.vectordrawable:vectordrawable:1.1.0-beta01" |     implementation "androidx.vectordrawable:vectordrawable:1.1.0-beta01" | ||||||
|     implementation "androidx.browser:browser:$android_version" |     implementation "androidx.browser:browser:$android_version" | ||||||
|     implementation "androidx.cardview:cardview:$android_version" |     implementation "androidx.cardview:cardview:$android_version" | ||||||
|     implementation 'androidx.constraintlayout:constraintlayout:2.0.0-alpha5' |     implementation 'androidx.constraintlayout:constraintlayout:2.1.0-alpha1' | ||||||
|  |     implementation 'org.jsoup:jsoup:1.13.1' | ||||||
|  |  | ||||||
|     //multidex |     //multidex | ||||||
|     implementation 'androidx.multidex:multidex:2.0.1' |     implementation 'androidx.multidex:multidex:2.0.1' | ||||||
| @@ -137,6 +138,9 @@ dependencies { | |||||||
|     // Pager |     // Pager | ||||||
|     implementation 'me.relex:circleindicator:2.0.0@aar' |     implementation 'me.relex:circleindicator:2.0.0@aar' | ||||||
|  |  | ||||||
|  |     //PhotoView | ||||||
|  |     implementation 'com.github.chrisbanes:PhotoView:2.0.0' | ||||||
|  |  | ||||||
|     implementation 'androidx.core:core-ktx:1.1.0-beta01' |     implementation 'androidx.core:core-ktx:1.1.0-beta01' | ||||||
|  |  | ||||||
|     implementation "androidx.lifecycle:lifecycle-livedata:$lifecycle_version" |     implementation "androidx.lifecycle:lifecycle-livedata:$lifecycle_version" | ||||||
| @@ -146,4 +150,4 @@ dependencies { | |||||||
|     kapt "androidx.room:room-compiler:$room_version" |     kapt "androidx.room:room-compiler:$room_version" | ||||||
|  |  | ||||||
|     implementation "android.arch.work:work-runtime-ktx:$work_version" |     implementation "android.arch.work:work-runtime-ktx:$work_version" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,226 @@ | |||||||
|  | { | ||||||
|  |   "formatVersion": 1, | ||||||
|  |   "database": { | ||||||
|  |     "version": 4, | ||||||
|  |     "identityHash": "9cf8b03d32f80dfd58160599a1df197d", | ||||||
|  |     "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, `icon` TEXT, `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": false | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "fieldPath": "icon", | ||||||
|  |             "columnName": "icon", | ||||||
|  |             "affinity": "TEXT", | ||||||
|  |             "notNull": false | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             "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, \"9cf8b03d32f80dfd58160599a1df197d\")" | ||||||
|  |     ] | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -62,6 +62,9 @@ | |||||||
|         <activity |         <activity | ||||||
|             android:name=".ReaderActivity"> |             android:name=".ReaderActivity"> | ||||||
|         </activity> |         </activity> | ||||||
|  |         <activity | ||||||
|  |             android:name=".ImageActivity"> | ||||||
|  |         </activity> | ||||||
|  |  | ||||||
|         <meta-data |         <meta-data | ||||||
|             android:name="apps.amine.bou.readerforselfoss.utils.glide.SelfSignedGlideModule" |             android:name="apps.amine.bou.readerforselfoss.utils.glide.SelfSignedGlideModule" | ||||||
|   | |||||||
| @@ -33,6 +33,7 @@ import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | |||||||
| import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | 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.persistence.migrations.MIGRATION_2_3 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_3_4 | ||||||
| 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 | ||||||
| @@ -164,7 +165,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).addMigrations(MIGRATION_2_3).build() |         ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).addMigrations(MIGRATION_3_4).build() | ||||||
|  |  | ||||||
|  |  | ||||||
|         customTabActivityHelper = CustomTabActivityHelper() |         customTabActivityHelper = CustomTabActivityHelper() | ||||||
|   | |||||||
| @@ -0,0 +1,52 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss | ||||||
|  |  | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.MenuItem | ||||||
|  | import androidx.appcompat.app.AppCompatActivity | ||||||
|  | import androidx.fragment.app.FragmentManager | ||||||
|  | import androidx.fragment.app.FragmentStatePagerAdapter | ||||||
|  | import apps.amine.bou.readerforselfoss.fragments.ImageFragment | ||||||
|  | import kotlinx.android.synthetic.main.activity_reader.* | ||||||
|  |  | ||||||
|  | class ImageActivity : AppCompatActivity() { | ||||||
|  |     private lateinit var allImages : ArrayList<String> | ||||||
|  |     private var position : Int = 0 | ||||||
|  |  | ||||||
|  |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |  | ||||||
|  |         setContentView(R.layout.activity_image) | ||||||
|  |  | ||||||
|  |         setSupportActionBar(toolBar) | ||||||
|  |         supportActionBar?.setDisplayShowTitleEnabled(false) | ||||||
|  |         supportActionBar?.setDisplayHomeAsUpEnabled(true) | ||||||
|  |  | ||||||
|  |         allImages = intent.getStringArrayListExtra("allImages") | ||||||
|  |         position = intent.getIntExtra("position", 0) | ||||||
|  |  | ||||||
|  |         pager.adapter = ScreenSlidePagerAdapter(supportFragmentManager) | ||||||
|  |         pager.currentItem = position | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onOptionsItemSelected(item: MenuItem): Boolean { | ||||||
|  |         when (item.itemId) { | ||||||
|  |             android.R.id.home -> { | ||||||
|  |                 onBackPressed() | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return super.onOptionsItemSelected(item) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private inner class ScreenSlidePagerAdapter(fm: FragmentManager) : FragmentStatePagerAdapter(fm, FragmentStatePagerAdapter.BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT) { | ||||||
|  |  | ||||||
|  |         override fun getCount(): Int { | ||||||
|  |             return allImages.size | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         override fun getItem(position: Int): ImageFragment { | ||||||
|  |             return ImageFragment.newInstance(allImages[position]) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -25,6 +25,7 @@ import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | |||||||
| import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | 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.persistence.migrations.MIGRATION_2_3 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_3_4 | ||||||
| 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 | ||||||
| @@ -81,7 +82,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).addMigrations(MIGRATION_2_3).build() |         ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).addMigrations(MIGRATION_3_4).build() | ||||||
|  |  | ||||||
|         val scoop = Scoop.getInstance() |         val scoop = Scoop.getInstance() | ||||||
|         scoop.bind(this, Toppings.PRIMARY.value, toolBar) |         scoop.bind(this, Toppings.PRIMARY.value, toolBar) | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ import android.app.Activity | |||||||
| import android.content.Context | import android.content.Context | ||||||
| import androidx.cardview.widget.CardView | import androidx.cardview.widget.CardView | ||||||
| import androidx.recyclerview.widget.RecyclerView | import androidx.recyclerview.widget.RecyclerView | ||||||
| import android.text.Html |  | ||||||
| import android.view.LayoutInflater | import android.view.LayoutInflater | ||||||
| import android.view.View | import android.view.View | ||||||
| import android.view.ViewGroup | import android.view.ViewGroup | ||||||
| @@ -69,7 +68,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 = itm.getTitleDecoded() | ||||||
|         holder.mView.title.setOnTouchListener(LinkOnTouchListener()) |         holder.mView.title.setOnTouchListener(LinkOnTouchListener()) | ||||||
|  |  | ||||||
|         holder.mView.title.setLinkTextColor(appColors.colorAccent) |         holder.mView.title.setLinkTextColor(appColors.colorAccent) | ||||||
| @@ -181,7 +180,7 @@ class ItemCardAdapter( | |||||||
|  |  | ||||||
|             mView.shareBtn.setOnClickListener { |             mView.shareBtn.setOnClickListener { | ||||||
|                 val item = items[adapterPosition] |                 val item = items[adapterPosition] | ||||||
|                 c.shareLink(item.getLinkDecoded(), item.title) |                 c.shareLink(item.getLinkDecoded(), item.getTitleDecoded()) | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             mView.browserBtn.setOnClickListener { |             mView.browserBtn.setOnClickListener { | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ import android.app.Activity | |||||||
| import android.content.Context | 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.Spannable | import android.text.Spannable | ||||||
| import android.text.style.ClickableSpan | import android.text.style.ClickableSpan | ||||||
| import android.util.TypedValue | import android.util.TypedValue | ||||||
| @@ -71,7 +70,7 @@ class ItemListAdapter( | |||||||
|         val itm = items[position] |         val itm = items[position] | ||||||
|  |  | ||||||
|  |  | ||||||
|         holder.mView.title.text = Html.fromHtml(itm.title) |         holder.mView.title.text = itm.getTitleDecoded() | ||||||
|  |  | ||||||
|         holder.mView.title.setOnTouchListener(LinkOnTouchListener()) |         holder.mView.title.setOnTouchListener(LinkOnTouchListener()) | ||||||
|  |  | ||||||
| @@ -80,23 +79,6 @@ class ItemListAdapter( | |||||||
|         holder.mView.sourceTitleAndDate.text = itm.sourceAndDateText() |         holder.mView.sourceTitleAndDate.text = itm.sourceAndDateText() | ||||||
|  |  | ||||||
|         if (itm.getThumbnail(c).isEmpty()) { |         if (itm.getThumbnail(c).isEmpty()) { | ||||||
|             val sizeInInt = 46 |  | ||||||
|             val sizeInDp = TypedValue.applyDimension( |  | ||||||
|                 TypedValue.COMPLEX_UNIT_DIP, sizeInInt.toFloat(), c.resources |  | ||||||
|                     .displayMetrics |  | ||||||
|             ).toInt() |  | ||||||
|  |  | ||||||
|             val marginInInt = 16 |  | ||||||
|             val marginInDp = TypedValue.applyDimension( |  | ||||||
|                 TypedValue.COMPLEX_UNIT_DIP, marginInInt.toFloat(), c.resources |  | ||||||
|                     .displayMetrics |  | ||||||
|             ).toInt() |  | ||||||
|  |  | ||||||
|             val params = holder.mView.itemImage.layoutParams as ViewGroup.MarginLayoutParams |  | ||||||
|             params.height = sizeInDp |  | ||||||
|             params.width = sizeInDp |  | ||||||
|             params.setMargins(marginInDp, 0, 0, 0) |  | ||||||
|             holder.mView.itemImage.layoutParams = params |  | ||||||
|  |  | ||||||
|             if (itm.getIcon(c).isEmpty()) { |             if (itm.getIcon(c).isEmpty()) { | ||||||
|                 val color = generator.getColor(itm.sourcetitle) |                 val color = generator.getColor(itm.sourcetitle) | ||||||
|   | |||||||
| @@ -4,9 +4,15 @@ import android.content.Context | |||||||
| import android.net.Uri | import android.net.Uri | ||||||
| import android.os.Parcel | import android.os.Parcel | ||||||
| import android.os.Parcelable | import android.os.Parcelable | ||||||
|  | import android.text.Html | ||||||
|  | import android.webkit.URLUtil | ||||||
|  | import org.jsoup.Jsoup | ||||||
|  |  | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | ||||||
|  | import com.bumptech.glide.Glide | ||||||
|  | import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||||
|  | import com.bumptech.glide.request.RequestOptions | ||||||
| import com.google.gson.annotations.SerializedName | import com.google.gson.annotations.SerializedName | ||||||
|  |  | ||||||
| private fun constructUrl(config: Config?, path: String, file: String?): String { | private fun constructUrl(config: Config?, path: String, file: String?): String { | ||||||
| @@ -67,8 +73,8 @@ data class Item( | |||||||
|     @SerializedName("content") val content: String, |     @SerializedName("content") val content: String, | ||||||
|     @SerializedName("unread") val unread: Boolean, |     @SerializedName("unread") val unread: Boolean, | ||||||
|     @SerializedName("starred") var starred: Boolean, |     @SerializedName("starred") var starred: Boolean, | ||||||
|     @SerializedName("thumbnail") val thumbnail: String, |     @SerializedName("thumbnail") val thumbnail: String?, | ||||||
|     @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: SelfossTagType |     @SerializedName("tags") val tags: SelfossTagType | ||||||
| @@ -127,6 +133,40 @@ data class Item( | |||||||
|         return constructUrl(config, "thumbnails", thumbnail) |         return constructUrl(config, "thumbnails", thumbnail) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     fun getImages() : ArrayList<String> { | ||||||
|  |         var allImages = ArrayList<String>() | ||||||
|  |  | ||||||
|  |         for ( image in Jsoup.parse(content).getElementsByTag("img")) { | ||||||
|  |             allImages.add(image.attr("src")) | ||||||
|  |         } | ||||||
|  |         return allImages | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun preloadImages(context: Context) : Boolean { | ||||||
|  |         val imageUrls = this.getImages() | ||||||
|  |  | ||||||
|  |         val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         try { | ||||||
|  |             for (url in imageUrls) { | ||||||
|  |                 if ( URLUtil.isValidUrl(url)) { | ||||||
|  |                     val image = Glide.with(context).asBitmap() | ||||||
|  |                             .apply(glideOptions) | ||||||
|  |                             .load(url).submit().get() | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } catch (e : Error) { | ||||||
|  |             return false | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fun getTitleDecoded(): String { | ||||||
|  |         return Html.fromHtml(title).toString() | ||||||
|  |     } | ||||||
|  |  | ||||||
|     // TODO: maybe find a better way to handle these kind of urls |     // TODO: maybe find a better way to handle these kind of urls | ||||||
|     fun getLinkDecoded(): String { |     fun getLinkDecoded(): String { | ||||||
|         var stringUrl: String |         var stringUrl: String | ||||||
|   | |||||||
| @@ -19,6 +19,7 @@ import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | |||||||
| import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | 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.persistence.migrations.MIGRATION_2_3 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_3_4 | ||||||
| import apps.amine.bou.readerforselfoss.utils.Config | import apps.amine.bou.readerforselfoss.utils.Config | ||||||
| import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | import apps.amine.bou.readerforselfoss.utils.persistence.toEntity | ||||||
| @@ -56,7 +57,7 @@ class LoadingWorker(val context: Context, params: WorkerParameters) : Worker(con | |||||||
|             db = Room.databaseBuilder( |             db = Room.databaseBuilder( | ||||||
|                 applicationContext, |                 applicationContext, | ||||||
|                 AppDatabase::class.java, "selfoss-database" |                 AppDatabase::class.java, "selfoss-database" | ||||||
|             ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).build() |             ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).addMigrations(MIGRATION_3_4).build() | ||||||
|  |  | ||||||
|             val api = SelfossApi( |             val api = SelfossApi( | ||||||
|                 this.context, |                 this.context, | ||||||
| @@ -104,6 +105,7 @@ class LoadingWorker(val context: Context, params: WorkerParameters) : Worker(con | |||||||
|                                     notificationManager.notify(2, newItemsNotification.build()) |                                     notificationManager.notify(2, newItemsNotification.build()) | ||||||
|                                 } |                                 } | ||||||
|                             } |                             } | ||||||
|  |                             apiItems.map {it.preloadImages(context)} | ||||||
|                         } |                         } | ||||||
|                         Timer("", false).schedule(4000) { |                         Timer("", false).schedule(4000) { | ||||||
|                             notificationManager.cancel(1) |                             notificationManager.cancel(1) | ||||||
|   | |||||||
| @@ -1,28 +1,28 @@ | |||||||
| package apps.amine.bou.readerforselfoss.fragments | package apps.amine.bou.readerforselfoss.fragments | ||||||
|  |  | ||||||
| import android.content.Context | import android.content.Context | ||||||
|  | import android.content.Intent | ||||||
| import android.content.SharedPreferences | import android.content.SharedPreferences | ||||||
| import android.content.res.ColorStateList | import android.content.res.ColorStateList | ||||||
| import android.content.res.TypedArray | import android.content.res.TypedArray | ||||||
|  | import android.graphics.Bitmap | ||||||
| import android.graphics.Typeface | import android.graphics.Typeface | ||||||
| import android.graphics.drawable.ColorDrawable | import android.graphics.drawable.ColorDrawable | ||||||
|  | import android.net.Uri | ||||||
| import android.os.Build | import android.os.Build | ||||||
| import android.os.Bundle | import android.os.Bundle | ||||||
| import android.preference.PreferenceManager | import android.preference.PreferenceManager | ||||||
| import android.view.InflateException | import android.view.* | ||||||
|  | import android.webkit.* | ||||||
| import androidx.browser.customtabs.CustomTabsIntent | import androidx.browser.customtabs.CustomTabsIntent | ||||||
| import com.google.android.material.floatingactionbutton.FloatingActionButton | import com.google.android.material.floatingactionbutton.FloatingActionButton | ||||||
| import androidx.fragment.app.Fragment | import androidx.fragment.app.Fragment | ||||||
| import androidx.core.content.ContextCompat | import androidx.core.content.ContextCompat | ||||||
| import androidx.core.widget.NestedScrollView | import androidx.core.widget.NestedScrollView | ||||||
| import android.view.LayoutInflater |  | ||||||
| import android.view.MenuItem |  | ||||||
| import android.view.View |  | ||||||
| import android.view.ViewGroup |  | ||||||
| import android.webkit.WebSettings |  | ||||||
| import androidx.appcompat.app.AlertDialog | import androidx.appcompat.app.AlertDialog | ||||||
| import androidx.core.content.res.ResourcesCompat | import androidx.core.content.res.ResourcesCompat | ||||||
| import androidx.room.Room | import androidx.room.Room | ||||||
|  | import apps.amine.bou.readerforselfoss.ImageActivity | ||||||
| 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 | ||||||
| @@ -33,11 +33,13 @@ import apps.amine.bou.readerforselfoss.persistence.database.AppDatabase | |||||||
| import apps.amine.bou.readerforselfoss.persistence.entities.ActionEntity | 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.persistence.migrations.MIGRATION_2_3 | ||||||
|  | import apps.amine.bou.readerforselfoss.persistence.migrations.MIGRATION_3_4 | ||||||
| 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.glide.loadMaybeBasicAuth | import apps.amine.bou.readerforselfoss.utils.glide.loadMaybeBasicAuth | ||||||
|  | import apps.amine.bou.readerforselfoss.utils.glide.getBitmapInputStream | ||||||
| import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | import apps.amine.bou.readerforselfoss.utils.isEmptyOrNullOrNullString | ||||||
| import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | import apps.amine.bou.readerforselfoss.utils.network.isNetworkAccessible | ||||||
| import apps.amine.bou.readerforselfoss.utils.openItemUrl | import apps.amine.bou.readerforselfoss.utils.openItemUrl | ||||||
| @@ -45,6 +47,7 @@ import apps.amine.bou.readerforselfoss.utils.shareLink | |||||||
| import apps.amine.bou.readerforselfoss.utils.sourceAndDateText | import apps.amine.bou.readerforselfoss.utils.sourceAndDateText | ||||||
| import apps.amine.bou.readerforselfoss.utils.succeeded | import apps.amine.bou.readerforselfoss.utils.succeeded | ||||||
| import com.bumptech.glide.Glide | import com.bumptech.glide.Glide | ||||||
|  | import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||||
| import com.bumptech.glide.request.RequestOptions | import com.bumptech.glide.request.RequestOptions | ||||||
| import com.github.rubensousa.floatingtoolbar.FloatingToolbar | import com.github.rubensousa.floatingtoolbar.FloatingToolbar | ||||||
| import kotlinx.android.synthetic.main.fragment_article.view.* | import kotlinx.android.synthetic.main.fragment_article.view.* | ||||||
| @@ -53,6 +56,8 @@ 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 java.util.concurrent.ExecutionException | ||||||
|  | import kotlin.collections.ArrayList | ||||||
| import kotlin.concurrent.thread | import kotlin.concurrent.thread | ||||||
|  |  | ||||||
| class ArticleFragment : Fragment() { | class ArticleFragment : Fragment() { | ||||||
| @@ -65,6 +70,7 @@ class ArticleFragment : Fragment() { | |||||||
|     private lateinit var contentSource: String |     private lateinit var contentSource: String | ||||||
|     private lateinit var contentImage: String |     private lateinit var contentImage: String | ||||||
|     private lateinit var contentTitle: String |     private lateinit var contentTitle: String | ||||||
|  |     private lateinit var allImages : ArrayList<String> | ||||||
|     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 | ||||||
| @@ -99,7 +105,7 @@ class ArticleFragment : Fragment() { | |||||||
|         db = Room.databaseBuilder( |         db = Room.databaseBuilder( | ||||||
|             context!!, |             context!!, | ||||||
|             AppDatabase::class.java, "selfoss-database" |             AppDatabase::class.java, "selfoss-database" | ||||||
|         ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).build() |         ).addMigrations(MIGRATION_1_2).addMigrations(MIGRATION_2_3).addMigrations(MIGRATION_3_4).build() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun onCreateView( |     override fun onCreateView( | ||||||
| @@ -113,9 +119,10 @@ class ArticleFragment : Fragment() { | |||||||
|  |  | ||||||
|             url = allItems[pageNumber.toInt()].getLinkDecoded() |             url = allItems[pageNumber.toInt()].getLinkDecoded() | ||||||
|             contentText = allItems[pageNumber.toInt()].content |             contentText = allItems[pageNumber.toInt()].content | ||||||
|             contentTitle = allItems[pageNumber.toInt()].title |             contentTitle = allItems[pageNumber.toInt()].getTitleDecoded() | ||||||
|             contentImage = allItems[pageNumber.toInt()].getThumbnail(activity!!) |             contentImage = allItems[pageNumber.toInt()].getThumbnail(activity!!) | ||||||
|             contentSource = allItems[pageNumber.toInt()].sourceAndDateText() |             contentSource = allItems[pageNumber.toInt()].sourceAndDateText() | ||||||
|  |             allImages = allItems[pageNumber.toInt()].getImages() | ||||||
|  |  | ||||||
|             prefs = PreferenceManager.getDefaultSharedPreferences(activity) |             prefs = PreferenceManager.getDefaultSharedPreferences(activity) | ||||||
|             editor = prefs.edit() |             editor = prefs.edit() | ||||||
| @@ -410,6 +417,47 @@ class ArticleFragment : Fragment() { | |||||||
|         rootView!!.webcontent.settings.loadWithOverviewMode = true |         rootView!!.webcontent.settings.loadWithOverviewMode = true | ||||||
|         rootView!!.webcontent.settings.javaScriptEnabled = false |         rootView!!.webcontent.settings.javaScriptEnabled = false | ||||||
|  |  | ||||||
|  |         rootView!!.webcontent.webViewClient = object : WebViewClient() { | ||||||
|  |             override fun shouldOverrideUrlLoading(view: WebView?, url : String): Boolean { | ||||||
|  |                 if (rootView!!.webcontent.hitTestResult.type != WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||||
|  |                     rootView!!.context.startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) | ||||||
|  |                 } | ||||||
|  |                 return true | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             override fun shouldInterceptRequest(view: WebView?, url: String): WebResourceResponse? { | ||||||
|  |                 val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||||
|  |                 if (url.toLowerCase().contains(".jpg") || url.toLowerCase().contains(".jpeg")) { | ||||||
|  |                     try { | ||||||
|  |                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||||
|  |                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.JPEG)) | ||||||
|  |                     }catch ( e : ExecutionException) {} | ||||||
|  |                 } | ||||||
|  |                 else if (url.toLowerCase().contains(".png")) { | ||||||
|  |                     try { | ||||||
|  |                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||||
|  |                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.PNG)) | ||||||
|  |                     }catch ( e : ExecutionException) {} | ||||||
|  |                 } | ||||||
|  |                 else if (url.toLowerCase().contains(".webp")) { | ||||||
|  |                     try { | ||||||
|  |                         val image = Glide.with(view).asBitmap().apply(glideOptions).load(url).submit().get() | ||||||
|  |                         return WebResourceResponse("image/jpg", "UTF-8", getBitmapInputStream(image, Bitmap.CompressFormat.WEBP)) | ||||||
|  |                     }catch ( e : ExecutionException) {} | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 return super.shouldInterceptRequest(view, url) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         val gestureDetector = GestureDetector(activity, object : GestureDetector.SimpleOnGestureListener() { | ||||||
|  |             override fun onSingleTapUp(e: MotionEvent?): Boolean { | ||||||
|  |                 return performClick() | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |         rootView!!.webcontent.setOnTouchListener { _, event -> gestureDetector.onTouchEvent(event)} | ||||||
|  |  | ||||||
|         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { |         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { | ||||||
|             rootView!!.webcontent.settings.layoutAlgorithm = |             rootView!!.webcontent.settings.layoutAlgorithm = | ||||||
|                     WebSettings.LayoutAlgorithm.TEXT_AUTOSIZING |                     WebSettings.LayoutAlgorithm.TEXT_AUTOSIZING | ||||||
| @@ -525,5 +573,20 @@ class ArticleFragment : Fragment() { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     fun performClick(): Boolean { | ||||||
|  |         if (rootView!!.webcontent.hitTestResult.type == WebView.HitTestResult.IMAGE_TYPE || | ||||||
|  |                 rootView!!.webcontent.hitTestResult.type == WebView.HitTestResult.SRC_IMAGE_ANCHOR_TYPE) { | ||||||
|  |  | ||||||
|  |             val position : Int = allImages.indexOf(rootView!!.webcontent.hitTestResult.extra) | ||||||
|  |  | ||||||
|  |             val intent = Intent(activity, ImageActivity::class.java) | ||||||
|  |             intent.putExtra("allImages", allImages) | ||||||
|  |             intent.putExtra("position", position) | ||||||
|  |             startActivity(intent) | ||||||
|  |             return false | ||||||
|  |         } | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,49 @@ | |||||||
|  | package apps.amine.bou.readerforselfoss.fragments | ||||||
|  |  | ||||||
|  | import android.os.Bundle | ||||||
|  | import android.view.* | ||||||
|  | import androidx.fragment.app.Fragment | ||||||
|  | import apps.amine.bou.readerforselfoss.R | ||||||
|  | import com.bumptech.glide.Glide | ||||||
|  | import com.bumptech.glide.load.engine.DiskCacheStrategy | ||||||
|  | import com.bumptech.glide.request.RequestOptions | ||||||
|  | import kotlinx.android.synthetic.main.fragment_image.view.* | ||||||
|  |  | ||||||
|  | class ImageFragment : Fragment() { | ||||||
|  |  | ||||||
|  |     private lateinit var imageUrl : String | ||||||
|  |     private val glideOptions = RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.ALL) | ||||||
|  |  | ||||||
|  |     override fun onCreate(savedInstanceState: Bundle?) { | ||||||
|  |         super.onCreate(savedInstanceState) | ||||||
|  |  | ||||||
|  |         imageUrl = arguments!!.getString("imageUrl") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { | ||||||
|  |         val view : View = inflater.inflate(R.layout.fragment_image, container, false) | ||||||
|  |  | ||||||
|  |         view.photoView.visibility = View.VISIBLE | ||||||
|  |         Glide.with(activity) | ||||||
|  |                 .asBitmap() | ||||||
|  |                 .apply(glideOptions) | ||||||
|  |                 .load(imageUrl) | ||||||
|  |                 .into(view.photoView) | ||||||
|  |  | ||||||
|  |         return view | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     companion object { | ||||||
|  |         private const val ARG_IMAGE = "imageUrl" | ||||||
|  |  | ||||||
|  |         fun newInstance( | ||||||
|  |                 imageUrl : String | ||||||
|  |         ): ImageFragment { | ||||||
|  |             val fragment = ImageFragment() | ||||||
|  |             val args = Bundle() | ||||||
|  |             args.putString(ARG_IMAGE, imageUrl) | ||||||
|  |             fragment.arguments = args | ||||||
|  |             return fragment | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -10,7 +10,7 @@ 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, ActionEntity::class], version = 3) | @Database(entities = [TagEntity::class, SourceEntity::class, ItemEntity::class, ActionEntity::class], version = 4) | ||||||
| abstract class AppDatabase : RoomDatabase() { | abstract class AppDatabase : RoomDatabase() { | ||||||
|     abstract fun drawerDataDao(): DrawerDataDao |     abstract fun drawerDataDao(): DrawerDataDao | ||||||
|  |  | ||||||
|   | |||||||
| @@ -20,9 +20,9 @@ data class ItemEntity( | |||||||
|     @ColumnInfo(name = "starred") |     @ColumnInfo(name = "starred") | ||||||
|     var starred: Boolean, |     var starred: Boolean, | ||||||
|     @ColumnInfo(name = "thumbnail") |     @ColumnInfo(name = "thumbnail") | ||||||
|     val thumbnail: String, |     val thumbnail: String?, | ||||||
|     @ColumnInfo(name = "icon") |     @ColumnInfo(name = "icon") | ||||||
|     val icon: String, |     val icon: String?, | ||||||
|     @ColumnInfo(name = "link") |     @ColumnInfo(name = "link") | ||||||
|     val link: String, |     val link: String, | ||||||
|     @ColumnInfo(name = "sourcetitle") |     @ColumnInfo(name = "sourcetitle") | ||||||
|   | |||||||
| @@ -14,3 +14,21 @@ val MIGRATION_2_3: Migration = object : Migration(2, 3) { | |||||||
|         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`))") |         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`))") | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | val MIGRATION_3_4: Migration = object : Migration(3, 4) { | ||||||
|  |     override fun migrate(database: SupportSQLiteDatabase) { | ||||||
|  |         // @see https://stackoverflow.com/questions/57392015/how-to-migrate-not-null-table-column-into-null-in-android-room-database | ||||||
|  |         // Create the new table | ||||||
|  |         database.execSQL("CREATE TABLE IF NOT EXISTS `itemstmp` (`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, `icon` TEXT, `link` TEXT NOT NULL, `sourcetitle` TEXT NOT NULL, `tags` TEXT NOT NULL, PRIMARY KEY(`id`))") | ||||||
|  |  | ||||||
|  |         // Copy the data | ||||||
|  |         database.execSQL( | ||||||
|  |                 "INSERT INTO itemstmp (`id`, `datetime`, `title`, `content`, `unread`, `starred`, `thumbnail`, `icon`, `link`, `sourcetitle`, `tags`) SELECT `id`, `datetime`, `title`, `content`, `unread`, `starred`, `thumbnail`, `icon`, `link`, `sourcetitle`, `tags` FROM items") | ||||||
|  |  | ||||||
|  |         // Remove the old table | ||||||
|  |         database.execSQL("DROP TABLE items") | ||||||
|  |  | ||||||
|  |         // Change the table name to the correct one | ||||||
|  |         database.execSQL("ALTER TABLE itemstmp RENAME TO items") | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -195,16 +195,6 @@ public class SettingsActivity extends AppCompatPreferenceActivity { | |||||||
|             }); |             }); | ||||||
|  |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public boolean onOptionsItemSelected(MenuItem item) { |  | ||||||
|             int id = item.getItemId(); |  | ||||||
|             if (id == android.R.id.home) { |  | ||||||
|                 getActivity().finish(); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|             return super.onOptionsItemSelected(item); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @TargetApi(Build.VERSION_CODES.HONEYCOMB) |     @TargetApi(Build.VERSION_CODES.HONEYCOMB) | ||||||
| @@ -245,16 +235,6 @@ public class SettingsActivity extends AppCompatPreferenceActivity { | |||||||
|                     } |                     } | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public boolean onOptionsItemSelected(MenuItem item) { |  | ||||||
|             int id = item.getItemId(); |  | ||||||
|             if (id == android.R.id.home) { |  | ||||||
|                 getActivity().finish(); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|             return super.onOptionsItemSelected(item); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -298,16 +278,6 @@ public class SettingsActivity extends AppCompatPreferenceActivity { | |||||||
|                 } |                 } | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public boolean onOptionsItemSelected(MenuItem item) { |  | ||||||
|             int id = item.getItemId(); |  | ||||||
|             if (id == android.R.id.home) { |  | ||||||
|                 getActivity().finish(); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|             return super.onOptionsItemSelected(item); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @TargetApi(Build.VERSION_CODES.HONEYCOMB) |     @TargetApi(Build.VERSION_CODES.HONEYCOMB) | ||||||
| @@ -322,10 +292,7 @@ public class SettingsActivity extends AppCompatPreferenceActivity { | |||||||
|         @Override |         @Override | ||||||
|         public boolean onOptionsItemSelected(MenuItem item) { |         public boolean onOptionsItemSelected(MenuItem item) { | ||||||
|             int id = item.getItemId(); |             int id = item.getItemId(); | ||||||
|             if (id == android.R.id.home) { |             if (id == R.id.clear) { | ||||||
|                 getActivity().finish(); |  | ||||||
|                 return true; |  | ||||||
|             } else if (id == R.id.clear) { |  | ||||||
|                 SharedPreferences pref = PreferenceManager.getDefaultSharedPreferences(getActivity()); |                 SharedPreferences pref = PreferenceManager.getDefaultSharedPreferences(getActivity()); | ||||||
|                 SharedPreferences.Editor editor = pref.edit(); |                 SharedPreferences.Editor editor = pref.edit(); | ||||||
|                 editor.remove("color_primary"); |                 editor.remove("color_primary"); | ||||||
| @@ -334,7 +301,7 @@ public class SettingsActivity extends AppCompatPreferenceActivity { | |||||||
|                 editor.remove("color_accent_dark"); |                 editor.remove("color_accent_dark"); | ||||||
|                 editor.remove("dark_theme"); |                 editor.remove("dark_theme"); | ||||||
|                 editor.apply(); |                 editor.apply(); | ||||||
|                 getActivity().finish(); |                 getActivity().recreate(); | ||||||
|             } |             } | ||||||
|             return super.onOptionsItemSelected(item); |             return super.onOptionsItemSelected(item); | ||||||
|         } |         } | ||||||
| @@ -353,16 +320,6 @@ public class SettingsActivity extends AppCompatPreferenceActivity { | |||||||
|             addPreferencesFromResource(R.xml.pref_offline); |             addPreferencesFromResource(R.xml.pref_offline); | ||||||
|             setHasOptionsMenu(true); |             setHasOptionsMenu(true); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public boolean onOptionsItemSelected(MenuItem item) { |  | ||||||
|             int id = item.getItemId(); |  | ||||||
|             if (id == android.R.id.home) { |  | ||||||
|                 getActivity().finish(); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|             return super.onOptionsItemSelected(item); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @TargetApi(Build.VERSION_CODES.HONEYCOMB) |     @TargetApi(Build.VERSION_CODES.HONEYCOMB) | ||||||
| @@ -373,16 +330,6 @@ public class SettingsActivity extends AppCompatPreferenceActivity { | |||||||
|             addPreferencesFromResource(R.xml.pref_experimental); |             addPreferencesFromResource(R.xml.pref_experimental); | ||||||
|             setHasOptionsMenu(true); |             setHasOptionsMenu(true); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public boolean onOptionsItemSelected(MenuItem item) { |  | ||||||
|             int id = item.getItemId(); |  | ||||||
|             if (id == android.R.id.home) { |  | ||||||
|                 getActivity().finish(); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|             return super.onOptionsItemSelected(item); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -390,7 +337,7 @@ public class SettingsActivity extends AppCompatPreferenceActivity { | |||||||
|     public boolean onOptionsItemSelected(MenuItem item) { |     public boolean onOptionsItemSelected(MenuItem item) { | ||||||
|         int id = item.getItemId(); |         int id = item.getItemId(); | ||||||
|         if (id == android.R.id.home) { |         if (id == android.R.id.home) { | ||||||
|             finish(); |             super.onBackPressed(); | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|         return super.onOptionsItemSelected(item); |         return super.onOptionsItemSelected(item); | ||||||
|   | |||||||
| @@ -14,6 +14,9 @@ import com.bumptech.glide.load.model.GlideUrl | |||||||
| import com.bumptech.glide.load.model.LazyHeaders | import com.bumptech.glide.load.model.LazyHeaders | ||||||
| import com.bumptech.glide.request.RequestOptions | import com.bumptech.glide.request.RequestOptions | ||||||
| import com.bumptech.glide.request.target.BitmapImageViewTarget | import com.bumptech.glide.request.target.BitmapImageViewTarget | ||||||
|  | import java.io.ByteArrayInputStream | ||||||
|  | import java.io.ByteArrayOutputStream | ||||||
|  | import java.io.InputStream | ||||||
|  |  | ||||||
| fun Context.bitmapCenterCrop(config: Config, url: String, iv: ImageView) = | fun Context.bitmapCenterCrop(config: Config, url: String, iv: ImageView) = | ||||||
|     Glide.with(this) |     Glide.with(this) | ||||||
| @@ -56,4 +59,11 @@ fun RequestManager.loadMaybeBasicAuth(config: Config, url: String): RequestBuild | |||||||
|     } |     } | ||||||
|     val glideUrl = GlideUrl(url, builder.build()) |     val glideUrl = GlideUrl(url, builder.build()) | ||||||
|     return this.load(glideUrl) |     return this.load(glideUrl) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun getBitmapInputStream(bitmap:Bitmap,compressFormat: Bitmap.CompressFormat): InputStream { | ||||||
|  |     val byteArrayOutputStream = ByteArrayOutputStream() | ||||||
|  |     bitmap.compress(compressFormat, 80, byteArrayOutputStream) | ||||||
|  |     val bitmapData: ByteArray = byteArrayOutputStream.toByteArray() | ||||||
|  |     return ByteArrayInputStream(bitmapData) | ||||||
| } | } | ||||||
| @@ -61,7 +61,7 @@ fun Item.toEntity(): ItemEntity = | |||||||
|     ItemEntity( |     ItemEntity( | ||||||
|         this.id, |         this.id, | ||||||
|         this.datetime, |         this.datetime, | ||||||
|         this.title, |         this.getTitleDecoded(), | ||||||
|         this.content, |         this.content, | ||||||
|         this.unread, |         this.unread, | ||||||
|         this.starred, |         this.starred, | ||||||
|   | |||||||
							
								
								
									
										33
									
								
								app/src/main/res/layout/activity_image.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								app/src/main/res/layout/activity_image.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|  |     android:layout_width="match_parent" | ||||||
|  |     android:layout_height="match_parent"> | ||||||
|  |  | ||||||
|  |     <com.google.android.material.appbar.AppBarLayout | ||||||
|  |         android:id="@+id/appBarLayout" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="wrap_content" | ||||||
|  |         app:layout_constraintEnd_toEndOf="parent" | ||||||
|  |         app:layout_constraintStart_toStartOf="parent" | ||||||
|  |         app:layout_constraintTop_toTopOf="parent"> | ||||||
|  |  | ||||||
|  |         <androidx.appcompat.widget.Toolbar | ||||||
|  |             android:id="@+id/toolBar" | ||||||
|  |             android:layout_width="match_parent" | ||||||
|  |             android:layout_height="?attr/actionBarSize" | ||||||
|  |             app:popupTheme="?attr/toolbarPopupTheme" | ||||||
|  |             app:theme="@style/ToolBarStyle" /> | ||||||
|  |  | ||||||
|  |     </com.google.android.material.appbar.AppBarLayout> | ||||||
|  |  | ||||||
|  |     <androidx.viewpager.widget.ViewPager | ||||||
|  |         android:id="@+id/pager" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="0dp" | ||||||
|  |         app:layout_constraintBottom_toBottomOf="parent" | ||||||
|  |         app:layout_constraintEnd_toEndOf="parent" | ||||||
|  |         app:layout_constraintStart_toStartOf="parent" | ||||||
|  |         app:layout_constraintTop_toBottomOf="@+id/appBarLayout" /> | ||||||
|  |  | ||||||
|  | </androidx.constraintlayout.widget.ConstraintLayout> | ||||||
							
								
								
									
										16
									
								
								app/src/main/res/layout/fragment_image.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								app/src/main/res/layout/fragment_image.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|  |     android:layout_width="match_parent" | ||||||
|  |     android:layout_height="match_parent"> | ||||||
|  |  | ||||||
|  |     <com.github.chrisbanes.photoview.PhotoView | ||||||
|  |         android:id="@+id/photoView" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:layout_centerVertical="true" | ||||||
|  |         android:layout_centerHorizontal="true" | ||||||
|  |         android:background="@android:color/black" | ||||||
|  |         app:srcCompat="@android:drawable/screen_background_dark" /> | ||||||
|  |  | ||||||
|  | </androidx.coordinatorlayout.widget.CoordinatorLayout> | ||||||
| @@ -3,14 +3,14 @@ | |||||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|     xmlns:tools="http://schemas.android.com/tools" |     xmlns:tools="http://schemas.android.com/tools" | ||||||
|     android:layout_width="match_parent" |     android:layout_width="match_parent" | ||||||
|     android:layout_height="wrap_content" |     android:layout_height="88dp"> | ||||||
|     android:minHeight="88dp"> |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     <ImageView |     <ImageView | ||||||
|         android:id="@+id/itemImage" |         android:id="@+id/itemImage" | ||||||
|         android:layout_width="88dp" |         android:layout_width="46dp" | ||||||
|         android:layout_height="88dp" |         android:layout_height="46dp" | ||||||
|  |         android:layout_marginStart="8dp" | ||||||
|  |         android:layout_marginTop="21dp" | ||||||
|         app:layout_constraintStart_toStartOf="parent" |         app:layout_constraintStart_toStartOf="parent" | ||||||
|         app:layout_constraintTop_toTopOf="parent" /> |         app:layout_constraintTop_toTopOf="parent" /> | ||||||
|  |  | ||||||
| @@ -18,9 +18,9 @@ | |||||||
|         android:id="@+id/title" |         android:id="@+id/title" | ||||||
|         android:layout_width="0dp" |         android:layout_width="0dp" | ||||||
|         android:layout_height="wrap_content" |         android:layout_height="wrap_content" | ||||||
|         android:layout_marginEnd="16dp" |         android:layout_marginStart="8dp" | ||||||
|         android:layout_marginStart="16dp" |  | ||||||
|         android:layout_marginTop="8dp" |         android:layout_marginTop="8dp" | ||||||
|  |         android:layout_marginEnd="16dp" | ||||||
|         android:ellipsize="end" |         android:ellipsize="end" | ||||||
|         android:fontFamily="sans-serif" |         android:fontFamily="sans-serif" | ||||||
|         android:gravity="start" |         android:gravity="start" | ||||||
| @@ -39,16 +39,17 @@ | |||||||
|         android:id="@+id/sourceTitleAndDate" |         android:id="@+id/sourceTitleAndDate" | ||||||
|         android:layout_width="0dp" |         android:layout_width="0dp" | ||||||
|         android:layout_height="wrap_content" |         android:layout_height="wrap_content" | ||||||
|         android:layout_marginStart="16dp" |         android:layout_marginStart="8dp" | ||||||
|         android:layout_marginTop="8dp" |         android:layout_marginTop="66dp" | ||||||
|         android:layout_marginEnd="16dp" |         android:layout_marginEnd="16dp" | ||||||
|         android:gravity="start" |         android:gravity="start" | ||||||
|  |         android:maxLines="1" | ||||||
|         android:textAlignment="viewStart" |         android:textAlignment="viewStart" | ||||||
|         android:textSize="14sp" |         android:textSize="14sp" | ||||||
|         app:layout_constraintEnd_toEndOf="parent" |         app:layout_constraintEnd_toEndOf="parent" | ||||||
|         app:layout_constraintHorizontal_bias="0.0" |         app:layout_constraintHorizontal_bias="0.0" | ||||||
|         app:layout_constraintStart_toEndOf="@+id/itemImage" |         app:layout_constraintStart_toEndOf="@+id/itemImage" | ||||||
|         app:layout_constraintTop_toBottomOf="@+id/title" |         app:layout_constraintTop_toTopOf="parent" | ||||||
|         tools:text="Google Actualité Il y a 5h" /> |         tools:text="Google Actualité Il y a 5h" /> | ||||||
|  |  | ||||||
| </androidx.constraintlayout.widget.ConstraintLayout> | </androidx.constraintlayout.widget.ConstraintLayout> | ||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Lector per a Selfoss"</string> |     <string name="app_name">"Lector per a Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Inicia la sessió"</string> |     <string name="title_activity_login">"Inicia la sessió"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Reader für selfoss"</string> |     <string name="app_name">"Reader für selfoss"</string> | ||||||
|     <string name="title_activity_login">"Anmelden"</string> |     <string name="title_activity_login">"Anmelden"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Lector para Selfoss"</string> |     <string name="app_name">"Lector para Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Iniciar sesión"</string> |     <string name="title_activity_login">"Iniciar sesión"</string> | ||||||
|   | |||||||
							
								
								
									
										163
									
								
								app/src/main/res/values-fa-rIR/strings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								app/src/main/res/values-fa-rIR/strings.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,163 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|  |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|  |     <string name="title_activity_login">"Log in"</string> | ||||||
|  |     <string name="prompt_password">"Password"</string> | ||||||
|  |     <string name="prompt_http_password">"HTTP Password"</string> | ||||||
|  |     <string name="action_sign_in">"Go"</string> | ||||||
|  |     <string name="error_invalid_password">"Password not long enough"</string> | ||||||
|  |     <string name="error_field_required">"Field required"</string> | ||||||
|  |     <string name="prompt_url">"Url"</string> | ||||||
|  |     <string name="withLoginSwitch">"Login required ?"</string> | ||||||
|  |     <string name="withHttpLoginSwitch">"HTTP Login required ?"</string> | ||||||
|  |     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> | ||||||
|  |     <string name="prompt_login">"Username"</string> | ||||||
|  |     <string name="prompt_http_login">"HTTP Username"</string> | ||||||
|  |     <string name="label_share">"Share"</string> | ||||||
|  |     <string name="readAll">"Read all"</string> | ||||||
|  |     <string name="action_disconnect">"Disconnect"</string> | ||||||
|  |     <string name="title_activity_settings">"Settings"</string> | ||||||
|  |     <string name="pref_header_general">"General"</string> | ||||||
|  |     <string name="add_source_hint_tags">"Tag1, Tag2, Tag3"</string> | ||||||
|  |     <string name="add_source_hint_url">"Link"</string> | ||||||
|  |     <string name="add_source_hint_name">"Name"</string> | ||||||
|  |     <string name="add_source">"Add a source"</string> | ||||||
|  |     <string name="add_source_save">"Save"</string> | ||||||
|  |     <string name="wrong_infos">"Check your details again."</string> | ||||||
|  |     <string name="all_posts_not_read">"All posts weren't read"</string> | ||||||
|  |     <string name="all_posts_read">"All posts were read"</string> | ||||||
|  |     <string name="cant_get_favs">"Can't get favorites"</string> | ||||||
|  |     <string name="cant_get_new_elements">"Can't get new articles"</string> | ||||||
|  |     <string name="cant_get_read">"Can't get read articles"</string> | ||||||
|  |     <string name="nothing_here">"Nothing here"</string> | ||||||
|  |     <string name="tab_new">"New"</string> | ||||||
|  |     <string name="tab_read">"All"</string> | ||||||
|  |     <string name="tab_favs">"Favorites"</string> | ||||||
|  |     <string name="action_about">"About"</string> | ||||||
|  |     <string name="marked_as_read">"Item read"</string> | ||||||
|  |     <string name="marked_as_unread">"Item unread"</string> | ||||||
|  |     <string name="undo_string">"Undo"</string> | ||||||
|  |     <string name="addStringNoUrl">"Log in to add sources."</string> | ||||||
|  |     <string name="cant_get_sources">"Can't get sources list."</string> | ||||||
|  |     <string name="cant_create_source">"Can't create source."</string> | ||||||
|  |     <string name="cant_get_spouts">"Can't get spouts list."</string> | ||||||
|  |     <string name="form_not_complete">"The form is not complete"</string> | ||||||
|  |     <string name="pref_header_links">"Links"</string> | ||||||
|  |     <string name="issue_tracker_link">"Issue Tracker"</string> | ||||||
|  |     <string name="issue_tracker_summary">"Report a bug or ask for a new feature"</string> | ||||||
|  |     <string name="warning_wrong_url">"WARNING"</string> | ||||||
|  |     <string name="pref_switch_card_view_title">"Card View"</string> | ||||||
|  |     <string name="cant_mark_favortie">"Can't mark article as favorite"</string> | ||||||
|  |     <string name="cant_unmark_favortie">"Can't remove item from favorite"</string> | ||||||
|  |     <string name="share">"Share"</string> | ||||||
|  |     <string name="rating_prompt_title">"Enjoying the app ?"</string> | ||||||
|  |     <string name="rating_prompt_yes">"Yes !"</string> | ||||||
|  |     <string name="rating_prompt_no">"Not really …"</string> | ||||||
|  |     <string name="rating_prompt_feedback_title">"Can you tell us why ?"</string> | ||||||
|  |     <string name="rating_prompt_feedback_yes">"OK !"</string> | ||||||
|  |     <string name="rating_prompt_feedback_no">"Not now."</string> | ||||||
|  |     <string name="rating_prompt_rating_title">"Great ! Can you rate us on the Store ?"</string> | ||||||
|  |     <string name="rating_prompt_rating_yes">"Sure !"</string> | ||||||
|  |     <string name="rating_prompt_rating_no">"Not right now."</string> | ||||||
|  |     <string name="rating_prompt_thanks">"Thanks, your feedback help enhance the app !"</string> | ||||||
|  |     <string name="switch_unread_count">"Display the unread count as a badge for the bottom bar."</string> | ||||||
|  |     <string name="switch_unread_count_title">"Display unread count"</string> | ||||||
|  |     <string name="display_all_counts_title">"Display count for favorite and read"</string> | ||||||
|  |     <string name="text_wrong_url">"You seem to be trying to use an invalid URL. Make sure it is correct, and if the problem persists, contact me (via the store contact link). Please note that the app needs you to be using Selfoss. You can't access RSS feeds without it."</string> | ||||||
|  |     <string name="pref_general_internal_browser_title">"Open links inside the app"</string> | ||||||
|  |     <string name="pref_general_internal_browser_on">"Articles will open inside the app"</string> | ||||||
|  |     <string name="pref_general_internal_browser_off">"Articles will open with your default browser"</string> | ||||||
|  |     <string name="prefer_article_viewer_title">"Use the article viewer"</string> | ||||||
|  |     <string name="prefer_article_viewer_on">"Will use the article viewer instead of the internal browser"</string> | ||||||
|  |     <string name="prefer_article_viewer_off">"Will use the internal browser instead of the article viewer"</string> | ||||||
|  |     <string name="pref_general_category_links">"Link handling"</string> | ||||||
|  |     <string name="pref_general_category_displaying">"Displaying"</string> | ||||||
|  |     <string name="pref_switch_card_view_on">"The articles will be displayed as cards"</string> | ||||||
|  |     <string name="pref_switch_card_view_off">"The articles will be displayed as a list"</string> | ||||||
|  |     <string name="menu_home_refresh">"Update remote"</string> | ||||||
|  |     <string name="refresh_success_response">"The remote is updated, you can now reload the articles list"</string> | ||||||
|  |     <string name="refresh_failer_message">"The update didn't work, try again later, or check your selfoss logs."</string> | ||||||
|  |     <string name="refresh_in_progress">"Refresh in progress"</string> | ||||||
|  |     <string name="card_height_title">Full height cards</string> | ||||||
|  |     <string name="card_height_on">Cards height will adjust to its content</string> | ||||||
|  |     <string name="card_height_off">Card height will be fixed</string> | ||||||
|  |     <string name="source_code">Source code</string> | ||||||
|  |     <string name="cant_mark_read">Can\'t mark article as read</string> | ||||||
|  |     <string name="cant_mark_unread">Can\'t mark article as unread</string> | ||||||
|  |     <string name="drawer_error_loading_tags">Error loading tags…</string> | ||||||
|  |     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||||
|  |     <string name="drawer_item_filters">Filters</string> | ||||||
|  |     <string name="drawer_action_clear">clear</string> | ||||||
|  |     <string name="drawer_item_tags">Tags</string> | ||||||
|  |     <string name="drawer_item_sources">Sources</string> | ||||||
|  |     <string name="drawer_action_edit">edit</string> | ||||||
|  |     <string name="no_tags_loaded">No tags loaded</string> | ||||||
|  |     <string name="no_sources_loaded">No sources loaded</string> | ||||||
|  |     <string name="drawer_loading">Loading …</string> | ||||||
|  |     <string name="menu_home_search">Search</string> | ||||||
|  |     <string name="can_delete_source">Can\'t delete the source…</string> | ||||||
|  |     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||||
|  |     <string name="pref_header_theme">Themes</string> | ||||||
|  |     <string name="default_theme">Default</string> | ||||||
|  |     <string name="default_dark_theme">Default/Dark</string> | ||||||
|  |     <string name="pref_header_debug">Debug</string> | ||||||
|  |     <string name="self_hosted_cert_switch">Using a self hosted certificate ?</string> | ||||||
|  |     <string name="self_signed_cert_warning">Due to security reasons, self signed certificates are not supported by default. By activating this, I\'ll not be responsible of any security problem you encounter.</string> | ||||||
|  |     <string name="pref_selfoss_category">Selfoss Api</string> | ||||||
|  |     <string name="pref_api_items_number_title">Loaded items number</string> | ||||||
|  |     <string name="pref_hidden_tags">Hidden Tags</string> | ||||||
|  |     <string name="summary_debug_identifier">Debug identifier</string> | ||||||
|  |     <string name="unique_id_to_clipboard">Identifier copied to your clipboard</string> | ||||||
|  |     <string name="display_header_drawer_summary">Display a header with the selfoss instance url on the lateral drawer.</string> | ||||||
|  |     <string name="display_header_drawer_title">Account header</string> | ||||||
|  |     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||||
|  |     <string name="translation">Translation</string> | ||||||
|  |     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> | ||||||
|  |     <string name="drawer_report_bug">Report a bug</string> | ||||||
|  |     <string name="items_number_should_be_number">The items number should be an integer.</string> | ||||||
|  |     <string name="reader_action_more">Read more</string> | ||||||
|  |     <string name="reader_action_open">Open in browser</string> | ||||||
|  |     <string name="reader_action_share">Share</string> | ||||||
|  |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
|  |     <string name="add_to_favs_reader">Add to favorites</string> | ||||||
|  |     <string name="remove_to_favs_reader">Remove from favorites</string> | ||||||
|  |     <string name="pref_content_reader_font_size">Article reader content font size</string> | ||||||
|  |     <string name="pref_header_viewer">Article viewer</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="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_acra_alwaysaccept">Automatically send crash reports</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_debug_crash_reports">Crash reports</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="drawer_item_hidden_tags">Hidden Tags</string> | ||||||
|  |     <string name="unmark">Mark item as unread</string> | ||||||
|  |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|  |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|  |     <string name="pref_switch_items_caching_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="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> | ||||||
|  |     <string name="new_items_channel_sync">New items notification</string> | ||||||
|  |     <string name="new_items_notification_title">New items !</string> | ||||||
|  |     <string name="new_items_notification_text">%1$d new items loaded.</string> | ||||||
|  |     <string name="pref_switch_notify_new_items">Notify on new items synced.</string> | ||||||
|  |     <string name="shortcut_offline">Offline</string> | ||||||
|  |     <string name="pref_api_timeout">Api Timeout</string> | ||||||
|  |     <string name="pref_header_experimental">Experimental</string> | ||||||
|  |     <string name="webview_dialog_issue_message">Webview not available. Disabling the article viewer to avoid any future crashes. Will load articles inside of your browser from now on.</string> | ||||||
|  |     <string name="webview_dialog_issue_title">Webview issue</string> | ||||||
|  |     <string name="reader_text_align_left">Align left</string> | ||||||
|  |     <string name="reader_text_align_justify">Justify</string> | ||||||
|  |     <string name="settings_reader_font">Reader font</string> | ||||||
|  | </resources> | ||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Login"</string> |     <string name="title_activity_login">"Login"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Lector para Selfoss"</string> |     <string name="app_name">"Lector para Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Acceder"</string> |     <string name="title_activity_login">"Acceder"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Masuk"</string> |     <string name="title_activity_login">"Masuk"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Lettore RSS per Selfoss"</string> |     <string name="app_name">"Lettore RSS per Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Accedi"</string> |     <string name="title_activity_login">"Accedi"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"로그인"</string> |     <string name="title_activity_login">"로그인"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Selfoss Reader"</string> |     <string name="app_name">"Selfoss Reader"</string> | ||||||
|     <string name="title_activity_login">"Inloggen"</string> |     <string name="title_activity_login">"Inloggen"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Reader for Selfoss"</string> |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Entrar"</string> |     <string name="title_activity_login">"Entrar"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Leitor para Selfoss"</string> |     <string name="app_name">"Leitor para Selfoss"</string> | ||||||
|     <string name="title_activity_login">"Iniciar sessão"</string> |     <string name="title_activity_login">"Iniciar sessão"</string> | ||||||
|   | |||||||
							
								
								
									
										163
									
								
								app/src/main/res/values-si-rLK/strings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								app/src/main/res/values-si-rLK/strings.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,163 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|  |     <string name="app_name">"Reader for Selfoss"</string> | ||||||
|  |     <string name="title_activity_login">"පිවිසෙන්න"</string> | ||||||
|  |     <string name="prompt_password">"මුර පදය"</string> | ||||||
|  |     <string name="prompt_http_password">"HTTP Password"</string> | ||||||
|  |     <string name="action_sign_in">"Go"</string> | ||||||
|  |     <string name="error_invalid_password">"Password not long enough"</string> | ||||||
|  |     <string name="error_field_required">"Field required"</string> | ||||||
|  |     <string name="prompt_url">"Url"</string> | ||||||
|  |     <string name="withLoginSwitch">"Login required ?"</string> | ||||||
|  |     <string name="withHttpLoginSwitch">"HTTP Login required ?"</string> | ||||||
|  |     <string name="login_url_problem">"Oops. You may need to add a \"/\" at the end of the url."</string> | ||||||
|  |     <string name="prompt_login">"පරිශීලක නාමය"</string> | ||||||
|  |     <string name="prompt_http_login">"HTTP Username"</string> | ||||||
|  |     <string name="label_share">"Share"</string> | ||||||
|  |     <string name="readAll">"Read all"</string> | ||||||
|  |     <string name="action_disconnect">"Disconnect"</string> | ||||||
|  |     <string name="title_activity_settings">"සැකසුම්"</string> | ||||||
|  |     <string name="pref_header_general">"General"</string> | ||||||
|  |     <string name="add_source_hint_tags">"Tag1, Tag2, Tag3"</string> | ||||||
|  |     <string name="add_source_hint_url">"Link"</string> | ||||||
|  |     <string name="add_source_hint_name">"නම"</string> | ||||||
|  |     <string name="add_source">"Add a source"</string> | ||||||
|  |     <string name="add_source_save">"සුරකින්න"</string> | ||||||
|  |     <string name="wrong_infos">"Check your details again."</string> | ||||||
|  |     <string name="all_posts_not_read">"All posts weren't read"</string> | ||||||
|  |     <string name="all_posts_read">"All posts were read"</string> | ||||||
|  |     <string name="cant_get_favs">"Can't get favorites"</string> | ||||||
|  |     <string name="cant_get_new_elements">"Can't get new articles"</string> | ||||||
|  |     <string name="cant_get_read">"Can't get read articles"</string> | ||||||
|  |     <string name="nothing_here">"Nothing here"</string> | ||||||
|  |     <string name="tab_new">"New"</string> | ||||||
|  |     <string name="tab_read">"සියල්ල"</string> | ||||||
|  |     <string name="tab_favs">"Favorites"</string> | ||||||
|  |     <string name="action_about">"මේ ගැන"</string> | ||||||
|  |     <string name="marked_as_read">"Item read"</string> | ||||||
|  |     <string name="marked_as_unread">"Item unread"</string> | ||||||
|  |     <string name="undo_string">"Undo"</string> | ||||||
|  |     <string name="addStringNoUrl">"Log in to add sources."</string> | ||||||
|  |     <string name="cant_get_sources">"Can't get sources list."</string> | ||||||
|  |     <string name="cant_create_source">"Can't create source."</string> | ||||||
|  |     <string name="cant_get_spouts">"Can't get spouts list."</string> | ||||||
|  |     <string name="form_not_complete">"The form is not complete"</string> | ||||||
|  |     <string name="pref_header_links">"Links"</string> | ||||||
|  |     <string name="issue_tracker_link">"Issue Tracker"</string> | ||||||
|  |     <string name="issue_tracker_summary">"Report a bug or ask for a new feature"</string> | ||||||
|  |     <string name="warning_wrong_url">"WARNING"</string> | ||||||
|  |     <string name="pref_switch_card_view_title">"Card View"</string> | ||||||
|  |     <string name="cant_mark_favortie">"Can't mark article as favorite"</string> | ||||||
|  |     <string name="cant_unmark_favortie">"Can't remove item from favorite"</string> | ||||||
|  |     <string name="share">"Share"</string> | ||||||
|  |     <string name="rating_prompt_title">"Enjoying the app ?"</string> | ||||||
|  |     <string name="rating_prompt_yes">"Yes !"</string> | ||||||
|  |     <string name="rating_prompt_no">"Not really …"</string> | ||||||
|  |     <string name="rating_prompt_feedback_title">"Can you tell us why ?"</string> | ||||||
|  |     <string name="rating_prompt_feedback_yes">"OK !"</string> | ||||||
|  |     <string name="rating_prompt_feedback_no">"Not now."</string> | ||||||
|  |     <string name="rating_prompt_rating_title">"Great ! Can you rate us on the Store ?"</string> | ||||||
|  |     <string name="rating_prompt_rating_yes">"Sure !"</string> | ||||||
|  |     <string name="rating_prompt_rating_no">"Not right now."</string> | ||||||
|  |     <string name="rating_prompt_thanks">"Thanks, your feedback help enhance the app !"</string> | ||||||
|  |     <string name="switch_unread_count">"Display the unread count as a badge for the bottom bar."</string> | ||||||
|  |     <string name="switch_unread_count_title">"Display unread count"</string> | ||||||
|  |     <string name="display_all_counts_title">"Display count for favorite and read"</string> | ||||||
|  |     <string name="text_wrong_url">"You seem to be trying to use an invalid URL. Make sure it is correct, and if the problem persists, contact me (via the store contact link). Please note that the app needs you to be using Selfoss. You can't access RSS feeds without it."</string> | ||||||
|  |     <string name="pref_general_internal_browser_title">"Open links inside the app"</string> | ||||||
|  |     <string name="pref_general_internal_browser_on">"Articles will open inside the app"</string> | ||||||
|  |     <string name="pref_general_internal_browser_off">"Articles will open with your default browser"</string> | ||||||
|  |     <string name="prefer_article_viewer_title">"Use the article viewer"</string> | ||||||
|  |     <string name="prefer_article_viewer_on">"Will use the article viewer instead of the internal browser"</string> | ||||||
|  |     <string name="prefer_article_viewer_off">"Will use the internal browser instead of the article viewer"</string> | ||||||
|  |     <string name="pref_general_category_links">"Link handling"</string> | ||||||
|  |     <string name="pref_general_category_displaying">"Displaying"</string> | ||||||
|  |     <string name="pref_switch_card_view_on">"The articles will be displayed as cards"</string> | ||||||
|  |     <string name="pref_switch_card_view_off">"The articles will be displayed as a list"</string> | ||||||
|  |     <string name="menu_home_refresh">"Update remote"</string> | ||||||
|  |     <string name="refresh_success_response">"The remote is updated, you can now reload the articles list"</string> | ||||||
|  |     <string name="refresh_failer_message">"The update didn't work, try again later, or check your selfoss logs."</string> | ||||||
|  |     <string name="refresh_in_progress">"Refresh in progress"</string> | ||||||
|  |     <string name="card_height_title">Full height cards</string> | ||||||
|  |     <string name="card_height_on">Cards height will adjust to its content</string> | ||||||
|  |     <string name="card_height_off">Card height will be fixed</string> | ||||||
|  |     <string name="source_code">Source code</string> | ||||||
|  |     <string name="cant_mark_read">Can\'t mark article as read</string> | ||||||
|  |     <string name="cant_mark_unread">Can\'t mark article as unread</string> | ||||||
|  |     <string name="drawer_error_loading_tags">Error loading tags…</string> | ||||||
|  |     <string name="drawer_error_loading_sources">Error loading sources…</string> | ||||||
|  |     <string name="drawer_item_filters">Filters</string> | ||||||
|  |     <string name="drawer_action_clear">clear</string> | ||||||
|  |     <string name="drawer_item_tags">Tags</string> | ||||||
|  |     <string name="drawer_item_sources">Sources</string> | ||||||
|  |     <string name="drawer_action_edit">edit</string> | ||||||
|  |     <string name="no_tags_loaded">No tags loaded</string> | ||||||
|  |     <string name="no_sources_loaded">No sources loaded</string> | ||||||
|  |     <string name="drawer_loading">Loading …</string> | ||||||
|  |     <string name="menu_home_search">Search</string> | ||||||
|  |     <string name="can_delete_source">Can\'t delete the source…</string> | ||||||
|  |     <string name="base_url_error">There was an issue when trying to communicate with your Selfoss Instance. If the issue persists, please get in touch with me.</string> | ||||||
|  |     <string name="pref_header_theme">Themes</string> | ||||||
|  |     <string name="default_theme">Default</string> | ||||||
|  |     <string name="default_dark_theme">Default/Dark</string> | ||||||
|  |     <string name="pref_header_debug">Debug</string> | ||||||
|  |     <string name="self_hosted_cert_switch">Using a self hosted certificate ?</string> | ||||||
|  |     <string name="self_signed_cert_warning">Due to security reasons, self signed certificates are not supported by default. By activating this, I\'ll not be responsible of any security problem you encounter.</string> | ||||||
|  |     <string name="pref_selfoss_category">Selfoss Api</string> | ||||||
|  |     <string name="pref_api_items_number_title">Loaded items number</string> | ||||||
|  |     <string name="pref_hidden_tags">Hidden Tags</string> | ||||||
|  |     <string name="summary_debug_identifier">Debug identifier</string> | ||||||
|  |     <string name="unique_id_to_clipboard">Identifier copied to your clipboard</string> | ||||||
|  |     <string name="display_header_drawer_summary">Display a header with the selfoss instance url on the lateral drawer.</string> | ||||||
|  |     <string name="display_header_drawer_title">Account header</string> | ||||||
|  |     <string name="pref_general_infinite_loading_title">Load more articles on scroll</string> | ||||||
|  |     <string name="translation">Translation</string> | ||||||
|  |     <string name="cant_open_invalid_url">The item url is invalid. I\'m looking into solving this issue so the app won\'t crash.</string> | ||||||
|  |     <string name="drawer_report_bug">Report a bug</string> | ||||||
|  |     <string name="items_number_should_be_number">The items number should be an integer.</string> | ||||||
|  |     <string name="reader_action_more">Read more</string> | ||||||
|  |     <string name="reader_action_open">Open in browser</string> | ||||||
|  |     <string name="reader_action_share">Share</string> | ||||||
|  |     <string name="pref_switch_actions_pager_scroll_on">Mark articles as read when swiping between articles.</string> | ||||||
|  |     <string name="add_to_favs_reader">Add to favorites</string> | ||||||
|  |     <string name="remove_to_favs_reader">Remove from favorites</string> | ||||||
|  |     <string name="pref_content_reader_font_size">Article reader content font size</string> | ||||||
|  |     <string name="pref_header_viewer">Article viewer</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="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_acra_alwaysaccept">Automatically send crash reports</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_debug_crash_reports">Crash reports</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="drawer_item_hidden_tags">Hidden Tags</string> | ||||||
|  |     <string name="unmark">Mark item as unread</string> | ||||||
|  |     <string name="pref_header_offline">Offline and cache</string> | ||||||
|  |     <string name="pref_switch_items_caching_off">Articles won\'t be saved to the device memory, and the app won\'t be usable offline.</string> | ||||||
|  |     <string name="pref_switch_items_caching_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="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> | ||||||
|  |     <string name="new_items_channel_sync">New items notification</string> | ||||||
|  |     <string name="new_items_notification_title">New items !</string> | ||||||
|  |     <string name="new_items_notification_text">%1$d new items loaded.</string> | ||||||
|  |     <string name="pref_switch_notify_new_items">Notify on new items synced.</string> | ||||||
|  |     <string name="shortcut_offline">Offline</string> | ||||||
|  |     <string name="pref_api_timeout">Api Timeout</string> | ||||||
|  |     <string name="pref_header_experimental">Experimental</string> | ||||||
|  |     <string name="webview_dialog_issue_message">Webview not available. Disabling the article viewer to avoid any future crashes. Will load articles inside of your browser from now on.</string> | ||||||
|  |     <string name="webview_dialog_issue_title">Webview issue</string> | ||||||
|  |     <string name="reader_text_align_left">Align left</string> | ||||||
|  |     <string name="reader_text_align_justify">Justify</string> | ||||||
|  |     <string name="settings_reader_font">Reader font</string> | ||||||
|  | </resources> | ||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Selfoss için okuyucu"</string> |     <string name="app_name">"Selfoss için okuyucu"</string> | ||||||
|     <string name="title_activity_login">"Giriş"</string> |     <string name="title_activity_login">"Giriş"</string> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Selfoss 阅读器"</string> |     <string name="app_name">"Selfoss 阅读器"</string> | ||||||
|     <string name="title_activity_login">"登录"</string> |     <string name="title_activity_login">"登录"</string> | ||||||
| @@ -36,7 +35,7 @@ | |||||||
|     <string name="tab_favs">"收藏夹"</string> |     <string name="tab_favs">"收藏夹"</string> | ||||||
|     <string name="action_about">"关于我们"</string> |     <string name="action_about">"关于我们"</string> | ||||||
|     <string name="marked_as_read">"已读"</string> |     <string name="marked_as_read">"已读"</string> | ||||||
|     <string name="marked_as_unread">"Item unread"</string> |     <string name="marked_as_unread">"未读条目"</string> | ||||||
|     <string name="undo_string">"撤销"</string> |     <string name="undo_string">"撤销"</string> | ||||||
|     <string name="addStringNoUrl">"登录以添加数据源。"</string> |     <string name="addStringNoUrl">"登录以添加数据源。"</string> | ||||||
|     <string name="cant_get_sources">"无法获取数据列表。"</string> |     <string name="cant_get_sources">"无法获取数据列表。"</string> | ||||||
| @@ -84,7 +83,7 @@ | |||||||
|     <string name="card_height_off">卡片高度将被固定</string> |     <string name="card_height_off">卡片高度将被固定</string> | ||||||
|     <string name="source_code">源代码</string> |     <string name="source_code">源代码</string> | ||||||
|     <string name="cant_mark_read">无法将文章标记为已读</string> |     <string name="cant_mark_read">无法将文章标记为已读</string> | ||||||
|     <string name="cant_mark_unread">Can\'t mark article as unread</string> |     <string name="cant_mark_unread">无法将文章标记为未读</string> | ||||||
|     <string name="drawer_error_loading_tags">加载标记时出错..。</string> |     <string name="drawer_error_loading_tags">加载标记时出错..。</string> | ||||||
|     <string name="drawer_error_loading_sources">加载源时出错..。</string> |     <string name="drawer_error_loading_sources">加载源时出错..。</string> | ||||||
|     <string name="drawer_item_filters">搜索条件</string> |     <string name="drawer_item_filters">搜索条件</string> | ||||||
| @@ -106,7 +105,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">隐藏段落</string> |     <string name="pref_hidden_tags">隐藏标签</string> | ||||||
|     <string name="summary_debug_identifier">除错标识符</string> |     <string name="summary_debug_identifier">除错标识符</string> | ||||||
|     <string name="unique_id_to_clipboard">复制到你的剪贴板的标识符</string> |     <string name="unique_id_to_clipboard">复制到你的剪贴板的标识符</string> | ||||||
|     <string name="display_header_drawer_summary">在侧边栏中显示带有 Selfoss 链接地址的页眉。</string> |     <string name="display_header_drawer_summary">在侧边栏中显示带有 Selfoss 链接地址的页眉。</string> | ||||||
| @@ -122,43 +121,43 @@ | |||||||
|     <string name="pref_switch_actions_pager_scroll_on">切换文章时将文章标记为已读。</string> |     <string name="pref_switch_actions_pager_scroll_on">切换文章时将文章标记为已读。</string> | ||||||
|     <string name="add_to_favs_reader">添加到收藏</string> |     <string name="add_to_favs_reader">添加到收藏</string> | ||||||
|     <string name="remove_to_favs_reader">从收藏中移除</string> |     <string name="remove_to_favs_reader">从收藏中移除</string> | ||||||
|     <string name="pref_content_reader_font_size">Article reader content font size</string> |     <string name="pref_content_reader_font_size">文章阅读器内容字体大小</string> | ||||||
|     <string name="pref_header_viewer">Article viewer</string> |     <string name="pref_header_viewer">文章查看器</string> | ||||||
|     <string name="refresh_dialog_message">This will refresh your Selfoss instance.</string> |     <string name="refresh_dialog_message">这将刷新您的 Selfoss 实例。</string> | ||||||
|     <string name="markall_dialog_message">這會使全部項目標示為已讀</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">滑动时标为已读</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">滑动时不标记文章为已读</string> | ||||||
|     <string name="pref_acra_alwaysaccept">自动发送錯誤报告</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">允许自动发送故障报告</string> | ||||||
|     <string name="pref_acra_alwaysaccept_disabled">Will ask everytime when sending crash reports.</string> |     <string name="pref_acra_alwaysaccept_disabled">每次发送崩溃报告时都会询问。</string> | ||||||
|     <string name="pref_debug_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">调试日志(这些日志将在没有对话框的情况下发送)</string> | ||||||
|     <string name="acra_login">Enable logging</string> |     <string name="acra_login">启用日志记录</string> | ||||||
|     <string name="drawer_item_hidden_tags">Hidden Tags</string> |     <string name="drawer_item_hidden_tags">隐藏标签</string> | ||||||
|     <string name="unmark">Mark item as unread</string> |     <string name="unmark">标记条目为未读</string> | ||||||
|     <string name="pref_header_offline">Offline and cache</string> |     <string name="pref_header_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_off">文章不会被保存到设备内存,应用程序在离线时将无法阅读它们</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">文章将被保存到设备内存并可在离线时使用</string> | ||||||
|     <string name="pref_switch_items_caching">Save items for offline use</string> |     <string name="pref_switch_items_caching">保存项目以便离线使用</string> | ||||||
|     <string name="no_network_connectivity">未连接</string> |     <string name="no_network_connectivity">未连接!</string> | ||||||
|     <string name="pref_switch_periodic_refresh">Sync articles</string> |     <string name="pref_switch_periodic_refresh">同步文章</string> | ||||||
|     <string name="pref_switch_periodic_refresh_off">Articles will not be synced in the background</string> |     <string name="pref_switch_periodic_refresh_off">文章将不会在后台同步</string> | ||||||
|     <string name="pref_switch_periodic_refresh_on">Articles will periodically be synced</string> |     <string name="pref_switch_periodic_refresh_on">将定期同步文章</string> | ||||||
|     <string name="pref_periodic_refresh_minutes_title"><![CDATA[Sync interval ( >= 15 minutes)]]></string> |     <string name="pref_periodic_refresh_minutes_title"><![CDATA[同步间隔 (>= 15分钟)]]></string> | ||||||
|     <string name="pref_switch_refresh_when_charging">Only refresh when phone is charging</string> |     <string name="pref_switch_refresh_when_charging">仅在手机充电时刷新</string> | ||||||
|     <string name="loading_notification_title">加载中...</string> |     <string name="loading_notification_title">加载中...</string> | ||||||
|     <string name="loading_notification_text">Selfoss is syncing your articles</string> |     <string name="loading_notification_text">Selfoss 正在同步您的文章</string> | ||||||
|     <string name="notification_channel_sync">同步通知</string> |     <string name="notification_channel_sync">同步通知</string> | ||||||
|     <string name="new_items_channel_sync">New items notification</string> |     <string name="new_items_channel_sync">新条目通知</string> | ||||||
|     <string name="new_items_notification_title">New items !</string> |     <string name="new_items_notification_title">新条目 !</string> | ||||||
|     <string name="new_items_notification_text">%1$d new items loaded.</string> |     <string name="new_items_notification_text">已加载%1$d新条目</string> | ||||||
|     <string name="pref_switch_notify_new_items">Notify on new items synced.</string> |     <string name="pref_switch_notify_new_items">通知已同步的新条目</string> | ||||||
|     <string name="shortcut_offline">Offline</string> |     <string name="shortcut_offline">离线</string> | ||||||
|     <string name="pref_api_timeout">Api Timeout</string> |     <string name="pref_api_timeout">Api 超时</string> | ||||||
|     <string name="pref_header_experimental">Experimental</string> |     <string name="pref_header_experimental">实验性功能</string> | ||||||
|     <string name="webview_dialog_issue_message">Webview not available. Disabling the article viewer to avoid any future crashes. Will load articles inside of your browser from now on.</string> |     <string name="webview_dialog_issue_message">Webview不可用。禁用文章查看器以避免将来发生崩溃。从现在起将在浏览器内加载文章。</string> | ||||||
|     <string name="webview_dialog_issue_title">Webview issue</string> |     <string name="webview_dialog_issue_title">Webview问题</string> | ||||||
|     <string name="reader_text_align_left">Align left</string> |     <string name="reader_text_align_left">左对齐</string> | ||||||
|     <string name="reader_text_align_justify">Justify</string> |     <string name="reader_text_align_justify">左右对齐</string> | ||||||
|     <string name="settings_reader_font">Reader font</string> |     <string name="settings_reader_font">阅读器字体</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <!--Generated by crowdin.com--> |  | ||||||
| <resources xmlns:tools="http://schemas.android.com/tools"> | <resources xmlns:tools="http://schemas.android.com/tools"> | ||||||
|     <string name="app_name">"Selfoss 阅读器"</string> |     <string name="app_name">"Selfoss 阅读器"</string> | ||||||
|     <string name="title_activity_login">"登录"</string> |     <string name="title_activity_login">"登录"</string> | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ buildscript { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     dependencies { |     dependencies { | ||||||
|         classpath 'com.android.tools.build:gradle:3.5.3' |         classpath 'com.android.tools.build:gradle:4.1.1' | ||||||
|         classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" |         classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| #Sat Feb 01 12:14:14 CET 2020 | #Sat Dec 12 19:38:31 CET 2020 | ||||||
| distributionBase=GRADLE_USER_HOME | distributionBase=GRADLE_USER_HOME | ||||||
| distributionPath=wrapper/dists | distributionPath=wrapper/dists | ||||||
| zipStoreBase=GRADLE_USER_HOME | zipStoreBase=GRADLE_USER_HOME | ||||||
| zipStorePath=wrapper/dists | zipStorePath=wrapper/dists | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-5.4.1-all.zip | distributionUrl=https\://services.gradle.org/distributions/gradle-6.5-all.zip | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								res/fr-card.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								res/fr-card.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 1.2 MiB | 
							
								
								
									
										
											BIN
										
									
								
								res/fr-list.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								res/fr-list.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 422 KiB | 
		Reference in New Issue
	
	Block a user