【发布时间】:2019-03-09 06:11:54
【问题描述】:
我正在尝试从名为“posts”的 Firestore 集合中检索帖子,其中包含 帖子创建者的用户 ID 和帖子描述,这可以通过同时使用 StreamBuilder 和 FutureBuilder(不推荐,因为它只获取一次快照,并且在字段更改时不会更新)。
但是,我想用 帖子创建者的用户 ID 查询另一个名为“用户”的集合,并检索与用户 ID 匹配的文档。
这是我的第一个方法:
StreamBuilder<QuerySnapshot>(
stream:Firestore.instance.collection("posts").snapshots(),
builder: (BuildContext context, AsyncSnapshot<QuerySnapshot> snapshot) {
if (!snapshot.hasData) {
return Center(
child: _showProgressBar,
);
}
List<DocumentSnapshot> reversedDocuments = snapshot.data.documents.reversed.toList();
return ListView.builder(
itemCount: reversedDocuments.length,
itemBuilder: (BuildContext context, int index){
String postAuthorID = reversedDocuments[index].data["postAuthorID"].toString();
String postAuthorName = '';
Firestore.instance.collection("users")
.where("userID", isEqualTo: postAuthorID).snapshots().listen((dataSnapshot) {
print(postAuthorName = dataSnapshot.documents[0].data["username"]);
setState(() {
postAuthorName = dataSnapshot.documents[0].data["username"];
});
});
String desc = reversedDocuments[index].data["post_desc"].toString();
return new ListTile(
title: Container(
child: Row(
children: <Widget>[
Expanded(
child: Card(
child: new Column(
children: <Widget>[
ListTile(
title: Text(postAuthorName, //Here, the value is not changed, it holds empty space.
style: TextStyle(
fontSize: 20.0,
),
),
subtitle: Text(desc),
),
)
了解ListView.builder()只能根据DocumentSnapshot列表渲染item,无法处理builder内部的查询。
经过多次研究: 我尝试了许多替代方法,例如尝试在 initState() 中构建列表,尝试使用 Nested Stream Builder:
return StreamBuilder<QuerySnapshot>(
stream: Firestore.instance.collection('posts').snapshots(),
builder: (context, snapshot1){
return StreamBuilder<QuerySnapshot>(
stream: Firestore.instance.collection("users").snapshots(),
builder: (context, snapshot2){
return ListView.builder(
itemCount: snapshot1.data.documents.length,
itemBuilder: (context, index){
String desc = snapshot1.data.documents[index].data['post_description'].toString();
String taskAuthorID = snapshot1.data.documents[index].data['post_authorID'].toString();
var usersMap = snapshot2.data.documents.asMap();
String authorName;
username.forEach((len, snap){
print("Position: $len, Data: ${snap.data["username"]}");
if(snap.documentID == post_AuthorID){
authorName = snap.data["username"].toString();
}
});
return ListTile(
title: Text(desc),
subtitle: Text(authorName), //Crashes here...
);
},
);
}
);
}
);
尝试使用 Stream Group 并无法找到完成此操作的方法,因为它只是组合了两个流,但我希望通过第一个流中的值获取第二个流。
这是我的 Firebase 集合截图:
我知道这是一件很简单的事情,但还是找不到任何教程或文章来实现这一点。
【问题讨论】:
-
无需调用
setState,这是在构建方法中。不过,一般来说,您应该远离在构建方法中查询 Firestore。查看this excellent answer,了解如何保持您的构建方法纯净。 -
在创建构建之前,是否有任何替代方法可以从帖子中查询用户 ID 并显示其详细信息?
-
为什么不将该查询放入您的
initState方法中? -
好吧,我需要从“posts”集合中包含的帖子中获取帖子创建者的 ID,然后使用“users”集合查询该 ID,然后检索用户的姓名、图片或任何其他值。现在我应该在 initStae 方法中调用哪个?
标签: listview dart flutter google-cloud-firestore