首页 > 解决方案 > 将 RecyclerViews itemAnimator 设置为 null 不会删除动画

问题描述

我的外部RecyclerView崩溃要么

IllegalArgumentException: Scrapped or attached views may not be recycled. isScrap:false isAttached:true...

或者

IllegalStateException: The specified child already has a parent. You must call removeView() on the child's parent first.

就像标题所暗示的那样,我在第一个RecyclerView的列表项布局中有一个RecyclerView. 此布局用于显示消息,内部用于显示消息RecyclerView附带的附件。内部RecyclerViews可见性设置为GONE或者VISIBLE取决于消息是否有任何附件。简化的外部列表项布局如下所示

ConstraintLayout
    TextView
    TextView
    TextView
    RecyclerView

处理内部的适配器部分RecyclerView看起来像这样

private fun bindFiles(message: Message?) = with(itemView) {
      if (message != null && message.attachments.isNotEmpty())
      {
            sent_message_attachments.setAsVisible()
            sent_message_attachments.layoutManager = GridLayoutManager(this.context,Math.min(message.attachments.size,3))
            sent_message_attachments.adapter = AttachmentAdapter(message.attachments)
            sent_message_attachments.itemAnimator = null
            sent_message_attachments.setHasFixedSize(true)
      }
      else{
            sent_message_attachments.setAsGone()
            sent_message_attachments.adapter = null
            sent_message_attachments.layoutManager = null
      }
    }

该错误与我在内部适配器中获取附件的方式有关,因为一旦我禁用启动下载过程的部分,一切都很好。从设备加载图像时没有问题,但是一旦我开始下载过程,一切都会变糟。这是处理图像并在内部适配器中启动下载过程的部分。我有视频和其他文件类型的功能,它们几乎完全相同,但布局略有不同。

private fun bindImage(item: HFile?) = with(itemView) {
      if (item != null)
      {
        if (item.isOnDevice && !item.path.isNullOrEmpty())
        {
          if (item.isGif)
          {
            attachment_image.displayGif(File(item.path))
          }
          else
          {
            attachment_image.displayImage(File(item.path))
          }
        }
        else
        {
          //TODO: Add option to load images manually
          FileHandler(item.id).downloadFileAsObservable(false)
              .subscribeOn(Schedulers.io())
              .observeOn(AndroidSchedulers.mainThread())
              .subscribe(
                  { progress ->
                    //TODO: Show download process
                  },
                  { error -> 
                  error.printStackTrace()
                  //TODO: Enable manual retry 
                  },
                  { notifyItemChanged(adapterPosition)} //onComplete
              )
        }
      }
  }

我在DiscussionListAdapter加载讨论肖像(个人资料图片等)时使用与上述相同的结构,并且没有相同的问题。

这些是用于膨胀 viewHolders 和显示图像的扩展函数

fun ViewGroup.inflate(layoutRes: Int): View
{
  return LayoutInflater.from(context).inflate(layoutRes, this, false)
}

fun ImageView.displayGif(file:File){
  GlideApp.with(context).asGif().load(file).transforms(CenterCrop(), RoundedCorners(30)).into(this)
}

fun ImageView.displayImage(file:File){
  GlideApp.with(context).load(file).transforms(CenterCrop(), RoundedCorners(30)).into(this)
}

在过去的几天里,我一直在做这个,只是无法理解它。非常感谢任何方向的任何帮助。我知道我的解释可能有点到处都是,所以只要在需要时要求澄清:)

更新

我现在已经能够用 aGridLayoutRecyclerView. 可以安全地假设嵌套RecyclerViews不是这里的罪魁祸首。我什至试图放弃处理加载图像并IntentService为该过程创建一个的 Rx-piece,但同样的崩溃仍然发生。

我的意思是,我没有使用GridLayout另一个适配器来填充嵌套,而是RecyclerView只使用一个适配器来填充消息并膨胀和填充附件的视图,并将这些视图附加到嵌套的GridLayout.

当我开始下载文件然后将视图(应该显示下载的文件)滚动到屏幕外时,就会发生崩溃。该视图应该被回收,但由于某种原因,下载过程(在我的测试用例中只需要大约 100ms-400ms)导致应用程序抛出原始问题中提到的两个错误之一。值得注意的是,我正在使用Realm并且适配器将RealmResults<Message>列表作为其数据集。我的演示者在列表中查找更改,然后在需要时通知适配器(由于 的实现而更改IntentService)。

