diff --git a/README.md b/README.md index 5f535bbd..aa98a063 100644 --- a/README.md +++ b/README.md @@ -12,7 +12,7 @@ An open source podcast instrument, attuned to Puccini ![Puccini](./images/Puccin [Amazon](https://www.amazon.com/%E8%B4%BE%E8%A5%BF%E6%9E%97-Podcini-R/dp/B0D9WR8P13) #### Podcini.R 6.10 allows creating synthetic podcast and shelving any episdes to any synthetic podcasts -#### Podcini.R version 6.5 as a major step forward brings YouTube contents in the app. Channels can be searched, received from share, subscribed. Since 6.6, podcasts, playlists as well as single media from Youtube and YT Music can be shared to Podcini. For more see the Youtube section below or the changelogs +#### Podcini.R version 6.5 as a major step forward brings YouTube contents in the app. Channels can be searched, received from share, subscribed. Podcasts, playlists as well as single media from Youtube and YT Music can be shared to Podcini. For more see the Youtube section below or the changelogs That means finally: [Nessun dorma](https://www.youtube.com/watch?v=cWc7vYjgnTs) #### For Podcini to show up on car's HUD with Android Auto, please read AnroidAuto.md for instructions. #### If you need to cast to an external speaker, you should install the "play" apk, not the "free" apk, that's about the difference between the two. diff --git a/app/build.gradle b/app/build.gradle index ef76a654..c073e74e 100644 --- a/app/build.gradle +++ b/app/build.gradle @@ -31,8 +31,8 @@ android { testApplicationId "ac.mdiq.podcini.tests" testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" - versionCode 3020277 - versionName "6.11.7" + versionCode 3020278 + versionName "6.12.0" applicationId "ac.mdiq.podcini.R" def commit = "" diff --git a/app/src/androidTest/kotlin/ac/test/podcini/service/playback/MediaPlayerBaseTest.kt b/app/src/androidTest/kotlin/ac/test/podcini/service/playback/MediaPlayerBaseTest.kt index 9be520eb..749b2fbf 100644 --- a/app/src/androidTest/kotlin/ac/test/podcini/service/playback/MediaPlayerBaseTest.kt +++ b/app/src/androidTest/kotlin/ac/test/podcini/service/playback/MediaPlayerBaseTest.kt @@ -106,7 +106,7 @@ class MediaPlayerBaseTest { VolumeAdaptionSetting.OFF, null, null) f.preferences = prefs f.episodes.clear() - val i = Episode(0, "t", "i", "l", Date(), Episode.PlayState.UNPLAYED.code, f) + val i = Episode(0, "t", "i", "l", Date(), PlayState.UNPLAYED.code, f) f.episodes.add(i) val media = EpisodeMedia(0, i, 0, 0, 0, "audio/wav", fileUrl, downloadUrl, fileUrl != null, null, 0, 0) i.setMedia(media) diff --git a/app/src/androidTest/kotlin/ac/test/podcini/service/playback/TaskManagerTest.kt b/app/src/androidTest/kotlin/ac/test/podcini/service/playback/TaskManagerTest.kt index c4ea6406..b048fef6 100644 --- a/app/src/androidTest/kotlin/ac/test/podcini/service/playback/TaskManagerTest.kt +++ b/app/src/androidTest/kotlin/ac/test/podcini/service/playback/TaskManagerTest.kt @@ -65,7 +65,7 @@ class TaskManagerTest { val f = Feed(0, null, "title", "link", "d", null, null, null, null, "id", null, "null", "url") f.episodes.clear() for (i in 0 until NUM_ITEMS) { - f.episodes.add(Episode(0, pref + i, pref + i, "link", Date(), Episode.PlayState.PLAYED.code, f)) + f.episodes.add(Episode(0, pref + i, pref + i, "link", Date(), PlayState.PLAYED.code, f)) } // val adapter = getInstance() // adapter.open() diff --git a/app/src/androidTest/kotlin/ac/test/podcini/ui/UITestUtils.kt b/app/src/androidTest/kotlin/ac/test/podcini/ui/UITestUtils.kt index e8be5fc3..9d776275 100644 --- a/app/src/androidTest/kotlin/ac/test/podcini/ui/UITestUtils.kt +++ b/app/src/androidTest/kotlin/ac/test/podcini/ui/UITestUtils.kt @@ -3,6 +3,7 @@ package de.test.podcini.ui import ac.mdiq.podcini.storage.model.Feed import ac.mdiq.podcini.storage.model.Episode import ac.mdiq.podcini.storage.model.EpisodeMedia +import ac.mdiq.podcini.storage.model.PlayState import ac.mdiq.podcini.util.EventFlow import ac.mdiq.podcini.util.FlowEvent import android.content.Context @@ -113,7 +114,7 @@ class UITestUtils(private val context: Context) { val items: MutableList = ArrayList() for (j in 0 until NUM_ITEMS_PER_FEED) { val item = Episode(j.toLong(), "Feed " + (i + 1) + ": Item " + (j + 1), "item$j", - "http://example.com/feed$i/item/$j", Date(), Episode.PlayState.UNPLAYED.code, feed) + "http://example.com/feed$i/item/$j", Date(), PlayState.UNPLAYED.code, feed) items.add(item) if (!hostTextOnlyFeeds) { diff --git a/app/src/main/kotlin/ac/mdiq/podcini/net/download/service/DownloadServiceInterfaceImpl.kt b/app/src/main/kotlin/ac/mdiq/podcini/net/download/service/DownloadServiceInterfaceImpl.kt index 5c3a1960..e696bb1b 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/net/download/service/DownloadServiceInterfaceImpl.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/net/download/service/DownloadServiceInterfaceImpl.kt @@ -362,7 +362,7 @@ class DownloadServiceInterfaceImpl : DownloadServiceInterface() { .addTag(WORK_TAG_EPISODE_URL + item.media!!.downloadUrl) if (enqueueDownloadedEpisodes()) { if (item.feed?.preferences?.queue != null) - runBlocking { Queues.addToQueueSync(false, item, item.feed?.preferences?.queue) } + runBlocking { Queues.addToQueueSync(item, item.feed?.preferences?.queue) } workRequest.addTag(WORK_DATA_WAS_QUEUED) } workRequest.setInputData(Data.Builder().putLong(WORK_DATA_MEDIA_ID, item.media!!.id).build()) diff --git a/app/src/main/kotlin/ac/mdiq/podcini/net/feed/LocalFeedUpdater.kt b/app/src/main/kotlin/ac/mdiq/podcini/net/feed/LocalFeedUpdater.kt index ebabf73d..aa89b6c7 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/net/feed/LocalFeedUpdater.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/net/feed/LocalFeedUpdater.kt @@ -121,7 +121,7 @@ object LocalFeedUpdater { } private fun createFeedItem(feed: Feed, file: FastDocumentFile, context: Context): Episode { - val item = Episode(0L, file.name, UUID.randomUUID().toString(), file.name, Date(file.lastModified), Episode.PlayState.UNPLAYED.code, feed) + val item = Episode(0L, file.name, UUID.randomUUID().toString(), file.name, Date(file.lastModified), PlayState.UNPLAYED.code, feed) item.disableAutoDownload() val size = file.length val media = EpisodeMedia(0, item, 0, 0, size, file.type, diff --git a/app/src/main/kotlin/ac/mdiq/podcini/playback/base/InTheatre.kt b/app/src/main/kotlin/ac/mdiq/podcini/playback/base/InTheatre.kt index c200519e..68df9b7f 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/playback/base/InTheatre.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/playback/base/InTheatre.kt @@ -82,9 +82,7 @@ object InTheatre { } upsert(curQueue_) {} } - upsert(curQueue) { - it.update() - } + upsert(curQueue) { it.update() } } Logd(TAG, "starting curState") @@ -135,10 +133,12 @@ object InTheatre { val type = curState.curMediaType.toInt() if (type == EpisodeMedia.PLAYABLE_TYPE_FEEDMEDIA) { val mediaId = curState.curMediaId + Logd(TAG, "loadPlayableFromPreferences getting mediaId: $mediaId") if (mediaId != 0L) { curMedia = getEpisodeMedia(mediaId) if (curEpisode != null) curEpisode = (curMedia as EpisodeMedia).episodeOrFetch() } + Logd(TAG, "loadPlayableFromPreferences: curMedia: ${curMedia?.getIdentifier()}") } else Log.e(TAG, "Could not restore Playable object from preferences") } } diff --git a/app/src/main/kotlin/ac/mdiq/podcini/playback/service/PlaybackService.kt b/app/src/main/kotlin/ac/mdiq/podcini/playback/service/PlaybackService.kt index a4885558..567fdfb5 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/playback/service/PlaybackService.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/playback/service/PlaybackService.kt @@ -393,7 +393,7 @@ class PlaybackService : MediaLibraryService() { if (ended || smartMarkAsPlayed || autoSkipped || (skipped && !shouldSkipKeepEpisode())) { Logd(TAG, "onPostPlayback ended: $ended smartMarkAsPlayed: $smartMarkAsPlayed autoSkipped: $autoSkipped skipped: $skipped") // only mark the item as played if we're not keeping it anyways - item = setPlayStateSync(Episode.PlayState.PLAYED.code, ended || (skipped && smartMarkAsPlayed), item!!) + item = setPlayStateSync(PlayState.PLAYED.code, ended || (skipped && smartMarkAsPlayed), item!!) val action = item?.feed?.preferences?.autoDeleteAction val shouldAutoDelete = (action == AutoDeleteAction.ALWAYS || (action == AutoDeleteAction.GLOBAL && item?.feed != null && shouldAutoDeleteItem(item!!.feed!!))) @@ -1189,7 +1189,7 @@ class PlaybackService : MediaLibraryService() { if (media != null) { media.setPosition(position) media.setLastPlayedTime(System.currentTimeMillis()) - if (it.isNew) it.playState = Episode.PlayState.UNPLAYED.code + if (it.isNew) it.playState = PlayState.UNPLAYED.code if (media.startPosition >= 0 && media.getPosition() > media.startPosition) media.playedDuration = (media.playedDurationWhenStarted + media.getPosition() - media.startPosition) } diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/algorithms/AutoCleanups.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/algorithms/AutoCleanups.kt index c5029cd8..57d311ba 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/storage/algorithms/AutoCleanups.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/algorithms/AutoCleanups.kt @@ -12,6 +12,7 @@ import ac.mdiq.podcini.storage.database.Queues.getInQueueEpisodeIds import ac.mdiq.podcini.storage.model.Episode import ac.mdiq.podcini.storage.model.EpisodeFilter import ac.mdiq.podcini.storage.model.EpisodeSortOrder +import ac.mdiq.podcini.storage.model.PlayState import ac.mdiq.podcini.util.Logd import android.content.Context import android.util.Log @@ -182,7 +183,7 @@ object AutoCleanups { val idsInQueues = getInQueueEpisodeIds() val mostRecentDateForDeletion = calcMostRecentDateForDeletion(Date()) for (item in downloadedItems) { - if (item.media != null && item.media!!.downloaded && !idsInQueues.contains(item.id) && item.isPlayed() && !item.isFavorite) { + if (item.media != null && item.media!!.downloaded && !idsInQueues.contains(item.id) && item.playState >= PlayState.PLAYED.code && !item.isFavorite) { val media = item.media // make sure this candidate was played at least the proper amount of days prior to now if (media?.playbackCompletionDate != null && media.playbackCompletionDate!!.before(mostRecentDateForDeletion)) candidates.add(item) diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Episodes.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Episodes.kt index 8843dea3..fd3bcf38 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Episodes.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Episodes.kt @@ -18,11 +18,7 @@ import ac.mdiq.podcini.storage.database.RealmDB.realm import ac.mdiq.podcini.storage.database.RealmDB.runOnIOScope import ac.mdiq.podcini.storage.database.RealmDB.upsert import ac.mdiq.podcini.storage.database.RealmDB.upsertBlk -import ac.mdiq.podcini.storage.model.Episode -import ac.mdiq.podcini.storage.model.Episode.PlayState -import ac.mdiq.podcini.storage.model.EpisodeFilter -import ac.mdiq.podcini.storage.model.EpisodeMedia -import ac.mdiq.podcini.storage.model.EpisodeSortOrder +import ac.mdiq.podcini.storage.model.* import ac.mdiq.podcini.storage.utils.EpisodesPermutors.getPermutor import ac.mdiq.podcini.storage.utils.FilesUtils.getMediafilename import ac.mdiq.podcini.util.EventFlow @@ -287,7 +283,7 @@ object Episodes { var episode_ = episode if (!episode.isManaged()) episode_ = realm.query(Episode::class).query("id == $0", episode.id).first().find() ?: episode val result = upsert(episode_) { - if (played >= PlayState.NEW.code && played <= PlayState.BUILDING.code) it.playState = played + if (played != PlayState.UNSPECIFIED.code) it.playState = played else { if (it.playState == PlayState.PLAYED.code) it.playState = PlayState.UNPLAYED.code else it.playState = PlayState.PLAYED.code diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Feeds.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Feeds.kt index 647a3219..6f0bd98a 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Feeds.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Feeds.kt @@ -287,6 +287,7 @@ object Feeds { episode.feed = savedFeed episode.id = idLong++ episode.feedId = savedFeed.id + episode.playState = PlayState.NEW.code if (episode.media != null) { episode.media!!.id = episode.id if (!savedFeed.hasVideoMedia && episode.media!!.getMediaType() == MediaType.VIDEO) savedFeed.hasVideoMedia = true @@ -300,7 +301,7 @@ object Feeds { episode.setNew() if (savedFeed.preferences?.autoAddNewToQueue == true) { val q = savedFeed.preferences?.queue - if (q != null) runOnIOScope { addToQueueSync(false, episode, q) } + if (q != null) runOnIOScope { addToQueueSync(episode, q) } } } } diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/LogsAndStats.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/LogsAndStats.kt index bf384659..e398969b 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/LogsAndStats.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/LogsAndStats.kt @@ -56,7 +56,7 @@ object LogsAndStats { feedTotalTime += m.duration if (m.lastPlayedTime in timeFilterFrom.. 0 && m.playedDuration > 0) || m.episodeOrFetch()?.playState == Episode.PlayState.PLAYED.code || m.position > 0) { + if ((m.playbackCompletionTime > 0 && m.playedDuration > 0) || m.episodeOrFetch()?.playState == PlayState.PLAYED.code || m.position > 0) { episodesStarted += 1 feedPlayedTime += m.duration } diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Queues.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Queues.kt index 31c8eb6a..52683d0a 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Queues.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/Queues.kt @@ -6,6 +6,7 @@ import ac.mdiq.podcini.playback.base.InTheatre.curQueue import ac.mdiq.podcini.preferences.UserPreferences import ac.mdiq.podcini.preferences.UserPreferences.appPrefs import ac.mdiq.podcini.storage.database.Episodes.setPlayState +import ac.mdiq.podcini.storage.database.Episodes.setPlayStateSync import ac.mdiq.podcini.storage.database.RealmDB.realm import ac.mdiq.podcini.storage.database.RealmDB.runOnIOScope import ac.mdiq.podcini.storage.database.RealmDB.upsert @@ -77,9 +78,9 @@ object Queues { appPrefs.edit().putString(UserPreferences.Prefs.prefEnqueueLocation.name, location.name).apply() } - fun queueFromName(name: String): PlayQueue? { - return realm.query(PlayQueue::class).query("name == $0", name).first().find() - } +// fun queueFromName(name: String): PlayQueue? { +// return realm.query(PlayQueue::class).query("name == $0", name).first().find() +// } fun getInQueueEpisodeIds(): Set { Logd(TAG, "getQueueIDList() called") @@ -98,13 +99,13 @@ object Queues { * @param episodes the Episode objects that should be added to the queue. */ @UnstableApi @JvmStatic @Synchronized - fun addToQueue(markAsUnplayed: Boolean, vararg episodes: Episode) : Job { + fun addToQueue(vararg episodes: Episode) : Job { Logd(TAG, "addToQueue( ... ) called") return runOnIOScope { if (episodes.isEmpty()) return@runOnIOScope var queueModified = false - val markAsUnplayeds = mutableListOf() + val setInQueue = mutableListOf() val events: MutableList = ArrayList() val updatedItems: MutableList = ArrayList() val positionCalculator = EnqueuePositionPolicy(enqueueLocation) @@ -121,7 +122,7 @@ object Queues { updatedItems.add(episode) qItems.add(insertPosition, episode) queueModified = true - if (episode.isNew) markAsUnplayeds.add(episode) + if (episode.playState < PlayState.INQUEUE.code) setInQueue.add(episode) insertPosition++ } if (queueModified) { @@ -134,13 +135,13 @@ object Queues { } for (event in events) EventFlow.postEvent(event) - if (markAsUnplayed && markAsUnplayeds.size > 0) setPlayState(Episode.PlayState.UNPLAYED.code, false, *markAsUnplayeds.toTypedArray()) + setPlayState(PlayState.INQUEUE.code, false, *setInQueue.toTypedArray()) // if (performAutoDownload) autodownloadEpisodeMedia(context) } } } - suspend fun addToQueueSync(markAsUnplayed: Boolean, episode: Episode, queue_: PlayQueue? = null) { + suspend fun addToQueueSync(episode: Episode, queue_: PlayQueue? = null) { Logd(TAG, "addToQueueSync( ... ) called") val queue = queue_ ?: curQueue if (queue.episodeIds.contains(episode.id)) return @@ -157,7 +158,7 @@ object Queues { } if (queue.id == curQueue.id) curQueue = queueNew - if (markAsUnplayed && episode.isNew) setPlayState(Episode.PlayState.UNPLAYED.code, false, episode) + if (episode.playState < PlayState.INQUEUE.code) setPlayState(PlayState.INQUEUE.code, false, episode) if (queue.id == curQueue.id) EventFlow.postEvent(FlowEvent.QueueEvent.added(episode, insertPosition)) // if (performAutoDownload) autodownloadEpisodeMedia(context) } @@ -194,6 +195,9 @@ object Queues { it.episodeIds.clear() it.update() } + for (e in curQueue.episodes) { + if (e.playState < PlayState.SKIPPED.code) setPlayState(PlayState.SKIPPED.code, false, e) + } curQueue.episodes.clear() EventFlow.postEvent(FlowEvent.QueueEvent.cleared()) } @@ -230,7 +234,7 @@ object Queues { var queue = queue_ ?: curQueue if (queue.size() == 0) return - val events: MutableList = ArrayList() + val events: MutableList = mutableListOf() val indicesToRemove: MutableList = mutableListOf() val qItems = queue.episodes.toMutableList() val eList = episodes.toList() @@ -239,6 +243,7 @@ object Queues { if (indexOfItemWithId(eList, episode.id) >= 0) { Logd(TAG, "removing from queue: ${episode.id} ${episode.title}") indicesToRemove.add(i) + if (episode.playState < PlayState.SKIPPED.code) setPlayState(PlayState.SKIPPED.code, false, episode) if (queue.id == curQueue.id) events.add(FlowEvent.QueueEvent.removed(episode)) } } @@ -270,6 +275,10 @@ object Queues { if (q.size() == 0 || q.id == curQueue.id) continue idsInQueuesToRemove = q.episodeIds.intersect(episodeIds.toSet()).toMutableSet() if (idsInQueuesToRemove.isNotEmpty()) { + val eList = realm.query(Episode::class).query("id IN $0", idsInQueuesToRemove).find() + for (e in eList) { + if (e.playState < PlayState.SKIPPED.code) setPlayState(PlayState.SKIPPED.code, false, e) + } upsert(q) { it.idsBinList.removeAll(idsInQueuesToRemove) it.idsBinList.addAll(idsInQueuesToRemove) @@ -288,6 +297,10 @@ object Queues { } idsInQueuesToRemove = q.episodeIds.intersect(episodeIds.toSet()).toMutableSet() if (idsInQueuesToRemove.isNotEmpty()) { + val eList = realm.query(Episode::class).query("id IN $0", idsInQueuesToRemove).find() + for (e in eList) { + if (e.playState < PlayState.SKIPPED.code) setPlayState(PlayState.SKIPPED.code, false, e) + } curQueue = upsert(q) { it.idsBinList.removeAll(idsInQueuesToRemove) it.idsBinList.addAll(idsInQueuesToRemove) diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/RealmDB.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/RealmDB.kt index e6ac311a..7e751b6f 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/storage/database/RealmDB.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/database/RealmDB.kt @@ -40,7 +40,7 @@ object RealmDB { SubscriptionLog::class, Chapter::class)) .name("Podcini.realm") - .schemaVersion(27) + .schemaVersion(28) .migration({ mContext -> val oldRealm = mContext.oldRealm // old realm using the previous schema val newRealm = mContext.newRealm // new realm using the new schema @@ -104,8 +104,22 @@ object RealmDB { // ) // } } - }) - .build() + if (oldRealm.schemaVersion() < 28) { + Logd(TAG, "migrating DB from below 27") + mContext.enumerate(className = "Episode") { oldObject: DynamicRealmObject, newObject: DynamicMutableRealmObject? -> + newObject?.run { + if (oldObject.getValue(fieldName = "playState") == 1L) { + set("playState", 10L) + } else { + val media = oldObject.getObject(propertyName = "media") + var position = 0L + if (media != null) position = media.getValue(propertyName = "position", Long::class) ?: 0 + if (position > 0) set("playState", 5L) + } + } + } + } + }).build() realm = Realm.open(config) } diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/model/Episode.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/model/Episode.kt index f289ae98..750aebdd 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/storage/model/Episode.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/model/Episode.kt @@ -86,7 +86,7 @@ class Episode : RealmObject { var rating: Int = Rating.UNRATED.code @Ignore - var isFavorite: Boolean = (rating == 2) + var isFavorite: Boolean = (rating == Rating.FAVORITE.code) private set var comment: String = "" @@ -142,7 +142,7 @@ class Episode : RealmObject { val isRemote = mutableStateOf(false) constructor() { - this.playState = PlayState.UNPLAYED.code + this.playState = PlayState.NEW.code } /** @@ -313,19 +313,10 @@ class Episode : RealmObject { return result } - fun shiftRating(): Int { - val nr = rating + 1 - return if (nr <= Rating.FAVORITE.code) nr else Rating.TRASH.code - } - - enum class PlayState(val code: Int) { - UNSPECIFIED(-2), - NEW(-1), - UNPLAYED(0), - PLAYED(1), - BUILDING(2), - ABANDONED(3) - } +// fun shiftRating(): Int { +// val nr = rating + 1 +// return if (nr <= Rating.FAVORITE.code) nr else Rating.TRASH.code +// } companion object { val TAG: String = Episode::class.simpleName ?: "Anonymous" diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/model/EpisodeFilter.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/model/EpisodeFilter.kt index 305c7561..cf6af9f1 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/storage/model/EpisodeFilter.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/model/EpisodeFilter.kt @@ -38,8 +38,8 @@ class EpisodeFilter(vararg properties: String) : Serializable { fun matches(item: Episode): Boolean { when { showNew && !item.isNew -> return false - showPlayed && !item.isPlayed() -> return false - showUnplayed && item.isPlayed() -> return false + showPlayed && item.playState < PlayState.PLAYED.code -> return false + showUnplayed && item.playState >= PlayState.PLAYED.code -> return false showPaused && !item.isInProgress -> return false showNotPaused && item.isInProgress -> return false showDownloaded && !item.isDownloaded -> return false @@ -58,18 +58,18 @@ class EpisodeFilter(vararg properties: String) : Serializable { // filter on queues does not have a query string so it's not applied on query results, need to filter separately fun matchesForQueues(item: Episode): Boolean { - when { - showQueued && !inAnyQueue(item) -> return false - showNotQueued && inAnyQueue(item) -> return false - else -> return true - } + return when { + showQueued && !inAnyQueue(item) -> false + showNotQueued && inAnyQueue(item) -> false + else -> true + } } fun queryString(): String { val statements: MutableList = ArrayList() when { - showPlayed -> statements.add("playState == 1 ") - showUnplayed -> statements.add(" playState != 1 ") // Match "New" items (read = -1) as well + showPlayed -> statements.add("playState >= ${PlayState.PLAYED.code}") + showUnplayed -> statements.add(" playState < ${PlayState.PLAYED.code}> ") // Match "New" items (read = -1) as well showNew -> statements.add("playState == -1 ") } when { @@ -127,8 +127,8 @@ class EpisodeFilter(vararg properties: String) : Serializable { auto_downloadable, not_auto_downloadable } - companion object { + companion object { @JvmStatic fun unfiltered(): EpisodeFilter { return EpisodeFilter("") diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/model/Feed.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/model/Feed.kt index c14331d6..7f044050 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/storage/model/Feed.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/model/Feed.kt @@ -283,7 +283,7 @@ class Feed : RealmObject { } fun getVirtualQueueItems(): List { - var qString = "feedId == $id AND playState != ${Episode.PlayState.PLAYED.code}" + var qString = "feedId == $id AND playState != ${PlayState.PLAYED.code}" // TODO: perhaps need to set prefStreamOverDownload for youtube feeds if (type != FeedType.YOUTUBE.name && preferences?.prefStreamOverDownload != true) qString += " AND media.downloaded == true" val eList_ = realm.query(Episode::class, qString).find().toMutableList() diff --git a/app/src/main/kotlin/ac/mdiq/podcini/storage/model/PlayState.kt b/app/src/main/kotlin/ac/mdiq/podcini/storage/model/PlayState.kt new file mode 100644 index 00000000..6f7776ff --- /dev/null +++ b/app/src/main/kotlin/ac/mdiq/podcini/storage/model/PlayState.kt @@ -0,0 +1,23 @@ +package ac.mdiq.podcini.storage.model + +import ac.mdiq.podcini.R + +enum class PlayState(val code: Int, val res: Int, val userSet: Boolean) { + UNSPECIFIED(-10, R.drawable.ic_questionmark, false), + BUILDING(-2, R.drawable.baseline_build_24, false), + NEW(-1, R.drawable.baseline_fiber_new_24, false), + UNPLAYED(0, R.drawable.baseline_new_label_24, true), + LATER(1, R.drawable.baseline_watch_later_24, true), + SOON(2, R.drawable.baseline_local_play_24, true), + INQUEUE(3, R.drawable.ic_playlist_play_black, false), + INPROGRESS(5, R.drawable.baseline_play_circle_outline_24, false), + SKIPPED(6, R.drawable.ic_skip_24dp, true), + PLAYED(10, R.drawable.ic_mark_played, true), // was 1 + IGNORED(20, R.drawable.baseline_visibility_off_24, true); + + companion object { + fun fromCode(code: Int): PlayState { + return enumValues().firstOrNull { it.code == code } ?: UNSPECIFIED + } + } +} diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/EpisodeActionButton.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/EpisodeActionButton.kt index 6f06addb..6fbab966 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/EpisodeActionButton.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/EpisodeActionButton.kt @@ -5,16 +5,16 @@ import ac.mdiq.podcini.net.download.service.DownloadServiceInterface import ac.mdiq.podcini.net.utils.NetworkUtils import ac.mdiq.podcini.playback.PlaybackServiceStarter import ac.mdiq.podcini.playback.base.InTheatre -import ac.mdiq.podcini.preferences.UserPreferences.isStreamOverDownload import ac.mdiq.podcini.playback.base.InTheatre.isCurrentlyPlaying import ac.mdiq.podcini.playback.base.VideoMode import ac.mdiq.podcini.playback.service.PlaybackService import ac.mdiq.podcini.playback.service.PlaybackService.Companion.getPlayerActivityIntent import ac.mdiq.podcini.preferences.UsageStatistics import ac.mdiq.podcini.preferences.UserPreferences +import ac.mdiq.podcini.preferences.UserPreferences.isStreamOverDownload import ac.mdiq.podcini.preferences.UserPreferences.videoPlayMode import ac.mdiq.podcini.receiver.MediaButtonReceiver -import ac.mdiq.podcini.storage.database.Episodes +import ac.mdiq.podcini.storage.database.Episodes.setPlayStateSync import ac.mdiq.podcini.storage.database.RealmDB import ac.mdiq.podcini.storage.model.* import ac.mdiq.podcini.storage.utils.AudioMediaTools @@ -36,15 +36,19 @@ import android.widget.Toast import androidx.annotation.DrawableRes import androidx.annotation.OptIn import androidx.annotation.StringRes -import androidx.compose.foundation.Image -import androidx.compose.foundation.layout.* +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.wrapContentSize import androidx.compose.foundation.shape.RoundedCornerShape -import androidx.compose.material3.* -import androidx.compose.runtime.* +import androidx.compose.material3.Card +import androidx.compose.material3.Icon +import androidx.compose.material3.IconButton +import androidx.compose.runtime.Composable +import androidx.compose.runtime.mutableIntStateOf import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.vector.ImageVector -import androidx.compose.ui.res.painterResource import androidx.compose.ui.res.vectorResource import androidx.compose.ui.unit.dp import androidx.compose.ui.window.Dialog @@ -153,7 +157,7 @@ abstract class EpisodeActionButton internal constructor(@JvmField var item: Epis class VisitWebsiteActionButton(item: Episode) : EpisodeActionButton(item) { override val visibility: Boolean - get() = if (item.link.isNullOrEmpty()) false else true + get() = !item.link.isNullOrEmpty() override fun getLabel(): Int { return R.string.visit_website_label @@ -222,6 +226,7 @@ class PlayActionButton(item: Episode) : EpisodeActionButton(item) { } else { PlaybackService.clearCurTempSpeed() PlaybackServiceStarter(context, media).callEvenIfRunning(true).start() + item = runBlocking { setPlayStateSync(PlayState.INPROGRESS.code, false, item) } EventFlow.postEvent(FlowEvent.PlayEvent(item)) } playVideoIfNeeded(context, media) @@ -253,8 +258,7 @@ class DeleteActionButton(item: Episode) : EpisodeActionButton(item) { override val visibility: Boolean get() { - if (item.media != null && (item.media!!.downloaded || item.feed?.isLocalFeed == true)) return true - return false + return item.media != null && (item.media!!.downloaded || item.feed?.isLocalFeed == true) } override fun getLabel(): Int { @@ -291,7 +295,7 @@ class PauseActionButton(item: Episode) : EpisodeActionButton(item) { class DownloadActionButton(item: Episode) : EpisodeActionButton(item) { override val visibility: Boolean - get() = if (item.feed?.isLocalFeed == true) false else true + get() = item.feed?.isLocalFeed != true override fun getLabel(): Int { return R.string.download_label @@ -371,7 +375,10 @@ class StreamActionButton(item: Episode) : EpisodeActionButton(item) { fun stream(context: Context, media: Playable) { if (media !is EpisodeMedia || !InTheatre.isCurMedia(media)) PlaybackService.clearCurTempSpeed() PlaybackServiceStarter(context, media).shouldStreamThisTime(true).callEvenIfRunning(true).start() - if (media is EpisodeMedia && media.episode != null) EventFlow.postEvent(FlowEvent.PlayEvent(media.episode!!)) + if (media is EpisodeMedia && media.episode != null) { + val item = runBlocking { setPlayStateSync(PlayState.INPROGRESS.code, false, media.episode!!) } + EventFlow.postEvent(FlowEvent.PlayEvent(item)) + } playVideoIfNeeded(context, media) } } @@ -382,7 +389,7 @@ class TTSActionButton(item: Episode) : EpisodeActionButton(item) { private var readerText: String? = null override val visibility: Boolean - get() = if (item.link.isNullOrEmpty()) false else true + get() = !item.link.isNullOrEmpty() override fun getLabel(): Int { return R.string.TTS_label @@ -535,6 +542,7 @@ class PlayLocalActionButton(item: Episode) : EpisodeActionButton(item) { } else { PlaybackService.clearCurTempSpeed() PlaybackServiceStarter(context, media).callEvenIfRunning(true).start() + item = runBlocking { setPlayStateSync(PlayState.INPROGRESS.code, false, item) } EventFlow.postEvent(FlowEvent.PlayEvent(item)) } if (media.getMediaType() == MediaType.VIDEO) context.startActivity(getPlayerActivityIntent(context, @@ -542,23 +550,3 @@ class PlayLocalActionButton(item: Episode) : EpisodeActionButton(item) { actionState.value = getLabel() } } - -class MarkAsPlayedActionButton(item: Episode) : EpisodeActionButton(item) { - override val visibility: Boolean - get() = if (item.isPlayed()) false else true - - override fun getLabel(): Int { - return (if (item.media != null) R.string.mark_read_label else R.string.mark_read_no_media_label) - } - - override fun getDrawable(): Int { - return R.drawable.ic_check - } - - @UnstableApi - override fun onClick(context: Context) { - if (!item.isPlayed()) Episodes.setPlayState(Episode.PlayState.PLAYED.code, true, item) - actionState.value = getLabel() - } - -} \ No newline at end of file diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/SwipeAction.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/SwipeAction.kt index bc8dc597..391421b7 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/SwipeAction.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/SwipeAction.kt @@ -29,6 +29,7 @@ interface SwipeAction { START_DOWNLOAD, MARK_FAV, TOGGLE_PLAYED, + SET_PLAY_STATE, REMOVE_FROM_QUEUE, DELETE, REMOVE_FROM_HISTORY diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/SwipeActions.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/SwipeActions.kt index 6e4ab9e4..7eee1fd7 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/SwipeActions.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/actions/SwipeActions.kt @@ -1,10 +1,10 @@ package ac.mdiq.podcini.ui.actions +//import ac.mdiq.podcini.ui.dialog.SwipeActionsDialog import ac.mdiq.podcini.R import ac.mdiq.podcini.playback.base.InTheatre.curQueue import ac.mdiq.podcini.storage.database.Episodes.deleteMediaSync import ac.mdiq.podcini.storage.database.Episodes.setPlayState -import ac.mdiq.podcini.storage.database.Episodes.setPlayStateSync import ac.mdiq.podcini.storage.database.Episodes.shouldDeleteRemoveFromQueue import ac.mdiq.podcini.storage.database.Feeds.shouldAutoDeleteItem import ac.mdiq.podcini.storage.database.Queues.addToQueue @@ -16,21 +16,20 @@ import ac.mdiq.podcini.storage.database.RealmDB.upsert import ac.mdiq.podcini.storage.model.Episode import ac.mdiq.podcini.storage.model.EpisodeFilter import ac.mdiq.podcini.storage.model.EpisodeMedia +import ac.mdiq.podcini.storage.model.PlayState import ac.mdiq.podcini.storage.utils.EpisodeUtil -import ac.mdiq.podcini.ui.actions.SwipeAction.ActionTypes.NO_ACTION import ac.mdiq.podcini.ui.actions.SwipeAction.ActionTypes +import ac.mdiq.podcini.ui.actions.SwipeAction.ActionTypes.NO_ACTION import ac.mdiq.podcini.ui.activity.MainActivity import ac.mdiq.podcini.ui.compose.ChooseRatingDialog import ac.mdiq.podcini.ui.compose.CustomTheme +import ac.mdiq.podcini.ui.compose.PlayStateDialog import ac.mdiq.podcini.ui.fragment.* -//import ac.mdiq.podcini.ui.dialog.SwipeActionsDialog import ac.mdiq.podcini.ui.utils.LocalDeleteModal.deleteEpisodesWarnLocal import ac.mdiq.podcini.util.EventFlow import ac.mdiq.podcini.util.FlowEvent -import ac.mdiq.podcini.util.Logd import android.content.Context import android.content.SharedPreferences -import android.os.Handler import android.util.TypedValue import android.view.ViewGroup import androidx.annotation.OptIn @@ -47,7 +46,6 @@ import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.vector.ImageVector import androidx.compose.ui.platform.ComposeView import androidx.compose.ui.platform.LocalContext -import androidx.compose.ui.res.painterResource import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.vectorResource import androidx.compose.ui.text.style.TextAlign @@ -147,7 +145,7 @@ open class SwipeActions(private val fragment: Fragment, private val tag: String) @OptIn(UnstableApi::class) override fun performAction(item: Episode, fragment: Fragment, filter: EpisodeFilter) { - addToQueue( true, item) + addToQueue(item) } override fun willRemove(filter: EpisodeFilter, item: Episode): Boolean { @@ -398,7 +396,7 @@ open class SwipeActions(private val fragment: Fragment, private val tag: String) fun addToQueueAt(episode: Episode, index: Int) : Job { return runOnIOScope { if (curQueue.episodeIds.contains(episode.id)) return@runOnIOScope - if (episode.isNew) setPlayState(Episode.PlayState.UNPLAYED.code, false, episode) + if (episode.isNew) setPlayState(PlayState.UNPLAYED.code, false, episode) curQueue = upsert(curQueue) { it.episodeIds.add(index, episode.id) it.update() @@ -438,9 +436,9 @@ open class SwipeActions(private val fragment: Fragment, private val tag: String) } } - class TogglePlaybackStateSwipeAction : SwipeAction { + class SetPlaybackStateSwipeAction : SwipeAction { override fun getId(): String { - return ActionTypes.TOGGLE_PLAYED.name + return ActionTypes.SET_PLAY_STATE.name } override fun getActionIcon(): Int { @@ -452,45 +450,22 @@ open class SwipeActions(private val fragment: Fragment, private val tag: String) } override fun getTitle(context: Context): String { - return context.getString(R.string.toggle_played_label) + return context.getString(R.string.set_play_state_label) } override fun performAction(item: Episode, fragment: Fragment, filter: EpisodeFilter) { - val newState = if (item.playState == Episode.PlayState.UNPLAYED.code) Episode.PlayState.PLAYED.code else Episode.PlayState.UNPLAYED.code - - Logd("TogglePlaybackStateSwipeAction", "performAction( ${item.id} )") - // we're marking it as unplayed since the user didn't actually play it - // but they don't want it considered 'NEW' anymore - var item = runBlocking { setPlayStateSync(newState, false, item) } - - val h = Handler(fragment.requireContext().mainLooper) - val r = Runnable { - val media: EpisodeMedia? = item.media - val shouldAutoDelete = if (item.feed == null) false else shouldAutoDeleteItem(item.feed!!) - if (media != null && EpisodeUtil.hasAlmostEnded(media) && shouldAutoDelete) { - item = deleteMediaSync(fragment.requireContext(), item) - if (shouldDeleteRemoveFromQueue()) removeFromQueueSync(null, item) } - } - val playStateStringRes: Int = when (newState) { - Episode.PlayState.UNPLAYED.code -> if (item.playState == Episode.PlayState.NEW.code) R.string.removed_inbox_label //was new - else R.string.marked_as_unplayed_label //was played - Episode.PlayState.PLAYED.code -> R.string.marked_as_played_label - else -> if (item.playState == Episode.PlayState.NEW.code) R.string.removed_inbox_label - else R.string.marked_as_unplayed_label - } - val duration: Int = Snackbar.LENGTH_LONG - - if (willRemove(filter, item)) { - (fragment.activity as MainActivity).showSnackbarAbovePlayer( - playStateStringRes, duration) - .setAction(fragment.getString(R.string.undo)) { - setPlayState(item.playState, false, item) - // don't forget to cancel the thing that's going to remove the media - h.removeCallbacks(r) + var showPlayStateDialog by mutableStateOf(true) + val composeView = ComposeView(fragment.requireContext()).apply { + setContent { + CustomTheme(fragment.requireContext()) { + if (showPlayStateDialog) PlayStateDialog(listOf(item)) { + showPlayStateDialog = false + (fragment.view as? ViewGroup)?.removeView(this@apply) + } } + } } - - h.postDelayed(r, ceil((duration * 1.05f).toDouble()).toLong()) + (fragment.view as? ViewGroup)?.addView(composeView) } private fun delayedExecution(item: Episode, fragment: Fragment, duration: Float) = runBlocking { @@ -504,7 +479,7 @@ open class SwipeActions(private val fragment: Fragment, private val tag: String) } override fun willRemove(filter: EpisodeFilter, item: Episode): Boolean { - return if (item.playState == Episode.PlayState.NEW.code) filter.showPlayed || filter.showNew + return if (item.playState == PlayState.NEW.code) filter.showPlayed || filter.showNew else filter.showUnplayed || filter.showPlayed || filter.showNew } } @@ -523,7 +498,7 @@ open class SwipeActions(private val fragment: Fragment, private val tag: String) val swipeActions: List = listOf( NoActionSwipeAction(), ComboSwipeAction(), AddToQueueSwipeAction(), StartDownloadSwipeAction(), SetRatingSwipeAction(), - TogglePlaybackStateSwipeAction(), RemoveFromQueueSwipeAction(), + SetPlaybackStateSwipeAction(), RemoveFromQueueSwipeAction(), DeleteSwipeAction(), RemoveFromHistorySwipeAction()) private fun getPrefs(tag: String, defaultActions: String): Actions { diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/activity/MainActivity.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/activity/MainActivity.kt index 4107c485..71918cc9 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/activity/MainActivity.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/activity/MainActivity.kt @@ -633,6 +633,10 @@ class MainActivity : CastEnabledActivity() { } } + fun openDrawer() { + drawerLayout?.openDrawer(navDrawer) + } + private var eventSink: Job? = null private var eventStickySink: Job? = null private fun cancelFlowEvents() { diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/compose/EpisodesVM.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/compose/EpisodesVM.kt index 84cb9584..7e96cf8a 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/compose/EpisodesVM.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/compose/EpisodesVM.kt @@ -3,18 +3,27 @@ package ac.mdiq.podcini.ui.compose import ac.mdiq.podcini.R import ac.mdiq.podcini.net.download.DownloadStatus import ac.mdiq.podcini.net.download.service.DownloadServiceInterface +import ac.mdiq.podcini.net.sync.SynchronizationSettings.isProviderConnected +import ac.mdiq.podcini.net.sync.SynchronizationSettings.wifiSyncEnabledKey +import ac.mdiq.podcini.net.sync.model.EpisodeAction +import ac.mdiq.podcini.net.sync.queue.SynchronizationQueueSink import ac.mdiq.podcini.playback.base.InTheatre import ac.mdiq.podcini.playback.base.InTheatre.curQueue import ac.mdiq.podcini.playback.base.MediaPlayerBase.Companion.status import ac.mdiq.podcini.storage.database.Episodes +import ac.mdiq.podcini.storage.database.Episodes.deleteMediaSync import ac.mdiq.podcini.storage.database.Episodes.episodeFromStreamInfo import ac.mdiq.podcini.storage.database.Episodes.setPlayState +import ac.mdiq.podcini.storage.database.Episodes.setPlayStateSync +import ac.mdiq.podcini.storage.database.Episodes.shouldDeleteRemoveFromQueue import ac.mdiq.podcini.storage.database.Feeds.addToMiscSyndicate import ac.mdiq.podcini.storage.database.Feeds.addToYoutubeSyndicate +import ac.mdiq.podcini.storage.database.Feeds.shouldAutoDeleteItem import ac.mdiq.podcini.storage.database.Queues import ac.mdiq.podcini.storage.database.Queues.addToQueueSync import ac.mdiq.podcini.storage.database.Queues.removeFromAllQueuesQuiet import ac.mdiq.podcini.storage.database.Queues.removeFromQueue +import ac.mdiq.podcini.storage.database.Queues.removeFromQueueSync import ac.mdiq.podcini.storage.database.RealmDB.realm import ac.mdiq.podcini.storage.database.RealmDB.upsert import ac.mdiq.podcini.storage.database.RealmDB.upsertBlk @@ -22,6 +31,7 @@ import ac.mdiq.podcini.storage.model.* import ac.mdiq.podcini.storage.model.Feed.Companion.MAX_SYNTHETIC_ID import ac.mdiq.podcini.storage.model.Feed.Companion.newId import ac.mdiq.podcini.storage.utils.DurationConverter +import ac.mdiq.podcini.storage.utils.EpisodeUtil.hasAlmostEnded import ac.mdiq.podcini.storage.utils.ImageResourceUtils import ac.mdiq.podcini.ui.actions.EpisodeActionButton import ac.mdiq.podcini.ui.actions.EpisodeActionButton.Companion.forItem @@ -121,7 +131,7 @@ var queueChanged by mutableIntStateOf(0) @Stable class EpisodeVM(var episode: Episode) { var positionState by mutableStateOf(episode.media?.position?:0) - var playedState by mutableStateOf(episode.isPlayed()) + var playedState by mutableIntStateOf(episode.playState) var isPlayingState by mutableStateOf(false) var ratingState by mutableIntStateOf(episode.rating) var inProgressState by mutableStateOf(episode.isInProgress) @@ -157,9 +167,9 @@ class EpisodeVM(var episode: Episode) { Logd("EpisodeVM", "episodeMonitor UpdatedObject ${changes.obj.title} ${changes.changedFields.joinToString()}") if (episode.id == changes.obj.id) { withContext(Dispatchers.Main) { - playedState = changes.obj.isPlayed() + playedState = changes.obj.playState ratingState = changes.obj.rating -// episode = changes.obj // direct assignment doesn't update member like media?? + episode = changes.obj // direct assignment doesn't update member like media?? } Logd("EpisodeVM", "episodeMonitor $playedState $playedState ") } else Logd("EpisodeVM", "episodeMonitor index out bound") @@ -183,7 +193,7 @@ class EpisodeVM(var episode: Episode) { positionState = changes.obj.media?.position ?: 0 inProgressState = changes.obj.isInProgress Logd("EpisodeVM", "mediaMonitor $positionState $inProgressState ${episode.title}") -// episode = changes.obj + episode = changes.obj // Logd("EpisodeVM", "mediaMonitor downloaded: ${changes.obj.media?.downloaded} ${episode.media?.downloaded}") } } else Logd("EpisodeVM", "mediaMonitor index out bound") @@ -231,6 +241,61 @@ fun ChooseRatingDialog(selected: List, onDismissRequest: () -> Unit) { } } +@Composable +fun PlayStateDialog(selected: List, onDismissRequest: () -> Unit) { + val context = LocalContext.current + Dialog(onDismissRequest = onDismissRequest) { + Surface(shape = RoundedCornerShape(16.dp)) { + Column(modifier = Modifier.padding(16.dp), verticalArrangement = Arrangement.spacedBy(16.dp)) { + for (state in PlayState.entries) { + if (state.userSet) { + Row(verticalAlignment = Alignment.CenterVertically, modifier = Modifier.padding(4.dp) + .clickable { + for (item in selected) { + var item_ = runBlocking { setPlayStateSync(state.code, false, item) } + val media: EpisodeMedia? = item_.media + val shouldAutoDelete = if (item_.feed == null) false else shouldAutoDeleteItem(item_.feed!!) + if (media != null && hasAlmostEnded(media) && shouldAutoDelete) { + item_ = deleteMediaSync(context, item_) + if (shouldDeleteRemoveFromQueue()) removeFromQueueSync(null, item_) + } + when (state) { + PlayState.UNPLAYED -> { + if (isProviderConnected && item_?.feed?.isLocalFeed != true && item_?.media != null) { + val actionNew: EpisodeAction = EpisodeAction.Builder(item_!!, EpisodeAction.NEW).currentTimestamp().build() + SynchronizationQueueSink.enqueueEpisodeActionIfSyncActive(context, actionNew) + } + } + PlayState.PLAYED -> { + if (item_?.feed?.isLocalFeed != true && (isProviderConnected || wifiSyncEnabledKey)) { + val media: EpisodeMedia? = item_?.media + // not all items have media, Gpodder only cares about those that do + if (isProviderConnected && media != null) { + val actionPlay: EpisodeAction = EpisodeAction.Builder(item_!!, EpisodeAction.PLAY) + .currentTimestamp() + .started(media.getDuration() / 1000) + .position(media.getDuration() / 1000) + .total(media.getDuration() / 1000) + .build() + SynchronizationQueueSink.enqueueEpisodeActionIfSyncActive(context, actionPlay) + } + } + } + else -> {} + } + } + onDismissRequest() + }) { + Icon(imageVector = ImageVector.vectorResource(id = state.res), "") + Text(state.name, Modifier.padding(start = 4.dp)) + } + } + } + } + } + } +} + @Composable fun PutToQueueDialog(selected: List, onDismissRequest: () -> Unit) { val queues = realm.query(PlayQueue::class).find() @@ -271,7 +336,7 @@ fun PutToQueueDialog(selected: List, onDismissRequest: () -> Unit) { if (toRemoveCur.isNotEmpty()) EventFlow.postEvent(FlowEvent.QueueEvent.removed(toRemoveCur)) } selected.forEach { e -> - runBlocking { addToQueueSync(false, e, toQueue) } + runBlocking { addToQueueSync(e, toQueue) } } onDismissRequest() }) { @@ -366,6 +431,9 @@ fun EpisodeLazyColumn(activity: MainActivity, vms: List, feed: Feed? var showChooseRatingDialog by remember { mutableStateOf(false) } if (showChooseRatingDialog) ChooseRatingDialog(selected) { showChooseRatingDialog = false } + var showPlayStateDialog by remember { mutableStateOf(false) } + if (showPlayStateDialog) PlayStateDialog(selected) { showPlayStateDialog = false } + var showPutToQueueDialog by remember { mutableStateOf(false) } if (showPutToQueueDialog) PutToQueueDialog(selected) { showPutToQueueDialog = false } @@ -460,13 +528,14 @@ fun EpisodeLazyColumn(activity: MainActivity, vms: List, feed: Feed? { Row(modifier = Modifier .padding(horizontal = 16.dp) .clickable { + showPlayStateDialog = true isExpanded = false selectMode = false Logd(TAG, "ic_mark_played: ${selected.size}") - setPlayState(Episode.PlayState.UNSPECIFIED.code, false, *selected.toTypedArray()) +// setPlayState(PlayState.UNSPECIFIED.code, false, *selected.toTypedArray()) }, verticalAlignment = Alignment.CenterVertically) { Icon(imageVector = ImageVector.vectorResource(id = R.drawable.ic_mark_played), "Toggle played state") - Text(stringResource(id = R.string.toggle_played_label)) } }, + Text(stringResource(id = R.string.set_play_state_label)) } }, { Row(modifier = Modifier .padding(horizontal = 16.dp) .clickable { @@ -483,7 +552,7 @@ fun EpisodeLazyColumn(activity: MainActivity, vms: List, feed: Feed? isExpanded = false selectMode = false Logd(TAG, "ic_playlist_play: ${selected.size}") - Queues.addToQueue(true, *selected.toTypedArray()) + Queues.addToQueue(*selected.toTypedArray()) }, verticalAlignment = Alignment.CenterVertically) { Icon(imageVector = ImageVector.vectorResource(id = R.drawable.ic_playlist_play), "Add to active queue") Text(stringResource(id = R.string.add_to_queue_label)) } }, @@ -520,8 +589,7 @@ fun EpisodeLazyColumn(activity: MainActivity, vms: List, feed: Feed? ) if (selected.isNotEmpty() && selected[0].isRemote.value) options.add { - Row(modifier = Modifier - .padding(horizontal = 16.dp) + Row(modifier = Modifier.padding(horizontal = 16.dp) .clickable { isExpanded = false selectMode = false @@ -594,7 +662,7 @@ fun EpisodeLazyColumn(activity: MainActivity, vms: List, feed: Feed? ConstraintLayout(modifier = Modifier.width(56.dp).height(56.dp)) { val (imgvCover, checkMark) = createRefs() val imgLoc = remember(vm) { ImageResourceUtils.getEpisodeListImageLocation(vm.episode) } - Logd(TAG, "imgLoc: $imgLoc") +// Logd(TAG, "imgLoc: $imgLoc") AsyncImage(model = ImageRequest.Builder(context).data(imgLoc) .memoryCachePolicy(CachePolicy.ENABLED).placeholder(R.mipmap.ic_launcher).error(R.mipmap.ic_launcher).build(), contentDescription = "imgvCover", @@ -609,8 +677,8 @@ fun EpisodeLazyColumn(activity: MainActivity, vms: List, feed: Feed? if (selectMode) toggleSelected() else if (vm.episode.feed != null) activity.loadChildFragment(FeedInfoFragment.newInstance(vm.episode.feed!!)) })) - val alpha = if (vm.playedState) 1.0f else 0f - if (vm.playedState) Icon(imageVector = ImageVector.vectorResource(R.drawable.ic_check), tint = textColor, contentDescription = "played_mark", + val alpha = if (vm.playedState >= PlayState.SKIPPED.code) 1.0f else 0f + if (vm.playedState >= PlayState.SKIPPED.code) Icon(imageVector = ImageVector.vectorResource(R.drawable.ic_check), tint = textColor, contentDescription = "played_mark", modifier = Modifier.background(Color.Green).alpha(alpha) .constrainAs(checkMark) { bottom.linkTo(parent.bottom) @@ -640,14 +708,16 @@ fun EpisodeLazyColumn(activity: MainActivity, vms: List, feed: Feed? vms[index].inQueueState = curQueue.contains(vms[index].episode) } Row(verticalAlignment = Alignment.CenterVertically) { - Logd(TAG, "info row") - if (vm.episode.media?.getMediaType() == MediaType.VIDEO) - Icon(imageVector = ImageVector.vectorResource(R.drawable.ic_videocam), tint = textColor, contentDescription = "isVideo", modifier = Modifier.width(14.dp).height(14.dp)) +// Logd(TAG, "info row") val ratingIconRes = Rating.fromCode(vm.ratingState).res if (vm.ratingState != Rating.UNRATED.code) Icon(imageVector = ImageVector.vectorResource(ratingIconRes), tint = MaterialTheme.colorScheme.tertiary, contentDescription = "rating", modifier = Modifier.background(MaterialTheme.colorScheme.tertiaryContainer).width(14.dp).height(14.dp)) + val playStateRes = PlayState.fromCode(vm.playedState).res + Icon(imageVector = ImageVector.vectorResource(playStateRes), tint = textColor, contentDescription = "playState", modifier = Modifier.width(14.dp).height(14.dp)) if (vm.inQueueState) Icon(imageVector = ImageVector.vectorResource(R.drawable.ic_playlist_play), tint = textColor, contentDescription = "ivInPlaylist", modifier = Modifier.width(14.dp).height(14.dp)) + if (vm.episode.media?.getMediaType() == MediaType.VIDEO) + Icon(imageVector = ImageVector.vectorResource(R.drawable.ic_videocam), tint = textColor, contentDescription = "isVideo", modifier = Modifier.width(14.dp).height(14.dp)) val curContext = LocalContext.current val dur = remember { vm.episode.media?.getDuration() ?: 0 } val durText = remember { DurationConverter.getDurationStringLong(dur) } @@ -681,7 +751,7 @@ fun EpisodeLazyColumn(activity: MainActivity, vms: List, feed: Feed? detectTapGestures(onLongPress = { vms[index].showAltActionsDialog = true }, onTap = { vms[index].actionButton.onClick(activity) }) }, ) { - Logd(TAG, "button box") +// Logd(TAG, "button box") vm.actionRes = vm.actionButton.getDrawable() Icon(imageVector = ImageVector.vectorResource(vm.actionRes), tint = textColor, contentDescription = null, modifier = Modifier.width(28.dp).height(32.dp)) if (isDownloading() && vm.dlPercent >= 0) CircularProgressIndicator(progress = { 0.01f * vm.dlPercent }, diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/AudioPlayerFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/AudioPlayerFragment.kt index 51c7cf83..bf4ddf0d 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/AudioPlayerFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/AudioPlayerFragment.kt @@ -6,6 +6,8 @@ import ac.mdiq.podcini.playback.PlaybackServiceStarter import ac.mdiq.podcini.playback.ServiceStatusHandler import ac.mdiq.podcini.playback.base.InTheatre.curEpisode import ac.mdiq.podcini.playback.base.InTheatre.curMedia +import ac.mdiq.podcini.playback.base.InTheatre.curQueue +import ac.mdiq.podcini.playback.base.InTheatre.isCurrentlyPlaying import ac.mdiq.podcini.playback.base.MediaPlayerBase.Companion.status import ac.mdiq.podcini.playback.base.PlayerStatus import ac.mdiq.podcini.playback.base.VideoMode @@ -85,11 +87,8 @@ import coil.request.CachePolicy import coil.request.ImageRequest import com.google.android.material.bottomsheet.BottomSheetBehavior import com.google.android.material.snackbar.Snackbar -import kotlinx.coroutines.Dispatchers -import kotlinx.coroutines.Job +import kotlinx.coroutines.* import kotlinx.coroutines.flow.collectLatest -import kotlinx.coroutines.launch -import kotlinx.coroutines.withContext import net.dankito.readability4j.Readability4J import org.apache.commons.lang3.StringUtils import java.text.DecimalFormat @@ -108,6 +107,7 @@ class AudioPlayerFragment : Fragment() { private var prevItem: Episode? = null private var currentItem: Episode? = null + private var playButInit = false private var isShowPlay: Boolean = true private var showTimeLeft = false @@ -166,7 +166,10 @@ class AudioPlayerFragment : Fragment() { } } } - (activity as MainActivity).setPlayerVisible(false) + Logd(TAG, "curMedia: ${curMedia?.getIdentifier()}") + (activity as MainActivity).setPlayerVisible(curMedia != null) + if (curMedia != null) updateUi(curMedia!!) +// if (curMedia is EpisodeMedia) setIsShowPlay(isCurrentlyPlaying(curMedia as EpisodeMedia)) return composeView } @@ -188,7 +191,7 @@ class AudioPlayerFragment : Fragment() { if (curMedia == null) return if (playbackService == null) PlaybackServiceStarter(requireContext(), curMedia!!).start() } - val imgLoc_ = remember(currentItem) { imgLoc } + val imgLoc_ = remember(currentMedia) { imgLoc } AsyncImage(model = ImageRequest.Builder(context).data(imgLoc_) .memoryCachePolicy(CachePolicy.ENABLED).placeholder(R.mipmap.ic_launcher).error(R.mipmap.ic_launcher).build(), contentDescription = "imgvCover", @@ -502,9 +505,15 @@ class AudioPlayerFragment : Fragment() { txtvPlaybackSpeed = speedStr // binding.butPlaybackSpeed.setSpeed(event.newSpeed) TODO } + @UnstableApi fun onPositionUpdate(event: FlowEvent.PlaybackPositionEvent) { Logd(TAG, "onPositionUpdate") + if (!playButInit && playButRes == R.drawable.ic_play_48dp && curMedia is EpisodeMedia) { + playButRes = R.drawable.ic_pause + playButInit = true + } + if (curMedia?.getIdentifier() != event.media?.getIdentifier() || controller == null || curPositionFB == Playable.INVALID_TIME || curDurationFB == Playable.INVALID_TIME) return val converter = TimeSpeedConverter(curSpeedFB) currentPosition = converter.convert(event.position) @@ -520,6 +529,7 @@ class AudioPlayerFragment : Fragment() { sliderValue = event.position.toFloat() } + private fun onPlaybackServiceChanged(event: FlowEvent.PlaybackServiceEvent) { when (event.action) { FlowEvent.PlaybackServiceEvent.Action.SERVICE_SHUT_DOWN -> (activity as MainActivity).setPlayerVisible(false) @@ -527,6 +537,7 @@ class AudioPlayerFragment : Fragment() { // PlaybackServiceEvent.Action.SERVICE_RESTARTED -> (activity as MainActivity).setPlayerVisible(true) } } + @UnstableApi fun updateUi(media: Playable) { Logd(TAG, "updateUi called $media") @@ -578,7 +589,8 @@ class AudioPlayerFragment : Fragment() { withContext(Dispatchers.Main) { Logd(TAG, "subscribe: ${currentMedia?.getEpisodeTitle()}") displayMediaInfo(currentMedia!!) -// shownoteView.loadDataWithBaseURL("https://127.0.0.1", cleanedNotes?:"No notes", "text/html", "utf-8", "about:blank") + (activity as MainActivity).setPlayerVisible(curMedia != null) + // shownoteView.loadDataWithBaseURL("https://127.0.0.1", cleanedNotes?:"No notes", "text/html", "utf-8", "about:blank") Logd(TAG, "Webview loaded") } }.invokeOnCompletion { throwable -> @@ -738,7 +750,10 @@ class AudioPlayerFragment : Fragment() { private var loadItemsRunning = false fun loadMediaInfo() { + Logd(TAG, "loadMediaInfo() curMedia: ${curMedia?.getIdentifier()}") val actMain = (activity as MainActivity) + var i = 0 + while (curMedia == null && i++ < 6) runBlocking { delay(500) } if (curMedia == null) { if (actMain.isPlayerVisible()) actMain.setPlayerVisible(false) return @@ -747,6 +762,7 @@ class AudioPlayerFragment : Fragment() { loadItemsRunning = true if (!actMain.isPlayerVisible()) actMain.setPlayerVisible(true) val curMediaChanged = currentMedia == null || curMedia?.getIdentifier() != currentMedia?.getIdentifier() + if (curMedia?.getIdentifier() != currentMedia?.getIdentifier()) updateUi(curMedia!!) if (!isCollapsed && curMediaChanged) { updateDetails() Logd(TAG, "loadMediaInfo loading details ${curMedia?.getIdentifier()}") @@ -761,7 +777,6 @@ class AudioPlayerFragment : Fragment() { if (item != null) setItem(item) setChapterDividers() sleepTimerActive = isSleepTimerActive() - if (currentMedia != null) updateUi(currentMedia!!) // TODO: disable for now // if (!includingChapters) loadMediaInfo(true) }.invokeOnCompletion { throwable -> diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/BaseEpisodesFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/BaseEpisodesFragment.kt index db1785ae..ebd45a98 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/BaseEpisodesFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/BaseEpisodesFragment.kt @@ -1,7 +1,7 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R -import ac.mdiq.podcini.databinding.BaseEpisodesListFragmentBinding +import ac.mdiq.podcini.databinding.ComposeFragmentBinding import ac.mdiq.podcini.net.download.DownloadStatus import ac.mdiq.podcini.storage.model.Episode import ac.mdiq.podcini.storage.model.EpisodeFilter @@ -10,7 +10,10 @@ import ac.mdiq.podcini.ui.actions.SwipeAction import ac.mdiq.podcini.ui.actions.SwipeActions import ac.mdiq.podcini.ui.actions.SwipeActions.NoActionSwipeAction import ac.mdiq.podcini.ui.activity.MainActivity -import ac.mdiq.podcini.ui.compose.* +import ac.mdiq.podcini.ui.compose.CustomTheme +import ac.mdiq.podcini.ui.compose.EpisodeLazyColumn +import ac.mdiq.podcini.ui.compose.EpisodeVM +import ac.mdiq.podcini.ui.compose.InforBar import ac.mdiq.podcini.ui.utils.EmptyViewHandler import ac.mdiq.podcini.util.EventFlow import ac.mdiq.podcini.util.FlowEvent @@ -42,7 +45,7 @@ abstract class BaseEpisodesFragment : Fragment(), Toolbar.OnMenuItemClickListene protected var page: Int = 1 private var displayUpArrow = false - var _binding: BaseEpisodesListFragmentBinding? = null + var _binding: ComposeFragmentBinding? = null protected val binding get() = _binding!! protected var infoBarText = mutableStateOf("") @@ -59,7 +62,7 @@ abstract class BaseEpisodesFragment : Fragment(), Toolbar.OnMenuItemClickListene @UnstableApi override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { super.onCreateView(inflater, container, savedInstanceState) - _binding = BaseEpisodesListFragmentBinding.inflate(inflater) + _binding = ComposeFragmentBinding.inflate(inflater) Logd(TAG, "fragment onCreateView") toolbar = binding.toolbar @@ -79,7 +82,7 @@ abstract class BaseEpisodesFragment : Fragment(), Toolbar.OnMenuItemClickListene swipeActions = SwipeActions(this, TAG) lifecycle.addObserver(swipeActions) - binding.lazyColumn.setContent { + binding.mainView.setContent { CustomTheme(requireContext()) { Column { InforBar(infoBarText, leftAction = leftActionState, rightAction = rightActionState, actionConfig = {swipeActions.showDialog()}) diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/DownloadsFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/DownloadsFragment.kt index 3c936069..03a9267b 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/DownloadsFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/DownloadsFragment.kt @@ -1,7 +1,7 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R -import ac.mdiq.podcini.databinding.DownloadsFragmentBinding +import ac.mdiq.podcini.databinding.ComposeFragmentBinding import ac.mdiq.podcini.net.download.service.DownloadServiceInterface import ac.mdiq.podcini.preferences.UserPreferences import ac.mdiq.podcini.preferences.UserPreferences.appPrefs @@ -20,7 +20,10 @@ import ac.mdiq.podcini.ui.actions.SwipeAction import ac.mdiq.podcini.ui.actions.SwipeActions import ac.mdiq.podcini.ui.actions.SwipeActions.NoActionSwipeAction import ac.mdiq.podcini.ui.activity.MainActivity -import ac.mdiq.podcini.ui.compose.* +import ac.mdiq.podcini.ui.compose.CustomTheme +import ac.mdiq.podcini.ui.compose.EpisodeLazyColumn +import ac.mdiq.podcini.ui.compose.EpisodeVM +import ac.mdiq.podcini.ui.compose.InforBar import ac.mdiq.podcini.ui.dialog.EpisodeFilterDialog import ac.mdiq.podcini.ui.dialog.EpisodeSortDialog import ac.mdiq.podcini.ui.dialog.SwitchQueueDialog @@ -37,7 +40,8 @@ import android.view.ViewGroup import android.widget.Toast import androidx.appcompat.widget.Toolbar import androidx.compose.foundation.layout.Column -import androidx.compose.runtime.* +import androidx.compose.runtime.mutableStateListOf +import androidx.compose.runtime.mutableStateOf import androidx.fragment.app.Fragment import androidx.lifecycle.lifecycleScope import androidx.media3.common.util.UnstableApi @@ -56,7 +60,7 @@ import java.util.* */ @UnstableApi class DownloadsFragment : Fragment(), Toolbar.OnMenuItemClickListener { - private var _binding: DownloadsFragmentBinding? = null + private var _binding: ComposeFragmentBinding? = null private val binding get() = _binding!! private var runningDownloads: Set = HashSet() @@ -74,7 +78,7 @@ import java.util.* private var displayUpArrow = false @UnstableApi override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { - _binding = DownloadsFragmentBinding.inflate(inflater) + _binding = ComposeFragmentBinding.inflate(inflater) Logd(TAG, "fragment onCreateView") toolbar = binding.toolbar @@ -88,12 +92,11 @@ import java.util.* // } displayUpArrow = parentFragmentManager.backStackEntryCount != 0 if (savedInstanceState != null) displayUpArrow = savedInstanceState.getBoolean(KEY_UP_ARROW) - (activity as MainActivity).setupToolbarToggle(toolbar, displayUpArrow) swipeActions = SwipeActions(this, TAG) swipeActions.setFilter(EpisodeFilter(EpisodeFilter.States.downloaded.name)) - binding.lazyColumn.setContent { + binding.mainView.setContent { CustomTheme(requireContext()) { Column { InforBar(infoBarText, leftAction = leftActionState, rightAction = rightActionState, actionConfig = {swipeActions.showDialog()}) diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/EpisodeInfoFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/EpisodeInfoFragment.kt index 578304b7..02d6b540 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/EpisodeInfoFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/EpisodeInfoFragment.kt @@ -1,14 +1,10 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R +import ac.mdiq.podcini.databinding.ComposeFragmentBinding import ac.mdiq.podcini.databinding.EpisodeHomeFragmentBinding -import ac.mdiq.podcini.databinding.EpisodeInfoFragmentBinding import ac.mdiq.podcini.net.download.service.DownloadServiceInterface import ac.mdiq.podcini.net.download.service.PodciniHttpClient.getHttpClient -import ac.mdiq.podcini.net.sync.SynchronizationSettings.isProviderConnected -import ac.mdiq.podcini.net.sync.SynchronizationSettings.wifiSyncEnabledKey -import ac.mdiq.podcini.net.sync.model.EpisodeAction -import ac.mdiq.podcini.net.sync.queue.SynchronizationQueueSink import ac.mdiq.podcini.net.utils.NetworkUtils.fetchHtmlSource import ac.mdiq.podcini.net.utils.NetworkUtils.isEpisodeHeadDownloadAllowed import ac.mdiq.podcini.playback.base.InTheatre @@ -16,7 +12,6 @@ import ac.mdiq.podcini.playback.base.InTheatre.curQueue import ac.mdiq.podcini.playback.service.PlaybackService.Companion.seekTo import ac.mdiq.podcini.preferences.UsageStatistics import ac.mdiq.podcini.preferences.UserPreferences -import ac.mdiq.podcini.storage.database.Episodes.setPlayState import ac.mdiq.podcini.storage.database.Queues.addToQueue import ac.mdiq.podcini.storage.database.Queues.removeFromQueue import ac.mdiq.podcini.storage.database.RealmDB.realm @@ -24,18 +19,12 @@ import ac.mdiq.podcini.storage.database.RealmDB.runOnIOScope import ac.mdiq.podcini.storage.database.RealmDB.unmanaged import ac.mdiq.podcini.storage.database.RealmDB.upsert import ac.mdiq.podcini.storage.database.RealmDB.upsertBlk -import ac.mdiq.podcini.storage.model.Episode -import ac.mdiq.podcini.storage.model.EpisodeMedia -import ac.mdiq.podcini.storage.model.Feed -import ac.mdiq.podcini.storage.model.Rating +import ac.mdiq.podcini.storage.model.* import ac.mdiq.podcini.storage.utils.DurationConverter import ac.mdiq.podcini.storage.utils.ImageResourceUtils import ac.mdiq.podcini.ui.actions.* import ac.mdiq.podcini.ui.activity.MainActivity -import ac.mdiq.podcini.ui.compose.ChaptersDialog -import ac.mdiq.podcini.ui.compose.ChooseRatingDialog -import ac.mdiq.podcini.ui.compose.CustomTheme -import ac.mdiq.podcini.ui.compose.LargeTextEditingDialog +import ac.mdiq.podcini.ui.compose.* import ac.mdiq.podcini.ui.dialog.ShareDialog import ac.mdiq.podcini.ui.utils.ShownotesCleaner import ac.mdiq.podcini.ui.utils.ThemeUtils @@ -109,7 +98,7 @@ import java.util.* */ @UnstableApi class EpisodeInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { - private var _binding: EpisodeInfoFragmentBinding? = null + private var _binding: ComposeFragmentBinding? = null private val binding get() = _binding!! private var homeFragment: EpisodeHomeFragment? = null @@ -126,7 +115,7 @@ class EpisodeInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { private var hasMedia by mutableStateOf(true) var rating by mutableStateOf(episode?.rating ?: Rating.UNRATED.code) private var inQueue by mutableStateOf(if (episode != null) curQueue.contains(episode!!) else false) - var isPlayed by mutableStateOf(episode?.isPlayed() ?: false) + var isPlayed by mutableIntStateOf(episode?.playState ?: PlayState.UNSPECIFIED.code) private var webviewData by mutableStateOf("") @@ -141,7 +130,7 @@ class EpisodeInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { super.onCreateView(inflater, container, savedInstanceState) - _binding = EpisodeInfoFragmentBinding.inflate(inflater, container, false) + _binding = ComposeFragmentBinding.inflate(inflater, container, false) Logd(TAG, "fragment onCreateView") toolbar = binding.toolbar @@ -150,7 +139,7 @@ class EpisodeInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { toolbar.setNavigationOnClickListener { parentFragmentManager.popBackStack() } toolbar.setOnMenuItemClickListener(this) - binding.composeView.setContent{ + binding.mainView.setContent{ CustomTheme(requireContext()) { MainView() } @@ -188,6 +177,9 @@ class EpisodeInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { var showChaptersDialog by remember { mutableStateOf(false) } if (showChaptersDialog && episode?.media != null) ChaptersDialog(media = episode!!.media!!, onDismissRequest = {showChaptersDialog = false}) + var showPlayStateDialog by remember { mutableStateOf(false) } + if (showPlayStateDialog) PlayStateDialog(listOf(episode!!)) { showPlayStateDialog = false } + Column { Row(modifier = Modifier.padding(start = 16.dp, end = 16.dp), verticalAlignment = Alignment.CenterVertically) { val imgLoc = if (episode != null) ImageResourceUtils.getEpisodeListImageLocation(episode!!) else null @@ -200,39 +192,40 @@ class EpisodeInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { } Row(modifier = Modifier.padding(top = 4.dp), verticalAlignment = Alignment.CenterVertically) { Spacer(modifier = Modifier.weight(0.4f)) - val playedIconRes = if (!isPlayed) R.drawable.ic_mark_unplayed else R.drawable.ic_mark_played + val playedIconRes = PlayState.fromCode(isPlayed).res Icon(imageVector = ImageVector.vectorResource(playedIconRes), tint = MaterialTheme.colorScheme.tertiary, contentDescription = "isPlayed", modifier = Modifier.background(MaterialTheme.colorScheme.tertiaryContainer).width(24.dp).height(24.dp) .clickable(onClick = { - if (isPlayed) { - setPlayState(Episode.PlayState.UNPLAYED.code, false, episode!!) - if (isProviderConnected && episode?.feed?.isLocalFeed != true && episode?.media != null) { - val actionNew: EpisodeAction = EpisodeAction.Builder(episode!!, EpisodeAction.NEW).currentTimestamp().build() - SynchronizationQueueSink.enqueueEpisodeActionIfSyncActive(requireContext(), actionNew) - } - } else { - setPlayState(Episode.PlayState.PLAYED.code, true, episode!!) - if (episode?.feed?.isLocalFeed != true && (isProviderConnected || wifiSyncEnabledKey)) { - val media: EpisodeMedia? = episode?.media - // not all items have media, Gpodder only cares about those that do - if (isProviderConnected && media != null) { - val actionPlay: EpisodeAction = EpisodeAction.Builder(episode!!, EpisodeAction.PLAY) - .currentTimestamp() - .started(media.getDuration() / 1000) - .position(media.getDuration() / 1000) - .total(media.getDuration() / 1000) - .build() - SynchronizationQueueSink.enqueueEpisodeActionIfSyncActive(requireContext(), actionPlay) - } - } - } + showPlayStateDialog = true +// if (isPlayed) { +// setPlayState(PlayState.UNPLAYED.code, false, episode!!) +// if (isProviderConnected && episode?.feed?.isLocalFeed != true && episode?.media != null) { +// val actionNew: EpisodeAction = EpisodeAction.Builder(episode!!, EpisodeAction.NEW).currentTimestamp().build() +// SynchronizationQueueSink.enqueueEpisodeActionIfSyncActive(requireContext(), actionNew) +// } +// } else { +// setPlayState(PlayState.PLAYED.code, true, episode!!) +// if (episode?.feed?.isLocalFeed != true && (isProviderConnected || wifiSyncEnabledKey)) { +// val media: EpisodeMedia? = episode?.media +// // not all items have media, Gpodder only cares about those that do +// if (isProviderConnected && media != null) { +// val actionPlay: EpisodeAction = EpisodeAction.Builder(episode!!, EpisodeAction.PLAY) +// .currentTimestamp() +// .started(media.getDuration() / 1000) +// .position(media.getDuration() / 1000) +// .total(media.getDuration() / 1000) +// .build() +// SynchronizationQueueSink.enqueueEpisodeActionIfSyncActive(requireContext(), actionPlay) +// } +// } +// } })) if (episode?.media != null) { Spacer(modifier = Modifier.weight(0.2f)) val inQueueIconRes = if (inQueue) R.drawable.ic_playlist_play else R.drawable.ic_playlist_remove Icon(imageVector = ImageVector.vectorResource(inQueueIconRes), tint = MaterialTheme.colorScheme.tertiary, contentDescription = "inQueue", modifier = Modifier.background(MaterialTheme.colorScheme.tertiaryContainer).width(24.dp).height(24.dp).clickable(onClick = { - if (inQueue) removeFromQueue(episode!!) else addToQueue(true, episode!!) + if (inQueue) removeFromQueue(episode!!) else addToQueue(episode!!) })) } Spacer(modifier = Modifier.weight(0.2f)) @@ -625,7 +618,7 @@ class EpisodeInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { if (episode != null) { rating = episode!!.rating inQueue = curQueue.contains(episode!!) - isPlayed = episode!!.isPlayed() + isPlayed = episode!!.playState } onFragmentLoaded() itemLoaded = true diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/FeedEpisodesFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/FeedEpisodesFragment.kt index cf4e3c87..be7ddd51 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/FeedEpisodesFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/FeedEpisodesFragment.kt @@ -1,7 +1,7 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R -import ac.mdiq.podcini.databinding.FeedItemListFragmentBinding +import ac.mdiq.podcini.databinding.ComposeFragmentBinding import ac.mdiq.podcini.net.download.DownloadStatus import ac.mdiq.podcini.net.feed.FeedUpdateManager import ac.mdiq.podcini.preferences.UserPreferences @@ -10,11 +10,8 @@ import ac.mdiq.podcini.storage.database.RealmDB.realm import ac.mdiq.podcini.storage.database.RealmDB.runOnIOScope import ac.mdiq.podcini.storage.database.RealmDB.upsert import ac.mdiq.podcini.storage.database.RealmDB.upsertBlk -import ac.mdiq.podcini.storage.model.Episode -import ac.mdiq.podcini.storage.model.EpisodeFilter -import ac.mdiq.podcini.storage.model.EpisodeSortOrder +import ac.mdiq.podcini.storage.model.* import ac.mdiq.podcini.storage.model.EpisodeSortOrder.Companion.fromCode -import ac.mdiq.podcini.storage.model.Feed import ac.mdiq.podcini.storage.utils.EpisodesPermutors.getPermutor import ac.mdiq.podcini.ui.actions.SwipeAction import ac.mdiq.podcini.ui.actions.SwipeActions @@ -36,7 +33,10 @@ import android.view.* import android.widget.Toast import androidx.annotation.OptIn import androidx.appcompat.widget.Toolbar -import androidx.compose.foundation.* +import androidx.compose.foundation.ExperimentalFoundationApi +import androidx.compose.foundation.background +import androidx.compose.foundation.clickable +import androidx.compose.foundation.combinedClickable import androidx.compose.foundation.layout.* import androidx.compose.material3.Icon import androidx.compose.material3.MaterialTheme @@ -55,6 +55,7 @@ import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.text.style.TextOverflow import androidx.compose.ui.unit.dp import androidx.constraintlayout.compose.ConstraintLayout +import androidx.constraintlayout.compose.Dimension import androidx.fragment.app.Fragment import androidx.lifecycle.lifecycleScope import androidx.media3.common.util.UnstableApi @@ -72,7 +73,7 @@ import java.util.concurrent.Semaphore */ @UnstableApi class FeedEpisodesFragment : Fragment(), Toolbar.OnMenuItemClickListener { - private var _binding: FeedItemListFragmentBinding? = null + private var _binding: ComposeFragmentBinding? = null private val binding get() = _binding!! private lateinit var swipeActions: SwipeActions @@ -87,6 +88,7 @@ import java.util.concurrent.Semaphore private var headerCreated = false private var feedID: Long = 0 private var feed by mutableStateOf(null) + var rating by mutableStateOf(Rating.UNRATED.code) private val episodes = mutableStateListOf() private val vms = mutableStateListOf() @@ -112,7 +114,7 @@ import java.util.concurrent.Semaphore @UnstableApi override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { Logd(TAG, "fragment onCreateView") - _binding = FeedItemListFragmentBinding.inflate(inflater) + _binding = ComposeFragmentBinding.inflate(inflater) binding.toolbar.inflateMenu(R.menu.feed_episodes) binding.toolbar.setOnMenuItemClickListener(this) @@ -169,7 +171,7 @@ import java.util.concurrent.Semaphore requireActivity().supportFragmentManager.executePendingTransactions() } Column { - FeedEpisodesHeader(activity = (activity as MainActivity), feed = feed, filterButColor = filterButColor.value, filterClickCB = {filterClick()}, filterLongClickCB = {filterLongClick()}) + FeedEpisodesHeader(activity = (activity as MainActivity), filterButColor = filterButColor.value, filterClickCB = {filterClick()}, filterLongClickCB = {filterLongClick()}) InforBar(infoBarText, leftAction = leftActionState, rightAction = rightActionState, actionConfig = { swipeActions.showDialog() }) @@ -225,8 +227,14 @@ import java.util.concurrent.Semaphore @kotlin.OptIn(ExperimentalFoundationApi::class) @Composable - fun FeedEpisodesHeader(activity: MainActivity, feed: Feed?, filterButColor: Color, filterClickCB: ()->Unit, filterLongClickCB: ()->Unit) { + fun FeedEpisodesHeader(activity: MainActivity, filterButColor: Color, filterClickCB: ()->Unit, filterLongClickCB: ()->Unit) { val textColor = MaterialTheme.colorScheme.onSurface + var showChooseRatingDialog by remember { mutableStateOf(false) } + if (showChooseRatingDialog) ChooseRatingDialog(listOf(feed!!)) { + showChooseRatingDialog = false + feed = realm.query(Feed::class).query("id == $0", feed!!.id).first().find()!! + rating = feed!!.rating + } ConstraintLayout(modifier = Modifier.fillMaxWidth().height(120.dp)) { val (bgImage, bgColor, controlRow, image1, image2, imgvCover, taColumn) = createRefs() AsyncImage(model = feed?.imageUrl?:"", contentDescription = "bgImage", contentScale = ContentScale.FillBounds, @@ -238,7 +246,7 @@ import java.util.concurrent.Semaphore start.linkTo(parent.start) end.linkTo(parent.end) }) - Box(modifier = Modifier.fillMaxSize().background(MaterialTheme.colorScheme.surface.copy(alpha = 0.7f)) + Box(modifier = Modifier.fillMaxSize().background(MaterialTheme.colorScheme.surface.copy(alpha = 0.75f)) .constrainAs(bgColor) { bottom.linkTo(parent.bottom) top.linkTo(parent.top) @@ -249,20 +257,30 @@ import java.util.concurrent.Semaphore .constrainAs(controlRow) { bottom.linkTo(parent.bottom) start.linkTo(parent.start) + width = Dimension.fillToConstraints }, verticalAlignment = Alignment.CenterVertically) { Spacer(modifier = Modifier.weight(0.7f)) + val ratingIconRes = Rating.fromCode(rating).res + Icon(imageVector = ImageVector.vectorResource(ratingIconRes), tint = MaterialTheme.colorScheme.tertiary, contentDescription = "rating", + modifier = Modifier.background(MaterialTheme.colorScheme.tertiaryContainer).width(30.dp).height(30.dp).clickable(onClick = { + showChooseRatingDialog = true + })) + Spacer(modifier = Modifier.weight(0.2f)) + Icon(imageVector = ImageVector.vectorResource(R.drawable.arrows_sort), tint = textColor, contentDescription = "butSort", + modifier = Modifier.width(40.dp).height(40.dp).padding(3.dp).clickable(onClick = { SingleFeedSortDialog(feed).show(childFragmentManager, "SortDialog") })) + Spacer(modifier = Modifier.width(15.dp)) Icon(imageVector = ImageVector.vectorResource(R.drawable.ic_filter_white), tint = if (filterButColor == Color.White) textColor else filterButColor, contentDescription = "butFilter", modifier = Modifier.width(40.dp).height(40.dp).padding(3.dp).combinedClickable(onClick = filterClickCB, onLongClick = filterLongClickCB)) Spacer(modifier = Modifier.width(15.dp)) Icon(imageVector = ImageVector.vectorResource(R.drawable.ic_settings_white), tint = textColor, contentDescription = "butShowSettings", modifier = Modifier.width(40.dp).height(40.dp).padding(3.dp).clickable(onClick = { if (feed != null) { - val fragment = FeedSettingsFragment.newInstance(feed) + val fragment = FeedSettingsFragment.newInstance(feed!!) activity.loadChildFragment(fragment, TransitionEffect.SLIDE) } })) - Spacer(modifier = Modifier.weight(0.5f)) - Text(episodes.size.toString() + " / " + feed?.episodes?.size?.toString(), textAlign = TextAlign.Center, color = Color.White, style = MaterialTheme.typography.bodyLarge) + Spacer(modifier = Modifier.weight(0.4f)) + Text(episodes.size.toString() + " / " + feed?.episodes?.size?.toString(), textAlign = TextAlign.End, color = textColor, fontWeight = FontWeight.Bold, style = MaterialTheme.typography.bodyLarge) } // Icon(imageVector = ImageVector.vectorResource(R.drawable.ic_rounded_corner_left), contentDescription = "left_corner", // Modifier.width(12.dp).height(12.dp).constrainAs(image1) { @@ -274,21 +292,23 @@ import java.util.concurrent.Semaphore // bottom.linkTo(parent.bottom) // end.linkTo(parent.end) // }) - AsyncImage(model = feed?.imageUrl?:"", contentDescription = "imgvCover", error = painterResource(R.mipmap.ic_launcher), - modifier = Modifier.width(120.dp).height(120.dp).padding(start = 16.dp, end = 16.dp, bottom = 12.dp).constrainAs(imgvCover) { - bottom.linkTo(parent.bottom) - start.linkTo(parent.start) - }.clickable(onClick = { - if (feed != null) { - val fragment = FeedInfoFragment.newInstance(feed) - activity.loadChildFragment(fragment, TransitionEffect.SLIDE) - } - })) - Column(Modifier.fillMaxWidth().constrainAs(taColumn) { - top.linkTo(imgvCover.top) - start.linkTo(imgvCover.end) }) { - Text(feed?.title?:"", color = textColor, fontWeight = FontWeight.Bold, style = MaterialTheme.typography.bodyLarge, modifier = Modifier.fillMaxWidth(), maxLines = 2, overflow = TextOverflow.Ellipsis) - Text(feed?.author?:"", color = textColor, fontWeight = FontWeight.Bold, style = MaterialTheme.typography.bodyMedium, modifier = Modifier.fillMaxWidth(), maxLines = 1, overflow = TextOverflow.Ellipsis) + Row(verticalAlignment = Alignment.Top, modifier = Modifier.fillMaxWidth().padding(bottom = 12.dp).constrainAs(imgvCover) { + top.linkTo(parent.top) + start.linkTo(parent.start) + end.linkTo(parent.end) + width = Dimension.fillToConstraints + }) { + AsyncImage(model = feed?.imageUrl ?: "", contentDescription = "imgvCover", error = painterResource(R.mipmap.ic_launcher), + modifier = Modifier.width(100.dp).height(100.dp).padding(start = 16.dp, end = 16.dp).clickable(onClick = { + if (feed != null) { + val fragment = FeedInfoFragment.newInstance(feed!!) + activity.loadChildFragment(fragment, TransitionEffect.SLIDE) + } + })) + Column(Modifier.padding(top = 10.dp)) { + Text(feed?.title ?: "", color = textColor, fontWeight = FontWeight.Bold, style = MaterialTheme.typography.bodyLarge, modifier = Modifier.fillMaxWidth(), maxLines = 2, overflow = TextOverflow.Ellipsis) + Text(feed?.author ?: "", color = textColor, style = MaterialTheme.typography.bodyMedium, modifier = Modifier.fillMaxWidth(), maxLines = 1, overflow = TextOverflow.Ellipsis) + } } } } @@ -386,7 +406,7 @@ import java.util.concurrent.Semaphore } catch (e: InterruptedException) { throw RuntimeException(e) } }.start() } - R.id.sort_items -> SingleFeedSortDialog(feed).show(childFragmentManager, "SortDialog") +// R.id.sort_items -> SingleFeedSortDialog(feed).show(childFragmentManager, "SortDialog") // R.id.filter_items -> {} // R.id.settings -> { // if (feed != null) { @@ -651,6 +671,7 @@ import java.util.concurrent.Semaphore } withContext(Dispatchers.Main) { Logd(TAG, "loadItems subscribe called ${feed?.title}") + rating = feed?.rating ?: Rating.UNRATED.code swipeActions.setFilter(feed?.episodeFilter) refreshHeaderView() // if (feed != null) { diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/FeedInfoFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/FeedInfoFragment.kt index cd9ecee7..2327dfbd 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/FeedInfoFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/FeedInfoFragment.kt @@ -1,8 +1,8 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R +import ac.mdiq.podcini.databinding.ComposeFragmentBinding import ac.mdiq.podcini.databinding.EditTextDialogBinding -import ac.mdiq.podcini.databinding.FeedinfoBinding import ac.mdiq.podcini.net.feed.FeedUpdateManager.runOnce import ac.mdiq.podcini.net.feed.discovery.CombinedSearcher import ac.mdiq.podcini.net.utils.HtmlToPlainText @@ -42,8 +42,11 @@ import android.view.ViewGroup import androidx.activity.result.contract.ActivityResultContracts import androidx.appcompat.app.AlertDialog import androidx.appcompat.widget.Toolbar -import androidx.compose.foundation.* +import androidx.compose.foundation.background +import androidx.compose.foundation.clickable import androidx.compose.foundation.layout.* +import androidx.compose.foundation.rememberScrollState +import androidx.compose.foundation.verticalScroll import androidx.compose.material3.Button import androidx.compose.material3.Icon import androidx.compose.material3.MaterialTheme @@ -62,6 +65,7 @@ import androidx.compose.ui.text.input.TextFieldValue import androidx.compose.ui.text.style.TextOverflow import androidx.compose.ui.unit.dp import androidx.constraintlayout.compose.ConstraintLayout +import androidx.constraintlayout.compose.Dimension import androidx.documentfile.provider.DocumentFile import androidx.fragment.app.Fragment import androidx.fragment.compose.AndroidFragment @@ -80,7 +84,7 @@ import java.util.concurrent.ExecutionException @UnstableApi class FeedInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { - private var _binding: FeedinfoBinding? = null + private var _binding: ComposeFragmentBinding? = null private val binding get() = _binding!! private lateinit var feed: Feed @@ -96,7 +100,7 @@ class FeedInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { } override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { - _binding = FeedinfoBinding.inflate(inflater) + _binding = ComposeFragmentBinding.inflate(inflater) Logd(TAG, "fragment onCreateView") toolbar = binding.toolbar toolbar.title = "" @@ -108,7 +112,7 @@ class FeedInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { txtvAuthor = feed.author ?: "" txtvUrl = feed.downloadUrl - binding.mainUI.setContent { + binding.mainView.setContent { CustomTheme(requireContext()) { if (showRemoveFeedDialog) RemoveFeedDialog(listOf(feed), onDismissRequest = {showRemoveFeedDialog = false}) { (activity as MainActivity).loadFragment(UserPreferences.defaultPage, null) @@ -200,21 +204,23 @@ class FeedInfoFragment : Fragment(), Toolbar.OnMenuItemClickListener { // bottom.linkTo(parent.bottom) // end.linkTo(parent.end) // }) - AsyncImage(model = feed.imageUrl?:"", contentDescription = "imgvCover", error = painterResource(R.mipmap.ic_launcher), - modifier = Modifier.width(120.dp).height(120.dp).padding(start = 16.dp, end = 16.dp, bottom = 12.dp).constrainAs(imgvCover) { - bottom.linkTo(parent.bottom) - start.linkTo(parent.start) - }.clickable(onClick = { + Row(verticalAlignment = Alignment.Top, modifier = Modifier.fillMaxWidth().padding(bottom = 12.dp).constrainAs(imgvCover) { + top.linkTo(parent.top) + start.linkTo(parent.start) + end.linkTo(parent.end) + width = Dimension.fillToConstraints + }) { + AsyncImage(model = feed.imageUrl ?: "", contentDescription = "imgvCover", error = painterResource(R.mipmap.ic_launcher), + modifier = Modifier.width(100.dp).height(100.dp).padding(start = 16.dp, end = 16.dp).clickable(onClick = { // if (feed != null) { // val fragment = FeedInfoFragment.newInstance(feed) // (activity as MainActivity).loadChildFragment(fragment, TransitionEffect.SLIDE) // } - })) - Column(Modifier.constrainAs(taColumn) { - top.linkTo(imgvCover.top) - start.linkTo(imgvCover.end) }) { - Text(feed.title ?:"", color = textColor, fontWeight = FontWeight.Bold, style = MaterialTheme.typography.bodyLarge, modifier = Modifier.fillMaxWidth(), maxLines = 2, overflow = TextOverflow.Ellipsis) - Text(text = txtvAuthor, color = textColor, fontWeight = FontWeight.Bold, style = MaterialTheme.typography.bodyMedium, modifier = Modifier.fillMaxWidth(), maxLines = 1, overflow = TextOverflow.Ellipsis) + })) + Column(Modifier.padding(top = 10.dp)) { + Text(feed.title ?: "", color = textColor, fontWeight = FontWeight.Bold, style = MaterialTheme.typography.bodyLarge, modifier = Modifier.fillMaxWidth(), maxLines = 2, overflow = TextOverflow.Ellipsis) + Text(text = txtvAuthor, color = textColor, style = MaterialTheme.typography.bodyMedium, modifier = Modifier.fillMaxWidth(), maxLines = 1, overflow = TextOverflow.Ellipsis) + } } } } diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/LogsFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/LogsFragment.kt index 5a2f6c5f..09229034 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/LogsFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/LogsFragment.kt @@ -1,9 +1,8 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R -import ac.mdiq.podcini.databinding.LogsFragmentBinding +import ac.mdiq.podcini.databinding.ComposeFragmentBinding import ac.mdiq.podcini.net.feed.FeedUpdateManager -import ac.mdiq.podcini.storage.database.Episodes.getEpisodeByGuidOrUrl import ac.mdiq.podcini.storage.database.Feeds.getFeed import ac.mdiq.podcini.storage.database.Feeds.getFeedByTitleAndAuthor import ac.mdiq.podcini.storage.database.RealmDB.realm @@ -14,8 +13,8 @@ import ac.mdiq.podcini.storage.utils.DownloadResultComparator import ac.mdiq.podcini.ui.actions.DownloadActionButton import ac.mdiq.podcini.ui.activity.MainActivity import ac.mdiq.podcini.ui.activity.ShareReceiverActivity.Companion.receiveShared -import ac.mdiq.podcini.ui.compose.CustomTheme import ac.mdiq.podcini.ui.compose.ConfirmAddYoutubeEpisode +import ac.mdiq.podcini.ui.compose.CustomTheme import ac.mdiq.podcini.util.EventFlow import ac.mdiq.podcini.util.FlowEvent import ac.mdiq.podcini.util.Logd @@ -51,7 +50,6 @@ import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.vector.ImageVector -import androidx.compose.ui.res.painterResource import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.vectorResource import androidx.compose.ui.text.style.TextOverflow @@ -66,23 +64,26 @@ import kotlinx.coroutines.withContext import java.util.* class LogsFragment : Fragment(), Toolbar.OnMenuItemClickListener { - private var _binding: LogsFragmentBinding? = null + private var _binding: ComposeFragmentBinding? = null private val binding get() = _binding!! private val shareLogs = mutableStateListOf() private val subscriptionLogs = mutableStateListOf() private val downloadLogs = mutableStateListOf() -// private var showShared by mutableStateOf(true) -// private var showSubscription by mutableStateOf(false) + private var displayUpArrow = false override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { Logd(TAG, "fragment onCreateView") - _binding = LogsFragmentBinding.inflate(inflater) + _binding = ComposeFragmentBinding.inflate(inflater) binding.toolbar.inflateMenu(R.menu.logs) binding.toolbar.setOnMenuItemClickListener(this) - binding.lazyColumn.setContent { + displayUpArrow = parentFragmentManager.backStackEntryCount != 0 + if (savedInstanceState != null) displayUpArrow = savedInstanceState.getBoolean(KEY_UP_ARROW) + (activity as MainActivity).setupToolbarToggle(binding.toolbar, displayUpArrow) + + binding.mainView.setContent { CustomTheme(requireContext()) { when { downloadLogs.isNotEmpty() -> DownloadLogView() @@ -95,6 +96,11 @@ class LogsFragment : Fragment(), Toolbar.OnMenuItemClickListener { return binding.root } + override fun onSaveInstanceState(outState: Bundle) { + outState.putBoolean(KEY_UP_ARROW, displayUpArrow) + super.onSaveInstanceState(outState) + } + override fun onDestroyView() { Logd(TAG, "onDestroyView") _binding = null @@ -489,5 +495,6 @@ class LogsFragment : Fragment(), Toolbar.OnMenuItemClickListener { companion object { val TAG: String = LogsFragment::class.simpleName ?: "Anonymous" + private const val KEY_UP_ARROW = "up_arrow" } } diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/NavDrawerFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/NavDrawerFragment.kt index 205495b4..bf2abb40 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/NavDrawerFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/NavDrawerFragment.kt @@ -1,8 +1,6 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R -import ac.mdiq.podcini.databinding.NavListBinding -import ac.mdiq.podcini.playback.base.InTheatre.curQueue import ac.mdiq.podcini.preferences.UserPreferences.hiddenDrawerItems import ac.mdiq.podcini.storage.database.Episodes.getEpisodesCount import ac.mdiq.podcini.storage.database.Feeds.getFeedCount @@ -42,6 +40,7 @@ import androidx.compose.runtime.* import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.vector.ImageVector +import androidx.compose.ui.platform.ComposeView import androidx.compose.ui.res.painterResource import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.vectorResource @@ -57,30 +56,30 @@ import com.google.android.material.bottomsheet.BottomSheetBehavior import com.google.android.material.shape.MaterialShapeDrawable import com.google.android.material.shape.ShapeAppearanceModel import io.realm.kotlin.query.Sort -import kotlinx.coroutines.* +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.launch +import kotlinx.coroutines.withContext import kotlin.math.max class NavDrawerFragment : Fragment(), OnSharedPreferenceChangeListener { val TAG = this::class.simpleName ?: "Anonymous" - - private var _binding: NavListBinding? = null - private val binding get() = _binding!! private val feeds = mutableStateListOf() override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { super.onCreateView(inflater, container, savedInstanceState) - _binding = NavListBinding.inflate(inflater) checkHiddenItems() getRecentPodcasts() - binding.mainView.setContent { - CustomTheme(requireContext()) { - MainView() + val composeView = ComposeView(requireContext()).apply { + setContent { + CustomTheme(requireContext()) { + MainView() + } } } Logd(TAG, "fragment onCreateView") - setupDrawerRoundBackground(binding.root) - ViewCompat.setOnApplyWindowInsetsListener(binding.root) { view: View, insets: WindowInsetsCompat -> + setupDrawerRoundBackground(composeView) + ViewCompat.setOnApplyWindowInsetsListener(composeView) { view: View, insets: WindowInsetsCompat -> val bars: Insets = insets.getInsets(WindowInsetsCompat.Type.systemBars()) view.setPadding(bars.left, bars.top, bars.right, 0) var navigationBarHeight = 0f @@ -94,7 +93,7 @@ class NavDrawerFragment : Fragment(), OnSharedPreferenceChangeListener { insets } prefs!!.registerOnSharedPreferenceChangeListener(this) - return binding.root + return composeView } private fun checkHiddenItems() { @@ -171,7 +170,6 @@ class NavDrawerFragment : Fragment(), OnSharedPreferenceChangeListener { override fun onDestroyView() { Logd(TAG, "onDestroyView") - _binding = null prefs!!.unregisterOnSharedPreferenceChangeListener(this) super.onDestroyView() } @@ -239,7 +237,7 @@ class NavDrawerFragment : Fragment(), OnSharedPreferenceChangeListener { } fun getLastNavFragment(): String { - val lastFragment: String = prefs!!.getString(PREF_LAST_FRAGMENT_TAG, SubscriptionsFragment.TAG)?:"" + val lastFragment: String = prefs?.getString(PREF_LAST_FRAGMENT_TAG, SubscriptionsFragment.TAG)?:"" return lastFragment } diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/OnlineFeedFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/OnlineFeedFragment.kt index c0ba7767..96636898 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/OnlineFeedFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/OnlineFeedFragment.kt @@ -1,7 +1,7 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R -import ac.mdiq.podcini.databinding.OnlineFeedviewFragmentBinding +import ac.mdiq.podcini.databinding.ComposeFragmentBinding import ac.mdiq.podcini.net.download.service.DownloadServiceInterface import ac.mdiq.podcini.net.feed.FeedBuilder import ac.mdiq.podcini.net.feed.FeedUrlNotFoundException @@ -81,7 +81,7 @@ import kotlin.concurrent.Volatile */ @OptIn(UnstableApi::class) class OnlineFeedFragment : Fragment() { - private var _binding: OnlineFeedviewFragmentBinding? = null + private var _binding: ComposeFragmentBinding? = null private val binding get() = _binding!! private var displayUpArrow = false @@ -124,7 +124,7 @@ class OnlineFeedFragment : Fragment() { @OptIn(UnstableApi::class) override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { Logd(TAG, "fragment onCreateView") - _binding = OnlineFeedviewFragmentBinding.inflate(layoutInflater) + _binding = ComposeFragmentBinding.inflate(layoutInflater) displayUpArrow = parentFragmentManager.backStackEntryCount != 0 if (savedInstanceState != null) displayUpArrow = savedInstanceState.getBoolean(KEY_UP_ARROW) (activity as MainActivity).setupToolbarToggle(binding.toolbar, displayUpArrow) diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/QueuesFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/QueuesFragment.kt index 2e874b3b..a424e76d 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/QueuesFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/QueuesFragment.kt @@ -2,7 +2,7 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R import ac.mdiq.podcini.databinding.CheckboxDoNotShowAgainBinding -import ac.mdiq.podcini.databinding.QueueFragmentBinding +import ac.mdiq.podcini.databinding.ComposeFragmentBinding import ac.mdiq.podcini.net.download.DownloadStatus import ac.mdiq.podcini.net.feed.FeedUpdateManager import ac.mdiq.podcini.playback.base.InTheatre.curQueue @@ -85,7 +85,7 @@ import kotlin.math.max @UnstableApi class QueuesFragment : Fragment(), Toolbar.OnMenuItemClickListener { - private var _binding: QueueFragmentBinding? = null + private var _binding: ComposeFragmentBinding? = null private val binding get() = _binding!! private lateinit var emptyViewHandler: EmptyViewHandler @@ -126,7 +126,7 @@ import kotlin.math.max @UnstableApi override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { super.onCreateView(inflater, container, savedInstanceState) - _binding = QueueFragmentBinding.inflate(inflater) + _binding = ComposeFragmentBinding.inflate(inflater) Logd(TAG, "fragment onCreateView") toolbar = binding.toolbar diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/QuickDiscoveryFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/QuickDiscoveryFragment.kt index b08f2f21..de1d35a6 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/QuickDiscoveryFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/QuickDiscoveryFragment.kt @@ -2,7 +2,10 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.BuildConfig import ac.mdiq.podcini.R -import ac.mdiq.podcini.databinding.* +import ac.mdiq.podcini.databinding.ComposeFragmentBinding +import ac.mdiq.podcini.databinding.QuickFeedDiscoveryBinding +import ac.mdiq.podcini.databinding.QuickFeedDiscoveryItemBinding +import ac.mdiq.podcini.databinding.SelectCountryDialogBinding import ac.mdiq.podcini.net.feed.discovery.ItunesTopListLoader import ac.mdiq.podcini.net.feed.discovery.ItunesTopListLoader.Companion.prefs import ac.mdiq.podcini.net.feed.discovery.PodcastSearchResult @@ -10,9 +13,9 @@ import ac.mdiq.podcini.storage.database.Feeds.getFeedList import ac.mdiq.podcini.ui.activity.MainActivity import ac.mdiq.podcini.ui.compose.CustomTheme import ac.mdiq.podcini.ui.compose.OnlineFeedItem -import ac.mdiq.podcini.util.Logd import ac.mdiq.podcini.util.EventFlow import ac.mdiq.podcini.util.FlowEvent +import ac.mdiq.podcini.util.Logd import android.content.DialogInterface import android.os.Bundle import android.util.DisplayMetrics @@ -24,7 +27,6 @@ import android.view.ViewGroup import android.widget.* import androidx.annotation.OptIn import androidx.appcompat.widget.Toolbar -import androidx.compose.foundation.background import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.padding @@ -37,7 +39,6 @@ import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.runtime.* import androidx.compose.ui.Modifier -import androidx.compose.ui.graphics.Color import androidx.compose.ui.res.stringResource import androidx.compose.ui.unit.dp import androidx.constraintlayout.compose.ConstraintLayout @@ -255,7 +256,7 @@ class QuickDiscoveryFragment : Fragment(), AdapterView.OnItemClickListener { * Searches iTunes store for top podcasts and displays results in a list. */ class DiscoveryFragment : Fragment(), Toolbar.OnMenuItemClickListener { - private var _binding: FragmentSearchResultsBinding? = null + private var _binding: ComposeFragmentBinding? = null private val binding get() = _binding!! private lateinit var toolbar: MaterialToolbar @@ -295,7 +296,7 @@ class QuickDiscoveryFragment : Fragment(), AdapterView.OnItemClickListener { @OptIn(UnstableApi::class) override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { // Inflate the layout for this fragment - _binding = FragmentSearchResultsBinding.inflate(inflater) + _binding = ComposeFragmentBinding.inflate(inflater) Logd(TAG, "fragment onCreateView") binding.mainView.setContent { CustomTheme(requireContext()) { diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/SearchResultsFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/SearchResultsFragment.kt index 1b5a8be2..2b494745 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/SearchResultsFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/SearchResultsFragment.kt @@ -1,7 +1,7 @@ package ac.mdiq.podcini.ui.fragment import ac.mdiq.podcini.R -import ac.mdiq.podcini.databinding.FragmentSearchResultsBinding +import ac.mdiq.podcini.databinding.ComposeFragmentBinding import ac.mdiq.podcini.net.feed.discovery.PodcastSearchResult import ac.mdiq.podcini.net.feed.discovery.PodcastSearcher import ac.mdiq.podcini.net.feed.discovery.PodcastSearcherRegistry @@ -48,7 +48,7 @@ import kotlinx.coroutines.withContext class SearchResultsFragment : Fragment() { - private var _binding: FragmentSearchResultsBinding? = null + private var _binding: ComposeFragmentBinding? = null private val binding get() = _binding!! private var searchProvider: PodcastSearcher? = null @@ -74,7 +74,7 @@ class SearchResultsFragment : Fragment() { } @UnstableApi override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View { - _binding = FragmentSearchResultsBinding.inflate(inflater) + _binding = ComposeFragmentBinding.inflate(inflater) Logd(TAG, "fragment onCreateView") binding.mainView.setContent { CustomTheme(requireContext()) { diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/SubscriptionsFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/SubscriptionsFragment.kt index c0fb25dd..aeb7bd51 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/SubscriptionsFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/fragment/SubscriptionsFragment.kt @@ -381,7 +381,7 @@ class SubscriptionsFragment : Fragment(), Toolbar.OnMenuItemClickListener { val dir = 1 - 2*feedOrderDir // get from 0, 1 to 1, -1 val comparator: Comparator = when (feedOrder) { FeedSortOrder.UNPLAYED_NEW_OLD.index -> { - val queryString = "feedId == $0 AND (playState == ${Episode.PlayState.NEW.code} OR playState == ${Episode.PlayState.UNPLAYED.code})" + val queryString = "feedId == $0 AND (playState == ${PlayState.NEW.code} OR playState == ${PlayState.UNPLAYED.code})" val counterMap: MutableMap = mutableMapOf() for (f in feedList_) { val c = realm.query(Episode::class).query(queryString, f.id).count().find() @@ -403,7 +403,7 @@ class SubscriptionsFragment : Fragment(), Toolbar.OnMenuItemClickListener { } } FeedSortOrder.MOST_PLAYED.index -> { - val queryString = "feedId == $0 AND playState == ${Episode.PlayState.PLAYED.code}" + val queryString = "feedId == $0 AND playState == ${PlayState.PLAYED.code}" val counterMap: MutableMap = mutableMapOf() for (f in feedList_) { val c = realm.query(Episode::class).query(queryString, f.id).count().find() @@ -436,7 +436,7 @@ class SubscriptionsFragment : Fragment(), Toolbar.OnMenuItemClickListener { } FeedSortOrder.LAST_UPDATED_UNPLAYED_NEW_OLD.index -> { val queryString = - "feedId == $0 AND (playState == ${Episode.PlayState.NEW.code} OR playState == ${Episode.PlayState.UNPLAYED.code}) SORT(pubDate DESC)" + "feedId == $0 AND (playState == ${PlayState.NEW.code} OR playState == ${PlayState.UNPLAYED.code}) SORT(pubDate DESC)" val counterMap: MutableMap = mutableMapOf() for (f in feedList_) { val d = realm.query(Episode::class).query(queryString, f.id).first().find()?.pubDate ?: 0L @@ -458,7 +458,7 @@ class SubscriptionsFragment : Fragment(), Toolbar.OnMenuItemClickListener { } FeedSortOrder.MOST_DOWNLOADED_UNPLAYED.index -> { val queryString = - "feedId == $0 AND (playState == ${Episode.PlayState.NEW.code} OR playState == ${Episode.PlayState.UNPLAYED.code}) AND media.downloaded == true" + "feedId == $0 AND (playState == ${PlayState.NEW.code} OR playState == ${PlayState.UNPLAYED.code}) AND media.downloaded == true" val counterMap: MutableMap = mutableMapOf() for (f in feedList_) { val c = realm.query(Episode::class).query(queryString, f.id).count().find() @@ -469,7 +469,7 @@ class SubscriptionsFragment : Fragment(), Toolbar.OnMenuItemClickListener { } // doing FEED_ORDER_NEW else -> { - val queryString = "feedId == $0 AND playState == ${Episode.PlayState.NEW.code}" + val queryString = "feedId == $0 AND playState == ${PlayState.NEW.code}" val counterMap: MutableMap = mutableMapOf() for (f in feedList_) { val c = realm.query(Episode::class).query(queryString, f.id).count().find() @@ -954,13 +954,13 @@ class SubscriptionsFragment : Fragment(), Toolbar.OnMenuItemClickListener { } }) ) - if (feed.rating != Rating.UNRATED.code) - Icon(imageVector = ImageVector.vectorResource(Rating.fromCode(feed.rating).res), tint = MaterialTheme.colorScheme.tertiary, - contentDescription = "rating", - modifier = Modifier.background(MaterialTheme.colorScheme.tertiaryContainer).constrainAs(rating) { - start.linkTo(parent.start) - centerVerticallyTo(coverImage) - }) +// if (feed.rating != Rating.UNRATED.code) +// Icon(imageVector = ImageVector.vectorResource(Rating.fromCode(feed.rating).res), tint = MaterialTheme.colorScheme.tertiary, +// contentDescription = "rating", +// modifier = Modifier.background(MaterialTheme.colorScheme.tertiaryContainer).constrainAs(rating) { +// start.linkTo(parent.start) +// centerVerticallyTo(coverImage) +// }) } val textColor = MaterialTheme.colorScheme.onSurface Column(Modifier.weight(1f).padding(start = 10.dp).combinedClickable(onClick = { @@ -984,8 +984,13 @@ class SubscriptionsFragment : Fragment(), Toolbar.OnMenuItemClickListener { } Logd(TAG, "long clicked: ${feed.title}") })) { - Text(feed.title ?: "No title", color = textColor, maxLines = 1, overflow = TextOverflow.Ellipsis, - style = MaterialTheme.typography.bodyMedium.copy(fontWeight = FontWeight.Bold)) + Row { + if (feed.rating != Rating.UNRATED.code) + Icon(imageVector = ImageVector.vectorResource(Rating.fromCode(feed.rating).res), tint = MaterialTheme.colorScheme.tertiary, contentDescription = "rating", + modifier = Modifier.background(MaterialTheme.colorScheme.tertiaryContainer)) + Text(feed.title ?: "No title", color = textColor, maxLines = 1, overflow = TextOverflow.Ellipsis, + style = MaterialTheme.typography.bodyMedium.copy(fontWeight = FontWeight.Bold)) + } Text(feed.author ?: "No author", color = textColor, maxLines = 1, overflow = TextOverflow.Ellipsis, style = MaterialTheme.typography.bodyMedium) Row(Modifier.padding(top = 5.dp)) { val measureString = remember { NumberFormat.getInstance().format(feed.episodes.size.toLong()) + " : " + diff --git a/app/src/main/kotlin/ac/mdiq/podcini/ui/statistics/StatisticsFragment.kt b/app/src/main/kotlin/ac/mdiq/podcini/ui/statistics/StatisticsFragment.kt index 06bae5ac..fb383644 100644 --- a/app/src/main/kotlin/ac/mdiq/podcini/ui/statistics/StatisticsFragment.kt +++ b/app/src/main/kotlin/ac/mdiq/podcini/ui/statistics/StatisticsFragment.kt @@ -447,7 +447,7 @@ class StatisticsFragment : Fragment() { else { // progress import does not include playedDuration if (includeMarkedAsPlayed) { - if (m.playbackCompletionTime > 0 || m.episodeOrFetch()?.playState == Episode.PlayState.PLAYED.code) + if (m.playbackCompletionTime > 0 || m.episodeOrFetch()?.playState == PlayState.PLAYED.code) dur += m.duration else if (m.position > 0) dur += m.position } else dur += m.position diff --git a/app/src/main/res/drawable/baseline_build_24.xml b/app/src/main/res/drawable/baseline_build_24.xml new file mode 100644 index 00000000..cbc92c8d --- /dev/null +++ b/app/src/main/res/drawable/baseline_build_24.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/app/src/main/res/drawable/baseline_download_done_24.xml b/app/src/main/res/drawable/baseline_download_done_24.xml new file mode 100644 index 00000000..74b5d32e --- /dev/null +++ b/app/src/main/res/drawable/baseline_download_done_24.xml @@ -0,0 +1,7 @@ + + + + + + + diff --git a/app/src/main/res/drawable/baseline_fiber_new_24.xml b/app/src/main/res/drawable/baseline_fiber_new_24.xml new file mode 100644 index 00000000..979c0465 --- /dev/null +++ b/app/src/main/res/drawable/baseline_fiber_new_24.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/app/src/main/res/drawable/baseline_local_play_24.xml b/app/src/main/res/drawable/baseline_local_play_24.xml new file mode 100644 index 00000000..e8e09168 --- /dev/null +++ b/app/src/main/res/drawable/baseline_local_play_24.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/app/src/main/res/drawable/baseline_new_label_24.xml b/app/src/main/res/drawable/baseline_new_label_24.xml new file mode 100644 index 00000000..3eb7690c --- /dev/null +++ b/app/src/main/res/drawable/baseline_new_label_24.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/app/src/main/res/drawable/baseline_play_circle_outline_24.xml b/app/src/main/res/drawable/baseline_play_circle_outline_24.xml new file mode 100644 index 00000000..b1bdebfe --- /dev/null +++ b/app/src/main/res/drawable/baseline_play_circle_outline_24.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/app/src/main/res/drawable/baseline_play_disabled_24.xml b/app/src/main/res/drawable/baseline_play_disabled_24.xml new file mode 100644 index 00000000..fc25b014 --- /dev/null +++ b/app/src/main/res/drawable/baseline_play_disabled_24.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/app/src/main/res/drawable/baseline_visibility_off_24.xml b/app/src/main/res/drawable/baseline_visibility_off_24.xml new file mode 100644 index 00000000..5993ca39 --- /dev/null +++ b/app/src/main/res/drawable/baseline_visibility_off_24.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/app/src/main/res/drawable/baseline_watch_later_24.xml b/app/src/main/res/drawable/baseline_watch_later_24.xml new file mode 100644 index 00000000..711969d7 --- /dev/null +++ b/app/src/main/res/drawable/baseline_watch_later_24.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/app/src/main/res/layout/addfeed.xml b/app/src/main/res/layout/addfeed.xml index d0efb9d0..17a6e5d8 100644 --- a/app/src/main/res/layout/addfeed.xml +++ b/app/src/main/res/layout/addfeed.xml @@ -4,26 +4,18 @@ xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent" - android:orientation="vertical"> + android:orientation="vertical" + android:fitsSystemWindows="true"> - - - - - + android:minHeight="?android:attr/actionBarSize" + android:theme="?attr/actionBarTheme" + app:title="@string/add_feed_label" + app:navigationContentDescription="@string/toolbar_back_button_content_description" + app:navigationIcon="?homeAsUpIndicator" /> - - - - - - - - - - - - - - - - diff --git a/app/src/main/res/layout/episode_info_fragment.xml b/app/src/main/res/layout/compose_fragment.xml similarity index 72% rename from app/src/main/res/layout/episode_info_fragment.xml rename to app/src/main/res/layout/compose_fragment.xml index 179afb08..6305ceed 100644 --- a/app/src/main/res/layout/episode_info_fragment.xml +++ b/app/src/main/res/layout/compose_fragment.xml @@ -1,27 +1,24 @@ - + android:id="@+id/compose_fragment"> - \ No newline at end of file + diff --git a/app/src/main/res/layout/downloads_fragment.xml b/app/src/main/res/layout/downloads_fragment.xml deleted file mode 100644 index 2baf5dc7..00000000 --- a/app/src/main/res/layout/downloads_fragment.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - - - - - - - - - diff --git a/app/src/main/res/layout/feed_item_list_fragment.xml b/app/src/main/res/layout/feed_item_list_fragment.xml deleted file mode 100644 index 4b5533f4..00000000 --- a/app/src/main/res/layout/feed_item_list_fragment.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - - - - - - - - - diff --git a/app/src/main/res/layout/feedinfo.xml b/app/src/main/res/layout/feedinfo.xml deleted file mode 100644 index a2a01973..00000000 --- a/app/src/main/res/layout/feedinfo.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - - - - diff --git a/app/src/main/res/layout/fragment_search_results.xml b/app/src/main/res/layout/fragment_search_results.xml deleted file mode 100644 index 7f4aaa2c..00000000 --- a/app/src/main/res/layout/fragment_search_results.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - - - - diff --git a/app/src/main/res/layout/logs_fragment.xml b/app/src/main/res/layout/logs_fragment.xml deleted file mode 100644 index 8d2f2359..00000000 --- a/app/src/main/res/layout/logs_fragment.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - - diff --git a/app/src/main/res/layout/nav_list.xml b/app/src/main/res/layout/nav_list.xml deleted file mode 100644 index efd3b5b7..00000000 --- a/app/src/main/res/layout/nav_list.xml +++ /dev/null @@ -1,16 +0,0 @@ - - - - - - diff --git a/app/src/main/res/layout/online_feedview_fragment.xml b/app/src/main/res/layout/online_feedview_fragment.xml deleted file mode 100644 index dc052e07..00000000 --- a/app/src/main/res/layout/online_feedview_fragment.xml +++ /dev/null @@ -1,33 +0,0 @@ - - - - - - - - - - - - diff --git a/app/src/main/res/layout/queue_fragment.xml b/app/src/main/res/layout/queue_fragment.xml deleted file mode 100644 index 09f13f67..00000000 --- a/app/src/main/res/layout/queue_fragment.xml +++ /dev/null @@ -1,36 +0,0 @@ - - - - - - - - - - - - - - - - - diff --git a/app/src/main/res/layout/search_fragment.xml b/app/src/main/res/layout/search_fragment.xml index 48e719d1..08afe406 100644 --- a/app/src/main/res/layout/search_fragment.xml +++ b/app/src/main/res/layout/search_fragment.xml @@ -5,26 +5,18 @@ android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" + android:fitsSystemWindows="true" android:id="@+id/search_fragment"> - - - - - + android:minHeight="?android:attr/actionBarSize" + android:theme="?attr/actionBarTheme" + app:title="@string/search_label" + app:navigationContentDescription="@string/toolbar_back_button_content_description" + app:navigationIcon="?homeAsUpIndicator" /> - - - - - - - - - - - - - - - - - - diff --git a/app/src/main/res/menu/feed_episodes.xml b/app/src/main/res/menu/feed_episodes.xml index d916899b..caf86e97 100644 --- a/app/src/main/res/menu/feed_episodes.xml +++ b/app/src/main/res/menu/feed_episodes.xml @@ -8,12 +8,12 @@ android:icon="@drawable/playlist_play" custom:showAsAction="always"/> - - + + + + + + diff --git a/app/src/main/res/menu/feeditemlist_context.xml b/app/src/main/res/menu/feeditemlist_context.xml deleted file mode 100644 index 6a2e27d6..00000000 --- a/app/src/main/res/menu/feeditemlist_context.xml +++ /dev/null @@ -1,59 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/app/src/main/res/menu/multi_select_context_popup.xml b/app/src/main/res/menu/multi_select_context_popup.xml deleted file mode 100644 index 730b0101..00000000 --- a/app/src/main/res/menu/multi_select_context_popup.xml +++ /dev/null @@ -1,11 +0,0 @@ - - - - - - - \ No newline at end of file diff --git a/app/src/main/res/menu/multi_select_options.xml b/app/src/main/res/menu/multi_select_options.xml deleted file mode 100644 index 33755cbb..00000000 --- a/app/src/main/res/menu/multi_select_options.xml +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - diff --git a/app/src/main/res/menu/queue_context.xml b/app/src/main/res/menu/queue_context.xml deleted file mode 100644 index 522e712e..00000000 --- a/app/src/main/res/menu/queue_context.xml +++ /dev/null @@ -1,16 +0,0 @@ - - - - - - - - - - \ No newline at end of file diff --git a/app/src/main/res/values-ar/strings.xml b/app/src/main/res/values-ar/strings.xml index 2c7edc35..cbe75c51 100644 --- a/app/src/main/res/values-ar/strings.xml +++ b/app/src/main/res/values-ar/strings.xml @@ -228,31 +228,14 @@ تم مسح %d حلقات منزلة. - تم الإزالة من صندوق الوارد - علمها كمشغلة - فعل حالة التشغيل - تم الاستماع - لم يتم الاستماع بعد + + + + + علمها كمقروءة للإنتقال للتوقيتات, يجب أن تشغل الحلقة - - %d حلقة علمت كـ مقروءة. - %d حلقة علمت كـ مقروءة. - %d حلقتان علمتا كـ مقروءة. - %d حلقات علمت كـ مقروءة. - %d حلقات علمت كـ مقروءة. - %d حلقات علمت كمقروءة. - - علمها كغير مشغلة - علمها كغير مقروءة - - %d حلقة علمت بأنه لم يتم تشغيلها. - %d حلقة علمت بأنه لم يتم تشغيلها. - %d حلقتان علمتا بأنه لم يتم تشغيلها. - %d حلقات علمت بأنه لم يتم تشغيلها. - %d حلقة علمت بأنه لم يتم تشغيلها. - %d حلقات علمت بأنه لم يتم تشغيلها. - + اضف للائحة الاستماع %d حلقة أضيفت إلى لائحة الاستماع. diff --git a/app/src/main/res/values-ast/strings.xml b/app/src/main/res/values-ast/strings.xml index 82576a21..737500e8 100644 --- a/app/src/main/res/values-ast/strings.xml +++ b/app/src/main/res/values-ast/strings.xml @@ -104,9 +104,9 @@ Sentir Desaniciar Nun ye posible desaniciar el ficheru. Reaniciar el preséu podría ayudar. - Marcar como «Reprodúxose» - Marcar como «Ensin reproducir» - Marcar como «Ensin lleer» + + + Amestóse %d episodiu a la cola. Amestáronse %d episodios a la cola. diff --git a/app/src/main/res/values-br/strings.xml b/app/src/main/res/values-br/strings.xml index b1531dcf..2781ee45 100644 --- a/app/src/main/res/values-br/strings.xml +++ b/app/src/main/res/values-br/strings.xml @@ -193,29 +193,15 @@ Dilamet ez eus bet %d rann pellgarget. - Tennet eo bet eus ar voest degemer - Merkañ evel lennet - Lakaat da gemm stad al lenn - Lakaet war-well evel lennet - Lakaet war-well evel lakaet da baouez + + + + + Merkañ evel lennet Evit kemmañ al lec\'hiadur e rankit lenn ar rann - - %d rann merket evel lennet - %d rann merket evel lennet - %d rann merket evel lennet - %d rann merket evel lennet - %d rann merket evel lennet - - Merkañ evel anlennet - Merkañ evel anlennet - - %d rann merket evel anlennet. - %d rann merket evel anlennet. - %d rann merket evel anlennet. - %d rann merket evel anlennet. - %d rann merket evel anlennet. - + + Ouzhpennañ el lost %d rann ouzhpennet el lost diff --git a/app/src/main/res/values-ca/strings.xml b/app/src/main/res/values-ca/strings.xml index c35bc16e..7d5f5e6e 100644 --- a/app/src/main/res/values-ca/strings.xml +++ b/app/src/main/res/values-ca/strings.xml @@ -188,23 +188,14 @@ %d episodis baixats, suprimits. - Eliminat de la Safata d\'entrada - Marca com a reproduït - Commuta l\'estat de reproducció - Marca com a reproduït - Marca com a no reproduït + + + + + Marcar com a llegit Per a botar a posicions deus reproduir l\'episodi. - - %d episodi marcat com a reproduït. - %d episodis marcats com a reproduïts. - - Marca com a pendent - Marcar com a no llegit - - %d episodi marcat com a no llegit. - %d episodis marcats com a no llegits. - + Afegeix a la cua %d episodi afegit a la cua. diff --git a/app/src/main/res/values-cs/strings.xml b/app/src/main/res/values-cs/strings.xml index 085ecbc2..1d457a2e 100644 --- a/app/src/main/res/values-cs/strings.xml +++ b/app/src/main/res/values-cs/strings.xml @@ -217,27 +217,15 @@ %d stažených epizod smazáno. - Odebráno z nových - Označit jako poslechnuté - Přepnutí stavu přehrávání - Označeno jako poslechnuté - Označeno jako neposlechnuté + + + + + Označit jako poslechnuté Pro přeskočení na pozice musíte epizodu přehrát - - %d epizoda označena jako přehraná - %d epizody označeny jako přehrané - %d epizod označeno jako přehrané - %d epizod označeno jako přehrané - - Označit jako neposlechnuté - Označit jako nepřečtené - - %d epizoda označena jako neposlechnutá - %d epizody označeny jako neposlechnuté - %d epizod označeno jako neposlechnuté - %d epizod označeno jako neposlechnuté - + + Přidat do fronty %d epizoda přidána do fronty diff --git a/app/src/main/res/values-da/strings.xml b/app/src/main/res/values-da/strings.xml index 91a39a2a..ba2e4acf 100644 --- a/app/src/main/res/values-da/strings.xml +++ b/app/src/main/res/values-da/strings.xml @@ -199,23 +199,14 @@ %d overførte afsnit slettet. - Fjernet fra indbakken - Markér som afspillet - Skift afspilningstilstand - Markér som afspillet - Markér som uafspillet + + + + + Marker som læst For at springe til positioner, er du nødt til at afspille afsnittet - - %d afsnit markeret som afspillet. - %d afsnit markeret som afspillet. - - Markér som uafspillet - Marker som ulæst - - %d afsnit markeret som uafspillet. - %d afsnit markeret som uafspillede. - + Føj til kø %d afsnit føjet til køen. diff --git a/app/src/main/res/values-de/strings.xml b/app/src/main/res/values-de/strings.xml index d66dbabe..562cbedd 100644 --- a/app/src/main/res/values-de/strings.xml +++ b/app/src/main/res/values-de/strings.xml @@ -203,23 +203,14 @@ %d heruntergeladene Episoden gelöscht. - Aus dem Posteingang entfernt - Als gespielt markieren - Abgespielt-Zustand umschalten - Als gespielt markieren - Als ungespielt markieren + + + + + Als gelesen markieren Um auf eine Position zu springen, musst du die Episode abspielen - - %d Episode als gespielt markiert. - %d Episoden als gespielt markiert. - - Als ungespielt markieren - Als ungelesen markieren - - %d Episode als ungespielt markiert. - %d Episoden als ungespielt markiert. - + Zur Warteschlange hinzufügen %d Episode zur Warteschlange hinzugefügt. diff --git a/app/src/main/res/values-es/strings.xml b/app/src/main/res/values-es/strings.xml index bb6d0964..dadba586 100644 --- a/app/src/main/res/values-es/strings.xml +++ b/app/src/main/res/values-es/strings.xml @@ -211,25 +211,14 @@ Eliminados %d episodios descargados. - Borrado de la bandeja de entrada - Marcar como reproducido - Cambiar estado de reproducción - Marcar como reproducido - Marcar como no reproducido + + + + + Marcar como leído Para saltar a posiciones, necesitas reproducir el episodio - - %depisodio marcado como reproducido. - %depisodios marcados como reproducidos. - %depisodios marcados como reproducidos. - - Marcar como no reproducido - Marcar como no leído - - %d episodio marcado como no reproducido. - %d episodios marcados como no reproducidos. - %d episodios marcados como no reproducidos. - + Añadir a la cola %d episodio añadido a la cola. diff --git a/app/src/main/res/values-et/strings.xml b/app/src/main/res/values-et/strings.xml index 76bd171e..c9f09d0e 100644 --- a/app/src/main/res/values-et/strings.xml +++ b/app/src/main/res/values-et/strings.xml @@ -160,21 +160,11 @@ 1 allalaaditud saade kustutatud. %d allalaaditud saadet kustutatud. - Märgi kuulatuks - Märgitud kuulatuks - Märgitud kui kuulamata + + + Märgi loetuks Asukohale hüppamiseks pead saadet esitama - - %d saade märgiti kuulatuks. - %d saadet märgiti kuulatuks. - - Märgitud kui kuulamata - Märgi mitteloetuks - - %d saade märgiti kui kuulamata. - %d saadet märgiti kui kuulamata. - Lisa järjekorda %d saade lisati järjekorda. diff --git a/app/src/main/res/values-eu/strings.xml b/app/src/main/res/values-eu/strings.xml index 36053bd2..2a9c4f45 100644 --- a/app/src/main/res/values-eu/strings.xml +++ b/app/src/main/res/values-eu/strings.xml @@ -187,23 +187,14 @@ %d deskargatutako kapituluak ezabatu egin dira - Sarrerako ontzitik kenduta - Markatu entzundakotzat - Erreprodukzio-egoera txandakatu - Entzundakotzat markatua - Ez entzundakotzat markatua + + + + + Markatu iraurrita gisa Posizioetara jauzi egiteko, pasartea erreproduzitu behar duzu - - %d saio markatuta ikusita bezala. - %d saio markatuta ikusita bezala. - - Markatu ez entzundakotzat bezala - Markatu ez irakurrita bezala - - %d saio markatuta ikusita bezala. - %d saio markatuta ez ikusita bezala. - + Gehitu ilaran %d saio ilaran gehitua. diff --git a/app/src/main/res/values-fa/strings.xml b/app/src/main/res/values-fa/strings.xml index 95fac2f7..3c0413c5 100644 --- a/app/src/main/res/values-fa/strings.xml +++ b/app/src/main/res/values-fa/strings.xml @@ -194,23 +194,14 @@ %dقسمت بار گرفته حذف شد. - حذف شده از صندوق ورودی - علامت‌گذاری به‌عنوان پخش‌شده - وضعیت پخش را تغییر دهید - علامت‌گذاری شد به‌عنوان پخش‌شده - علامت‌گذاری شد به‌عنوان پخش‌نشده + + + + + علامت زدن به عنوان خوانده شده برای پرش به موقعیت‌ها، می‌توانید قسمت را پخش کنید - - %d قسمت به‌عنوان پخش‌شده علامت‌گذاری شد. - %d قسمت به‌عنوان پخش‌شده علامت‌گذاری شد. - - علامت‌گذاری به‌عنوان پخش‌نشده - علامت‌گذاری به‌عنوان نخوانده‌شده - - %d قسمت به‌عنوان پخش‌نشده علامت‌گذاری شد. - %d قسمت به‌عنوان پخش‌نشده علامت‌گذاری شد. - + افزودن به صف %d قسمت به صف اضافه شد. diff --git a/app/src/main/res/values-fi/strings.xml b/app/src/main/res/values-fi/strings.xml index adadbfd0..83c8bca4 100644 --- a/app/src/main/res/values-fi/strings.xml +++ b/app/src/main/res/values-fi/strings.xml @@ -187,23 +187,14 @@ %d ladattua jaksoa poistettu. - Poistettu saapuneista - Merkitse toistetuksi - Vaihda toistettu-tila - Merkattu toistetuksi - Merkattu toistamattomaksi + + + + + Merkitse luetuksi Jaksoa pitää toistaa siirtyäksesi kohtiin - - %d jakso merkitty soitetuksi. - %d jaksoa merkitty toistetuksi - - Merkitse toistamattomaksi - Merkitse lukemattomaksi - - %d jakso merkitty soittamattomaksi. - %d jaksoa merkitty toistamattomaksi. - + Lisää jonoon %d jakso lisätty jonoon. diff --git a/app/src/main/res/values-fr/strings.xml b/app/src/main/res/values-fr/strings.xml index 772fb2d8..4bf5625e 100644 --- a/app/src/main/res/values-fr/strings.xml +++ b/app/src/main/res/values-fr/strings.xml @@ -212,25 +212,14 @@ %d épisodes téléchargés supprimés. - Supprimé de la boîte de réception - Marquer comme lu - Modifier l\'état de lecture - Marqué comme lu - Marqué comme non lu + + + + + Marquer comme lu Pour changer la position l\'épisode doit être en cours de lecture - - %d épisode marqué comme lu. - %d épisodes marqués comme lus. - %d épisodes marqués comme lus. - - Marquer comme non lu - Marquer comme non lu - - %d épisode marqué comme non lu. - %d épisodes marqués comme non lus. - %d épisodes marqués comme non lus. - + Ajouter à la liste de lecture %d épisode ajouté à la liste de lecture. diff --git a/app/src/main/res/values-gl/strings.xml b/app/src/main/res/values-gl/strings.xml index fc7c7625..03eeaa03 100644 --- a/app/src/main/res/values-gl/strings.xml +++ b/app/src/main/res/values-gl/strings.xml @@ -199,23 +199,13 @@ Eliminados %d episodios descargados. - Retirado da lista de novidades - Marcar como reproducido - Cambiar estado de reprodución - Marcado como reproducido - Marcado como non reproducido + + + + + Marcar como lido Para ir á posición, debes reproducir o episodio - - %d episodio marcado como reproducido. - %d episodios marcados como reproducidos. - - Marcar como non reproducido - Marcar como non lido - - %d episodio marcado como non reproducido. - %d episodios marcados como non reproducidos. - Engadir á cola %d episodio engadido a cola. diff --git a/app/src/main/res/values-hu/strings.xml b/app/src/main/res/values-hu/strings.xml index 7992398a..1ad65afa 100644 --- a/app/src/main/res/values-hu/strings.xml +++ b/app/src/main/res/values-hu/strings.xml @@ -175,23 +175,14 @@ %d letöltött epizód törölve. - Eltávolítva a beérkezők közül - Megjelölés lejátszottként - Lejátszott állapot be/ki - Megjelölve lejátszottként - Megjelölve nem lejátszottként + + + + + Megjelölés olvasottként A pozíciókra ugráshoz le kell játszania az epizódot - - %d epizód megjelölve lejátszottként. - %d epizód megjelölve lejátszottként. - - Megjelölés nem lejátszottként - Megjelölés olvasatlanként - - %d epizód megjelölve nem lejátszottként. - %d epizód megjelölve nem lejátszottként. - + %d epizód sorbaállítva. %d epizód sorbaállítva. diff --git a/app/src/main/res/values-in/strings.xml b/app/src/main/res/values-in/strings.xml index f3a9c63c..7c403883 100644 --- a/app/src/main/res/values-in/strings.xml +++ b/app/src/main/res/values-in/strings.xml @@ -155,21 +155,14 @@ %depisode terunduh dihapus. - Dihapus dari kotak masuk - Tandai diputar - Aktifkan status diputar - Ditandai sebagai \'diputar\' - Ditandai sebagai \'belum diputar\' + + + + + Tandai dibaca Untuk melompat posisi waktu, Anda perlu memutar episode terlebih dahulu - - %d episode ditandai sebagai telah diputar. - - Tandai belum diputar - Tandai belum dibaca - - %d episode ditandai sebagai belum diputar - + %d episode ditambahkan ke antrian. diff --git a/app/src/main/res/values-it/strings.xml b/app/src/main/res/values-it/strings.xml index 72dc7b1c..c90b5888 100644 --- a/app/src/main/res/values-it/strings.xml +++ b/app/src/main/res/values-it/strings.xml @@ -212,25 +212,14 @@ %d episodi scaricati eliminati. - Rimosso dall\'inbox - Segna come riprodotto - Cambia stato di riproduzione - Segnato come riprodotto - Segnato come non riprodotto + + + + + Segna come letto Per saltare alla posizione devi riprodurre l\'episodio - - %d episodio segnato come riprodotto. - %d di episodi segnati come riprodotti. - %d episodi segnati come riprodotti. - - Segna come non riprodotto - Segna come non letto - - %d episodio segnato come non riprodotto. - %d di episodi segnati come non riprodotti. - %d episodi segnati come non riprodotti. - + Aggiungi alla coda %d episodio aggiunto alla coda. diff --git a/app/src/main/res/values-iw/strings.xml b/app/src/main/res/values-iw/strings.xml index a50bc7d4..94349bb4 100644 --- a/app/src/main/res/values-iw/strings.xml +++ b/app/src/main/res/values-iw/strings.xml @@ -217,27 +217,14 @@ %d פרקים שהורדו נמחקו. - הוסר מהדואר הנכנס - סימון כנצפה - החלפת מצב נגינה - סומן כהתנגן - סומן שלא התנגן + + + + + סימון כנקרא כדי לקפוץ למיקומים, עליך לנגן את הפרק - - פרק אחד סומן שנוגן. - %d פרקים סומנו שנוגנו. - %d פרקים סומנו שנוגנו. - %d פרקים סומנו שנוגנו. - - סימון כלא נוגן - סימון כלא נקרא - - פרק אחד סומן שטרם נוגן. - %d פרקים סומנו שטרם נוגנו. - %d פרקים סומנו שטרם נוגנו. - %d פרקים סומנו שטרם נוגנו. - + הוספה לתור פרק אחד נוסף לתור. diff --git a/app/src/main/res/values-ja/strings.xml b/app/src/main/res/values-ja/strings.xml index 7f0abc0f..c10a6fd5 100644 --- a/app/src/main/res/values-ja/strings.xml +++ b/app/src/main/res/values-ja/strings.xml @@ -147,21 +147,13 @@ ダウンロードした %d 項目のエピソードが削除されました。 - 受信トレイから取り除きました - 再生済としてマーク - 再生状態の切り替え - 再生済みとマーク済み - 未再生としてマーク + + + + + 既読としてマーク 位置にジャンプするには、エピソードを再生する必要があります - - %d エピソードを再生済にしました。 - - 未再生としてマーク - 未読としてマーク - - %d エピソードを未再生にしました。 - %d エピソードをキューに追加しました。 diff --git a/app/src/main/res/values-ko/strings.xml b/app/src/main/res/values-ko/strings.xml index 70e83a69..41f0564c 100644 --- a/app/src/main/res/values-ko/strings.xml +++ b/app/src/main/res/values-ko/strings.xml @@ -178,21 +178,14 @@ 다운로드한 %d개 에피소드 삭제함. - 새로 받음 목록에서 제거함 - 재생했다고 표시 - 재생함 상태 토글 - 재생했다고 표시 - 재생하지 않음으로 표시 + + + + + 읽었다고 표시 특정 재생 위치로 이동하려면, 에피소드를 재생해야 합니다 - - %d개 에피소드를 재생한 것으로 표시했습니다. - - 재생하지 않음으로 표시 - 읽지 않음으로 표시 - - %d개 에피소드를 재생하지 않은 것으로 표시했습니다. - + 대기열에 추가 %d개 에피소드를 대기열에 추가했습니다. diff --git a/app/src/main/res/values-lt/strings.xml b/app/src/main/res/values-lt/strings.xml index f416df91..ba662b65 100644 --- a/app/src/main/res/values-lt/strings.xml +++ b/app/src/main/res/values-lt/strings.xml @@ -151,25 +151,12 @@ %d atsiųsti epizodai ištrinti. %d atsiųsti epizodai ištrinti. - Pažymėti kaip perklausytą - Pažymėtas kaip perklausytas - Pažymėtas kaip neperklausytas + + + Pažymėti kaip perskaitytą Norint peršokti į poziciją reikia pradėti epizodo atkūrimą - - %d epizodas pažymėtas kaip perklausytas. - %d epizodai pažymėti kaip perklausyti. - %d epizodai pažymėti kaip perklausyti. - %d epizodai pažymėti kaip perklausyti. - - Pažymėti kaip neperklausytą - Pažymėti kaip neperskaitytą - - %d epizodas pažymėtas kaip neperklausytas. - %d epizodai pažymėti kaip neperklausyti. - %d epizodai pažymėti kaip neperklausyti. - %d epizodai pažymėti kaip neperklausyti. - + %d epizodas pridėtas į eilę. %d epizodai pridėti į eilę. diff --git a/app/src/main/res/values-nb/strings.xml b/app/src/main/res/values-nb/strings.xml index cd4119fc..01f45fca 100644 --- a/app/src/main/res/values-nb/strings.xml +++ b/app/src/main/res/values-nb/strings.xml @@ -183,23 +183,14 @@ %d nedlasted episoder slettet. - Fjernet fra innboksen - Marker som avspilt - Bytte avspilt status - Marker som avspilt - Marker som ikke avspilt + + + + + Merk som lest For å hoppe til posisjoner, må du spille episoden - - %d episode merket som avspilt. - %d episoder merket som avspilt. - - Marker som ikke avspilt - Merk som ulest - - %d episode merket som ikke avspilt. - %d episoder merket som ikke avspilt. - + Legg til i kø %d episode lagt til i køen. diff --git a/app/src/main/res/values-nl/strings.xml b/app/src/main/res/values-nl/strings.xml index 668746c7..393c257e 100644 --- a/app/src/main/res/values-nl/strings.xml +++ b/app/src/main/res/values-nl/strings.xml @@ -193,23 +193,14 @@ %d gedownloade afleveringen verwijderd. - Verwijderd uit Postvak IN - Als afgespeeld markeren - Afspeelstatus wijzigen - Gemarkeerd als afgespeeld - Gemarkeerd als niet-afgespeeld + + + + + Markeren als gelezen Speel de aflevering af om naar posities te springen - - %d aflevering gemarkeerd als afgespeeld. - %d afleveringen gemarkeerd als afgespeeld. - - Als niet-afgespeeld markeren - Markeren als ongelezen - - %d aflevering gemarkeerd als niet-afgespeeld. - %d afleveringen gemarkeerd als niet-afgespeeld. - + Toevoegen aan wachtrij %d aflevering toegevoegd aan de wachtrij. diff --git a/app/src/main/res/values-pl/strings.xml b/app/src/main/res/values-pl/strings.xml index 2cf70c2f..32015d22 100644 --- a/app/src/main/res/values-pl/strings.xml +++ b/app/src/main/res/values-pl/strings.xml @@ -204,27 +204,14 @@ Usunięto %d pobranych odcinków. - Usunięto ze skrzynki odbiorczej - Oznacz jako odtworzone - Pokaż status odtworzenia - Oznaczone jako odtworzone - Oznaczone jako nieodtworzone + + + + + Oznacz jako przeczytane Aby skoczyć do konkretnej pozycji, musisz odtworzyć odcinek - - Oznaczono %d odcinek jako odtworzony. - Oznaczono %d odcinki(ów) jako odtworzone. - Oznaczono %d odcinki(ów) jako odtworzone. - Oznaczono %d odcinki(ów) jako odtworzone. - - Oznacz jako nieodtworzone - Oznacz jako nieprzeczytane - - Oznaczono %d odcinek jako nieodtworzony. - Oznaczono %d odcinki(ów) jako nieodtworzone. - Oznaczono %d odcinki(ów) jako nieodtworzone. - Oznaczono %d odcinki(ów) jako nieodtworzone. - + Dodaj do kolejki %d odcinek dodano do kolejki. diff --git a/app/src/main/res/values-pt-rBR/strings.xml b/app/src/main/res/values-pt-rBR/strings.xml index 665fe18e..780cabad 100644 --- a/app/src/main/res/values-pt-rBR/strings.xml +++ b/app/src/main/res/values-pt-rBR/strings.xml @@ -199,25 +199,14 @@ %d episódios baixados deletados. - Removido da caixa de entrada - Marcar como reproduzido - Alterar estado de reprodução - Marcado como reproduzido - Marcado como não reproduzido + + + + + Marcar como lido Para pular para as posições, você precisa reproduzir o episódio - - %d episódio marcado como reproduzido. - %d episódios marcados como reproduzidos. - %d episódios marcados como reproduzidos. - - Marcar como não reproduzido - Marcar como não reproduzido - - %d episódio marcado como não reproduzido. - %d episódios marcados como não reproduzidos. - %d episódios marcados como não reproduzidos. - + Adicionar à fila %d episódio adicionado à fila. diff --git a/app/src/main/res/values-pt/strings.xml b/app/src/main/res/values-pt/strings.xml index cfd024d1..26310d62 100644 --- a/app/src/main/res/values-pt/strings.xml +++ b/app/src/main/res/values-pt/strings.xml @@ -211,25 +211,14 @@ %d episódios eliminados - Removido da caixa de entrada - Marcar como reproduzido - Alternar estado de reprodução - Marcado como reproduzido - Marcado como não reproduzido + + + + + Marcar como lido Se quiser ir para uma posição, tem que reproduzir o episódio - - %d episódio marcado como reproduzido - %d episódios marcados como reproduzidos - %d episódios marcados como reproduzidos - - Marcar como não reproduzido - Marcar como não lido - - %d episódio marcado como não reproduzido - %d episódios marcados como não reproduzidos - %d episódios marcados como não reproduzidos - + Adicionar à fila %d episódio adicionado à fila diff --git a/app/src/main/res/values-ro/strings.xml b/app/src/main/res/values-ro/strings.xml index 68c278cf..a82f18ee 100644 --- a/app/src/main/res/values-ro/strings.xml +++ b/app/src/main/res/values-ro/strings.xml @@ -208,25 +208,14 @@ %d de episoade descărcare au fost șterse. - Eliminat din inbox - Marchează ca redat - Comută starea de redare - Marcat ca redat - Marcat ca neredat + + + + + Marcați ca citit Pentru a sări către o poziție, trebuie sa începeți redarea episodului - - Un episod marcat ca redat. - %d episoade marcate ca redate. - %d de episoade marcate ca redate. - - Marchează ca neredat - Marchează ca necitit. - - Un episod marcat ca neredat. - %d episoade marcate ca neredate. - %d de episoade marcate ca neredate. - + Adaugă la coadă Un episod adăugat la coadă. diff --git a/app/src/main/res/values-ru/strings.xml b/app/src/main/res/values-ru/strings.xml index 2cd53cd3..f31fc2b9 100644 --- a/app/src/main/res/values-ru/strings.xml +++ b/app/src/main/res/values-ru/strings.xml @@ -205,27 +205,14 @@ %d загруженных выпусков удалёно. - Убрано из входящих - Отметить как прослушанное - Переключить состояние \"прослушано\" - Отмечено как прослушанное - Отметить как непрослушанное + + + + + Отметить как прочитанное Для переходов в выпуске нужно слушать выпуск - - %d выпуск отмечен как прослушанный. - %d выпуска отмечены как прослушанные. - %d выпусков отмечены как прослушанные. - %d выпусков отмечено как прослушанные. - - Отметить как непрослушанное - Отметить как непрочитанное - - %d выпуск отмечен непрослушанный. - %d выпуска отмечены непрослушанные. - %d выпусков отмечены непрослушанные. - %d выпусков отмечено непрослушанные. - + Добавить в очередь %d выпуск добавлен в очередь. diff --git a/app/src/main/res/values-sk/strings.xml b/app/src/main/res/values-sk/strings.xml index d818df71..d65494b9 100644 --- a/app/src/main/res/values-sk/strings.xml +++ b/app/src/main/res/values-sk/strings.xml @@ -217,27 +217,14 @@ %dstiahnutých epizód zmazaných. - Odstránené zo schránky - Označiť ako prehrané - Prepnúť stav prehratia - Označené ako prehrané - Označené ako neprehrané + + + + + Označiť ako prečítané Preskočenie na určitú pozíciu funguje len pri prehrávaní epizódy. - - %d epizóda bola označená ako prehraná. - %d epizódy boli označené ako prehrané. - %d epizód bolo označených ako prehrané. - %d epizód bolo označených ako prehrané. - - Označiť ako neprehrané - Označiť ako prehrané - - %d epizóda bola označená ako neprehraná. - %d epizódy bolo označené ako neprehrané. - %d epizód bolo označených ako neprehrané. - %d epizód bolo označených ako neprehrané. - + Pridať do poradia %d epizóda bola pridaná do poradia diff --git a/app/src/main/res/values-sl/strings.xml b/app/src/main/res/values-sl/strings.xml index 0a1c6339..dfe6c18b 100644 --- a/app/src/main/res/values-sl/strings.xml +++ b/app/src/main/res/values-sl/strings.xml @@ -142,25 +142,12 @@ Tok Briši Ni bilo mogoče brisati datoteke. Ponovni zagon naprave bi lahko pomagal. - Označi kot predvajan - Označi kot predvajano. - Označi kot ne predvajano. + + + Označi kot prebrano Če želite skočiti na to mesto, morate predvajati epizodo - - %d označi kot predvajano. - %d označi kot predvajani. - %d označi kot predvajano. - %d označi kot predvajano. - - Označi kot ne predvajano. - Pusti ne prebrano. - - %d epizodo označi kot ne predvajano. - %depizodi označi kot ne predvajani. - %d epizode označi kot ne predvajane. - %d epizode označi kot ne predvajane. - + %d epizoda je dodana v čakalno vrsto. %d epizodi sta dodani v čakalno vrsto. diff --git a/app/src/main/res/values-sv/strings.xml b/app/src/main/res/values-sv/strings.xml index fe186e31..8d20ad5a 100644 --- a/app/src/main/res/values-sv/strings.xml +++ b/app/src/main/res/values-sv/strings.xml @@ -199,23 +199,14 @@ %d nedladdade episoder borttagna. - Tog bort från inkorg - Markera som spelad - Växla spelläge - Markera som spelad - Markera som ospelad + + + + + Markera som läst Spela episoden för att kunna hoppa till olika positioner - - 1%d episod markerad som spelad. - %d episoder markerade som spelade. - - Markera som ospelad - Markera som ospelad - - 1%d episod markerad som ospelad. - %d episoder markerade som ospelade. - + Lägg till i kön 1%d episod tillagd i kön. diff --git a/app/src/main/res/values-tr/strings.xml b/app/src/main/res/values-tr/strings.xml index 4eda2a21..131c8984 100644 --- a/app/src/main/res/values-tr/strings.xml +++ b/app/src/main/res/values-tr/strings.xml @@ -192,23 +192,14 @@ %d indirilmiş bölüm silindi. - Gelen kutusundan silindi - Oynatıldı olarak işaretle - Oynatma durumunu değiştir - Çalındı olarak işaretlendi - Çalınmadı olarak işaretlendi + + + + + Okundu olarak işaretle Başka bir konuma atlamak için bölümü oynatmanız gerekir. - - %d bölüm oynatıldı olarak işaretlendi. - %d bölüm oynatıldı olarak işaretlendi. - - Oynatılmadı olarak işaretle - Okunmadı olarak işaretle - - %d bölüm oynanmamış olarak işaretlendi. - %d bölüm oynanmamış olarak işaretlendi. - + Sıraya ekle %d bölüm kuyruğa eklendi. diff --git a/app/src/main/res/values-uk/strings.xml b/app/src/main/res/values-uk/strings.xml index 92b9b0d0..ee0edc43 100644 --- a/app/src/main/res/values-uk/strings.xml +++ b/app/src/main/res/values-uk/strings.xml @@ -217,27 +217,14 @@ %d завантажених епізодів видалено. - Вилучено з вхідних - Позначити як відтворений - Перемикнути стан прослуханості - Позначено як відтворений - Позначено як невідтворений + + + + + Позначити як прочитане Щоб перейти до позиції, потрібно відтворити епізод - - %d епізод помічено як відтворений - %d епізоди помічено як відтворені - %d епізодів помічено як відтворені - %d епізодів помічено як відтворені - - Позначити як невідтворений - Позначити як непрочитане - - %d епізод помічено як невідтворений. - %d епізоди помічено як невідтворені. - %d епізодів помічено як невідтворені. - %d епізодів помічено як невідтворені. - + Додати до черги %d епізод додано до черги. diff --git a/app/src/main/res/values-zh-rCN/strings.xml b/app/src/main/res/values-zh-rCN/strings.xml index 0bc3b43f..7c727009 100644 --- a/app/src/main/res/values-zh-rCN/strings.xml +++ b/app/src/main/res/values-zh-rCN/strings.xml @@ -194,21 +194,14 @@ 删除了 %d 个已下载的节目。 - 已从收件箱删除 - 标记已播放 - 切换播放状态 - 标记为已播放 - 标记为未播放 + + + + + 标记为已读 要跳转到某处,你需要播放这一集 - - 已将%d个节目标记为已播放 - - 标记未播放 - 标为未读 - - 已将%d个节目标记为未播放 - + 加至队列 已将%d个节目添加到序列中 diff --git a/app/src/main/res/values-zh-rTW/strings.xml b/app/src/main/res/values-zh-rTW/strings.xml index 94d6aae1..3f78856d 100644 --- a/app/src/main/res/values-zh-rTW/strings.xml +++ b/app/src/main/res/values-zh-rTW/strings.xml @@ -105,17 +105,10 @@ 串流播放 刪除 刪除文件失敗。重啟設備試試看。 - 標記為已播放 + 標示為已讀 若想指定播放位置,請先播放該單集 - - 共有 %d 集標示為已播放。 - - 標記為未播放 - 標示為未讀 - - 共有 %d 集標示為未播放。 - + 已將 %d 集加入待播清單。 diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index a9d19dcc..f53418d5 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -270,27 +270,15 @@ My opinion Cancelled on - Removed from inbox - Mark as played - Toggle played state - Marked as played - Marked as unplayed + Set played state + Mark as read To jump to positions, you need to play the episode %d episode marked as favorite. %d episodes marked as favorite. - - %d episode marked as played. - %d episodes marked as played. - - Mark as unplayed - Mark as unread - - %d episode marked as unplayed. - %d episodes marked as unplayed. - + Shelve to synthetic Add to active queue diff --git a/app/src/test/kotlin/ac/mdiq/podcini/feed/FeedItemMother.kt b/app/src/test/kotlin/ac/mdiq/podcini/feed/FeedItemMother.kt index 10ffe102..7c712ba5 100644 --- a/app/src/test/kotlin/ac/mdiq/podcini/feed/FeedItemMother.kt +++ b/app/src/test/kotlin/ac/mdiq/podcini/feed/FeedItemMother.kt @@ -8,7 +8,7 @@ internal object FeedItemMother { @JvmStatic fun anyFeedItemWithImage(): Episode { - val item = Episode(0, "Item", "Item", "url", Date(), Episode.PlayState.PLAYED.code, FeedMother.anyFeed()) + val item = Episode(0, "Item", "Item", "url", Date(), PlayState.PLAYED.code, FeedMother.anyFeed()) item.imageUrl = (IMAGE_URL) return item } diff --git a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbCleanupTests.kt b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbCleanupTests.kt index 67296a9d..b6e39254 100644 --- a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbCleanupTests.kt +++ b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbCleanupTests.kt @@ -87,7 +87,7 @@ open class DbCleanupTests { val items: MutableList = ArrayList() feed.episodes.addAll(items) val files: MutableList = ArrayList() - populateItems(numItems, feed, items, files, Episode.PlayState.PLAYED.code, false, false) + populateItems(numItems, feed, items, files, PlayState.PLAYED.code, false, false) performAutoCleanup(context) for (i in files.indices) { @@ -104,7 +104,7 @@ open class DbCleanupTests { for (i in 0 until numItems) { val itemDate = Date((numItems - i).toLong()) var playbackCompletionDate: Date? = null - if (itemState == Episode.PlayState.PLAYED.code) { + if (itemState == PlayState.PLAYED.code) { playbackCompletionDate = itemDate } val item = Episode(0, "title", "id$i", "link", itemDate, itemState, feed) @@ -140,7 +140,7 @@ open class DbCleanupTests { val items: MutableList = ArrayList() feed.episodes.addAll(items) val files: MutableList = ArrayList() - populateItems(numItems, feed, items, files, Episode.PlayState.UNPLAYED.code, false, false) + populateItems(numItems, feed, items, files, PlayState.UNPLAYED.code, false, false) performAutoCleanup(context) for (file in files) { @@ -157,7 +157,7 @@ open class DbCleanupTests { val items: MutableList = ArrayList() feed.episodes.addAll(items) val files: MutableList = ArrayList() - populateItems(numItems, feed, items, files, Episode.PlayState.PLAYED.code, true, false) + populateItems(numItems, feed, items, files, PlayState.PLAYED.code, true, false) performAutoCleanup(context) for (file in files) { @@ -198,7 +198,7 @@ open class DbCleanupTests { val items: MutableList = ArrayList() feed.episodes.addAll(items) val files: MutableList = ArrayList() - populateItems(numItems, feed, items, files, Episode.PlayState.PLAYED.code, false, true) + populateItems(numItems, feed, items, files, PlayState.PLAYED.code, false, true) performAutoCleanup(context) for (file in files) { diff --git a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbNullCleanupAlgorithmTest.kt b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbNullCleanupAlgorithmTest.kt index e90ffdc8..2d18f4d4 100644 --- a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbNullCleanupAlgorithmTest.kt +++ b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbNullCleanupAlgorithmTest.kt @@ -81,7 +81,7 @@ class DbNullCleanupAlgorithmTest { feed.episodes.addAll(items) val files: MutableList = ArrayList() for (i in 0 until numItems) { - val item = Episode(0, "title", "id$i", "link", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "title", "id$i", "link", Date(), PlayState.PLAYED.code, feed) val f = File(destFolder, "file $i") Assert.assertTrue(f.createNewFile()) diff --git a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbPlayQueueCleanupAlgorithmTest.kt b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbPlayQueueCleanupAlgorithmTest.kt index dcdb50aa..ad7eab37 100644 --- a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbPlayQueueCleanupAlgorithmTest.kt +++ b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbPlayQueueCleanupAlgorithmTest.kt @@ -33,7 +33,7 @@ class DbPlayQueueCleanupAlgorithmTest : DbCleanupTests() { val items: MutableList = ArrayList() feed.episodes.addAll(items) val files: MutableList = ArrayList() - populateItems(numItems, feed, items, files, Episode.PlayState.UNPLAYED.code, false, false) + populateItems(numItems, feed, items, files, PlayState.UNPLAYED.code, false, false) performAutoCleanup(context) for (i in files.indices) { diff --git a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbTasksTest.kt b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbTasksTest.kt index 9915895d..784e2f53 100644 --- a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbTasksTest.kt +++ b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbTasksTest.kt @@ -61,7 +61,7 @@ class DbTasksTest { feed.episodes.clear() for (i in 0 until numItems) { feed.episodes.add(Episode(0, "item $i", "id $i", "link $i", - Date(), Episode.PlayState.UNPLAYED.code, feed)) + Date(), PlayState.UNPLAYED.code, feed)) } val newFeed = updateFeed(context, feed, false) @@ -97,7 +97,7 @@ class DbTasksTest { feed.episodes.clear() for (i in 0 until numItemsOld) { feed.episodes.add(Episode(0, "item $i", "id $i", "link $i", - Date(i.toLong()), Episode.PlayState.PLAYED.code, feed)) + Date(i.toLong()), PlayState.PLAYED.code, feed)) } // val adapter = getInstance() // adapter.open() @@ -117,7 +117,7 @@ class DbTasksTest { for (i in numItemsOld until numItemsNew + numItemsOld) { feed.episodes.add(0, Episode(0, "item $i", "id $i", "link $i", - Date(i.toLong()), Episode.PlayState.UNPLAYED.code, feed)) + Date(i.toLong()), PlayState.UNPLAYED.code, feed)) } val newFeed = updateFeed(context, feed, false) @@ -134,7 +134,7 @@ class DbTasksTest { @Test fun testUpdateFeedMediaUrlResetState() { val feed = Feed("url", null, "title") - val item = Episode(0, "item", "id", "link", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "item", "id", "link", Date(), PlayState.PLAYED.code, feed) feed.episodes.add(item) // val adapter = getInstance() @@ -166,7 +166,7 @@ class DbTasksTest { feed.episodes.clear() for (i in 0..9) { feed.episodes.add( - Episode(0, "item $i", "id $i", "link $i", Date(i.toLong()), Episode.PlayState.PLAYED.code, feed)) + Episode(0, "item $i", "id $i", "link $i", Date(i.toLong()), PlayState.PLAYED.code, feed)) } // val adapter = getInstance() // adapter.open() @@ -188,7 +188,7 @@ class DbTasksTest { feed.episodes.clear() for (i in 0..9) { val item = - Episode(0, "item $i", "id $i", "link $i", Date(i.toLong()), Episode.PlayState.PLAYED.code, feed) + Episode(0, "item $i", "id $i", "link $i", Date(i.toLong()), PlayState.PLAYED.code, feed) val media = EpisodeMedia(item, "download url $i", 123, "media/mp3") item.setMedia(media) feed.episodes.add(item) @@ -244,7 +244,7 @@ class DbTasksTest { val items: MutableList = ArrayList(numFeedItems) for (i in 1..numFeedItems) { val item = Episode(0, "item $i of $title", "id$title$i", "link", - Date(), Episode.PlayState.UNPLAYED.code, feed) + Date(), PlayState.UNPLAYED.code, feed) items.add(item) } feed.episodes.addAll(items) diff --git a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbWriterTest.kt b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbWriterTest.kt index 4a5cd39f..5b59b566 100644 --- a/app/src/test/kotlin/ac/mdiq/podcini/storage/DbWriterTest.kt +++ b/app/src/test/kotlin/ac/mdiq/podcini/storage/DbWriterTest.kt @@ -97,7 +97,7 @@ class DbWriterTest { val feed = Feed("url", null, "title") val items: MutableList = ArrayList() feed.episodes.addAll(items) - val item = Episode(0, "Item", "Item", "url", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "Item", "Item", "url", Date(), PlayState.PLAYED.code, feed) items.add(item) val media = EpisodeMedia(0, item, duration, 1, 1, "mime_type", "dummy path", "download_url", true, null, 0, 0) @@ -138,7 +138,7 @@ class DbWriterTest { val feed = Feed("url", null, "title") val items: MutableList = ArrayList() feed.episodes.addAll(items) - val item = Episode(0, "Item", "Item", "url", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "Item", "Item", "url", Date(), PlayState.PLAYED.code, feed) var media: EpisodeMedia? = EpisodeMedia(0, item, 1, 1, 1, "mime_type", dest.absolutePath, "download_url", true, null, 0, 0) @@ -176,7 +176,7 @@ class DbWriterTest { val feed = Feed("url", null, "title") val items: MutableList = ArrayList() feed.episodes.addAll(items) - val item = Episode(0, "Item", "Item", "url", Date(), Episode.PlayState.UNPLAYED.code, feed) + val item = Episode(0, "Item", "Item", "url", Date(), PlayState.UNPLAYED.code, feed) var media: EpisodeMedia? = EpisodeMedia(0, item, 1, 1, 1, "mime_type", dest.absolutePath, "download_url", true, null, 0, 0) @@ -218,7 +218,7 @@ class DbWriterTest { val itemFiles: MutableList = ArrayList() // create items with downloaded media files for (i in 0..9) { - val item = Episode(0, "Item $i", "Item$i", "url", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "Item $i", "Item$i", "url", Date(), PlayState.PLAYED.code, feed) feed.episodes.add(item) val enc = File(destFolder, "file $i") @@ -308,7 +308,7 @@ class DbWriterTest { // create items for (i in 0..9) { - val item = Episode(0, "Item $i", "Item$i", "url", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "Item $i", "Item$i", "url", Date(), PlayState.PLAYED.code, feed) feed.episodes.add(item) } @@ -352,7 +352,7 @@ class DbWriterTest { // create items with downloaded media files for (i in 0..9) { - val item = Episode(0, "Item $i", "Item$i", "url", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "Item $i", "Item$i", "url", Date(), PlayState.PLAYED.code, feed) feed.episodes.add(item) val enc = File(destFolder, "file $i") val media = EpisodeMedia(0, item, 1, 1, 1, "mime_type", @@ -415,7 +415,7 @@ class DbWriterTest { // create items with downloaded media files for (i in 0..9) { - val item = Episode(0, "Item $i", "Item$i", "url", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "Item $i", "Item$i", "url", Date(), PlayState.PLAYED.code, feed) feed.episodes.add(item) val enc = File(destFolder, "file $i") val media = EpisodeMedia(0, item, 1, 1, 1, "mime_type", @@ -463,7 +463,7 @@ class DbWriterTest { // create items for (i in 0..9) { - val item = Episode(0, "Item $i", "Item$i", "url", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "Item $i", "Item$i", "url", Date(), PlayState.PLAYED.code, feed) item.setMedia(EpisodeMedia(item, "", 0, "")) feed.episodes.add(item) } @@ -494,7 +494,7 @@ class DbWriterTest { private fun playbackHistorySetup(playbackCompletionDate: Date?): EpisodeMedia { val feed = Feed("url", null, "title") feed.episodes.clear() - val item = Episode(0, "title", "id", "link", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "title", "id", "link", Date(), PlayState.PLAYED.code, feed) val media = EpisodeMedia(0, item, 10, 0, 1, "mime", null, "url", false, playbackCompletionDate, 0, 0) feed.episodes.add(item) @@ -547,7 +547,7 @@ class DbWriterTest { val feed = Feed("url", null, "title") feed.episodes.clear() for (i in 0 until numItems) { - val item = Episode(0, "title $i", "id $i", "link $i", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "title $i", "id $i", "link $i", Date(), PlayState.PLAYED.code, feed) item.setMedia(EpisodeMedia(item, "", 0, "")) feed.episodes.add(item) } @@ -576,7 +576,7 @@ class DbWriterTest { fun testAddQueueItemSingleItem() { val feed = Feed("url", null, "title") feed.episodes.clear() - val item = Episode(0, "title", "id", "link", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "title", "id", "link", Date(), PlayState.PLAYED.code, feed) item.setMedia(EpisodeMedia(item, "", 0, "")) feed.episodes.add(item) @@ -587,7 +587,7 @@ class DbWriterTest { Assert.assertTrue(item.id != 0L) runBlocking { - val job = addToQueue(true, item) + val job = addToQueue(item) withTimeout(TIMEOUT*1000) { job.join() } } @@ -605,7 +605,7 @@ class DbWriterTest { fun testAddQueueItemSingleItemAlreadyInQueue() { val feed = Feed("url", null, "title") feed.episodes.clear() - val item = Episode(0, "title", "id", "link", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "title", "id", "link", Date(), PlayState.PLAYED.code, feed) item.setMedia(EpisodeMedia(item, "", 0, "")) feed.episodes.add(item) @@ -616,7 +616,7 @@ class DbWriterTest { Assert.assertTrue(item.id != 0L) runBlocking { - val job = addToQueue(true, item) + val job = addToQueue(item) withTimeout(TIMEOUT*1000) { job.join() } } @@ -629,7 +629,7 @@ class DbWriterTest { // adapter.close() runBlocking { - val job = addToQueue(true, item) + val job = addToQueue(item) withTimeout(TIMEOUT*1000) { job.join() } } // adapter = getInstance() @@ -766,7 +766,7 @@ class DbWriterTest { feed.episodes.clear() for (i in 0 until numItems) { val item = Episode(0, "title $i", "id $i", "link $i", - Date(), Episode.PlayState.PLAYED.code, feed) + Date(), PlayState.PLAYED.code, feed) item.setMedia(EpisodeMedia(item, "", 0, "")) feed.episodes.add(item) } @@ -817,7 +817,7 @@ class DbWriterTest { val feed = Feed("url", null, "title") feed.episodes.clear() for (i in 0 until numItems) { - val item = Episode(0, "title $i", "id $i", "link $i", Date(), Episode.PlayState.NEW.code, feed) + val item = Episode(0, "title $i", "id $i", "link $i", Date(), PlayState.NEW.code, feed) item.setMedia(EpisodeMedia(item, "", 0, "")) feed.episodes.add(item) } @@ -848,7 +848,7 @@ class DbWriterTest { val feed = Feed("url", null, "title") feed.episodes.clear() for (i in 0 until numItems) { - val item = Episode(0, "title $i", "id $i", "link $i", Date(), Episode.PlayState.PLAYED.code, feed) + val item = Episode(0, "title $i", "id $i", "link $i", Date(), PlayState.PLAYED.code, feed) item.setMedia(EpisodeMedia(item, "", 0, "")) feed.episodes.add(item) } diff --git a/app/src/test/kotlin/ac/mdiq/podcini/storage/EpisodeDuplicateGuesserTest.kt b/app/src/test/kotlin/ac/mdiq/podcini/storage/EpisodeDuplicateGuesserTest.kt index 287bc5fd..673717a8 100644 --- a/app/src/test/kotlin/ac/mdiq/podcini/storage/EpisodeDuplicateGuesserTest.kt +++ b/app/src/test/kotlin/ac/mdiq/podcini/storage/EpisodeDuplicateGuesserTest.kt @@ -57,7 +57,7 @@ class EpisodeDuplicateGuesserTest { private fun item(guid: String, title: String, downloadUrl: String, date: Long, duration: Long, mime: String ): Episode { - val item = Episode(0, title, guid, "link", Date(date), Episode.PlayState.PLAYED.code, null) + val item = Episode(0, title, guid, "link", Date(date), PlayState.PLAYED.code, null) val media = EpisodeMedia(item, downloadUrl, duration, mime) item.setMedia(media) return item diff --git a/app/src/test/kotlin/ac/mdiq/podcini/storage/ExceptFavoriteCleanupAlgorithmTest.kt b/app/src/test/kotlin/ac/mdiq/podcini/storage/ExceptFavoriteCleanupAlgorithmTest.kt index d91c9837..3db6cdce 100644 --- a/app/src/test/kotlin/ac/mdiq/podcini/storage/ExceptFavoriteCleanupAlgorithmTest.kt +++ b/app/src/test/kotlin/ac/mdiq/podcini/storage/ExceptFavoriteCleanupAlgorithmTest.kt @@ -29,7 +29,7 @@ class ExceptFavoriteCleanupAlgorithmTest : DbCleanupTests() { val items: MutableList = ArrayList() feed.episodes.addAll(items) val files: MutableList = ArrayList() - populateItems(numberOfItems, feed, items, files, Episode.PlayState.UNPLAYED.code, false, false) + populateItems(numberOfItems, feed, items, files, PlayState.UNPLAYED.code, false, false) performAutoCleanup(context) for (i in files.indices) { @@ -48,7 +48,7 @@ class ExceptFavoriteCleanupAlgorithmTest : DbCleanupTests() { val items: MutableList = ArrayList() feed.episodes.addAll(items) val files: MutableList = ArrayList() - populateItems(numberOfItems, feed, items, files, Episode.PlayState.UNPLAYED.code, true, false) + populateItems(numberOfItems, feed, items, files, PlayState.UNPLAYED.code, true, false) performAutoCleanup(context) for (i in files.indices) { @@ -67,7 +67,7 @@ class ExceptFavoriteCleanupAlgorithmTest : DbCleanupTests() { val items: MutableList = ArrayList() feed.episodes.addAll(items) val files: MutableList = ArrayList() - populateItems(numberOfItems, feed, items, files, Episode.PlayState.UNPLAYED.code, false, true) + populateItems(numberOfItems, feed, items, files, PlayState.UNPLAYED.code, false, true) performAutoCleanup(context) for (i in files.indices) { diff --git a/app/src/test/kotlin/ac/mdiq/podcini/storage/ItemEnqueuePositionCalculatorTest.kt b/app/src/test/kotlin/ac/mdiq/podcini/storage/ItemEnqueuePositionCalculatorTest.kt index 0d23606b..ee0feb53 100644 --- a/app/src/test/kotlin/ac/mdiq/podcini/storage/ItemEnqueuePositionCalculatorTest.kt +++ b/app/src/test/kotlin/ac/mdiq/podcini/storage/ItemEnqueuePositionCalculatorTest.kt @@ -57,7 +57,7 @@ object ItemEnqueuePositionCalculatorTest { fun createFeedItem(id: Long): Episode { val item = Episode(id, "Item$id", "ItemId$id", "url", - Date(), Episode.PlayState.PLAYED.code, anyFeed()) + Date(), PlayState.PLAYED.code, anyFeed()) val media = EpisodeMedia(item, "http://download.url.net/$id", 1234567, "audio/mpeg") media.id = item.id item.setMedia(media) diff --git a/changelog.md b/changelog.md index fed7f72a..801b16de 100644 --- a/changelog.md +++ b/changelog.md @@ -1,3 +1,18 @@ +# 6.12.0 + +* created a new play state system: Unspecified, Building, New, Unplayed, Later, Soon, InQueue, InProgress, Skipped, Played, Ignored, + * among which Unplayed, Later, Soon, Skipped, Played, Ignored are settable by the user + * when an episode is started to play, its state is set to InProgress + * when episode is added to a queue, its state is set to InQueue, when it's removed from a queue, the state (if lower than Skipped) is set to Skipped + * in Episode filter, for now, Unplayed is all lower than Played, while Played includes Played, Ignore +* fixed issues from last release: action button not update, icon at PlayUI not update +* re-worked the header layouts of FeedInfo and FeedEpisodes +* tuned to show PlayerUI in proper conditions +* enabled open NavDrawer in LogsFragment +* in list view of Subscriptions, moved the rating icon to the title +* NavDrawer is fully in Compose +* cleaned out some unused or redundant stuff + # 6.11.7 * added author and title info in SharedLog diff --git a/fastlane/metadata/android/en-US/changelogs/3020278.txt b/fastlane/metadata/android/en-US/changelogs/3020278.txt new file mode 100644 index 00000000..06d6ab73 --- /dev/null +++ b/fastlane/metadata/android/en-US/changelogs/3020278.txt @@ -0,0 +1,14 @@ + Version 6.12.0 + +* created a new play state system: Unspecified, Building, New, Unplayed, Later, Soon, InQueue, InProgress, Skipped, Played, Ignored, + * among which Unplayed, Later, Soon, Skipped, Played, Ignored are settable by the user + * when an episode is started to play, its state is set to InProgress + * when episode is added to a queue, its state is set to InQueue, when it's removed from a queue, the state (if lower than Skipped) is set to Skipped + * in Episode filter, for now, Unplayed is all lower than Played, while Played includes Played, Ignore +* fixed issues from last release: action button not update, icon at PlayUI not update +* re-worked the header layouts of FeedInfo and FeedEpisodes +* tuned to show PlayerUI in proper conditions +* enabled open NavDrawer in LogsFragment +* in list view of Subscriptions, moved the rating icon to the title +* NavDrawer is fully in Compose +* cleaned out some unused or redundant stuff