"use client"; import { useState, useEffect } from "react"; import { Dialog, DialogContent, DialogHeader, DialogTitle } from "@/components/ui/dialog"; import { Button } from "@/components/ui/button"; import { Input } from "@/components/ui/input"; import { Textarea } from "@/components/ui/textarea"; import { Label } from "@/components/ui/label"; import { Switch } from "@/components/ui/switch"; // import { useCommonCode } from "@/hooks/useCommonCode"; // 제거: 상태 공유 문제 해결 // import { useMultiLang } from "@/hooks/useMultiLang"; // 무한 루프 방지를 위해 임시 제거 import { LoadingSpinner } from "@/components/common/LoadingSpinner"; import { CodeInfo, CreateCodeRequest, UpdateCodeRequest } from "@/types/commonCode"; interface CodeFormModalProps { isOpen: boolean; onClose: () => void; categoryCode: string; editingCode?: CodeInfo | null; // 수정할 코드 객체 (null이면 새 코드) codes: CodeInfo[]; onCreateCode: (categoryCode: string, data: CreateCodeRequest) => Promise; onUpdateCode: (categoryCode: string, codeValue: string, data: UpdateCodeRequest) => Promise; } export function CodeFormModal({ isOpen, onClose, categoryCode, editingCode, codes, onCreateCode, onUpdateCode, }: CodeFormModalProps) { // const { getText } = useMultiLang(); // 무한 루프 방지를 위해 임시 제거 // const { codes, createCode, updateCode } = useCommonCode(); // 제거: props로 전달받음 // 폼 상태 const [formData, setFormData] = useState({ codeValue: "", codeName: "", codeNameEng: "", description: "", sortOrder: 1, isActive: true, }); const [loading, setLoading] = useState(false); const [errors, setErrors] = useState>({}); // 모달 열릴 때 데이터 초기화 useEffect(() => { console.log("🚀 CodeFormModal useEffect 실행:", { isOpen, editingCode, categoryCode, }); if (isOpen) { if (editingCode) { // 수정 모드: 전달받은 코드 데이터 사용 console.log("🔄 수정 모드 - 기존 데이터 로드:", editingCode); setFormData({ codeValue: editingCode.code_value, codeName: editingCode.code_name, codeNameEng: editingCode.code_name_eng || "", description: editingCode.description || "", sortOrder: editingCode.sort_order, isActive: editingCode.is_active === "Y", }); } else { // 새 코드 모드: 초기값 설정 const maxSortOrder = codes.length > 0 ? Math.max(...codes.map((c) => c.sort_order)) : 0; console.log("✨ 새 코드 모드 - 초기값 설정, 다음 순서:", maxSortOrder + 1); setFormData({ codeValue: "", codeName: "", codeNameEng: "", description: "", sortOrder: maxSortOrder + 1, isActive: true, }); } setErrors({}); } }, [isOpen, editingCode, codes, categoryCode]); // 실시간 필드 검증 const validateField = (fieldName: string, value: string) => { const newErrors = { ...errors }; switch (fieldName) { case "codeValue": if (!value.trim()) { newErrors.codeValue = "필수 입력 항목입니다."; } else if (value.length > 50) { newErrors.codeValue = "코드값은 50자 이하로 입력해주세요."; } else if (!/^[A-Z0-9_]+$/.test(value)) { newErrors.codeValue = "대문자, 숫자, 언더스코어(_)만 사용 가능합니다."; } else { delete newErrors.codeValue; } break; case "codeName": if (!value.trim()) { newErrors.codeName = "필수 입력 항목입니다."; } else if (value.length > 100) { newErrors.codeName = "코드명은 100자 이하로 입력해주세요."; } else { delete newErrors.codeName; } break; case "codeNameEng": if (value && value.length > 100) { newErrors.codeNameEng = "영문명은 100자 이하로 입력해주세요."; } else { delete newErrors.codeNameEng; } break; case "description": if (value && value.length > 500) { newErrors.description = "설명은 500자 이하로 입력해주세요."; } else { delete newErrors.description; } break; } setErrors(newErrors); }; // 전체 폼 검증 const validateForm = () => { const newErrors: Record = {}; // 필수 필드 검증 if (!formData.codeValue.trim()) { newErrors.codeValue = "필수 입력 항목입니다."; } else if (!/^[A-Z0-9_]+$/.test(formData.codeValue)) { newErrors.codeValue = "대문자, 숫자, 언더스코어(_)만 사용 가능합니다."; } if (!formData.codeName.trim()) { newErrors.codeName = "필수 입력 항목입니다."; } // 중복 검사 (신규 생성 시) if (!editingCode) { const existingCode = codes.find((c) => c.code_value === formData.codeValue); if (existingCode) { newErrors.codeValue = "이미 존재하는 코드값입니다."; } } setErrors(newErrors); return Object.keys(newErrors).length === 0; }; // 폼 제출 핸들러 const handleSubmit = async (e: React.FormEvent) => { e.preventDefault(); if (!validateForm()) return; setLoading(true); try { if (editingCode) { // 수정 await onUpdateCode(categoryCode, editingCode.code_value, { codeName: formData.codeName, codeNameEng: formData.codeNameEng, description: formData.description, sortOrder: formData.sortOrder, isActive: formData.isActive, }); } else { // 생성 await onCreateCode(categoryCode, { codeValue: formData.codeValue, codeName: formData.codeName, codeNameEng: formData.codeNameEng, description: formData.description, sortOrder: formData.sortOrder, }); } onClose(); } catch (error) { console.error("코드 저장 오류:", error); // 에러 처리는 useCommonCode 훅에서 처리됨 } finally { setLoading(false); } }; // 입력값 변경 핸들러 (실시간 검증 포함) const handleChange = (field: string, value: string | number | boolean) => { setFormData((prev) => ({ ...prev, [field]: value })); // 실시간 검증 (문자열 필드만) if (typeof value === "string") { validateField(field, value); } else { // 에러 제거 (숫자, 불린 필드) if (errors[field]) { setErrors((prev) => ({ ...prev, [field]: "" })); } } }; return ( {editingCode ? "코드 수정" : "새 코드"}
{/* 코드값 */}
handleChange("codeValue", e.target.value.toUpperCase())} disabled={!!editingCode || loading} placeholder={"코드값을 입력하세요 (예: USER_ACTIVE)"} className={errors.codeValue ? "border-red-500" : ""} /> {errors.codeValue &&

{errors.codeValue}

}
{/* 코드명 */}
handleChange("codeName", e.target.value)} disabled={loading} placeholder={"코드명을 입력하세요"} className={errors.codeName ? "border-red-500" : ""} /> {errors.codeName &&

{errors.codeName}

}
{/* 영문명 */}
handleChange("codeNameEng", e.target.value)} disabled={loading} placeholder={"코드 영문명을 입력하세요"} className={errors.codeNameEng ? "border-red-500" : ""} /> {errors.codeNameEng &&

{errors.codeNameEng}

}
{/* 설명 */}