mirror of
https://github.com/smmarty/friflex_flutter_starter.git
synced 2026-02-05 03:32:18 +00:00
139 lines
5.0 KiB
Dart
139 lines
5.0 KiB
Dart
import 'dart:async';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
import 'package:friflex_starter/app/app_env.dart';
|
|
import 'package:friflex_starter/app/app_root.dart';
|
|
import 'package:friflex_starter/di/di_container.dart';
|
|
import 'package:friflex_starter/features/debug/debug_service.dart';
|
|
import 'package:friflex_starter/features/debug/i_debug_service.dart';
|
|
import 'package:friflex_starter/features/error/error_screen.dart';
|
|
import 'package:friflex_starter/runner/timer_runner.dart';
|
|
import 'package:go_router/go_router.dart';
|
|
|
|
part 'errors_handlers.dart';
|
|
|
|
/// Класс, реализующий раннер для конфигурирования приложения при запуске
|
|
///
|
|
/// Порядок инициализации:
|
|
/// 1. _initApp - инициализация конфигурации приложения
|
|
/// 2. инициализация репозиториев приложения (будет позже)
|
|
/// 3. runApp - запуск приложения
|
|
/// 4. _onAppLoaded - после запуска приложения
|
|
class AppRunner {
|
|
/// Создает экземпляр раннера приложения
|
|
///
|
|
/// Принимает:
|
|
/// - [env] - тип окружения сборки приложения
|
|
AppRunner(this.env);
|
|
|
|
/// Тип окружения сборки приложения¬
|
|
final AppEnv env;
|
|
|
|
/// Контейнер зависимостей приложения
|
|
late IDebugService _debugService;
|
|
|
|
/// Роутер приложения
|
|
late GoRouter router;
|
|
|
|
/// Таймер для отслеживания времени инициализации приложения
|
|
late TimerRunner _timerRunner;
|
|
|
|
/// Метод для запуска приложения
|
|
Future<void> run(List<String> arguments) async {
|
|
try {
|
|
WidgetsFlutterBinding.ensureInitialized();
|
|
// Инициализация сервиса отладки
|
|
_debugService = DebugService();
|
|
|
|
_timerRunner = TimerRunner(_debugService);
|
|
|
|
// Инициализация Talker для логирования Bloc
|
|
Bloc.observer = _debugService.blocObserver;
|
|
|
|
// Инициализация приложения
|
|
await _initApp();
|
|
|
|
final diContainer = await _initDependencies(
|
|
debugService: _debugService,
|
|
env: env,
|
|
timerRunner: _timerRunner,
|
|
);
|
|
// Инициализация метода обработки ошибок
|
|
_initErrorHandlers(_debugService);
|
|
runApp(AppRoot(diContainer: diContainer));
|
|
await _onAppLoaded();
|
|
} on Object catch (e, stackTrace) {
|
|
await _onAppLoaded();
|
|
_timerRunner.stop();
|
|
|
|
/// Если произошла ошибка при инициализации приложения,
|
|
/// то запускаем экран ошибки
|
|
runApp(
|
|
ErrorScreen(
|
|
error: e,
|
|
stackTrace: stackTrace,
|
|
onRetry: () => run(arguments),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
/// Метод инициализации приложения,
|
|
/// выполняется до запуска приложения
|
|
Future<void> _initApp() async {
|
|
// Запрет на поворот экрана
|
|
await SystemChrome.setPreferredOrientations([DeviceOrientation.portraitUp]);
|
|
|
|
// Заморозка первого кадра (сплеш)
|
|
WidgetsBinding.instance.deferFirstFrame();
|
|
}
|
|
|
|
/// Метод срабатывает после запуска приложения
|
|
Future<void> _onAppLoaded() async {
|
|
// Разморозка первого кадра (сплеш)
|
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
WidgetsBinding.instance.allowFirstFrame();
|
|
});
|
|
}
|
|
|
|
// Метод для инициализации зависимостей приложения
|
|
Future<DiContainer> _initDependencies({
|
|
required IDebugService debugService,
|
|
required AppEnv env,
|
|
required TimerRunner timerRunner,
|
|
}) async {
|
|
debugService.log(() => 'Тип сборки: ${env.name}');
|
|
final diContainer = DiContainer(env: env, dService: debugService);
|
|
await diContainer
|
|
.init(
|
|
onProgress: (name) => timerRunner.logOnProgress(name),
|
|
onComplete: (name) {
|
|
timerRunner
|
|
..logOnComplete(name)
|
|
..stop();
|
|
},
|
|
onError: (message, error, [stackTrace]) {
|
|
timerRunner.stop();
|
|
_debugService.logError(
|
|
message,
|
|
error: error,
|
|
stackTrace: stackTrace,
|
|
);
|
|
throw Exception('Ошибка инициализации зависимостей: $message');
|
|
},
|
|
)
|
|
.timeout(
|
|
const Duration(seconds: 7),
|
|
onTimeout: () {
|
|
throw Exception(
|
|
'Превышено время ожидания инициализации зависимостей',
|
|
);
|
|
},
|
|
);
|
|
return diContainer;
|
|
}
|
|
}
|