Вид списка показывает элемент только после обновления

#android #kotlin #listview #android-fragments #android-adapter

Вопрос:

Я начал изучать android и однажды столкнулся с подобной проблемой: в моем фрагменте у меня есть editText и ListView . Я установил стандартный адаптер с ArraList<String> данными и увидел, что ListView элементы отображаются только после того, как я вызову нажатие клавиатуры editText . Я пытался найти какую-либо информацию об этом на многих сайтах по программированию, но столкнулся с неудачей.

Что у меня за проблема?

Мой xml-файл:

 <LinearLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    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"
    tools:context=".ui.forMainActivity.searchDish.SearchShopFragment"
    android:orientation="vertical">

    <LinearLayout
        android:id="@ id/linearLayoutCompat"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginVertical="10dp"
        android:weightSum="5">

        <EditText
            android:id="@ id/for_search_shop"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_weight="4" />

        <ImageButton
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:src="@drawable/ic_baseline_search_24" />

    </LinearLayout>

    <LinearLayout

        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical">

        <ListView
            android:id="@ id/for_list_of_categories"
            android:layout_width="match_parent"
            app:layout_behavior="@string/appbar_scrolling_view_behavior"
            android:layout_height="match_parent" />
    </LinearLayout>

</LinearLayout>
 

Мой фрагмент файла:

 override fun onCreateView(
    inflater: LayoutInflater, container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {
    arrayOfNames = ArrayList()
    db.collection("restaurants")
        .get()
        .addOnSuccessListener { documents ->
            for (document in documents) {
                arrayOfNames.add(document.data["name"].toString())
            }
        }
    val root = inflater.inflate(R.layout.fragment_search_shop, container, false)
    val adapter = ArrayAdapter(requireContext(), android.R.layout.simple_list_item_1, arrayOfNames)
    root.findViewById<ListView>(R.id.for_list_of_categories).adapter = adapter
    return root
}
 

Ответ №1:

Вам необходимо обновить адаптер синхронно с данными, так как addOnSuccessListener обратный вызов вызывается после установки адаптера, поэтому в приведенной ниже строке кода arrayOfNames список по-прежнему пуст.

 val adapter = ArrayAdapter(requireContext(), 
                           android.R.layout.simple_list_item_1, arrayOfNames)
 

Чтобы исправить это, вам нужно либо:

Установите адаптер в пределах обратного addOnSuccessListener вызова

 override fun onCreateView(
    inflater: LayoutInflater, container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {
    val root = inflater.inflate(R.layout.fragment_search_shop, container, false)
    arrayOfNames = ArrayList()
    db.collection("restaurants")
        .get()
        .addOnSuccessListener { documents ->
            for (document in documents) {
                arrayOfNames.add(document.data["name"].toString())
            }
            val adapter = ArrayAdapter(requireContext(), android.R.layout.simple_list_item_1, arrayOfNames)
            root.findViewById<ListView>(R.id.for_list_of_categories).adapter = adapter
        }
    return root
}
 

Или adapter.notifyDatasetChanged() в рамках обратного вызова.

 override fun onCreateView(
    inflater: LayoutInflater, container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {
    val root = inflater.inflate(R.layout.fragment_search_shop, container, false)
    val adapter = ArrayAdapter(requireContext(), android.R.layout.simple_list_item_1, arrayOfNames)
    root.findViewById<ListView>(R.id.for_list_of_categories).adapter = adapter
    arrayOfNames = ArrayList()
    db.collection("restaurants")
        .get()
        .addOnSuccessListener { documents ->
            for (document in documents) {
                arrayOfNames.add(document.data["name"].toString())
            }
            adapter.notifyDataSetChanged()
        }
    return root
}
 

Комментарии:

1. @yachek Рад, что смог помочь.. если вы нашли этот ответ на свой вопрос, пожалуйста, примите ответ, нажав на левую галочку.