- .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 문서를 보강했다
143 lines
4.7 KiB
Dart
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);
|
|
}
|
|
}
|
|
}
|