- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我努力达到的结果
我将 Firestore 服务器用作餐厅订单系统的 Flutter 应用程序的后端。在实现显示数据库实时更改的屏幕时,我发现自己遇到了一些挑战。
所以我有一个名为“表格”的集合,其中包含多个文档(不同的餐厅可能有不同的名称)。这些文档中的每一个都有几个字段,例如'登记时间'。此外,每个表文档还有一个订单集合(“子集合”)。
我想要一个Stream
在我的 Flutter 屏幕上,它接收所有表格,包括他们的订单。屏幕应该给出所有表格的概览,这意味着它应该在任何时候更新...
我有一个模型 Order
和 Table
类(class)。 Table 类的一个实例应该有一个 List<Order> _orders
其中包含存储在子集合中的所有订单。 Stream
应该给我提供 List<Table>
.
注意:我尝试使用 Flutter Web 来实现这一点,但这可能不会对我的问题的解决产生影响。
当前进度
到目前为止,我实际上几乎成功地解决了这个挑战。我设法得到一个 Stream
将所有表格显示在屏幕上。
在大多数情况下,它最初会成功加载数据。有时屏幕的第一次加载会导致无限加载 CircularProgressIndicator 而数据未加载。
当另一个表被添加到集合中时,屏幕会更新。添加订单不会立即起作用,但如果我第二次重新加载页面或在我将另一个订单添加到订单子(monad)集合后它会起作用。似乎错误的发生与时间无关。
似乎更新订单文档根本不起作用/重新加载订单状态。
在任何情况下都不会出现错误信息。
当前代码
类 TableInfo
对于流:
import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:flutter/material.dart' as m;
import '../../../../../domain/infrastructure/firestore_infrastructure.dart';
import '../../../../../domain/models/orders_models/order.dart';
import '../../../../../domain/models/orders_models/table.dart';
import '../../../../../locator.dart';
///Class for getting all tables and orders in realtime and managing the current filter settings.
class TableInfo extends m.ChangeNotifier {
///List of all tables
List<Table> _tables = [];
///List of all orders
List<Order> _orders = [];
final CollectionReference _tableCollection = locator<CustomFirestore>().tablesRef;
Stream<QuerySnapshot> get _stream => _tableCollection.snapshots();
///returns all tables in the Firestore `tables` collection of the restaurant.
///Also sets a listener for the orders and matches them from the [_orders] list.
Stream<List<Table>> getTableStream(
{void Function(Order savedOrder) onOrderSaved}) async* {
print("Reloading");
final tableDocs = await _tableCollection.get().then((value) => value.docs);
for (QueryDocumentSnapshot tableDoc in tableDocs) {
_tableCollection
.doc(tableDoc.id)
.collection(CustomFirestore.ORDERS_COLLECTION)
.snapshots()
.listen((snapshot) =>
_saveOrders(snapshot, tableDoc.id, onOrderSaved: onOrderSaved));
}
final tableStream =
_stream.map((snapShot) => _unfilteredTablesFromTableSnapshot(snapShot));
yield* tableStream;
}
///Help function to get all tables with ordes from a table doc snapshot.
List<Table> _unfilteredTablesFromTableSnapshot(QuerySnapshot snapShot) {
return snapShot.docs.map((tableDocument) {
final table = Table.fromSnapshot(tableDocument);
table.addOrders(_orders.where((element) => element.tableId == table.id),
overwrite: true);
_tables.add(table);
return table;
}).toList();
}
///Function to save the orders from a snapshot of an order collection of a single table.
void _saveOrders(QuerySnapshot orderCollectionSnapshot, String tableId,
{void Function(Order savedOrder) onOrderSaved}) {
final orderDocs = orderCollectionSnapshot.docs;
print(
"Saving ${orderDocs.length} orders for $tableId at ${DateTime.now()} :)");
for (QueryDocumentSnapshot orderDoc in orderDocs) {
final order = Order.fromSnapshot(orderDoc, tableId);
final existing = _orders.firstWhere(
(element) => element.orderId == order.orderId,
orElse: () => null);
if (existing != null) _orders.remove(existing);
_orders.add(order);
if (onOrderSaved != null) onOrderSaved(order);
}
}
}
如您所见,我首先获取所有表文档,然后为子集合“订单”添加一个监听器,称为 _saveOrders
.看起来代码首先获得订单,这就是我在 Table.fromSnapshot
中初始化表的原因使用 _orders = [] 然后从 _unfilteredTablesFromTableSnapshot
内的订单列表中添加订单方法。
这里是屏幕的相关部分build
方法(显然放在脚手架中):
final tableInfo = Provider.of<TableInfo>(context);
return StreamProvider.value(
value: tableInfo.getTableStream(),
catchError: (ctx, error) {
print("Error occured! $error");
print(error.runtimeType);
//throw error;
return [
t.Table(
id: 'fehler',
name: 'Fehler $error',
status: t.TableStatus.Free,
orders: [],
checkinTime: DateTime.now())
];
},
builder: (context, child) {
final allUnfilteredTables = Provider.of<
List<t.Table>>(
context); //corresponds to TableInfo.filteredTableStream
///while the Tables are loading:
if (allUnfilteredTables == null)
return Center(
child: CircularProgressIndicator());
///if there are no tables at all:
if (allUnfilteredTables.isEmpty)
return Text(
"No tables detected.");
///if an error occured:
if (allUnfilteredTables[0].id == 'fehler')
return Text(allUnfilteredTables[0].name);
return GridView.builder(
gridDelegate:
SliverGridDelegateWithFixedCrossAxisCount(
crossAxisSpacing: (50 / 1920) *
constraints.maxWidth,
crossAxisCount: isTablet ? 3 : 6),
itemCount: allUnfilteredTables.length,
itemBuilder: (ctx, index) {
t.Table table = allUnfilteredTables[index];
return TableMiniDisplay(table);
},
);
});
我已经尝试过的
我已经阅读了很多有关相关问题的文章,但找不到任何对我有很大帮助的答案来解决我剩下的错误。我也尝试使用 rxdart并使用 CombinedStream 但无法运行它。
实时同步的主要部分是有效的,所以我想只有很少的东西能阻止我成功,我感谢大家花时间阅读我的问题。我很感激任何可以帮助我的想法或代码示例。
(此外,如果您有任何其他改进代码的建议,请随时发表评论:))
提前致谢!
干杯,大卫
最佳答案
我建议您使用两个 Streambuilder 包装您的小部件。只需使用第一个来流式传输表,使用第二个来访问每个表中的订单。不要在每次要访问数据时都添加 Streambuilder。仅使用两个流生成数据并通过变量传递。
这是示例代码:
database.dart
import 'package:cloud_firestore/cloud_firestore.dart';
class Tables {
CollectionReference tablesReference =
FirebaseFirestore.instance.collection("tables");
Stream<QuerySnapshot> getTables() {
// Returns all tables
return tablesReference.snapshots();
}
Stream<QuerySnapshot> getOrdersFromTables(String tableName) {
// Returns specific table orders
return tablesReference.doc(tableName).collection("orders").snapshots();
}
}
main.dart
import 'package:firebase_core/firebase_core.dart';
import 'package:flutter/material.dart';
import 'tables.dart';
void main() {
runApp(MyApp());
}
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
debugShowCheckedModeBanner: false,
theme: ThemeData.dark(),
// Initialize FlutterFire:
home: FutureBuilder(
future: Firebase.initializeApp(),
builder: (context, snapshot) {
// Check for errors
if (snapshot.hasError) {
return Center(
child: Text("Error"),
);
}
// Once complete, show your application
if (snapshot.connectionState == ConnectionState.done) {
return TablesPage();
}
// Otherwise, show something whilst waiting for initialization to complete
return Center(
child: CircularProgressIndicator(),
);
}),
);
}
}
表格.dart
import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:flutter/material.dart';
import 'package:intl/intl.dart';
import 'orders.dart';
import 'database.dart';
class TablesPage extends StatefulWidget {
@override
_TablesPageState createState() => _TablesPageState();
}
class _TablesPageState extends State<TablesPage> {
@override
Widget build(BuildContext context) {
// Get Tables
return Scaffold(
appBar: AppBar(
title: Text("Tables"),
),
body: StreamBuilder<QuerySnapshot>(
stream: Tables().getTables(),
builder: (context, tables) {
if (tables.hasError)
return Text(tables.error);
else if (tables.hasData) {
if (tables.data.docs.isEmpty) {
return Text("No tables.");
} else {
return ListView.builder(
itemCount: tables.data.docs.length,
itemBuilder: (context, index) {
// Get Orders
return StreamBuilder<QuerySnapshot>(
stream: Tables()
.getOrdersFromTables(tables.data.docs[index].id),
builder: (context, orders) {
if (orders.hasData) {
return ListTile(
title:
Text("Table ${tables.data.docs[index].id}"),
subtitle: Text(DateFormat("HH:mm").format(tables
.data.docs[index]["checkinTime"]
.toDate())),
trailing: Text(
"Total orders: ${orders.data.docs.length.toString()}"),
onTap: () {
// Navigate to Orders Page
Navigator.push(
context,
MaterialPageRoute(
builder: (context) => Orders(
orders: orders.data.docs,
),
),
);
});
} else {
return Container();
}
});
},
);
}
} else {
print(tables
.connectionState); //is stuck in ConnectionState.waiting or ConnectionState.active
return Center(child: CircularProgressIndicator());
}
},
),
);
}
}
订单.dart
import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:flutter/material.dart';
class Orders extends StatelessWidget {
const Orders({
Key key,
@required this.orders,
}) : super(key: key);
final List<QueryDocumentSnapshot> orders;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text("Orders"),
),
body: ListView.builder(
itemCount: orders.length,
itemBuilder: (context, index) {
return ListTile(
title: Text(orders[index].id),
subtitle: Text(orders[index]["barStatus"]),
leading: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Text((index+1).toString()),
],
),
);
},
),
);
}
}
关于firebase - 在 Flutter 中收听 Firestore 集合及其子集合,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65044932/
我刚刚设置了新的 Firebase 托管以及我的自定义域和我的网页等... 一切都很好,除了真正困扰我的是默认域: projectname.web.app projectname.firebaseap
Firebase 云消息传递和 Firebase 应用内消息传递有什么区别?它们都会向您的应用发送发生了某些事情的通知。我查看了他们的文档,这些是他们的单行描述: Firebase 云消息传递: Fi
是否有任何工具可以帮助将数据从开发迁移到登台再到生产?如果没有,是否有计划 build 它们? 我知道您可以从 Forge 导出 JSON 和导入 JSON,但这不包括授权和安全设置。 最佳答案 我们
firebase 网络应用和 firebase 托管有什么区别?据我所知,网络应用程序用于实际上只是浏览器的应用程序,并且 firebase 托管仅用于网站。 最佳答案 Firebase for We
我有两个 firebase 项目 F1 和 F2。我正在使用基于密码的身份验证来识别 F1 上的用户。我希望在 F2 中识别相同的用户。所以我正在考虑以下基于 this question 的场景: C
正在使用 firebase 云功能 构建服务器,我需要其他服务器来调用我的服务器端点。最重要的是,我需要这些调用我的服务器的第三方域进行身份验证。 问题 1:如何创建这样的服务帐户,在调用中包含服务帐
我开发了一个应用程序来测试使用 flutter 和谷歌身份验证的谷歌登录功能。该项目是一个封闭的项目,只有我可以访问它。但最近我看到有一个来自未知电子邮件 ID 的谷歌登录。用户如何在没有构建我的应用
我有 2 个存储库,一个用于 firebase 功能,另一个用于静态 firebase 托管 react 站点。他们都使用相同的firebase项目。 myfirebaseproject --- fi
背景 -我正在设置一项新功能,允许用户选择他们所在的城市,因为我的应用程序是一个公共(public)交通应用程序。我希望城市位于单独的数据库中,为此我在我的 firebase 项目中创建了一个辅助数据
我构建了一个网络平台,允许用户使用一个用户帐户注册多个网络应用程序。每个应用程序都有自己独立的 Firebase 数据库。是否可以使用 firebase 身份验证并在各种 firebase 数据库之间
在 Firebase 控制台内的 Cloud Messaging View 下,用户可以创建测试通知。此功能还允许您安排通知将发送到一个设备或一组设备的时间。 是否可以创建和发送预定 使用 Fireb
我们正在为我们最近的项目使用 firebase 平台。我们最近推出并一直在监控使用情况。 我可以在 firebase 控制台中看到 firebase 调用的总数,但找不到查看每个函数调用次数的方法。
我正在开发一个使用 Webpack 捆绑模块的 Web 应用程序。无论出于何种原因,在应用程序中引入 Firebase 都会导致 Webpack 抛出错误。当 Webpack 尝试加载 Firebas
我无法在 firebase 控制台中提交支持请求。 登录 Firebase 控制台 导航至 https://firebase.google.com/support/contact/troublesho
我安装 Firebase CLI (firebase-tools) https://github.com/firebase/firebase-tools通过此代码: curl -sL firebase
在我的应用程序中,我尝试使用Firebase crash10.2.1,但无法获得编译的好处。我不断收到消息: 'Failed to resolve:com.google.firebase:fireba
我的 firebase 应用有一个注册用户列表。这些是通过电子邮件和密码身份验证创建的。 我想将 Firebase 数据和用户列表传输到另一个 Firebase 应用。 传输 firebase 数据很
如果我将数据插入到 firebase 中的 node/a/0 中。 结果将把a视为数组a[0]。 同样,如果我在 node/a/1 中设置数据,第一个数组将变为 null "a" : [
我是 Firebase 新手,我正在尝试 Firebase 数据库规则和 Firebase 身份验证。 我尝试使用 firebase 身份验证来执行 firebase 数据库规则。 因此,我创建了一封
所以场景如下: 当我从 Firebase 存储访问文件时: 我从存储桶获取文件(.html、.png、.zip 等)(尺寸较小,不超过 2mb)。 将该文件存储在我的本地存储中,这样应用就不需要再次下
我是一名优秀的程序员,十分优秀!