import 'dart:async'; import 'dart:io'; import 'package:adaptive_theme/adaptive_theme.dart'; import 'package:device_info_plus/device_info_plus.dart'; import 'package:equatable/equatable.dart'; import 'package:feature_discovery/feature_discovery.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/scheduler.dart'; import 'package:flutter/services.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_local_notifications/flutter_local_notifications.dart'; import 'package:flutter_siri_suggestions/flutter_siri_suggestions.dart'; import 'package:hacki/blocs/blocs.dart'; import 'package:hacki/config/constants.dart'; import 'package:hacki/config/custom_router.dart'; import 'package:hacki/config/locator.dart'; import 'package:hacki/cubits/cubits.dart'; import 'package:hacki/services/fetcher.dart'; import 'package:hacki/styles/styles.dart'; import 'package:hacki/utils/theme_util.dart'; import 'package:hive/hive.dart'; import 'package:hydrated_bloc/hydrated_bloc.dart'; import 'package:logger/logger.dart'; import 'package:path_provider/path_provider.dart'; import 'package:rxdart/rxdart.dart' show BehaviorSubject; import 'package:visibility_detector/visibility_detector.dart'; import 'package:workmanager/workmanager.dart'; // For receiving payload event from local notifications. final BehaviorSubject selectNotificationSubject = BehaviorSubject(); // For receiving payload event from siri suggestions. final BehaviorSubject siriSuggestionSubject = BehaviorSubject(); late final bool isTesting; void notificationReceiver(NotificationResponse details) => selectNotificationSubject.add(details.payload); Future main({bool testing = false}) async { WidgetsFlutterBinding.ensureInitialized(); isTesting = testing; final Directory tempDir = await getTemporaryDirectory(); final String tempPath = tempDir.path; Hive.init(tempPath); await setUpLocator(); EquatableConfig.stringify = true; FlutterError.onError = (FlutterErrorDetails details) { locator.get().e( details.summary, error: details.exceptionAsString(), stackTrace: details.stack, ); }; final HydratedStorage storage = await HydratedStorage.build( storageDirectory: kIsWeb ? HydratedStorage.webStorageDirectory : await getTemporaryDirectory(), ); if (Platform.isIOS) { unawaited( Workmanager().initialize( fetcherCallbackDispatcher, ), ); final FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); const AndroidInitializationSettings initializationSettingsAndroid = AndroidInitializationSettings('@mipmap/ic_launcher'); const DarwinInitializationSettings initializationSettingsIOS = DarwinInitializationSettings(); const InitializationSettings initializationSettings = InitializationSettings( android: initializationSettingsAndroid, iOS: initializationSettingsIOS, ); await flutterLocalNotificationsPlugin.initialize( initializationSettings, onDidReceiveBackgroundNotificationResponse: notificationReceiver, onDidReceiveNotificationResponse: notificationReceiver, ); await flutterLocalNotificationsPlugin .resolvePlatformSpecificImplementation< IOSFlutterLocalNotificationsPlugin>() ?.requestPermissions( alert: true, badge: true, sound: true, ); FlutterSiriSuggestions.instance.configure( onLaunch: (Map message) async { final String? storyId = message['key'] as String?; if (storyId == null) return; siriSuggestionSubject.add(storyId); }, ); } else if (Platform.isAndroid) { final DeviceInfoPlugin deviceInfoPlugin = DeviceInfoPlugin(); final AndroidDeviceInfo androidInfo = await deviceInfoPlugin.androidInfo; final int sdk = androidInfo.version.sdkInt; if (sdk > 28) { SystemChrome.setSystemUIOverlayStyle( const SystemUiOverlayStyle( statusBarColor: Palette.transparent, systemNavigationBarColor: Palette.transparent, systemNavigationBarDividerColor: Palette.transparent, ), ); } await SystemChrome.setEnabledSystemUIMode( SystemUiMode.edgeToEdge, overlays: [SystemUiOverlay.top], ); } final AdaptiveThemeMode? savedThemeMode = await AdaptiveTheme.getThemeMode(); // Uncomment this line to log events from bloc/cubit. // Bloc.observer = CustomBlocObserver(); HydratedBloc.storage = storage; VisibilityDetectorController.instance.updateInterval = Durations.ms200; runApp( HackiApp( savedThemeMode: savedThemeMode, ), ); } class HackiApp extends StatelessWidget { const HackiApp({ super.key, this.savedThemeMode, }); final AdaptiveThemeMode? savedThemeMode; @override Widget build(BuildContext context) { return MultiBlocProvider( providers: >[ BlocProvider( lazy: false, create: (BuildContext context) => PreferenceCubit(), ), BlocProvider( lazy: false, create: (BuildContext context) => FilterCubit(), ), BlocProvider( create: (BuildContext context) => StoriesBloc( preferenceCubit: context.read(), filterCubit: context.read(), ), ), BlocProvider( lazy: false, create: (BuildContext context) => AuthBloc(), ), BlocProvider( lazy: false, create: (BuildContext context) => HistoryCubit( authBloc: context.read(), ), ), BlocProvider( lazy: false, create: (BuildContext context) => FavCubit( authBloc: context.read(), ), ), BlocProvider( lazy: false, create: (BuildContext context) => BlocklistCubit(), ), BlocProvider( lazy: false, create: (BuildContext context) => SearchCubit(), ), BlocProvider( lazy: false, create: (BuildContext context) => NotificationCubit( authBloc: context.read(), preferenceCubit: context.read(), ), ), BlocProvider( lazy: false, create: (BuildContext context) => PinCubit(), ), BlocProvider( lazy: false, create: (BuildContext context) => SplitViewCubit(), ), BlocProvider( lazy: false, create: (BuildContext context) => ReminderCubit()..init(), ), BlocProvider( lazy: false, create: (BuildContext context) => PostCubit(), ), BlocProvider( lazy: false, create: (BuildContext context) => EditCubit(), ), BlocProvider( create: (BuildContext context) => TabCubit( preferenceCubit: context.read(), )..init(), ), ], child: BlocBuilder( buildWhen: (PreferenceState previous, PreferenceState current) => previous.appColor != current.appColor || previous.font != current.font || previous.textScaleFactor != current.textScaleFactor || previous.material3Enabled != current.material3Enabled, builder: (BuildContext context, PreferenceState state) { return AdaptiveTheme( key: ValueKey( '''${state.appColor}${state.font}${state.material3Enabled}''', ), light: ThemeData( primaryColor: state.appColor, colorScheme: ColorScheme.fromSwatch( primarySwatch: state.appColor, ), fontFamily: state.font.name, ), dark: ThemeData( brightness: Brightness.dark, primaryColor: state.appColor, colorScheme: ColorScheme.fromSwatch( primarySwatch: state.appColor, brightness: Brightness.dark, ), canvasColor: Palette.black, fontFamily: state.font.name, ), initial: savedThemeMode ?? AdaptiveThemeMode.system, builder: (ThemeData theme, ThemeData darkTheme) { return FutureBuilder( future: AdaptiveTheme.getThemeMode(), builder: ( BuildContext context, AsyncSnapshot snapshot, ) { final AdaptiveThemeMode? mode = snapshot.data; ThemeUtil.updateStatusBarSetting( SchedulerBinding .instance.platformDispatcher.platformBrightness, mode, ); final bool isDarkModeEnabled = mode == AdaptiveThemeMode.dark || (mode == AdaptiveThemeMode.system && View.of(context) .platformDispatcher .platformBrightness == Brightness.dark); return FeatureDiscovery( child: MediaQuery( data: MediaQuery.of(context).copyWith( textScaleFactor: state.textScaleFactor == 1 ? null : state.textScaleFactor, ), child: MaterialApp.router( key: Key(state.appColor.hashCode.toString()), title: 'Hacki', debugShowCheckedModeBanner: false, theme: (isDarkModeEnabled ? darkTheme : theme).copyWith( useMaterial3: state.material3Enabled, dividerTheme: state.material3Enabled ? DividerThemeData( color: Palette.grey.withOpacity(0.2), ) : null, switchTheme: state.material3Enabled ? SwitchThemeData( trackColor: MaterialStateProperty.resolveWith( (Set states) { if (states .contains(MaterialState.selected)) { return null; } else { return Palette.grey.withOpacity(0.2); } }, ), ) : null, bottomSheetTheme: state.material3Enabled ? BottomSheetThemeData( modalElevation: 0, backgroundColor: isDarkModeEnabled ? Palette.black : Palette.white, shape: const RoundedRectangleBorder(), ) : null, inputDecorationTheme: state.material3Enabled ? InputDecorationTheme( enabledBorder: UnderlineInputBorder( borderSide: BorderSide( color: isDarkModeEnabled ? Palette.white : Palette.black, ), ), ) : null, ), routerConfig: router, ), ), ); }, ); }, ); }, ), ); } }