Значения EditText в RecyclerView меняются при добавлении элементов в RecyclerView - PullRequest
0 голосов
/ 10 июля 2020

У меня есть RecyclerView, который содержит список представлений с полями EditText в них.

Я хочу иметь возможность изменять значение в EditTexts и обновлять мой набор данных. Кажется, я это делаю, но когда я добавляю новый элемент в список, значения в EditTexts смешиваются.

Я не могу понять, почему это происходит. Есть мысли?

Основное действие

class MainActivity : AppCompatActivity() {
    private lateinit var recyclerView: RecyclerView
    private lateinit var viewAdapter: RecyclerView.Adapter<*>
    private lateinit var viewManager: RecyclerView.LayoutManager

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

        val myDataSet = ArrayList<Person>()
        myDataSet.add(Person("Tommy"))
        myDataSet.add(Person("John"))
        myDataSet.add(Person("Suzie"))
        myDataSet.add(Person("Leslie"))

        viewManager = LinearLayoutManager(this)
        viewAdapter = PersonAdapter(myDataSet)

        recyclerView = findViewById<RecyclerView>(R.id.rvPersons).apply {
            setHasFixedSize(true)

            layoutManager = viewManager
            adapter = viewAdapter
        }

        val button = findViewById<Button>(R.id.myButton)

        button.setOnClickListener {
            myDataSet.add(Person("new person"))
            myDataSet.forEach {person ->
                Log.i("my-tag", person.name)
            }
            rvPersons.adapter?.notifyDataSetChanged()
        }
    }
}

Адаптер людей

class PersonAdapter(private val myDataSet: ArrayList<Person>) :
    RecyclerView.Adapter<PersonAdapter.MyViewHolder>() {

    class MyViewHolder(view: View) : RecyclerView.ViewHolder(view) {
        val editText: EditText = view.my_edittext
        var textView: TextView = view.my_textView
    }

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

    override fun onBindViewHolder(holder: MyViewHolder, position: Int) {
        holder.editText.setText(myDataSet[position].name)
        holder.editText.doAfterTextChanged { text ->
            if (!text.isNullOrBlank()) {
                myDataSet[position].name = text.toString()
            }
        }
        holder.textView.text = myDataSet[position].name
    }

    override fun getItemCount(): Int = myDataSet.size
}

Ответы [ 2 ]

2 голосов
/ 11 июля 2020

Поскольку все старые TextWachers остаются зарегистрированными из-за вызовов onBindViewHolder, данные должны быть повреждены. Сохраните ссылку на TextWatcher в MyViewHolder и сначала удалите ее в onBindViewHolder ().

class PersonAdapter(private val myDataSet: ArrayList<Person>) :
    RecyclerView.Adapter<PersonAdapter.MyViewHolder>() {

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

        val editText: EditText = view.my_edittext
        var textView: TextView = view.my_textView
        var watcher: TextWatcher? = null
    }

    override fun onBindViewHolder(holder: MyViewHolder, position: Int) {

        holder.editText.removeTextChangedListener(holder.watcher)

        holder.editText.setText(myDataSet[position].name)
        holder.watcher = holder.editText.doAfterTextChanged { text ->
            if (!text.isNullOrBlank()) {
                myDataSet[position].name = text.toString()
            }
        }
        holder.textView.text = myDataSet[position].name
    }
    
    :
}
0 голосов
/ 11 июля 2020

Вы должны позвонить this.notifyDataSetChanged(); после добавления нового текста в editText.

...