import 'package:provider/provider.dart'; import 'package:twonly/globals.dart'; import 'package:twonly/src/localization/generated/app_localizations.dart'; import 'package:twonly/src/providers/connection_provider.dart'; import 'package:twonly/src/providers/settings_change_provider.dart'; import 'package:twonly/src/services/notification_service.dart'; import 'package:twonly/src/utils/storage.dart'; import 'package:twonly/src/views/onboarding/onboarding_view.dart'; import 'package:twonly/src/views/home_view.dart'; import 'package:twonly/src/views/onboarding/register_view.dart'; import 'package:flutter/material.dart'; import 'package:flutter_localizations/flutter_localizations.dart'; import 'dart:async'; // these global function can be called from anywhere to update // the ui when something changed. The callbacks will be set by // App widget. // this callback is called by the apiProvider Function(bool) globalCallbackConnectionState = (a) {}; bool globalIsAppInBackground = true; int globalBestFriendUserId = -1; // these two callbacks are called on updated to the corresponding database /// The Widget that configures your application. class App extends StatefulWidget { const App({super.key}); @override State createState() => _AppState(); } class _AppState extends State with WidgetsBindingObserver { bool wasPaused = false; @override void initState() { super.initState(); globalIsAppInBackground = false; WidgetsBinding.instance.addObserver(this); // register global callbacks to the widget tree globalCallbackConnectionState = (update) { context.read().updateConnectionState(update); setUserPlan(); setupNotificationWithUsers(); }; initAsync(); } Future setUserPlan() async { final user = await getUser(); globalBestFriendUserId = -1; if (user != null && context.mounted) { if (user.myBestFriendContactId != null) { final contact = await twonlyDatabase.contactsDao .getContactByUserId(user.myBestFriendContactId!) .getSingleOrNull(); if (contact != null) { if (contact.alsoBestFriend) { globalBestFriendUserId = user.myBestFriendContactId ?? 0; } } } context.read().updatePlan(user.subscriptionPlan); } } Future initAsync() async { setUserPlan(); apiProvider.connect(); } @override void didChangeAppLifecycleState(AppLifecycleState state) { super.didChangeAppLifecycleState(state); if (state == AppLifecycleState.resumed) { if (wasPaused) { globalIsAppInBackground = false; twonlyDatabase.markUpdated(); apiProvider.connect(); } } else if (state == AppLifecycleState.paused) { wasPaused = true; globalIsAppInBackground = true; } } @override void dispose() { WidgetsBinding.instance.removeObserver(this); globalCallbackConnectionState = (a) {}; super.dispose(); } @override Widget build(BuildContext context) { return ListenableBuilder( listenable: context.watch(), builder: (BuildContext context, Widget? child) { return MaterialApp( restorationScopeId: 'app', localizationsDelegates: const [ AppLocalizations.delegate, GlobalMaterialLocalizations.delegate, GlobalWidgetsLocalizations.delegate, GlobalCupertinoLocalizations.delegate, ], supportedLocales: const [ Locale('en', ''), Locale('de', ''), ], onGenerateTitle: (BuildContext context) => "twonly", theme: ThemeData( colorScheme: ColorScheme.fromSeed(seedColor: const Color(0xFF57CC99)), inputDecorationTheme: const InputDecorationTheme(border: OutlineInputBorder())), darkTheme: ThemeData.dark().copyWith( colorScheme: ColorScheme.fromSeed( brightness: Brightness.dark, seedColor: const Color(0xFF57CC99), ), inputDecorationTheme: const InputDecorationTheme( border: OutlineInputBorder(), ), ), themeMode: context.watch().themeMode, initialRoute: '/', routes: { "/": (context) => AppMainWidget(initialPage: 1), "/chats": (context) => AppMainWidget(initialPage: 0) }, ); }, ); } } class AppMainWidget extends StatefulWidget { const AppMainWidget({super.key, required this.initialPage}); final int initialPage; @override State createState() => _AppMainWidgetState(); } class _AppMainWidgetState extends State { Future userCreated = isUserCreated(); bool showOnboarding = true; @override Widget build(BuildContext context) { return Stack( children: [ FutureBuilder( future: userCreated, builder: (context, snapshot) { if (!snapshot.hasData) { return Center(child: Container()); } if (snapshot.data!) { return HomeView( initialPage: widget.initialPage, ); } return showOnboarding ? OnboardingView( callbackOnSuccess: () { setState(() { showOnboarding = false; }); }, ) : RegisterView( callbackOnSuccess: () { setState(() { userCreated = isUserCreated(); }); }, ); }, ), ], ); } }