"use client"; import React, { useState, useRef, useEffect } from "react"; import { ComponentRendererProps } from "@/types/component"; import { ButtonPrimaryConfig } from "./types"; import { ButtonActionExecutor, ButtonActionContext, ButtonActionType, DEFAULT_BUTTON_ACTIONS, } from "@/lib/utils/buttonActions"; import { AlertDialog, AlertDialogAction, AlertDialogCancel, AlertDialogContent, AlertDialogDescription, AlertDialogFooter, AlertDialogHeader, AlertDialogTitle, } from "@/components/ui/alert-dialog"; import { toast } from "sonner"; import { filterDOMProps } from "@/lib/utils/domPropsFilter"; export interface ButtonPrimaryComponentProps extends ComponentRendererProps { config?: ButtonPrimaryConfig; // 추가 props screenId?: number; tableName?: string; onRefresh?: () => void; onClose?: () => void; // 폼 데이터 관련 originalData?: Record; // 부분 업데이트용 원본 데이터 // 테이블 선택된 행 정보 (다중 선택 액션용) selectedRows?: any[]; selectedRowsData?: any[]; } /** * ButtonPrimary 컴포넌트 * button-primary 컴포넌트입니다 */ export const ButtonPrimaryComponent: React.FC = ({ component, isDesignMode = false, isSelected = false, isInteractive = false, onClick, onDragStart, onDragEnd, config, className, style, formData, originalData, onFormDataChange, screenId, tableName, onRefresh, onClose, selectedRows, selectedRowsData, ...props }) => { // 확인 다이얼로그 상태 const [showConfirmDialog, setShowConfirmDialog] = useState(false); const [pendingAction, setPendingAction] = useState<{ type: ButtonActionType; config: any; context: ButtonActionContext; } | null>(null); // 토스트 정리를 위한 ref const currentLoadingToastRef = useRef(); // 컴포넌트 언마운트 시 토스트 정리 useEffect(() => { return () => { if (currentLoadingToastRef.current !== undefined) { console.log("🧹 컴포넌트 언마운트 시 토스트 정리"); toast.dismiss(currentLoadingToastRef.current); currentLoadingToastRef.current = undefined; } }; }, []); // 삭제 액션 감지 로직 (실제 필드명 사용) const isDeleteAction = () => { const deleteKeywords = ['삭제', 'delete', 'remove', '제거', 'del']; return ( component.componentConfig?.action?.type === 'delete' || component.config?.action?.type === 'delete' || component.webTypeConfig?.actionType === 'delete' || component.text?.toLowerCase().includes('삭제') || component.text?.toLowerCase().includes('delete') || component.label?.toLowerCase().includes('삭제') || component.label?.toLowerCase().includes('delete') || deleteKeywords.some(keyword => component.config?.buttonText?.toLowerCase().includes(keyword) || component.config?.text?.toLowerCase().includes(keyword) ) ); }; // 삭제 액션일 때 라벨 색상 자동 설정 useEffect(() => { if (isDeleteAction() && !component.style?.labelColor) { // 삭제 액션이고 라벨 색상이 설정되지 않은 경우 빨간색으로 자동 설정 if (component.style) { component.style.labelColor = '#ef4444'; } else { component.style = { labelColor: '#ef4444' }; } } }, [component.componentConfig?.action?.type, component.config?.action?.type, component.webTypeConfig?.actionType]); // 컴포넌트 설정 const componentConfig = { ...config, ...component.config, } as ButtonPrimaryConfig; // 🎨 동적 색상 설정 (속성편집 모달의 "색상" 필드와 연동) const getLabelColor = () => { if (isDeleteAction()) { return component.style?.labelColor || '#ef4444'; // 빨간색 기본값 (Tailwind red-500) } return component.style?.labelColor || '#3b83f6'; // 기본 파란색 (Tailwind blue-500) }; const buttonColor = getLabelColor(); // 그라데이션용 어두운 색상 계산 const getDarkColor = (baseColor: string) => { const hex = baseColor.replace('#', ''); const r = Math.max(0, parseInt(hex.substr(0, 2), 16) - 40); const g = Math.max(0, parseInt(hex.substr(2, 2), 16) - 40); const b = Math.max(0, parseInt(hex.substr(4, 2), 16) - 40); return `#${r.toString(16).padStart(2, '0')}${g.toString(16).padStart(2, '0')}${b.toString(16).padStart(2, '0')}`; }; const buttonDarkColor = getDarkColor(buttonColor); console.log("🎨 동적 색상 연동:", { labelColor: component.style?.labelColor, buttonColor, buttonDarkColor, }); // 액션 설정 처리 - DB에서 문자열로 저장된 액션을 객체로 변환 const processedConfig = { ...componentConfig }; if (componentConfig.action && typeof componentConfig.action === "string") { const actionType = componentConfig.action as ButtonActionType; processedConfig.action = { ...DEFAULT_BUTTON_ACTIONS[actionType], type: actionType, // 🔥 제어관리 설정 추가 (webTypeConfig에서 가져옴) enableDataflowControl: component.webTypeConfig?.enableDataflowControl, dataflowConfig: component.webTypeConfig?.dataflowConfig, }; } else if (componentConfig.action && typeof componentConfig.action === "object") { // 🔥 이미 객체인 경우에도 제어관리 설정 추가 processedConfig.action = { ...componentConfig.action, enableDataflowControl: component.webTypeConfig?.enableDataflowControl, dataflowConfig: component.webTypeConfig?.dataflowConfig, }; } console.log("🔧 버튼 컴포넌트 설정:", { originalConfig: componentConfig, processedConfig, actionConfig: processedConfig.action, webTypeConfig: component.webTypeConfig, enableDataflowControl: component.webTypeConfig?.enableDataflowControl, dataflowConfig: component.webTypeConfig?.dataflowConfig, screenId, tableName, onRefresh, onClose, selectedRows, selectedRowsData, }); // 스타일 계산 (위치는 RealtimePreviewDynamic에서 처리하므로 제외) const componentStyle: React.CSSProperties = { width: "100%", height: "100%", ...component.style, ...style, }; // 디자인 모드 스타일 if (isDesignMode) { componentStyle.border = "1px dashed #cbd5e1"; componentStyle.borderColor = isSelected ? "#3b82f6" : "#cbd5e1"; } // 확인 다이얼로그가 필요한 액션 타입들 const confirmationRequiredActions: ButtonActionType[] = ["save", "submit", "delete"]; // 실제 액션 실행 함수 const executeAction = async (actionConfig: any, context: ButtonActionContext) => { console.log("🚀 executeAction 시작:", { actionConfig, context }); try { // 기존 토스트가 있다면 먼저 제거 if (currentLoadingToastRef.current !== undefined) { console.log("📱 기존 토스트 제거"); toast.dismiss(currentLoadingToastRef.current); currentLoadingToastRef.current = undefined; } // 추가 안전장치: 모든 로딩 토스트 제거 toast.dismiss(); // edit 액션을 제외하고만 로딩 토스트 표시 if (actionConfig.type !== "edit") { console.log("📱 로딩 토스트 표시 시작"); currentLoadingToastRef.current = toast.loading( actionConfig.type === "save" ? "저장 중..." : actionConfig.type === "delete" ? "삭제 중..." : actionConfig.type === "submit" ? "제출 중..." : "처리 중...", { duration: Infinity, // 명시적으로 무한대로 설정 }, ); console.log("📱 로딩 토스트 ID:", currentLoadingToastRef.current); } console.log("⚡ ButtonActionExecutor.executeAction 호출 시작"); const success = await ButtonActionExecutor.executeAction(actionConfig, context); console.log("⚡ ButtonActionExecutor.executeAction 완료, success:", success); // 로딩 토스트 제거 (있는 경우에만) if (currentLoadingToastRef.current !== undefined) { console.log("📱 로딩 토스트 제거 시도, ID:", currentLoadingToastRef.current); toast.dismiss(currentLoadingToastRef.current); currentLoadingToastRef.current = undefined; } // edit 액션은 조용히 처리 (모달 열기만 하므로 토스트 불필요) if (actionConfig.type !== "edit") { const successMessage = actionConfig.successMessage || (actionConfig.type === "save" ? "저장되었습니다." : actionConfig.type === "delete" ? "삭제되었습니다." : actionConfig.type === "submit" ? "제출되었습니다." : "완료되었습니다."); console.log("🎉 성공 토스트 표시:", successMessage); toast.success(successMessage); } else { console.log("🔕 edit 액션은 조용히 처리 (토스트 없음)"); } console.log("✅ 버튼 액션 실행 성공:", actionConfig.type); } catch (error) { console.log("❌ executeAction catch 블록 진입:", error); // 로딩 토스트 제거 if (currentLoadingToastRef.current !== undefined) { console.log("📱 오류 시 로딩 토스트 제거, ID:", currentLoadingToastRef.current); toast.dismiss(currentLoadingToastRef.current); currentLoadingToastRef.current = undefined; } console.error("❌ 버튼 액션 실행 오류:", error); // 오류 토스트 표시 const errorMessage = actionConfig.errorMessage || (actionConfig.type === "save" ? "저장 중 오류가 발생했습니다." : actionConfig.type === "delete" ? "삭제 중 오류가 발생했습니다." : actionConfig.type === "submit" ? "제출 중 오류가 발생했습니다." : "처리 중 오류가 발생했습니다."); console.log("💥 오류 토스트 표시:", errorMessage); toast.error(errorMessage); } }; // 이벤트 핸들러 const handleClick = async (e: React.MouseEvent) => { e.stopPropagation(); // 디자인 모드에서는 기본 onClick만 실행 if (isDesignMode) { onClick?.(); return; } // 인터랙티브 모드에서 액션 실행 if (isInteractive && processedConfig.action) { const context: ButtonActionContext = { formData: formData || {}, originalData: originalData || {}, // 부분 업데이트용 원본 데이터 추가 screenId, tableName, onFormDataChange, onRefresh, onClose, // 테이블 선택된 행 정보 추가 selectedRows, selectedRowsData, }; // 확인이 필요한 액션인지 확인 if (confirmationRequiredActions.includes(processedConfig.action.type)) { // 확인 다이얼로그 표시 setPendingAction({ type: processedConfig.action.type, config: processedConfig.action, context, }); setShowConfirmDialog(true); } else { // 확인이 필요하지 않은 액션은 바로 실행 await executeAction(processedConfig.action, context); } } else { // 액션이 설정되지 않은 경우 기본 onClick 실행 onClick?.(); } }; // 확인 다이얼로그에서 확인 버튼 클릭 시 const handleConfirmAction = async () => { if (pendingAction) { await executeAction(pendingAction.config, pendingAction.context); } setShowConfirmDialog(false); setPendingAction(null); }; // 확인 다이얼로그에서 취소 버튼 클릭 시 const handleCancelAction = () => { setShowConfirmDialog(false); setPendingAction(null); }; // DOM에 전달하면 안 되는 React-specific props 필터링 const { selectedScreen, onZoneComponentDrop, onZoneClick, componentConfig: _componentConfig, component: _component, isSelected: _isSelected, onClick: _onClick, onDragStart: _onDragStart, onDragEnd: _onDragEnd, size: _size, position: _position, style: _style, screenId: _screenId, tableName: _tableName, onRefresh: _onRefresh, onClose: _onClose, selectedRows: _selectedRows, selectedRowsData: _selectedRowsData, onSelectedRowsChange: _onSelectedRowsChange, originalData: _originalData, // 부분 업데이트용 원본 데이터 필터링 refreshKey: _refreshKey, // 필터링 추가 isInModal: _isInModal, // 필터링 추가 mode: _mode, // 필터링 추가 ...domProps } = props; // 다이얼로그 메시지 생성 const getConfirmMessage = () => { if (!pendingAction) return ""; const customMessage = pendingAction.config.confirmMessage; if (customMessage) return customMessage; switch (pendingAction.type) { case "save": return "변경사항을 저장하시겠습니까?"; case "delete": return "정말로 삭제하시겠습니까? 이 작업은 되돌릴 수 없습니다."; case "submit": return "제출하시겠습니까?"; default: return "이 작업을 실행하시겠습니까?"; } }; const getConfirmTitle = () => { if (!pendingAction) return ""; switch (pendingAction.type) { case "save": return "저장 확인"; case "delete": return "삭제 확인"; case "submit": return "제출 확인"; default: return "작업 확인"; } }; // DOM 안전한 props만 필터링 const safeDomProps = filterDOMProps(domProps); return ( <>
{/* 확인 다이얼로그 */} {getConfirmTitle()} {getConfirmMessage()} 취소 {pendingAction?.type === "save" ? "저장" : pendingAction?.type === "delete" ? "삭제" : pendingAction?.type === "submit" ? "제출" : "확인"} ); }; /** * ButtonPrimary 래퍼 컴포넌트 * 추가적인 로직이나 상태 관리가 필요한 경우 사용 */ export const ButtonPrimaryWrapper: React.FC = (props) => { return ; };