forked from Louvorg/ReaderForSelfoss-multiplatform
Compare commits
10 Commits
v122123601
...
v122123621
Author | SHA1 | Date | |
---|---|---|---|
417a33eb25 | |||
2e7f7f23b3 | |||
e5e182761e | |||
a094d88799 | |||
e51915d1cd | |||
3a654f6ede | |||
5227751dca | |||
27eafe4ff4 | |||
8c83a9408b | |||
fe2410f719 |
@ -35,7 +35,6 @@ steps:
|
|||||||
trigger:
|
trigger:
|
||||||
event:
|
event:
|
||||||
- push
|
- push
|
||||||
- pull_request
|
|
||||||
|
|
||||||
---
|
---
|
||||||
kind: pipeline
|
kind: pipeline
|
||||||
@ -77,10 +76,7 @@ steps:
|
|||||||
from_secret: privateKey
|
from_secret: privateKey
|
||||||
command_timeout: 2m
|
command_timeout: 2m
|
||||||
script:
|
script:
|
||||||
- cd /home/ubuntu
|
- cd /home/ubuntu && sudo rm -rf /var/www/amine/version.txt && sudo chown www-data:www-data ./version.txt && sudo mv version.txt /var/www/amine/
|
||||||
- sudo rm -rf /var/www/amine/version.txt
|
|
||||||
- sudo chown www-data:www-data ./version.txt
|
|
||||||
- sudo mv version.txt /var/www/amine/
|
|
||||||
|
|
||||||
trigger:
|
trigger:
|
||||||
event:
|
event:
|
||||||
|
@ -3,10 +3,7 @@ package bou.amine.apps.readerforselfossv2.android
|
|||||||
import android.app.NotificationChannel
|
import android.app.NotificationChannel
|
||||||
import android.app.NotificationManager
|
import android.app.NotificationManager
|
||||||
import android.content.Context
|
import android.content.Context
|
||||||
import android.graphics.drawable.Drawable
|
|
||||||
import android.net.Uri
|
|
||||||
import android.os.Build
|
import android.os.Build
|
||||||
import android.widget.ImageView
|
|
||||||
import android.widget.Toast
|
import android.widget.Toast
|
||||||
import androidx.lifecycle.DefaultLifecycleObserver
|
import androidx.lifecycle.DefaultLifecycleObserver
|
||||||
import androidx.lifecycle.LifecycleOwner
|
import androidx.lifecycle.LifecycleOwner
|
||||||
@ -18,8 +15,6 @@ import bou.amine.apps.readerforselfossv2.dao.DriverFactory
|
|||||||
import bou.amine.apps.readerforselfossv2.dao.ReaderForSelfossDB
|
import bou.amine.apps.readerforselfossv2.dao.ReaderForSelfossDB
|
||||||
import bou.amine.apps.readerforselfossv2.repository.Repository
|
import bou.amine.apps.readerforselfossv2.repository.Repository
|
||||||
import bou.amine.apps.readerforselfossv2.service.AppSettingsService
|
import bou.amine.apps.readerforselfossv2.service.AppSettingsService
|
||||||
import com.bumptech.glide.Glide
|
|
||||||
import com.bumptech.glide.request.RequestOptions
|
|
||||||
import com.github.ln_12.library.ConnectivityStatus
|
import com.github.ln_12.library.ConnectivityStatus
|
||||||
import io.github.aakira.napier.DebugAntilog
|
import io.github.aakira.napier.DebugAntilog
|
||||||
import io.github.aakira.napier.Napier
|
import io.github.aakira.napier.Napier
|
||||||
@ -83,6 +78,8 @@ class MyApp : MultiDexApplication(), DIAware {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
repository.migrate(driverFactory)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun attachBaseContext(base: Context?) {
|
override fun attachBaseContext(base: Context?) {
|
||||||
|
@ -62,7 +62,7 @@ class ItemCardAdapter(
|
|||||||
|
|
||||||
binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent))
|
binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent))
|
||||||
|
|
||||||
binding.sourceTitleAndDate.text = itm.sourceAndDateText()
|
binding.sourceTitleAndDate.text = itm.sourceAuthorAndDate()
|
||||||
|
|
||||||
if (!appSettingsService.isFullHeightCardsEnabled()) {
|
if (!appSettingsService.isFullHeightCardsEnabled()) {
|
||||||
binding.itemImage.maxHeight = imageMaxHeight
|
binding.itemImage.maxHeight = imageMaxHeight
|
||||||
|
@ -51,7 +51,7 @@ class ItemListAdapter(
|
|||||||
|
|
||||||
binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent))
|
binding.title.setLinkTextColor(c.resources.getColor(R.color.colorAccent))
|
||||||
|
|
||||||
binding.sourceTitleAndDate.text = itm.sourceAndDateText()
|
binding.sourceTitleAndDate.text = itm.sourceAuthorAndDate()
|
||||||
|
|
||||||
if (itm.getThumbnail(repository.baseUrl).isEmpty()) {
|
if (itm.getThumbnail(repository.baseUrl).isEmpty()) {
|
||||||
|
|
||||||
|
@ -78,9 +78,9 @@ class SourcesListAdapter(
|
|||||||
val deleteBtn: Button = mView.findViewById(R.id.deleteBtn)
|
val deleteBtn: Button = mView.findViewById(R.id.deleteBtn)
|
||||||
|
|
||||||
deleteBtn.setOnClickListener {
|
deleteBtn.setOnClickListener {
|
||||||
val (id) = items[bindingAdapterPosition]
|
val (id, title) = items[bindingAdapterPosition]
|
||||||
CoroutineScope(Dispatchers.IO).launch {
|
CoroutineScope(Dispatchers.IO).launch {
|
||||||
val successfullyDeletedSource = repository.deleteSource(id)
|
val successfullyDeletedSource = repository.deleteSource(id, title)
|
||||||
if (successfullyDeletedSource) {
|
if (successfullyDeletedSource) {
|
||||||
items.removeAt(bindingAdapterPosition)
|
items.removeAt(bindingAdapterPosition)
|
||||||
notifyItemRemoved(bindingAdapterPosition)
|
notifyItemRemoved(bindingAdapterPosition)
|
||||||
|
@ -101,7 +101,7 @@ class ArticleFragment : Fragment(), DIAware {
|
|||||||
contentText = item.content
|
contentText = item.content
|
||||||
contentTitle = item.title.getHtmlDecoded()
|
contentTitle = item.title.getHtmlDecoded()
|
||||||
contentImage = item.getThumbnail(repository.baseUrl)
|
contentImage = item.getThumbnail(repository.baseUrl)
|
||||||
contentSource = item.sourceAndDateText()
|
contentSource = item.sourceAuthorAndDate()
|
||||||
allImages = item.getImages()
|
allImages = item.getImages()
|
||||||
|
|
||||||
fontSize = appSettingsService.getFontSize()
|
fontSize = appSettingsService.getFontSize()
|
||||||
|
@ -16,7 +16,8 @@ fun SelfossModel.Item.toParcelable() : ParecelableItem =
|
|||||||
this.icon,
|
this.icon,
|
||||||
this.link,
|
this.link,
|
||||||
this.sourcetitle,
|
this.sourcetitle,
|
||||||
this.tags.joinToString(",")
|
this.tags.joinToString(","),
|
||||||
|
this.author
|
||||||
)
|
)
|
||||||
fun ParecelableItem.toModel() : SelfossModel.Item =
|
fun ParecelableItem.toModel() : SelfossModel.Item =
|
||||||
SelfossModel.Item(
|
SelfossModel.Item(
|
||||||
@ -30,7 +31,8 @@ fun ParecelableItem.toModel() : SelfossModel.Item =
|
|||||||
this.icon,
|
this.icon,
|
||||||
this.link,
|
this.link,
|
||||||
this.sourcetitle,
|
this.sourcetitle,
|
||||||
this.tags.split(",")
|
this.tags.split(","),
|
||||||
|
this.author
|
||||||
)
|
)
|
||||||
data class ParecelableItem(
|
data class ParecelableItem(
|
||||||
val id: Int,
|
val id: Int,
|
||||||
@ -43,7 +45,8 @@ data class ParecelableItem(
|
|||||||
val icon: String?,
|
val icon: String?,
|
||||||
val link: String,
|
val link: String,
|
||||||
val sourcetitle: String,
|
val sourcetitle: String,
|
||||||
val tags: String
|
val tags: String,
|
||||||
|
val author: String?
|
||||||
) : Parcelable {
|
) : Parcelable {
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
@ -65,7 +68,8 @@ data class ParecelableItem(
|
|||||||
icon = source.readString(),
|
icon = source.readString(),
|
||||||
link = source.readString().orEmpty(),
|
link = source.readString().orEmpty(),
|
||||||
sourcetitle = source.readString().orEmpty(),
|
sourcetitle = source.readString().orEmpty(),
|
||||||
tags = source.readString().orEmpty()
|
tags = source.readString().orEmpty(),
|
||||||
|
author = source.readString().orEmpty()
|
||||||
)
|
)
|
||||||
|
|
||||||
override fun describeContents() = 0
|
override fun describeContents() = 0
|
||||||
@ -82,5 +86,6 @@ data class ParecelableItem(
|
|||||||
dest.writeString(link)
|
dest.writeString(link)
|
||||||
dest.writeString(sourcetitle)
|
dest.writeString(sourcetitle)
|
||||||
dest.writeString(tags)
|
dest.writeString(tags)
|
||||||
|
dest.writeString(author)
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -58,6 +58,7 @@ class RepositoryTest {
|
|||||||
data = SelfossModel.Stats(NUMBER_ARTICLES, NUMBER_UNREAD, NUMBER_STARRED)
|
data = SelfossModel.Stats(NUMBER_ARTICLES, NUMBER_UNREAD, NUMBER_STARRED)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
every { db.itemsQueries.deleteItemsWhereSource(any()) } returns Unit
|
||||||
every { db.itemsQueries.items().executeAsList() } returns generateTestDBItems()
|
every { db.itemsQueries.items().executeAsList() } returns generateTestDBItems()
|
||||||
every { db.tagsQueries.deleteAllTags() } returns Unit
|
every { db.tagsQueries.deleteAllTags() } returns Unit
|
||||||
every { db.tagsQueries.transaction(any(), any()) } returns Unit
|
every { db.tagsQueries.transaction(any(), any()) } returns Unit
|
||||||
@ -798,10 +799,11 @@ class RepositoryTest {
|
|||||||
initializeRepository()
|
initializeRepository()
|
||||||
var response: Boolean
|
var response: Boolean
|
||||||
runBlocking {
|
runBlocking {
|
||||||
response = repository.deleteSource(5)
|
response = repository.deleteSource(5, "src")
|
||||||
}
|
}
|
||||||
|
|
||||||
coVerify(exactly = 1) { api.deleteSource(5) }
|
coVerify(exactly = 1) { api.deleteSource(5) }
|
||||||
|
coVerify(exactly = 1) { db.itemsQueries.deleteItemsWhereSource("src") }
|
||||||
assertSame(true, response)
|
assertSame(true, response)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -812,10 +814,11 @@ class RepositoryTest {
|
|||||||
initializeRepository()
|
initializeRepository()
|
||||||
var response: Boolean
|
var response: Boolean
|
||||||
runBlocking {
|
runBlocking {
|
||||||
response = repository.deleteSource(5)
|
response = repository.deleteSource(5, "src")
|
||||||
}
|
}
|
||||||
|
|
||||||
coVerify(exactly = 1) { api.deleteSource(5) }
|
coVerify(exactly = 1) { api.deleteSource(5) }
|
||||||
|
coVerify(exactly = 0) { db.itemsQueries.deleteItemsWhereSource("src") }
|
||||||
assertSame(false, response)
|
assertSame(false, response)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -826,10 +829,11 @@ class RepositoryTest {
|
|||||||
initializeRepository(MutableStateFlow(false))
|
initializeRepository(MutableStateFlow(false))
|
||||||
var response: Boolean
|
var response: Boolean
|
||||||
runBlocking {
|
runBlocking {
|
||||||
response = repository.deleteSource(5)
|
response = repository.deleteSource(5, "src")
|
||||||
}
|
}
|
||||||
|
|
||||||
coVerify(exactly = 0) { api.deleteSource(5) }
|
coVerify(exactly = 0) { api.deleteSource(5) }
|
||||||
|
coVerify(exactly = 1) { db.itemsQueries.deleteItemsWhereSource("src") }
|
||||||
assertSame(false, response)
|
assertSame(false, response)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -17,7 +17,8 @@ fun generateTestDBItems(item: FakeItemParameters = FakeItemParameters()): List<I
|
|||||||
icon = item.icon,
|
icon = item.icon,
|
||||||
link = item.link,
|
link = item.link,
|
||||||
sourcetitle = item.sourcetitle,
|
sourcetitle = item.sourcetitle,
|
||||||
tags = item.tags
|
tags = item.tags,
|
||||||
|
author = item.author
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -35,7 +36,8 @@ fun generateTestApiItem(item: FakeItemParameters = FakeItemParameters()): List<S
|
|||||||
icon = item.icon,
|
icon = item.icon,
|
||||||
link = item.link,
|
link = item.link,
|
||||||
sourcetitle = item.sourcetitle,
|
sourcetitle = item.sourcetitle,
|
||||||
tags = item.tags.split(',')
|
tags = item.tags.split(','),
|
||||||
|
author = item.author
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -54,4 +56,5 @@ class FakeItemParameters {
|
|||||||
"https://ilblogdellasci.wordpress.com/2022/09/09/etica-della-ricerca-sotto-i-riflettori/"
|
"https://ilblogdellasci.wordpress.com/2022/09/09/etica-della-ricerca-sotto-i-riflettori/"
|
||||||
var sourcetitle = "La Chimica e la Società"
|
var sourcetitle = "La Chimica e la Società"
|
||||||
var tags = "Chimica, Testing"
|
var tags = "Chimica, Testing"
|
||||||
|
var author = "Someone important"
|
||||||
}
|
}
|
@ -57,7 +57,6 @@ class SelfossModel {
|
|||||||
val error: String,
|
val error: String,
|
||||||
val icon: String?
|
val icon: String?
|
||||||
)
|
)
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
data class Item(
|
data class Item(
|
||||||
val id: Int,
|
val id: Int,
|
||||||
@ -73,7 +72,8 @@ class SelfossModel {
|
|||||||
val link: String,
|
val link: String,
|
||||||
val sourcetitle: String,
|
val sourcetitle: String,
|
||||||
@Serializable(with = TagsListSerializer::class)
|
@Serializable(with = TagsListSerializer::class)
|
||||||
val tags: List<String>
|
val tags: List<String>,
|
||||||
|
val author: String?
|
||||||
) {
|
) {
|
||||||
// TODO: maybe find a better way to handle these kind of urls
|
// TODO: maybe find a better way to handle these kind of urls
|
||||||
fun getLinkDecoded(): String {
|
fun getLinkDecoded(): String {
|
||||||
@ -102,8 +102,14 @@ class SelfossModel {
|
|||||||
return stringUrl
|
return stringUrl
|
||||||
}
|
}
|
||||||
|
|
||||||
fun sourceAndDateText(): String =
|
fun sourceAuthorAndDate(): String {
|
||||||
this.sourcetitle.getHtmlDecoded() + DateUtils.parseRelativeDate(this.datetime)
|
var txt = this.sourcetitle.getHtmlDecoded()
|
||||||
|
if (!this.author.isNullOrBlank()) {
|
||||||
|
txt += " (by ${this.author}) "
|
||||||
|
}
|
||||||
|
txt += DateUtils.parseRelativeDate(this.datetime)
|
||||||
|
return txt
|
||||||
|
}
|
||||||
|
|
||||||
fun toggleStar(): Item {
|
fun toggleStar(): Item {
|
||||||
this.starred = !this.starred
|
this.starred = !this.starred
|
||||||
@ -111,6 +117,7 @@ class SelfossModel {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// TODO: this seems to be super slow.
|
// TODO: this seems to be super slow.
|
||||||
object TagsListSerializer : KSerializer<List<String>> {
|
object TagsListSerializer : KSerializer<List<String>> {
|
||||||
override fun deserialize(decoder: Decoder): List<String> {
|
override fun deserialize(decoder: Decoder): List<String> {
|
||||||
|
@ -359,13 +359,20 @@ class Repository(
|
|||||||
return response
|
return response
|
||||||
}
|
}
|
||||||
|
|
||||||
suspend fun deleteSource(id: Int): Boolean {
|
suspend fun deleteSource(id: Int, title: String): Boolean {
|
||||||
var success = false
|
var success = false
|
||||||
if (isNetworkAvailable()) {
|
if (isNetworkAvailable()) {
|
||||||
val response = api.deleteSource(id)
|
val response = api.deleteSource(id)
|
||||||
success = response.isSuccess
|
success = response.isSuccess
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We filter on success or if the network isn't available
|
||||||
|
if (success || !isNetworkAvailable()) {
|
||||||
|
items = ArrayList(items.filter { it.sourcetitle != title })
|
||||||
|
setReaderItems(items)
|
||||||
|
db.itemsQueries.deleteItemsWhereSource(title)
|
||||||
|
}
|
||||||
|
|
||||||
return success
|
return success
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -504,6 +511,7 @@ class Repository(
|
|||||||
item.link,
|
item.link,
|
||||||
item.sourcetitle,
|
item.sourcetitle,
|
||||||
item.tags.joinToString(","),
|
item.tags.joinToString(","),
|
||||||
|
item.author,
|
||||||
item.id.toString()
|
item.id.toString()
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -568,4 +576,8 @@ class Repository(
|
|||||||
fun getReaderItems(): ArrayList<SelfossModel.Item> {
|
fun getReaderItems(): ArrayList<SelfossModel.Item> {
|
||||||
return _readerItems
|
return _readerItems
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun migrate(driverFactory: DriverFactory) {
|
||||||
|
ReaderForSelfossDB.Schema.migrate(driverFactory.createDriver(), 0, 1)
|
||||||
|
}
|
||||||
}
|
}
|
@ -51,7 +51,8 @@ fun ITEM.toView(): SelfossModel.Item =
|
|||||||
this.icon,
|
this.icon,
|
||||||
this.link,
|
this.link,
|
||||||
this.sourcetitle,
|
this.sourcetitle,
|
||||||
this.tags.split(",")
|
this.tags.split(","),
|
||||||
|
this.author
|
||||||
)
|
)
|
||||||
|
|
||||||
fun SelfossModel.Item.toEntity(): ITEM =
|
fun SelfossModel.Item.toEntity(): ITEM =
|
||||||
@ -66,5 +67,6 @@ fun SelfossModel.Item.toEntity(): ITEM =
|
|||||||
this.icon,
|
this.icon,
|
||||||
this.link,
|
this.link,
|
||||||
this.sourcetitle.getHtmlDecoded(),
|
this.sourcetitle.getHtmlDecoded(),
|
||||||
this.tags.joinToString(",")
|
this.tags.joinToString(","),
|
||||||
|
this.author
|
||||||
)
|
)
|
@ -0,0 +1,6 @@
|
|||||||
|
CREATE TABLE ITEM_BACKUP AS SELECT `id`, `datetime`, `title`, `content`,
|
||||||
|
`unread`, `starred`, `thumbnail`, `icon`, `link`, `sourcetitle`,
|
||||||
|
`tags` FROM ITEM;
|
||||||
|
ALTER TABLE ITEM_BACKUP ADD COLUMN `author` TEXT;
|
||||||
|
DROP TABLE ITEM;
|
||||||
|
ALTER TABLE ITEM_BACKUP RENAME TO ITEM;
|
@ -10,6 +10,7 @@ CREATE TABLE ITEM (
|
|||||||
`link` TEXT NOT NULL,
|
`link` TEXT NOT NULL,
|
||||||
`sourcetitle` TEXT NOT NULL,
|
`sourcetitle` TEXT NOT NULL,
|
||||||
`tags` TEXT NOT NULL,
|
`tags` TEXT NOT NULL,
|
||||||
|
`author` TEXT,
|
||||||
PRIMARY KEY(`id`)
|
PRIMARY KEY(`id`)
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -26,5 +27,8 @@ INSERT OR REPLACE INTO ITEM VALUES ?;
|
|||||||
deleteItem:
|
deleteItem:
|
||||||
DELETE FROM ITEM WHERE `id` = ?;
|
DELETE FROM ITEM WHERE `id` = ?;
|
||||||
|
|
||||||
|
deleteItemsWhereSource:
|
||||||
|
DELETE FROM ITEM WHERE `sourcetitle` = ?;
|
||||||
|
|
||||||
updateItem:
|
updateItem:
|
||||||
UPDATE ITEM SET `datetime` = ?, `title` = ?, `content` = ?, `unread` = ?, `starred` = ?, `thumbnail` = ?, `icon` = ?, `link` = ?, `sourcetitle` = ?, `tags` = ? WHERE `id` = ?;
|
UPDATE ITEM SET `datetime` = ?, `title` = ?, `content` = ?, `unread` = ?, `starred` = ?, `thumbnail` = ?, `icon` = ?, `link` = ?, `sourcetitle` = ?, `tags` = ?, `author` = ? WHERE `id` = ?;
|
Reference in New Issue
Block a user