Files
superport_v2/lib/main.dart
JiWoong Sul 47cc62a33d feat(inventory): 재고 현황 요약/상세 플로우를 릴리스
- lib/features/inventory/summary 계층과 warehouse select 위젯을 추가해 목록/상세, 자동 새로고침, 필터, 상세 시트를 구현

- PermissionBootstrapper, scope 파서, 라우트 가드로 inventory.view 기반 권한 부여와 메뉴 노출을 통합(lib/core, lib/main.dart 등)

- Inventory Summary API/QA/Audit 문서와 PR 템플릿, CHANGELOG를 신규 스펙과 검증 커맨드로 업데이트

- DTO 직렬화 의존성을 추가하고 Golden·Widget·단위 테스트를 작성했으며 flutter analyze / flutter test --coverage를 통과
2025-11-09 01:13:10 +09:00

114 lines
3.7 KiB
Dart

import 'dart:async';
import 'package:flutter/material.dart';
import 'package:flutter_localizations/flutter_localizations.dart';
import 'package:get_it/get_it.dart';
import 'package:shadcn_ui/shadcn_ui.dart';
import 'core/config/environment.dart';
import 'core/permissions/permission_bootstrapper.dart';
import 'core/permissions/permission_manager.dart';
import 'core/routing/app_router.dart';
import 'core/theme/superport_shad_theme.dart';
import 'core/theme/theme_controller.dart';
import 'features/auth/application/auth_service.dart';
import 'features/masters/group/domain/repositories/group_repository.dart';
import 'features/masters/group_permission/domain/repositories/group_permission_repository.dart';
import 'injection_container.dart';
/// Superport 애플리케이션 진입점. 환경 초기화 후 앱 위젯을 실행한다.
Future<void> main() async {
WidgetsFlutterBinding.ensureInitialized();
await Environment.initialize();
await initInjection(baseUrl: Environment.baseUrl);
final authService = GetIt.I<AuthService>();
try {
await authService.refreshSession();
} catch (error, stackTrace) {
// 초기 자동 로그인 갱신이 실패하면 세션을 정리하고 로그인 화면으로 진입한다.
debugPrint('세션 갱신 실패: $error');
debugPrintStack(stackTrace: stackTrace);
await authService.clearSession();
}
runApp(const SuperportApp());
}
/// 전체 앱을 구성하는 루트 위젯.
class SuperportApp extends StatefulWidget {
const SuperportApp({super.key});
@override
State<SuperportApp> createState() => _SuperportAppState();
}
/// 테마/권한 스코프를 초기화하고 라우터를 구성한다.
class _SuperportAppState extends State<SuperportApp> {
late final ThemeController _themeController;
late final PermissionManager _permissionManager;
@override
void initState() {
super.initState();
_themeController = ThemeController();
_permissionManager = PermissionManager();
if (GetIt.I.isRegistered<PermissionManager>()) {
GetIt.I.unregister<PermissionManager>();
}
GetIt.I.registerSingleton<PermissionManager>(_permissionManager);
unawaited(_restorePermissions());
}
@override
void dispose() {
if (GetIt.I.isRegistered<PermissionManager>()) {
GetIt.I.unregister<PermissionManager>();
}
_themeController.dispose();
_permissionManager.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
return PermissionScope(
manager: _permissionManager,
child: ThemeControllerScope(
controller: _themeController,
child: AnimatedBuilder(
animation: _themeController,
builder: (context, _) {
return ShadApp.router(
title: 'Superport v2',
routerConfig: appRouter,
debugShowCheckedModeBanner: false,
supportedLocales: const [Locale('ko', 'KR'), Locale('en', 'US')],
localizationsDelegates: const [
GlobalMaterialLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
],
theme: SuperportShadTheme.light(),
darkTheme: SuperportShadTheme.dark(),
themeMode: _themeController.mode,
);
},
),
),
);
}
Future<void> _restorePermissions() async {
final authService = GetIt.I<AuthService>();
final session = authService.session;
if (session == null) {
return;
}
final bootstrapper = PermissionBootstrapper(
manager: _permissionManager,
groupRepository: GetIt.I<GroupRepository>(),
groupPermissionRepository: GetIt.I<GroupPermissionRepository>(),
);
await bootstrapper.apply(session);
}
}