如何使用 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 检索数据?