5

recyclerView.getLayoutManager().smoothScrollToPosition(recyclerView,null,0)只在mDiffer.submitlist(list)完成“差异化”并对列表更新/更改进行动画处理后才尝试调用。

是否有onAfterSubmitList(Callback callback)我可以用来实现此目的的 AsyncListDiffer 功能?

如果没有,有没有办法找出什么时候submitList()完成它的任务,这样我就可以把我scrollToPosition(0)的放在那里了?

4

2 回答 2

5

2020 年 7 月更新:

谷歌为此添加了回调!请参阅:https://developer.android.com/reference/androidx/recyclerview/widget/AsyncListDiffer#submitList(java.util.List%3CT%3E,%20java.lang.Runnable)

以前的答案,来自糟糕的过去:

首先,我不敢相信谷歌没有为此提供回调。

我深入研究了 的源代码AsyncListDiffer,发现当所有 RecyclerView 更新完成时,可以接收回调——但是接收这个回调的方式很奇怪。

您需要创建 的子类BatchingListUpdateCallback,然后将您的ListUpdateCallback(或AdapterListUpdateCallback在大多数情况下)包装在其中。

然后,您应该覆盖dispatchLastEvent. AsyncListDiffer将调用它,但其中一个更新已被发送。在你的dispatchLastEvent,你会想要调用超级实现,所以你不会破坏任何东西。您还需要调用自定义回调,这是您的方式。

在 Kotlin 中看起来像:

class OnDiffDoneListUpdateCallback(
    listUpdateCallback: ListUpdateCallback,
    private val onDiffDoneCallback: () -> Unit
) : BatchingListUpdateCallback(listUpdateCallback) {
    override fun dispatchLastEvent() {
        super.dispatchLastEvent()
        onDiffDoneCallback()
    }
}

最后一步是OnDiffDoneListUpdateCallbackAsyncListDiffer. 为此,您需要AsyncListDiffer为自己初始化 - 如果您正在使用ListAdapter或类似的东西,您需要重构以便您可以控制AsyncListDiffer.

在 Kotlin 中,这看起来像:

private val asyncListDiffer = AsyncListDiffer<ItemType>(
        OnDiffDoneListUpdateCallback(AdapterListUpdateCallback(adapter)) {
            // here's your callback
            doTheStuffAfterDiffIsDone()
        },
        AsyncDifferConfig.Builder<ItemType>(diffCallback).build()
    )

编辑:当然,我忘记了边缘情况!

有时,dispatchLastEvent不调用,因为AsyncListDiffer认为更新微不足道。这是它所做的检查:

    if (newList == mList) {
        ...
        return;
    }

    // fast simple remove all
    if (newList == null) {
        ...
        mUpdateCallback.onRemoved(0, countRemoved);
        return;
    }

    // fast simple first insert
    if (mList == null) {
        ...
        mUpdateCallback.onInserted(0, newList.size());
        return;
    }

我建议在你打电话之前自己做这些检查asyncListDiffer.submitList(list)。但当然,这不可能那么容易!mList是私有的,getCurrentList如果为空,将返回一个空列表mList,所以对这个检查没有用。相反,您必须使用反射来访问它:

    val listField = AsyncListDiffer::class.java.getDeclaredField("mList")
    listField.isAccessible = true
    val asyncDifferList = listField.get(asyncListDiffer) as List<ItemType>?

    asyncListDiffer.submitList(newList)

    if(asyncDifferList == null) {
        onDiffDone()
    }
    if(newList == null) {
        onDiffDone()
    }
    if(newList == asyncDifferList) {
        onDiffDone()
    }

编辑 2:现在,我知道你在说什么 - 肯定有一种更简单、更简单的方法来做到这一点?答案是……是的!只需将整个AsyncListDiffer类复制到您的项目中,然后自己添加回调!

于 2020-01-21T20:24:14.873 回答
0

registerAdapterDataObserver您可以通过聆听方法(或您的需要)从方法中获得优势,即:

listAdapter.registerAdapterDataObserver(object : RecyclerView.AdapterDataObserver() {
            override fun onChanged() {
                //
            }

            override fun onItemRangeChanged(positionStart: Int, itemCount: Int) {

            }

            override fun onItemRangeChanged(positionStart: Int, itemCount: Int, payload: Any?) {

            }

            override fun onItemRangeInserted(positionStart: Int, itemCount: Int) {

            }

            override fun onItemRangeMoved(fromPosition: Int, toPosition: Int, itemCount: Int) {

            }

            override fun onItemRangeRemoved(positionStart: Int, itemCount: Int) {

            }
        })

您可以使用 清除您adapterObserver曾经需要的寄存器unregisterAdapterDataObserver

于 2019-03-25T05:08:19.523 回答