Как установить currentIndex в BottomNavigationBar в флаттер из BLOC?
Я хотел бы использовать StatelessWidget с BottomNavigationBar, которым я бы управлял из BLOC. Я могу подключить корпус Scaffold и onTap BottomNavigationBar к BLOC (см. Код). Но я не понимаю, как можно установить currentIndex BottomNavigationBar из BLOC (из Observable).
Есть ли хорошее решение или мне нужно использовать StatefulWidget, как в /questions/42978104/bottomnavigationbar-s-shablonom-bloc/42978106#42978106 который похож на мой пример.
Код BLOC:
class Bloc {
final _currentTabSink = PublishSubject<int>();
final _currentTabIndex = BehaviorSubject<int>();
Observable<int> get currentTabIndex => _currentTabIndex.stream;
Function(int) get setTabIndex => _currentTabSink.sink.add;
Bloc() {
_currentTabSink.stream.pipe(_currentTabIndex);
}
dispose() {
_currentTabSink.close();
_currentTabIndex.close();
}
}
Код виджета:
class MyWidget extends StatelessWidget {
@override
Widget build(BuildContext context) {
final bloc = Provider.of(context);
final List<Widget> _children = [
AaWidget(),
BbWidget(),
CcWidget(),
DdWidget(),
EeWidget()
];
return Scaffold(
appBar: AppBar(
title: Text(Localizations.of(context).appName),
),
body: setBody(_children, bloc), // hook to BLOC
bottomNavigationBar: BottomNavigationBar(
currentIndex: 1, // ?? how to hook up to BLOC ??
onTap: (index) {
bloc.setTabIndex(index); // hook to BLOC
},
items: [
addAa(context),
addBb(context),
addCc(context),
addDd(context),
addEE(context),
],
));
}
Widget setBody(List<Widget> children, Bloc bloc) {
return StreamBuilder(
stream: bloc.currentTabIndex,
builder: (context, AsyncSnapshot<int> snapshot) {
if (snapshot.hasData) {
return children[snapshot.data];
}
},
);
}
...
}
1 ответ
Я думаю, вам нужно обернуть Scaffold
в StreamBuilder
с начальным набором данных, а затем вы можете получить доступ к данным снимка. Примерно так (мне пришлось внести некоторые изменения в ваш код, чтобы он собирался для меня)
class MyWidget extends StatelessWidget {
@override
Widget build(BuildContext context) {
final bloc = Bloc();
final List<Widget> _children = [
Container(color: Colors.blue, child: Center(child: Text("1"))),
Container(color: Colors.red, child: Center(child: Text("2"))),
Container(color: Colors.green, child: Center(child: Text("3"))),
Container(color: Colors.yellow, child: Center(child: Text("4"))),
Container(color: Colors.orange, child: Center(child: Text("5"))),
];
return StreamBuilder<int>(
stream: bloc.currentTabIndex,
initialData: 0,
builder: (BuildContext context, AsyncSnapshot<int> snapshot) {
if (!snapshot.hasData) {
return Container();
}
return Scaffold(
appBar: AppBar(
title: Text("Test"),
),
body: _children[snapshot.data],
bottomNavigationBar: BottomNavigationBar(
currentIndex: snapshot.data,
onTap: (index) {
bloc.setTabIndex(index);
},
items: _children
.map((child) => BottomNavigationBarItem(
icon: Icon(Icons.add),
title: Text("Test"),
backgroundColor: Colors.black))
.toList(),
),
);
},
);
}
}
Это работает для меня, единственным недостатком здесь является то, что вы будете восстанавливать AppBar
без необходимости каждый раз, когда изменяется индекс.