如何使用 Flutter 从 Firestore 中的 ListView 获得无限滚动

Posted

技术标签:

【中文标题】如何使用 Flutter 从 Firestore 中的 ListView 获得无限滚动【英文标题】:How to get an infinite scroll with ListView from Firestore with Flutter 【发布时间】:2019-06-25 02:53:26 【问题描述】:

我正在 Flutter 中使用 Firestore 和 ListView。 对于列表中的某些项目,每个项目都可以正常工作,但我向下滚动的距离超出了看到的限制,我收到了很多消息:“该方法被调用为 null”。 似乎 ListView.builder 没有正确处理来自 Firebase 的所有日期请求或类似的东西。

这是代码:

import 'package:flutter/material.dart';
import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:father_home_flutter/screen_audio_selected.dart';
import 'package:father_home_flutter/model/utils.dart';

class AudioscreenList extends StatelessWidget 
  static const String _collectionRussian = 'speech-ru';
  static const String _loadingTextRussian = 'Loading...';
  static const String _speechTitle = 'speechTitle';
  static const String _speechSubtitle = 'speechSubtitle';
  static const String _audioLink = 'audioLink';
  static const String _pastorCode = 'pastorCode';
  static const String _pastorName = 'pastorName';
  static const String _id = "id";

  @override
  Widget build(BuildContext context) 
    return Scaffold(
        body: StreamBuilder(
            stream: Firestore.instance
                .collection(_collectionRussian)
                .limit(100)
                .orderBy(_id, descending: true)
                .snapshots(),
            builder: (context, snapshot) 
              if (!snapshot.hasData)
                return const Center(
                    child: Text(
                  _loadingTextRussian,
                  style: TextStyle(fontSize: 25.0, color: Colors.grey),
                ));
              return ListView.builder(
                itemCount: snapshot.data.documents.length,
                itemBuilder: (BuildContext context, int i) =>
                    _buildRow(context, snapshot.data.documents[i]),
              );
            ));
  

  Widget _buildRow(BuildContext context, DocumentSnapshot document) 
    return Container(
        margin: const EdgeInsets.fromLTRB(0.0, 8.0, 0.0, 0.0),
        height: 90.0,
        child: ListTile(
            leading: Padding(
                padding: EdgeInsets.fromLTRB(0.0, 20.0, 0.0, 0.0),
                child: Hero(
                    tag: document[_audioLink],
                    child: new ClipOval(
                        child: Container(
                      width: 70.0,
                      height: 70.0,
                      child: Image.asset(
                        Utils.getImagePastor(document[_pastorCode]),
                        fit: BoxFit.cover,
                      ),
                    )))),
            title: Text(document[_speechTitle]),
            subtitle:
                Text(document[_speechSubtitle] + " - " + document[_pastorName]),
            onTap: () => onPressed(context, document[_speechTitle],
                document[_pastorCode], document[_audioLink])));
  

  onPressed(BuildContext context, String speechTitle, String pastorCode,
      String audioLink) 
    Navigator.push(
        context,
        MaterialPageRoute(
            builder: (context) =>
                ScreenAudioSelected(speechTitle, pastorCode, audioLink)));
  

这是模拟器上的问题:

我在网上寻找如何处理它的方法,但我只是找到了模拟服务器请求的示例,例如这个示例 https://flutter-academy.com/flutter-listview-infinite-scrolling/

有人遇到过同样的问题或知道如何解决吗?

【问题讨论】:

【参考方案1】:

好的,在网上查找后,我找到了解决此问题的方法,感谢link

如果有人需要,我在这里写完整的代码:

    class AudioScreenList extends StatefulWidget 
  @override
  _AudioScreenListState createState() => _AudioScreenListState();


class _AudioScreenListState extends State<AudioScreenList> 
  bool isPerformingRequest = false;

  static const String _collectionRussian = 'speech-ru';
  static const String _loadingTextRussian = 'Loading...';
  static const String _speechTitle = 'speechTitle';
  static const String _speechSubtitle = 'speechSubtitle';
  static const String _audioLink = 'audioLink';
  static const String _pastorCode = 'pastorCode';
  static const String _pastorName = 'pastorName';
  static const String _id = "id";

  @override
  Widget build(BuildContext context) 
    return Scaffold(
        body: StreamBuilder(
            stream: Firestore.instance
                .collection(_collectionRussian)
                .orderBy(_id, descending: true)
                .snapshots(),
            builder: (context, AsyncSnapshot<QuerySnapshot> snapshot) 
              if (!snapshot.hasData)
                return const Center(
                    child: Text(
                  _loadingTextRussian,
                  style: TextStyle(fontSize: 25.0, color: Colors.grey),
                ));
              return ListView(children: getExpenseItems(snapshot));
            ));
  

  getExpenseItems(AsyncSnapshot<QuerySnapshot> snapshot) 
    return snapshot.data.documents
        .map((doc) => new Container(
            margin: const EdgeInsets.fromLTRB(0.0, 8.0, 0.0, 0.0),
            child: ListTile(
                leading: Padding(
                    padding: EdgeInsets.fromLTRB(0.0, 20.0, 0.0, 0.0),
                    child: Hero(
                        tag: doc[_audioLink],
                        child: new ClipOval(
                            child: Container(
                          width: 70.0,
                          height: 70.0,
                          child: Image.asset(
                            Utils.getImagePastor(doc[_pastorCode]),
                            fit: BoxFit.cover,
                          ),
                        )))),
                title: new Text(doc[_speechTitle]),
                subtitle: new Text(doc[_speechSubtitle].toString() +
                    " - " +
                    doc[_pastorName].toString()),
                onTap: () => onPressed(context, doc[_speechTitle],
                    doc[_pastorCode], doc[_audioLink]))))
        .toList();
  

  onPressed(BuildContext context, String speechTitle, String pastorCode,
      String audioLink) 
    Navigator.push(
        context,
        MaterialPageRoute(
            builder: (context) =>
                ScreenAudioSelected(speechTitle, pastorCode, audioLink)));
  

【讨论】:

以上是关于如何使用 Flutter 从 Firestore 中的 ListView 获得无限滚动的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 graphql 从 Firebase 使用 Flutter 从 Cloud Firestore 获取数据?

如何使用flutter从firestore中删除/更新数组对象中的指定索引

如何在flutter中将数据从firestore传递给Typeahead

如何使用 Flutter 中的 UserID 从 Firebase Firestore 检索数据?

Flutter - 如何从 Firestore 返回当前用户

如何从flutter中的firestore文档字段中获取数据?