如何在Dart的流构建器中切换正在收听的流?

时间:2019-09-15 06:25:06

标签: firebase flutter dart stream google-cloud-firestore

基本上,当页面第一次加载时,我想使用1个通过地理位置查询的流,但是要检查变量socialVisible为true的文档,然后在用户单击按钮进行切换时,我希望该流仍然通过地理位置,但现在使用字段profVisible检查文档。

从理论上讲,这非常简单,每当按下相应的按钮时,我都会设置状态,使socialPressed = true和profPressed = false,然后在streambuilder中,我使用三元运算符根据真值确定应使用哪个流of socialPressed和profPressed。

但是,这仅在第一次单击时有效,socialPressed的初始值为true,因此初始流为社交流,然后当我单击以转到教授时,它将成功切换流。但是在这段时间之后,当我尝试返回社交网站时,教授教授的所有文档都保留了,而不是被社交网站所取代?即使我将初始流切换为教授,也会发生这种情况,而且我不明白为什么会发生这种情况?

StreamBuilder(
            stream:socialPressed==true?socStream:proStream,
            builder:
                (context, AsyncSnapshot<List<DocumentSnapshot>> snapshots) {
              if (!snapshots.hasData) {
                return Container(
                    alignment: FractionalOffset.center,
                    child: CircularProgressIndicator());
              } else {
               return Container(
                    child: Container(
                      child: (snapshots.data.length == 0)
                          Container(
                        height: MediaQuery.of(context).size.height * 2 / 3,
                        child: ListView.builder(
                          itemBuilder: (context, index) {
                            DocumentSnapshot doc = snapshots.data[index];
                          return UserTile('this is where i use the doc data')},
                          itemCount: snapshots.data.length,
                        ),),));}},)

这是我代码的基本结构,我删除了不相关的信息。

编辑: 请求了按钮回调代码:

 FloatingActionButton(
                shape: RoundedRectangleBorder(
                    borderRadius: BorderRadius.all(Radius.circular(16.0))),
                onPressed: () {
                  setState(() {
                    likeType='social';
                    socialPressed=!socialPressed;
                  });
                },
                elevation: 0,
                heroTag: 'socialButton',
                backgroundColor: socialPressed==false?Colors.grey[100]:Color(0xFFe0bdff),
                child: Icon(
                  Entypo.drink,
                  color: Color(0xFF8803fc),
                ),
              ),
              FloatingActionButton(
                shape: RoundedRectangleBorder(
                    borderRadius: BorderRadius.all(Radius.circular(16.0))),
                onPressed: () {
                  setState(() {
                    likeType='prof';
                    socialPressed=!socialPressed;
                  });
                },
                elevation: 0,
                heroTag: 'profButton',
                backgroundColor: socialPressed==true?Colors.grey[100]:Color(0xFFb9ebe9),
                child: Row(
                  children: <Widget>[
                    SizedBox(
                      width: MediaQuery.of(context).size.width / 30,
                    ),
                    Icon(
                      FontAwesome.graduation_cap,
                      color: Color(0xFF096664),
                    ),
                  ],

                ),
              ),

2 个答案:

答案 0 :(得分:0)

您使用什么逻辑来设置状态?似乎您正在使用两个变量socialPressedprofPressed。如果只有两种状态,我会考虑仅使用一个变量。

没有您的代码,我们只能推测出什么地方出了问题。但这应该在按钮的onPressed回调中起作用:

onPressed: () => setState(() { socialPressed = !socialPressed; })

答案 1 :(得分:0)

我也面临这个问题。我正在使用RX Dart,并尝试根据布尔变量切换流。

我最初是在流中使用PublishSubject。切换到“行为主体”已为我解决了此问题。目前,我尚不足以了解这种方法的工作原理,但会尝试对其进行进一步探索,并在可能的情况下进行编辑。

先前的代码:

final _activeOrdersFetcher = PublishSubject<OrderListModel>();
final _completedOrdersFetcher = PublishSubject<OrderListModel>();

StreamBuilder<OrderListModel>(
      stream: isActive
          ? _ordersBloc.getActiveOrders
          : _ordersBloc.getCompletedOrders

新代码更改:

final _activeOrdersFetcher = BehaviorSubject<OrderListModel>();
final _completedOrdersFetcher = BehaviorSubject<OrderListModel>();