如何在streambuilder中查询firestore文档并更新listview

Posted

技术标签:

【中文标题】如何在streambuilder中查询firestore文档并更新listview【英文标题】:How to query firestore document inside streambuilder and update the listview 【发布时间】: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 集合截图:

Firestore“帖子”集合:

Firestore“用户”集合:

我知道这是一件很简单的事情,但还是找不到任何教程或文章来实现这一点。

【问题讨论】:

无需调用setState,这是在构建方法中。不过,一般来说,您应该远离在构建方法中查询 Firestore。查看this excellent answer,了解如何保持您的构建方法纯净。 在创建构建之前,是否有任何替代方法可以从帖子中查询用户 ID 并显示其详细信息? 为什么不将该查询放入您的initState 方法中? 好吧,我需要从“posts”集合中包含的帖子中获取帖子创建者的 ID,然后使用“users”集合查询该 ID,然后检索用户的姓名、图片或任何其他值。现在我应该在 initStae 方法中调用哪个? 【参考方案1】:

因为我花了几个小时试图弄清楚这一点,所以为将来的人发帖 - 希望它可以拯救其他人。

首先我建议阅读Streams:https://www.dartlang.org/tutorials/language/streams 这将有所帮助,而且是简短的阅读

很自然的想法是在外部StreamBuilder 内部有一个嵌套的StreamBuilder,如果内部StreamBuilder 接收数据导致ListView 的大小不会改变,这很好。您可以在没有数据时创建一个具有固定大小的容器,然后在数据丰富的小部件准备好时呈现它。就我而言,我想为“外部”集合和“内部”集合中的每个文档创建一个Card。例如,我有一个组集合,每个组都有用户。我想要这样的视图:

[
  Group_A header card,
  Group_A's User_1 card,
  Group_A's User_2 card,
  Group_B header card,
  Group_B's User_1 card,
  Group_B's User_2 card,
]

嵌套的StreamBuilder 方法呈现数据,但滚动ListView.builder 是个问题。滚动时,我猜高度计算为 (group_header_card_height + inner_listview_no_data_height)。当内部ListView 接收到数据时,它会扩展列表高度以适应和滚动抖动。这是不可接受的用户体验。

解决方案要点:

