feat(dialog): 상세 팝업 SuperportDetailDialog 통합

- SuperportDetailDialog 위젯과 showSuperportDetailDialog 헬퍼를 추가하고 metadata/섹션 패턴을 표준화

- 결재/재고/마스터 각 상세 다이얼로그를 dialogs 디렉터리에 신설하고 기존 페이지를 신규 팝업으로 전환

- SuperportTable 행 선택과 우편번호 검색 다이얼로그 onRowTap 보정을 통해 헤더 오프셋 버그를 제거

- 상세 다이얼로그 및 트랜잭션/상세 뷰 전용 위젯 테스트와 tester_extensions 유틸을 추가하여 회귀를 방지

- detail_dialog_unification_plan.md로 작업 배경과 필드 통합 계획을 문서화
This commit is contained in:
JiWoong Sul
2025-11-07 19:02:43 +09:00
parent 1f78171294
commit 2f8b529506
64 changed files with 13721 additions and 7545 deletions

View File

@@ -14,6 +14,7 @@ import 'package:superport_v2/features/approvals/domain/usecases/save_approval_te
import 'package:superport_v2/features/approvals/template/presentation/pages/approval_template_page.dart';
import 'package:superport_v2/features/approvals/shared/domain/entities/approval_approver_candidate.dart';
import 'package:superport_v2/features/approvals/shared/domain/repositories/approval_approver_repository.dart';
import 'package:superport_v2/features/approvals/shared/widgets/approver_autocomplete_field.dart';
import 'package:superport_v2/features/auth/application/auth_service.dart';
import 'package:superport_v2/features/auth/domain/entities/auth_session.dart';
import 'package:superport_v2/features/auth/domain/entities/authenticated_user.dart';
@@ -169,8 +170,8 @@ void main() {
await tester.pump();
await tester.pumpAndSettle();
expect(find.text('AP_INBOUND'), findsOneWidget);
expect(find.text('입고 템플릿'), findsOneWidget);
expect(find.text('AP_INBOUND'), findsWidgets);
expect(find.text('입고 템플릿'), findsWidgets);
expect(find.textContaining('1. 최승인'), findsOneWidget);
verify(
@@ -198,7 +199,7 @@ void main() {
when(
() => repository.create(any(), steps: any(named: 'steps')),
).thenAnswer((_) async => buildTemplate());
).thenAnswer((_) async => buildTemplate().copyWith(name: '신규 템플릿'));
await tester.pumpWidget(_buildApp(const ApprovalTemplatePage()));
await tester.pump();
@@ -207,43 +208,26 @@ void main() {
await tester.tap(find.text('템플릿 생성'));
await tester.pumpAndSettle();
final dialogFieldsFinder = find.descendant(
of: find.byType(Dialog),
matching: find.byType(EditableText),
skipOffstage: false,
);
final dialogFieldElements = dialogFieldsFinder.evaluate().toList();
expect(dialogFieldElements.length, greaterThanOrEqualTo(4));
await tester.enterText(
find.byWidget(dialogFieldElements[0].widget),
'AP_NEW',
);
await tester.enterText(
find.byWidget(dialogFieldElements[1].widget),
find.byKey(const ValueKey('template_form_name')),
'신규 템플릿',
);
final stepFieldsFinder = find.descendant(
of: find.byKey(const ValueKey('step_field_0')),
matching: find.byType(EditableText),
skipOffstage: false,
final approverField = tester.widget<ApprovalApproverAutocompleteField>(
find.byType(ApprovalApproverAutocompleteField).first,
);
final stepFieldElements = stepFieldsFinder.evaluate().toList();
expect(stepFieldElements.length, greaterThanOrEqualTo(2));
await tester.enterText(find.byWidget(stepFieldElements[1].widget), '33');
await tester.testTextInput.receiveAction(TextInputAction.done);
approverField.idController.text = '33';
await tester.pump();
await tester.tap(find.text('생성 완료'));
final createButtonFinder = find.widgetWithText(ShadButton, '등록').last;
final createButton = tester.widget<ShadButton>(createButtonFinder);
createButton.onPressed?.call();
await tester.pump();
await tester.pumpAndSettle();
verify(
() => repository.create(any(), steps: any(named: 'steps')),
).called(1);
expect(find.text('템플릿 "신규 템플릿"을 생성했습니다.'), findsOneWidget);
expect(find.textContaining('템플릿 "신규 템플릿"'), findsOneWidget);
});
testWidgets('보기 버튼을 눌러 템플릿 단계를 미리본다', (tester) async {
@@ -273,25 +257,25 @@ void main() {
await tester.pump();
await tester.pumpAndSettle();
final previewFinder = find.text('보기', skipOffstage: false);
await tester.dragUntilVisible(
previewFinder,
find.text(template.name),
const Offset(-200, 0),
);
await tester.pumpAndSettle();
await tester.tap(previewFinder);
await tester.tap(find.text(template.name));
await tester.pump();
await tester.pumpAndSettle();
expect(find.text(template.name), findsWidgets);
final detailTabsState =
tester.state(find.byWidgetPredicate((widget) => widget is ShadTabs))
as dynamic;
detailTabsState.controller.select('steps');
await tester.pump();
await tester.pumpAndSettle();
expect(find.textContaining('사번 E001'), findsOneWidget);
verify(
() => repository.fetchDetail(template.id, includeSteps: true),
).called(1);
await tester.tap(find.byTooltip('닫기'));
await tester.pumpAndSettle();
});
testWidgets('수정 플로우에서 fetchDetail 후 update를 호출한다', (tester) async {
@@ -334,31 +318,25 @@ void main() {
repository.list(page: 1, pageSize: 20, query: null, isActive: null),
).called(1);
await tester.dragUntilVisible(
find.text('수정'),
find.text('입고 템플릿'),
const Offset(-200, 0),
);
await tester.pumpAndSettle();
await tester.tap(find.text('수정').first);
await tester.tap(find.text('입고 템플릿'));
await tester.pump();
await tester.pumpAndSettle();
final editDialogFields = find.descendant(
of: find.byType(Dialog),
matching: find.byType(EditableText),
skipOffstage: false,
);
final editFieldElements = editDialogFields.evaluate().toList();
expect(editFieldElements.length, greaterThanOrEqualTo(1));
final tabsState =
tester.state(find.byWidgetPredicate((widget) => widget is ShadTabs))
as dynamic;
tabsState.controller.select('edit');
await tester.pump();
await tester.pumpAndSettle();
await tester.enterText(
find.byWidget(editFieldElements[0].widget),
find.byKey(const ValueKey('template_form_name')),
'수정된 템플릿',
);
await tester.tap(find.text('수정 완료'));
final saveButtonFinder = find.widgetWithText(ShadButton, '저장').last;
final saveButton = tester.widget<ShadButton>(saveButtonFinder);
saveButton.onPressed?.call();
await tester.pump();
await tester.pumpAndSettle();
@@ -367,7 +345,7 @@ void main() {
() => repository.update(10, any(), steps: any(named: 'steps')),
).called(1);
expect(find.text('템플릿 "입고 템플릿"을(를) 수정했습니다.'), findsOneWidget);
expect(find.text('템플릿 "수정된 템플릿"을(를) 수정했습니다.'), findsOneWidget);
});
});
}