## 🔧 주요 수정사항 ### 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>
194 lines
6.0 KiB
Dart
194 lines
6.0 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:superport/utils/constants.dart';
|
|
import 'package:superport/core/extensions/license_expiry_summary_extensions.dart';
|
|
import 'package:superport/data/models/dashboard/license_expiry_summary.dart';
|
|
|
|
/// 라이선스 만료 알림 배너 위젯
|
|
class LicenseExpiryAlert extends StatelessWidget {
|
|
final LicenseExpirySummary summary;
|
|
|
|
const LicenseExpiryAlert({
|
|
super.key,
|
|
required this.summary,
|
|
});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
if (summary.alertLevel == 0) {
|
|
return const SizedBox.shrink(); // 알림이 필요없으면 숨김
|
|
}
|
|
|
|
return Container(
|
|
margin: const EdgeInsets.all(16.0),
|
|
decoration: BoxDecoration(
|
|
color: _getAlertBackgroundColor(summary.alertLevel),
|
|
borderRadius: BorderRadius.circular(8.0),
|
|
border: Border.all(
|
|
color: _getAlertBorderColor(summary.alertLevel),
|
|
width: 1.0,
|
|
),
|
|
),
|
|
child: Material(
|
|
color: Colors.transparent,
|
|
child: InkWell(
|
|
onTap: () => _navigateToLicenses(context),
|
|
borderRadius: BorderRadius.circular(8.0),
|
|
child: Padding(
|
|
padding: const EdgeInsets.all(16.0),
|
|
child: Row(
|
|
children: [
|
|
Icon(
|
|
_getAlertIcon(summary.alertLevel),
|
|
color: _getAlertIconColor(summary.alertLevel),
|
|
size: 24,
|
|
),
|
|
const SizedBox(width: 12),
|
|
Expanded(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(
|
|
_getAlertTitle(summary.alertLevel),
|
|
style: TextStyle(
|
|
fontSize: 16,
|
|
fontWeight: FontWeight.bold,
|
|
color: _getAlertTextColor(summary.alertLevel),
|
|
),
|
|
),
|
|
const SizedBox(height: 4),
|
|
Text(
|
|
summary.alertMessage,
|
|
style: TextStyle(
|
|
fontSize: 14,
|
|
color: _getAlertTextColor(summary.alertLevel).withOpacity(0.8),
|
|
),
|
|
),
|
|
if (summary.alertLevel > 1) ...[
|
|
const SizedBox(height: 8),
|
|
Text(
|
|
'상세 내용을 확인하려면 탭하세요',
|
|
style: TextStyle(
|
|
fontSize: 12,
|
|
fontStyle: FontStyle.italic,
|
|
color: _getAlertTextColor(summary.alertLevel).withOpacity(0.6),
|
|
),
|
|
),
|
|
],
|
|
],
|
|
),
|
|
),
|
|
_buildStatsBadges(),
|
|
const SizedBox(width: 8),
|
|
Icon(
|
|
Icons.arrow_forward_ios,
|
|
size: 16,
|
|
color: _getAlertTextColor(summary.alertLevel).withOpacity(0.6),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
/// 통계 배지들 생성
|
|
Widget _buildStatsBadges() {
|
|
return Row(
|
|
children: [
|
|
if (summary.expired > 0)
|
|
_buildBadge('만료 ${summary.expired}', Colors.red),
|
|
if (summary.expiring7Days > 0)
|
|
_buildBadge('7일 ${summary.expiring7Days}', Colors.orange),
|
|
if (summary.expiring30Days > 0)
|
|
_buildBadge('30일 ${summary.expiring30Days}', Colors.yellow[700]!),
|
|
],
|
|
);
|
|
}
|
|
|
|
/// 개별 배지 생성
|
|
Widget _buildBadge(String text, Color color) {
|
|
return Container(
|
|
margin: const EdgeInsets.only(left: 4),
|
|
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 4),
|
|
decoration: BoxDecoration(
|
|
color: color.withOpacity(0.2),
|
|
borderRadius: BorderRadius.circular(12),
|
|
border: Border.all(color: color.withOpacity(0.5)),
|
|
),
|
|
child: Text(
|
|
text,
|
|
style: TextStyle(
|
|
fontSize: 10,
|
|
fontWeight: FontWeight.bold,
|
|
color: Colors.black87,
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
/// 라이선스 화면으로 이동
|
|
void _navigateToLicenses(BuildContext context) {
|
|
Navigator.pushNamed(context, Routes.licenses);
|
|
}
|
|
|
|
/// 알림 레벨별 배경색
|
|
Color _getAlertBackgroundColor(int level) {
|
|
switch (level) {
|
|
case 3: return Colors.red.shade50;
|
|
case 2: return Colors.orange.shade50;
|
|
case 1: return Colors.yellow.shade50;
|
|
default: return Colors.green.shade50;
|
|
}
|
|
}
|
|
|
|
/// 알림 레벨별 테두리색
|
|
Color _getAlertBorderColor(int level) {
|
|
switch (level) {
|
|
case 3: return Colors.red.shade200;
|
|
case 2: return Colors.orange.shade200;
|
|
case 1: return Colors.yellow.shade200;
|
|
default: return Colors.green.shade200;
|
|
}
|
|
}
|
|
|
|
/// 알림 레벨별 아이콘
|
|
IconData _getAlertIcon(int level) {
|
|
switch (level) {
|
|
case 3: return Icons.error;
|
|
case 2: return Icons.warning;
|
|
case 1: return Icons.info;
|
|
default: return Icons.check_circle;
|
|
}
|
|
}
|
|
|
|
/// 알림 레벨별 아이콘 색상
|
|
Color _getAlertIconColor(int level) {
|
|
switch (level) {
|
|
case 3: return Colors.red.shade600;
|
|
case 2: return Colors.orange.shade600;
|
|
case 1: return Colors.yellow.shade700;
|
|
default: return Colors.green.shade600;
|
|
}
|
|
}
|
|
|
|
/// 알림 레벨별 텍스트 색상
|
|
Color _getAlertTextColor(int level) {
|
|
switch (level) {
|
|
case 3: return Colors.red.shade800;
|
|
case 2: return Colors.orange.shade800;
|
|
case 1: return Colors.yellow.shade800;
|
|
default: return Colors.green.shade800;
|
|
}
|
|
}
|
|
|
|
/// 알림 레벨별 타이틀
|
|
String _getAlertTitle(int level) {
|
|
switch (level) {
|
|
case 3: return '라이선스 만료 위험';
|
|
case 2: return '라이선스 만료 경고';
|
|
case 1: return '라이선스 만료 주의';
|
|
default: return '라이선스 정상';
|
|
}
|
|
}
|
|
} |