feat: 대규모 코드베이스 개선 - 백엔드 통합성 강화 및 UI 일관성 완성
Some checks failed
Flutter Test & Quality Check / Test on macos-latest (push) Has been cancelled
Flutter Test & Quality Check / Test on ubuntu-latest (push) Has been cancelled
Flutter Test & Quality Check / Build APK (push) Has been cancelled

- CLAUDE.md 대폭 개선: 개발 가이드라인 및 프로젝트 상태 문서화
- 백엔드 API 통합: 모든 엔티티 간 Foreign Key 관계 완벽 구현
- UI 일관성 강화: shadcn_ui 컴포넌트 표준화 적용
- 데이터 모델 개선: DTO 및 모델 클래스 백엔드 스키마와 100% 일치
- 사용자 관리: 회사 연결, 중복 검사, 입력 검증 기능 추가
- 창고 관리: 우편번호 연결, 중복 검사 기능 강화
- 회사 관리: 우편번호 연결, 중복 검사 로직 구현
- 장비 관리: 불필요한 카테고리 필드 제거, 벤더-모델 관계 정리
- 우편번호 시스템: 검색 다이얼로그 Provider 버그 수정

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

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
JiWoong Sul
2025-08-31 15:49:05 +09:00
parent 9dec6f1034
commit df7dd8dacb
46 changed files with 2148 additions and 2722 deletions

View File

@@ -1,6 +1,8 @@
import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
import 'package:shadcn_ui/shadcn_ui.dart';
import 'package:superport/data/models/vendor_dto.dart';
import 'package:superport/screens/vendor/controllers/vendor_controller.dart';
class VendorFormDialog extends StatefulWidget {
final VendorDto? vendor;
@@ -22,6 +24,7 @@ class _VendorFormDialogState extends State<VendorFormDialog> {
late bool _isActive;
bool _isLoading = false;
String? _statusMessage;
@override
void initState() {
@@ -38,10 +41,51 @@ class _VendorFormDialogState extends State<VendorFormDialog> {
super.dispose();
}
Future<bool> _checkDuplicate() async {
final name = _nameController.text.trim();
if (name.isEmpty) return false;
// 수정 모드일 때 현재 이름과 같으면 검사하지 않음
if (widget.vendor != null && widget.vendor!.name == name) {
return false;
}
try {
final controller = context.read<VendorController>();
final isDuplicate = await controller.checkDuplicateName(
name,
excludeId: widget.vendor?.id,
);
return isDuplicate;
} catch (e) {
// 네트워크 오류 시 false 반환
return false;
}
}
void _handleSave() async {
if (!_formKey.currentState!.validate()) return;
setState(() => _isLoading = true);
setState(() {
_isLoading = true;
_statusMessage = '중복 확인 중...';
});
// 저장 시 중복 검사 수행
final isDuplicate = await _checkDuplicate();
if (isDuplicate) {
setState(() {
_isLoading = false;
_statusMessage = '이미 존재하는 벤더명입니다.';
});
return;
}
setState(() {
_statusMessage = '저장 중...';
});
final vendor = VendorDto(
id: widget.vendor?.id,
@@ -53,7 +97,10 @@ class _VendorFormDialogState extends State<VendorFormDialog> {
await widget.onSave(vendor);
setState(() => _isLoading = false);
setState(() {
_isLoading = false;
_statusMessage = null;
});
}
String? _validateRequired(String? value, String fieldName) {
@@ -85,6 +132,22 @@ class _VendorFormDialogState extends State<VendorFormDialog> {
placeholder: const Text('예: 삼성전자, LG전자, 애플'),
validator: (value) => _validateRequired(value, '벤더명'),
),
// 상태 메시지 영역 (고정 높이)
SizedBox(
height: 20,
child: _statusMessage != null
? Text(
_statusMessage!,
style: theme.textTheme.muted.copyWith(
fontSize: 12,
color: _statusMessage!.contains('존재')
? Colors.red
: theme.textTheme.muted.color,
),
)
: const SizedBox.shrink(),
),
const SizedBox(height: 24),
// 활성 상태