- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我创建了一个简单的聊天风格的 recyclerview,比如 messenger 只是 super 简化。
这一切都很完美,但是当我添加一个lottie动画来显示另一个人正在打字时,它会导致recyclerview出现很多不稳定。
我在插入新消息时使用了 scrollToPosition 函数,因此它会自动向下滚动到底部并使消息可见。当我添加 lottie 文件时,这个错误出现了,它不会正确滚动到底部,可能只是向下滚动一半。
此外,我注意到当我向下移动列表时,来自列表索引 0 的消息将显示为最新索引,因此它现在重复了。当我从 xml 中删除 lottie 动画时,所有这些异常都会自行修复。但我希望能够像地球上所有其他聊天应用程序一样显示打字动画。如果有人可以帮助我了解出了什么问题,我将不胜感激。
Video snippet of scrollToPosition failing with lottie animation
为了澄清当我从 xml 中删除 lottie 动画时,scrollToPosition 会按预期工作,并且它们不是异常行为。
View 持有者 XML
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="@dimen/_6sdp"
android:layout_marginTop="@dimen/_12sdp">
<ImageView
android:id="@+id/icon"
android:layout_width="@dimen/_24sdp"
android:layout_height="@dimen/_24sdp"
android:layout_marginStart="@dimen/_6sdp"
android:layout_marginBottom="4dp"
android:contentDescription="User Icon"
app:layout_constraintBottom_toBottomOf="@+id/messageCv"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="@+id/messageCv"
app:layout_constraintVertical_bias="1.0"
app:srcCompat="@drawable/feelingfit_app_icon_legacy_circular_v1" />
<!--<com.airbnb.lottie.LottieAnimationView
android:id="@+id/typingAnimation"
android:layout_width="@dimen/_30sdp"
android:layout_height="@dimen/_30sdp"
android:layout_marginStart="@dimen/_12sdp"
app:lottie_autoPlay="true"
app:lottie_loop="true"
app:lottie_repeatCount="3"
android:visibility="visible"
app:lottie_rawRes="@raw/typing"
app:lottie_cacheStrategy="strong"
app:layout_constraintBottom_toBottomOf="@+id/icon"
app:layout_constraintStart_toEndOf="@+id/icon"
app:layout_constraintTop_toTopOf="@+id/icon"
/>-->
<com.google.android.material.card.MaterialCardView
android:id="@+id/messageCv"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:layout_marginEnd="@dimen/_40sdp"
android:visibility="invisible"
app:cardBackgroundColor="@color/light_blue_50"
app:cardCornerRadius="@dimen/_10sdp"
app:cardElevation="0dp"
app:cardPreventCornerOverlap="false"
app:cardUseCompatPadding="true"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toEndOf="@+id/icon"
app:layout_constraintTop_toTopOf="parent">
<LinearLayout
android:id="@+id/layout_gchat_container_other"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical">
<com.google.android.material.textview.MaterialTextView
android:id="@+id/message"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginBottom="@dimen/_6sdp"
android:maxWidth="@dimen/_230sdp"
android:paddingLeft="@dimen/_10sdp"
android:paddingTop="@dimen/_6sdp"
android:paddingRight="@dimen/_10sdp"
android:text="Message"
android:textColor="#000000"
android:textSize="@dimen/text_size_normal" />
</LinearLayout>
</com.google.android.material.card.MaterialCardView>
<com.google.android.material.textview.MaterialTextView
android:id="@+id/timestamp"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="@dimen/_10sdp"
android:layout_marginTop="@dimen/_4sdp"
android:layout_marginBottom="@dimen/_4sdp"
android:text="Today at 110:00pm"
android:textColor="#C0C0C0"
android:textSize="@dimen/_8sdp"
android:visibility="invisible"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintStart_toStartOf="@+id/messageCv"
app:layout_constraintTop_toBottomOf="@+id/messageCv" />
</androidx.constraintlayout.widget.ConstraintLayout>
适配器
class ReplaceAndEmbraceAdapter
constructor(
private val dateUtil: DateUtil,
) : RecyclerView.Adapter<RecyclerView.ViewHolder>() {
private var messageList: MutableList<ReplaceAppModel> = mutableListOf()
private val appMessage: Int = 1
private val userMessage: Int = 2
inner class ChatDiffCallback(
private val oldList: List<ReplaceAppModel>,
private val newList: List<ReplaceAppModel>
) : DiffUtil.Callback() {
override fun areItemsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean {
return oldList[oldItemPosition] == newList[newItemPosition]
}
override fun getOldListSize(): Int = oldList.size
override fun getNewListSize(): Int = newList.size
override fun areContentsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean {
return oldList[oldItemPosition] == newList[newItemPosition]
}
}
fun addChat(list: List<ReplaceAppModel>) = apply {
val diffCallback = ChatDiffCallback(messageList, list)
val diffResult = DiffUtil.calculateDiff(diffCallback)
messageList.clear()
messageList.addAll(list)
diffResult.dispatchUpdatesTo(this)
}
/* fun addChat(list: List<ReplaceAppModel>) {
messageList = list
notifyItemInserted(messageList.size)
}*/
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {
return when (viewType) {
appMessage -> {
MessageAppViewHolder(
RecyclerViewReplaceAndEmbraceAppBinding.inflate(
LayoutInflater.from(parent.context),
parent,
false
),
dateUtil
)
}
else -> MessageUserViewHolder(
RecyclerViewReplaceAndEmbraceUserBinding.inflate(
LayoutInflater.from(parent.context),
parent,
false
)
)
}
}
override fun onBindViewHolder(holder: RecyclerView.ViewHolder, position: Int) {
when (holder) {
is MessageAppViewHolder -> {
holder.bind(messageList?.get(position))
}
is MessageUserViewHolder -> {
holder.bind(messageList?.get(position))
}
}
}
override fun getItemCount(): Int {
return if (!messageList.isNullOrEmpty()) {
messageList!!.size
} else {
0
}
}
override fun getItemViewType(position: Int): Int = messageList[position].id
class MessageAppViewHolder
constructor(
private val binding: RecyclerViewReplaceAndEmbraceAppBinding,
private val dateUtil: DateUtil
) : RecyclerView.ViewHolder(binding.root) {
fun bind(item: ReplaceAppModel?) = with(itemView) {
binding?.apply {
typingAnimation.addAnimatorListener(object : Animator.AnimatorListener {
override fun onAnimationRepeat(animation: Animator?) {
}
override fun onAnimationEnd(animation: Animator?) {
typingAnimation.visibility = View.INVISIBLE
message.text =
HtmlCompat.fromHtml(item?.message!!, HtmlCompat.FROM_HTML_MODE_LEGACY)
icon.setImageResource(R.drawable.feelingfit_app_icon_legacy_circular_v1)
timestamp.text = dateUtil.getCurrentTimestampEDDMMMYYYYHHMM()
timestamp.visibility = View.VISIBLE
messageCv.visibility = View.VISIBLE
messageCv.fadeIn()
timestamp.fadeIn()
}
override fun onAnimationCancel(animation: Animator?) {
}
override fun onAnimationStart(animation: Animator?) {
}
})
}
}
}
最佳答案
在查看了我认为可用的代码之后,问题在于约束,所以首先尝试更改 typingAnimation
的约束和图标 w.r.t 父而不是 messageCv
.
但是看了视频后,我觉得你不用把动画放到recyclerView/adapter里。把它放在recyclerView下面就行了。我的意思是在 recyclerView 和 EditText 之间,并在您想要显示/隐藏动画时更改可见性。
如果您仍想在回收站 View 中显示动画,那么您应该为动画 View 创建一个单独的 View 。这意味着您将拥有三个 viewType 和 viewHolders。如果您在 recyclerView 中显示它,这是一个更好的方法。
关于android - 将 Lottie 动画添加到 Android recyclerview 会导致不稳定的结果,包括 View 显示在错误的位置和 scrollToPosition 错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67975909/
我在另一个 RecyclerView(parentRecyclerView) 中实现了 RecyclerView(childRecyclerView),图片说明了所有的实现: 我想编写一个 espre
我正在构建一个应用程序,它使用ViewPager2在页面之间水平滚动,每个页面都有一个垂直滚动的内容RecyclerView,其中每个列表的第一个位置都有一行水平滚动小部件,例如附加到寻呼机指示器类型
我正在尝试实现一个水平的 recyclerview 并且 recyclerview 的每个项目都将是一个具有网格布局的垂直 recyclerview。我面临的问题是,当我尝试垂直滚动子 recycle
我使用显示条目列表的 RecyclerView。每个条目都托管另一个 RecyclerView,它是一个图像列表。 我现在想让这个嵌套的 RecyclerView 可点击,而不是它的项目,而是整个 V
我正在尝试将手机中的图像加载到我的回收 View 中。 到目前为止,我已经创建了一个 recyclerview.adapter 和一个 gridlayoutmanager,并将它们附加到我的 recy
我需要一个可扩展 View ,其中父/可扩展标题将具有文本和下拉箭头。每个此类标题的子项应该是在网格结构中具有多个项目的 View 。 (有点像带有 GridLayoutManager 的回收器 Vi
我正在使用显示不同类别列表的 RecyclerView。每行项目还包含 RecyclerView 以显示类别项目列表。父级 RecyclerView 由垂直 LinearLayoutManager 填
我正在更新一个旧的 Android 项目,现在我从 RecyclerView 中反复收到这条日志语句: W/RecyclerView:RecyclerView 不支持滚动到绝对位置。改为使用 scro
在我的 Adapter 中,我调用 LayoutManager.ChildAt(position) 来获取 itemview,但是我得到的 View 不是匹配的 itemview,当我调用 notif
我正在从 ListView 迁移到 RecyclerView,但是在 SQLite 中输入一些数据后,我的列表没有使用 notifyDataSetChanged() 更新;所以我总是要调用setAda
我正在用 Google Now Cards 的风格创建一组 5 张卡片。我首先关注的是总体布局。 我正在使用 CardView 和 RecyclerView ,我想要实现的是这样的: 这是我的 Car
在我的应用程序中,我有一个 RecyclerView,其中包含一个简单的项目 View ,每行包含一个 TextView 和一个 Spinner。 用户从操作栏中单击“保存”后,我需要遍历所有项目并获
我正在尝试检查 RecyclerView 中是否可以看到某些特定项目;但我无法实现。请帮助我确定我的项目是否在 RecyclerView 中完全可见。 mrecylerView.addOnScroll
我有一个提要片段,它的主要元素是帖子的 RecyclerView。 我正在使用 Lisa Wray 的 Groupie 库管理回收站 https://github.com/lisawray/group
我有这两个类似的行为回收器 View 适配器,它们之间的唯一区别是 onclick 方法和传递给它们的对象。所以,我正在考虑将类 B 设计为从回收器 View 适配器继承,这样我就可以更改构造函数来执
这很奇怪。我对这个问题感到沮丧。我将数据保存在 ArrayList 中。当我将数据放在 RecyclerView 上时。数据未显示在列表中。这是我的代码 listOrder.forEach { ord
编辑我通过使用 CoordinatorLayout 和 AppBarLayout 作为 header 和 TabLayout 的包装解决了这个问题。本来应该很明显,但是哦,好吧。 原始问题:我有一个设
我有一个回收 View 。我想从其适配器类更新回收器 View 。我尝试了 notiftDataSetChanged( ) 但它只适用于主类。下面是 recyclerview 适配器类的代码。 pac
我正在使用具有不同 viewholders 的 recyclerview(具有垂直 LinearLayout),其中一个有另一个 RecyclerView(带有水平 LL)。在第二个 recycler
我想创建一个与 Google Play Store 或 Netflix App 类似的布局,其中在单个 RecyclerView 中基本上有多个水平 RecyclerView。如果这不是他们做事的方式
我是一名优秀的程序员,十分优秀!