feat(approvals): Approval Flow v2 프런트엔드 전면 개편

- 환경/라우터 모듈에 approval_flow_v2 토글을 추가하고 FeatureFlags 초기화를 연결 (.env*, lib/core/**)
- ApiClient 빌더·ApiRoutes 확장과 ApprovalRepositoryRemote 리팩터링으로 include·액션 시그니처를 정합화
- ApprovalFlow·ApprovalDraft 엔티티/레포/유즈케이스를 도입해 서버 초안과 단계 액션(승인·회수·재상신)을 지원
- Approval 컨트롤러·히스토리·템플릿 페이지와 공유 위젯을 재작성해 감사 로그·회수 UX·템플릿 CRUD를 반영
- Inbound/Outbound/Rental 컨트롤러·페이지에 결재 섹션을 삽입하고 대시보드 pending 카드 요약을 갱신
- SuperportDialog·FormField 등 공통 위젯을 보강하고 승인 위젯 가이드를 추가해 UI 가이드를 정리
- 결재/재고 테스트 픽스처와 단위·위젯·통합 테스트를 확장하고 flutter_test_config로 스테이징 호스트를 허용
- Approval Flow 레포트/플랜 문서를 업데이트하고 ApprovalFlow_System_Integration_and_ChangePlan.md를 추가
- 실행: flutter analyze, flutter test
This commit is contained in:
JiWoong Sul
2025-10-31 01:05:39 +09:00
parent 259b056072
commit d76f765814
133 changed files with 13878 additions and 947 deletions

View File

@@ -0,0 +1,55 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:superport_v2/features/approvals/domain/entities/approval_draft.dart';
void main() {
group('ApprovalDraftSaveInput', () {
test('metadata에 상태 ID를 병합한다', () {
final input = ApprovalDraftSaveInput(
requesterId: 99,
statusId: 7,
metadata: const {'foo': 'bar'},
steps: [ApprovalDraftStep(stepOrder: 1, approverId: 2)],
);
final json = input.toJson();
expect(json['metadata'], isA<Map<String, dynamic>>());
final metadata = json['metadata'] as Map<String, dynamic>;
expect(metadata['_client_state'], isA<Map<String, dynamic>>());
final client = metadata['_client_state'] as Map<String, dynamic>;
expect(client['status_id'], 7);
expect(metadata['foo'], 'bar');
});
});
group('ApprovalDraftDetail', () {
test('toSubmissionInput이 metadata에서 상태 ID를 복원한다', () {
final detail = ApprovalDraftDetail(
id: 1,
requesterId: 10,
savedAt: DateTime.utc(2025, 1, 1),
payload: ApprovalDraftPayload(
templateId: 5,
title: '임시 저장',
metadata: const {
'_client_state': {'status_id': 3},
'note': 'memo',
},
steps: [ApprovalDraftStep(stepOrder: 1, approverId: 42)],
),
);
final submission = detail.toSubmissionInput(defaultStatusId: 1);
expect(submission.requesterId, 10);
expect(submission.statusId, 3);
expect(submission.templateId, 5);
expect(submission.metadata, isA<Map<String, dynamic>>());
final metadata = submission.metadata!;
expect(metadata.containsKey('_client_state'), isFalse);
expect(metadata['note'], 'memo');
expect(submission.steps, hasLength(1));
expect(submission.steps.first.approverId, 42);
});
});
}