【问题标题】:Flutter web Firebase onAuthStateChanges gets triggered every time I refresh the app每次刷新应用程序时都会触发 Flutter web Firebase onAuthStateChanges
【发布时间】:2020-12-29 05:36:09
【问题描述】:

我正在使用 Flutter Web 开发一个网络应用。

每次我刷新页面时,都会触发 onAuthStateChanges 并执行仅在身份验证状态发生更改时才应触发的功能。

这种行为是有意的吗?

为了订阅 onAuthStateChanges,我用一个有状态的小部件包装了我的主小部件并监听了 onAuthStateChanges,如下所示: Flutter Stream Builder Triggered when Navigator Pop or Push is Called

以下是我的代码:

我的包装:

/// Wrapper for stateful functionality to provide onInit calls in stateles widget
class StatefulWrapper extends StatefulWidget {
  final Function onInit;
  final Widget child;
  const StatefulWrapper({@required this.onInit, @required this.child});
  @override
  _StatefulWrapperState createState() => _StatefulWrapperState();
}

class _StatefulWrapperState extends State<StatefulWrapper> {
  @override
  void initState() {
    if (widget.onInit != null) {
      widget.onInit();
    }
    super.initState();
  }

  @override
  Widget build(BuildContext context) {
    return widget.child;
  }
}

我在 main.dart 中的小部件:

Widget build(BuildContext context) {
    var router = MyRouter.Router();
    var authenticationService = locator<AuthenticationService>();
    var sharedPreferences = locator<SharedPreferencesService>();

    return StatefulWrapper(
      onInit: () {
        authenticationService.listenToAuthChanges();
      },
      child: FutureBuilder(
          future: sharedPreferences.getSharedPreferences('ruolo'),
          builder: (context, snapshot) {
            return MaterialApp(
              builder: ExtendedNavigator<MyRouter.Router>(
                initialRoute: MyRouter.Routes.startupView,
                guards: [AuthGuard(), AdminGuard()],
                router: router,
                navigatorKey: locator<NavigationService>().navigatorKey,
              ),
              title: 'Flutter Demo',
              theme: ThemeData(
                primarySwatch: Colors.blue,
                visualDensity: VisualDensity.adaptivePlatformDensity,
              ),
              onGenerateRoute: router.onGenerateRoute,
            );
          }),
    );
  }

这是来自 authenticationService 的方法:

  void listenToAuthChanges() {
    FirebaseAuth.instance.authStateChanges().listen((firebaseUser) async {
      if (firebaseUser == null) {
        navigationService.navigateTo(Routes.login);
      } else {
         navigationService.replaceWith(Routes.home);
      }
    });
  }

当我在浏览器中刷新页面时,会触发此函数的侦听器,如果我没有登录,它会导航到登录,否则在主页上。

这是故意的吗?对我来说,只有当它收听的内容发生变化时,监听器才应该触发。

【问题讨论】:

  • 如果您刷新页面,您实际上是在重新启动整个应用程序。这是有意的,否则你无法在启动时获得身份验证状态。
  • 好吧,但是为什么会触发函数的handler呢?对我来说,这对我来说没有任何意义。
  • 我的意思是,只有当该事件发生时才应该触发侦听器的处理程序。当我刷新时,我的登录状态没有改变,它和以前一样。
  • 如果他们将其删除,则无法确定应用启动时的身份验证状态。当您刷新网页时,它不会“记住”之前的身份验证状态。您正在重新启动整个应用程序。

标签: firebase flutter firebase-authentication flutter-web


【解决方案1】:

当应用加载时,Firebase 会检查用户的身份验证状态。在该过程开始时,调用任何具有空值的活动身份验证状态侦听器。如果用户可以通过身份验证,则使用用户配置文件对象调用侦听器。

所以您所看到的确实按预期工作。

【讨论】:

  • 我还有一个问题。使用我在这里显示的代码,触发器似乎被触发了不止一次。我想某处应该有一个错误,因为这不是故意的,对吧?
  • 您的回调通常会在页面加载时使用null 调用,然后在稍后与实际用户一起调用。
  • 感谢您的解释。但是,我注意到一种与您向我暴露的行为不同的行为。当我登录和注销时,只调用一次监听器。当我刷新我的页面时,会多次调用监听器,但如果我已登录,则会使用用户的值多次调用它。如果我没有登录,它会被多次调用为 null 。以下是刷新页面的示例: 触发了侦听器 - 时间戳:2020-09-11 09:17:53.741 电子邮件为:mail@mail.com 触发了侦听器 - 时间戳:2020-09-11 09:17:53.755 电子邮件为: mail@mail.com
  • 我在移动设备上使用过 firebase,我喜欢它,但这种在网络上的行为让我很难过。由于我需要保存我的用户身份验证状态,我想使用监听器将他重定向到正确的页面。然而,监听器多次调用,使得网络应用程序有几个无用和烦人的重定向。解决这个问题的标准方法是什么?到目前为止,我在侦听器中没有任何重定向,但我 sed 了一个共享首选项(loggedIn),并且我有一个启动屏幕来检查这个变量并相应地重定向到它。但我觉得这是一种解决方法,而不是正确的方法。
  • 我发现的另一个解决方案是检查 onAuthStateChange 中的时间戳。如果时间戳大于 500 毫秒,则进行导航。但我仍然认为这是一个肮脏的解决方法,而不是解决方案。
猜你喜欢
  • 2021-07-05
  • 2016-06-29
  • 1970-01-01
  • 1970-01-01
  • 2011-10-04
  • 1970-01-01
  • 2020-02-02
  • 2018-05-07
  • 2021-09-13
相关资源
最近更新 更多