如何在 RecyclerView 中突出显示并自动单击最近添加的项目

Posted

技术标签:

【中文标题】如何在 RecyclerView 中突出显示并自动单击最近添加的项目【英文标题】:How to highlight and auto perform click on recently added item in RecyclerView 【发布时间】:2021-11-01 13:09:42 【问题描述】:

我有两个场景,所以第一个是我想突出显示并自动单击 RecyclerView 中最近添加的项目,第二个是如果用户选择或单击任何项​​目,此时也应该突出显示和点击所以第二个场景我已经实现了,但我不知道如何突出显示并自动执行单击列表中最后添加的项目。

我的适配器类

class CarItemAdapter(
        private val activity: MainActivity,
        private val carList: ArrayList<MyEntity>,
        private val myViewModel: MyViewModel,
        private val settings: Settings
) :
        RecyclerView.Adapter<RecyclerView.ViewHolder>() 
    var listener: OnItemsClickListener? = null
    var rowIndex = 0

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder 
        val layoutInflater = LayoutInflater.from(parent.context)
        return when (viewType) 
            VIEW_TYPE_ADDITEM -> ViewHolderAddCarItem(SettingsCarAddItemTileBinding.inflate(layoutInflater, parent, false))
            VIEW_TYPE_ITEM -> ViewHolderCarItem(SettingsCarItemTileBinding.inflate(layoutInflater, parent, false))
            else -> throw IllegalArgumentException("Invalid view type")
        
    

    override fun onBindViewHolder(holder: RecyclerView.ViewHolder, position: Int) 

        when (holder) 
            is ViewHolderCarItem -> holder.bind(carList[position -1], position-1)
            is ViewHolderAddCarItem -> holder.bind()
        
    

    inner class ViewHolderAddCarItem internal constructor(private var binding: SettingsCarAddItemTileBinding) : RecyclerView.ViewHolder(binding.root) 
        fun bind() 
            binding.lifecycleOwner = activity
            binding.handler = EventHandler()
            binding.userSpecificCarSubtitle.setText(R.string.user_specific_car_subtitle)
        

        inner class EventHandler 
            fun onItemClicked() 
                activity.showSubPage(AddVehiclePage(activity), true)
            
        
    

    inner class ViewHolderCarItem internal constructor(private var binding: SettingsCarItemTileBinding) : RecyclerView.ViewHolder(binding.root) 
        fun bind(myEntity: MyEntity, position: Int) 

            binding.lifecycleOwner = activity
            val bmp = BitmapFactory.decodeByteArray(myEntity.image, 0, Objects.requireNonNull<ByteArray>(myEntity.image).size)
            binding.carImage.setImageBitmap(bmp)
            binding.carName.text = myEntity.label

            binding.layoutCar.setOnClickListener 
                listener?.onItemClick(myEntity)
                settings.carNumber = myEntity.carNumber
                settings.selectedCarType = activity.getString(R.string.settings_category_view_sub_title_for_user_specific_car)
                settings.isCarSelected = true
                rowIndex = position
                notifyDataSetChanged()
            

            binding.deleteButton.setOnClickListener 
                GlobalScope.launch(Dispatchers.IO)  bbdCarViewModel.deleteByFinNumber(myEntity.carNumber) 
                settings.isBBDCarSelected = false
                carList.removeAt(position)
                notifyDataSetChanged()
            

            if (rowIndex == position) 
                binding.layoutCar.setBackgroundColor(activity.getColor(R.color.petrol))
                binding.carName.setTextColor(activity.getColor(R.color.white))
             else 
                binding.layoutCar.setBackgroundColor(activity.getColor(R.color.coolGray))
                binding.carName.setTextColor(activity.getColor(R.color.coolGray_80k))
            
        
    

    override fun getItemViewType(position: Int): Int 
        return if (position == 0 && position <= carList.size) 
            VIEW_TYPE_ADDITEM
         else VIEW_TYPE_ITEM
    

    override fun getItemCount(): Int 
        return carList.size + 1
    

    interface OnItemsClickListener 
        fun onItemClick(myEntity: MyEntity)
    

    fun setWhenClickListener(listener: OnItemsClickListener) 
        this.listener = listener
    

    companion object 
        const val VIEW_TYPE_ITEM = 0
        const val VIEW_TYPE_ADDITEM = 1
    

请帮帮我

【问题讨论】:

