feat(split-panel-layout2): 복수 검색 컬럼 지원 기능 추가

- SearchColumnConfig 타입 추가 (types.ts)
- 좌측/우측 패널 모두 여러 검색 컬럼 설정 가능
- ConfigPanel에 검색 컬럼 추가/삭제 UI 구현
- 검색 시 OR 조건으로 여러 컬럼 동시 검색
- 기존 searchColumn 단일 설정과 하위 호환성 유지
This commit is contained in:
SeongHyun Kim
2025-12-03 18:43:01 +09:00
parent 700623aa78
commit 294c61e0e3
3 changed files with 587 additions and 234 deletions

View File

@@ -98,13 +98,35 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
const loadTables = useCallback(async () => {
setTablesLoading(true);
try {
const response = await apiClient.get("/table/list?userLang=KR");
const tableList = response.data?.data || response.data || [];
if (Array.isArray(tableList)) {
setTables(tableList);
const response = await apiClient.get("/table-management/tables");
console.log("[loadTables] API 응답:", response.data);
let tableList: any[] = [];
if (response.data?.success && Array.isArray(response.data?.data)) {
tableList = response.data.data;
} else if (Array.isArray(response.data?.data)) {
tableList = response.data.data;
} else if (Array.isArray(response.data)) {
tableList = response.data;
}
console.log("[loadTables] 추출된 테이블 목록:", tableList);
if (tableList.length > 0) {
// 백엔드에서 카멜케이스(tableName)로 반환하므로 둘 다 처리
const transformedTables = tableList.map((t: any) => ({
table_name: t.tableName ?? t.table_name ?? t.name ?? "",
table_comment: t.displayName ?? t.table_comment ?? t.description ?? "",
}));
console.log("[loadTables] 변환된 테이블 목록:", transformedTables);
setTables(transformedTables);
} else {
console.warn("[loadTables] 테이블 목록이 비어있습니다");
setTables([]);
}
} catch (error) {
console.error("테이블 목록 로드 실패:", error);
setTables([]);
} finally {
setTablesLoading(false);
}
@@ -114,20 +136,38 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
const loadScreens = useCallback(async () => {
setScreensLoading(true);
try {
const response = await apiClient.get("/screen/list");
// size를 크게 설정하여 모든 화면 가져오기
const response = await apiClient.get("/screen-management/screens?size=1000");
console.log("[loadScreens] API 응답:", response.data);
const screenList = response.data?.data || response.data || [];
if (Array.isArray(screenList)) {
// API 응답 구조: { success, data: [...], total, page, size }
let screenList: any[] = [];
if (response.data?.success && Array.isArray(response.data?.data)) {
screenList = response.data.data;
} else if (Array.isArray(response.data?.data)) {
screenList = response.data.data;
} else if (Array.isArray(response.data)) {
screenList = response.data;
}
console.log("[loadScreens] 추출된 화면 목록:", screenList);
if (screenList.length > 0) {
// 백엔드에서 카멜케이스(screenId, screenName)로 반환하므로 둘 다 처리
const transformedScreens = screenList.map((s: any) => ({
screen_id: s.screen_id || s.id,
screen_name: s.screen_name || s.name,
screen_code: s.screen_code || s.code || "",
screen_id: s.screenId ?? s.screen_id ?? s.id,
screen_name: s.screenName ?? s.screen_name ?? s.name ?? `화면 ${s.screenId || s.screen_id || s.id}`,
screen_code: s.screenCode ?? s.screen_code ?? s.code ?? "",
}));
console.log("[loadScreens] 변환된 화면 목록:", transformedScreens);
setScreens(transformedScreens);
} else {
console.warn("[loadScreens] 화면 목록이 비어있습니다");
setScreens([]);
}
} catch (error) {
console.error("화면 목록 로드 실패:", error);
setScreens([]);
} finally {
setScreensLoading(false);
}
@@ -137,17 +177,52 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
const loadColumns = useCallback(async (tableName: string, side: "left" | "right") => {
if (!tableName) return;
try {
const response = await apiClient.get(`/table/${tableName}/columns`);
const columnList = response.data?.data || response.data || [];
if (Array.isArray(columnList)) {
const response = await apiClient.get(`/table-management/tables/${tableName}/columns?size=200`);
console.log(`[loadColumns] ${side} API 응답:`, response.data);
// API 응답 구조: { success, data: { columns: [...], total, page, totalPages } }
let columnList: any[] = [];
if (response.data?.success && response.data?.data?.columns) {
columnList = response.data.data.columns;
} else if (Array.isArray(response.data?.data?.columns)) {
columnList = response.data.data.columns;
} else if (Array.isArray(response.data?.data)) {
columnList = response.data.data;
} else if (Array.isArray(response.data)) {
columnList = response.data;
}
console.log(`[loadColumns] ${side} 추출된 컬럼 목록:`, columnList);
if (columnList.length > 0) {
// 백엔드에서 카멜케이스(columnName)로 반환하므로 둘 다 처리
const transformedColumns = columnList.map((c: any) => ({
column_name: c.columnName ?? c.column_name ?? c.name ?? "",
data_type: c.dataType ?? c.data_type ?? c.type ?? "",
column_comment: c.displayName ?? c.column_comment ?? c.label ?? "",
}));
console.log(`[loadColumns] ${side} 변환된 컬럼 목록:`, transformedColumns);
if (side === "left") {
setLeftColumns(columnList);
setLeftColumns(transformedColumns);
} else {
setRightColumns(columnList);
setRightColumns(transformedColumns);
}
} else {
console.warn(`[loadColumns] ${side} 컬럼 목록이 비어있습니다`);
if (side === "left") {
setLeftColumns([]);
} else {
setRightColumns([]);
}
}
} catch (error) {
console.error(`${side} 컬럼 목록 로드 실패:`, error);
if (side === "left") {
setLeftColumns([]);
} else {
setRightColumns([]);
}
}
}, []);
@@ -177,59 +252,63 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
placeholder: string;
open: boolean;
onOpenChange: (open: boolean) => void;
}> = ({ value, onValueChange, placeholder, open, onOpenChange }) => (
<Popover open={open} onOpenChange={onOpenChange}>
<PopoverTrigger asChild>
<Button
variant="outline"
role="combobox"
aria-expanded={open}
disabled={tablesLoading}
className="w-full justify-between h-9 text-sm"
>
{tablesLoading ? (
"로딩 중..."
) : value ? (
tables.find((t) => t.table_name === value)?.table_comment || value
) : (
placeholder
)}
<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50" />
</Button>
</PopoverTrigger>
<PopoverContent className="w-full p-0" align="start">
<Command>
<CommandInput placeholder="테이블 검색..." className="h-9" />
<CommandList>
<CommandEmpty> </CommandEmpty>
<CommandGroup>
{tables.map((table) => (
<CommandItem
key={table.table_name}
value={table.table_name}
onSelect={(selectedValue) => {
onValueChange(selectedValue);
onOpenChange(false);
}}
>
<Check
className={cn(
"mr-2 h-4 w-4",
value === table.table_name ? "opacity-100" : "opacity-0"
)}
/>
<span className="flex flex-col">
<span>{table.table_comment || table.table_name}</span>
<span className="text-xs text-muted-foreground">{table.table_name}</span>
</span>
</CommandItem>
))}
</CommandGroup>
</CommandList>
</Command>
</PopoverContent>
</Popover>
);
}> = ({ value, onValueChange, placeholder, open, onOpenChange }) => {
const selectedTable = tables.find((t) => t.table_name === value);
return (
<Popover open={open} onOpenChange={onOpenChange}>
<PopoverTrigger asChild>
<Button
variant="outline"
role="combobox"
aria-expanded={open}
disabled={tablesLoading}
className="h-9 w-full justify-between text-sm"
>
{tablesLoading
? "로딩 중..."
: selectedTable
? selectedTable.table_comment || selectedTable.table_name
: value || placeholder}
<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50" />
</Button>
</PopoverTrigger>
<PopoverContent className="w-full p-0" align="start">
<Command>
<CommandInput placeholder="테이블 검색..." className="h-9" />
<CommandList>
<CommandEmpty>
{tables.length === 0 ? "테이블 목록을 불러오는 중..." : "검색 결과가 없습니다"}
</CommandEmpty>
<CommandGroup>
{tables.map((table, index) => (
<CommandItem
key={`table-${table.table_name || index}`}
value={table.table_name}
onSelect={(selectedValue) => {
onValueChange(selectedValue);
onOpenChange(false);
}}
>
<Check
className={cn(
"mr-2 h-4 w-4",
value === table.table_name ? "opacity-100" : "opacity-0"
)}
/>
<span className="flex flex-col">
<span>{table.table_comment || table.table_name}</span>
<span className="text-xs text-muted-foreground">{table.table_name}</span>
</span>
</CommandItem>
))}
</CommandGroup>
</CommandList>
</Command>
</PopoverContent>
</Popover>
);
};
// 화면 선택 컴포넌트
const ScreenSelect: React.FC<{
@@ -238,64 +317,70 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
placeholder: string;
open: boolean;
onOpenChange: (open: boolean) => void;
}> = ({ value, onValueChange, placeholder, open, onOpenChange }) => (
<Popover open={open} onOpenChange={onOpenChange}>
<PopoverTrigger asChild>
<Button
variant="outline"
role="combobox"
aria-expanded={open}
disabled={screensLoading}
className="w-full justify-between h-9 text-sm"
>
{screensLoading ? (
"로딩 중..."
) : value ? (
screens.find((s) => s.screen_id === value)?.screen_name || `화면 ${value}`
) : (
placeholder
)}
<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50" />
</Button>
</PopoverTrigger>
<PopoverContent className="w-full p-0" align="start">
<Command>
<CommandInput placeholder="화면 검색..." className="h-9" />
<CommandList>
<CommandEmpty> </CommandEmpty>
<CommandGroup>
{screens.map((screen, index) => (
<CommandItem
key={`screen-${screen.screen_id ?? index}`}
value={`${screen.screen_id}-${screen.screen_name}`}
onSelect={(selectedValue: string) => {
const screenId = parseInt(selectedValue.split("-")[0]);
console.log("[ScreenSelect] onSelect:", { selectedValue, screenId, screen });
onValueChange(screenId);
onOpenChange(false);
}}
className="flex items-center"
>
<div className="flex items-center w-full">
<Check
className={cn(
"mr-2 h-4 w-4 flex-shrink-0",
value === screen.screen_id ? "opacity-100" : "opacity-0"
)}
/>
<span className="flex flex-col">
<span>{screen.screen_name}</span>
<span className="text-xs text-muted-foreground">{screen.screen_code}</span>
</span>
</div>
</CommandItem>
))}
</CommandGroup>
</CommandList>
</Command>
</PopoverContent>
</Popover>
);
}> = ({ value, onValueChange, placeholder, open, onOpenChange }) => {
const selectedScreen = screens.find((s) => s.screen_id === value);
return (
<Popover open={open} onOpenChange={onOpenChange}>
<PopoverTrigger asChild>
<Button
variant="outline"
role="combobox"
aria-expanded={open}
disabled={screensLoading}
className="w-full justify-between h-9 text-sm"
>
{screensLoading
? "로딩 중..."
: selectedScreen
? selectedScreen.screen_name
: value
? `화면 ${value}`
: placeholder}
<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50" />
</Button>
</PopoverTrigger>
<PopoverContent className="w-full p-0" align="start">
<Command>
<CommandInput placeholder="화면 검색..." className="h-9" />
<CommandList>
<CommandEmpty>
{screens.length === 0 ? "화면 목록을 불러오는 중..." : "검색 결과가 없습니다"}
</CommandEmpty>
<CommandGroup>
{screens.map((screen, index) => (
<CommandItem
key={`screen-${screen.screen_id ?? index}`}
value={`${screen.screen_id}-${screen.screen_name}`}
onSelect={(selectedValue: string) => {
const screenId = parseInt(selectedValue.split("-")[0]);
console.log("[ScreenSelect] onSelect:", { selectedValue, screenId, screen });
onValueChange(isNaN(screenId) ? undefined : screenId);
onOpenChange(false);
}}
className="flex items-center"
>
<div className="flex items-center w-full">
<Check
className={cn(
"mr-2 h-4 w-4 shrink-0",
value === screen.screen_id ? "opacity-100" : "opacity-0"
)}
/>
<span className="flex flex-col">
<span>{screen.screen_name}</span>
<span className="text-xs text-muted-foreground">{screen.screen_code}</span>
</span>
</div>
</CommandItem>
))}
</CommandGroup>
</CommandList>
</Command>
</PopoverContent>
</Popover>
);
};
// 컬럼 선택 컴포넌트
const ColumnSelect: React.FC<{
@@ -303,20 +388,36 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
value: string;
onValueChange: (value: string) => void;
placeholder: string;
}> = ({ columns, value, onValueChange, placeholder }) => (
<Select value={value || ""} onValueChange={onValueChange}>
<SelectTrigger className="h-9 text-sm">
<SelectValue placeholder={placeholder} />
</SelectTrigger>
<SelectContent>
{columns.map((col) => (
<SelectItem key={col.column_name} value={col.column_name}>
{col.column_comment || col.column_name}
</SelectItem>
))}
</SelectContent>
</Select>
);
}> = ({ columns, value, onValueChange, placeholder }) => {
// 현재 선택된 값의 라벨 찾기
const selectedColumn = columns.find((col) => col.column_name === value);
const displayValue = selectedColumn
? selectedColumn.column_comment || selectedColumn.column_name
: value || "";
return (
<Select value={value || ""} onValueChange={onValueChange}>
<SelectTrigger className="h-9 text-sm min-w-[120px]">
<SelectValue placeholder={placeholder}>
{displayValue || placeholder}
</SelectValue>
</SelectTrigger>
<SelectContent>
{columns.length === 0 ? (
<SelectItem value="_empty" disabled>
</SelectItem>
) : (
columns.map((col) => (
<SelectItem key={col.column_name} value={col.column_name}>
{col.column_comment || col.column_name}
</SelectItem>
))
)}
</SelectContent>
</Select>
);
};
// 표시 컬럼 추가
const addDisplayColumn = (side: "left" | "right") => {
@@ -405,30 +506,52 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
<div className="flex items-center justify-between mb-2">
<Label className="text-xs"> </Label>
<Button size="sm" variant="ghost" className="h-6 text-xs" onClick={() => addDisplayColumn("left")}>
<Plus className="h-3 w-3 mr-1" />
<Plus className="mr-1 h-3 w-3" />
</Button>
</div>
<div className="space-y-2">
<div className="space-y-3">
{(config.leftPanel?.displayColumns || []).map((col, index) => (
<div key={index} className="flex gap-2 items-center">
<div key={index} className="space-y-2 rounded-md border p-3">
<div className="flex items-center justify-between">
<span className="text-xs font-medium text-muted-foreground"> {index + 1}</span>
<Button
size="sm"
variant="ghost"
className="h-6 w-6 p-0"
onClick={() => removeDisplayColumn("left", index)}
>
<X className="h-3 w-3" />
</Button>
</div>
<ColumnSelect
columns={leftColumns}
value={col.name}
onValueChange={(value) => updateDisplayColumn("left", index, "name", value)}
placeholder="컬럼"
placeholder="컬럼 선택"
/>
<Input
value={col.label || ""}
onChange={(e) => updateDisplayColumn("left", index, "label", e.target.value)}
placeholder="라벨"
className="h-9 text-sm flex-1"
/>
<Button size="sm" variant="ghost" className="h-9 w-9 p-0" onClick={() => removeDisplayColumn("left", index)}>
<X className="h-4 w-4" />
</Button>
<div>
<Label className="text-xs text-muted-foreground"> </Label>
<Select
value={col.displayRow || "name"}
onValueChange={(value) => updateDisplayColumn("left", index, "displayRow", value)}
>
<SelectTrigger className="h-8 text-xs">
<SelectValue />
</SelectTrigger>
<SelectContent>
<SelectItem value="name"> (Name Row)</SelectItem>
<SelectItem value="info"> (Info Row)</SelectItem>
</SelectContent>
</Select>
</div>
</div>
))}
{(config.leftPanel?.displayColumns || []).length === 0 && (
<div className="rounded-md border py-4 text-center text-xs text-muted-foreground">
</div>
)}
</div>
</div>
@@ -440,6 +563,61 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
/>
</div>
{config.leftPanel?.showSearch && (
<div>
<div className="mb-2 flex items-center justify-between">
<Label className="text-xs"> </Label>
<Button
size="sm"
variant="ghost"
className="h-6 text-xs"
onClick={() => {
const current = config.leftPanel?.searchColumns || [];
updateConfig("leftPanel.searchColumns", [...current, { columnName: "", label: "" }]);
}}
>
<Plus className="mr-1 h-3 w-3" />
</Button>
</div>
<div className="space-y-2">
{(config.leftPanel?.searchColumns || []).map((searchCol, index) => (
<div key={index} className="flex items-center gap-2">
<ColumnSelect
columns={leftColumns}
value={searchCol.columnName}
onValueChange={(value) => {
const current = [...(config.leftPanel?.searchColumns || [])];
current[index] = { ...current[index], columnName: value };
updateConfig("leftPanel.searchColumns", current);
}}
placeholder="컬럼 선택"
/>
<Button
size="sm"
variant="ghost"
className="h-8 w-8 shrink-0 p-0"
onClick={() => {
const current = config.leftPanel?.searchColumns || [];
updateConfig(
"leftPanel.searchColumns",
current.filter((_, i) => i !== index)
);
}}
>
<X className="h-3 w-3" />
</Button>
</div>
))}
{(config.leftPanel?.searchColumns || []).length === 0 && (
<div className="rounded-md border py-3 text-center text-xs text-muted-foreground">
</div>
)}
</div>
</div>
)}
<div className="flex items-center justify-between">
<Label className="text-xs"> </Label>
<Switch
@@ -505,30 +683,52 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
<div className="flex items-center justify-between mb-2">
<Label className="text-xs"> </Label>
<Button size="sm" variant="ghost" className="h-6 text-xs" onClick={() => addDisplayColumn("right")}>
<Plus className="h-3 w-3 mr-1" />
<Plus className="mr-1 h-3 w-3" />
</Button>
</div>
<div className="space-y-2">
<div className="space-y-3">
{(config.rightPanel?.displayColumns || []).map((col, index) => (
<div key={index} className="flex gap-2 items-center">
<div key={index} className="rounded-md border p-3 space-y-2">
<div className="flex items-center justify-between">
<span className="text-xs font-medium text-muted-foreground"> {index + 1}</span>
<Button
size="sm"
variant="ghost"
className="h-6 w-6 p-0"
onClick={() => removeDisplayColumn("right", index)}
>
<X className="h-3 w-3" />
</Button>
</div>
<ColumnSelect
columns={rightColumns}
value={col.name}
onValueChange={(value) => updateDisplayColumn("right", index, "name", value)}
placeholder="컬럼"
placeholder="컬럼 선택"
/>
<Input
value={col.label || ""}
onChange={(e) => updateDisplayColumn("right", index, "label", e.target.value)}
placeholder="라벨"
className="h-9 text-sm flex-1"
/>
<Button size="sm" variant="ghost" className="h-9 w-9 p-0" onClick={() => removeDisplayColumn("right", index)}>
<X className="h-4 w-4" />
</Button>
<div>
<Label className="text-xs text-muted-foreground"> </Label>
<Select
value={col.displayRow || "info"}
onValueChange={(value) => updateDisplayColumn("right", index, "displayRow", value)}
>
<SelectTrigger className="h-8 text-xs">
<SelectValue />
</SelectTrigger>
<SelectContent>
<SelectItem value="name"> (Name Row)</SelectItem>
<SelectItem value="info"> (Info Row)</SelectItem>
</SelectContent>
</Select>
</div>
</div>
))}
{(config.rightPanel?.displayColumns || []).length === 0 && (
<div className="text-center py-4 text-xs text-muted-foreground border rounded-md">
</div>
)}
</div>
</div>
@@ -540,6 +740,61 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
/>
</div>
{config.rightPanel?.showSearch && (
<div>
<div className="mb-2 flex items-center justify-between">
<Label className="text-xs"> </Label>
<Button
size="sm"
variant="ghost"
className="h-6 text-xs"
onClick={() => {
const current = config.rightPanel?.searchColumns || [];
updateConfig("rightPanel.searchColumns", [...current, { columnName: "", label: "" }]);
}}
>
<Plus className="mr-1 h-3 w-3" />
</Button>
</div>
<div className="space-y-2">
{(config.rightPanel?.searchColumns || []).map((searchCol, index) => (
<div key={index} className="flex items-center gap-2">
<ColumnSelect
columns={rightColumns}
value={searchCol.columnName}
onValueChange={(value) => {
const current = [...(config.rightPanel?.searchColumns || [])];
current[index] = { ...current[index], columnName: value };
updateConfig("rightPanel.searchColumns", current);
}}
placeholder="컬럼 선택"
/>
<Button
size="sm"
variant="ghost"
className="h-8 w-8 shrink-0 p-0"
onClick={() => {
const current = config.rightPanel?.searchColumns || [];
updateConfig(
"rightPanel.searchColumns",
current.filter((_, i) => i !== index)
);
}}
>
<X className="h-3 w-3" />
</Button>
</div>
))}
{(config.rightPanel?.searchColumns || []).length === 0 && (
<div className="rounded-md border py-3 text-center text-xs text-muted-foreground">
</div>
)}
</div>
</div>
)}
<div className="flex items-center justify-between">
<Label className="text-xs"> </Label>
<Switch
@@ -576,7 +831,14 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
{/* 연결 설정 */}
<div className="space-y-4">
<h4 className="font-medium text-sm border-b pb-2"> ()</h4>
<h4 className="border-b pb-2 text-sm font-medium"> ()</h4>
{/* 설명 */}
<div className="rounded-md bg-muted/50 p-3 text-xs text-muted-foreground">
<p className="mb-1 font-medium text-foreground"> </p>
<p> .</p>
<p className="mt-1 text-[10px]">: 부서(dept_code) </p>
</div>
<div className="space-y-3">
<div>
@@ -604,19 +866,31 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
{/* 데이터 전달 설정 */}
<div className="space-y-4">
<div className="flex items-center justify-between border-b pb-2">
<h4 className="font-medium text-sm"> </h4>
<h4 className="text-sm font-medium"> </h4>
<Button size="sm" variant="ghost" className="h-6 text-xs" onClick={addDataTransferField}>
<Plus className="h-3 w-3 mr-1" />
<Plus className="mr-1 h-3 w-3" />
</Button>
</div>
{/* 설명 */}
<div className="rounded-md bg-muted/50 p-3 text-xs text-muted-foreground">
<p className="mb-1 font-medium text-foreground"> </p>
<p> .</p>
<p className="mt-1 text-[10px]">: dept_code를 dept_code </p>
</div>
<div className="space-y-3">
{(config.dataTransferFields || []).map((field, index) => (
<div key={index} className="space-y-2 p-3 border rounded-md">
<div key={index} className="space-y-2 rounded-md border p-3">
<div className="flex items-center justify-between">
<span className="text-xs font-medium"> {index + 1}</span>
<Button size="sm" variant="ghost" className="h-6 w-6 p-0" onClick={() => removeDataTransferField(index)}>
<Button
size="sm"
variant="ghost"
className="h-6 w-6 p-0"
onClick={() => removeDataTransferField(index)}
>
<X className="h-3 w-3" />
</Button>
</div>
@@ -640,6 +914,11 @@ export const SplitPanelLayout2ConfigPanel: React.FC<SplitPanelLayout2ConfigPanel
</div>
</div>
))}
{(config.dataTransferFields || []).length === 0 && (
<div className="rounded-md border py-4 text-center text-xs text-muted-foreground">
</div>
)}
</div>
</div>