From 905a9f62c38b214660907090b1cc8f7df63bd8ba Mon Sep 17 00:00:00 2001 From: DDD1542 Date: Tue, 13 Jan 2026 13:28:50 +0900 Subject: [PATCH] =?UTF-8?q?feat:=20=ED=94=84=EB=A6=AC=EB=B7=B0=20=EB=AA=A8?= =?UTF-8?q?=EB=93=9C=EC=97=90=EC=84=9C=20=ED=9A=8C=EC=82=AC=20=EC=BD=94?= =?UTF-8?q?=EB=93=9C=20=EC=98=A4=EB=B2=84=EB=9D=BC=EC=9D=B4=EB=93=9C=20?= =?UTF-8?q?=EA=B8=B0=EB=8A=A5=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - 최고 관리자만 다른 회사 코드로 오버라이드 가능하도록 로직 개선 - entityJoinController 및 tableManagementController에서 회사 코드 오버라이드 처리 추가 - 관련 API 호출 시 오버라이드된 회사 코드 적용 - 프리뷰 모드 감지 및 UI 개선을 위한 코드 추가 --- backend-node/package-lock.json | 13 +- .../src/controllers/entityJoinController.ts | 18 +- .../src/controllers/screenGroupController.ts | 2 +- .../controllers/tableManagementController.ts | 18 +- docs/화면관계_시각화_개선_보고서.md | 55 + docs/화면설정모달_개선_완료_보고서.md | 535 ++++ .../app/(main)/screens/[screenId]/page.tsx | 46 +- .../components/screen/ScreenRelationFlow.tsx | 27 + .../components/screen/ScreenSettingModal.tsx | 2825 +++++++++++++++-- .../screen/panels/DataFlowPanel.tsx | 2 + .../screen/panels/FieldJoinPanel.tsx | 2 + frontend/lib/api/entityJoin.ts | 15 +- .../SplitPanelLayoutComponent.tsx | 6 + .../table-list/TableListComponent.tsx | 6 + frontend/package-lock.json | 54 +- frontend/package.json | 1 + 16 files changed, 3328 insertions(+), 297 deletions(-) create mode 100644 docs/화면설정모달_개선_완료_보고서.md diff --git a/backend-node/package-lock.json b/backend-node/package-lock.json index f826a86a..d801ddbb 100644 --- a/backend-node/package-lock.json +++ b/backend-node/package-lock.json @@ -1043,6 +1043,7 @@ "integrity": "sha512-2BCOP7TN8M+gVDj7/ht3hsaO/B/n5oDbiAyyvnRlNOs+u1o+JWNYTQrmpuNp1/Wq2gcFrI01JAW+paEKDMx/CA==", "dev": true, "license": "MIT", + "peer": true, "dependencies": { "@babel/code-frame": "^7.27.1", "@babel/generator": "^7.28.3", @@ -2370,6 +2371,7 @@ "resolved": "https://registry.npmjs.org/@redis/client/-/client-1.6.1.tgz", "integrity": "sha512-/KCsg3xSlR+nCK8/8ZYSknYxvXHwubJrU82F3Lm1Fp6789VQ0/3RJKfsmRXjqfaTA++23CvC3hqmqe/2GEt6Kw==", "license": "MIT", + "peer": true, "dependencies": { "cluster-key-slot": "1.1.2", "generic-pool": "3.9.0", @@ -3463,6 +3465,7 @@ "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.17.tgz", "integrity": "sha512-gfehUI8N1z92kygssiuWvLiwcbOB3IRktR6hTDgJlXMYh5OvkPSRmgfoBUmfZt+vhwJtX7v1Yw4KvvAf7c5QKQ==", "license": "MIT", + "peer": true, "dependencies": { "undici-types": "~6.21.0" } @@ -3699,6 +3702,7 @@ "integrity": "sha512-tbsV1jPne5CkFQCgPBcDOt30ItF7aJoZL997JSF7MhGQqOeT3svWRYxiqlfA5RUdlHN6Fi+EI9bxqbdyAUZjYQ==", "dev": true, "license": "BSD-2-Clause", + "peer": true, "dependencies": { "@typescript-eslint/scope-manager": "6.21.0", "@typescript-eslint/types": "6.21.0", @@ -3916,6 +3920,7 @@ "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", "dev": true, "license": "MIT", + "peer": true, "bin": { "acorn": "bin/acorn" }, @@ -4442,6 +4447,7 @@ } ], "license": "MIT", + "peer": true, "dependencies": { "baseline-browser-mapping": "^2.8.3", "caniuse-lite": "^1.0.30001741", @@ -5652,6 +5658,7 @@ "deprecated": "This version is no longer supported. Please see https://eslint.org/version-support for other options.", "dev": true, "license": "MIT", + "peer": true, "dependencies": { "@eslint-community/eslint-utils": "^4.2.0", "@eslint-community/regexpp": "^4.6.1", @@ -7414,6 +7421,7 @@ "integrity": "sha512-NIy3oAFp9shda19hy4HK0HRTWKtPJmGdnvywu01nOqNC2vZg+Z+fvJDxpMQA88eb2I9EcafcdjYgsDthnYTvGw==", "dev": true, "license": "MIT", + "peer": true, "dependencies": { "@jest/core": "^29.7.0", "@jest/types": "^29.6.3", @@ -8383,7 +8391,6 @@ "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", "license": "MIT", - "peer": true, "dependencies": { "js-tokens": "^3.0.0 || ^4.0.0" }, @@ -9272,6 +9279,7 @@ "resolved": "https://registry.npmjs.org/pg/-/pg-8.16.3.tgz", "integrity": "sha512-enxc1h0jA/aq5oSDMvqyW3q89ra6XIIDZgCX9vkMrnz5DFTw/Ny3Li2lFQ+pt3L6MCgm/5o2o8HW9hiJji+xvw==", "license": "MIT", + "peer": true, "dependencies": { "pg-connection-string": "^2.9.1", "pg-pool": "^3.10.1", @@ -10122,7 +10130,6 @@ "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz", "integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==", "license": "MIT", - "peer": true, "dependencies": { "loose-envify": "^1.1.0" } @@ -10931,6 +10938,7 @@ "integrity": "sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==", "dev": true, "license": "MIT", + "peer": true, "dependencies": { "@cspotcode/source-map-support": "^0.8.0", "@tsconfig/node10": "^1.0.7", @@ -11036,6 +11044,7 @@ "integrity": "sha512-CWBzXQrc/qOkhidw1OzBTQuYRbfyxDXJMVJ1XNwUHGROVmuaeiEm3OslpZ1RV96d7SKKjZKrSJu3+t/xlw3R9A==", "dev": true, "license": "Apache-2.0", + "peer": true, "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" diff --git a/backend-node/src/controllers/entityJoinController.ts b/backend-node/src/controllers/entityJoinController.ts index fbb88750..4a541456 100644 --- a/backend-node/src/controllers/entityJoinController.ts +++ b/backend-node/src/controllers/entityJoinController.ts @@ -66,11 +66,23 @@ export class EntityJoinController { const userField = parsedAutoFilter.userField || "companyCode"; const userValue = ((req as any).user as any)[userField]; - if (userValue) { - searchConditions[filterColumn] = userValue; + // 🆕 프리뷰용 회사 코드 오버라이드 (최고 관리자만 허용) + let finalCompanyCode = userValue; + if (parsedAutoFilter.companyCodeOverride && userValue === "*") { + // 최고 관리자만 다른 회사 코드로 오버라이드 가능 + finalCompanyCode = parsedAutoFilter.companyCodeOverride; + logger.info("🔓 최고 관리자 회사 코드 오버라이드:", { + originalCompanyCode: userValue, + overrideCompanyCode: parsedAutoFilter.companyCodeOverride, + tableName, + }); + } + + if (finalCompanyCode) { + searchConditions[filterColumn] = finalCompanyCode; logger.info("🔒 Entity 조인에 멀티테넌시 필터 적용:", { filterColumn, - userValue, + finalCompanyCode, tableName, }); } diff --git a/backend-node/src/controllers/screenGroupController.ts b/backend-node/src/controllers/screenGroupController.ts index 518de7e8..52464ed4 100644 --- a/backend-node/src/controllers/screenGroupController.ts +++ b/backend-node/src/controllers/screenGroupController.ts @@ -1652,7 +1652,7 @@ export const getScreenSubTables = async (req: Request, res: Response) => { }); } }); - + rightPanelResult.rows.forEach((row: any) => { const screenId = row.screen_id; const mainTable = row.main_table; diff --git a/backend-node/src/controllers/tableManagementController.ts b/backend-node/src/controllers/tableManagementController.ts index 7c84898b..5bbec536 100644 --- a/backend-node/src/controllers/tableManagementController.ts +++ b/backend-node/src/controllers/tableManagementController.ts @@ -775,13 +775,25 @@ export async function getTableData( const userField = autoFilter?.userField || "companyCode"; const userValue = (req.user as any)[userField]; - if (userValue) { - enhancedSearch[filterColumn] = userValue; + // 🆕 프리뷰용 회사 코드 오버라이드 (최고 관리자만 허용) + let finalCompanyCode = userValue; + if (autoFilter?.companyCodeOverride && userValue === "*") { + // 최고 관리자만 다른 회사 코드로 오버라이드 가능 + finalCompanyCode = autoFilter.companyCodeOverride; + logger.info("🔓 최고 관리자 회사 코드 오버라이드:", { + originalCompanyCode: userValue, + overrideCompanyCode: autoFilter.companyCodeOverride, + tableName, + }); + } + + if (finalCompanyCode) { + enhancedSearch[filterColumn] = finalCompanyCode; logger.info("🔍 현재 사용자 필터 적용:", { filterColumn, userField, - userValue, + userValue: finalCompanyCode, tableName, }); } else { diff --git a/docs/화면관계_시각화_개선_보고서.md b/docs/화면관계_시각화_개선_보고서.md index 83411d7f..27946afa 100644 --- a/docs/화면관계_시각화_개선_보고서.md +++ b/docs/화면관계_시각화_개선_보고서.md @@ -1682,6 +1682,61 @@ frontend/ --- +## 화면 설정 모달 개선 (2026-01-12) + +### 개요 + +화면 노드 우클릭 시 열리는 설정 모달을 대폭 개선했습니다. + +### 주요 변경 사항 + +#### 1. 테이블 정보 시각화 개선 + +| 항목 | 변경 내용 | +|------|----------| +| 메인 테이블 | 아코디언 형식으로 모든 컬럼 표시 | +| 필터 테이블 | 아코디언 형식 + 필터/조인 키 색상 구분 | +| 사용 중 컬럼 | 파란색 배경 + "필드" 배지로 강조 | + +#### 2. 화면 프리뷰 상시 표시 + +- 모달 레이아웃: 좌측 40% (탭) / 우측 60% (프리뷰) +- 탭 전환해도 프리뷰 항상 표시 + +#### 3. 줌/드래그 기능 (react-zoom-pan-pinch 라이브러리) + +```bash +npm install react-zoom-pan-pinch +``` + +| 기능 | 동작 | +|------|------| +| 휠 스크롤 | 마우스 포인터 기준 확대/축소 (20%~300%) | +| 드래그 | 화면 이동 | +| 클릭 | iframe 내부 버튼/목록 상호작용 | + +#### 4. 프리뷰 company_code 전달 문제 해결 + +| 문제 | 해결 | +|------|------| +| 최고 관리자로 다른 회사 프리뷰 불가 | `companyCodeOverride` 파라미터 도입 | +| URL 파라미터 무시됨 | 백엔드에서 admin 전용 오버라이드 처리 | + +### 관련 파일 + +| 파일 | 변경 내용 | +|------|----------| +| `ScreenSettingModal.tsx` | 전체 UI 개선, 줌/드래그 기능 | +| `entityJoin.ts` | `companyCodeOverride` 파라미터 추가 | +| `SplitPanelLayoutComponent.tsx` | `companyCode` prop 추가 | +| `entityJoinController.ts` | `companyCodeOverride` 처리 로직 | + +### 상세 문서 + +- [화면설정모달_개선_완료_보고서.md](./화면설정모달_개선_완료_보고서.md) + +--- + ## 관련 문서 - [멀티테넌시 구현 가이드](.cursor/rules/multi-tenancy-guide.mdc) diff --git a/docs/화면설정모달_개선_완료_보고서.md b/docs/화면설정모달_개선_완료_보고서.md new file mode 100644 index 00000000..493c3f37 --- /dev/null +++ b/docs/화면설정모달_개선_완료_보고서.md @@ -0,0 +1,535 @@ +# 화면 설정 모달 개선 완료 보고서 + +## 개요 +화면 관리에서 화면 노드 우클릭 시 열리는 설정 모달을 대폭 개선하여, 테이블 정보 시각화, 필드 매핑 확인, 컬럼 변경/추가/제거 기능, 조인 설정 기능, 실시간 프리뷰 기능을 강화했습니다. + +## 주요 개선 사항 + +### 1. 화면 개요 탭 통합 개선 + +#### 1.1 필드 매핑 탭 → 개요 탭 통합 +- 기존 "필드 매핑" 탭 제거 +- 필드 매핑 정보를 개요 탭의 메인/필터 테이블 아코디언에 통합 표시 +- 더 직관적이고 간결한 UI 제공 + +#### 1.2 메인 테이블 아코디언 +- 메인 테이블(예: `customer_mng`)을 아코디언 형식으로 표시 +- 클릭 시 테이블의 모든 컬럼 정보 표시 +- **1열 레이아웃**: 컬럼 정보를 세로로 배치 +- 화면에서 사용 중인 컬럼은 **파란색 배경 + "필드" 배지**로 강조 +- **컬럼 정렬**: + - 사용중인 필드가 상단에 표시 + - 화면에 표시되는 순서대로 정렬 (y좌표 기준) + - 미사용 컬럼은 하단에 표시 + +#### 1.3 필터 테이블 아코디언 +- 필터 테이블(예: `customer_item_mapping`)을 아코디언 형식으로 표시 +- 클릭 시 테이블의 모든 컬럼 정보 표시 +- 컬럼별 색상 구분: + - **파란색**: 화면에서 사용 중인 컬럼 (필드) + - **보라색**: 필터 키 컬럼 (WHERE 절에 사용) + - **주황색**: 조인 키 컬럼 (JOIN 조건에 사용) +- **다중 배지 표시**: 컬럼이 필드이면서 조인/필터 키인 경우 배지 동시 표시 +- 필터 연결 정보 표시 (예: `→ customer_mng`) + +#### 1.4 컬럼 레이아웃 순서 +- **순서**: `컬럼명 | 배지 | 데이터타입` +- 예: `거래처 코드` `[필드]` `character varying` +- 데이터타입은 오른쪽 정렬 + +#### 1.5 클릭 스타일 개선 +- **테두리 제거**: ring-2, ring-offset 등 제거 +- **강조 색상 연하게**: + - 선택됨: `bg-blue-100 border-blue-300` + - 미선택: `bg-blue-50 border-blue-200` +- 더 부드러운 시각적 피드백 + +#### 1.6 패널 높이 동기화 +- 왼쪽(컬럼 목록)과 오른쪽(설정 패널) 동일한 `max-h-[350px]` 적용 +- `overflow-y-auto`로 스크롤 처리 +- `items-stretch`로 양쪽 패널 높이 동기화 + +### 2. 컬럼 변경 기능 + +#### 2.1 인라인 컬럼 편집 +- 사용중인 필드(파란색 배경)를 클릭하면 우측에 "컬럼 설정" 패널 표시 +- 패널 정보: + - **화면 필드**: 컬럼 한글명 표시 (예: "거래처 코드") + - **현재 컬럼**: 영문 컬럼명 표시 (예: `customer_code`) + - **컬럼 변경**: 드롭다운으로 다른 컬럼 선택 +- 검색 기능으로 컬럼 빠르게 찾기 + +#### 2.2 실시간 반영 +- 컬럼 변경 후 **페이지 새로고침 없이** 실시간 반영 +- `onRefresh` 콜백으로 데이터 리로드 + iframe 새로고침 +- 더 빠른 사용자 경험 + +#### 2.3 변경사항 저장 +- `screenApi.saveLayout()` 사용하여 **화면 디자이너와 동일한 테이블에 저장** +- 저장 위치: + - `componentConfig.leftPanel.columns` (분할 패널) + - `componentConfig.rightPanel.columns` (분할 패널) + - `usedColumns` 배열 + - `bindField` 필드 + - `fieldMapping` 배열 + +### 3. 필드 추가/제거 기능 (신규) + +#### 3.1 필드 추가 +- 비필드 컬럼(회색/흰색 배경) 클릭 +- "컬럼 설정" 패널에 컬럼 정보 표시 +- **"필드로 추가"** 버튼 클릭 → 해당 컬럼이 화면 필드로 추가됨 +- 버튼 스타일: `text-blue-600 border-blue-300 hover:bg-blue-50` (파란색 테두리) + +#### 3.2 필드 제거 +- 기존 필드(파란색 배경) 클릭 +- "컬럼 설정" 패널에 필드 정보 표시 +- **"필드에서 제거"** 버튼 클릭 → 해당 필드가 화면에서 제거됨 +- 버튼 스타일: `text-red-600 border-red-300 hover:bg-red-50` (빨간색 테두리) + +#### 3.3 저장 로직 +```typescript +// 필드 추가: 배열에 새 컬럼 추가 +if (isAddingField) { + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: [...leftColumns, { name: newColumn, columnName: newColumn }], + }, + }, + }; +} + +// 필드 제거: 배열에서 해당 컬럼 제거 +if (isRemovingField) { + const filteredColumns = leftColumns.filter((_, i) => i !== columnIdx); + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: filteredColumns, + }, + }, + }; +} +``` + +#### 3.4 적용 범위 +- 메인 테이블 아코디언: 필드 추가/제거 가능 +- 필터 테이블 아코디언: 필드 추가/제거 가능 +- `usedColumns`, `componentConfig.usedColumns`, `componentConfig.columns`, `leftPanel.columns`, `rightPanel.columns` 모두 지원 + +### 4. 화면 프리뷰 상시 표시 + +#### 4.1 레이아웃 변경 +- 기존: 탭으로 프리뷰 전환 +- 개선: **모달 우측에 프리뷰 상시 표시** +- 모달 크기 확대 (1600px 최대 너비) +- 좌측 40% (탭 콘텐츠) / 우측 60% (프리뷰) + +#### 4.2 줌/드래그/클릭 기능 (react-zoom-pan-pinch 라이브러리) +- **휠 스크롤**: 마우스 포인터 위치 기준 확대/축소 (20% ~ 300%) +- **드래그**: 마우스 왼쪽 버튼으로 화면 이동 (5px 이상 이동 시) +- **클릭**: iframe 내부 요소 정상 클릭 가능 + - 버튼, 셀렉트박스, 체크박스, 테이블 행 클릭 + - 인풋박스/텍스트박스 포커스 및 입력 + - X버튼(닫기) 등 SVG 아이콘 버튼 클릭 + +#### 4.3 클릭 좌표 보정 시스템 +- 줌 상태에서도 정확한 클릭 위치 계산 +- `designWidth / rect.width` 비율로 좌표 변환 +- 오버레이 방식으로 드래그와 클릭 분리 처리 + +### 5. 필드+조인 컬럼 스타일 개선 + +#### 5.1 다중 역할 컬럼 표시 +- 컬럼이 **필드이면서 조인 키**인 경우: + - **파란색 배경** (필드 기준) + - **왼쪽에 주황색 세로 선** (`border-l-4 border-l-orange-500`) + - 배지: `조인` `필드` 동시 표시 +- 컬럼이 **필드이면서 필터 키**인 경우: + - **파란색 배경** (필드 기준) + - **왼쪽에 보라색 세로 선** (`border-l-4 border-l-purple-400`) + - 배지: `필터` `필드` 동시 표시 + +#### 5.2 조인 컬럼도 필드로 인식 +- `filterTableColumnMappings` 생성 시 조인 컬럼(`ft.joinColumnRefs`)도 포함 +- 조인 테이블 데이터를 화면에서 보여주므로 필드로 간주 + +#### 5.3 컬럼 설정 패널 - 조인 정보 표시 +- 조인 키 클릭 시 패널에 조인 정보 표시: + - **대상 테이블**: item_info (실제 참조 테이블) + - **연결 컬럼**: item_number (참조 컬럼) + +### 6. 조인 관계 설정/수정 기능 + +#### 6.1 기능 설명 +- 컬럼 설정 패널에서 **조인 관계 직접 수정** 가능 +- **모든 컬럼에서 조인 설정 가능** (기존 조인 키가 아닌 컬럼도 포함) +- 테이블 타입 관리(`column_labels` 테이블)와 동일한 저장 위치 사용 + +#### 6.2 저장 테이블 +``` +column_labels 테이블: +├── reference_table (참조 테이블명) +├── reference_column (참조 컬럼 - 보통 PK) +└── display_column (화면에 표시할 컬럼) +``` + +#### 6.3 구현된 UI +1. **컬럼 클릭** → 컬럼 설정 패널 표시 +2. **"조인" 섹션 확인**: + - 조인 설정 있음: "편집" 버튼 + - 조인 설정 없음: "추가" 버튼 +3. **드롭다운으로 설정** (모두 검색 가능): + - 대상 테이블: 전체 테이블 목록에서 검색/선택 + - 연결 컬럼 (PK): 선택한 테이블의 컬럼 중 검색/선택 + - 표시 컬럼: 화면에 표시할 컬럼 검색/선택 +4. **저장 버튼** → `column_labels` 테이블에 저장 +5. **취소 버튼** → 편집 취소 + +#### 6.4 검색 가능한 드롭다운 +- Popover + Command 컴포넌트 사용 +- 실시간 텍스트 검색 지원 +- 대상 테이블, 연결 컬럼, 표시 컬럼 모두 검색 가능 + +#### 6.5 API 연동 +- **테이블 목록 조회**: `tableManagementApi.getTableList()` +- **컬럼 목록 조회**: `tableManagementApi.getColumnList(tableName)` +- **저장**: `tableManagementApi.updateColumnSettings(tableName, columnName, settings)` + +#### 6.6 메인 테이블에도 조인 설정 적용 +- 메인 테이블 아코디언에서도 조인 설정 가능 +- 필터 테이블과 동일한 UI/기능 제공 + +#### 6.7 조인 데이터 소스 수정 +- 기존 조인 키 클릭 시 `joinRef.refTable` 값을 사용 +- 예: `품목 ID` → `item_info` (실제 참조 테이블) +- `mainTable` 대신 `joinRef.refTable` 사용으로 정확한 테이블 표시 + +### 7. 배지 순서 및 스타일 +- **배지 순서**: `필터` → `조인` → `필드` (필드가 맨 뒤) +- **조인 배지**: 주황색 배경 (`bg-orange-200 text-orange-700`) +- **필터 배지**: 보라색 배경 (`bg-purple-200 text-purple-700`) +- **필드 배지**: 파란색 배경 (`bg-blue-500 text-white`) + +### 8. 컴포넌트 통합 리팩토링 + +#### 8.1 `TableColumnAccordion` 통합 컴포넌트 +- 기존 `MainTableAccordion`과 `FilterTableAccordion`을 하나의 컴포넌트로 통합 +- `tableType` prop으로 "main" 또는 "filter" 구분 +- 코드 중복 제거 및 유지보수성 향상 + +#### 8.2 Props 구조 +```typescript +interface TableColumnAccordionProps { + tableName: string; + tableLabel?: string; + tableType: "main" | "filter"; + columnMappings?: ColumnMapping[]; + onColumnChange?: (fieldLabel: string, oldColumn: string, newColumn: string) => void; + onColumnReorder?: (newOrder: string[]) => void; + onJoinSettingSaved?: () => void; + // 필터 테이블 전용 props + mainTable?: string; + filterKeyMapping?: FilterKeyMapping; + joinColumnRefs?: JoinColumnRef[]; +} +``` + +#### 8.3 동적 테마 적용 +- 메인 테이블: 파란색 테마 (`blue`) +- 필터 테이블: 보라색 테마 (`purple`) +- `themeColor`, `themeIcon`, `themeBadge` 변수로 동적 스타일 적용 + +### 9. 드래그 앤 드롭 컬럼 순서 변경 + +#### 9.1 기능 설명 +- 사용 중인 컬럼(필드)을 드래그하여 순서 변경 가능 +- 드래그 중에는 시각적으로만 순서 변경, **드롭 시에만 저장** +- 드래그 취소(영역 밖으로 나간 경우) 시 원래 순서로 복원 + +#### 9.2 드래그 상태 관리 +```typescript +// 드래그 상태 +const [draggedIndex, setDraggedIndex] = useState(null); +const [localColumnOrder, setLocalColumnOrder] = useState(null); +``` + +#### 9.3 드래그 핸들러 +```typescript +// 드래그 시작: 현재 순서를 로컬 상태로 저장 +const handleDragStart = (e: React.DragEvent, index: number) => { + setDraggedIndex(index); + const usedColumns = sortedColumns.filter(col => columnMappingMap.has(col.columnName.toLowerCase())); + setLocalColumnOrder(usedColumns.map(col => col.columnName)); +}; + +// 드래그 중: 로컬 순서만 변경 (저장하지 않음) +const handleDragOver = (e: React.DragEvent, hoverIndex: number) => { + if (draggedIndex === null || draggedIndex === hoverIndex || !localColumnOrder) return; + const newOrder = [...localColumnOrder]; + const draggedItem = newOrder[draggedIndex]; + newOrder.splice(draggedIndex, 1); + newOrder.splice(hoverIndex, 0, draggedItem); + setDraggedIndex(hoverIndex); + setLocalColumnOrder(newOrder); +}; + +// 드롭: 최종 순서로 저장 +const handleDrop = (e: React.DragEvent) => { + if (localColumnOrder && onColumnReorder) { + onColumnReorder(localColumnOrder); + } + setDraggedIndex(null); + setLocalColumnOrder(null); +}; + +// 드래그 취소 +const handleDragEnd = () => { + setDraggedIndex(null); + setLocalColumnOrder(null); +}; +``` + +#### 9.4 시각적 피드백 +- 드래그 가능한 컬럼: `cursor-grab active:cursor-grabbing` +- 드래그 중인 컬럼: `opacity-50 scale-95` +- 드래그 중 실시간 순서 변경 표시 + +#### 9.5 저장 로직 (`handleColumnReorder`) +```typescript +const handleColumnReorder = async (tableType: "main" | "filter", newOrder: string[]) => { + const currentLayout = await screenApi.getLayout(screenId); + + const updatedComponents = currentLayout.components.map((comp: any) => { + // leftPanel.columns 순서 변경 + if (comp.componentConfig?.leftPanel?.columns) { + const leftColumns = comp.componentConfig.leftPanel.columns; + const reorderedColumns = newOrder.map(colName => + leftColumns.find((col: any) => col.name?.toLowerCase() === colName.toLowerCase()) + ).filter(Boolean); + const remainingColumns = leftColumns.filter((col: any) => + !newOrder.some(n => n.toLowerCase() === col.name?.toLowerCase()) + ); + + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: [...reorderedColumns, ...remainingColumns], + }, + }, + }; + } + return comp; + }); + + await screenApi.saveLayout(screenId, { ...currentLayout, components: updatedComponents }); + onRefresh?.(); +}; +``` + +#### 9.6 지원 범위 +- 메인 테이블: `onColumnReorder={(newOrder) => handleColumnReorder("main", newOrder)}` +- 필터 테이블: `onColumnReorder={(newOrder) => handleColumnReorder("filter", newOrder)}` +- 지원 배열: + - `componentConfig.leftPanel.columns` + - `componentConfig.rightPanel.columns` + - `componentConfig.usedColumns` + - `componentConfig.columns` + +## 기술 스택 + +### 신규 의존성 +```bash +npm install react-zoom-pan-pinch +``` + +### 사용된 컴포넌트 +- `TransformWrapper`, `TransformComponent` - 줌/드래그 기능 +- `Accordion`, `AccordionContent`, `AccordionItem`, `AccordionTrigger` - 아코디언 UI +- `Popover`, `PopoverTrigger`, `PopoverContent` - 드롭다운 컨테이너 +- `Command`, `CommandInput`, `CommandList`, `CommandItem`, `CommandEmpty` - 검색 가능한 선택 UI +- `tableManagementApi.getColumnList()` - 테이블 컬럼 정보 조회 +- `tableManagementApi.getTableList()` - 테이블 목록 조회 +- `tableManagementApi.updateColumnSettings()` - 조인 설정 저장 +- `screenApi.saveLayout()` - 레이아웃 저장 +- `screenApi.getLayout()` - 레이아웃 조회 + +### 핵심 로직 + +#### 컬럼 변경/추가/제거 +```typescript +const handleColumnChange = async (fieldLabel: string, oldColumn: string, newColumn: string) => { + const isAddingField = fieldLabel === "__NEW_FIELD__"; + const isRemovingField = newColumn === "__REMOVE_FIELD__"; + + const currentLayout = await screenApi.getLayout(screenId); + + const updatedComponents = currentLayout.components.map((comp: any) => { + if (comp.componentConfig?.leftPanel?.columns) { + const leftColumns = comp.componentConfig.leftPanel.columns; + + // 필드 추가 + if (isAddingField) { + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: [...leftColumns, { name: newColumn, columnName: newColumn }], + }, + }, + }; + } + + // 필드 제거 + const columnIdx = leftColumns.findIndex((col: any) => ...); + if (columnIdx !== -1 && isRemovingField) { + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: leftColumns.filter((_, i) => i !== columnIdx), + }, + }, + }; + } + + // 컬럼 변경 + if (columnIdx !== -1) { + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: leftColumns.map((col, i) => + i === columnIdx ? { ...col, name: newColumn } : col + ), + }, + }, + }; + } + } + return comp; + }); + + await screenApi.saveLayout(screenId, { ...currentLayout, components: updatedComponents }); + onRefresh?.(); +}; +``` + +#### 조인 설정 편집기 (JoinSettingEditor) +```tsx + +``` + +## 파일 변경 목록 + +| 파일 | 변경 내용 | +|------|----------| +| `frontend/components/screen/ScreenSettingModal.tsx` | 전체 UI 개선, 줌/드래그 기능, 컬럼 변경/추가/제거 기능, 조인 설정 기능, 필드 매핑 통합, 실시간 반영 | +| `frontend/components/screen/ScreenRelationFlow.tsx` | `filterKeyMapping`, `joinColumnRefs` 데이터 전달 | +| `frontend/lib/api/entityJoin.ts` | `companyCodeOverride` 파라미터 추가 | +| `frontend/lib/api/screen.ts` | `saveLayout`, `getLayout` API 사용 | +| `frontend/lib/api/tableManagement.ts` | `getTableList`, `getColumnList`, `updateColumnSettings` API | +| `frontend/lib/registry/components/split-panel-layout/SplitPanelLayoutComponent.tsx` | `companyCode` prop 추가 | +| `backend-node/src/controllers/entityJoinController.ts` | `companyCodeOverride` 처리 로직 추가 | + +## 사용 방법 + +### 화면 설정 모달 열기 +1. 화면 관리 페이지에서 화면 그룹 선택 +2. 화면 노드 우클릭 → 컨텍스트 메뉴 표시 +3. "화면 설정" 선택 → 모달 열림 +4. 좌측 탭에서 정보 확인/수정, 우측에서 실시간 프리뷰 + +### 프리뷰 영역 조작 +- **휠 스크롤**: 확대/축소 (5% 단위) +- **마우스 드래그**: 화면 이동 (5px 이상 움직여야 드래그로 인식) +- **짧은 클릭**: iframe 내부 요소 클릭 + +### 컬럼 변경 +1. 메인/필터 테이블 아코디언 펼치기 +2. 파란색 배경의 "필드" 컬럼 클릭 +3. 우측 "컬럼 설정" 패널 확인 +4. "컬럼 변경" 드롭다운에서 새 컬럼 선택 +5. **실시간 반영** (페이지 새로고침 없음) + +### 필드 추가 +1. 메인/필터 테이블 아코디언 펼치기 +2. 회색/흰색 배경의 비필드 컬럼 클릭 +3. 우측 패널에서 **"필드로 추가"** 버튼 클릭 +4. 해당 컬럼이 화면 필드로 추가됨 + +### 필드 제거 +1. 메인/필터 테이블 아코디언 펼치기 +2. 파란색 배경의 필드 컬럼 클릭 +3. 우측 패널에서 **"필드에서 제거"** 버튼 클릭 +4. 해당 필드가 화면에서 제거됨 + +### 조인 설정 추가/편집 +1. 메인/필터 테이블 아코디언 펼치기 +2. 아무 컬럼 클릭 (조인 키가 아니어도 됨) +3. 우측 패널의 "조인" 섹션에서: + - 조인 없음: **"추가"** 버튼 클릭 + - 조인 있음: **"편집"** 버튼 클릭 +4. 대상 테이블 선택 (검색 가능) +5. 연결 컬럼 (PK) 선택 (검색 가능) +6. 표시 컬럼 선택 (검색 가능) +7. **"저장"** 버튼 클릭 + +### 컬럼 순서 변경 (드래그 앤 드롭) +1. 메인/필터 테이블 아코디언 펼치기 +2. 파란색 배경의 "필드" 컬럼을 드래그 시작 +3. 원하는 위치로 드래그하여 이동 (실시간으로 순서 변경 표시) +4. 마우스를 놓으면 (드롭) 순서가 저장됨 +5. 드래그 취소하려면 컬럼 영역 밖으로 드래그 + +**참고:** +- 사용 중인 필드만 드래그 가능 (파란색 배경) +- 미사용 컬럼은 드래그 불가 +- 드래그 중에는 저장되지 않고, 드롭 시에만 저장됨 + +--- + +## 완료일 +2026-01-13 + +## 변경 이력 +- 2026-01-12: 최초 작성 (줌/드래그/클릭, company_code 전달) +- 2026-01-12: 컬럼 변경 기능 추가, 필드 매핑 통합, UI 개선 (1열 레이아웃, 배지 변경) +- 2026-01-12: 실시간 반영 구현 (reload 제거), 레이아웃 순서 변경, 스타일 개선 +- 2026-01-12: 필드+조인 컬럼 스타일 개선 (파란배경 + 왼쪽 주황선), 조인 정보 패널 표시 +- 2026-01-12: 조인 관계 설정/수정 기능 구현 완료 (column_labels 테이블 저장) +- 2026-01-13: 필드 추가/제거 기능 구현 +- 2026-01-13: 검색 가능한 조인 설정 드롭다운 (Command 컴포넌트) +- 2026-01-13: 모든 컬럼에서 조인 설정 가능 (범용성 패치) +- 2026-01-13: 메인 테이블에도 조인 설정 기능 추가 +- 2026-01-13: 조인 라인 색상 주황색으로 변경 (`border-l-orange-500`) +- 2026-01-13: 조인 데이터 소스 수정 (`joinRef.refTable` 사용) +- 2026-01-13: 패널 높이 동기화 (`max-h-[350px]`, `items-stretch`) +- 2026-01-13: `MainTableAccordion`과 `FilterTableAccordion`을 `TableColumnAccordion`으로 통합 +- 2026-01-13: 드래그 앤 드롭 컬럼 순서 변경 기능 구현 +- 2026-01-13: 드래그 중에는 로컬 상태만 변경, 드롭 시에만 저장하도록 최적화 diff --git a/frontend/app/(main)/screens/[screenId]/page.tsx b/frontend/app/(main)/screens/[screenId]/page.tsx index 99634357..c96f7483 100644 --- a/frontend/app/(main)/screens/[screenId]/page.tsx +++ b/frontend/app/(main)/screens/[screenId]/page.tsx @@ -35,6 +35,9 @@ function ScreenViewPage() { // URL 쿼리에서 프리뷰용 company_code 가져오기 const previewCompanyCode = searchParams.get("company_code"); + + // 프리뷰 모드 감지 (iframe에서 로드될 때) + const isPreviewMode = searchParams.get("preview") === "true"; // 🆕 현재 로그인한 사용자 정보 const { user, userName, companyCode: authCompanyCode } = useAuth(); @@ -239,27 +242,40 @@ function ScreenViewPage() { const designWidth = layout?.screenResolution?.width || 1200; const designHeight = layout?.screenResolution?.height || 800; - // 컨테이너의 실제 크기 - const containerWidth = containerRef.current.offsetWidth; - const containerHeight = containerRef.current.offsetHeight; + // 컨테이너의 실제 크기 (프리뷰 모드에서는 window 크기 사용) + let containerWidth: number; + let containerHeight: number; + + if (isPreviewMode) { + // iframe에서는 window 크기를 직접 사용 + containerWidth = window.innerWidth; + containerHeight = window.innerHeight; + } else { + containerWidth = containerRef.current.offsetWidth; + containerHeight = containerRef.current.offsetHeight; + } - // 여백 설정: 좌우 16px씩 (총 32px), 상단 패딩 32px (pt-8) - const MARGIN_X = 32; - const availableWidth = containerWidth - MARGIN_X; - - // 가로 기준 스케일 계산 (좌우 여백 16px씩 고정) - const newScale = availableWidth / designWidth; + let newScale: number; + + if (isPreviewMode) { + // 프리뷰 모드: 가로/세로 모두 fit하도록 (여백 없이) + const scaleX = containerWidth / designWidth; + const scaleY = containerHeight / designHeight; + newScale = Math.min(scaleX, scaleY, 1); // 최대 1배율 + } else { + // 일반 모드: 가로 기준 스케일 (좌우 여백 16px씩 고정) + const MARGIN_X = 32; + const availableWidth = containerWidth - MARGIN_X; + newScale = availableWidth / designWidth; + } // console.log("📐 스케일 계산:", { // containerWidth, // containerHeight, - // MARGIN_X, - // availableWidth, // designWidth, // designHeight, // finalScale: newScale, - // "스케일된 화면 크기": `${designWidth * newScale}px × ${designHeight * newScale}px`, - // "실제 좌우 여백": `${(containerWidth - designWidth * newScale) / 2}px씩`, + // isPreviewMode, // }); setScale(newScale); @@ -278,7 +294,7 @@ function ScreenViewPage() { return () => { clearTimeout(timer); }; - }, [layout, isMobile]); + }, [layout, isMobile, isPreviewMode]); if (loading) { return ( @@ -316,7 +332,7 @@ function ScreenViewPage() { -
+
{/* 레이아웃 준비 중 로딩 표시 */} {!layoutReady && (
diff --git a/frontend/components/screen/ScreenRelationFlow.tsx b/frontend/components/screen/ScreenRelationFlow.tsx index f3b14d56..d7f814d3 100644 --- a/frontend/components/screen/ScreenRelationFlow.tsx +++ b/frontend/components/screen/ScreenRelationFlow.tsx @@ -1107,10 +1107,37 @@ function ScreenRelationFlowInner({ screen, selectedGroup, initialFocusedScreenId ); const tableNodeData = tableNode?.data as TableNodeData | undefined; + // 필터 키 매핑 정보 추출 (leftColumn → foreignKey) + let filterKeyMapping: { + mainTableColumn: string; + mainTableColumnLabel?: string; + filterTableColumn: string; + filterTableColumnLabel?: string; + } | undefined = undefined; + + if (subTableData?.leftColumn && subTableData?.foreignKey) { + // 메인 테이블 컬럼 한글명 조회 + const mainTable = subTablesDataMap[screenId]?.mainTable; + const mainTableCols = mainTable ? tableColumns[mainTable] : []; + const mainColInfo = mainTableCols?.find(c => c.columnName === subTableData.leftColumn); + + // 필터 테이블 컬럼 한글명 조회 + const filterTableCols = tableColumns[tableName] || []; + const filterColInfo = filterTableCols?.find(c => c.columnName === subTableData.foreignKey); + + filterKeyMapping = { + mainTableColumn: subTableData.leftColumn, + mainTableColumnLabel: mainColInfo?.displayName, + filterTableColumn: subTableData.foreignKey, + filterTableColumnLabel: filterColInfo?.displayName, + }; + } + return { tableName, tableLabel: subTableData?.tableLabel || tableNodeData?.label || tableName, filterColumns: subTableData?.filterColumns || tableNodeData?.filterColumns || [], + filterKeyMapping, joinColumnRefs: subTableData?.joinColumnRefs || tableNodeData?.joinColumnRefs || [], }; }); diff --git a/frontend/components/screen/ScreenSettingModal.tsx b/frontend/components/screen/ScreenSettingModal.tsx index 6b6dea5f..79342ef6 100644 --- a/frontend/components/screen/ScreenSettingModal.tsx +++ b/frontend/components/screen/ScreenSettingModal.tsx @@ -1,6 +1,6 @@ "use client"; -import React, { useState, useEffect, useCallback, useMemo } from "react"; +import React, { useState, useEffect, useCallback, useMemo, useRef } from "react"; import { Dialog, DialogContent, @@ -45,6 +45,7 @@ import { Eye, Save, Plus, + Minus, Pencil, Trash2, RefreshCw, @@ -58,6 +59,13 @@ import { ChevronDown, ChevronRight, Filter, + RotateCcw, + X, + Zap, + MousePointer, + Globe, + Workflow, + Info, } from "lucide-react"; import { getDataFlows, @@ -68,7 +76,19 @@ import { getMultipleScreenLayoutSummary, LayoutItem, } from "@/lib/api/screenGroup"; -import { tableManagementApi, ColumnTypeInfo, TableInfo } from "@/lib/api/tableManagement"; +import { tableManagementApi, ColumnTypeInfo, TableInfo, ColumnSettings } from "@/lib/api/tableManagement"; +import { screenApi } from "@/lib/api/screen"; +import { TransformWrapper, TransformComponent } from "react-zoom-pan-pinch"; +import { ExternalCallConfigAPI, ExternalCallConfig } from "@/lib/api/externalCallConfig"; +import { getFlowDefinitions } from "@/lib/api/flow"; +import { FlowDefinition } from "@/types/flow"; +import { + Select, + SelectContent, + SelectItem, + SelectTrigger, + SelectValue, +} from "@/components/ui/select"; // ============================================================ // 타입 정의 @@ -78,6 +98,13 @@ interface FilterTableInfo { tableName: string; tableLabel?: string; filterColumns?: string[]; + // 필터 키 매핑 정보 (메인 테이블.컬럼 → 필터 테이블.컬럼) + filterKeyMapping?: { + mainTableColumn: string; // 메인 테이블의 컬럼 (leftColumn) + mainTableColumnLabel?: string; + filterTableColumn: string; // 필터 테이블의 컬럼 (foreignKey) + filterTableColumnLabel?: string; + }; joinColumnRefs?: Array<{ column: string; refTable: string; @@ -217,6 +244,7 @@ export function ScreenSettingModal({ const [loading, setLoading] = useState(false); const [dataFlows, setDataFlows] = useState([]); const [layoutItems, setLayoutItems] = useState([]); + const [iframeKey, setIframeKey] = useState(0); // iframe 새로고침용 키 // 데이터 로드 const loadData = useCallback(async () => { @@ -249,15 +277,15 @@ export function ScreenSettingModal({ } }, [isOpen, screenId, loadData]); - // 새로고침 - const handleRefresh = () => { + // 새로고침 (데이터 + iframe) + const handleRefresh = useCallback(() => { loadData(); - toast.success("새로고침 완료"); - }; + setIframeKey(prev => prev + 1); // iframe 새로고침 + }, [loadData]); return ( - + @@ -268,120 +296,254 @@ export function ScreenSettingModal({ - -
- - - - 화면 개요 - - - - 필드 매핑 - - - - 데이터 흐름 - - - - 화면 프리뷰 - - - +
+ + + + 개요 + + + + 제어 관리 + + + + 데이터 흐름 + + + +
+ + {/* 탭 1: 화면 개요 */} + + + + + {/* 탭 2: 제어 관리 */} + + + + + {/* 탭 3: 데이터 흐름 */} + + + +
- {/* 탭 1: 화면 개요 */} - - - - - {/* 탭 2: 필드 매핑 */} - - - - - {/* 탭 3: 데이터 흐름 */} - - - - - {/* 탭 4: 화면 프리뷰 */} - - - -
+ {/* 오른쪽: 화면 프리뷰 (60%, 항상 표시) */} +
+ +
+
); } // ============================================================ -// 필터 테이블 아코디언 컴포넌트 +// 통합 테이블 컬럼 아코디언 컴포넌트 // ============================================================ -interface FilterTableAccordionProps { - filterTable: FilterTableInfo; - mainTable?: string; +interface ColumnMapping { + columnName: string; + fieldLabel?: string; + order: number; // 화면 순서 (y 좌표 기준) } -function FilterTableAccordion({ filterTable: ft, mainTable }: FilterTableAccordionProps) { +interface JoinColumnRef { + column: string; + refTable: string; + refTableLabel?: string; + refColumn: string; + displayColumn?: string; +} + +interface FilterKeyMapping { + mainTableColumn: string; + mainTableColumnLabel?: string; + filterTableColumn: string; + filterTableColumnLabel?: string; +} + +interface TableColumnAccordionProps { + // 공통 props + tableName: string; + tableLabel?: string; + tableType: "main" | "filter"; // 테이블 타입 + columnMappings?: ColumnMapping[]; + onColumnChange?: (fieldLabel: string, oldColumn: string, newColumn: string) => void; + onColumnReorder?: (newOrder: string[]) => void; // 컬럼 순서 변경 콜백 + onJoinSettingSaved?: () => void; + + // 필터 테이블 전용 props (optional) + mainTable?: string; // 메인 테이블명 (필터 테이블에서 필터 연결 정보 표시용) + filterKeyMapping?: FilterKeyMapping; + joinColumnRefs?: JoinColumnRef[]; +} + +function TableColumnAccordion({ + tableName, + tableLabel, + tableType, + columnMappings = [], + onColumnChange, + onColumnReorder, + onJoinSettingSaved, + mainTable, + filterKeyMapping, + joinColumnRefs = [], +}: TableColumnAccordionProps) { + // columnMappings를 Map으로 변환 (컬럼명 → 매핑정보) + const columnMappingMap = useMemo(() => { + const map = new Map(); + columnMappings.forEach(m => map.set(m.columnName.toLowerCase(), m)); + return map; + }, [columnMappings]); + const [isOpen, setIsOpen] = useState(false); const [columns, setColumns] = useState([]); const [loadingColumns, setLoadingColumns] = useState(false); + + // 편집 중인 필드 + const [editingField, setEditingField] = useState(null); + + // 조인 설정 관련 상태 + const [allTables, setAllTables] = useState([]); + const [refTableColumns, setRefTableColumns] = useState([]); + const [loadingRefColumns, setLoadingRefColumns] = useState(false); + const [savingJoinSetting, setSavingJoinSetting] = useState(false); + + // 조인 설정 편집 상태 + const [editingJoin, setEditingJoin] = useState<{ + columnName: string; + referenceTable: string; + referenceColumn: string; + displayColumn: string; + } | null>(null); + + // 드래그 앤 드롭 상태 + const [draggedIndex, setDraggedIndex] = useState(null); + const [localColumnOrder, setLocalColumnOrder] = useState(null); // 드래그 중 로컬 순서 + + // 스타일 설정 (테이블 타입별) + const isMain = tableType === "main"; + const themeColor = isMain ? "blue" : "purple"; + const themeIcon = isMain ? Table2 : Filter; + const themeBadge = isMain ? "메인" : "필터"; + + // 필터 테이블용 플래그 + const hasJoinRefs = joinColumnRefs && joinColumnRefs.length > 0; + const hasFilterKey = !!filterKeyMapping; + + // 정렬된 컬럼 목록 + const sortedColumns = useMemo(() => { + if (columns.length === 0) return []; + + if (isMain) { + // 메인: 사용 중 → 안 쓰는 컬럼 + const used: (ColumnTypeInfo & { mapping: ColumnMapping })[] = []; + const unused: ColumnTypeInfo[] = []; + + columns.forEach(col => { + const mapping = columnMappingMap.get(col.columnName.toLowerCase()); + if (mapping) { + used.push({ ...col, mapping }); + } else { + unused.push(col); + } + }); + + used.sort((a, b) => a.mapping.order - b.mapping.order); + return [...used, ...unused]; + } else { + // 필터: 필터키 → 조인키 → 필드 → 안 쓰는 컬럼 + const filterKeys: ColumnTypeInfo[] = []; + const joinKeys: ColumnTypeInfo[] = []; + const fieldCols: (ColumnTypeInfo & { mapping: ColumnMapping })[] = []; + const unused: ColumnTypeInfo[] = []; + + columns.forEach(col => { + const colNameLower = col.columnName.toLowerCase(); + const isFilterKey = filterKeyMapping?.filterTableColumn?.toLowerCase() === colNameLower; + const isJoinKey = joinColumnRefs?.some(j => j.column.toLowerCase() === colNameLower); + const mapping = columnMappingMap.get(colNameLower); + + if (isFilterKey) { + filterKeys.push(col); + } else if (isJoinKey) { + joinKeys.push(col); + } else if (mapping) { + fieldCols.push({ ...col, mapping }); + } else { + unused.push(col); + } + }); + + fieldCols.sort((a, b) => a.mapping.order - b.mapping.order); + return [...filterKeys, ...joinKeys, ...fieldCols, ...unused]; + } + }, [columns, columnMappingMap, isMain, filterKeyMapping, joinColumnRefs]); - const hasJoinRefs = ft.joinColumnRefs && ft.joinColumnRefs.length > 0; - const hasFilterColumns = ft.filterColumns && ft.filterColumns.length > 0; - - // 아코디언 열릴 때 테이블 컬럼 로드 + // 아코디언 열릴 때 테이블 컬럼 + 전체 테이블 목록 로드 const handleToggle = async () => { const newIsOpen = !isOpen; setIsOpen(newIsOpen); - // 처음 열릴 때 컬럼 로드 - if (newIsOpen && columns.length === 0 && ft.tableName) { + if (newIsOpen && columns.length === 0 && tableName) { setLoadingColumns(true); try { - const result = await tableManagementApi.getColumnList(ft.tableName); + const result = await tableManagementApi.getColumnList(tableName); if (result.success && result.data && result.data.columns) { setColumns(result.data.columns); } + + if (allTables.length === 0) { + const tablesResult = await tableManagementApi.getTableList(); + if (tablesResult.success && tablesResult.data) { + setAllTables(tablesResult.data); + } + } } catch (error) { console.error("테이블 컬럼 로드 실패:", error); } finally { @@ -389,145 +551,734 @@ function FilterTableAccordion({ filterTable: ft, mainTable }: FilterTableAccordi } } }; + + // 참조 테이블 선택 시 해당 테이블의 컬럼 로드 + const loadRefTableColumns = useCallback(async (refTableName: string) => { + if (!refTableName) { + setRefTableColumns([]); + return; + } + + setLoadingRefColumns(true); + try { + const result = await tableManagementApi.getColumnList(refTableName); + if (result.success && result.data && result.data.columns) { + setRefTableColumns(result.data.columns); + } + } catch (error) { + console.error("참조 테이블 컬럼 로드 실패:", error); + } finally { + setLoadingRefColumns(false); + } + }, []); + + // 조인 설정 저장 + const handleSaveJoinSetting = useCallback(async () => { + if (!editingJoin || !tableName) return; + + setSavingJoinSetting(true); + try { + const settings: ColumnSettings = { + columnLabel: columns.find(c => c.columnName === editingJoin.columnName)?.displayName || editingJoin.columnName, + webType: "entity", + detailSettings: JSON.stringify({}), + codeCategory: "", + codeValue: "", + referenceTable: editingJoin.referenceTable, + referenceColumn: editingJoin.referenceColumn, + displayColumn: editingJoin.displayColumn, + }; + + const result = await tableManagementApi.updateColumnSettings( + tableName, + editingJoin.columnName, + settings + ); + + if (result.success) { + toast.success("조인 설정이 저장되었습니다."); + setEditingJoin(null); + onJoinSettingSaved?.(); + } else { + toast.error(result.message || "조인 설정 저장에 실패했습니다."); + } + } catch (error) { + console.error("조인 설정 저장 실패:", error); + toast.error("조인 설정 저장에 실패했습니다."); + } finally { + setSavingJoinSetting(false); + } + }, [editingJoin, tableName, columns, onJoinSettingSaved]); + + // 조인 설정 편집 시작 + const startEditingJoin = useCallback((columnName: string, currentRefTable?: string, currentRefColumn?: string, currentDisplayColumn?: string) => { + setEditingJoin({ + columnName, + referenceTable: currentRefTable || "", + referenceColumn: currentRefColumn || "", + displayColumn: currentDisplayColumn || "", + }); + + if (currentRefTable) { + loadRefTableColumns(currentRefTable); + } + }, [loadRefTableColumns]); + + // 드래그 앤 드롭 핸들러 + const handleDragStart = useCallback((e: React.DragEvent, index: number) => { + setDraggedIndex(index); + e.dataTransfer.effectAllowed = "move"; + e.dataTransfer.setData("text/plain", String(index)); + + // 드래그 시작 시 현재 순서를 로컬 상태로 저장 + const usedColumns = sortedColumns.filter(col => { + const colNameLower = col.columnName.toLowerCase(); + return columnMappingMap.has(colNameLower); + }); + setLocalColumnOrder(usedColumns.map(col => col.columnName)); + }, [sortedColumns, columnMappingMap]); + + const handleDragOver = useCallback((e: React.DragEvent, hoverIndex: number) => { + e.preventDefault(); + e.dataTransfer.dropEffect = "move"; + + if (draggedIndex === null || draggedIndex === hoverIndex || !localColumnOrder) return; + + // 사용 중인 컬럼 수 체크 + if (hoverIndex >= localColumnOrder.length || draggedIndex >= localColumnOrder.length) return; + + // 로컬 순서만 변경 (저장하지 않음) + const newOrder = [...localColumnOrder]; + const draggedItem = newOrder[draggedIndex]; + newOrder.splice(draggedIndex, 1); + newOrder.splice(hoverIndex, 0, draggedItem); + + setDraggedIndex(hoverIndex); + setLocalColumnOrder(newOrder); + }, [draggedIndex, localColumnOrder]); + + const handleDrop = useCallback((e: React.DragEvent) => { + e.preventDefault(); + + // 드롭 시 최종 순서로 저장 + if (localColumnOrder && onColumnReorder) { + onColumnReorder(localColumnOrder); + } + + setDraggedIndex(null); + setLocalColumnOrder(null); + }, [localColumnOrder, onColumnReorder]); + + const handleDragEnd = useCallback(() => { + // 드래그 취소 시 (드롭 영역 밖으로 나간 경우) + setDraggedIndex(null); + setLocalColumnOrder(null); + }, []); + + // 컬럼의 특수 상태 확인 (필터 테이블용) + const getColumnState = (colNameLower: string) => { + const isFilterKey = filterKeyMapping?.filterTableColumn?.toLowerCase() === colNameLower; + const joinRef = joinColumnRefs?.find(j => j.column.toLowerCase() === colNameLower); + const isJoinKey = !!joinRef; + const mapping = columnMappingMap.get(colNameLower); + const isUsed = !!mapping; + + return { isFilterKey, isJoinKey, joinRef, isUsed, mapping }; + }; + + const ThemeIcon = themeIcon; return ( -
- {/* 헤더 - 클릭하면 펼쳐짐 */} +
+ {/* 헤더 */} {/* 펼쳐진 내용 */} {isOpen && ( -
- {/* 필터 키 설명 */} -
- {ft.tableLabel || ft.tableName}의 데이터를 기준으로 필터링됩니다. -
+
+ {/* 필터 연결 정보 (필터 테이블만) */} + {!isMain && filterKeyMapping && ( +
+ 필터 + + {mainTable}.{filterKeyMapping.mainTableColumnLabel || filterKeyMapping.mainTableColumn} + + = + + {tableName}.{filterKeyMapping.filterTableColumnLabel || filterKeyMapping.filterTableColumn} + +
+ )} {/* 테이블 컬럼 정보 */}
-
- - 테이블 컬럼 ({loadingColumns ? "로딩중..." : `${columns.length}개`}) -
- {loadingColumns ? ( -
- +
+
+ + 테이블 컬럼 ({loadingColumns ? "로딩중..." : `${columns.length}개`})
- ) : columns.length > 0 ? ( -
- {columns.slice(0, 10).map((col, cIdx) => ( -
- {col.displayName || col.columnName} - - ({col.dataType}) - + {columnMappings.length > 0 && ( +
+
+
+ 화면에서 사용 ({columnMappings.length}개)
- ))} - {columns.length > 10 && ( -
- +{columns.length - 10}개 더 -
- )} +
+ )} +
+ + {loadingColumns ? ( +
+ +
+ ) : sortedColumns.length > 0 ? ( +
+ {/* 왼쪽: 컬럼 목록 */} +
+ {(() => { + // 드래그 중일 때 로컬 순서 적용 + let displayColumns = sortedColumns; + if (localColumnOrder && localColumnOrder.length > 0) { + // 사용 중인 컬럼들을 localColumnOrder에 따라 재정렬 + const usedCols = sortedColumns.filter(col => columnMappingMap.has(col.columnName.toLowerCase())); + const unusedCols = sortedColumns.filter(col => !columnMappingMap.has(col.columnName.toLowerCase())); + + const reorderedUsed = localColumnOrder + .map(name => usedCols.find(col => col.columnName.toLowerCase() === name.toLowerCase())) + .filter(Boolean) as typeof usedCols; + + displayColumns = [...reorderedUsed, ...unusedCols]; + } + + return displayColumns.map((col, cIdx) => { + const colNameLower = col.columnName.toLowerCase(); + const { isFilterKey, isJoinKey, isUsed, mapping } = getColumnState(colNameLower); + const isSelected = editingField === (mapping?.fieldLabel || col.columnName); + const isDragging = draggedIndex === cIdx; + + // 드래그 가능 여부 (사용 중인 컬럼만) + const canDrag = isUsed && !!onColumnReorder; + + // 스타일 결정 + let baseClass = ""; + let leftBorderClass = ""; + + if (isUsed) { + baseClass = isSelected + ? "bg-blue-100 border-blue-300" + : "bg-blue-50 border-blue-200 hover:bg-blue-100 hover:border-blue-300"; + if (isJoinKey) { + leftBorderClass = "border-l-4 border-l-orange-500"; + } else if (isFilterKey) { + leftBorderClass = "border-l-4 border-l-purple-400"; + } + } else if (isJoinKey) { + baseClass = isSelected + ? "bg-orange-100 border-orange-400" + : "bg-orange-50 border-orange-200 hover:bg-orange-100 hover:border-orange-300"; + } else if (isFilterKey) { + baseClass = isSelected + ? "bg-purple-100 border-purple-400" + : "bg-purple-50 border-purple-200 hover:bg-purple-100 hover:border-purple-300"; + } else { + baseClass = isSelected + ? "bg-gray-100 border-gray-400" + : "bg-gray-50 border-gray-200 hover:bg-gray-100"; + } + + return ( +
handleDragStart(e, cIdx) : undefined} + onDragOver={canDrag ? (e) => handleDragOver(e, cIdx) : undefined} + onDrop={canDrag ? handleDrop : undefined} + onDragEnd={canDrag ? handleDragEnd : undefined} + onClick={() => { + setEditingField(mapping?.fieldLabel || col.columnName); + setEditingJoin(null); + }} + className={`flex items-center justify-between gap-2 text-xs rounded px-2 py-1.5 border transition-all cursor-pointer ${baseClass} ${leftBorderClass} ${isDragging ? "opacity-50 scale-95" : ""} ${canDrag ? "cursor-grab active:cursor-grabbing" : ""}`} + > + + {col.displayName || col.columnName} + +
+ {isFilterKey && ( + 필터 + )} + {isJoinKey && ( + 조인 + )} + {isUsed && ( + 필드 + )} + + {col.dataType?.split("(")[0]} + +
+
+ ); + }); + })()} +
+ + {/* 오른쪽: 컬럼 설정 패널 */} +
+ {editingField ? (() => { + const selectedMapping = columnMappings.find(m => m.fieldLabel === editingField); + const selectedColumn = selectedMapping + ? columns.find(c => c.columnName.toLowerCase() === selectedMapping.columnName?.toLowerCase()) + : columns.find(c => (c.displayName || c.columnName) === editingField || c.columnName === editingField); + const colNameLower = selectedColumn?.columnName?.toLowerCase() || editingField.toLowerCase(); + const { isFilterKey, isJoinKey, joinRef, isUsed } = getColumnState(colNameLower); + + // 조인 정보 - joinColumnRefs에서 먼저 찾고, 없으면 selectedColumn에서 가져옴 + const hasJoinSetting = isJoinKey || !!selectedColumn?.referenceTable; + + return ( +
+
컬럼 설정
+ + {/* 화면 필드 정보 (필드인 경우만) */} + {isUsed && ( + <> +
+ 화면 필드 +
+ {selectedColumn?.displayName || selectedMapping?.columnName || editingField} +
+
+ +
+ 현재 컬럼 +
+ {selectedMapping?.columnName || "-"} +
+
+ +
+ 컬럼 변경 + + + + + + + + + 없음 + + {columns.map((c) => ( + { + if (onColumnChange && selectedMapping) { + onColumnChange(editingField, selectedMapping.columnName, c.columnName); + } + setEditingField(null); + }} + > + + {c.displayName || c.columnName} + + ))} + + + + + +
+ + {/* 필드에서 제거 */} + + + )} + + {/* 컬럼 기본 정보 (필드가 아닌 경우) */} + {!isUsed && ( +
+
+
+ 컬럼명 +
{selectedColumn?.columnName || editingField}
+
+
+ 데이터 타입 +
{selectedColumn?.dataType || "-"}
+
+
+ + +
+ )} + + {/* 조인 설정 */} +
+
+
+ + 조인 + + + {editingJoin && editingJoin.columnName === (selectedColumn?.columnName || editingField) ? "연결 편집" : (hasJoinSetting ? "연결 정보" : "연결 설정")} + +
+ {editingJoin && editingJoin.columnName === (selectedColumn?.columnName || editingField) ? ( +
+ + +
+ ) : ( + + )} +
+ + {editingJoin && editingJoin.columnName === (selectedColumn?.columnName || editingField) ? ( + + ) : hasJoinSetting ? ( +
+
+ 대상 테이블: + {isJoinKey && joinRef ? joinRef.refTable : selectedColumn?.referenceTable} +
+
+ 연결 컬럼: + {isJoinKey && joinRef ? joinRef.refColumn : selectedColumn?.referenceColumn} +
+
+ ) : ( +
조인 설정이 없습니다.
+ )} +
+ + {/* 필터 정보 (필터 키인 경우) - 필터 테이블에서만 */} + {!isMain && isFilterKey && filterKeyMapping && ( +
+
+ 필터 + 필터링 정보 +
+
+
+ 대상 테이블: + {mainTable} +
+
+ 연결 컬럼: + {filterKeyMapping.mainTableColumn} +
+
+
+ )} +
+ ); + })() : ( +
+ 필드를 선택하세요 +
+ )} +
) : ( -
- 컬럼 정보 없음 -
+
컬럼 정보 없음
)}
- - {/* 필터 컬럼 매핑 */} - {hasFilterColumns && ( -
-
- - 필터 키 매핑 -
-
- {ft.filterColumns!.map((col, cIdx) => ( -
- - {mainTable}.{col} - - - - {ft.tableLabel || ft.tableName}.{col} - -
- ))} -
-
- )} - - {/* 조인 관계 */} - {hasJoinRefs && ( -
-
- - 조인 관계 ({ft.joinColumnRefs!.length}개) -
-
- {ft.joinColumnRefs!.map((join, jIdx) => ( -
- - {ft.tableLabel || ft.tableName}.{join.column} - - - - {join.refTableLabel || join.refTable}.{join.refColumn} - -
- ))} -
-
- )}
)}
); } +// ============================================================ +// 조인 설정 편집 컴포넌트 (검색 가능한 Combobox 사용) +// ============================================================ + +interface JoinSettingEditorProps { + editingJoin: { + columnName: string; + referenceTable: string; + referenceColumn: string; + displayColumn: string; + }; + setEditingJoin: React.Dispatch>; + allTables: TableInfo[]; + refTableColumns: ColumnTypeInfo[]; + loadingRefColumns: boolean; + savingJoinSetting: boolean; + loadRefTableColumns: (tableName: string) => void; + handleSaveJoinSetting: () => void; +} + +function JoinSettingEditor({ + editingJoin, + setEditingJoin, + allTables, + refTableColumns, + loadingRefColumns, + savingJoinSetting, + loadRefTableColumns, + handleSaveJoinSetting, +}: JoinSettingEditorProps) { + const [tableSearchOpen, setTableSearchOpen] = useState(false); + const [refColSearchOpen, setRefColSearchOpen] = useState(false); + const [displayColSearchOpen, setDisplayColSearchOpen] = useState(false); + + const selectedTable = allTables.find(t => t.tableName === editingJoin.referenceTable); + const selectedRefCol = refTableColumns.find(c => c.columnName === editingJoin.referenceColumn); + const selectedDisplayCol = refTableColumns.find(c => c.columnName === editingJoin.displayColumn); + + return ( +
+ {/* 대상 테이블 선택 - 검색 가능 Combobox */} +
+ 대상 테이블 + + + + + + + + + 테이블을 찾을 수 없습니다. + + {allTables.map(t => ( + { + setEditingJoin({ ...editingJoin, referenceTable: t.tableName, referenceColumn: "", displayColumn: "" }); + loadRefTableColumns(t.tableName); + setTableSearchOpen(false); + }} + className="text-xs" + > + + {t.displayName || t.tableName} + + ))} + + + + + +
+ + {/* 연결 컬럼 선택 - 검색 가능 Combobox */} +
+ 연결 컬럼 (PK) + + + + + + + + + 컬럼을 찾을 수 없습니다. + + {refTableColumns.map(c => ( + { + setEditingJoin({ ...editingJoin, referenceColumn: c.columnName }); + setRefColSearchOpen(false); + }} + className="text-xs" + > + + {c.displayName || c.columnName} + + ))} + + + + + +
+ + {/* 표시 컬럼 선택 - 검색 가능 Combobox */} +
+ 표시 컬럼 + + + + + + + + + 컬럼을 찾을 수 없습니다. + + {refTableColumns.map(c => ( + { + setEditingJoin({ ...editingJoin, displayColumn: c.columnName }); + setDisplayColSearchOpen(false); + }} + className="text-xs" + > + + {c.displayName || c.columnName} + + ))} + + + + + +
+ +
+ ); +} + // ============================================================ // 탭 1: 화면 개요 // ============================================================ @@ -543,6 +1294,7 @@ interface OverviewTabProps { dataFlows: DataFlow[]; layoutItems: LayoutItem[]; // 컴포넌트 컬럼 정보 추가 loading: boolean; + onRefresh?: () => void; // 컬럼 변경 후 새로고침 콜백 } function OverviewTab({ @@ -556,7 +1308,574 @@ function OverviewTab({ dataFlows, layoutItems, loading, + onRefresh, }: OverviewTabProps) { + const [isSavingColumn, setIsSavingColumn] = useState(false); + + // 컬럼 변경 저장 함수 - 화면 디자이너와 동일한 방식 + const handleColumnChange = useCallback(async (fieldLabel: string, oldColumn: string, newColumn: string) => { + console.log("[handleColumnChange] 시작", { screenId, fieldLabel, oldColumn, newColumn }); + + if (!screenId) { + toast.error("화면 정보가 없습니다."); + return; + } + + // 필드 추가/제거 처리 + const isAddingField = fieldLabel === "__NEW_FIELD__"; + const isRemovingField = newColumn === "__REMOVE_FIELD__"; + + setIsSavingColumn(true); + try { + // 1. 현재 레이아웃 가져오기 + console.log("[handleColumnChange] 레이아웃 조회 시작", { screenId }); + const currentLayout = await screenApi.getLayout(screenId); + console.log("[handleColumnChange] 레이아웃 조회 완료", { + hasLayout: !!currentLayout, + hasComponents: !!currentLayout?.components, + componentCount: currentLayout?.components?.length + }); + + if (!currentLayout?.components) { + toast.error("레이아웃 정보를 불러올 수 없습니다."); + console.error("[handleColumnChange] 레이아웃 정보 없음", { currentLayout }); + return; + } + + // 2. 레이아웃에서 해당 컬럼 변경 + let columnChanged = false; + + // 디버깅: 각 컴포넌트의 구조 확인 + console.log("[handleColumnChange] 컴포넌트 구조 분석 시작"); + currentLayout.components.forEach((comp: any, i: number) => { + console.log(`[handleColumnChange] 컴포넌트 ${i}:`, { + id: comp.id, + componentType: comp.componentType, + hasUsedColumns: !!comp.usedColumns, + usedColumns: comp.usedColumns, + hasComponentConfig: !!comp.componentConfig, + componentConfigKeys: comp.componentConfig ? Object.keys(comp.componentConfig) : [], + componentConfigColumns: comp.componentConfig?.columns, + componentConfigUsedColumns: comp.componentConfig?.usedColumns, + columnName: comp.columnName, + bindField: comp.bindField, + }); + }); + + const updatedComponents = currentLayout.components.map((comp: any) => { + // usedColumns 배열이 있는 컴포넌트에서 oldColumn을 newColumn으로 교체 + if (comp.usedColumns && Array.isArray(comp.usedColumns)) { + // 필드 추가 + if (isAddingField) { + console.log("[handleColumnChange] usedColumns에 필드 추가", { compId: comp.id, newColumn }); + columnChanged = true; + return { + ...comp, + usedColumns: [...comp.usedColumns, newColumn], + }; + } + + const idx = comp.usedColumns.findIndex( + (col: string) => col.toLowerCase() === oldColumn.toLowerCase() + ); + if (idx !== -1) { + console.log("[handleColumnChange] usedColumns에서 찾음", { compId: comp.id, idx, isRemovingField }); + columnChanged = true; + + // 필드 제거 + if (isRemovingField) { + return { + ...comp, + usedColumns: comp.usedColumns.filter((_: string, i: number) => i !== idx), + }; + } + + // 컬럼 변경 + return { + ...comp, + usedColumns: comp.usedColumns.map((col: string, i: number) => + i === idx ? newColumn : col + ), + }; + } + } + + // componentConfig 내부의 usedColumns도 확인 + if (comp.componentConfig?.usedColumns && Array.isArray(comp.componentConfig.usedColumns)) { + // 필드 추가 + if (isAddingField && !columnChanged) { + console.log("[handleColumnChange] componentConfig.usedColumns에 필드 추가", { compId: comp.id, newColumn }); + columnChanged = true; + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + usedColumns: [...comp.componentConfig.usedColumns, newColumn], + }, + }; + } + + const idx = comp.componentConfig.usedColumns.findIndex( + (col: string) => col.toLowerCase() === oldColumn.toLowerCase() + ); + if (idx !== -1) { + console.log("[handleColumnChange] componentConfig.usedColumns에서 찾음", { compId: comp.id, idx, isRemovingField }); + columnChanged = true; + + // 필드 제거 + if (isRemovingField) { + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + usedColumns: comp.componentConfig.usedColumns.filter((_: string, i: number) => i !== idx), + }, + }; + } + + // 컬럼 변경 + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + usedColumns: comp.componentConfig.usedColumns.map((col: string, i: number) => + i === idx ? newColumn : col + ), + }, + }; + } + } + + // componentConfig.columns 배열도 확인 (컬럼 설정 형태) + if (comp.componentConfig?.columns && Array.isArray(comp.componentConfig.columns)) { + // 필드 추가 + if (isAddingField && !columnChanged) { + console.log("[handleColumnChange] componentConfig.columns에 필드 추가", { compId: comp.id, newColumn }); + columnChanged = true; + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + columns: [...comp.componentConfig.columns, { field: newColumn, columnName: newColumn }], + }, + }; + } + + const columnIdx = comp.componentConfig.columns.findIndex( + (col: any) => { + const colName = typeof col === 'string' ? col : (col.field || col.columnName || col.name); + return colName?.toLowerCase() === oldColumn.toLowerCase(); + } + ); + if (columnIdx !== -1) { + console.log("[handleColumnChange] componentConfig.columns에서 찾음", { compId: comp.id, columnIdx, isRemovingField }); + columnChanged = true; + + // 필드 제거 + if (isRemovingField) { + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + columns: comp.componentConfig.columns.filter((_: any, i: number) => i !== columnIdx), + }, + }; + } + + // 컬럼 변경 + const updatedColumns = comp.componentConfig.columns.map((col: any, i: number) => { + if (i !== columnIdx) return col; + if (typeof col === 'string') return newColumn; + return { ...col, field: newColumn, columnName: newColumn }; + }); + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + columns: updatedColumns, + }, + }; + } + } + + // columnName 필드 체크 (위젯 컴포넌트) + if (comp.columnName?.toLowerCase() === oldColumn.toLowerCase()) { + console.log("[handleColumnChange] columnName에서 찾음", { compId: comp.id }); + columnChanged = true; + return { + ...comp, + columnName: newColumn, + }; + } + + // bindField 필드 체크 (바인딩 필드) + if (comp.bindField?.toLowerCase() === oldColumn.toLowerCase()) { + console.log("[handleColumnChange] bindField에서 찾음", { compId: comp.id }); + columnChanged = true; + return { + ...comp, + bindField: newColumn, + }; + } + + // split-panel-layout의 leftPanel.columns 검사 + if (comp.componentConfig?.leftPanel?.columns && Array.isArray(comp.componentConfig.leftPanel.columns)) { + const leftColumns = comp.componentConfig.leftPanel.columns; + console.log("[handleColumnChange] leftPanel.columns 검사:", { + compId: comp.id, + leftColumnsCount: leftColumns.length, + leftColumnsContent: leftColumns.map((col: any) => typeof col === 'string' ? col : (col.name || col.columnName || col.field)), + searchingFor: isAddingField ? newColumn : oldColumn.toLowerCase(), + isAddingField, + isRemovingField, + }); + + // 필드 추가: 배열에 새 컬럼 추가 + if (isAddingField) { + console.log("[handleColumnChange] 필드 추가", { compId: comp.id, newColumn }); + columnChanged = true; + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: [...leftColumns, { name: newColumn, columnName: newColumn }], + }, + }, + }; + } + + const columnIdx = leftColumns.findIndex((col: any) => { + const colName = typeof col === 'string' ? col : (col.name || col.columnName || col.field); + return colName?.toLowerCase() === oldColumn.toLowerCase(); + }); + if (columnIdx !== -1) { + console.log("[handleColumnChange] leftPanel.columns에서 찾음", { compId: comp.id, columnIdx, isRemovingField }); + columnChanged = true; + + // 필드 제거: 배열에서 해당 컬럼 제거 + if (isRemovingField) { + const filteredColumns = leftColumns.filter((_: any, i: number) => i !== columnIdx); + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: filteredColumns, + }, + }, + }; + } + + // 컬럼 변경 + const updatedLeftColumns = leftColumns.map((col: any, i: number) => { + if (i !== columnIdx) return col; + if (typeof col === 'string') return newColumn; + // 객체인 경우 name/columnName 필드 업데이트 + return { ...col, name: newColumn, columnName: newColumn }; + }); + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: updatedLeftColumns, + }, + }, + }; + } + } + + // split-panel-layout의 rightPanel.columns 검사 + if (comp.componentConfig?.rightPanel?.columns && Array.isArray(comp.componentConfig.rightPanel.columns)) { + const rightColumns = comp.componentConfig.rightPanel.columns; + + // 필드 추가: 배열에 새 컬럼 추가 + if (isAddingField && !columnChanged) { + console.log("[handleColumnChange] 필드 추가 (rightPanel)", { compId: comp.id, newColumn }); + columnChanged = true; + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + rightPanel: { + ...comp.componentConfig.rightPanel, + columns: [...rightColumns, { name: newColumn, columnName: newColumn }], + }, + }, + }; + } + + const columnIdx = rightColumns.findIndex((col: any) => { + const colName = typeof col === 'string' ? col : (col.name || col.columnName || col.field); + return colName?.toLowerCase() === oldColumn.toLowerCase(); + }); + if (columnIdx !== -1) { + console.log("[handleColumnChange] rightPanel.columns에서 찾음", { compId: comp.id, columnIdx, isRemovingField }); + columnChanged = true; + + // 필드 제거 + if (isRemovingField) { + const filteredColumns = rightColumns.filter((_: any, i: number) => i !== columnIdx); + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + rightPanel: { + ...comp.componentConfig.rightPanel, + columns: filteredColumns, + }, + }, + }; + } + + // 컬럼 변경 + const updatedRightColumns = rightColumns.map((col: any, i: number) => { + if (i !== columnIdx) return col; + if (typeof col === 'string') return newColumn; + return { ...col, name: newColumn, columnName: newColumn }; + }); + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + rightPanel: { + ...comp.componentConfig.rightPanel, + columns: updatedRightColumns, + }, + }, + }; + } + } + + return comp; + }); + + if (!columnChanged) { + toast.warning("변경할 컬럼을 찾을 수 없습니다."); + console.warn("[handleColumnChange] 변경할 컬럼 없음", { oldColumn, newColumn }); + return; + } + + // 3. 저장 + console.log("[handleColumnChange] 저장 시작", { + screenId, + componentCount: updatedComponents.length + }); + await screenApi.saveLayout(screenId, { + ...currentLayout, + components: updatedComponents, + }); + console.log("[handleColumnChange] 저장 완료"); + + if (isAddingField) { + toast.success(`필드가 추가되었습니다: ${newColumn}`); + } else if (isRemovingField) { + toast.success(`필드가 제거되었습니다: ${oldColumn}`); + } else { + toast.success(`컬럼이 변경되었습니다: ${oldColumn} → ${newColumn}`); + } + + // 실시간 반영을 위해 콜백 호출 + onRefresh?.(); + } catch (error) { + console.error("컬럼 변경 저장 실패:", error); + toast.error("컬럼 변경 저장에 실패했습니다."); + } finally { + setIsSavingColumn(false); + } + }, [screenId, onRefresh]); + + // 컬럼 순서 변경 저장 함수 + const handleColumnReorder = useCallback(async (tableType: "main" | "filter", newOrder: string[]) => { + console.log("[handleColumnReorder] 시작", { screenId, tableType, newOrder }); + + if (!screenId) { + console.warn("[handleColumnReorder] screenId 없음"); + return; + } + + try { + // 1. 현재 레이아웃 가져오기 + const currentLayout = await screenApi.getLayout(screenId); + + if (!currentLayout?.components) { + console.error("[handleColumnReorder] 레이아웃 정보 없음"); + return; + } + + // 2. 레이아웃에서 해당 컬럼들의 순서 변경 + let orderChanged = false; + + const updatedComponents = currentLayout.components.map((comp: any) => { + // split-panel-layout의 leftPanel.columns 순서 변경 + if (comp.componentConfig?.leftPanel?.columns && Array.isArray(comp.componentConfig.leftPanel.columns)) { + const leftColumns = comp.componentConfig.leftPanel.columns as any[]; + + // newOrder에 따라 leftColumns 재정렬 + const reorderedColumns = newOrder.map(colName => { + return leftColumns.find((col: any) => { + const name = typeof col === 'string' ? col : (col.name || col.columnName || col.field); + return name?.toLowerCase() === colName.toLowerCase(); + }); + }).filter(Boolean); + + // 원래 없던 컬럼들 유지 (newOrder에 없는 컬럼들) + const remainingColumns = leftColumns.filter((col: any) => { + const name = typeof col === 'string' ? col : (col.name || col.columnName || col.field); + return !newOrder.some(n => n.toLowerCase() === name?.toLowerCase()); + }); + + if (reorderedColumns.length > 0) { + orderChanged = true; + console.log("[handleColumnReorder] leftPanel.columns 순서 변경", { + compId: comp.id, + before: leftColumns.map((c: any) => typeof c === 'string' ? c : (c.name || c.columnName)), + after: [...reorderedColumns, ...remainingColumns].map((c: any) => typeof c === 'string' ? c : (c.name || c.columnName)), + }); + + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + leftPanel: { + ...comp.componentConfig.leftPanel, + columns: [...reorderedColumns, ...remainingColumns], + }, + }, + }; + } + } + + // rightPanel.columns 순서 변경 + if (comp.componentConfig?.rightPanel?.columns && Array.isArray(comp.componentConfig.rightPanel.columns)) { + const rightColumns = comp.componentConfig.rightPanel.columns as any[]; + + const reorderedColumns = newOrder.map(colName => { + return rightColumns.find((col: any) => { + const name = typeof col === 'string' ? col : (col.name || col.columnName || col.field); + return name?.toLowerCase() === colName.toLowerCase(); + }); + }).filter(Boolean); + + const remainingColumns = rightColumns.filter((col: any) => { + const name = typeof col === 'string' ? col : (col.name || col.columnName || col.field); + return !newOrder.some(n => n.toLowerCase() === name?.toLowerCase()); + }); + + if (reorderedColumns.length > 0) { + orderChanged = true; + console.log("[handleColumnReorder] rightPanel.columns 순서 변경", { + compId: comp.id, + before: rightColumns.map((c: any) => typeof c === 'string' ? c : (c.name || c.columnName)), + after: [...reorderedColumns, ...remainingColumns].map((c: any) => typeof c === 'string' ? c : (c.name || c.columnName)), + }); + + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + rightPanel: { + ...comp.componentConfig.rightPanel, + columns: [...reorderedColumns, ...remainingColumns], + }, + }, + }; + } + } + + // componentConfig.usedColumns 순서 변경 + if (comp.componentConfig?.usedColumns && Array.isArray(comp.componentConfig.usedColumns)) { + const usedColumns = comp.componentConfig.usedColumns as string[]; + + const reorderedColumns = newOrder.filter(colName => + usedColumns.some(c => c.toLowerCase() === colName.toLowerCase()) + ); + + const remainingColumns = usedColumns.filter(c => + !newOrder.some(n => n.toLowerCase() === c.toLowerCase()) + ); + + if (reorderedColumns.length > 0) { + orderChanged = true; + console.log("[handleColumnReorder] usedColumns 순서 변경", { + compId: comp.id, + before: usedColumns, + after: [...reorderedColumns, ...remainingColumns], + }); + + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + usedColumns: [...reorderedColumns, ...remainingColumns], + }, + }; + } + } + + // componentConfig.columns 순서 변경 + if (comp.componentConfig?.columns && Array.isArray(comp.componentConfig.columns)) { + const columns = comp.componentConfig.columns as any[]; + + const reorderedColumns = newOrder.map(colName => { + return columns.find((col: any) => { + const name = typeof col === 'string' ? col : (col.field || col.columnName || col.name); + return name?.toLowerCase() === colName.toLowerCase(); + }); + }).filter(Boolean); + + const remainingColumns = columns.filter((col: any) => { + const name = typeof col === 'string' ? col : (col.field || col.columnName || col.name); + return !newOrder.some(n => n.toLowerCase() === name?.toLowerCase()); + }); + + if (reorderedColumns.length > 0) { + orderChanged = true; + console.log("[handleColumnReorder] componentConfig.columns 순서 변경", { + compId: comp.id, + before: columns.map((c: any) => typeof c === 'string' ? c : (c.field || c.columnName)), + after: [...reorderedColumns, ...remainingColumns].map((c: any) => typeof c === 'string' ? c : (c.field || c.columnName)), + }); + + return { + ...comp, + componentConfig: { + ...comp.componentConfig, + columns: [...reorderedColumns, ...remainingColumns], + }, + }; + } + } + + return comp; + }); + + if (!orderChanged) { + console.log("[handleColumnReorder] 순서 변경 없음"); + return; + } + + // 3. 레이아웃 저장 + console.log("[handleColumnReorder] 레이아웃 저장"); + await screenApi.saveLayout(screenId, { + ...currentLayout, + components: updatedComponents, + }); + + console.log("[handleColumnReorder] 순서 변경 저장 완료"); + + // 실시간 반영을 위해 콜백 호출 + onRefresh?.(); + } catch (error) { + console.error("[handleColumnReorder] 순서 변경 저장 실패:", error); + toast.error("컬럼 순서 변경 저장에 실패했습니다."); + } + }, [screenId, onRefresh]); + // 통계 계산 (layoutItems의 컬럼 수도 포함) const stats = useMemo(() => { const totalJoins = filterTables.reduce( @@ -612,25 +1931,38 @@ function OverviewTab({
- {/* 메인 테이블 */} + {/* 메인 테이블 (아코디언 형식) */}