检查 if( carList.size()-1 == position ),然后你会捕捉到列表中最后添加的项目。 是的,这样做也是如此,但这样我的最后两个项目就会突出显示。 【参考方案1】:

我们可以为适配器类添加占位符,并通过唯一的 id 在其中存储突出显示的项目。然后在绑定函数中,我们可以突出显示具有这些 id 的项目:

1。添加占位符

我们可以使用 占位符 来保存突出显示的项目来处理活动状态。

它可以是一个 HashSet ,Long 的类型是用于项目 id 的,我们将在 init 块中将其设置为稳定。

class CarItemAdapter(...) : ... 
...
val activeHolder: HashSet<Long> = HashSet()
    
init 
    // Represents each item in the data set with a unique indentifier
    setHasStableIds(true)
 
...


2。将最近添加的项目添加到 placeHolder

现在我们为每件商品设置了一个唯一标识符,并为每个商品设置了一个占位符 保持突出显示项目的活动状态。

我们希望最后添加的项目被突出显示,所以假设您在 EventHandler 的 onclick 方法中添加新项目,更新如下:

inner class EventHandler 
            fun onItemClicked() 
                with(activeHolder)                  
                    val lastItemId = getItemId(carList.size - 1)
                    // Remove last item's id from our holder to prevent last two items being highlighted.
                    if (contains(lastItemId)) remove(lastItemId) 
                    // Suppose we add new item to carList here
                    carList.add(newItem)
                    // add last item id of new list to the holder   
                    add(getItemId(carList.size - 1))   
                
                notifyDataSetChanged()
                
            
        

在上面的代码中,我们假设将新项目添加到列表中是在 carList.add(newItem) 内,如果不是这种情况,请将上面的 with() 代码块包裹在 carList.add(newCar) 周围。


3。通过 onClick 操作向占位符添加和删除项目

这个很简单,只是一个单行:


binding.layoutCar.setOnClickListener 
    ...
    with(activeHolder)  if (contains(itemId)) remove(itemId) else add(itemId) 
    ...
notifyDataSetChanged()

4。压轴;执行突出显示


    inner class ViewHolderCarItem internal constructor(private var binding: SettingsCarItemTileBinding) : RecyclerView.ViewHolder(binding.root) 
        fun bind(myEntity: MyEntity, position: Int) 
            ...

            binding.layoutCar.setOnClickListener 
                ...
                with(activeHolder)  if (contains(itemId)) remove(itemId) else add(itemId) 
                ...
                notifyDataSetChanged()
            

            binding.deleteButton.setOnClickListener 
                   ...
            

            // Activates last item, or item's clicked
            itemView.isActivated = activeHolder.contains(itemId)

            if(itemView.isActivated) 
                // Do anything you want by highlighted items.
            

        
    


我希望这个答案有所帮助。


更新

为了执行点击最后一个突出显示的项目:

在活动的 onCreate() 方法中:

lifecycle.coroutineScope.launch 
            lifecycle.repeatOnLifecycle(Lifecycle.State.STARTED) 
                delay(300)
                recyclerView.findViewHolderForAdapterPosition(recyclerView.size-1)?.itemView?.performClick()
           

注意:对于 repeatOnLifecycle() 方法,你需要 androidx.lifecycle:lifecycle-runtime-ktx:2.4.0-alpha01 依赖。

【讨论】:

当然我会试试你建议的答案。但是我们如何在突出显示的列表中执行自动点击最近添加的项目? 将 binding.layoutCar.setOnClickListener 中的所有代码抽象为一个函数,并在 binding.layoutCar.setOnClickListener 块内以及 EventHandler 的内部 onItemClick 内调用该函数 所以 setOnClickListener 只会在我们点击任何项目时调用,但没有用户点击我想对最后一个突出显示的项目执行点击操作。那么我们该怎么做呢?? 通过itemView上的performClick()方法,刚刚更新了答案。 实际上我的子页面在java类中,所以我不能使用协程。为什么你使用协程和延迟来执行自动点击?

以上是关于如何在 RecyclerView 中突出显示并自动单击最近添加的项目的主要内容,如果未能解决你的问题,请参考以下文章

RecyclerView 显示两个突出显示的项目

为啥在recyclerview android中滚动后突出显示的项目丢失

如何突出显示 Recycler View 的选定项目?

如何将 QTreeWidget 的子项设置为在单击其父项时自动选择并突出显示

IntelliJ:如何在 Eclipse 中自动突出显示变量

如何在 Rider 中更改自动完成突出显示颜色