diff --git a/lib/widgets/app.dart b/lib/widgets/app.dart index c58848a369..8f76e66e4c 100644 --- a/lib/widgets/app.dart +++ b/lib/widgets/app.dart @@ -63,6 +63,22 @@ class ZulipApp extends StatefulWidget { /// to be mounted. static final navigatorKey = GlobalKey(); + /// The [ScaffoldMessengerState] for the app. + /// + /// This is null during the app's early startup, while [ready] is still false. + /// + /// For code that exists entirely outside the widget tree and has no natural + /// [BuildContext] of its own, this enables controlling snack bars. + /// Where a relevant [BuildContext] does exist, prefer using that instead, + /// with [ScaffoldMessenger.of]. + static ScaffoldMessengerState? get scaffoldMessenger { + final context = navigatorKey.currentContext; + if (context == null) return null; + // Not maybeOf; we use MaterialApp, which provides ScaffoldMessenger, + // so it's a bug if navigatorKey is mounted somewhere lacking that. + return ScaffoldMessenger.of(context); + } + /// Reset the state of [ZulipApp] statics, for testing. /// /// TODO refactor this better, perhaps unify with ZulipBinding diff --git a/test/flutter_checks.dart b/test/flutter_checks.dart index d0906d849b..48666f1ac9 100644 --- a/test/flutter_checks.dart +++ b/test/flutter_checks.dart @@ -2,6 +2,7 @@ library; import 'package:checks/checks.dart'; +import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; @@ -51,7 +52,7 @@ extension RouteSettingsChecks on Subject { Subject get arguments => has((s) => s.arguments, 'arguments'); } -extension ValueNotifierChecks on Subject> { +extension ValueListenableChecks on Subject> { Subject get value => has((c) => c.value, 'value'); } diff --git a/test/widgets/app_test.dart b/test/widgets/app_test.dart index 578fd4b3ed..13e2c98d54 100644 --- a/test/widgets/app_test.dart +++ b/test/widgets/app_test.dart @@ -154,4 +154,15 @@ void main() { ..bottom.isLessThan(2 / 3 * screenHeight); }); }); + + group('scaffoldMessenger', () { + testWidgets('scaffoldMessenger becomes non-null after startup', (tester) async { + await tester.pumpWidget(const ZulipApp()); + check(ZulipApp.scaffoldMessenger).isNull(); + check(ZulipApp.ready).value.isFalse(); + await tester.pump(); + check(ZulipApp.scaffoldMessenger).isNotNull(); + check(ZulipApp.ready).value.isTrue(); + }); + }); }