ERP-node/frontend/lib/registry/components/split-panel-layout/SplitPanelLayoutConfigPanel...

1514 lines
64 KiB
TypeScript

"use client";
import React, { useState, useMemo, useEffect } from "react";
import { Label } from "@/components/ui/label";
import { Input } from "@/components/ui/input";
import { Switch } from "@/components/ui/switch";
import { Checkbox } from "@/components/ui/checkbox";
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select";
import { Slider } from "@/components/ui/slider";
import { Command, CommandEmpty, CommandGroup, CommandInput, CommandItem, CommandList } from "@/components/ui/command";
import { Popover, PopoverContent, PopoverTrigger } from "@/components/ui/popover";
import { Button } from "@/components/ui/button";
// Accordion 제거 - 단순 섹션으로 변경
import { Check, ChevronsUpDown, X, Database, GripVertical } from "lucide-react";
import { cn } from "@/lib/utils";
import { SplitPanelLayoutConfig } from "./types";
import { TableInfo, ColumnInfo } from "@/types/screen";
import { tableTypeApi } from "@/lib/api/screen";
import { DataFilterConfigPanel } from "@/components/screen/config-panels/DataFilterConfigPanel";
interface SplitPanelLayoutConfigPanelProps {
config: SplitPanelLayoutConfig;
onChange: (config: SplitPanelLayoutConfig) => void;
tables?: TableInfo[]; // 전체 테이블 목록 (선택적)
screenTableName?: string; // 현재 화면의 테이블명 (좌측 패널에서 사용)
menuObjid?: number; // 🆕 메뉴 OBJID (카테고리 값 조회 시 필요)
}
/**
* 그룹핑 기준 컬럼 선택 컴포넌트
*/
const GroupByColumnsSelector: React.FC<{
tableName?: string;
selectedColumns: string[];
onChange: (columns: string[]) => void;
}> = ({ tableName, selectedColumns, onChange }) => {
const [columns, setColumns] = useState<any[]>([]); // ColumnTypeInfo 타입
const [loading, setLoading] = useState(false);
useEffect(() => {
if (!tableName) {
setColumns([]);
return;
}
const loadColumns = async () => {
setLoading(true);
try {
const { tableManagementApi } = await import("@/lib/api/tableManagement");
const response = await tableManagementApi.getColumnList(tableName);
if (response.success && response.data && response.data.columns) {
setColumns(response.data.columns);
}
} catch (error) {
console.error("컬럼 정보 로드 실패:", error);
} finally {
setLoading(false);
}
};
loadColumns();
}, [tableName]);
const toggleColumn = (columnName: string) => {
const newSelection = selectedColumns.includes(columnName)
? selectedColumns.filter((c) => c !== columnName)
: [...selectedColumns, columnName];
onChange(newSelection);
};
if (!tableName) {
return (
<div className="rounded-md border border-dashed p-3">
<p className="text-muted-foreground text-center text-xs"> </p>
</div>
);
}
return (
<div>
<Label className="text-xs"> </Label>
{loading ? (
<div className="rounded-md border p-3">
<p className="text-muted-foreground text-center text-xs"> ...</p>
</div>
) : columns.length === 0 ? (
<div className="rounded-md border border-dashed p-3">
<p className="text-muted-foreground text-center text-xs"> </p>
</div>
) : (
<div className="max-h-[200px] space-y-1 overflow-y-auto rounded-md border p-3">
{columns.map((col) => (
<div key={col.columnName} className="flex items-center gap-2">
<Checkbox
id={`groupby-${col.columnName}`}
checked={selectedColumns.includes(col.columnName)}
onCheckedChange={() => toggleColumn(col.columnName)}
/>
<label htmlFor={`groupby-${col.columnName}`} className="flex-1 cursor-pointer text-xs">
{col.columnLabel || col.columnName}
<span className="text-muted-foreground ml-1">({col.columnName})</span>
</label>
</div>
))}
</div>
)}
<p className="text-muted-foreground mt-1 text-[10px]">
: {selectedColumns.length > 0 ? selectedColumns.join(", ") : "없음"}
<br />
</p>
</div>
);
};
/**
* 화면 선택 Combobox 컴포넌트
*/
const ScreenSelector: React.FC<{
value?: number;
onChange: (screenId?: number) => void;
}> = ({ value, onChange }) => {
const [open, setOpen] = useState(false);
const [screens, setScreens] = useState<Array<{ screenId: number; screenName: string; screenCode: string }>>([]);
const [loading, setLoading] = useState(false);
useEffect(() => {
const loadScreens = async () => {
setLoading(true);
try {
const { screenApi } = await import("@/lib/api/screen");
const response = await screenApi.getScreens({ page: 1, size: 1000 });
setScreens(
response.data.map((s) => ({ screenId: s.screenId, screenName: s.screenName, screenCode: s.screenCode })),
);
} catch (error) {
console.error("화면 목록 로드 실패:", error);
} finally {
setLoading(false);
}
};
loadScreens();
}, []);
const selectedScreen = screens.find((s) => s.screenId === value);
return (
<Popover open={open} onOpenChange={setOpen}>
<PopoverTrigger asChild>
<Button
variant="outline"
role="combobox"
aria-expanded={open}
className="h-8 w-full justify-between text-xs"
disabled={loading}
>
{loading ? "로딩 중..." : selectedScreen ? selectedScreen.screenName : "화면 선택"}
<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50" />
</Button>
</PopoverTrigger>
<PopoverContent className="w-[400px] p-0" align="start">
<Command>
<CommandInput placeholder="화면 검색..." className="text-xs" />
<CommandList>
<CommandEmpty className="py-6 text-center text-xs"> .</CommandEmpty>
<CommandGroup className="max-h-[300px] overflow-auto">
{screens.map((screen) => (
<CommandItem
key={screen.screenId}
value={`${screen.screenName.toLowerCase()} ${screen.screenCode.toLowerCase()} ${screen.screenId}`}
onSelect={() => {
onChange(screen.screenId === value ? undefined : screen.screenId);
setOpen(false);
}}
className="text-xs"
>
<Check className={cn("mr-2 h-4 w-4", value === screen.screenId ? "opacity-100" : "opacity-0")} />
<div className="flex flex-col">
<span className="font-medium">{screen.screenName}</span>
<span className="text-muted-foreground text-[10px]">{screen.screenCode}</span>
</div>
</CommandItem>
))}
</CommandGroup>
</CommandList>
</Command>
</PopoverContent>
</Popover>
);
};
/**
* SplitPanelLayout 설정 패널
*/
export const SplitPanelLayoutConfigPanel: React.FC<SplitPanelLayoutConfigPanelProps> = ({
config,
onChange,
tables = [], // 기본값 빈 배열 (현재 화면 테이블만)
screenTableName, // 현재 화면의 테이블명
menuObjid, // 🆕 메뉴 OBJID
}) => {
const [rightTableOpen, setRightTableOpen] = useState(false);
const [loadedTableColumns, setLoadedTableColumns] = useState<Record<string, ColumnInfo[]>>({});
const [loadingColumns, setLoadingColumns] = useState<Record<string, boolean>>({});
const [allTables, setAllTables] = useState<any[]>([]); // 조인 모드용 전체 테이블 목록
// 엔티티 참조 테이블 컬럼
type EntityRefTable = { tableName: string; columns: ColumnInfo[] };
const [entityReferenceTables, setEntityReferenceTables] = useState<Record<string, EntityRefTable[]>>({});
// 🆕 입력 필드용 로컬 상태
const [isUserEditing, setIsUserEditing] = useState(false);
const [localTitles, setLocalTitles] = useState({
left: config.leftPanel?.title || "",
right: config.rightPanel?.title || "",
});
// 관계 타입
const relationshipType = config.rightPanel?.relation?.type || "detail";
// config 변경 시 로컬 타이틀 동기화 (사용자가 입력 중이 아닐 때만)
useEffect(() => {
if (!isUserEditing) {
setLocalTitles({
left: config.leftPanel?.title || "",
right: config.rightPanel?.title || "",
});
}
}, [config.leftPanel?.title, config.rightPanel?.title, isUserEditing]);
// 조인 모드일 때만 전체 테이블 목록 로드
useEffect(() => {
if (relationshipType === "join") {
const loadAllTables = async () => {
try {
const { tableManagementApi } = await import("@/lib/api/tableManagement");
const response = await tableManagementApi.getTableList();
if (response.success && response.data) {
console.log("✅ 분할패널 조인 모드: 전체 테이블 목록 로드", response.data.length, "개");
setAllTables(response.data);
}
} catch (error) {
console.error("❌ 전체 테이블 목록 로드 실패:", error);
}
};
loadAllTables();
} else {
// 상세 모드일 때는 기본 테이블만 사용
setAllTables([]);
}
}, [relationshipType]);
// screenTableName이 변경되면 좌측 패널 테이블을 항상 화면 테이블로 설정
useEffect(() => {
if (screenTableName) {
// 좌측 패널은 항상 현재 화면의 테이블 사용
if (config.leftPanel?.tableName !== screenTableName) {
updateLeftPanel({ tableName: screenTableName });
}
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [screenTableName]);
// 테이블 컬럼 로드 함수
const loadTableColumns = async (tableName: string) => {
if (loadedTableColumns[tableName] || loadingColumns[tableName]) {
return; // 이미 로드되었거나 로딩 중
}
setLoadingColumns((prev) => ({ ...prev, [tableName]: true }));
try {
const columnsResponse = await tableTypeApi.getColumns(tableName);
console.log(`📊 테이블 ${tableName} 컬럼 응답:`, columnsResponse);
const columns: ColumnInfo[] = (columnsResponse || []).map((col: any) => ({
tableName: col.tableName || tableName,
columnName: col.columnName || col.column_name,
columnLabel: col.displayName || col.columnLabel || col.column_label || col.columnName || col.column_name,
dataType: col.dataType || col.data_type || col.dbType,
webType: col.webType || col.web_type,
input_type: col.inputType || col.input_type,
widgetType: col.widgetType || col.widget_type || col.webType || col.web_type,
isNullable: col.isNullable || col.is_nullable,
required: col.required !== undefined ? col.required : col.isNullable === "NO" || col.is_nullable === "NO",
columnDefault: col.columnDefault || col.column_default,
characterMaximumLength: col.characterMaximumLength || col.character_maximum_length,
isPrimaryKey: col.isPrimaryKey || false, // PK 여부 추가
codeCategory: col.codeCategory || col.code_category,
codeValue: col.codeValue || col.code_value,
referenceTable: col.referenceTable || col.reference_table, // 🆕 참조 테이블
referenceColumn: col.referenceColumn || col.reference_column, // 🆕 참조 컬럼
displayColumn: col.displayColumn || col.display_column, // 🆕 표시 컬럼
}));
console.log(`✅ 테이블 ${tableName} 컬럼 ${columns.length}개 로드됨:`, columns);
setLoadedTableColumns((prev) => ({ ...prev, [tableName]: columns }));
// 🆕 엔티티 타입 컬럼의 참조 테이블 컬럼도 로드
await loadEntityReferenceColumns(tableName, columns);
} catch (error) {
console.error(`테이블 ${tableName} 컬럼 로드 실패:`, error);
setLoadedTableColumns((prev) => ({ ...prev, [tableName]: [] }));
} finally {
setLoadingColumns((prev) => ({ ...prev, [tableName]: false }));
}
};
// 🆕 엔티티 참조 테이블의 컬럼 로드
const loadEntityReferenceColumns = async (sourceTableName: string, columns: ColumnInfo[]) => {
const entityColumns = columns.filter(
(col) => (col.input_type === "entity" || col.webType === "entity") && col.referenceTable,
);
if (entityColumns.length === 0) {
return;
}
console.log(
`🔗 테이블 ${sourceTableName}의 엔티티 참조 ${entityColumns.length}개 발견:`,
entityColumns.map((c) => `${c.columnName} -> ${c.referenceTable}`),
);
const referenceTableData: Array<{ tableName: string; columns: ColumnInfo[] }> = [];
// 각 참조 테이블의 컬럼 로드
for (const entityCol of entityColumns) {
const refTableName = entityCol.referenceTable!;
// 이미 로드했으면 스킵
if (referenceTableData.some((t) => t.tableName === refTableName)) continue;
try {
const refColumnsResponse = await tableTypeApi.getColumns(refTableName);
const refColumns: ColumnInfo[] = (refColumnsResponse || []).map((col: any) => ({
tableName: col.tableName || refTableName,
columnName: col.columnName || col.column_name,
columnLabel: col.displayName || col.columnLabel || col.column_label || col.columnName || col.column_name,
dataType: col.dataType || col.data_type || col.dbType,
input_type: col.inputType || col.input_type,
}));
referenceTableData.push({ tableName: refTableName, columns: refColumns });
console.log(` ✅ 참조 테이블 ${refTableName} 컬럼 ${refColumns.length}개 로드됨`);
} catch (error) {
console.error(` ❌ 참조 테이블 ${refTableName} 컬럼 로드 실패:`, error);
}
}
// 참조 테이블 정보 저장
setEntityReferenceTables((prev) => ({
...prev,
[sourceTableName]: referenceTableData,
}));
console.log(`✅ [엔티티 참조] ${sourceTableName}의 참조 테이블 저장 완료:`, {
sourceTableName,
referenceTableCount: referenceTableData.length,
referenceTables: referenceTableData.map((t) => `${t.tableName}(${t.columns.length}개)`),
});
};
// 좌측/우측 테이블이 변경되면 해당 테이블의 컬럼 로드
useEffect(() => {
if (config.leftPanel?.tableName) {
loadTableColumns(config.leftPanel.tableName);
}
}, [config.leftPanel?.tableName]);
useEffect(() => {
if (config.rightPanel?.tableName) {
loadTableColumns(config.rightPanel.tableName);
}
}, [config.rightPanel?.tableName]);
console.log("🔧 SplitPanelLayoutConfigPanel 렌더링");
console.log(" - config:", config);
console.log(" - tables:", tables);
console.log(" - tablesCount:", tables.length);
console.log(" - screenTableName:", screenTableName);
console.log(" - leftTable:", config.leftPanel?.tableName);
console.log(" - rightTable:", config.rightPanel?.tableName);
const updateConfig = (updates: Partial<SplitPanelLayoutConfig>) => {
const newConfig = { ...config, ...updates };
console.log("🔄 Config 업데이트:", newConfig);
onChange(newConfig);
};
const updateLeftPanel = (updates: Partial<SplitPanelLayoutConfig["leftPanel"]>) => {
const newConfig = {
...config,
leftPanel: { ...config.leftPanel, ...updates },
};
console.log("🔄 Left Panel 업데이트:", newConfig);
onChange(newConfig);
};
const updateRightPanel = (updates: Partial<SplitPanelLayoutConfig["rightPanel"]>) => {
const newConfig = {
...config,
rightPanel: { ...config.rightPanel, ...updates },
};
console.log("🔄 Right Panel 업데이트:", newConfig);
onChange(newConfig);
};
// 좌측 테이블명
const leftTableName = config.leftPanel?.tableName || screenTableName || "";
// 좌측 테이블 컬럼 (로드된 컬럼 사용)
const leftTableColumns = useMemo(() => {
return leftTableName ? loadedTableColumns[leftTableName] || [] : [];
}, [loadedTableColumns, leftTableName]);
// 우측 테이블명
const rightTableName = config.rightPanel?.tableName || "";
// 우측 테이블 컬럼 (로드된 컬럼 사용)
const rightTableColumns = useMemo(() => {
return rightTableName ? loadedTableColumns[rightTableName] || [] : [];
}, [loadedTableColumns, rightTableName]);
// 테이블 데이터 로딩 상태 확인
if (!tables || tables.length === 0) {
return (
<div className="rounded-lg border p-4">
<p className="text-sm font-medium"> .</p>
<p className="mt-1 text-xs text-gray-600">
.
</p>
</div>
);
}
// 조인 모드에서 우측 테이블 선택 시 사용할 테이블 목록
const availableRightTables = relationshipType === "join" ? allTables : tables;
console.log("📊 분할패널 테이블 목록 상태:");
console.log(" - relationshipType:", relationshipType);
console.log(" - allTables:", allTables.length, "개");
console.log(" - availableRightTables:", availableRightTables.length, "개");
return (
<div className="space-y-4">
{/* 관계 타입 선택 */}
<div className="space-y-3">
<h3 className="text-sm font-semibold"> </h3>
<Select
value={relationshipType}
onValueChange={(value: "join" | "detail") => {
// 상세 모드로 변경 시 우측 테이블을 현재 화면 테이블로 설정
if (value === "detail" && screenTableName) {
updateRightPanel({
relation: { ...config.rightPanel?.relation, type: value },
tableName: screenTableName,
});
} else {
updateRightPanel({
relation: { ...config.rightPanel?.relation, type: value },
});
}
}}
>
<SelectTrigger className="bg-white">
<SelectValue placeholder="관계 타입 선택" />
</SelectTrigger>
<SelectContent>
<SelectItem value="detail">
<div className="flex flex-col">
<span className="font-medium"> (DETAIL)</span>
<span className="text-xs text-gray-500"> ( )</span>
</div>
</SelectItem>
<SelectItem value="join">
<div className="flex flex-col">
<span className="font-medium"> (JOIN)</span>
<span className="text-xs text-gray-500"> ( )</span>
</div>
</SelectItem>
</SelectContent>
</Select>
</div>
{/* 좌측 패널 설정 */}
<div className="mt-4 space-y-4 border-t pt-4">
<h3 className="text-sm font-semibold"> ()</h3>
<div className="space-y-2">
<Label> </Label>
<Input
value={localTitles.left}
onChange={(e) => {
setIsUserEditing(true);
setLocalTitles((prev) => ({ ...prev, left: e.target.value }));
}}
onBlur={() => {
setIsUserEditing(false);
updateLeftPanel({ title: localTitles.left });
}}
placeholder="좌측 패널 제목"
/>
</div>
<div className="space-y-2">
<Label> (px)</Label>
<Input
type="number"
value={config.leftPanel?.panelHeaderHeight || 48}
onChange={(e) => updateLeftPanel({ panelHeaderHeight: parseInt(e.target.value) || 48 })}
placeholder="48"
min={32}
max={120}
/>
<p className="text-muted-foreground text-xs"> (기본: 48px)</p>
</div>
<div className="space-y-2">
<Label> </Label>
<Select
value={config.leftPanel?.displayMode || "list"}
onValueChange={(value: "list" | "table") => updateLeftPanel({ displayMode: value })}
>
<SelectTrigger className="bg-white">
<SelectValue placeholder="표시 모드 선택" />
</SelectTrigger>
<SelectContent>
<SelectItem value="list">
<div className="flex flex-col">
<span className="font-medium"> (LIST)</span>
<span className="text-xs text-gray-500"> ()</span>
</div>
</SelectItem>
<SelectItem value="table">
<div className="flex flex-col">
<span className="font-medium"> (TABLE)</span>
<span className="text-xs text-gray-500"> </span>
</div>
</SelectItem>
</SelectContent>
</Select>
</div>
{/* 좌측 패널 표시 컬럼 설정 - 체크박스 방식 */}
<div className="space-y-2">
<Label className="text-xs font-medium"> </Label>
{/* 컬럼 체크박스 목록 */}
<div className="max-h-60 space-y-0.5 overflow-y-auto rounded-md border p-2">
{leftTableColumns.length === 0 ? (
<p className="text-muted-foreground py-2 text-center text-xs"> ...</p>
) : (
leftTableColumns
.filter((column) => !["company_code", "company_name"].includes(column.columnName))
.map((column) => {
const isSelected = (config.leftPanel?.columns || []).some((c) => c.name === column.columnName);
return (
<div
key={column.columnName}
className={cn(
"hover:bg-muted/50 flex cursor-pointer items-center gap-2 rounded px-2 py-1.5",
isSelected && "bg-primary/10",
)}
onClick={() => {
const currentColumns = config.leftPanel?.columns || [];
if (isSelected) {
// 제거
const newColumns = currentColumns.filter((c) => c.name !== column.columnName);
updateLeftPanel({ columns: newColumns });
} else {
// 추가
const newColumn = {
name: column.columnName,
label: column.columnLabel || column.columnName,
width: 100,
};
updateLeftPanel({ columns: [...currentColumns, newColumn] });
}
}}
>
<Checkbox
checked={isSelected}
onCheckedChange={() => {
const currentColumns = config.leftPanel?.columns || [];
if (isSelected) {
const newColumns = currentColumns.filter((c) => c.name !== column.columnName);
updateLeftPanel({ columns: newColumns });
} else {
const newColumn = {
name: column.columnName,
label: column.columnLabel || column.columnName,
width: 100,
};
updateLeftPanel({ columns: [...currentColumns, newColumn] });
}
}}
className="pointer-events-none h-3.5 w-3.5 shrink-0"
/>
<Database className="text-muted-foreground h-3 w-3 shrink-0" />
<span className="truncate text-xs">{column.columnLabel || column.columnName}</span>
</div>
);
})
)}
</div>
{/* 선택된 컬럼 상세 설정 */}
{(config.leftPanel?.columns || []).length > 0 && (
<div className="space-y-2">
<Label className="text-xs font-medium"> ({(config.leftPanel?.columns || []).length})</Label>
<div className="max-h-48 space-y-1.5 overflow-y-auto">
{(config.leftPanel?.columns || []).map((col, index) => {
const column = leftTableColumns.find((c) => c.columnName === col.name);
const isTableMode = config.leftPanel?.displayMode === "table";
// 숫자 타입 판별
const dbNumericTypes = [
"numeric",
"decimal",
"integer",
"bigint",
"double precision",
"real",
"smallint",
"int4",
"int8",
"float4",
"float8",
];
const inputNumericTypes = ["number", "decimal", "currency", "integer"];
const isNumeric =
column &&
(dbNumericTypes.includes(column.dataType?.toLowerCase() || "") ||
inputNumericTypes.includes(column.input_type?.toLowerCase() || "") ||
inputNumericTypes.includes(column.webType?.toLowerCase() || ""));
return (
<div key={col.name} className="bg-muted/30 flex items-center gap-2 rounded-md border p-2">
<GripVertical className="text-muted-foreground h-3 w-3 shrink-0 cursor-grab" />
<Database className="text-muted-foreground h-3 w-3 shrink-0" />
<Input
value={col.label}
onChange={(e) => {
const newColumns = [...(config.leftPanel?.columns || [])];
newColumns[index] = { ...newColumns[index], label: e.target.value };
updateLeftPanel({ columns: newColumns });
}}
placeholder="제목"
className="h-6 flex-1 text-xs"
/>
<Input
value={col.width || ""}
onChange={(e) => {
const newColumns = [...(config.leftPanel?.columns || [])];
newColumns[index] = { ...newColumns[index], width: parseInt(e.target.value) || 100 };
updateLeftPanel({ columns: newColumns });
}}
placeholder="너비"
className="h-6 w-14 text-xs"
/>
{/* 숫자 타입: 천단위 구분자 체크박스 */}
{isNumeric && (
<label
className="flex shrink-0 cursor-pointer items-center gap-1 text-[10px]"
title="천 단위 구분자 (,)"
>
<input
type="checkbox"
checked={col.format?.thousandSeparator ?? false}
onChange={(e) => {
const newColumns = [...(config.leftPanel?.columns || [])];
newColumns[index] = {
...newColumns[index],
format: {
...newColumns[index].format,
type: "number",
thousandSeparator: e.target.checked,
},
};
updateLeftPanel({ columns: newColumns });
}}
className="h-3 w-3"
/>
,
</label>
)}
<Button
type="button"
variant="ghost"
size="sm"
onClick={() => {
const newColumns = (config.leftPanel?.columns || []).filter((_, i) => i !== index);
updateLeftPanel({ columns: newColumns });
}}
className="text-destructive h-6 w-6 shrink-0 p-0"
>
<X className="h-3 w-3" />
</Button>
</div>
);
})}
</div>
</div>
)}
</div>
</div>
{/* 좌측 패널 데이터 필터링 */}
<div className="mt-4 space-y-4 border-t pt-4">
<h3 className="text-sm font-semibold"> </h3>
<p className="text-muted-foreground text-xs"> </p>
<DataFilterConfigPanel
tableName={config.leftPanel?.tableName || screenTableName}
columns={leftTableColumns.map(
(col) =>
({
columnName: col.columnName,
columnLabel: col.columnLabel || col.columnName,
dataType: col.dataType || "text",
input_type: (col as any).input_type,
}) as any,
)}
config={config.leftPanel?.dataFilter}
onConfigChange={(dataFilter) => updateLeftPanel({ dataFilter })}
menuObjid={menuObjid} // 🆕 메뉴 OBJID 전달
/>
</div>
{/* 우측 패널 설정 */}
<div className="mt-4 space-y-4 border-t pt-4">
<h3 className="text-sm font-semibold"> ({relationshipType === "detail" ? "상세" : "조인"})</h3>
<div className="space-y-2">
<Label> </Label>
<Input
value={localTitles.right}
onChange={(e) => {
setIsUserEditing(true);
setLocalTitles((prev) => ({ ...prev, right: e.target.value }));
}}
onBlur={() => {
setIsUserEditing(false);
updateRightPanel({ title: localTitles.right });
}}
placeholder="우측 패널 제목"
/>
</div>
<div className="space-y-2">
<Label> (px)</Label>
<Input
type="number"
value={config.rightPanel?.panelHeaderHeight || 48}
onChange={(e) => updateRightPanel({ panelHeaderHeight: parseInt(e.target.value) || 48 })}
placeholder="48"
min={32}
max={120}
/>
<p className="text-muted-foreground text-xs"> (기본: 48px)</p>
</div>
{/* 관계 타입에 따라 테이블 선택 UI 변경 */}
{relationshipType === "detail" ? (
// 상세 모드: 좌측과 동일한 테이블 (자동 설정)
<div className="space-y-2">
<Label> ( )</Label>
<div className="rounded-lg border border-gray-200 bg-gray-50 p-3">
<p className="text-sm font-medium text-gray-900">
{config.leftPanel?.tableName || screenTableName || "테이블이 지정되지 않음"}
</p>
<p className="mt-1 text-xs text-gray-500"> </p>
</div>
</div>
) : (
// 조인 모드: 전체 테이블에서 선택 가능
<div className="space-y-2">
<Label> ( )</Label>
<Popover open={rightTableOpen} onOpenChange={setRightTableOpen}>
<PopoverTrigger asChild>
<Button
variant="outline"
role="combobox"
aria-expanded={rightTableOpen}
className="w-full justify-between"
>
{config.rightPanel?.tableName || "테이블을 선택하세요"}
<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50" />
</Button>
</PopoverTrigger>
<PopoverContent className="w-full p-0">
<Command>
<CommandInput placeholder="테이블 검색..." />
<CommandEmpty> .</CommandEmpty>
<CommandGroup className="max-h-[200px] overflow-auto">
{availableRightTables.map((table) => (
<CommandItem
key={table.tableName}
value={`${table.displayName || ""} ${table.tableName}`}
onSelect={() => {
updateRightPanel({ tableName: table.tableName });
setRightTableOpen(false);
}}
>
<Check
className={cn(
"mr-2 h-4 w-4",
config.rightPanel?.tableName === table.tableName ? "opacity-100" : "opacity-0",
)}
/>
{table.displayName || table.tableName}
{table.displayName && <span className="ml-2 text-xs text-gray-500">({table.tableName})</span>}
</CommandItem>
))}
</CommandGroup>
</Command>
</PopoverContent>
</Popover>
</div>
)}
<div className="space-y-2">
<Label> </Label>
<Select
value={config.rightPanel?.displayMode || "list"}
onValueChange={(value: "list" | "table") => updateRightPanel({ displayMode: value })}
>
<SelectTrigger className="bg-white">
<SelectValue placeholder="표시 모드 선택" />
</SelectTrigger>
<SelectContent>
<SelectItem value="list">
<div className="flex flex-col">
<span className="font-medium"> (LIST)</span>
<span className="text-xs text-gray-500"> ()</span>
</div>
</SelectItem>
<SelectItem value="table">
<div className="flex flex-col">
<span className="font-medium"> (TABLE)</span>
<span className="text-xs text-gray-500"> </span>
</div>
</SelectItem>
</SelectContent>
</Select>
</div>
{/* 요약 표시 설정 (LIST 모드에서만) */}
{(config.rightPanel?.displayMode || "list") === "list" && (
<div className="space-y-3 rounded-lg border border-gray-200 bg-gray-50 p-3">
<Label className="text-sm font-semibold"> </Label>
<div className="space-y-2">
<Label className="text-xs"> </Label>
<Input
type="number"
min="1"
max="10"
value={config.rightPanel?.summaryColumnCount ?? 3}
onChange={(e) => {
const value = parseInt(e.target.value) || 3;
updateRightPanel({ summaryColumnCount: value });
}}
className="bg-white"
/>
<p className="text-xs text-gray-500"> (기본: 3개)</p>
</div>
<div className="flex items-center justify-between space-x-2">
<div className="flex-1">
<Label className="text-xs"> </Label>
<p className="text-xs text-gray-500"> </p>
</div>
<Checkbox
checked={config.rightPanel?.summaryShowLabel ?? true}
onCheckedChange={(checked) => {
updateRightPanel({ summaryShowLabel: checked as boolean });
}}
/>
</div>
</div>
)}
{/* 엔티티 설정 선택 - 조인 모드에서만 표시 */}
{relationshipType !== "detail" && (
<div className="space-y-2">
<Label className="text-xs font-medium"> ( )</Label>
<p className="text-muted-foreground text-[10px]">
</p>
<Select
value={config.rightPanel?.relation?.foreignKey || ""}
onValueChange={(value) => {
// 선택된 엔티티 컬럼 정보 찾기
const entityColumn = rightTableColumns.find((col) => col.columnName === value);
if (entityColumn) {
updateRightPanel({
relation: {
...config.rightPanel?.relation,
foreignKey: value,
// 참조 테이블과 컬럼은 엔티티 설정에서 자동으로 가져옴
},
});
}
}}
>
<SelectTrigger className="h-8 text-xs">
<SelectValue placeholder="엔티티 컬럼 선택" />
</SelectTrigger>
<SelectContent>
{rightTableColumns
.filter((col) => {
// 엔티티 타입 컬럼만 표시 (input_type이 entity인 경우)
const inputType = col.input_type?.toLowerCase() || col.webType?.toLowerCase() || "";
return inputType === "entity" || inputType === "code";
})
.map((column) => (
<SelectItem key={column.columnName} value={column.columnName}>
<div className="flex items-center gap-2">
<span>{column.columnLabel || column.columnName}</span>
<span className="text-muted-foreground text-[10px]">({column.columnName})</span>
</div>
</SelectItem>
))}
{rightTableColumns.filter((col) => {
const inputType = col.input_type?.toLowerCase() || col.webType?.toLowerCase() || "";
return inputType === "entity" || inputType === "code";
}).length === 0 && (
<div className="text-muted-foreground px-2 py-4 text-center text-xs">
.
<br />
.
</div>
)}
</SelectContent>
</Select>
{config.rightPanel?.relation?.foreignKey && (
<p className="text-muted-foreground text-[10px]"> .</p>
)}
</div>
)}
{/* 우측 패널 표시 컬럼 설정 - 체크박스 방식 */}
<div className="space-y-2">
<Label className="text-xs font-medium"> </Label>
{/* 컬럼 체크박스 목록 */}
<div className="max-h-60 space-y-0.5 overflow-y-auto rounded-md border p-2">
{rightTableColumns.length === 0 ? (
<p className="text-muted-foreground py-2 text-center text-xs"> </p>
) : (
rightTableColumns
.filter((column) => !["company_code", "company_name"].includes(column.columnName))
.map((column) => {
const isSelected = (config.rightPanel?.columns || []).some((c) => c.name === column.columnName);
return (
<div
key={column.columnName}
className={cn(
"hover:bg-muted/50 flex cursor-pointer items-center gap-2 rounded px-2 py-1.5",
isSelected && "bg-primary/10",
)}
onClick={() => {
const currentColumns = config.rightPanel?.columns || [];
if (isSelected) {
const newColumns = currentColumns.filter((c) => c.name !== column.columnName);
updateRightPanel({ columns: newColumns });
} else {
const newColumn = {
name: column.columnName,
label: column.columnLabel || column.columnName,
width: 100,
};
updateRightPanel({ columns: [...currentColumns, newColumn] });
}
}}
>
<Checkbox
checked={isSelected}
onCheckedChange={() => {
const currentColumns = config.rightPanel?.columns || [];
if (isSelected) {
const newColumns = currentColumns.filter((c) => c.name !== column.columnName);
updateRightPanel({ columns: newColumns });
} else {
const newColumn = {
name: column.columnName,
label: column.columnLabel || column.columnName,
width: 100,
};
updateRightPanel({ columns: [...currentColumns, newColumn] });
}
}}
className="pointer-events-none h-3.5 w-3.5 shrink-0"
/>
<Database className="text-muted-foreground h-3 w-3 shrink-0" />
<span className="truncate text-xs">{column.columnLabel || column.columnName}</span>
</div>
);
})
)}
</div>
{/* 선택된 컬럼 상세 설정 */}
{(config.rightPanel?.columns || []).length > 0 && (
<div className="space-y-2">
<Label className="text-xs font-medium"> ({(config.rightPanel?.columns || []).length})</Label>
<div className="max-h-48 space-y-1.5 overflow-y-auto">
{(config.rightPanel?.columns || []).map((col, index) => {
const column = rightTableColumns.find((c) => c.columnName === col.name);
// 숫자 타입 판별
const dbNumericTypes = [
"numeric",
"decimal",
"integer",
"bigint",
"double precision",
"real",
"smallint",
"int4",
"int8",
"float4",
"float8",
];
const inputNumericTypes = ["number", "decimal", "currency", "integer"];
const isNumeric =
column &&
(dbNumericTypes.includes(column.dataType?.toLowerCase() || "") ||
inputNumericTypes.includes(column.input_type?.toLowerCase() || "") ||
inputNumericTypes.includes(column.webType?.toLowerCase() || ""));
return (
<div key={col.name} className="bg-muted/30 flex items-center gap-2 rounded-md border p-2">
<GripVertical className="text-muted-foreground h-3 w-3 shrink-0 cursor-grab" />
<Database className="text-muted-foreground h-3 w-3 shrink-0" />
<Input
value={col.label}
onChange={(e) => {
const newColumns = [...(config.rightPanel?.columns || [])];
newColumns[index] = { ...newColumns[index], label: e.target.value };
updateRightPanel({ columns: newColumns });
}}
placeholder="제목"
className="h-6 flex-1 text-xs"
/>
<Input
value={col.width || ""}
onChange={(e) => {
const newColumns = [...(config.rightPanel?.columns || [])];
newColumns[index] = { ...newColumns[index], width: parseInt(e.target.value) || 100 };
updateRightPanel({ columns: newColumns });
}}
placeholder="너비"
className="h-6 w-14 text-xs"
/>
{/* 숫자 타입: 천단위 구분자 체크박스 */}
{isNumeric && (
<label
className="flex shrink-0 cursor-pointer items-center gap-1 text-[10px]"
title="천 단위 구분자 (,)"
>
<input
type="checkbox"
checked={col.format?.thousandSeparator ?? false}
onChange={(e) => {
const newColumns = [...(config.rightPanel?.columns || [])];
newColumns[index] = {
...newColumns[index],
format: {
...newColumns[index].format,
type: "number",
thousandSeparator: e.target.checked,
},
};
updateRightPanel({ columns: newColumns });
}}
className="h-3 w-3"
/>
,
</label>
)}
<Button
type="button"
variant="ghost"
size="sm"
onClick={() => {
const newColumns = (config.rightPanel?.columns || []).filter((_, i) => i !== index);
updateRightPanel({ columns: newColumns });
}}
className="text-destructive h-6 w-6 shrink-0 p-0"
>
<X className="h-3 w-3" />
</Button>
</div>
);
})}
</div>
</div>
)}
</div>
</div>
{/* 우측 패널 데이터 필터링 */}
<div className="mt-4 space-y-4 border-t pt-4">
<h3 className="text-sm font-semibold"> </h3>
<p className="text-muted-foreground text-xs"> </p>
<DataFilterConfigPanel
tableName={config.rightPanel?.tableName}
columns={rightTableColumns.map(
(col) =>
({
columnName: col.columnName,
columnLabel: col.columnLabel || col.columnName,
dataType: col.dataType || "text",
input_type: (col as any).input_type,
}) as any,
)}
config={config.rightPanel?.dataFilter}
onConfigChange={(dataFilter) => updateRightPanel({ dataFilter })}
menuObjid={menuObjid} // 🆕 메뉴 OBJID 전달
/>
</div>
{/* 우측 패널 중복 제거 */}
<div className="mt-4 space-y-4 border-t pt-4">
<div className="flex items-center justify-between">
<div>
<h3 className="text-sm font-semibold"> </h3>
<p className="text-muted-foreground text-xs"> </p>
</div>
<Switch
checked={config.rightPanel?.deduplication?.enabled ?? false}
onCheckedChange={(checked) => {
if (checked) {
updateRightPanel({
deduplication: {
enabled: true,
groupByColumn: "",
keepStrategy: "latest",
sortColumn: "start_date",
},
});
} else {
updateRightPanel({ deduplication: undefined });
}
}}
/>
</div>
{config.rightPanel?.deduplication?.enabled && (
<div className="space-y-3 border-l-2 pl-4">
{/* 중복 제거 기준 컬럼 */}
<div>
<Label className="text-xs"> </Label>
<Select
value={config.rightPanel?.deduplication?.groupByColumn || ""}
onValueChange={(value) =>
updateRightPanel({
deduplication: { ...config.rightPanel?.deduplication!, groupByColumn: value },
})
}
>
<SelectTrigger className="h-8 text-xs">
<SelectValue placeholder="기준 컬럼 선택" />
</SelectTrigger>
<SelectContent>
{rightTableColumns.map((col) => (
<SelectItem key={col.columnName} value={col.columnName}>
{col.columnLabel || col.columnName}
</SelectItem>
))}
</SelectContent>
</Select>
<p className="text-muted-foreground mt-1 text-[10px]">
</p>
</div>
{/* 유지 전략 */}
<div>
<Label className="text-xs"> </Label>
<Select
value={config.rightPanel?.deduplication?.keepStrategy || "latest"}
onValueChange={(value: any) =>
updateRightPanel({
deduplication: { ...config.rightPanel?.deduplication!, keepStrategy: value },
})
}
>
<SelectTrigger className="h-8 text-xs">
<SelectValue />
</SelectTrigger>
<SelectContent>
<SelectItem value="latest"> ( )</SelectItem>
<SelectItem value="earliest"> ( )</SelectItem>
<SelectItem value="current_date"> ( )</SelectItem>
<SelectItem value="base_price"> </SelectItem>
</SelectContent>
</Select>
<p className="text-muted-foreground mt-1 text-[10px]">
{config.rightPanel?.deduplication?.keepStrategy === "latest" &&
"가장 최근에 추가된 데이터를 표시합니다"}
{config.rightPanel?.deduplication?.keepStrategy === "earliest" &&
"가장 먼저 추가된 데이터를 표시합니다"}
{config.rightPanel?.deduplication?.keepStrategy === "current_date" &&
"오늘 날짜 기준으로 유효한 기간의 데이터를 표시합니다"}
{config.rightPanel?.deduplication?.keepStrategy === "base_price" &&
"기준단가(base_price)로 체크된 데이터를 표시합니다"}
</p>
</div>
{/* 정렬 기준 컬럼 (latest/earliest만) */}
{(config.rightPanel?.deduplication?.keepStrategy === "latest" ||
config.rightPanel?.deduplication?.keepStrategy === "earliest") && (
<div>
<Label className="text-xs"> </Label>
<Select
value={config.rightPanel?.deduplication?.sortColumn || ""}
onValueChange={(value) =>
updateRightPanel({
deduplication: { ...config.rightPanel?.deduplication!, sortColumn: value },
})
}
>
<SelectTrigger className="h-8 text-xs">
<SelectValue placeholder="정렬 기준 컬럼 선택" />
</SelectTrigger>
<SelectContent>
{rightTableColumns
.filter(
(col) =>
col.dataType === "date" || col.dataType === "timestamp" || col.columnName.includes("date"),
)
.map((col) => (
<SelectItem key={col.columnName} value={col.columnName}>
{col.columnLabel || col.columnName}
</SelectItem>
))}
</SelectContent>
</Select>
<p className="text-muted-foreground mt-1 text-[10px]">
/ ( )
</p>
</div>
)}
</div>
)}
</div>
{/* 🆕 우측 패널 수정 버튼 설정 */}
<div className="mt-4 space-y-4 border-t pt-4">
<div className="flex items-center justify-between">
<div>
<h3 className="text-sm font-semibold"> </h3>
<p className="text-muted-foreground text-xs"> </p>
</div>
<Switch
checked={config.rightPanel?.editButton?.enabled ?? true}
onCheckedChange={(checked) => {
updateRightPanel({
editButton: {
enabled: checked,
mode: config.rightPanel?.editButton?.mode || "auto",
buttonLabel: config.rightPanel?.editButton?.buttonLabel,
buttonVariant: config.rightPanel?.editButton?.buttonVariant,
},
});
}}
/>
</div>
{(config.rightPanel?.editButton?.enabled ?? true) && (
<div className="space-y-3 border-l-2 pl-4">
{/* 수정 모드 */}
<div>
<Label className="text-xs"> </Label>
<Select
value={config.rightPanel?.editButton?.mode || "auto"}
onValueChange={(value: "auto" | "modal") =>
updateRightPanel({
editButton: {
...config.rightPanel?.editButton,
mode: value,
enabled: config.rightPanel?.editButton?.enabled ?? true,
},
})
}
>
<SelectTrigger className="h-8 text-xs">
<SelectValue />
</SelectTrigger>
<SelectContent>
<SelectItem value="auto"> ()</SelectItem>
<SelectItem value="modal"> </SelectItem>
</SelectContent>
</Select>
<p className="text-muted-foreground mt-1 text-[10px]">
{config.rightPanel?.editButton?.mode === "modal"
? "지정한 화면을 모달로 열어 데이터를 수정합니다"
: "현재 위치에서 직접 데이터를 수정합니다"}
</p>
</div>
{/* 모달 화면 선택 (modal 모드일 때만) */}
{config.rightPanel?.editButton?.mode === "modal" && (
<div>
<Label className="text-xs"> </Label>
<ScreenSelector
value={config.rightPanel?.editButton?.modalScreenId}
onChange={(screenId) =>
updateRightPanel({
editButton: {
...config.rightPanel?.editButton!,
modalScreenId: screenId,
},
})
}
/>
<p className="text-muted-foreground mt-1 text-[10px]"> </p>
</div>
)}
{/* 버튼 라벨 */}
<div>
<Label className="text-xs"> </Label>
<Input
value={config.rightPanel?.editButton?.buttonLabel || "수정"}
onChange={(e) =>
updateRightPanel({
editButton: {
...config.rightPanel?.editButton!,
buttonLabel: e.target.value,
enabled: config.rightPanel?.editButton?.enabled ?? true,
mode: config.rightPanel?.editButton?.mode || "auto",
},
})
}
className="h-8 text-xs"
placeholder="수정"
/>
</div>
{/* 버튼 스타일 */}
<div>
<Label className="text-xs"> </Label>
<Select
value={config.rightPanel?.editButton?.buttonVariant || "outline"}
onValueChange={(value: any) =>
updateRightPanel({
editButton: {
...config.rightPanel?.editButton!,
buttonVariant: value,
enabled: config.rightPanel?.editButton?.enabled ?? true,
mode: config.rightPanel?.editButton?.mode || "auto",
},
})
}
>
<SelectTrigger className="h-8 text-xs">
<SelectValue />
</SelectTrigger>
<SelectContent>
<SelectItem value="default"> ()</SelectItem>
<SelectItem value="outline"></SelectItem>
<SelectItem value="ghost"></SelectItem>
</SelectContent>
</Select>
</div>
{/* 🆕 그룹핑 기준 컬럼 설정 (modal 모드일 때만 표시) */}
{config.rightPanel?.editButton?.mode === "modal" && (
<GroupByColumnsSelector
tableName={config.rightPanel?.tableName}
selectedColumns={config.rightPanel?.editButton?.groupByColumns || []}
onChange={(columns) => {
updateRightPanel({
editButton: {
...config.rightPanel?.editButton!,
groupByColumns: columns,
enabled: config.rightPanel?.editButton?.enabled ?? true,
mode: config.rightPanel?.editButton?.mode || "auto",
},
});
}}
/>
)}
</div>
)}
</div>
{/* 🆕 우측 패널 삭제 버튼 설정 */}
<div className="space-y-3 rounded-lg border border-red-200 bg-red-50 p-3">
<div className="flex items-center justify-between">
<div className="flex items-center gap-2">
<h3 className="text-sm font-semibold"> </h3>
</div>
<Switch
checked={config.rightPanel?.deleteButton?.enabled ?? true}
onCheckedChange={(checked) => {
updateRightPanel({
deleteButton: {
enabled: checked,
buttonLabel: config.rightPanel?.deleteButton?.buttonLabel,
buttonVariant: config.rightPanel?.deleteButton?.buttonVariant,
},
});
}}
/>
</div>
{(config.rightPanel?.deleteButton?.enabled ?? true) && (
<div className="space-y-3">
<div className="grid grid-cols-2 gap-3">
{/* 버튼 라벨 */}
<div className="space-y-1">
<Label className="text-xs"> </Label>
<Input
value={config.rightPanel?.deleteButton?.buttonLabel || ""}
placeholder="삭제"
onChange={(e) => {
updateRightPanel({
deleteButton: {
...config.rightPanel?.deleteButton!,
buttonLabel: e.target.value || undefined,
enabled: config.rightPanel?.deleteButton?.enabled ?? true,
},
});
}}
className="h-8 text-xs"
/>
</div>
{/* 버튼 스타일 */}
<div className="space-y-1">
<Label className="text-xs"> </Label>
<Select
value={config.rightPanel?.deleteButton?.buttonVariant || "ghost"}
onValueChange={(value: "default" | "outline" | "ghost" | "destructive") => {
updateRightPanel({
deleteButton: {
...config.rightPanel?.deleteButton!,
buttonVariant: value,
enabled: config.rightPanel?.deleteButton?.enabled ?? true,
},
});
}}
>
<SelectTrigger className="h-8 text-xs">
<SelectValue />
</SelectTrigger>
<SelectContent>
<SelectItem value="ghost">Ghost ()</SelectItem>
<SelectItem value="outline">Outline</SelectItem>
<SelectItem value="default">Default</SelectItem>
<SelectItem value="destructive">Destructive ()</SelectItem>
</SelectContent>
</Select>
</div>
</div>
{/* 삭제 확인 메시지 */}
<div className="space-y-1">
<Label className="text-xs"> </Label>
<Input
value={config.rightPanel?.deleteButton?.confirmMessage || ""}
placeholder="정말 삭제하시겠습니까?"
onChange={(e) => {
updateRightPanel({
deleteButton: {
...config.rightPanel?.deleteButton!,
confirmMessage: e.target.value || undefined,
enabled: config.rightPanel?.deleteButton?.enabled ?? true,
},
});
}}
className="h-8 text-xs"
/>
</div>
</div>
)}
</div>
{/* 레이아웃 설정 */}
<div className="mt-4 space-y-4 border-t pt-4">
<div className="space-y-2">
<Label> : {config.splitRatio || 30}%</Label>
<Slider
value={[config.splitRatio || 30]}
onValueChange={(value) => updateConfig({ splitRatio: value[0] })}
min={20}
max={80}
step={5}
/>
</div>
<div className="flex items-center justify-between">
<Label> </Label>
<Switch
checked={config.resizable ?? true}
onCheckedChange={(checked) => updateConfig({ resizable: checked })}
/>
</div>
<div className="flex items-center justify-between">
<Label> </Label>
<Switch
checked={config.autoLoad ?? true}
onCheckedChange={(checked) => updateConfig({ autoLoad: checked })}
/>
</div>
</div>
</div>
);
};