StreamProvider 侦听的 _MapStream<FirebaseUser, User> 抛出了一个异常<User>,但没有提供"catchError"



<---更新--->

找到了解决方案。检查下方的答案

<---更新--->

有什么想法吗?为什么会发生这种情况,是什么原因导致了这个错误,以及我该如何修复它?

我在YouTube上关注TheNetNinja的教程,我看到人们也有类似的问题,但不完全一样。

当应用程序在没有注销的情况下突然关闭(使用Android Studio中的停止按钮(,然后重新启动时,仅当使用电子邮件和pass登录时,才会引发异常。对于谷歌登录,它似乎运行良好。此外,如果我在Firebase控制台中手动创建一个新用户并首次登录,则不会发生这种情况。

例外:

════════ Exception caught by provider ══════════════════════════════════════════════════════════════
The following assertion was thrown:
An exception was throw by _MapStream<FirebaseUser, User> listened by
StreamProvider<User>, but no `catchError` was provided.
Exception:
RangeError (index): Invalid value: Only valid value is 0: 1
════════════════════════════════════════════════════════════════════════════════════════════════════

我在我的pubspec.yaml中使用firebase_auth: ^0.16.0,我想这可能与版本有关。

当我登录时,我得到这个输出

I/BiChannelGoogleApi(17821): [FirebaseAuth: ] getGoogleApiForMethod() returned Gms: com.google.firebase.auth.api.internal.zzaq@cd1fa83
D/FirebaseAuth(17821): Notifying id token listeners about user ( 1EMwk8483hQWnSQwJeQYm9AcpiD2 ).

这看起来像是帐户被验证并登录,但屏幕没有改变,所以我想它在验证过程中被卡住了。

奇怪的是,如果我正常登录并重新启动应用程序,它会正常工作。如果我继续重新启动它,它会正常工作,但如果我突然停止它,我会再次出现异常。

这是我的身份验证码

import 'package:firebase_auth/firebase_auth.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:google_sign_in/google_sign_in.dart';
import 'package:common_ui_module/Utils.dart';
import 'package:user_module/Role.dart';
import 'package:user_module/User.dart';
import 'package:localization_module/AppLocalizations.dart';
class AuthService {
final FirebaseAuth _auth = FirebaseAuth.instance;
return user != null
? User(uid: user.uid)
: null;
}
Stream<User> get user {
// map FireBaseUsers to User objects
return _auth.onAuthStateChanged.map(_userFromFireBaseUser);
}

// sign in with email and pass
Future<User> signInWithEmailAndPassword(String email, String pass) async {
try {
AuthResult result =
await _auth.signInWithEmailAndPassword(email: email, password: pass);
FirebaseUser user = result.user;
assert(await user.getIdToken() != null);
print(
"Signed in: ${user.email} , pass: $pass, phone: ${user.phoneNumber}");
/// return the mapped user
return _userFromFireBaseUser(user);
} catch (e) {
print(e.toString());
return null;
}
}

// sign out
Future signOut() async {
try {
return await _auth.signOut();
} catch (e) {
print(e.toString());
return null;
}
}
}

这是在登录和主屏幕之间切换的包装器

class Wrapper extends StatelessWidget {
@override
Widget build(BuildContext context) {
final User user = Provider.of<User>(context);
if (user != null) {
return RestaurantOrdersScreen();
} else {
return LoginScreen();
}
}
}

我已经将主要应用程序封装在一个MultiProvider中:

Widget build(BuildContext context) {
return MultiProvider(
providers: [
/// PROVIDER FOR AUTHENTICATION
StreamProvider<User>.value(
value: AuthService().user,
),
//...
// some other providers
],
child: MaterialApp(
home: SplashScreen(),
),
);
}

终于找到了解决方案!

对我来说,当我试图操作displayName时,就是这种分裂。这就是为什么我得到的索引超出范围的错误。

List<String> name = ["", ""];
if (user != null && user.displayName != null) {
name = user.displayName.split(" ");
}
return user != null
? User(
id: 0,
token: user.uid,
firstName: name[0],
lastName: name[1],
...

因此,如果您遇到这种错误,请在尝试将FirebaseUser转换为您自己的自定义User对象时仔细检查您的User属性操作。

相关内容

最新更新