我试图将tableview单元格组织成列表中项目元素的部分(dueTime)。 Firebase是我的后端,每个项目都有一个名为dueTime的子节点,其中包含一个时间字符串。我创建了这些部分并让它们显示出来但我需要将实际的项目分开。目前,当我运行我的代码时,所有显示的都是部分。
override func numberOfSectionsInTableView(tableView: UITableView) -> Int {
let tasksRef = ref.childByAppendingPath("tasks")
tasksRef.observeSingleEventOfType(.Value, withBlock: {snapshot in
var dueTimesArray = [String]()
for task in snapshot.children.allObjects as! [FDataSnapshot] {
let times = task.value["dueTime"] as! String
dueTimesArray.append(times)
}
self.sectionTimes = dueTimesArray
})
let uniqueSectionTimes = Array(Set(sectionTimes))
return uniqueSectionTimes.count
}
override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
var uniqueSectionTimes = Array(Set(sectionTimes))
let tasksRef = Firebase(url: "\(self.ref)/tasks")
tasksRef.queryOrderedByChild("dueTime").queryEqualToValue(uniqueSectionTimes[section]).observeEventType(.Value, withBlock: { snapshot in
var newTasks = [Task]()
for task in snapshot.children.allObjects as! [FDataSnapshot] {
let tasks = Task(snapshot: task)
newTasks.append(tasks)
}
self.sectionTasks = newTasks
})
return self.sectionTasks.count
}
override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCellWithIdentifier("TaskCell", forIndexPath: indexPath) as! TaskCell
// Configure the cell...
cell.selectionStyle = .None
// let uniqueSectionTimes = Array(Set(sectionTimes))
// let times = self.sectionTasks[uniqueSectionTimes[indexPath.section]]
let task = tasks[indexPath.row]
cell.label.text = task.title
ref.childByAppendingPath("tasks").observeEventType(.Value, withBlock: { snapshot in
if task.done == true {
cell.checkBox.image = UIImage(named: "checkedbox")
cell.detailLabel.text = "Completed By: \(task.completedBy)"
}
else {
cell.checkBox.image = UIImage(named: "uncheckedbox")
cell.detailLabel.text = ""
}
})
cell.delegate = self
return cell
}
override func tableView(tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
let uniqueSectionTimes = Array(Set(sectionTimes))
return uniqueSectionTimes[section]
}
我觉得问题的根源可能在numberOfRowsInSection和cellForRowAtIndexPath中。从numberOfRowsInSection开始,当我将self.sectionTasks
设置为newTasks
后,我得到6个数组(等于部分的数量),其中所有正确的任务都在正确的数组中。但是,当我打印self.sectionTasks.count
时,我得到了' 0'六次对我来说没有意义。我不知道在cellForRowAtIndexPath
做什么。我似乎无法在任何地方找到一个很好的教程来解释它。
更新1:
我也在numberOfRows
override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
var uniqueSectionTimes = Array(Set(sectionTimes))
let tasksRef = Firebase(url: "\(self.ref)/tasks")
tasksRef.queryOrderedByChild("dueTime").queryEqualToValue(uniqueSectionTimes[section]).observeEventType(.Value, withBlock: { snapshot in
var newTasks = [String]()
for task in snapshot.children.allObjects as! [FDataSnapshot] {
let tasks = task.value["title"] as! String
newTasks.append(tasks)
}
// for task in snapshot.children.allObjects as! [FDataSnapshot] {
// let tasks = Task(snapshot: task)
// newTasks.append(tasks)
// }
self.sectionTasks = newTasks
})
print(sectionTasks.count)
return self.sectionTasks.count
}
我得到了同样的结果。基本上,这种方式只给出了项目的标题,而不是每个数组中的整个项目。但它仍然告诉我所有阵列的计数都是0。
更新2:
在实现以下代码之后,我现在正在获取每个部分下的所有重复任务。我想我需要以某种方式过滤任务,但我不知道在哪里或如何做到这一点。
func queryDueTimes(uniqueSectionTimes:Array<String>) {
let tasksRef = Firebase(url: "\(self.ref)/tasks")
tasksRef.queryOrderedByChild("dueTime").observeEventType(.Value, withBlock: { snapshot in
var newTasks = [Task]()
for task in snapshot.children.allObjects as! [FDataSnapshot] {
let times = Task(snapshot: task)
newTasks.append(times)
}
self.sectionTasks = newTasks
print(self.sectionTasks)
self.tableView.reloadData()
})
}
我正在viewDidLoad()
中运行此函数,并且正在获取一个包含每个任务内部所有元素的大数组。我想我需要使用&#34; &#34;节&#34;来自numberOfRows
的元素,但我不确定如何。另外,我正在查看的教程显示了cellForRow
中indexPath.section的使用,但我不确定如何实现其中任何一个。
解决方案:
我最终完全更改了查询代码,因此我从本地数组而不是Firebase中提取。
func querySections() -> [String] {
var sectionsArray = [String]()
for task in tasks {
let dueTimes = task.dueTime
sectionsArray.append(dueTimes)
}
let uniqueSectionsArray = Array(Set(sectionsArray)).sort()
return uniqueSectionsArray
}
func queryDueTimes(section:Int) -> [Task] {
var sectionItems = [Task]()
for task in tasks {
let dueTimes = task.dueTime
if dueTimes == querySections()[section] {
sectionItems.append(task)
}
}
return sectionItems
}
// MARK: - Table view data source
override func numberOfSectionsInTableView(tableView: UITableView) -> Int {
return querySections().count
}
override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return queryDueTimes(section).count
}
override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCellWithIdentifier("TaskCell", forIndexPath: indexPath) as! TaskCell
// Configure the cell...
cell.selectionStyle = .None
let times = queryDueTimes(indexPath.section)
let task = times[indexPath.row]
cell.label.text = task.title
if task.done == true {
cell.checkBox.image = UIImage(named: "checkedbox")
cell.detailLabel.text = "Completed By: \(task.completedBy)"
}
else {
cell.checkBox.image = UIImage(named: "uncheckedbox")
cell.detailLabel.text = ""
}
cell.delegate = self
return cell
}
override func tableView(tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
return querySections()[section]
}
答案 0 :(得分:2)
这是因为对taskRef.query...
的调用是异步的。此时numberOfRowsInSection
方法预计您已经知道自己拥有的行数。
将查询代码移动到其他方法中,例如
func queryDueTimes(uniqueSectionTimes: Array) {
let tasksRef = Firebase(url: "\(self.ref)/tasks")
tasksRef.queryOrderedByChild("dueTime").queryEqualToValue(uniqueSectionTimes[section]).observeEventType(.Value, withBlock: { snapshot in
var newTasks = [String]()
for task in snapshot.children.allObjects as! [FDataSnapshot] {
let tasks = task.value["title"] as! String
newTasks.append(tasks)
}
// for task in snapshot.children.allObjects as! [FDataSnapshot] {
// let tasks = Task(snapshot: task)
// newTasks.append(tasks)
// }
self.sectionTasks = newTasks
//IMPORTANT: reload your table here:
self.tableView.reloadData()
})
}
然后,从视图控制器的queryDueTimes
方法调用viewDidLoad()
方法:
func viewDidLoad()
{
super.viewDidLoad()
var uniqueSectionTimes = Array(Set(sectionTimes))
self.queryDueTimes(uniqueSectionTimes)
}