feat(WEB): 거래처 DevFill 자동 채우기 기능 추가

- 거래처 샘플 데이터 생성기 추가 (clientData.ts)
- DevFillContext에 'client' 페이지 타입 추가
- DevToolbar에 기준정보 섹션 (녹색 테마) 추가
- IntegratedDetailTemplate forwardRef 지원으로 외부 폼 데이터 조작 가능
- ClientDetailClientV2에서 DevFill 등록 로직 구현
This commit is contained in:
2026-01-22 23:03:45 +09:00
parent 2e9aa74b72
commit 1ba0561ee9
6 changed files with 339 additions and 24 deletions

View File

@@ -17,7 +17,8 @@ import React, { createContext, useContext, useState, useCallback, useEffect, Rea
// 지원하는 페이지 타입
export type DevFillPageType =
| 'quote' | 'order' | 'workOrder' | 'workOrderComplete' | 'shipment' // 판매/생산 플로우
| 'deposit' | 'withdrawal' | 'purchaseApproval' | 'cardTransaction'; // 회계 플로우
| 'deposit' | 'withdrawal' | 'purchaseApproval' | 'cardTransaction' // 회계 플로우
| 'client'; // 기준정보
// 폼 채우기 함수 타입
type FillFormFunction = (data?: unknown) => void | Promise<void>;

View File

@@ -33,6 +33,8 @@ import {
ArrowUpFromLine, // 출금
Receipt, // 매입(지출결의서)
CreditCard, // 카드
// 기준정보 아이콘
Building2, // 거래처
} from 'lucide-react';
import { Button } from '@/components/ui/button';
import { Badge } from '@/components/ui/badge';
@@ -55,6 +57,8 @@ const PAGE_PATTERNS: { pattern: RegExp; type: DevFillPageType; label: string }[]
{ pattern: /\/accounting\/withdrawals\/new/, type: 'withdrawal', label: '출금' },
{ pattern: /\/approval\/draft\/new/, type: 'purchaseApproval', label: '매입' },
{ pattern: /\/accounting\/card-transactions\/new/, type: 'cardTransaction', label: '카드' },
// 기준정보
{ pattern: /\/client-management-sales-admin\/new/, type: 'client', label: '거래처' },
];
// 플로우 단계 정의
@@ -74,6 +78,11 @@ const ACCOUNTING_STEPS: { type: DevFillPageType; label: string; icon: typeof Fil
{ type: 'cardTransaction', label: '카드', icon: CreditCard, path: '/accounting/card-transactions/new', fillEnabled: true },
];
// 기준정보 단계 정의
const MASTER_DATA_STEPS: { type: DevFillPageType; label: string; icon: typeof FileText; path: string; fillEnabled: boolean }[] = [
{ type: 'client', label: '거래처', icon: Building2, path: '/sales/client-management-sales-admin/new', fillEnabled: true },
];
export function DevToolbar() {
const pathname = usePathname();
const router = useRouter();
@@ -324,11 +333,61 @@ export function DevToolbar() {
</div>
)}
{/* 기준정보 버튼 영역 */}
{isExpanded && (
<div className="flex items-center gap-2 px-3 pb-3 border-t border-yellow-300 pt-3">
<span className="text-xs text-yellow-600 font-medium mr-1">:</span>
{MASTER_DATA_STEPS.map((step) => {
const Icon = step.icon;
const isActive = activePage === step.type;
const isRegistered = hasRegisteredForm(step.type);
const isCurrentLoading = isLoading === step.type;
// 활성화된 페이지: 폼 채우기 (fillEnabled가 true인 경우만)
if (isActive && step.fillEnabled) {
return (
<Button
key={step.type}
size="sm"
variant="default"
disabled={!isRegistered || isCurrentLoading}
className="bg-green-500 hover:bg-green-600 text-white border-green-600"
onClick={() => handleFillForm(step.type)}
title="폼 자동 채우기"
>
{isCurrentLoading ? (
<Loader2 className="w-4 h-4 mr-1 animate-spin" />
) : (
<Icon className="w-4 h-4 mr-1" />
)}
{step.label}
</Button>
);
}
// 비활성화된 페이지: 해당 페이지로 이동
return (
<Button
key={step.type}
size="sm"
variant="outline"
className={`border-green-300 text-green-700 hover:bg-green-100 hover:border-green-500 ${isActive ? 'bg-green-100 border-green-500' : ''}`}
onClick={() => handleNavigate(step.path)}
title={`${step.label} 페이지로 이동`}
>
<Icon className="w-4 h-4 mr-1" />
{step.label}
</Button>
);
})}
</div>
)}
{/* 안내 메시지 */}
{isExpanded && !activePage && (
<div className="px-3 pb-3">
<p className="text-xs text-yellow-600">
//////
///////
</p>
</div>
)}

View File

