android - 使用 Firebase 时,ListView 仅在 onCreate() 方法中更新
问题描述
我正在编写待办事项列表,并且我希望 ListView 在用户向列表中添加新任务或删除它们时显示更改。我不知道为什么,但是 adapter.notifyDataSetChanged() 仅在应用程序使用 onCreate() 方法(打开,更改为水平)时才有效。
我试图将 adapter.notifyDataSetChanged() 放在任何地方,但它不会更新内容。只有在将任务添加到数据库(它显示在 Firebase 控制台上)并更改为水平/重新启动后,您才能在 ListView 中看到新的输入。我还创建了按钮,它在 onClick() 方法中仅使用 adapter.notifyDataSetChanged() 。
MainActivity.kt:
package com.example.toodoo
import android.app.AlertDialog
import android.support.v7.app.AppCompatActivity
import android.os.Bundle
import android.support.design.widget.FloatingActionButton
import android.util.Log
import android.view.View
import android.widget.EditText
import android.widget.ListView
import android.widget.Toast
import com.google.firebase.database.*
class MainActivity : AppCompatActivity(), ItemRowListener {
//Get Access to Firebase database, no need of any URL, Firebase
//identifies the connection via the package name of the app
lateinit var mDatabase: DatabaseReference
var toDoItemList: MutableList<ToDoItem>? = null
lateinit var adapter: ToDoItemAdapter
private var listViewItems: ListView? = null
override fun modifyItemState(itemObjectId: String, isDone: Boolean) {
val itemReference = mDatabase.child(Constants.FIREBASE_ITEM).child(itemObjectId)
itemReference.child("done").setValue(isDone);
}
//delete an item
override fun onItemDelete(itemObjectId: String) {
//get child reference in database via the ObjectID
val itemReference = mDatabase.child(Constants.FIREBASE_ITEM).child(itemObjectId)
//deletion can be done via removeValue() method
itemReference.removeValue()
}
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
//reference for FAB
val fab = findViewById<View>(R.id.fab) as FloatingActionButton
listViewItems = findViewById<View>(R.id.items_list) as ListView
//Adding click listener for FAB
fab.setOnClickListener { view ->
//Show Dialog here to add new Item
addNewItemDialog()
}
mDatabase = FirebaseDatabase.getInstance().reference
toDoItemList = mutableListOf<ToDoItem>()
adapter = ToDoItemAdapter(this, toDoItemList!!)
listViewItems!!.setAdapter(adapter)
mDatabase.orderByKey().addListenerForSingleValueEvent(itemListener)
}
private fun addNewItemDialog() {
val alert = AlertDialog.Builder(this)
val itemEditText = EditText(this)
alert.setMessage("Add New Item")
alert.setTitle("Enter To Do Item Text")
alert.setView(itemEditText)
alert.setPositiveButton("Submit") { dialog, positiveButton ->
val todoItem = ToDoItem.create()
todoItem.itemText = itemEditText.text.toString()
todoItem.done = false
//We first make a push so that a new item is made with a unique ID
val newItem = mDatabase.child(Constants.FIREBASE_ITEM).push()
todoItem.objectId = newItem.key
//then, we used the reference to set the value on that ID
newItem.setValue(todoItem)
dialog.dismiss()
Toast.makeText(this, "Item saved with ID " + todoItem.objectId, Toast.LENGTH_SHORT).show()
}
alert.show()
}
var itemListener: ValueEventListener = object : ValueEventListener {
override fun onDataChange(dataSnapshot: DataSnapshot) {
// Get Post object and use the values to update the UI
addDataToList(dataSnapshot)
}
override fun onCancelled(databaseError: DatabaseError) {
// Getting Item failed, log a message
Log.w("MainActivity", "loadItem:onCancelled", databaseError.toException())
}
}
private fun addDataToList(dataSnapshot: DataSnapshot) {
val items = dataSnapshot.children.iterator()
//Check if current database contains any collection
if (items.hasNext()) {
val toDoListindex = items.next()
val itemsIterator = toDoListindex.children.iterator()
//check if the collection has any to do items or not
while (itemsIterator.hasNext()) {
//get current item
val currentItem = itemsIterator.next()
val todoItem = ToDoItem.create()
//get current data in a map
val map = currentItem.getValue() as HashMap<String, Any>
//key will return Firebase ID
todoItem.objectId = currentItem.key
todoItem.done = map.get("done") as Boolean?
todoItem.itemText = map.get("itemText") as String?
toDoItemList!!.add(todoItem);
}
}
//alert adapter that has changed
adapter.notifyDataSetChanged()
}
fun onClick(view: View){
adapter.notifyDataSetChanged()
}
}
我认为这可能也是将 Firebase 数据与 ListView 同步的问题,但我是第一次使用它,我无法解决这个问题。这是我第一次使用该数据库,所以我错过了一些东西。
解决方案
您正在为数据添加一个监听器:
mDatabase.orderByKey().addListenerForSingleValueEvent(itemListener)
由于您使用的是addListenerForSingleValueEvent
,这仅侦听当前数据,然后删除侦听器。因此,在加载当前数据后,您将不再监听更改。
要继续侦听初始数据和之后的更改,请使用:
mDatabase.orderByKey().addValueEventListener(itemListener)
请注意,您可能希望清空toDoItemList
,addDataToList
否则只要有更改,您就会一次又一次地添加所有现有项目。
或者使用addChildEventListener
,它可以为您提供有关子节点已更改内容的更详细信息,以便您对适配器执行细粒度更新。
推荐阅读
- javascript - v-options 保持相同的选项 [VUE]
- node.js - 如何加密 nodeJs 中应用作索引的字符串
- flask - flask-sqlalchemy - 在不同情况下改变懒惰
- python - python postgresql 通过网络复制到/从压缩流
- c - 用指针输入定义函数
- python - 修复库中的 Pandas 面板弃用以下载多个股票代码
- xamarin - Xamarin.Auth NUGet 包未安装
- java - If-then-else 语句使用
- python - python中自定义类对象的参数传递:按引用调用还是按值调用评估?
- java - Java array.sort(arr) 输出 0 而不是 arr