mirror of
https://github.com/TeamNewPipe/NewPipe
synced 2025-01-10 17:30:31 +00:00
Implemented UI highlighting and "new feed items"-notification
Fixed format
This commit is contained in:
parent
676bc02d52
commit
02789122a0
@ -300,18 +300,18 @@ private fun View.animateLightSlideAndAlpha(enterOrExit: Boolean, duration: Long,
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun View.slideUp(
|
fun View.slideUp(
|
||||||
duration: Long,
|
duration: Long,
|
||||||
delay: Long,
|
delay: Long,
|
||||||
@FloatRange(from = 0.0, to = 1.0) translationPercent: Float
|
@FloatRange(from = 0.0, to = 1.0) translationPercent: Float
|
||||||
) {
|
) {
|
||||||
slideUp(duration, delay, translationPercent)
|
slideUp(duration, delay, translationPercent)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun View.slideUp(
|
fun View.slideUp(
|
||||||
duration: Long,
|
duration: Long,
|
||||||
delay: Long = 0L,
|
delay: Long = 0L,
|
||||||
@FloatRange(from = 0.0, to = 1.0) translationPercent: Float = 1.0F,
|
@FloatRange(from = 0.0, to = 1.0) translationPercent: Float = 1.0F,
|
||||||
execOnEnd: Runnable? = null
|
execOnEnd: Runnable? = null
|
||||||
) {
|
) {
|
||||||
val newTranslationY = (resources.displayMetrics.heightPixels * translationPercent).toInt()
|
val newTranslationY = (resources.displayMetrics.heightPixels * translationPercent).toInt()
|
||||||
animate().setListener(null).cancel()
|
animate().setListener(null).cancel()
|
||||||
|
@ -40,8 +40,10 @@ import androidx.core.view.isVisible
|
|||||||
import androidx.lifecycle.ViewModelProvider
|
import androidx.lifecycle.ViewModelProvider
|
||||||
import androidx.preference.PreferenceManager
|
import androidx.preference.PreferenceManager
|
||||||
import androidx.recyclerview.widget.GridLayoutManager
|
import androidx.recyclerview.widget.GridLayoutManager
|
||||||
import com.xwray.groupie.GroupieAdapter
|
import androidx.recyclerview.widget.RecyclerView
|
||||||
|
import com.xwray.groupie.GroupAdapter
|
||||||
import com.xwray.groupie.Item
|
import com.xwray.groupie.Item
|
||||||
|
import com.xwray.groupie.OnAsyncUpdateListener
|
||||||
import com.xwray.groupie.OnItemClickListener
|
import com.xwray.groupie.OnItemClickListener
|
||||||
import com.xwray.groupie.OnItemLongClickListener
|
import com.xwray.groupie.OnItemLongClickListener
|
||||||
import icepick.State
|
import icepick.State
|
||||||
@ -65,6 +67,7 @@ import org.schabi.newpipe.fragments.BaseStateFragment
|
|||||||
import org.schabi.newpipe.info_list.InfoItemDialog
|
import org.schabi.newpipe.info_list.InfoItemDialog
|
||||||
import org.schabi.newpipe.ktx.animate
|
import org.schabi.newpipe.ktx.animate
|
||||||
import org.schabi.newpipe.ktx.animateHideRecyclerViewAllowingScrolling
|
import org.schabi.newpipe.ktx.animateHideRecyclerViewAllowingScrolling
|
||||||
|
import org.schabi.newpipe.ktx.slideUp
|
||||||
import org.schabi.newpipe.local.feed.item.StreamItem
|
import org.schabi.newpipe.local.feed.item.StreamItem
|
||||||
import org.schabi.newpipe.local.feed.service.FeedLoadService
|
import org.schabi.newpipe.local.feed.service.FeedLoadService
|
||||||
import org.schabi.newpipe.local.subscription.SubscriptionManager
|
import org.schabi.newpipe.local.subscription.SubscriptionManager
|
||||||
@ -76,6 +79,7 @@ import org.schabi.newpipe.util.ThemeHelper.getGridSpanCountStreams
|
|||||||
import org.schabi.newpipe.util.ThemeHelper.shouldUseGridLayout
|
import org.schabi.newpipe.util.ThemeHelper.shouldUseGridLayout
|
||||||
import java.time.OffsetDateTime
|
import java.time.OffsetDateTime
|
||||||
import java.util.ArrayList
|
import java.util.ArrayList
|
||||||
|
import java.util.function.Consumer
|
||||||
|
|
||||||
class FeedFragment : BaseStateFragment<FeedState>() {
|
class FeedFragment : BaseStateFragment<FeedState>() {
|
||||||
private var _feedBinding: FragmentFeedBinding? = null
|
private var _feedBinding: FragmentFeedBinding? = null
|
||||||
@ -97,6 +101,8 @@ class FeedFragment : BaseStateFragment<FeedState>() {
|
|||||||
private var updateListViewModeOnResume = false
|
private var updateListViewModeOnResume = false
|
||||||
private var isRefreshing = false
|
private var isRefreshing = false
|
||||||
|
|
||||||
|
private var lastNewItemsCount = 0
|
||||||
|
|
||||||
init {
|
init {
|
||||||
setHasOptionsMenu(true)
|
setHasOptionsMenu(true)
|
||||||
}
|
}
|
||||||
@ -136,6 +142,20 @@ class FeedFragment : BaseStateFragment<FeedState>() {
|
|||||||
setOnItemLongClickListener(listenerStreamItem)
|
setOnItemLongClickListener(listenerStreamItem)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
feedBinding.itemsList.addOnScrollListener(object : RecyclerView.OnScrollListener() {
|
||||||
|
override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {
|
||||||
|
// Check if we scrolled to the top
|
||||||
|
if (newState == RecyclerView.SCROLL_STATE_IDLE &&
|
||||||
|
!recyclerView.canScrollVertically(-1)
|
||||||
|
) {
|
||||||
|
|
||||||
|
if (feedBinding.newItemsLoadedLayout.isVisible) {
|
||||||
|
hideNewItemsLoaded(true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
feedBinding.itemsList.adapter = groupAdapter
|
feedBinding.itemsList.adapter = groupAdapter
|
||||||
setupListViewMode()
|
setupListViewMode()
|
||||||
}
|
}
|
||||||
@ -171,6 +191,10 @@ class FeedFragment : BaseStateFragment<FeedState>() {
|
|||||||
super.initListeners()
|
super.initListeners()
|
||||||
feedBinding.refreshRootView.setOnClickListener { reloadContent() }
|
feedBinding.refreshRootView.setOnClickListener { reloadContent() }
|
||||||
feedBinding.swipeRefreshLayout.setOnRefreshListener { reloadContent() }
|
feedBinding.swipeRefreshLayout.setOnRefreshListener { reloadContent() }
|
||||||
|
feedBinding.newItemsLoadedButton.setOnClickListener {
|
||||||
|
hideNewItemsLoaded(true)
|
||||||
|
feedBinding.itemsList.scrollToPosition(0)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// /////////////////////////////////////////////////////////////////////////
|
// /////////////////////////////////////////////////////////////////////////
|
||||||
@ -400,7 +424,17 @@ class FeedFragment : BaseStateFragment<FeedState>() {
|
|||||||
}
|
}
|
||||||
loadedState.items.forEach { it.itemVersion = itemVersion }
|
loadedState.items.forEach { it.itemVersion = itemVersion }
|
||||||
|
|
||||||
groupAdapter.updateAsync(loadedState.items, false, null)
|
// This need to be saved in a variable as the update occurs async
|
||||||
|
val oldOldestSubscriptionUpdate = oldestSubscriptionUpdate
|
||||||
|
|
||||||
|
groupAdapter.updateAsync(
|
||||||
|
loadedState.items, false,
|
||||||
|
OnAsyncUpdateListener {
|
||||||
|
oldOldestSubscriptionUpdate?.run {
|
||||||
|
highlightNewItemsAfter(oldOldestSubscriptionUpdate)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
listState?.run {
|
listState?.run {
|
||||||
feedBinding.itemsList.layoutManager?.onRestoreInstanceState(listState)
|
feedBinding.itemsList.layoutManager?.onRestoreInstanceState(listState)
|
||||||
@ -522,6 +556,94 @@ class FeedFragment : BaseStateFragment<FeedState>() {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Highlights all items that are after the specified time
|
||||||
|
*/
|
||||||
|
private fun highlightNewItemsAfter(updateTime: OffsetDateTime) {
|
||||||
|
var highlightCount = 0
|
||||||
|
|
||||||
|
var doCheck = true
|
||||||
|
|
||||||
|
for (i in 0 until groupAdapter.itemCount) {
|
||||||
|
val item = groupAdapter.getItem(i) as StreamItem
|
||||||
|
|
||||||
|
var resid = R.attr.selectableItemBackground
|
||||||
|
if (doCheck) {
|
||||||
|
if (item.streamWithState.stream.uploadDate?.isAfter(updateTime) != false) {
|
||||||
|
resid = R.attr.dashed_border
|
||||||
|
highlightCount++
|
||||||
|
} else {
|
||||||
|
// Increases execution time due to the order of the items (newest always on top)
|
||||||
|
// Once a item is is before the updateTime we can skip all following items
|
||||||
|
doCheck = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The highlighter has to be always set
|
||||||
|
// When it's only set on items that are highlighted it will highlight all items
|
||||||
|
// due to the fact that itemRoot is getting recycled
|
||||||
|
item.execBindEnd = Consumer { viewBinding ->
|
||||||
|
val context = viewBinding.itemRoot.context
|
||||||
|
viewBinding.itemRoot.background =
|
||||||
|
androidx.core.content.ContextCompat.getDrawable(
|
||||||
|
context,
|
||||||
|
android.util.TypedValue().apply {
|
||||||
|
context.theme.resolveAttribute(
|
||||||
|
resid,
|
||||||
|
this,
|
||||||
|
true
|
||||||
|
)
|
||||||
|
}.resourceId
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Force updates all items so that the highlighting is correct
|
||||||
|
// If this isn't done visible items that are already highlighted will stay in a highlighted
|
||||||
|
// state until the user scrolls them out of the visible area which causes a update/bind-call
|
||||||
|
groupAdapter.notifyItemRangeChanged(
|
||||||
|
0,
|
||||||
|
groupAdapter.itemCount.coerceAtMost(highlightCount.coerceAtLeast(lastNewItemsCount))
|
||||||
|
)
|
||||||
|
|
||||||
|
if (highlightCount > 0) {
|
||||||
|
showNewItemsLoaded()
|
||||||
|
}
|
||||||
|
|
||||||
|
lastNewItemsCount = highlightCount
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun showNewItemsLoaded() {
|
||||||
|
feedBinding.newItemsLoadedLayout.clearAnimation()
|
||||||
|
feedBinding.newItemsLoadedLayout
|
||||||
|
.slideUp(
|
||||||
|
250L,
|
||||||
|
delay = 100,
|
||||||
|
execOnEnd = {
|
||||||
|
// Hide the new items-"popup" after 10s
|
||||||
|
hideNewItemsLoaded(true, 10000)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun hideNewItemsLoaded(animate: Boolean, delay: Long = 0) {
|
||||||
|
feedBinding.newItemsLoadedLayout.clearAnimation()
|
||||||
|
if (animate) {
|
||||||
|
feedBinding.newItemsLoadedLayout.animate(
|
||||||
|
false,
|
||||||
|
200,
|
||||||
|
delay = delay,
|
||||||
|
execOnEnd = {
|
||||||
|
// Make the layout invisible so that the onScroll toTop method
|
||||||
|
// only does necessary work
|
||||||
|
feedBinding.newItemsLoadedLayout.isVisible = false
|
||||||
|
}
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
feedBinding.newItemsLoadedLayout.isVisible = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// /////////////////////////////////////////////////////////////////////////
|
// /////////////////////////////////////////////////////////////////////////
|
||||||
// Load Service Handling
|
// Load Service Handling
|
||||||
// /////////////////////////////////////////////////////////////////////////
|
// /////////////////////////////////////////////////////////////////////////
|
||||||
@ -529,6 +651,8 @@ class FeedFragment : BaseStateFragment<FeedState>() {
|
|||||||
override fun doInitialLoadLogic() {}
|
override fun doInitialLoadLogic() {}
|
||||||
|
|
||||||
override fun reloadContent() {
|
override fun reloadContent() {
|
||||||
|
hideNewItemsLoaded(false)
|
||||||
|
|
||||||
getActivity()?.startService(
|
getActivity()?.startService(
|
||||||
Intent(requireContext(), FeedLoadService::class.java).apply {
|
Intent(requireContext(), FeedLoadService::class.java).apply {
|
||||||
putExtra(FeedLoadService.EXTRA_GROUP_ID, groupId)
|
putExtra(FeedLoadService.EXTRA_GROUP_ID, groupId)
|
||||||
|
@ -43,7 +43,7 @@ class FeedViewModel(
|
|||||||
private var combineDisposable = Flowable
|
private var combineDisposable = Flowable
|
||||||
.combineLatest(
|
.combineLatest(
|
||||||
FeedEventManager.events(),
|
FeedEventManager.events(),
|
||||||
toggleShowPlayedItemsFlowable,
|
toggleShowPlayedItemsFlowable,
|
||||||
feedDatabaseManager.notLoadedCount(groupId),
|
feedDatabaseManager.notLoadedCount(groupId),
|
||||||
feedDatabaseManager.oldestSubscriptionUpdate(groupId),
|
feedDatabaseManager.oldestSubscriptionUpdate(groupId),
|
||||||
|
|
||||||
@ -58,8 +58,8 @@ class FeedViewModel(
|
|||||||
.map { (event, showPlayedItems, notLoadedCount, oldestUpdate) ->
|
.map { (event, showPlayedItems, notLoadedCount, oldestUpdate) ->
|
||||||
var streamItems = if (event is SuccessResultEvent || event is IdleEvent)
|
var streamItems = if (event is SuccessResultEvent || event is IdleEvent)
|
||||||
feedDatabaseManager
|
feedDatabaseManager
|
||||||
.getStreams(groupId, showPlayedItems)
|
.getStreams(groupId, showPlayedItems)
|
||||||
.blockingGet(arrayListOf())
|
.blockingGet(arrayListOf())
|
||||||
else
|
else
|
||||||
arrayListOf()
|
arrayListOf()
|
||||||
|
|
||||||
@ -87,16 +87,18 @@ class FeedViewModel(
|
|||||||
}
|
}
|
||||||
|
|
||||||
private data class CombineResultEventHolder(
|
private data class CombineResultEventHolder(
|
||||||
val t1: FeedEventManager.Event,
|
val t1: FeedEventManager.Event,
|
||||||
val t2: Boolean,
|
val t2: Boolean,
|
||||||
val t3: Long,
|
val t3: Long,
|
||||||
val t4: OffsetDateTime?)
|
val t4: OffsetDateTime?
|
||||||
|
)
|
||||||
|
|
||||||
private data class CombineResultDataHolder(
|
private data class CombineResultDataHolder(
|
||||||
val t1: FeedEventManager.Event,
|
val t1: FeedEventManager.Event,
|
||||||
val t2: List<StreamWithState>,
|
val t2: List<StreamWithState>,
|
||||||
val t3: Long,
|
val t3: Long,
|
||||||
val t4: OffsetDateTime?)
|
val t4: OffsetDateTime?
|
||||||
|
)
|
||||||
|
|
||||||
fun togglePlayedItems(showPlayedItems: Boolean) {
|
fun togglePlayedItems(showPlayedItems: Boolean) {
|
||||||
toggleShowPlayedItems.onNext(showPlayedItems)
|
toggleShowPlayedItems.onNext(showPlayedItems)
|
||||||
|
@ -19,6 +19,7 @@ import org.schabi.newpipe.util.Localization
|
|||||||
import org.schabi.newpipe.util.PicassoHelper
|
import org.schabi.newpipe.util.PicassoHelper
|
||||||
import org.schabi.newpipe.util.StreamTypeUtil
|
import org.schabi.newpipe.util.StreamTypeUtil
|
||||||
import java.util.concurrent.TimeUnit
|
import java.util.concurrent.TimeUnit
|
||||||
|
import java.util.function.Consumer
|
||||||
|
|
||||||
data class StreamItem(
|
data class StreamItem(
|
||||||
val streamWithState: StreamWithState,
|
val streamWithState: StreamWithState,
|
||||||
@ -31,6 +32,12 @@ data class StreamItem(
|
|||||||
private val stream: StreamEntity = streamWithState.stream
|
private val stream: StreamEntity = streamWithState.stream
|
||||||
private val stateProgressTime: Long? = streamWithState.stateProgressMillis
|
private val stateProgressTime: Long? = streamWithState.stateProgressMillis
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Will be executed at the end of the [StreamItem.bind] (with (ListStreamItemBinding,Int)).
|
||||||
|
* Can be used e.g. for highlighting a item.
|
||||||
|
*/
|
||||||
|
var execBindEnd: Consumer<ListStreamItemBinding>? = null
|
||||||
|
|
||||||
override fun getId(): Long = stream.uid
|
override fun getId(): Long = stream.uid
|
||||||
|
|
||||||
enum class ItemVersion { NORMAL, MINI, GRID }
|
enum class ItemVersion { NORMAL, MINI, GRID }
|
||||||
@ -97,6 +104,8 @@ data class StreamItem(
|
|||||||
viewBinding.itemAdditionalDetails.text =
|
viewBinding.itemAdditionalDetails.text =
|
||||||
getStreamInfoDetailLine(viewBinding.itemAdditionalDetails.context)
|
getStreamInfoDetailLine(viewBinding.itemAdditionalDetails.context)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
execBindEnd?.accept(viewBinding)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun isLongClickable() = when (stream.streamType) {
|
override fun isLongClickable() = when (stream.streamType) {
|
||||||
|
@ -87,6 +87,26 @@
|
|||||||
|
|
||||||
</androidx.swiperefreshlayout.widget.SwipeRefreshLayout>
|
</androidx.swiperefreshlayout.widget.SwipeRefreshLayout>
|
||||||
|
|
||||||
|
<androidx.constraintlayout.widget.ConstraintLayout
|
||||||
|
android:id="@+id/new_items_loaded_layout"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_alignParentBottom="true"
|
||||||
|
android:visibility="gone"
|
||||||
|
tools:visibility="visible">
|
||||||
|
|
||||||
|
<Button
|
||||||
|
android:id="@+id/new_items_loaded_button"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:text="@string/feed_new_items"
|
||||||
|
android:textSize="12sp"
|
||||||
|
android:theme="@style/ServiceColoredButton"
|
||||||
|
app:layout_constraintBottom_toBottomOf="parent"
|
||||||
|
app:layout_constraintLeft_toLeftOf="parent"
|
||||||
|
app:layout_constraintRight_toRightOf="parent" />
|
||||||
|
</androidx.constraintlayout.widget.ConstraintLayout>
|
||||||
|
|
||||||
<LinearLayout
|
<LinearLayout
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
|
@ -636,6 +636,7 @@
|
|||||||
<string name="feed_subscription_not_loaded_count">Not loaded: %d</string>
|
<string name="feed_subscription_not_loaded_count">Not loaded: %d</string>
|
||||||
<string name="feed_notification_loading">Loading feed…</string>
|
<string name="feed_notification_loading">Loading feed…</string>
|
||||||
<string name="feed_processing_message">Processing feed…</string>
|
<string name="feed_processing_message">Processing feed…</string>
|
||||||
|
<string name="feed_new_items">New feed items</string>
|
||||||
<string name="feed_group_dialog_select_subscriptions">Select subscriptions</string>
|
<string name="feed_group_dialog_select_subscriptions">Select subscriptions</string>
|
||||||
<string name="feed_group_dialog_empty_selection">No subscription selected</string>
|
<string name="feed_group_dialog_empty_selection">No subscription selected</string>
|
||||||
<plurals name="feed_group_dialog_selection_count">
|
<plurals name="feed_group_dialog_selection_count">
|
||||||
|
Loading…
Reference in New Issue
Block a user