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,125 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:superport_v2/features/inventory/transactions/data/dtos/stock_transaction_dto.dart';
void main() {
group('StockTransactionDto', () {
test('결재 상세 정보를 포함해 파싱한다', () {
final dto = StockTransactionDto.fromJson({
'id': 9001,
'transaction_no': 'TRX-202511100001',
'transaction_date': '2025-09-18',
'transaction_type': {'id': 1, 'name': '입고'},
'transaction_status': {'id': 1, 'name': '초안'},
'warehouse': {
'id': 1,
'warehouse_code': 'WH-001',
'warehouse_name': '1센터',
},
'created_by': {
'id': 7,
'employee_no': 'E20250001',
'employee_name': '김상신',
},
'lines': [
{
'id': 12001,
'line_no': 1,
'product': {
'id': 101,
'product_code': 'P100',
'product_name': '샘플',
},
'quantity': 10,
'unit_price': 1200,
},
],
'customers': [],
'approval': {
'id': 5001,
'approval_no': 'APP-202511100001',
'status': {
'id': 1,
'name': '대기',
'is_blocking_next': true,
'is_terminal': false,
},
'current_step': {
'id': 7001,
'step_order': 1,
'status': {
'id': 2,
'name': '진행중',
'is_blocking_next': true,
'is_terminal': false,
},
'approver': {'id': 21, 'employee_no': 'E2025002', 'name': '박검토'},
'assigned_at': '2025-09-18T06:05:00Z',
},
'requester': {'id': 7, 'employee_no': 'E20250001', 'name': '김상신'},
'requested_at': '2025-09-18T06:00:00Z',
'note': '입고 결재',
'template_name': '입고 결재 기본',
'steps': [
{
'id': 7201,
'step_order': 1,
'status': {
'id': 3,
'name': '승인',
'is_blocking_next': false,
'is_terminal': false,
},
'approver': {'id': 21, 'employee_no': 'E2025002', 'name': '박검토'},
'assigned_at': '2025-09-18T06:05:00Z',
'decided_at': '2025-09-18T06:10:00Z',
},
{
'id': 7202,
'step_order': 2,
'status': {
'id': 1,
'name': '대기',
'is_blocking_next': true,
'is_terminal': false,
},
'approver': {'id': 22, 'employee_no': 'E2025003', 'name': '이승인'},
'assigned_at': '2025-09-18T06:10:00Z',
},
],
'histories': [
{
'id': 93001,
'action': {'id': 1, 'name': '상신'},
'to_status': {
'id': 1,
'name': '대기',
'is_blocking_next': true,
'is_terminal': false,
},
'approver': {'id': 7, 'employee_no': 'E20250001', 'name': '김상신'},
'action_at': '2025-09-18T06:00:00Z',
},
],
'created_at': '2025-09-18T06:00:00Z',
'updated_at': '2025-09-18T06:05:00Z',
},
});
final entity = dto.toEntity();
expect(entity.approval, isNotNull);
final approval = entity.approval!;
expect(approval.approvalNo, 'APP-202511100001');
expect(approval.status.name, '대기');
expect(approval.currentStep?.stepOrder, 1);
expect(approval.currentStep?.approver.name, '박검토');
expect(approval.steps.length, 2);
expect(approval.histories.length, 1);
expect(approval.requester.name, '김상신');
expect(
approval.requestedAt.toUtc().toIso8601String(),
'2025-09-18T06:00:00.000Z',
);
});
});
}