@@ -0,0 +1,165 @@
/**
* 거래처 샘플 데이터 생성기
*/
import {
randomPick,
randomInt,
randomPhone,
} from './index';
// ===== 상수 정의 =====
// 회사명 접두어
const COMPANY_PREFIXES = [
'(주)', '주식회사 ', '', '(유)', '유한회사 ',
];
// 회사명
const COMPANY_NAMES = [
'대한철강', '신성산업', '한국정밀', '동양기계', '서울금속',
'부산화학', '인천전자', '광주물산', '대구섬유', '울산중공업',
'경기식품', '강원목재', '충청농산', '전북수산', '제주관광',
'삼성전자', 'LG화학', '현대중공업', 'SK하이닉스', '포스코',
'한화솔루션', '롯데케미칼', 'CJ대한통운', 'GS칼텍스', 'S-Oil',
];
// 대표자 성
const LAST_NAMES = ['김', '이', '박', '최', '정', '강', '조', '윤', '장', '임', '한', '오', '서', '신', '권', '황', '안', '송', '류', '홍'];
// 대표자 이름
const FIRST_NAMES = ['영수', '민수', '철수', '영희', '민희', '수진', '지영', '성호', '현우', '준호', '서연', '지민', '하늘', '도윤', '서준', '예준', '민준', '지후', '수아', '하윤'];
// 업태
const BUSINESS_TYPES = [
'제조업', '도소매업', '건설업', '서비스업', '운수업',
'정보통신업', '금융업', '부동산업', '전문서비스업', '교육서비스업',
];
// 종목
const BUSINESS_ITEMS = [
'철강', '기계', '전자부품', '화학제품', '섬유',
'식품', '자동차부품', '건설자재', '플라스틱', '금속가공',
'소프트웨어', '컨설팅', '물류', '무역', '반도체',
];
// 거래처 유형
const CLIENT_TYPES = ['매입', '매출', '매입매출'] as const;
// 주소 - 시도
const CITIES = ['서울', '부산', '인천', '대구', '광주', '대전', '울산', '세종', '경기', '강원', '충북', '충남', '전북', '전남', '경북', '경남', '제주'];
// 주소 - 구/시
const DISTRICTS = ['강남구', '서초구', '송파구', '중구', '종로구', '영등포구', '마포구', '성동구', '동작구', '관악구'];
// 주소 - 상세
const STREET_NAMES = ['테헤란로', '강남대로', '영동대로', '올림픽로', '삼성로', '선릉로', '봉은사로', '도산대로', '언주로', '논현로'];
// 담당자명
const MANAGER_NAMES = ['김담당', '이과장', '박대리', '최주임', '정사원', '강매니저', '조팀장', '윤실장', '장차장', '임부장'];
// 메모
const MEMOS = [
'우량 거래처',
'장기 거래 예정',
'결제 조건 협의 필요',
'월 정기 발주 거래처',
'신규 거래처 - 신용 확인 필요',
'현금 결제 선호',
'분기별 결제',
'',
];
// ===== 유틸리티 함수 =====
// 랜덤 사업자등록번호 생성 (형식: 000-00-00000)
function generateBusinessNo(): string {
const part1 = String(randomInt(100, 999));
const part2 = String(randomInt(10, 99));
const part3 = String(randomInt(10000, 99999));
return `${part1}-${part2}-${part3}`;
}
// 랜덤 이름 생성
function generateName(): string {
return randomPick(LAST_NAMES) + randomPick(FIRST_NAMES);
}
// 랜덤 회사명 생성
function generateCompanyName(): string {
return randomPick(COMPANY_PREFIXES) + randomPick(COMPANY_NAMES);
}
// 랜덤 주소 생성
function generateAddress(): string {
const city = randomPick(CITIES);
const district = randomPick(DISTRICTS);
const street = randomPick(STREET_NAMES);
const buildingNo = randomInt(1, 500);
return `${city} ${district} ${street} ${buildingNo}`;
}
// 랜덤 이메일 생성
function generateEmail(companyName: string): string {
const domains = ['company.com', 'corp.co.kr', 'business.com', 'enterprise.kr', 'inc.co.kr'];
// 회사명에서 영문/숫자만 추출하거나 기본값 사용
const cleanName = companyName.replace(/[^a-zA-Z0-9]/g, '').toLowerCase() || 'contact';
const prefix = cleanName.slice(0, 10) || 'info';
return `${prefix}@${randomPick(domains)}`;
}
// 랜덤 팩스번호 생성
function generateFax(): string {
const areaCode = randomPick(['02', '031', '032', '051', '053', '062', '042', '052']);
const middle = randomInt(100, 999);
const last = randomInt(1000, 9999);
return `${areaCode}-${middle}-${last}`;
}
// ===== 타입 정의 =====
export interface ClientFormData {
businessNo: string;
clientCode: string;
name: string;
representative: string;
clientType: '매입' | '매출' | '매입매출';
businessType: string;
businessItem: string;
address: string;
phone: string;
mobile: string;
fax: string;
email: string;
managerName: string;
managerTel: string;
systemManager: string;
memo: string;
isActive: string; // 'true' | 'false'
}
// ===== 메인 생성 함수 =====
export function generateClientData(): ClientFormData {
const companyName = generateCompanyName();
return {
businessNo: generateBusinessNo(),
clientCode: '', // 자동 생성되므로 빈 값
name: companyName,
representative: generateName(),
clientType: randomPick(CLIENT_TYPES),
businessType: randomPick(BUSINESS_TYPES),
businessItem: randomPick(BUSINESS_ITEMS),
address: generateAddress(),
phone: generateFax(), // 회사 전화는 지역번호 형식
mobile: randomPhone(),
fax: generateFax(),
email: generateEmail(companyName),
managerName: randomPick(MANAGER_NAMES),
managerTel: randomPhone(),
systemManager: randomPick(MANAGER_NAMES),
memo: randomPick(MEMOS),
isActive: 'true',
};
}