fix(inventory): 상세 편집 플로우 안정화

- inbound/outbound/rental controller에 fetchTransactionDetail을 추가해 상세 동기화를 지원

- 각 페이지 초기화 시 결재 초안 로딩 권한을 PermissionScope에서 확인하도록 수정

- 상세 패널의 수정 버튼이 모달과 연동되도록 흐름을 정리하고 생성/수정 후 상세 데이터를 재조회

- 기존 결재 메모 필드는 등록 이후 수정 불가하도록 UI와 입력 상태를 비활성화

- 신규 상세-수정 위젯 테스트와 리포지토리 스텁 fetchDetail 구현을 추가

- flutter analyze, flutter test를 실행해 회귀를 점검
This commit is contained in:
JiWoong Sul
2025-11-11 16:28:49 +09:00
parent 04c6bc9a2e
commit d603fd5c17
10 changed files with 354 additions and 38 deletions

View File

@@ -207,4 +207,46 @@ void main() {
expect(find.textContaining('자동완성에서 선택'), findsOneWidget);
expect(find.textContaining('창고를 선택'), findsOneWidget);
});
testWidgets('입고 상세의 수정 버튼은 수정 모달을 연다', (tester) async {
final view = tester.view;
view.physicalSize = const Size(1280, 900);
view.devicePixelRatio = 1.0;
addTearDown(() {
view.resetPhysicalSize();
view.resetDevicePixelRatio();
});
await tester.pumpWidget(
MaterialApp(
home: ScaffoldMessenger(
child: PermissionScope(
manager: PermissionManager(),
child: ShadTheme(
data: SuperportShadTheme.light(),
child: Scaffold(
body: InboundPage(routeUri: Uri.parse('/inventory/inbound')),
),
),
),
),
),
);
await tester.pumpAndSettle();
await tester.tap(find.text('TX-20240301-001').first);
await tester.pumpAndSettle();
expect(find.text('입고 상세'), findsOneWidget);
final editButton = find.widgetWithText(ShadButton, '수정').last;
await tester.ensureVisible(editButton);
await tester.tap(editButton);
await tester.pump();
await tester.pumpAndSettle(const Duration(milliseconds: 500));
expect(find.text('입고 수정'), findsOneWidget);
expect(find.widgetWithText(ShadButton, '저장'), findsWidgets);
});
}

View File

@@ -66,4 +66,45 @@ void main() {
expect(find.textContaining('자동완성에서 선택'), findsOneWidget);
expect(find.text('창고를 선택하세요.'), findsOneWidget);
});
testWidgets('출고 상세의 수정 버튼은 수정 모달을 연다', (tester) async {
final view = tester.view;
view.physicalSize = const Size(1280, 900);
view.devicePixelRatio = 1.0;
addTearDown(() {
view.resetPhysicalSize();
view.resetDevicePixelRatio();
});
await tester.pumpWidget(
MaterialApp(
home: ScaffoldMessenger(
child: PermissionScope(
manager: PermissionManager(),
child: ShadTheme(
data: SuperportShadTheme.light(),
child: Scaffold(
body: OutboundPage(routeUri: Uri.parse('/inventory/outbound')),
),
),
),
),
),
);
await tester.pumpAndSettle();
await tester.tap(find.text('TX-20240302-010').first);
await tester.pumpAndSettle();
expect(find.text('출고 상세'), findsOneWidget);
final editButton = find.widgetWithText(ShadButton, '수정').last;
await tester.ensureVisible(editButton);
await tester.tap(editButton);
await tester.pump();
await tester.pumpAndSettle(const Duration(milliseconds: 500));
expect(find.text('출고 수정'), findsOneWidget);
expect(find.widgetWithText(ShadButton, '저장'), findsWidgets);
});
}

View File

@@ -66,4 +66,45 @@ void main() {
expect(find.textContaining('자동완성에서 선택'), findsOneWidget);
expect(find.text('최소 1개의 고객사를 선택하세요.'), findsOneWidget);
});
testWidgets('대여 상세의 수정 버튼은 수정 모달을 연다', (tester) async {
final view = tester.view;
view.physicalSize = const Size(1280, 900);
view.devicePixelRatio = 1.0;
addTearDown(() {
view.resetPhysicalSize();
view.resetDevicePixelRatio();
});
await tester.pumpWidget(
MaterialApp(
home: ScaffoldMessenger(
child: PermissionScope(
manager: PermissionManager(),
child: ShadTheme(
data: SuperportShadTheme.light(),
child: Scaffold(
body: RentalPage(routeUri: Uri.parse('/inventory/rental')),
),
),
),
),
),
);
await tester.pumpAndSettle();
await tester.tap(find.text('TX-20240305-030').first);
await tester.pumpAndSettle();
expect(find.text('대여 상세'), findsOneWidget);
final editButton = find.widgetWithText(ShadButton, '수정').last;
await tester.ensureVisible(editButton);
await tester.tap(editButton);
await tester.pump();
await tester.pumpAndSettle(const Duration(milliseconds: 500));
expect(find.text('대여 수정'), findsOneWidget);
expect(find.widgetWithText(ShadButton, '저장'), findsWidgets);
});
}