메인 테이블

{mainTable ? ( -
- -
-
{mainTableLabel || mainTable}
- {mainTableLabel && mainTable !== mainTableLabel && ( -
{mainTable}
- )} -
- - 메인 - -
+ a.y - b.y) // 화면 순서대로 정렬 + .flatMap((item, idx) => + (item.usedColumns || []).map(col => ({ + columnName: col, + fieldLabel: col, // 컬럼명 자체를 식별자로 사용 (UI에서 columnLabel 표시) + order: idx * 100 + (item.usedColumns?.indexOf(col) || 0), // 순서 유지 + })) + ) + // 중복 제거 (첫 번째 매핑만 유지) + .filter((mapping, idx, arr) => + arr.findIndex(m => m.columnName.toLowerCase() === mapping.columnName.toLowerCase()) === idx + ) + } + onColumnChange={handleColumnChange} + onColumnReorder={(newOrder) => handleColumnReorder("main", newOrder)} + onJoinSettingSaved={onRefresh} + /> ) : (
메인 테이블이 설정되지 않았습니다. @@ -646,13 +1978,49 @@ function OverviewTab({ {filterTables.length > 0 ? (
- {filterTables.map((ft, idx) => ( - - ))} + {filterTables.map((ft, idx) => { + // 이 필터 테이블에서 사용되는 컬럼 매핑 정보 추출 + // 1. layoutItems의 usedColumns에서 추출 + const usedColumnMappings: ColumnMapping[] = layoutItems + .slice() + .sort((a, b) => a.y - b.y) + .flatMap((item, itemIdx) => + (item.usedColumns || []).map(col => ({ + columnName: col, + fieldLabel: col, + order: itemIdx * 100 + (item.usedColumns?.indexOf(col) || 0), + })) + ); + + // 2. 조인 컬럼도 필드로 추가 (화면에서 조인 테이블 데이터를 보여주므로) + const joinColumnMappings: ColumnMapping[] = (ft.joinColumnRefs || []).map((ref, refIdx) => ({ + columnName: ref.column, + fieldLabel: ref.column, + order: 1000 + refIdx, // 조인 컬럼은 후순위 + })); + + // 3. 합치고 중복 제거 + const filterTableColumnMappings: ColumnMapping[] = [...usedColumnMappings, ...joinColumnMappings] + .filter((mapping, i, arr) => + arr.findIndex(m => m.columnName.toLowerCase() === mapping.columnName.toLowerCase()) === i + ); + + return ( + handleColumnReorder("filter", newOrder)} + onJoinSettingSaved={onRefresh} + /> + ); + })}
) : (
@@ -717,6 +2085,43 @@ function FieldMappingTab({ layoutItems, loading, }: FieldMappingTabProps) { + // 편집 모드 상태 + const [isEditMode, setIsEditMode] = useState(false); + // 테이블 컬럼 목록 (편집용) + const [tableColumns, setTableColumns] = useState([]); + const [loadingTableColumns, setLoadingTableColumns] = useState(false); + // 편집 중인 컬럼 정보 + const [editingColumn, setEditingColumn] = useState<{ + componentIdx: number; + columnIdx: number; + currentColumn: string; + } | null>(null); + const [editPopoverOpen, setEditPopoverOpen] = useState(false); + + // 테이블 컬럼 로드 + const loadTableColumns = useCallback(async () => { + if (!mainTable || tableColumns.length > 0) return; + + setLoadingTableColumns(true); + try { + const result = await tableManagementApi.getColumnList(mainTable); + if (result.success && result.data?.columns) { + setTableColumns(result.data.columns); + } + } catch (error) { + console.error("테이블 컬럼 로드 실패:", error); + } finally { + setLoadingTableColumns(false); + } + }, [mainTable, tableColumns.length]); + + // 편집 모드 진입 시 컬럼 로드 + useEffect(() => { + if (isEditMode) { + loadTableColumns(); + } + }, [isEditMode, loadTableColumns]); + // 화면 컴포넌트에서 사용하는 컬럼 정보 추출 const componentColumns = useMemo(() => { const result: Array<{ @@ -748,6 +2153,15 @@ function FieldMappingTab({ }); return allColumns.size; }, [componentColumns]); + + // 컬럼명 → 표시명 매핑 (테이블 컬럼에서 추출) + const columnDisplayMap = useMemo(() => { + const map: Record = {}; + tableColumns.forEach((tc) => { + map[tc.columnName] = tc.displayName || tc.columnName; + }); + return map; + }, [tableColumns]); // 컴포넌트 타입별 그룹핑 (기존 fieldMappings용) const groupedMappings = useMemo(() => { @@ -782,12 +2196,34 @@ function FieldMappingTab({

화면 컴포넌트별 컬럼 사용 현황

- 각 컴포넌트에서 사용하는 테이블 컬럼을 확인합니다. + {isEditMode + ? "컬럼을 클릭하여 매핑을 변경할 수 있습니다." + : "각 컴포넌트에서 사용하는 테이블 컬럼을 확인합니다."}

- - 총 {totalColumns}개 컬럼 - +
+ + 총 {totalColumns}개 컬럼 + + +
{componentColumns.length === 0 ? ( @@ -800,9 +2236,10 @@ function FieldMappingTab({ {componentColumns.map((comp, idx) => (
-
+ {/* 컴포넌트 헤더 */} +
@@ -813,28 +2250,132 @@ function FieldMappingTab({
- {comp.columns.length}개 컬럼 + {comp.columns.length}개 필드
-
+ + {/* 필드 → 컬럼 매핑 테이블 */} +
+ {/* 테이블 헤더 */} +
+ 필드명 (화면 표시) + + 컬럼명 (데이터베이스) +
+ + {/* 매핑 행들 */} {comp.columns.map((col, cIdx) => { const isJoinColumn = comp.joinColumns.includes(col); + const displayName = columnDisplayMap[col] || col; + const isEditing = editingColumn?.componentIdx === idx && editingColumn?.columnIdx === cIdx; + return ( - - {col} - {isJoinColumn && ( - + {/* 필드명 (화면 표시) */} +
+ + {displayName} + + {isJoinColumn && ( + + 조인 + + )} +
+ + {/* 화살표 */} +
+ +
+ + {/* 컬럼명 (데이터베이스) */} + {isEditMode ? ( + { + if (open) { + setEditingColumn({ componentIdx: idx, columnIdx: cIdx, currentColumn: col }); + } else { + setEditingColumn(null); + } + setEditPopoverOpen(open); + }} + > + + + + + + + + + 컬럼을 찾을 수 없습니다. + + + {loadingTableColumns ? ( +
+ +
+ ) : ( + tableColumns.map((tableCol) => ( + { + toast.info(`컬럼 변경: ${col} → ${value}`, { + description: "저장 기능은 아직 구현 중입니다." + }); + setEditPopoverOpen(false); + setEditingColumn(null); + }} + className="text-xs" + > +
+ + {tableCol.displayName || tableCol.columnName} + + {tableCol.displayName && tableCol.displayName !== tableCol.columnName && ( + + {tableCol.columnName} + + )} +
+ {tableCol.columnName === col && ( + + )} +
+ )) + )} +
+
+
+
+
+ ) : ( + {col} )} -
+
); })}
@@ -1210,6 +2751,627 @@ function DataFlowTab({ ); } +// ============================================================ +// 탭: 제어 관리 +// ============================================================ + +interface ButtonControlInfo { + id: string; + label: string; + actionType: string; + targetTable?: string; + operations?: string[]; + confirmMessage?: string; + hasDataflowControl?: boolean; + dataflowControlMode?: string; + linkedExternalCall?: { + id: number; + name: string; + }; + linkedFlow?: { + id: number; + name: string; + }; +} + +interface ControlManagementTabProps { + screenId: number; + layoutItems: LayoutItem[]; + loading: boolean; + onRefresh: () => void; +} + +function ControlManagementTab({ + screenId, + layoutItems, + loading: parentLoading, + onRefresh, +}: ControlManagementTabProps) { + const [buttonControls, setButtonControls] = useState([]); + const [externalCalls, setExternalCalls] = useState([]); + const [flows, setFlows] = useState([]); + const [loading, setLoading] = useState(false); + const [expandedButton, setExpandedButton] = useState(null); + const [editingButton, setEditingButton] = useState(null); + const [editedValues, setEditedValues] = useState>({}); + + // 테이블 목록 조회 + const [tableList, setTableList] = useState([]); + + // 데이터 로드 + const loadData = useCallback(async () => { + setLoading(true); + try { + // 1. 화면 레이아웃에서 버튼 정보 추출 + const layoutResponse = await screenApi.getLayout(screenId); + console.log("[제어관리] 레이아웃 응답:", layoutResponse); + + if (layoutResponse?.components) { + const buttons: ButtonControlInfo[] = []; + + // 컴포넌트에서 버튼 추출 (다양한 필드 확인) + const extractButtons = (components: any[], depth = 0) => { + for (const comp of components) { + // 버튼 컴포넌트 필터링 (다양한 조건 확인) + const isButton = + comp.webType === "button" || + comp.componentType === "button" || + comp.type === "button" || + comp.componentKind?.includes("button") || + comp.widgetType === "button"; + + if (isButton) { + const config = comp.componentConfig || {}; + const webTypeConfig = comp.webTypeConfig || {}; + const action = config.action || {}; + + console.log("[제어관리] 버튼 발견:", comp); + + buttons.push({ + id: comp.id || comp.componentId || `btn-${buttons.length}`, + label: config.text || comp.label || comp.title || comp.name || "버튼", + actionType: typeof action === "string" ? action : (action.type || "custom"), + targetTable: config.tableName || webTypeConfig.tableName || comp.tableName, + operations: action.operations || [], + confirmMessage: action.confirmMessage || config.confirmMessage, + hasDataflowControl: webTypeConfig.enableDataflowControl, + dataflowControlMode: webTypeConfig.dataflowConfig?.controlMode, + linkedExternalCall: undefined, // TODO: 연결 정보 조회 + linkedFlow: webTypeConfig.dataflowConfig?.flowConfig ? { + id: webTypeConfig.dataflowConfig.flowConfig.flowId, + name: webTypeConfig.dataflowConfig.flowConfig.flowName, + } : undefined, + }); + } + + // 자식 컴포넌트 처리 (여러 필드 확인) + if (comp.children && Array.isArray(comp.children)) { + extractButtons(comp.children, depth + 1); + } + // componentConfig 내 중첩된 컴포넌트 확인 + if (comp.componentConfig?.children && Array.isArray(comp.componentConfig.children)) { + extractButtons(comp.componentConfig.children, depth + 1); + } + // items 배열 확인 (일부 레이아웃에서 사용) + if (comp.items && Array.isArray(comp.items)) { + extractButtons(comp.items, depth + 1); + } + } + }; + + extractButtons(layoutResponse.components); + console.log("[제어관리] 추출된 버튼:", buttons); + setButtonControls(buttons); + } + + // 2. 외부 호출 목록 조회 + const externalResponse = await ExternalCallConfigAPI.getConfigs({ is_active: "Y" }); + if (externalResponse.success && externalResponse.data) { + setExternalCalls(externalResponse.data); + } + + // 3. 플로우 목록 조회 + const flowResponse = await getFlowDefinitions({ isActive: true }); + if (flowResponse.success && flowResponse.data) { + setFlows(flowResponse.data); + } + + // 4. 테이블 목록 조회 + const tableResponse = await tableManagementApi.getTableList(); + if (tableResponse.success && tableResponse.data) { + setTableList(tableResponse.data); + } + } catch (error) { + console.error("제어 관리 데이터 로드 실패:", error); + toast.error("데이터 로드 실패"); + } finally { + setLoading(false); + } + }, [screenId]); + + useEffect(() => { + loadData(); + }, [loadData]); + + // 버튼 설정 저장 + const handleSaveButton = async (buttonId: string) => { + const values = editedValues[buttonId]; + if (!values) return; + + try { + // 레이아웃에서 해당 버튼 찾아서 업데이트 + const layoutResponse = await screenApi.getLayout(screenId); + if (!layoutResponse?.components) { + toast.error("레이아웃을 불러올 수 없습니다"); + return; + } + + // 버튼 컴포넌트 업데이트 + const updateButton = (components: any[]): boolean => { + for (const comp of components) { + if ((comp.id === buttonId || comp.componentId === buttonId) && + (comp.webType === "button" || comp.componentKind?.includes("button"))) { + // componentConfig 업데이트 + if (!comp.componentConfig) comp.componentConfig = {}; + if (!comp.componentConfig.action) comp.componentConfig.action = {}; + + if (values.targetTable) { + comp.componentConfig.tableName = values.targetTable; + } + if (values.confirmMessage !== undefined) { + comp.componentConfig.action.confirmMessage = values.confirmMessage; + } + if (values.operations) { + comp.componentConfig.action.operations = values.operations; + } + + // webTypeConfig 업데이트 (플로우 연동) + if (!comp.webTypeConfig) comp.webTypeConfig = {}; + if (values.linkedFlowId) { + comp.webTypeConfig.enableDataflowControl = true; + comp.webTypeConfig.dataflowConfig = { + controlMode: "flow", + flowConfig: { + flowId: values.linkedFlowId, + flowName: flows.find(f => f.id === values.linkedFlowId)?.name || "", + executionTiming: values.flowTiming || "after", + }, + }; + } else if (values.linkedFlowId === null) { + // 플로우 연동 해제 + comp.webTypeConfig.enableDataflowControl = false; + delete comp.webTypeConfig.dataflowConfig; + } + + return true; + } + + if (comp.children && Array.isArray(comp.children)) { + if (updateButton(comp.children)) return true; + } + } + return false; + }; + + if (updateButton(layoutResponse.components)) { + // 레이아웃 저장 + await screenApi.saveLayout(screenId, layoutResponse); + toast.success("버튼 설정이 저장되었습니다"); + setEditingButton(null); + setEditedValues(prev => { + const next = { ...prev }; + delete next[buttonId]; + return next; + }); + loadData(); + onRefresh(); + } else { + toast.error("버튼을 찾을 수 없습니다"); + } + } catch (error) { + console.error("버튼 설정 저장 실패:", error); + toast.error("저장 실패"); + } + }; + + // 액션 타입 라벨 + const getActionTypeLabel = (type: string) => { + const labels: Record = { + save: "저장", + delete: "삭제", + refresh: "새로고침", + reset: "초기화", + submit: "제출", + cancel: "취소", + close: "닫기", + navigate: "이동", + popup: "팝업", + custom: "커스텀", + }; + return labels[type] || type; + }; + + // 액션 타입 색상 + const getActionTypeColor = (type: string) => { + switch (type) { + case "save": + return "bg-green-100 text-green-700"; + case "delete": + return "bg-red-100 text-red-700"; + case "refresh": + return "bg-blue-100 text-blue-700"; + case "submit": + return "bg-purple-100 text-purple-700"; + default: + return "bg-gray-100 text-gray-700"; + } + }; + + if (loading || parentLoading) { + return ( +
+ +
+ ); + } + + return ( +
+ {/* 버튼 액션 설정 */} +
+
+ + 버튼 액션 설정 + + {buttonControls.length}개 + +
+ +
+ {buttonControls.length === 0 ? ( +
+ +

버튼이 없습니다

+

화면 디자이너에서 버튼을 추가하세요

+
+ ) : ( +
+ {buttonControls.map((btn) => ( +
+ {/* 버튼 헤더 */} +
setExpandedButton(expandedButton === btn.id ? null : btn.id)} + > + {expandedButton === btn.id ? ( + + ) : ( + + )} + [{btn.label}] + + {getActionTypeLabel(btn.actionType)} + + {btn.targetTable && ( + + → {btn.targetTable} + + )} + {btn.hasDataflowControl && ( + + + 제어 연동 + + )} + +
+ + {/* 버튼 상세 (확장 시) */} + {expandedButton === btn.id && ( +
+
+ {/* 대상 테이블 */} +
+ + {editingButton === btn.id ? ( + + ) : ( + + {btn.targetTable || 미설정} + + )} +
+ + {/* 확인 메시지 */} +
+ + {editingButton === btn.id ? ( + setEditedValues(prev => ({ + ...prev, + [btn.id]: { ...prev[btn.id], confirmMessage: e.target.value } + }))} + className="h-7 text-xs" + placeholder="예: 정말 저장하시겠습니까?" + /> + ) : ( + + {btn.confirmMessage || 없음} + + )} +
+ + {/* 플로우 연동 */} +
+ + {editingButton === btn.id ? ( + + ) : ( + + {btn.linkedFlow ? ( + + + {btn.linkedFlow.name} + + ) : ( + 없음 + )} + + )} +
+ + {/* 편집/저장 버튼 */} +
+ {editingButton === btn.id ? ( + <> + + + + ) : ( + + )} +
+
+
+ )} +
+ ))} +
+ )} +
+
+ + {/* 외부 연동 */} +
+
+ + 외부 연동 + + {externalCalls.filter(e => e.is_active === "Y").length}개 활성 + +
+ +
+ {externalCalls.length === 0 ? ( +
+ +

외부 호출 설정이 없습니다

+ +
+ ) : ( +
+ {externalCalls.slice(0, 5).map((call) => ( +
+ + {call.call_type} + + {call.config_name} + +
+ ))} + {externalCalls.length > 5 && ( +
+ +
+ )} +
+ )} +
+ +
+
+ + 버튼에 외부 호출을 연결하려면 버튼 편집에서 설정하세요 +
+
+
+ + {/* 플로우 연동 */} +
+
+ + 플로우 연동 + + {flows.length}개 + +
+ +
+ {flows.length === 0 ? ( +
+ +

플로우가 없습니다

+ +
+ ) : ( +
+ {flows.slice(0, 5).map((flow) => ( +
+ + 플로우 + + {flow.name} + + {flow.tableName} + + +
+ ))} + {flows.length > 5 && ( +
+ +
+ )} +
+ )} +
+ +
+
+ + 버튼에 플로우를 연결하려면 버튼 편집에서 설정하세요 +
+
+
+
+ ); +} + // ============================================================ // 탭 4: 화면 프리뷰 (iframe) // ============================================================ @@ -1218,11 +3380,52 @@ interface PreviewTabProps { screenId: number; screenName: string; companyCode?: string; + iframeKey?: number; // iframe 새로고침용 키 } -function PreviewTab({ screenId, screenName, companyCode }: PreviewTabProps) { +function PreviewTab({ screenId, screenName, companyCode, iframeKey = 0 }: PreviewTabProps) { const [loading, setLoading] = useState(true); const [error, setError] = useState(null); + const containerRef = useRef(null); + + // 화면 디자인 크기 (모달 프리뷰에 맞춘 크기) + const designWidth = 1200; + const designHeight = 750; + + // 컨테이너에 맞는 초기 스케일 계산 + const [initialScale, setInitialScale] = useState(0.7); + + // 컨테이너 크기에 맞춰 초기 스케일 계산 + useEffect(() => { + const updateInitialScale = () => { + if (containerRef.current) { + const containerWidth = containerRef.current.offsetWidth; + const containerHeight = containerRef.current.offsetHeight; + + // 여백 5px씩만 적용하여 꽉 차게 + const scaleX = (containerWidth - 10) / designWidth; + const scaleY = (containerHeight - 10) / designHeight; + const newScale = Math.min(scaleX, scaleY); + + setInitialScale(newScale); + } + }; + + // 초기 측정 (약간의 딜레이) + const timer = setTimeout(updateInitialScale, 200); + + // 리사이즈 감지 + const resizeObserver = new ResizeObserver(updateInitialScale); + if (containerRef.current) { + resizeObserver.observe(containerRef.current); + } + + return () => { + clearTimeout(timer); + resizeObserver.disconnect(); + }; + }, []); + // 화면 URL 생성 (preview=true로 사이드바 없이 화면만 표시, company_code 전달) const previewUrl = useMemo(() => { @@ -1253,19 +3456,17 @@ function PreviewTab({ screenId, screenName, companyCode }: PreviewTabProps) { }; return ( -
- {/* 상단 툴바 */} -
-
- - 화면 프리뷰 - - Screen ID: {screenId} - +
+ {/* 상단 툴바 (최소화) */} +
+
+ + {screenName} + (휠: 확대/축소, 드래그: 이동)
-
+
-
- {/* iframe 영역 */} -
+ {/* iframe 영역 - Ctrl+휠로 확대/축소, 내부 버튼/목록 클릭 가능 */} +
{loading && (
@@ -1318,21 +3521,109 @@ function PreviewTab({ screenId, screenName, companyCode }: PreviewTabProps) {
) : ( -