app_widget.dart 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. import 'package:appflowy/plugins/document/presentation/more/cubit/document_appearance_cubit.dart';
  2. import 'package:appflowy_editor/appflowy_editor.dart' hide Log;
  3. import 'package:easy_localization/easy_localization.dart';
  4. import 'package:flowy_infra/theme.dart';
  5. import 'package:flowy_infra_ui/flowy_infra_ui.dart';
  6. import 'package:appflowy_backend/log.dart';
  7. import 'package:appflowy_backend/protobuf/flowy-user/protobuf.dart';
  8. import 'package:flutter/material.dart';
  9. import 'package:flutter_bloc/flutter_bloc.dart';
  10. import '../../user/application/user_settings_service.dart';
  11. import '../../workspace/application/appearance.dart';
  12. import '../startup.dart';
  13. class InitAppWidgetTask extends LaunchTask {
  14. const InitAppWidgetTask();
  15. @override
  16. LaunchTaskType get type => LaunchTaskType.appLauncher;
  17. @override
  18. Future<void> initialize(LaunchContext context) async {
  19. final widget = context.getIt<EntryPoint>().create(context.config);
  20. final appearanceSetting =
  21. await UserSettingsBackendService().getAppearanceSetting();
  22. // If the passed-in context is not the same as the context of the
  23. // application widget, the application widget will be rebuilt.
  24. final app = ApplicationWidget(
  25. key: ValueKey(context),
  26. appearanceSetting: appearanceSetting,
  27. appTheme: await appTheme(appearanceSetting.theme),
  28. child: widget,
  29. );
  30. Bloc.observer = ApplicationBlocObserver();
  31. runApp(
  32. EasyLocalization(
  33. supportedLocales: const [
  34. // In alphabetical order
  35. Locale('ar', 'SA'),
  36. Locale('ca', 'ES'),
  37. Locale('de', 'DE'),
  38. Locale('en'),
  39. Locale('es', 'VE'),
  40. Locale('eu', 'ES'),
  41. Locale('fr', 'FR'),
  42. Locale('fr', 'CA'),
  43. Locale('hu', 'HU'),
  44. Locale('id', 'ID'),
  45. Locale('it', 'IT'),
  46. Locale('ja', 'JP'),
  47. Locale('ko', 'KR'),
  48. Locale('pl', 'PL'),
  49. Locale('pt', 'BR'),
  50. Locale('ru', 'RU'),
  51. Locale('sv'),
  52. Locale('tr', 'TR'),
  53. Locale('ur'),
  54. Locale('zh', 'CN'),
  55. Locale('zh', 'TW'),
  56. Locale('fa'),
  57. ],
  58. path: 'assets/translations',
  59. fallbackLocale: const Locale('en'),
  60. useFallbackTranslations: true,
  61. saveLocale: false,
  62. child: app,
  63. ),
  64. );
  65. return Future(() => {});
  66. }
  67. }
  68. class ApplicationWidget extends StatelessWidget {
  69. final Widget child;
  70. final AppearanceSettingsPB appearanceSetting;
  71. final AppTheme appTheme;
  72. const ApplicationWidget({
  73. Key? key,
  74. required this.child,
  75. required this.appTheme,
  76. required this.appearanceSetting,
  77. }) : super(key: key);
  78. @override
  79. Widget build(BuildContext context) {
  80. final cubit = AppearanceSettingsCubit(appearanceSetting, appTheme)
  81. ..readLocaleWhenAppLaunch(context);
  82. return MultiBlocProvider(
  83. providers: [
  84. BlocProvider.value(value: cubit),
  85. BlocProvider<DocumentAppearanceCubit>(
  86. create: (_) => DocumentAppearanceCubit()..fetch(),
  87. ),
  88. ],
  89. child: BlocBuilder<AppearanceSettingsCubit, AppearanceSettingsState>(
  90. builder: (context, state) => MaterialApp(
  91. builder: overlayManagerBuilder(),
  92. debugShowCheckedModeBanner: false,
  93. theme: state.lightTheme,
  94. darkTheme: state.darkTheme,
  95. themeMode: state.themeMode,
  96. localizationsDelegates: context.localizationDelegates +
  97. [AppFlowyEditorLocalizations.delegate],
  98. supportedLocales: context.supportedLocales,
  99. locale: state.locale,
  100. navigatorKey: AppGlobals.rootNavKey,
  101. home: child,
  102. ),
  103. ),
  104. );
  105. }
  106. }
  107. class AppGlobals {
  108. static GlobalKey<NavigatorState> rootNavKey = GlobalKey();
  109. static NavigatorState get nav => rootNavKey.currentState!;
  110. }
  111. class ApplicationBlocObserver extends BlocObserver {
  112. @override
  113. // ignore: unnecessary_overrides
  114. void onTransition(Bloc bloc, Transition transition) {
  115. // Log.debug("[current]: ${transition.currentState} \n\n[next]: ${transition.nextState}");
  116. // Log.debug("${transition.nextState}");
  117. super.onTransition(bloc, transition);
  118. }
  119. @override
  120. void onError(BlocBase bloc, Object error, StackTrace stackTrace) {
  121. Log.debug(error);
  122. super.onError(bloc, error, stackTrace);
  123. }
  124. // @override
  125. // void onEvent(Bloc bloc, Object? event) {
  126. // Log.debug("$event");
  127. // super.onEvent(bloc, event);
  128. // }
  129. }
  130. Future<AppTheme> appTheme(String themeName) async {
  131. if (themeName.isEmpty) {
  132. return AppTheme.fallback;
  133. } else {
  134. try {
  135. return await AppTheme.fromName(themeName);
  136. } catch (e) {
  137. Log.error(e);
  138. return AppTheme.fallback;
  139. }
  140. }
  141. }