【问题标题】:SelectionTracker does not select anything in my RecyclerviewSelectionTracker 没有在我的 Recyclerview 中选择任何内容
【发布时间】:2019-01-17 08:16:35
【问题描述】:

当我点击项目时,没有任何反应 我尝试更改颜色,显示所选元素的数量,显示日志 什么都没有发生

MyViewHolder

class MyViewHolder(view: View) : RecyclerView.ViewHolder(view) {

val name: TextView = view.list_item_name
val phone: TextView = view.list_item_phone

// More code here
fun getItemDetails(): ItemDetailsLookup.ItemDetails<Long> =
    object : ItemDetailsLookup.ItemDetails<Long>() {
        override fun getSelectionKey(): Long? = itemId

        override fun getPosition(): Int = adapterPosition

    }

}

我的 ItemDetailsLookup

class MuLookup(private val rv: RecyclerView) : ItemDetailsLookup<Long>() {
    override fun getItemDetails(event: MotionEvent): ItemDetails<Long>? {
        val view = rv.findChildViewUnder(event.x, event.y)
        if (view != null) {
            return (rv.getChildViewHolder(view) as MyViewHolder)
                .getItemDetails()
        }
        return null
    }
}

我的适配器

class PersonAdapter(
    private val listItems: List<Person>,
    private val context: Context
) : RecyclerView.Adapter<MyViewHolder>() {

    init {
        setHasStableIds(true)
    }

    private var tracker: SelectionTracker<Long>? = null

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): MyViewHolder {
        return MyViewHolder(
            LayoutInflater.from(context)
                .inflate(R.layout.list_item_jf, parent, false)
        )
    }

    override fun getItemCount(): Int {
        return listItems.size
    }

    override fun onBindViewHolder(holder: MyViewHolder, position: Int) {
        holder.name.text = listItems[position].name
        holder.phone.text = listItems[position].phone

        val parent = holder.name.parent as LinearLayout

        if (tracker!!.isSelected(position.toLong())) {
            parent.background = ColorDrawable(
                Color.parseColor("#80deea")
            )
        } else {
            Timber.i("zeze ${tracker!!.selection}")
            // Reset color to white if not selected
            parent.background = ColorDrawable(Color.WHITE)
        }

    }

    override fun getItemId(position: Int): Long {
        return position.toLong()
    }

    fun setTracker(tracker: SelectionTracker<Long>?) {
        this.tracker = tracker
    }

}

我的活动

class MainActivity : AppCompatActivity() {

    private var tracker: SelectionTracker<Long>? = null

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        if (savedInstanceState != null)
            tracker?.onRestoreInstanceState(savedInstanceState)

        val myList = listOf(
            Person("Alice", "555-0111"),
            Person("Bob", "555-0119"),
            Person("Carol", "555-0141"),
            Person("Dan", "555-0155"),
            Person("Eric", "555-0180"),
            Person("Craig", "555-0145")
        )

        my_rv.layoutManager = LinearLayoutManager(this)
        my_rv.setHasFixedSize(true)

        val adapter = PersonAdapter(myList, this)
        my_rv.adapter = adapter

        tracker = SelectionTracker.Builder<Long>(
            "selection-1",
            my_rv,
            StableIdKeyProvider(my_rv),
            MuLookup(my_rv),
            StorageStrategy.createLongStorage()
        ).withSelectionPredicate(
            SelectionPredicates.createSelectAnything()
        ).build()

        tracker?.addObserver(
            object : SelectionTracker.SelectionObserver<Long>() {
                override fun onSelectionChanged() {
                    val nItems: Int? = tracker?.selection?.size()

                    if (nItems != null && nItems > 0) {

                        // Change title and color of action bar

                        title = "$nItems items selected"
                        supportActionBar?.setBackgroundDrawable(
                            ColorDrawable(Color.parseColor("#ef6c00"))
                        )
                    } else {

                        // Reset color and title to default values

                        title = "RVSelection"
                        supportActionBar?.setBackgroundDrawable(
                            ColorDrawable(Color.parseColor("#126c00"))
                        )
                    }
                }
            })

        adapter.setTracker(tracker)


    }

    override fun onSaveInstanceState(outState: Bundle) {
        super.onSaveInstanceState(outState)

        tracker?.onSaveInstanceState(outState)
    }
}

谢谢

解决方案:跟踪器工作我像 Khaled Qasem 建议的那样在我的视图上添加了一个 setOnClickListener

override fun onBindViewHolder(holder: MyViewHolder, position: Int) {
    holder.name.text = listItems[position].name
    holder.phone.text = listItems[position].phone

    val parent = holder.name.parent as LinearLayout

    holder.name.setOnClickListener {
        Timber.i("zeze ici")
        if (tracker!!.isSelected(position.toLong())) {
            Timber.i("zeze la")
            parent.background = ColorDrawable(
                Color.parseColor("#80deea")
            )
        } else {
            Timber.i("zeze ${tracker!!.selection}")
            // Reset color to white if not selected
            parent.background = ColorDrawable(Color.WHITE)
        }
    }
}

supportActionBar 中显示颜色变化和项目数

【问题讨论】:

    标签: android kotlin android-recyclerview tracker


    【解决方案1】:

    问题是 Recyclerview 选择仅在最初选择一个项目时才会激活,然后您只需点击它们即可选择其他项目。

    如果您总是需要用户选择一个项目,或者选择一个项目,您可以使用selectionTracker 从回收器视图中最初选择一个项目:

    selectionTracker.select(someItemId)
    

    现在,您只需点按一下即可从列表中选择其他项目。

    【讨论】:

      【解决方案2】:

      您只需要在创建视图持有者时添加一个单击侦听器即可:

      override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): MyViewHolder {
      
          val myViewHolder = MyViewHolder(
              LayoutInflater.from(context)
                  .inflate(R.layout.list_item_jf, parent, false)
          )
          myViewHolder.setOnClickListener {
              // Add what you what you need here
          }
          return myViewHolder
      }
      

      【讨论】:

        【解决方案3】:

        就我而言,解决方案是告诉用户长按第一个选择:

        长按进行第一项设置,然后点按其他设置进行多项 选择

        我的实现有所不同,但我也观察到 SelectionTracker 从我的 RecyclerView 中没有选择任何内容。我无法从一些SelectionTracker example code 中理解它应该如何工作。该示例展示了如何在 ViewHolder 被回收时设置高亮:

        override fun onBindViewHolder(holder: ViewHolder, position: Int) {
            val number = list[position]
            tracker?.let {
                holder.bind(number, it.isSelected(position.toLong()))
            }
        }
        
        fun bind(value: Int, isActivated: Boolean = false) {
            text.text = value.toString()
            itemView.isActivated = isActivated
        }
        

        但是当用户点击该行时,isActivated 应该如何设置为 true?

        答案是用户不应该点击;用户应该长按:

        要开始选择项目,我们必须首先通过长按任何项目来激活多选模式。

        【讨论】:

        • 我怎样才能一键选择项目?
        • @Andrey See my answer
        • 选择单个项目使用SelectionPredicates.createSelectSingleAnything()而不是SelectionPredicates.createSelectAnything()
        • 如何仅在用户长按某个项目时强制选择?
        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多