我正在尝试用我的帖子中的评论填充我的commentsTable。我具有以下数据库结构JSON:
{
"posts" : {
"-Lhu-XRs806sXSEQS2BF" : {
"reports" : 0,
"text" : "How can I improve my data structure?",
"timestamp" : 1561120090116,
"title" : "Hello Stack Exchange",
"userID" : "nyRBXSyyDhc1Qkypou0Iz0iMsyr1"
},
"-Lhu-fI6DMSZvy8EdIgM" : {
"reports" : 0,
"text" : "As in Libre",
"timestamp" : 1561120126347,
"title" : "Free",
"userID" : "nyRBXSyyDhc1Qkypou0Iz0iMsyr1"
},
"comments" : {
"-Lhu-hXISy-0N2V4ES-a" : {
"reports" : 0,
"timestamp" : 1561120135594,
"userID" : "nyRBXSyyDhc1Qkypou0Iz0iMsyr1"
},
"-Lhu-j1cR6V407tyUYY1" : {
"reports" : 0,
"timestamp" : 1561120141801,
"userID" : "nyRBXSyyDhc1Qkypou0Iz0iMsyr1"
},
"-Lhu-lrJp9H8SQowlYWz" : {
"reports" : 0,
"timestamp" : 1561120153314,
"userID" : "nyRBXSyyDhc1Qkypou0Iz0iMsyr1"
},
"posts" : {
"-Lhu-XRs806sXSEQS2BF" : {
"comments" : {
"-Lhu-hXISy-0N2V4ES-_" : "How is it going?",
"-Lhu-j1cR6V407tyUYY0" : "It’s good to see you"
}
},
"-Lhu-fI6DMSZvy8EdIgM" : {
"comments" : {
"-Lhu-lrJp9H8SQowlYWy" : "Richard Stallman"
}
}
}
}
}
}
以及以下注释类:
class Comment {
var id:String
var text:String
init(id: String, text:String) {
self.id = id
self.text = text
}
}
考虑到您的建议后,这是我的代码:
var comments = [Comment] ()
@IBOutlet weak var commentsTable: UITableView!
@IBOutlet weak var commentPlaceHolder: UILabel!
@IBOutlet weak var newCommentLabel: UITextView!
weak var delegate:NewPostVCDelegate?
let ref = Database.database().reference().child("posts")
@IBAction func reply(_ sender: UIButton) {
let userID = (Auth.auth().currentUser?.uid)!
addComment(toPostId: post!.id, andComment: newCommentLabel.text, commentByUid: userID)
loadComments(forPostId: post!.id)
comments.removeAll()
commentsTable.reloadData()
newCommentLabel.text = String()
commentPlaceHolder.isHidden = false
}
func addComment(toPostId: String, andComment: String, commentByUid: String) {
let commentsRef = self.ref.child("comments") //ref to the comments node
let thisCommentRef = commentsRef.child(toPostId) //ref to a node with postId as key
let commentToAddRef = thisCommentRef.childByAutoId() //each comment will have it's own key
let d = [
"comment_text": andComment,
"comment_by_uid": commentByUid]
commentToAddRef.setValue(d)
}
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
loadComments(forPostId: post!.id)
}
func loadComments(forPostId: String) {
let ref = self.ref.child("comments")
let thisPostRef = ref.child(forPostId)
thisPostRef.observeSingleEvent(of: .value, with: { snapshot in
let allComments = snapshot.children.allObjects as! [DataSnapshot]
for commentSnap in allComments {
let commenterUid = commentSnap.childSnapshot(forPath: "comment_by_uid").value as? String ?? "No uid"
let commentText = commentSnap.childSnapshot(forPath: "comment_text").value as? String ?? "No comment"
let aComment = Comment(id: commenterUid, text: commentText)
self.comments.append(aComment)
print(commenterUid, commentText)
}
self.commentsTable.reloadData()
})
}
func adjustUITextViewHeight(arg : UITextView) {
arg.translatesAutoresizingMaskIntoConstraints = true
arg.sizeToFit()
arg.isScrollEnabled = false
}
override func viewDidLoad() {
super.viewDidLoad()
self.commentsTable.dataSource = self
let cellNib = UINib(nibName: "CommentTableViewCell", bundle: nil)
commentsTable.register(cellNib, forCellReuseIdentifier: "postCell")
view.addSubview(commentsTable)
commentsTable.register(LoadingCell.self, forCellReuseIdentifier: "loadingCell")
self.commentsTable.delegate = self
mainText.isEditable = false
titleText.isEditable = false
commentsTable.register(cellNib, forCellReuseIdentifier: "postCell")
view.addSubview(commentsTable)
commentsTable.register(LoadingCell.self, forCellReuseIdentifier: "loadingCell")
print(delegate!)
}
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return comments.count
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
if indexPath.section == 0 {
let cell = tableView.dequeueReusableCell(withIdentifier: "postCell", for: indexPath) as! CommentTableViewCell
cell.set(comment: comments[indexPath.row])
return cell
} else {
let cell = tableView.dequeueReusableCell(withIdentifier: "loadingCell", for: indexPath) as! LoadingCell
cell.spinner.startAnimating()
return cell
}
}
func textViewDidChange(_ commentView: UITextView) {
commentPlaceHolder.isHidden = !newCommentLabel.text.isEmpty
}
结合Jay的评论,使视图运行起来。我必须添加comment.removAll(),因此它没有在commentsTable中多次打印出注释。但是,func textViewDidChange
不再起作用。我不确定该如何解决。我尝试调用函数没有运气。也许代表变更影响了这一点?
答案 0 :(得分:1)
此答案基于问题中的数据,然后是后续评论。
对数据进行非规范化是NoSQL数据库中的标准做法,但在这种情况下,问题中的结构可能比所需的更为复杂。
这是问题
给出一系列帖子,每个帖子都有评论,您如何加载 每个帖子的评论都将显示在tableView中。
我将从拟议的结构开始进行反向工作
posts
post_0 //created with .childByAutoId
creator_uid: "the uid of whoever created this post"
post_title: "My post about posting"
comments
post_0 //ties back to the key of the post in the posts node
comment_0 //created with .childByAutoId
comment_by_uid: "whoever created this comment"
comment_text: "comment about this post"
comment_1
comment_by_uid: "whoever created this comment"
comment_text: "comment about this post"
此结构将评论从引用的帖子中分离出来。在comment节点内,每个节点的关键字是posts节点中的post_id。这样一来,无需过多开销即可将帖子加载到tableView中,例如,如果您要在detailView中显示评论,请加载特定帖子的所有评论。
请注意,发布节点和注释节点键是使用.childByAutoId()创建的
现在是工作流程。假设用户正在创建新帖子,并且已经输入了该帖子的标题和其他信息。调用它以在Firebase中创建帖子。
func createPost(withTitle: String, andCreatorUid: String) {
let postsRef = self.ref.child("posts")
let thisPost = postsRef.childByAutoId()
let d = [
"post_title": withTitle,
"creator_uid": andCreatorUid
]
thisPost.setValue(d)
}
这是一个棘手的问题-我要做的是让posts节点具有观察者。添加新帖子时,我收到该事件,创建一个包含有关该帖子的信息的PostsClass对象,并将其添加到我的dataSource数组中,然后刷新我的tableView。通过这样做,我还获得了节点的密钥(该节点是使用.childByAutoId创建的)。
另一个用户看到了该帖子并想对此发表评论,因此他们点击该帖子以输入评论。以下代码将其注释存储在Firebase中。
func addComment(toPostId: String, andComment: String, commentByUid: String) {
let commentsRef = self.ref.child("comments") //ref to the comments node
let thisCommentRef = commentsRef.child(toPostId) //ref to a node with postId as key
let commentToAddRef = thisCommentRef.childByAutoId() //each comment will have it's own key
let d = [
"comment_text": andComment,
"comment_by_uid": commentByUid]
commentToAddRef.setValue(d)
}
toPostId是帖子的键,该帖子是从他们选择为其添加注释的PostClass对象获得的。
最后,要专门回答这个问题,这里是针对特定帖子的评论。
func loadComments(forPostId: String) {
let ref = self.ref.child("comments")
let thisPostRef = ref.child(forPostId)
thisPostRef.observeSingleEvent(of: .value, with: { snapshot in
let allComments = snapshot.children.allObjects as! [DataSnapshot]
for commentSnap in allComments {
let commenterUid = commentSnap.childSnapshot(forPath: "comment_by_uid").value as? String ?? "No uid"
let commentText = commentSnap.childSnapshot(forPath: "comment_text").value as? String ?? "No comment"
//create a commentClass object, update properties and add to dataSourceArray
print(commenterUid, commentText)
}
//tableView reload
})
}
注意:
我有一个var类, ref ,所以self.ref指向我的根Firebase节点。您需要将其设置为指向您的
我在此答案中使用了post_0和comment_0作为节点键名,因为它比.emB-AutoLhzJD3tPL0xcnUDMaOZ 之类的键更易于阅读和理解。 / p>
答案 1 :(得分:0)
在Comment(id: childDataSnapshot.key, text: comments)
行中,您正在传递注释(它是数组属性),但是您需要传递文本。要重新加载数据,请使用commentsTable.reloadData()
。
对于cellForRowAt
中的错误,请仔细检查您需要传递的内容以配置单元格,并检查您传递的是set(comment
的类型