LiveData не обновляет данные - PullRequest
       31

LiveData не обновляет данные

0 голосов
/ 21 февраля 2020

Я хотел бы попросить вас о помощи. Я пишу приложение, которое использует MVVM и архитектуру LiveData. Внутри ViewPager у меня есть 3 фрагмента, отображающие данные, которые поступают из ViewModel. И я заметил, что после подключения viewModel к действию и к фрагменту данные обновляются только при запуске действия, но затем Observe не обновляет данные, даже если данные изменились. После вызова следующего запроса к серверу, внутри onDataSet я отправляю соответствующее время и получаю с сервера данные JSON, которые он анализирует и передает ViewModel. Почему Fragment обновляет данные только один раз в начале, а после ничего не меняется?

Это действие, в котором хранятся фрагменты

class MainActivity : AppCompatActivity(), DatePickerDialog.OnDateSetListener {

    private lateinit var currencyViewModel: CurrencyViewModel
    private lateinit var viewPager: ViewPager2
    private lateinit var tabLayout: TabLayout
    private lateinit var navigationView: NavigationView
    private lateinit var floatingActionButton: FloatingActionButton

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

        currencyViewModel = ViewModelProvider
            .AndroidViewModelFactory(application)
            .create(CurrencyViewModel::class.java)

        viewPager = findViewById(R.id.viewPager)
        tabLayout = findViewById(R.id.tabLayout)
        navigationView = findViewById(R.id.navigationView)
        floatingActionButton = findViewById(R.id.floatingActionButton)

        val viewPagerAdapter = CurrencyViewPagerAdapter(this)
        viewPager.adapter = viewPagerAdapter

        TabLayoutMediator(tabLayout
            ,viewPager
            ,TabLayoutMediator.TabConfigurationStrategy {
                    tab, position ->
                when(position){
                    0 -> tab.text = "Tabela A"
                    1 -> tab.text = "Tabela B"
                    2 -> tab.text = "Tabela C"
                }
            }).attach()
        floatingActionButton.setOnClickListener {
            val dialog = CalendarFragment()
            dialog.show(fm, "DatePickerDialog")
        }
    }
    override fun onDateSet(view: DatePicker?, year: Int, month: Int, dayOfMonth: Int) {
        //Convert year,month,day to millisecounds
        val c = Calendar.getInstance()
        c.set(year,month,dayOfMonth)
        val dayInMillis = c.time.time
        val today = Calendar.getInstance()

        if(checkIsDateAfterToday(today, c)){
            CoroutineScope(Dispatchers.Main).launch {
                currencyViewModel.setTableA(dayInMillis)
            }
        }

    }

Это ViewModel, общая для действия и фрагмента

class CurrencyViewModel : ViewModel() {

    private val repository = CurrencyRepository()
    val tableA: MutableLiveData<Array<TableA>> by lazy {
        MutableLiveData<Array<TableA>>().also {
            loadTableA(Date().time)
        }
    }

    private fun loadTableA(time: Long) {
            CoroutineScope(Dispatchers.Main).launch {
               val loadedData = CoroutineScope(Dispatchers.IO).async {
                    repository.getTableA(time)
                }.await()
                tableA.value = loadedData
            }
    }

   fun setTableA(time: Long){
       loadTableA(time)
   }

}

И это фрагмент, который отображает данные в recyclerView

class TableAFragment : Fragment() {

    private lateinit var currencyViewModel: CurrencyViewModel
    private lateinit var recyclerViewA: RecyclerView

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {

        return inflater.inflate(R.layout.fragment_table_a, container, false)
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        currencyViewModel = ViewModelProvider.AndroidViewModelFactory
            .getInstance(requireActivity().application)
            .create(CurrencyViewModel::class.java)

    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)

        recyclerViewA = view.findViewById(R.id.recyclerView_A)
        recyclerViewA.layoutManager = LinearLayoutManager(requireContext())

        currencyViewModel.tableA.observe(viewLifecycleOwner, androidx.lifecycle.Observer{
            val nbpAdapter = NBPAdapter(it)
            recyclerViewA.adapter = nbpAdapter
        })

    }

}

1 Ответ

1 голос
/ 22 февраля 2020

Ваш экземпляр ViewModel неверен.

Должен быть

currencyViewModel = ViewModelProvider(this).get<CurrencyViewModel>() // lifecycle-ktx

и во фрагменте:

currencyViewModel = ViewModelProvider(requireActivity()).get<CurrencyViewModel>() // lifecycle-ktx
...