带有 StreamBuilder 错误的 Flutter AlertDialog
Flutter AlertDialog with StreamBuilder Error
当我尝试在单击图标时弹出的 AlertDialog 框中传输数据时出现错误。我知道这个问题涉及在不允许的情况下发生 streambuilder 构建,但我还没有找到解决此问题的方法。我的目标是能够单击一个图标并出现一个弹出框,显示一些流数据或带有取消按钮和清除数据的按钮的最新实例。
enum WarningInfo { CANCEL, CLEAR }
class WarningStatus extends StatefulWidget {
final ROSBridgeClient client;
WarningStatus({Key key, this.client}) : super(key: key);
@override
State<StatefulWidget> createState() => WarningStatusState();
}
class WarningStatusState extends State<WarningStatus> {
static const String _FAULT_TOPIC = '/active_faults';
//color (gray) when icon has not been enabled
Color _iconColor = Color(0xff4e4e4e);
@override
void initState() {
super.initState();
}
Future<WarningInfo> _asyncWarningBox(BuildContext context) async {
return showDialog<WarningInfo>(
context: context,
barrierDismissible:
true, // user must tap button for close dialog if false!
builder: (BuildContext context) {
return AlertDialog(
// content: StatefulBuilder(),
title: Text('Robot Warnings'),
// content: const Text('No warnings'),
content: StatefulBuilder(
builder: (BuildContext context, StateSetter setState) {
return StreamBuilder(
stream: widget.client.getTopicStream(_FAULT_TOPIC),
builder: (context, snapshot) {
if (!snapshot.hasData) {
return Text('None');
}
return _buildFaultOutput(snapshot);
});
}),
actions: <Widget>[
FlatButton(
child: const Text('CANCEL'),
onPressed: () {
Navigator.of(context).pop(WarningInfo.CANCEL);
},
),
FlatButton(
child: const Text(
'CLEAR WARNINGS',
style: TextStyle(color: Colors.red),
),
onPressed: () {
// Navigator.of(context).pop(WarningInfo.CLEAR);
_clearFaults();
},
)
],
);
},
);
}
@override
Widget build(BuildContext context) {
widget.client.subscribe(_FAULT_TOPIC, 'rover_safety/FaultListMsg');
return SizedBox(
child: Column(
mainAxisSize: MainAxisSize.min,
children: <Widget>[
IconButton(
icon: Icon(
CustomIcons.attention,
color: _iconColor,
),
disabledColor: Color(0xff4e4e4e),
iconSize: 80,
tooltip: "Warnings",
onPressed: () async {
final WarningInfo action = await _asyncWarningBox(context);
print("Confirm Action $action");
},
// onPressed: () {
// setState(() {
// _iconColor = Colors.yellow;
// });
// },
),
// Text("Fault Messages:"),
// SizedBox(
// height: 10.0,
// ),
// StreamBuilder(
// stream: widget.client.getTopicStream(_FAULT_TOPIC),
// builder: (context, snapshot) {
// if (!snapshot.hasData) {
// return CircularProgressIndicator();
// }
// return _buildFaultOutput(snapshot);
// },
// ),
// SizedBox(
// height: 10.0,
// ),
// ButtonBar(
// mainAxisSize: MainAxisSize.min,
// children: <Widget>[
// RaisedButton(
// child: const Text("Clear Faults"),
// onPressed: _clearFaults,
// ),
// ],
// )
],
),
);
}
Widget _buildFaultOutput(snapshot) {
setState(() {
_iconColor = Colors.blue;
});
var msg = snapshot.data;
// print(payload);
var faultList = msg["fault_list"];
print("Number of faults: ${faultList.length}");
// if (faultListLength>0){
// var faults = faultList[0];
// var motorCurrent = faults["fault"];
// var severity = faults["severity"];
// print(faults);
// return Text("Number of Faults: $faultListLength\n$motorCurrent\nseverity: $severity");
// }
// else{
// return Text("No faults");
// }
if (faultList.length == 0) {
return Text("No faults");
} else {
for (var i = 0; i < faultList.length; i++) {
var faults = faultList[i];
var motorCurrent = faults["fault"];
var severity = faults["severity"];
print(faults);
print(i);
return Text(
"Number of Faults: ${faultList.length}\n$motorCurrent\nseverity: $severity");
}
}
}
void _clearFaults() {
widget.client.publish('/clear_faults', 'std_msgs/Empty');
}
@override
void dispose() {
super.dispose();
}
}
错误:
I/flutter ( 1382): Another exception was thrown: setState() or markNeedsBuild() called during build.
错误:
I/flutter ( 2967): ══╡ EXCEPTION CAUGHT BY WIDGETS LIBRARY ╞═══════════════════════════════════════════════════════════
I/flutter ( 2967): The following assertion was thrown building StreamBuilder<dynamic>(dirty, state:
I/flutter ( 2967): _StreamBuilderBaseState<dynamic, AsyncSnapshot<dynamic>>#dc80d):
I/flutter ( 2967): setState() or markNeedsBuild() called during build.
I/flutter ( 2967): This WarningStatus widget cannot be marked as needing to build because the framework is already in
I/flutter ( 2967): the process of building widgets. A widget can be marked as needing to be built during the build
I/flutter ( 2967): phase only if one of its ancestors is currently building. This exception is allowed because the
I/flutter ( 2967): framework builds parent widgets before children, which means a dirty descendant will always be
I/flutter ( 2967): built. Otherwise, the framework might not visit this widget during this build phase.
I/flutter ( 2967): The widget on which setState() or markNeedsBuild() was called was:
I/flutter ( 2967): WarningStatus
I/flutter ( 2967): The widget which was currently being built when the offending call was made was:
I/flutter ( 2967): StreamBuilder<dynamic>
I/flutter ( 2967):
I/flutter ( 2967): The relevant error-causing widget was:
I/flutter ( 2967): StreamBuilder<dynamic> file:///home/megan/rr_app/lib/widgets/warnings.dart:45:20
I/flutter ( 2967):
I/flutter ( 2967): When the exception was thrown, this was the stack:
I/flutter ( 2967): #0 Element.markNeedsBuild.<anonymous closure> (package:flutter/src/widgets/framework.dart:4167:11)
I/flutter ( 2967): #1 Element.markNeedsBuild (package:flutter/src/widgets/framework.dart:4182:6)
I/flutter ( 2967): #2 State.setState (package:flutter/src/widgets/framework.dart:1253:14)
I/flutter ( 2967): #3 WarningStatusState._buildFaultOutput (package:rrapp/widgets/warnings.dart:135:5)
I/flutter ( 2967): #4 WarningStatusState._asyncWarningBox.<anonymous closure>.<anonymous closure>.<anonymous closure> (package:rrapp/widgets/warnings.dart:51:26)
I/flutter ( 2967): #5 StreamBuilder.build (package:flutter/src/widgets/async.dart:509:81)
I/flutter ( 2967): #6 _StreamBuilderBaseState.build (package:flutter/src/widgets/async.dart:127:48)
I/flutter ( 2967): #7 StatefulElement.build (package:flutter/src/widgets/framework.dart:4619:28)
I/flutter ( 2967): #8 ComponentElement.performRebuild (package:flutter/src/widgets/framework.dart:4502:15)
I/flutter ( 2967): #9 StatefulElement.performRebuild (package:flutter/src/widgets/framework.dart:4675:11)
I/flutter ( 2967): #10 Element.rebuild (package:flutter/src/widgets/framework.dart:4218:5)
I/flutter ( 2967): #11 BuildOwner.buildScope (package:flutter/src/widgets/framework.dart:2627:33)
I/flutter ( 2967): #12 WidgetsBinding.drawFrame (package:flutter/src/widgets/binding.dart:883:20)
I/flutter ( 2967): #13 RendererBinding._handlePersistentFrameCallback (package:flutter/src/rendering/binding.dart:284:5)
I/flutter ( 2967): #14 SchedulerBinding._invokeFrameCallback (package:flutter/src/scheduler/binding.dart:1108:15)
I/flutter ( 2967): #15 SchedulerBinding.handleDrawFrame (package:flutter/src/scheduler/binding.dart:1047:9)
I/flutter ( 2967): #16 SchedulerBinding._handleDrawFrame (package:flutter/src/scheduler/binding.dart:963:5)
I/flutter ( 2967): #20 _invoke (dart:ui/hooks.dart:261:10)
I/flutter ( 2967): #21 _drawFrame (dart:ui/hooks.dart:219:3)
I/flutter ( 2967): (elided 3 frames from dart:async)
I/flutter ( 2967):
I/flutter ( 2967): ════════════════════════════════════════════════════════════════════════════════════════════════════
I/flutter ( 2967): Another exception was thrown: setState() or markNeedsBuild() called during build.
您在错误的位置调用了 setState,....
只需从 Widget _buildFaultOutput(snapshot)
中删除 setState 并将其简化为 _iconColor = Colors.blue
。
最后补充一点
setState(() => _iconColor = Colors.blue );
在你的任何 AlertDialog actions's
FlatButton
之后 Navigator.pop(context)
...
当我尝试在单击图标时弹出的 AlertDialog 框中传输数据时出现错误。我知道这个问题涉及在不允许的情况下发生 streambuilder 构建,但我还没有找到解决此问题的方法。我的目标是能够单击一个图标并出现一个弹出框,显示一些流数据或带有取消按钮和清除数据的按钮的最新实例。
enum WarningInfo { CANCEL, CLEAR }
class WarningStatus extends StatefulWidget {
final ROSBridgeClient client;
WarningStatus({Key key, this.client}) : super(key: key);
@override
State<StatefulWidget> createState() => WarningStatusState();
}
class WarningStatusState extends State<WarningStatus> {
static const String _FAULT_TOPIC = '/active_faults';
//color (gray) when icon has not been enabled
Color _iconColor = Color(0xff4e4e4e);
@override
void initState() {
super.initState();
}
Future<WarningInfo> _asyncWarningBox(BuildContext context) async {
return showDialog<WarningInfo>(
context: context,
barrierDismissible:
true, // user must tap button for close dialog if false!
builder: (BuildContext context) {
return AlertDialog(
// content: StatefulBuilder(),
title: Text('Robot Warnings'),
// content: const Text('No warnings'),
content: StatefulBuilder(
builder: (BuildContext context, StateSetter setState) {
return StreamBuilder(
stream: widget.client.getTopicStream(_FAULT_TOPIC),
builder: (context, snapshot) {
if (!snapshot.hasData) {
return Text('None');
}
return _buildFaultOutput(snapshot);
});
}),
actions: <Widget>[
FlatButton(
child: const Text('CANCEL'),
onPressed: () {
Navigator.of(context).pop(WarningInfo.CANCEL);
},
),
FlatButton(
child: const Text(
'CLEAR WARNINGS',
style: TextStyle(color: Colors.red),
),
onPressed: () {
// Navigator.of(context).pop(WarningInfo.CLEAR);
_clearFaults();
},
)
],
);
},
);
}
@override
Widget build(BuildContext context) {
widget.client.subscribe(_FAULT_TOPIC, 'rover_safety/FaultListMsg');
return SizedBox(
child: Column(
mainAxisSize: MainAxisSize.min,
children: <Widget>[
IconButton(
icon: Icon(
CustomIcons.attention,
color: _iconColor,
),
disabledColor: Color(0xff4e4e4e),
iconSize: 80,
tooltip: "Warnings",
onPressed: () async {
final WarningInfo action = await _asyncWarningBox(context);
print("Confirm Action $action");
},
// onPressed: () {
// setState(() {
// _iconColor = Colors.yellow;
// });
// },
),
// Text("Fault Messages:"),
// SizedBox(
// height: 10.0,
// ),
// StreamBuilder(
// stream: widget.client.getTopicStream(_FAULT_TOPIC),
// builder: (context, snapshot) {
// if (!snapshot.hasData) {
// return CircularProgressIndicator();
// }
// return _buildFaultOutput(snapshot);
// },
// ),
// SizedBox(
// height: 10.0,
// ),
// ButtonBar(
// mainAxisSize: MainAxisSize.min,
// children: <Widget>[
// RaisedButton(
// child: const Text("Clear Faults"),
// onPressed: _clearFaults,
// ),
// ],
// )
],
),
);
}
Widget _buildFaultOutput(snapshot) {
setState(() {
_iconColor = Colors.blue;
});
var msg = snapshot.data;
// print(payload);
var faultList = msg["fault_list"];
print("Number of faults: ${faultList.length}");
// if (faultListLength>0){
// var faults = faultList[0];
// var motorCurrent = faults["fault"];
// var severity = faults["severity"];
// print(faults);
// return Text("Number of Faults: $faultListLength\n$motorCurrent\nseverity: $severity");
// }
// else{
// return Text("No faults");
// }
if (faultList.length == 0) {
return Text("No faults");
} else {
for (var i = 0; i < faultList.length; i++) {
var faults = faultList[i];
var motorCurrent = faults["fault"];
var severity = faults["severity"];
print(faults);
print(i);
return Text(
"Number of Faults: ${faultList.length}\n$motorCurrent\nseverity: $severity");
}
}
}
void _clearFaults() {
widget.client.publish('/clear_faults', 'std_msgs/Empty');
}
@override
void dispose() {
super.dispose();
}
}
错误:
I/flutter ( 1382): Another exception was thrown: setState() or markNeedsBuild() called during build.
错误:
I/flutter ( 2967): ══╡ EXCEPTION CAUGHT BY WIDGETS LIBRARY ╞═══════════════════════════════════════════════════════════
I/flutter ( 2967): The following assertion was thrown building StreamBuilder<dynamic>(dirty, state:
I/flutter ( 2967): _StreamBuilderBaseState<dynamic, AsyncSnapshot<dynamic>>#dc80d):
I/flutter ( 2967): setState() or markNeedsBuild() called during build.
I/flutter ( 2967): This WarningStatus widget cannot be marked as needing to build because the framework is already in
I/flutter ( 2967): the process of building widgets. A widget can be marked as needing to be built during the build
I/flutter ( 2967): phase only if one of its ancestors is currently building. This exception is allowed because the
I/flutter ( 2967): framework builds parent widgets before children, which means a dirty descendant will always be
I/flutter ( 2967): built. Otherwise, the framework might not visit this widget during this build phase.
I/flutter ( 2967): The widget on which setState() or markNeedsBuild() was called was:
I/flutter ( 2967): WarningStatus
I/flutter ( 2967): The widget which was currently being built when the offending call was made was:
I/flutter ( 2967): StreamBuilder<dynamic>
I/flutter ( 2967):
I/flutter ( 2967): The relevant error-causing widget was:
I/flutter ( 2967): StreamBuilder<dynamic> file:///home/megan/rr_app/lib/widgets/warnings.dart:45:20
I/flutter ( 2967):
I/flutter ( 2967): When the exception was thrown, this was the stack:
I/flutter ( 2967): #0 Element.markNeedsBuild.<anonymous closure> (package:flutter/src/widgets/framework.dart:4167:11)
I/flutter ( 2967): #1 Element.markNeedsBuild (package:flutter/src/widgets/framework.dart:4182:6)
I/flutter ( 2967): #2 State.setState (package:flutter/src/widgets/framework.dart:1253:14)
I/flutter ( 2967): #3 WarningStatusState._buildFaultOutput (package:rrapp/widgets/warnings.dart:135:5)
I/flutter ( 2967): #4 WarningStatusState._asyncWarningBox.<anonymous closure>.<anonymous closure>.<anonymous closure> (package:rrapp/widgets/warnings.dart:51:26)
I/flutter ( 2967): #5 StreamBuilder.build (package:flutter/src/widgets/async.dart:509:81)
I/flutter ( 2967): #6 _StreamBuilderBaseState.build (package:flutter/src/widgets/async.dart:127:48)
I/flutter ( 2967): #7 StatefulElement.build (package:flutter/src/widgets/framework.dart:4619:28)
I/flutter ( 2967): #8 ComponentElement.performRebuild (package:flutter/src/widgets/framework.dart:4502:15)
I/flutter ( 2967): #9 StatefulElement.performRebuild (package:flutter/src/widgets/framework.dart:4675:11)
I/flutter ( 2967): #10 Element.rebuild (package:flutter/src/widgets/framework.dart:4218:5)
I/flutter ( 2967): #11 BuildOwner.buildScope (package:flutter/src/widgets/framework.dart:2627:33)
I/flutter ( 2967): #12 WidgetsBinding.drawFrame (package:flutter/src/widgets/binding.dart:883:20)
I/flutter ( 2967): #13 RendererBinding._handlePersistentFrameCallback (package:flutter/src/rendering/binding.dart:284:5)
I/flutter ( 2967): #14 SchedulerBinding._invokeFrameCallback (package:flutter/src/scheduler/binding.dart:1108:15)
I/flutter ( 2967): #15 SchedulerBinding.handleDrawFrame (package:flutter/src/scheduler/binding.dart:1047:9)
I/flutter ( 2967): #16 SchedulerBinding._handleDrawFrame (package:flutter/src/scheduler/binding.dart:963:5)
I/flutter ( 2967): #20 _invoke (dart:ui/hooks.dart:261:10)
I/flutter ( 2967): #21 _drawFrame (dart:ui/hooks.dart:219:3)
I/flutter ( 2967): (elided 3 frames from dart:async)
I/flutter ( 2967):
I/flutter ( 2967): ════════════════════════════════════════════════════════════════════════════════════════════════════
I/flutter ( 2967): Another exception was thrown: setState() or markNeedsBuild() called during build.
您在错误的位置调用了 setState,....
只需从 Widget _buildFaultOutput(snapshot)
中删除 setState 并将其简化为 _iconColor = Colors.blue
。
最后补充一点
setState(() => _iconColor = Colors.blue );
在你的任何 AlertDialog actions's
FlatButton
之后 Navigator.pop(context)
...