Files
superport/lib/screens/overview/overview_screen_redesign.dart
JiWoong Sul e0bc5894b2 UI 전체 리디자인 및 개선사항 적용
## 주요 변경사항:

### UI/UX 개선
- shadcn/ui 스타일 기반의 새로운 디자인 시스템 도입
- 모든 주요 화면에 대한 리디자인 구현 완료
  - 로그인 화면: 모던한 카드 스타일 적용
  - 대시보드: 통계 카드와 차트를 활용한 개요 화면
  - 리스트 화면들: 일관된 테이블 디자인과 검색/필터 기능
- 다크모드 지원을 위한 테마 시스템 구축

### 기능 개선
- Equipment List: 고급 필터링 (상태, 담당자별)
- Company List: 검색 및 정렬 기능 강화
- User List: 역할별 필터링 추가
- License List: 만료일 기반 상태 표시
- Warehouse Location: 재고 수준 시각화

### 기술적 개선
- 재사용 가능한 컴포넌트 라이브러리 구축
- 일관된 코드 패턴 가이드라인 작성
- 프로젝트 구조 분석 및 문서화

### 문서화
- 프로젝트 분석 문서 추가
- UI 리디자인 진행 상황 문서
- 코드 패턴 가이드 작성
- Equipment 기능 격차 분석 및 구현 계획

### 삭제/리팩토링
- goods_list.dart 제거 (equipment_list로 통합)
- 불필요한 import 및 코드 정리

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-07-07 19:45:32 +09:00

512 lines
16 KiB
Dart

