gpt4 book ai didi

dart - 在 StreamBuilder 中使用 SnackBar 的奇特方式是什么?

转载 作者:IT老高 更新时间:2023-10-28 12:43:45 24 4
gpt4 key购买 nike

我正在为我的应用程序实现 Bloc 模式,我必须显示 SnackBar,它会在未验证登录时显示错误消息。

但我无法在小部件的构建阶段显示 SnackBar。我寻找了很多解决方案,但我找不到。

使用此功能最有效的方法是什么?

我的代码

import 'package:chat_app/auth/auth_bloc.dart';
import 'package:chat_app/auth/auth_state.dart';
import 'package:chat_app/main_page.dart';
import 'package:flutter/material.dart';

void main() => runApp(App());

class App extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Chat App',
home: MyApp(),
debugShowCheckedModeBanner: false,
);
}
}

class MyApp extends StatefulWidget {
@override
_MyAppState createState() => _MyAppState();
}

class _MyAppState extends State<MyApp> {

final GlobalKey<FormState> _formKey = GlobalKey<FormState>();
final GlobalKey<ScaffoldState> _scaffoldKey = GlobalKey<ScaffoldState>();
final AuthBloc _bloc = AuthBloc();

final _emailController = TextEditingController();
final _passwordController = TextEditingController();

@override
void dispose() {
_emailController.dispose();
_passwordController.dispose();
_bloc.dispose();
super.dispose();
}

@override
Widget build(BuildContext context) {
return Scaffold(
key: _scaffoldKey,
appBar: AppBar(title: Text('Chat Example')),
body: StreamBuilder(
initialData: AuthInitializing(),
stream: _bloc.authStream,
builder: (BuildContext context, AsyncSnapshot<AuthState> snapshot){
AuthState state = snapshot.data;
if(state is AuthUnauthenticated){
_showErrorMessage(state.errorMessage);
}
if(state is AuthAuthenticated){
_moveNextPage(context);
}
return Form(
key: _formKey,
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 20.0),
child: Column(
children: <Widget>[
TextFormField(
controller: _emailController,
keyboardType: TextInputType.emailAddress,
decoration: InputDecoration(
border: OutlineInputBorder(),
labelText: '이메일',
),
),
SizedBox(height: 20.0),
TextFormField(
controller: _passwordController,
keyboardType: TextInputType.text,
obscureText: true,
decoration: InputDecoration(
border: OutlineInputBorder(),
labelText: '비밀번호'
),
),
SizedBox(height: 20.0),
RaisedButton(
child: Text('로그인',style: TextStyle(color: Colors.white),),
onPressed: () => _bloc.addLoginData(_emailController.text, _passwordController.text),
color: Theme.of(context).primaryColor,
),
SizedBox(height: 15.0),
state is AuthLoading ? _progressBar() : Container()
],
),
),
);
},
),
);
}

void _showErrorMessage(String message){
_scaffoldKey.currentState.showSnackBar(SnackBar(
content: Text(message),
));
}

void _moveNextPage(BuildContext context) {
Navigator.pushReplacement(context, MaterialPageRoute(
builder: (_) => MainPage()
));
}

Widget _progressBar() {
return Center(
child: CircularProgressIndicator(),
);
}
}

堆栈跟踪

I/flutter (30505): ══╡ EXCEPTION CAUGHT BY WIDGETS LIBRARY ╞═══════════════════════════════════════════════════════════ I/flutter (30505): The following assertion was thrown building StreamBuilder(dirty, state: I/flutter (30505): _StreamBuilderBaseState>#bd8b2): I/flutter (30505): setState() or markNeedsBuild() called during build. I/flutter (30505): This Scaffold widget cannot be marked as needing to build because the framework is already in the I/flutter (30505): process of building widgets. A widget can be marked as needing to be built during the build phase I/flutter (30505): only if one of its ancestors is currently building. This exception is allowed because the framework I/flutter (30505): builds parent widgets before children, which means a dirty descendant will always be built. I/flutter (30505): Otherwise, the framework might not visit this widget during this build phase. I/flutter (30505): The widget on which setState() or markNeedsBuild() was called was: I/flutter (30505):
Scaffold-[LabeledGlobalKey#5bdc5](state: ScaffoldState#61be4(tickers: tracking 2 I/flutter (30505): tickers))

最佳答案

首先,您必须确保始终返回一个小部件

然后您可以将 SnackBar 安排在帧的末尾

if(state is AuthUnauthenticated){
WidgetsBinding.instance.addPostFrameCallback((_) => _showErrorMessage(state.errorMessage));
return Container();
}

您还应该检查 data 是否为 null o snapshot 是否有数据。

关于dart - 在 StreamBuilder 中使用 SnackBar 的奇特方式是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54230331/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com