这就是我能够一次又一次地重现这种情况的方式:

  1. 打开包含带有附件的消息的讨论
  2. 开始向上滚动以获取更多消息
  3. 传递带有附件的消息并在仍在加载时将其滚动到屏幕外
  4. 碰撞

如果我停止并等待下载完成并且一切都按预期工作,则不会发生崩溃。图像/视频/文件会使用正确的缩略图进行更新,如果我将其滚动到视野之外,应用程序不会崩溃。

更新 2

我尝试将嵌套交换ViewGroup为单个ImageView只是为了查看嵌套内的问题。瞧!它仍然崩溃。现在我真的很困惑,因为DiscussionListAdapter我之前提到的内容完全相同,而且它的作用就像一个魅力......我的搜索仍在继续。我希望有人,有一天会从我的痛苦中受益。

更新 3

我开始记录函数中每个ViewHolder的父级。就像预期的那样,在应用程序崩溃并吐出之前,onBindViewHolder()我得到了nullsafter 。nullsnulls

04-26 21:54:50.718 27075-27075/com.hailer.hailer.dev D/MsgAdapter: Parent of ViewHolder: android.view.ViewOverlay$OverlayViewGroup{82a9fbc V.E...... .......D 0,0-1440,2168}

毕竟我的疯狂是有办法的!但这只会提出更多问题。为什么ViewOverlay在这里使用?作为 RecyclerView 的一部分,还是作为黑魔法师的一部分,计划剥夺我的理智?

边注

我深入RecyclerViews研究代码以检查是否能找到这个ViewOverlay谜团的原因。我发现只RecyclerView调用了适配器onCreateViewHolder()函数两次。两次都提供自己作为parent函数的参数。所以没有运气......到底是什么导致项目视图拥有ViewOverlay它的父级?父对象是一个不可变的值,因此将 设置为父对象的唯一方法ViewOverlay是构造一个新对象ViewHolder并将ViewOverlay对象作为父对象。

更新 4

有时我会惊讶于自己的愚蠢。使用ViewOverlay是因为项目正在被动画化。我什至不认为这是一个选项,因为我已经itemAnimatorRecyclerViewas设置了null,但由于某些奇怪的原因,它不起作用。这些项目仍在动画中,这导致了整个游戏。那么这可能是什么原因呢?(我是如何选择忽略移动项目的,我不知道,但是当我强迫应用程序一遍又一遍地下载同一张图片时,动画变得非常清晰,整个列表都变得混乱了。)

MyDiscussionInstanceFragment包含有问题的 RecyclerView 和嵌套的 ConstraintLayout,后者又包含EditText用户输入和发送按钮。

    val v = inflater.inflate(R.layout.fragment_discussion_instance, container, false)
    val lm = LinearLayoutManager(context)
    lm.reverseLayout = true
    v.disc_instance_messages_list.layoutManager = lm
    v.disc_instance_messages_list.itemAnimator = null
    v.disc_instance_messages_list.adapter = mPresenter.messageAdapter

这是处理RecyclerView. 我肯定会设置itemAnimatoras null,但动画不会停止!我已经尝试animateLayoutChanges在根目录ConstraintLayout上设置 xml 属性,RecyclerView但它们都不起作用。值得一提的是,我还检查了程序的不同状态是否RecyclerView有一个itemAnimator,每次我检查动画师时,它都是空的。那么我的动画是什么RecyclerView?!

标签: androidandroid-recyclerviewkotlinrx-kotlin

解决方案


我终于弄清楚是什么原因造成的。在我的DiscussionInstanceView我有一个小动画,它通过关键帧动画view进出视野。ConstraintLayout此视图仅显示聊天记录的下载进度,并且仅在首次打开讨论时使用一次。但是因为每次我的数据集更新时我都会调用隐藏该视图,所以我强制ConstraintLayout触发动画序列,从而在数据集更新期间使所有内容都动画化。我只是添加了一个简单的检查,我是否正在下载历史记录,这个问题得到了解决。


推荐阅读