import 'package:flutter/material.dart';
import 'package:superport/screens/common/theme_shadcn.dart';
import 'package:superport/screens/common/components/shadcn_components.dart';
import 'package:superport/screens/overview/controllers/overview_controller.dart';
import 'package:superport/services/mock_data_service.dart';
/// shadcn/ui 스타일로 재설계된 대시보드 화면
class OverviewScreenRedesign extends StatefulWidget {
const OverviewScreenRedesign({Key? key}) : super(key: key);
@override
State<OverviewScreenRedesign> createState() => _OverviewScreenRedesignState();
}
class _OverviewScreenRedesignState extends State<OverviewScreenRedesign> {
late final OverviewController _controller;
bool _isLoading = true;
@override
void initState() {
super.initState();
_controller = OverviewController(dataService: MockDataService());
_loadData();
}
Future<void> _loadData() async {
setState(() {
_isLoading = true;
});
await Future.delayed(const Duration(milliseconds: 800));
_controller.loadData();
setState(() {
_isLoading = false;
});
}
@override
Widget build(BuildContext context) {
if (_isLoading) {
return _buildLoadingState();
}
return Container(
color: ShadcnTheme.background,
child: SingleChildScrollView(
padding: const EdgeInsets.all(24),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// 환영 섹션
ShadcnCard(
padding: const EdgeInsets.all(32),
child: Row(
children: [
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text('안녕하세요, 관리자님! 👋', style: ShadcnTheme.headingH3),
const SizedBox(height: 8),
Text(
'오늘의 포트 운영 현황을 확인해보세요.',
style: ShadcnTheme.bodyMuted,
),
const SizedBox(height: 16),
Row(
children: [
ShadcnBadge(
text: '실시간 모니터링',
variant: ShadcnBadgeVariant.success,
),
const SizedBox(width: 8),
ShadcnBadge(
text: '업데이트됨',
variant: ShadcnBadgeVariant.outline,
),
],
),
],
),
),
],
),
),
const SizedBox(height: 32),
// 통계 카드 그리드 (반응형)
LayoutBuilder(
builder: (context, constraints) {
final crossAxisCount =
constraints.maxWidth > 1200
? 4
: constraints.maxWidth > 800
? 2
: 1;
return GridView.count(
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
crossAxisCount: crossAxisCount,
crossAxisSpacing: 16,
mainAxisSpacing: 16,
childAspectRatio: 1.5,
children: [
_buildStatCard(
'총 회사 수',
'${_controller.totalCompanies}',
Icons.business,
ShadcnTheme.gradient1,
),
_buildStatCard(
'총 사용자 수',
'${_controller.totalUsers}',
Icons.people,
ShadcnTheme.gradient2,
),
_buildStatCard(
'입고 장비',
'${_controller.totalEquipmentIn}',
Icons.inventory,
ShadcnTheme.success,
),
_buildStatCard(
'출고 장비',
'${_controller.totalEquipmentOut}',
Icons.local_shipping,
ShadcnTheme.warning,
),
],
);
},
),
const SizedBox(height: 32),
// 하단 콘텐츠
LayoutBuilder(
builder: (context, constraints) {
if (constraints.maxWidth > 1000) {
// 큰 화면: 가로로 배치
return Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Expanded(flex: 2, child: _buildLeftColumn()),
const SizedBox(width: 24),
Expanded(flex: 1, child: _buildRightColumn()),
],
);
} else {
// 작은 화면: 세로로 배치
return Column(
children: [
_buildLeftColumn(),
const SizedBox(height: 24),
_buildRightColumn(),
],
);
}
},
),
],
),
),
);
}
Widget _buildLoadingState() {
return Container(
color: ShadcnTheme.background,
child: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
CircularProgressIndicator(color: ShadcnTheme.primary),
const SizedBox(height: ShadcnTheme.spacing4),
Text('대시보드를 불러오는 중...', style: ShadcnTheme.bodyMuted),
],
),
),
);
}
Widget _buildLeftColumn() {
return Column(
children: [
// 차트 카드
ShadcnCard(
padding: const EdgeInsets.all(24),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text('월별 활동 현황', style: ShadcnTheme.headingH4),
Text('최근 6개월 데이터', style: ShadcnTheme.bodyMuted),
],
),
ShadcnButton(
text: '상세보기',
onPressed: () {},
variant: ShadcnButtonVariant.ghost,
size: ShadcnButtonSize.small,
),
],
),
const SizedBox(height: 24),
Container(
height: 200,
decoration: BoxDecoration(
color: ShadcnTheme.muted,
borderRadius: BorderRadius.circular(8),
),
child: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
Icons.analytics,
size: 48,
color: ShadcnTheme.mutedForeground,
),
const SizedBox(height: 12),
Text('차트 영역', style: ShadcnTheme.bodyMuted),
Text(
'fl_chart 라이브러리로 구현 예정',
style: ShadcnTheme.bodySmall,
),
],
),
),
),
],
),
),
const SizedBox(height: 24),
// 최근 활동
ShadcnCard(
padding: const EdgeInsets.all(24),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text('최근 활동', style: ShadcnTheme.headingH4),
ShadcnButton(
text: '전체보기',
onPressed: () {},
variant: ShadcnButtonVariant.ghost,
size: ShadcnButtonSize.small,
),
],
),
const SizedBox(height: 16),
...List.generate(5, (index) => _buildActivityItem(index)),
],
),
),
],
);
}
Widget _buildRightColumn() {
return Column(
children: [
// 빠른 작업
ShadcnCard(
padding: const EdgeInsets.all(24),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text('빠른 작업', style: ShadcnTheme.headingH4),
const SizedBox(height: 16),
_buildQuickActionButton(Icons.add_box, '장비 입고', '새 장비 등록'),
const SizedBox(height: 12),
_buildQuickActionButton(
Icons.local_shipping,
'장비 출고',
'장비 대여 처리',
),
const SizedBox(height: 12),
_buildQuickActionButton(
Icons.business_center,
'회사 등록',
'새 회사 추가',
),
],
),
),
const SizedBox(height: 24),
// 시스템 상태
ShadcnCard(
padding: const EdgeInsets.all(24),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text('시스템 상태', style: ShadcnTheme.headingH4),
const SizedBox(height: 16),
_buildStatusItem('서버 상태', '정상'),
_buildStatusItem('데이터베이스', '정상'),
_buildStatusItem('네트워크', '정상'),
_buildStatusItem('백업', '완료'),
],
),
),
],
);
}
Widget _buildStatCard(
String title,
String value,
IconData icon,
Color color,
) {
return ShadcnCard(
padding: const EdgeInsets.all(24),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Container(
padding: const EdgeInsets.all(12),
decoration: BoxDecoration(
color: color.withOpacity(0.1),
borderRadius: BorderRadius.circular(6),
),
child: Icon(icon, color: color, size: 20),
),
Container(
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 4),
decoration: BoxDecoration(
color: ShadcnTheme.success.withOpacity(0.1),
borderRadius: BorderRadius.circular(12),
),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
Icon(
Icons.trending_up,
size: 12,
color: ShadcnTheme.success,
),
const SizedBox(width: 4),
Text(
'+2.3%',
style: ShadcnTheme.labelSmall.copyWith(
color: ShadcnTheme.success,
),
),
],
),
),
],
),
const SizedBox(height: 16),
Text(value, style: ShadcnTheme.headingH2),
const SizedBox(height: 4),
Text(title, style: ShadcnTheme.bodyMedium),
Text('등록된 항목', style: ShadcnTheme.bodySmall),
],
),
);
}
Widget _buildActivityItem(int index) {
final activities = [
{
'icon': Icons.inventory,
'title': '장비 입고 처리',
'subtitle': '크레인 #CR-001 입고 완료',
'time': '2분 전',
},
{
'icon': Icons.local_shipping,
'title': '장비 출고 처리',
'subtitle': '포클레인 #FK-005 출고 완료',
'time': '5분 전',
},
{
'icon': Icons.business,
'title': '회사 등록',
'subtitle': '새로운 회사 "ABC건설" 등록',
'time': '10분 전',
},
{
'icon': Icons.person_add,
'title': '사용자 추가',
'subtitle': '신규 사용자 계정 생성',
'time': '15분 전',
},
{
'icon': Icons.settings,
'title': '시스템 점검',
'subtitle': '정기 시스템 점검 완료',
'time': '30분 전',
},
];
final activity = activities[index];
return Padding(
padding: const EdgeInsets.symmetric(vertical: 8),
child: Row(
children: [
Container(
padding: const EdgeInsets.all(8),
decoration: BoxDecoration(
color: ShadcnTheme.success.withOpacity(0.1),
borderRadius: BorderRadius.circular(6),
),
child: Icon(
activity['icon'] as IconData,
color: ShadcnTheme.success,
size: 16,
),
),
const SizedBox(width: 12),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
activity['title'] as String,
style: ShadcnTheme.bodyMedium,
),
Text(
activity['subtitle'] as String,
style: ShadcnTheme.bodySmall,
),
],
),
),
Text(activity['time'] as String, style: ShadcnTheme.bodySmall),
],
),
);
}
Widget _buildQuickActionButton(IconData icon, String title, String subtitle) {
return GestureDetector(
onTap: () {
// 실제 기능 구현
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text('$title 기능은 개발 중입니다.'),
backgroundColor: ShadcnTheme.info,
),
);
},
child: Container(
width: double.infinity,
padding: const EdgeInsets.all(16),
decoration: BoxDecoration(
border: Border.all(color: ShadcnTheme.border),
borderRadius: BorderRadius.circular(6),
),
child: Row(
children: [
Icon(icon, color: ShadcnTheme.primary),
const SizedBox(width: 12),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(title, style: ShadcnTheme.bodyMedium),
Text(subtitle, style: ShadcnTheme.bodySmall),
],
),
),
Icon(
Icons.arrow_forward_ios,
size: 16,
color: ShadcnTheme.mutedForeground,
),
],
),
),
);
}
Widget _buildStatusItem(String label, String status) {
return Padding(
padding: const EdgeInsets.symmetric(vertical: 8),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(label, style: ShadcnTheme.bodyMedium),
ShadcnBadge(
text: status,
variant: ShadcnBadgeVariant.success,
size: ShadcnBadgeSize.small,
),
],
),
);
}
}