应在StreamBuilderbuilder 执行之前获取所有数据。这意味着您的 Stream 需要包含来自两个集合的数据 虽然Stream 可以容纳多个项目,但您需要Stream&lt;List&lt;MyCompoundObject&gt;&gt;。对此答案的评论 (https://***.com/a/53903960/608347) 有帮助

我采取的方法基本上是

    创建组到用户列表对的流

    一个。查询群组

    b.对于每个组,获取适当的 userList

    c。返回包装每一对的自定义对象列表

    StreamBuilder 正常,但在 group-to-userList 对象而不是 QuerySnapshots

它看起来像什么

复合助手对象:

class GroupWithUsers 
  final Group group;
  final List<User> users;

  GroupWithUsers(this.group, this.users);

StreamBuilder

    Stream<List<GroupWithUsers>> stream = Firestore.instance
        .collection(GROUP_COLLECTION_NAME)
        .orderBy('createdAt', descending: true)
        .snapshots()
        .asyncMap((QuerySnapshot groupSnap) => groupsToPairs(groupSnap));

    return StreamBuilder(
        stream: stream,
        builder: (BuildContext c, AsyncSnapshot<List<GroupWithUsers>> snapshot) 
            // build whatever
    );

本质上,“为每个组创建一对”处理所有类型的转换

  Future<List<GroupWithUsers>> groupsToPairs(QuerySnapshot groupSnap) 
    return Future.wait(groupSnap.documents.map((DocumentSnapshot groupDoc) async 
      return await groupToPair(groupDoc);
    ).toList());
  

最后,获取Users 并构建我们的助手的实际内部查询

Future<GroupWithUsers> groupToPair(DocumentSnapshot groupDoc) 
    return Firestore.instance
        .collection(USER_COLLECTION_NAME)
        .where('groupId', isEqualTo: groupDoc.documentID)
        .orderBy('createdAt', descending: false)
        .getDocuments()
        .then((usersSnap) 
      List<User> users = [];
      for (var doc in usersSnap.documents) 
        users.add(User.from(doc));
      

      return GroupWithUser(Group.from(groupDoc), users);
    );
  

【讨论】:

非常感谢兄弟,你救了我的命。这种方法对我来说几乎不需要修改。【参考方案2】:

我发布了一个类似的问题,后来找到了解决方案:将 itemBuilder 返回的小部件设置为有状态并在其中使用 FutureBuilder。

Additional query for every DocumentSnapshot within StreamBuilder

这是我的代码。在您的情况下,您可能希望使用新的 Stateful 小部件而不是 ListTile,因此您可以添加 FutureBuilder 来调用异步函数。

StreamBuilder(
                  stream: Firestore.instance
                      .collection("messages").snapshots(),
                  builder: (context, snapshot) 
                    switch (snapshot.connectionState) 
                      case ConnectionState.none:
                      case ConnectionState.waiting:
                        return Center(
                          child: PlatformProgressIndicator(),
                        );
                      default:
                        return ListView.builder(
                          reverse: true,
                          itemCount: snapshot.data.documents.length,
                          itemBuilder: (context, index) 
                            List rev = snapshot.data.documents.reversed.toList();
                            ChatMessageModel message = ChatMessageModel.fromSnapshot(rev[index]);
                            return ChatMessage(message);
                          ,
                        );
                    
                  ,
                )


class ChatMessage extends StatefulWidget 
  final ChatMessageModel _message;
  ChatMessage(this._message);
  @override
  _ChatMessageState createState() => _ChatMessageState(_message);


class _ChatMessageState extends State<ChatMessage> 
  final ChatMessageModel _message;

  _ChatMessageState(this._message);

  Future<ChatMessageModel> _load() async 
    await _message.loadUser();
    return _message;
  

  @override
  Widget build(BuildContext context) 

    return Container(
      margin: const EdgeInsets.symmetric(horizontal: 10.0, vertical: 10.0),
      child: FutureBuilder(
        future: _load(),
        builder: (context, AsyncSnapshot<ChatMessageModel>message) 
          if (!message.hasData)
            return Container();
          return Row(
            children: <Widget>[
              Container(
                margin: const EdgeInsets.only(right: 16.0),
                child: GestureDetector(
                  child: CircleAvatar(
                    backgroundImage: NetworkImage(message.data.user.pictureUrl),
                  ),
                  onTap: () 
                    Navigator.of(context)
                        .push(MaterialPageRoute(builder: (context) => 
                        ProfileScreen(message.data.user)));
                  ,
                ),
              ),
              Expanded(
                child: Column(
                  crossAxisAlignment: CrossAxisAlignment.start,
                  children: <Widget>[
                    Text(
                      message.data.user.name,
                      style: Theme.of(context).textTheme.subhead,
                    ),
                    Container(
                        margin: const EdgeInsets.only(top: 5.0),
                        child: _message.mediaUrl != null
                            ? Image.network(_message.mediaUrl, width: 250.0)
                            : Text(_message.text))
                  ],
                ),
              )
            ],
          );
        ,
      ),
    );
  

class ChatMessageModel 
  String id;
  String userId;
  String text;
  String mediaUrl;
  int createdAt;
  String replyId;
  UserModel user;

  ChatMessageModel(String text, String mediaUrl, String userId) 
    this.text = text;
    this.mediaUrl = mediaUrl;
    this.userId = userId;
  

  ChatMessageModel.fromSnapshot(DocumentSnapshot snapshot) 
    this.id = snapshot.documentID;
    this.text = snapshot.data["text"];
    this.mediaUrl = snapshot.data["mediaUrl"];
    this.createdAt = snapshot.data["createdAt"];
    this.replyId = snapshot.data["replyId"];
    this.userId = snapshot.data["userId"];
  

  Map toMap() 
    Map<String, dynamic> map = 
      "text": this.text,
      "mediaUrl": this.mediaUrl,
      "userId": this.userId,
      "createdAt": this.createdAt
    ;
    return map;

  

  Future<void> loadUser() async 
    DocumentSnapshot ds = await Firestore.instance
        .collection("users").document(this.userId).get();
    if (ds != null)
      this.user = UserModel.fromSnapshot(ds);
  


【讨论】:

嘿,感谢您的回复,但是,我更喜欢 StreamBuilder,因为 FutureBuilder 只被调用一次,而且我希望每次 Firestore 发生变化时更新我的​​帖子。 @sanjay 你仍然会使用你的主 StreamBuilder。 FutureBuilder 将用于获取您需要的其他信息,并且会在 StreamBuilder 检测到 Firestore 中的更改时触发。 也许可以,但我希望能够从一个集合(帖子)文档的值中执行查询,以从另一个名为 users 的集合中获取文档 ID。你的 ChatMessageModel 有没有类似的操作? @sanjay 这就是我正在做的事情。 StreamBuilder 监听新消息,当有新消息进入时,会创建一个新的 ChatMessage 小部件,其 FutureBuilder 会为消息的所有者加载用户数据。 你能把 ChatMessageModel 的代码也贴出来,让我明白发生了什么吗?

以上是关于如何在streambuilder中查询firestore文档并更新listview的主要内容,如果未能解决你的问题,请参考以下文章

StreamBuilder 的 Firebase 查询出现并立即消失

无法从 StreamBuilder (Flutter) 查询子集合

使用 StreamBuilder 和从 Firebase Firestore 数据库查询时,ListView 不会更新/获取数据

Flutter - 使用 StreamBuilder 从 firebase 获取数据后,如何使用 ScrollController 跳转到列表视图的底部?

在 Streambuilder 中的集合上使用 where() 不断返回 null

如何在颤动中直接将数据从 StreamBuilder 获取到 GridView 中?