Files
superport_v2/lib/main.dart
JiWoong Sul 753f76e952 feat(menu-permissions): 메뉴 API 연동으로 사이드바 권한 정비
- .env.development.example과 lib/core/config/environment.dart, lib/core/permissions/permission_manager.dart에서 PERMISSION__ 폴백을 view 전용으로 좁히고 기본 정책을 명시적으로 거부하도록 재정비했다

- lib/core/navigation/*, lib/core/routing/app_router.dart, lib/widgets/app_shell.dart, lib/main.dart에서 메뉴 매니페스트·카탈로그를 도입해 /menus 응답을 캐싱하고 라우터·사이드바·Breadcrumb가 동일 menu_code/route_path를 쓰도록 리팩터링했다

- lib/core/permissions/permission_resources.dart와 그룹 권한/메뉴 마스터 모듈을 menu_code 기반 CRUD 및 Catalog 경로 정합성 검사로 전환하고 PermissionSynchronizer·PermissionBootstrapper를 확장했다

- test/helpers/test_permissions.dart, test/widgets/app_shell_test.dart 등 신규 구조를 반영하는 테스트·골든과 doc/frontend_menu_permission_tasks.md 문서를 보강했다
2025-11-12 18:29:03 +09:00

143 lines
4.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/navigation/menu_catalog.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 'features/masters/menu/domain/repositories/menu_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;
late final MenuCatalog _menuCatalog;
@override
void initState() {
super.initState();
_themeController = ThemeController();
_permissionManager = PermissionManager();
_menuCatalog = MenuCatalog(repository: GetIt.I<MenuRepository>());
if (GetIt.I.isRegistered<PermissionManager>()) {
GetIt.I.unregister<PermissionManager>();
}
GetIt.I.registerSingleton<PermissionManager>(_permissionManager);
if (GetIt.I.isRegistered<MenuCatalog>()) {
GetIt.I.unregister<MenuCatalog>();
}
GetIt.I.registerSingleton<MenuCatalog>(_menuCatalog);
unawaited(_restorePermissions());
unawaited(_preloadMenus());
}
@override
void dispose() {
if (GetIt.I.isRegistered<PermissionManager>()) {
GetIt.I.unregister<PermissionManager>();
}
if (GetIt.I.isRegistered<MenuCatalog>()) {
GetIt.I.unregister<MenuCatalog>();
}
_themeController.dispose();
_permissionManager.dispose();
_menuCatalog.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
return PermissionScope(
manager: _permissionManager,
child: MenuCatalogScope(
catalog: _menuCatalog,
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>(),
menuCatalog: _menuCatalog,
);
await bootstrapper.apply(session);
}
Future<void> _preloadMenus() async {
try {
await _menuCatalog.refresh();
} catch (error, stackTrace) {
debugPrint('메뉴 목록 초기화 실패: $error');
debugPrintStack(stackTrace: stackTrace);
}
}
}