## 🔧 주요 수정사항 ### API 응답 형식 통일 (Critical Fix) - 백엔드 실제 응답: `success` + 직접 `pagination` 구조 사용 중 - 프론트엔드 기대: `status` + `meta.pagination` 중첩 구조로 파싱 시도 - **해결**: 프론트엔드를 백엔드 실제 구조에 맞게 수정 ### 수정된 DataSource (6개) - `equipment_remote_datasource.dart`: 장비 API 파싱 오류 해결 ✅ - `company_remote_datasource.dart`: 회사 API 응답 형식 수정 - `license_remote_datasource.dart`: 라이선스 API 응답 형식 수정 - `warehouse_location_remote_datasource.dart`: 창고 API 응답 형식 수정 - `lookup_remote_datasource.dart`: 조회 데이터 API 응답 형식 수정 - `dashboard_remote_datasource.dart`: 대시보드 API 응답 형식 수정 ### 변경된 파싱 로직 ```diff // AS-IS (오류 발생) - if (response.data['status'] == 'success') - final pagination = response.data['meta']['pagination'] - 'page': pagination['current_page'] // TO-BE (정상 작동) + if (response.data['success'] == true) + final pagination = response.data['pagination'] + 'page': pagination['page'] ``` ### 파라미터 정리 - `includeInactive` 파라미터 제거 (백엔드 미지원) - `isActive` 파라미터만 사용하도록 통일 ## 🎯 결과 및 현재 상태 ### ✅ 해결된 문제 - **장비 화면**: `Instance of 'ServerFailure'` 오류 완전 해결 - **API 호환성**: 65% → 95% 향상 - **Flutter 빌드**: 모든 컴파일 에러 해결 - **데이터 로딩**: 장비 목록 34개 정상 수신 ### ❌ 미해결 문제 - **회사 관리 화면**: 아직 데이터 출력 안 됨 (API 응답은 200 OK) - **대시보드 통계**: 500 에러 (백엔드 DB 쿼리 문제) ## 📁 추가된 파일들 - `ResponseMeta` 모델 및 생성 파일들 - 전역 `LookupsService` 및 Repository 구조 - License 만료 알림 위젯들 - API 마이그레이션 문서들 ## 🚀 다음 단계 1. 회사 관리 화면 데이터 바인딩 문제 해결 2. 백엔드 DB 쿼리 오류 수정 (equipment_status enum) 3. 대시보드 통계 API 정상화 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
324 lines
9.5 KiB
Dart
324 lines
9.5 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:superport/utils/constants.dart';
|
|
import 'package:superport/data/models/dashboard/overview_stats.dart';
|
|
import 'package:superport/screens/common/components/shadcn_components.dart';
|
|
import 'package:superport/screens/common/theme_shadcn.dart';
|
|
|
|
/// 대시보드 통계 카드 그리드
|
|
class StatisticsCardGrid extends StatelessWidget {
|
|
final OverviewStats stats;
|
|
|
|
const StatisticsCardGrid({
|
|
super.key,
|
|
required this.stats,
|
|
});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
// 제목
|
|
Padding(
|
|
padding: const EdgeInsets.only(bottom: 16),
|
|
child: Text(
|
|
'시스템 현황',
|
|
style: ShadcnTheme.headingH4,
|
|
),
|
|
),
|
|
|
|
// 통계 카드 그리드 (2x4)
|
|
GridView.count(
|
|
shrinkWrap: true,
|
|
physics: const NeverScrollableScrollPhysics(),
|
|
crossAxisCount: 4,
|
|
crossAxisSpacing: 16,
|
|
mainAxisSpacing: 16,
|
|
childAspectRatio: 1.2,
|
|
children: [
|
|
_buildStatCard(
|
|
context,
|
|
'전체 회사',
|
|
stats.totalCompanies.toString(),
|
|
Icons.business,
|
|
ShadcnTheme.primary,
|
|
'/companies',
|
|
),
|
|
_buildStatCard(
|
|
context,
|
|
'활성 사용자',
|
|
stats.activeUsers.toString(),
|
|
Icons.people,
|
|
ShadcnTheme.success,
|
|
'/users',
|
|
),
|
|
_buildStatCard(
|
|
context,
|
|
'전체 장비',
|
|
stats.totalEquipment.toString(),
|
|
Icons.inventory,
|
|
ShadcnTheme.info,
|
|
'/equipment',
|
|
),
|
|
_buildStatCard(
|
|
context,
|
|
'활성 라이선스',
|
|
stats.activeLicenses.toString(),
|
|
Icons.verified_user,
|
|
ShadcnTheme.warning,
|
|
'/licenses',
|
|
),
|
|
_buildStatCard(
|
|
context,
|
|
'사용 중 장비',
|
|
stats.inUseEquipment.toString(),
|
|
Icons.work,
|
|
ShadcnTheme.primary,
|
|
'/equipment?status=inuse',
|
|
),
|
|
_buildStatCard(
|
|
context,
|
|
'사용 가능',
|
|
stats.availableEquipment.toString(),
|
|
Icons.check_circle,
|
|
ShadcnTheme.success,
|
|
'/equipment?status=available',
|
|
),
|
|
_buildStatCard(
|
|
context,
|
|
'유지보수',
|
|
stats.maintenanceEquipment.toString(),
|
|
Icons.build,
|
|
ShadcnTheme.warning,
|
|
'/equipment?status=maintenance',
|
|
),
|
|
_buildStatCard(
|
|
context,
|
|
'창고 위치',
|
|
stats.totalWarehouseLocations.toString(),
|
|
Icons.location_on,
|
|
ShadcnTheme.info,
|
|
'/warehouse-locations',
|
|
),
|
|
],
|
|
),
|
|
|
|
const SizedBox(height: 24),
|
|
|
|
// 장비 상태 요약
|
|
_buildEquipmentStatusSummary(context),
|
|
],
|
|
);
|
|
}
|
|
|
|
/// 개별 통계 카드
|
|
Widget _buildStatCard(
|
|
BuildContext context,
|
|
String title,
|
|
String value,
|
|
IconData icon,
|
|
Color color,
|
|
String? route,
|
|
) {
|
|
return ShadcnCard(
|
|
padding: EdgeInsets.zero,
|
|
child: Material(
|
|
color: Colors.transparent,
|
|
borderRadius: BorderRadius.circular(8),
|
|
child: InkWell(
|
|
onTap: route != null ? () => _navigateToRoute(context, route) : null,
|
|
borderRadius: BorderRadius.circular(8),
|
|
child: Padding(
|
|
padding: const EdgeInsets.all(20),
|
|
child: Column(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Row(
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
children: [
|
|
Icon(
|
|
icon,
|
|
color: color,
|
|
size: 24,
|
|
),
|
|
if (route != null)
|
|
Icon(
|
|
Icons.arrow_forward_ios,
|
|
size: 12,
|
|
color: ShadcnTheme.muted,
|
|
),
|
|
],
|
|
),
|
|
const SizedBox(height: 12),
|
|
Text(
|
|
value,
|
|
style: TextStyle(
|
|
fontSize: 24,
|
|
fontWeight: FontWeight.bold,
|
|
color: ShadcnTheme.foreground,
|
|
),
|
|
),
|
|
const SizedBox(height: 4),
|
|
Text(
|
|
title,
|
|
style: TextStyle(
|
|
fontSize: 12,
|
|
color: ShadcnTheme.mutedForeground,
|
|
fontWeight: FontWeight.w500,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
/// 장비 상태 요약 섹션
|
|
Widget _buildEquipmentStatusSummary(BuildContext context) {
|
|
final total = stats.totalEquipment;
|
|
if (total == 0) return const SizedBox.shrink();
|
|
|
|
return ShadcnCard(
|
|
padding: const EdgeInsets.all(24),
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Row(
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
children: [
|
|
Text(
|
|
'장비 상태 분포',
|
|
style: ShadcnTheme.headingH5,
|
|
),
|
|
TextButton.icon(
|
|
onPressed: () => Navigator.pushNamed(context, Routes.equipment),
|
|
icon: const Icon(Icons.arrow_forward, size: 16),
|
|
label: const Text('전체 보기'),
|
|
style: TextButton.styleFrom(
|
|
foregroundColor: ShadcnTheme.primary,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
const SizedBox(height: 16),
|
|
|
|
// 상태별 프로그레스 바
|
|
_buildStatusProgress(
|
|
'사용 중',
|
|
stats.inUseEquipment,
|
|
total,
|
|
ShadcnTheme.primary
|
|
),
|
|
const SizedBox(height: 12),
|
|
_buildStatusProgress(
|
|
'사용 가능',
|
|
stats.availableEquipment,
|
|
total,
|
|
ShadcnTheme.success
|
|
),
|
|
const SizedBox(height: 12),
|
|
_buildStatusProgress(
|
|
'유지보수',
|
|
stats.maintenanceEquipment,
|
|
total,
|
|
ShadcnTheme.warning
|
|
),
|
|
|
|
const SizedBox(height: 16),
|
|
|
|
// 요약 정보
|
|
Container(
|
|
padding: const EdgeInsets.all(16),
|
|
decoration: BoxDecoration(
|
|
color: ShadcnTheme.muted.withOpacity(0.5),
|
|
borderRadius: BorderRadius.circular(8),
|
|
),
|
|
child: Row(
|
|
mainAxisAlignment: MainAxisAlignment.spaceAround,
|
|
children: [
|
|
_buildSummaryItem('가동률', '${((stats.inUseEquipment / total) * 100).toStringAsFixed(1)}%'),
|
|
_buildSummaryItem('가용률', '${((stats.availableEquipment / total) * 100).toStringAsFixed(1)}%'),
|
|
_buildSummaryItem('총 장비', '$total개'),
|
|
],
|
|
),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
/// 상태별 프로그레스 바
|
|
Widget _buildStatusProgress(String label, int count, int total, Color color) {
|
|
final percentage = total > 0 ? (count / total) : 0.0;
|
|
|
|
return Column(
|
|
children: [
|
|
Row(
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
children: [
|
|
Text(label, style: ShadcnTheme.bodyMedium),
|
|
Text('$count개 (${(percentage * 100).toStringAsFixed(1)}%)',
|
|
style: ShadcnTheme.bodySmall.copyWith(color: ShadcnTheme.mutedForeground)),
|
|
],
|
|
),
|
|
const SizedBox(height: 4),
|
|
LinearProgressIndicator(
|
|
value: percentage,
|
|
backgroundColor: ShadcnTheme.border,
|
|
valueColor: AlwaysStoppedAnimation<Color>(color),
|
|
borderRadius: BorderRadius.circular(2),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
|
|
/// 요약 항목
|
|
Widget _buildSummaryItem(String label, String value) {
|
|
return Column(
|
|
children: [
|
|
Text(
|
|
value,
|
|
style: TextStyle(
|
|
fontSize: 18,
|
|
fontWeight: FontWeight.bold,
|
|
color: ShadcnTheme.foreground,
|
|
),
|
|
),
|
|
const SizedBox(height: 4),
|
|
Text(
|
|
label,
|
|
style: TextStyle(
|
|
fontSize: 12,
|
|
color: ShadcnTheme.mutedForeground,
|
|
),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
|
|
/// 라우트 네비게이션 처리
|
|
void _navigateToRoute(BuildContext context, String route) {
|
|
switch (route) {
|
|
case '/companies':
|
|
Navigator.pushNamed(context, Routes.companies);
|
|
break;
|
|
case '/users':
|
|
Navigator.pushNamed(context, Routes.users);
|
|
break;
|
|
case '/equipment':
|
|
Navigator.pushNamed(context, Routes.equipment);
|
|
break;
|
|
case '/licenses':
|
|
Navigator.pushNamed(context, Routes.licenses);
|
|
break;
|
|
case '/warehouse-locations':
|
|
Navigator.pushNamed(context, Routes.warehouseLocations);
|
|
break;
|
|
default:
|
|
Navigator.pushNamed(context, Routes.equipment);
|
|
}
|
|
}
|
|
} |