결재 단계 목록 화면과 테스트 도입

This commit is contained in:
JiWoong Sul
2025-09-25 16:41:22 +09:00
parent 86d3f5bf21
commit 35b9002688
10 changed files with 981 additions and 24 deletions

View File

@@ -0,0 +1,76 @@
import '../../../../core/common/models/paginated_result.dart';
import '../../../domain/entities/approval.dart';
import '../../data/dtos/approval_dto.dart';
import '../../domain/entities/approval_step_record.dart';
class ApprovalStepRecordDto {
ApprovalStepRecordDto({
required this.approvalId,
required this.approvalNo,
this.transactionNo,
this.templateName,
required this.step,
});
final int approvalId;
final String approvalNo;
final String? transactionNo;
final String? templateName;
final ApprovalStep step;
factory ApprovalStepRecordDto.fromJson(Map<String, dynamic> json) {
final approvalData = json['approval'] as Map<String, dynamic>?;
final approvalId =
json['approval_id'] as int? ?? approvalData?['id'] as int? ?? 0;
final approvalNo =
json['approval_no'] as String? ??
approvalData?['approval_no'] as String? ??
approvalData?['approvalNo'] as String? ??
'-';
final transactionNo =
json['transaction_no'] as String? ??
approvalData?['transaction_no'] as String? ??
approvalData?['transactionNo'] as String?;
final templateName =
json['template_name'] as String? ??
approvalData?['template_name'] as String? ??
approvalData?['templateName'] as String?;
final step = ApprovalStepDto.fromJson(json).toEntity();
return ApprovalStepRecordDto(
approvalId: approvalId,
approvalNo: approvalNo,
transactionNo: transactionNo,
templateName: templateName,
step: step,
);
}
ApprovalStepRecord toEntity() {
return ApprovalStepRecord(
approvalId: approvalId,
approvalNo: approvalNo,
transactionNo: transactionNo,
templateName: templateName,
step: step,
);
}
static PaginatedResult<ApprovalStepRecord> parsePaginated(
Map<String, dynamic>? json,
) {
final items = (json?['items'] as List<dynamic>? ?? [])
.whereType<Map<String, dynamic>>()
.map(ApprovalStepRecordDto.fromJson)
.map((dto) => dto.toEntity())
.toList();
return PaginatedResult<ApprovalStepRecord>(
items: items,
page: json?['page'] as int? ?? 1,
pageSize: json?['page_size'] as int? ?? items.length,
total: json?['total'] as int? ?? items.length,
);
}
}

View File

@@ -0,0 +1,51 @@
import 'package:dio/dio.dart';
import '../../../../core/common/models/paginated_result.dart';
import '../../../../core/network/api_client.dart';
import '../../domain/entities/approval_step_record.dart';
import '../../domain/repositories/approval_step_repository.dart';
import '../dtos/approval_step_record_dto.dart';
class ApprovalStepRepositoryRemote implements ApprovalStepRepository {
ApprovalStepRepositoryRemote({required ApiClient apiClient})
: _api = apiClient;
final ApiClient _api;
static const _basePath = '/approval-steps';
@override
Future<PaginatedResult<ApprovalStepRecord>> list({
int page = 1,
int pageSize = 20,
String? query,
int? statusId,
int? approverId,
int? approvalId,
}) async {
final response = await _api.get<Map<String, dynamic>>(
_basePath,
query: {
'page': page,
'page_size': pageSize,
if (query != null && query.isNotEmpty) 'q': query,
if (statusId != null) 'status_id': statusId,
if (approverId != null) 'approver_id': approverId,
if (approvalId != null) 'approval_id': approvalId,
},
options: Options(responseType: ResponseType.json),
);
return ApprovalStepRecordDto.parsePaginated(response.data);
}
@override
Future<ApprovalStepRecord> fetchDetail(int id) async {
final response = await _api.get<Map<String, dynamic>>(
'$_basePath/$id',
options: Options(responseType: ResponseType.json),
);
final data = (response.data?['data'] as Map<String, dynamic>?) ?? const {};
return ApprovalStepRecordDto.fromJson(data).toEntity();
}
}