繁体   English   中英

如何在 Fragment 的 Recycler 视图中访问每个项目的共享首选项

[英]How to access Shared Preferences for each item in Recycler View in Fragment

我想将按钮的共享首选项存储在片段使用的回收器视图适配器中。

class RecyclerViewAdapter : RecyclerView.Adapter<RecyclerViewAdapter.ViewHolder>() {

private val name = arrayOf("Android Dev W1",
    "Android Dev W2", "Web Dev W1", "Learn Kotlin B1",
    "Android Dev W3", "Android Dev W4", "Web Dev W2",
    "Learn DSA")

private val date = arrayOf("20:00 18th May, 2021", "04:00 18th May, 2021",
    "07:00 19th May, 2021", "04:00 20th May, 2021",
    "01:00 21th May, 2021", "10:00 21th May, 2021",
    "20:00 27th May, 2021", "10:00 30th May, 2021")

private val btn = arrayOf(false, false, false, false, false, false, false, false)

inner class ViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {

    var itemName: TextView = itemView.findViewById(R.id.workshopName)
    var itemDate: TextView = itemView.findViewById(R.id.workshopDate)
    var button: Button = itemView.findViewById(R.id.applyBtn)

    val prefs = itemView.context.getSharedPreferences("prefs", MODE_PRIVATE)

    init {

        for(i in 0 .. 7) {
            //checking via sharedPreferences if this app is being run first time or not
            btn[i] = prefs.getBoolean("registered$i", false)
            if (btn[i]) {
                hideButton()
            } else {
                showButton()
            }
        }

    }

     fun hideButton() {
         button.text = "Applied"
         button.isClickable = false
         button.setBackgroundColor(Color.parseColor("#FF3700E9"))
    }

    private fun showButton() {
        button.text = "Apply"
        button.isClickable = true
        button.setBackgroundColor(Color.parseColor("#FF6200EE"))
    }
}

override fun onCreateViewHolder(viewGroup: ViewGroup, i: Int): ViewHolder {
    val v = LayoutInflater.from(viewGroup.context)
        .inflate(R.layout.frame_textview, viewGroup, false)
    return ViewHolder(v)
}

override fun onBindViewHolder(viewHolder: ViewHolder, i: Int) {
    viewHolder.itemName.text = name[i]
    viewHolder.itemDate.text = date[i]
    viewHolder.button.setOnClickListener {
        viewHolder.prefs.edit().putBoolean("registered$i", true).apply()
        btn[i] = true
        viewHolder.hideButton()
    }

}

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

正在使用此适配器的片段

class AvailableWorkshops : Fragment() {

override fun onCreateView(
    inflater: LayoutInflater, container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {
    // Inflate the layout for this fragment
    return inflater.inflate(R.layout.fragment_available_workshops, container, false)
}

override fun onViewCreated(itemView: View, savedInstanceState: Bundle?) {
    super.onViewCreated(itemView, savedInstanceState)
    val recyclerView = itemView.findViewById<RecyclerView>(R.id.recycler_view)
    recyclerView.apply {
        // set a LinearLayoutManager to handle Android
        // RecyclerView behavior
        layoutManager = LinearLayoutManager(activity)
        // set the custom adapter to the RecyclerView
        adapter = RecyclerViewAdapter()
        }
    }
}

例如,此适配器中有 8 个项目,我单击第 5 个项目,它将调用 onClickListener() 并更改完全没问题的按钮属性。

但是,当我们重新启动应用程序时,会更改回默认按钮属性(共享首选项不起作用)

同样,如果我单击每个项目并重新启动应用程序,它现在将显示每个项目的新更改(共享首选项在这里工作)。

我想存储如果我单击第 5 项,然后在应用程序重新启动时它应该只更改第 5 项的显示,而不是每个项目的显示。

我想存储如果我单击第 5 项,然后在应用程序重新启动时它应该只更改第 5 项的显示,而不是每个项目的显示。

你的问题

ViewHolder代表RecyclerView中的一项 当您初始化ViewHolder时,您将首选项应用到同一个按钮7 次。

inner class ViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
  
 //...

 init {
        // ** THIS IS NOT HIDING OR SHOWING SEVEN DIFFERENT BUTTONS - THIS IS HIDING OR
        // SHOWING THIS VIEWHOLDER'S ONE BUTTON SEVEN TIMES **

        for(i in 0 .. 7) {
            //checking via sharedPreferences if this app is being run first time or not
            btn[i] = prefs.getBoolean("registered$i", false)
            if (btn[i]) {
                hideButton()
            } else {
                showButton()
            }
        }

    }

一个解法

删除ViewHolder中的init块并将隐藏/显示逻辑移动到设置每个实例onBindViewHolder方法。

override fun onBindViewHolder(viewHolder: ViewHolder, i: Int) {
    viewHolder.itemName.text = name[i]
    viewHolder.itemDate.text = date[i]

    // ** SHOW OR HIDE THIS INSTANCE'S BUTTON - AND ONLY THIS INSTANCE'S BUTTON **
    if (viewHolder.prefs.getBoolean("registered$i", false)) {
        viewHolder.hideButton()
    } else {
        viewHolder.showButton()
    }

    viewHolder.button.setOnClickListener {
        viewHolder.prefs.edit().putBoolean("registered$i", true).apply()
        btn[i] = true
        viewHolder.hideButton()
    }
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM