Merge origin/main into ksh
This commit is contained in:
commit
430723df59
|
|
@ -140,7 +140,7 @@ if (comp.componentType === "my-new-component") {
|
||||||
if (config?.title) {
|
if (config?.title) {
|
||||||
addLabel({
|
addLabel({
|
||||||
id: `${comp.id}_title`,
|
id: `${comp.id}_title`,
|
||||||
componentId: `${comp.id}_title`,
|
componentId: `${comp.id}_title`,-
|
||||||
label: config.title,
|
label: config.title,
|
||||||
type: "title",
|
type: "title",
|
||||||
parentType: "my-new-component",
|
parentType: "my-new-component",
|
||||||
|
|
|
||||||
|
|
@ -1418,69 +1418,31 @@ export async function updateMenu(
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 메뉴 삭제
|
* 재귀적으로 모든 하위 메뉴 ID를 수집하는 헬퍼 함수
|
||||||
*/
|
*/
|
||||||
export async function deleteMenu(
|
async function collectAllChildMenuIds(parentObjid: number): Promise<number[]> {
|
||||||
req: AuthenticatedRequest,
|
const allIds: number[] = [];
|
||||||
res: Response
|
|
||||||
): Promise<void> {
|
|
||||||
try {
|
|
||||||
const { menuId } = req.params;
|
|
||||||
logger.info(`메뉴 삭제 요청: menuId = ${menuId}`, { user: req.user });
|
|
||||||
|
|
||||||
// 사용자의 company_code 확인
|
// 직접 자식 메뉴들 조회
|
||||||
if (!req.user?.companyCode) {
|
const children = await query<any>(
|
||||||
res.status(400).json({
|
`SELECT objid FROM menu_info WHERE parent_obj_id = $1`,
|
||||||
success: false,
|
[parentObjid]
|
||||||
message: "사용자의 회사 코드를 찾을 수 없습니다.",
|
|
||||||
error: "Missing company_code",
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const userCompanyCode = req.user.companyCode;
|
|
||||||
const userType = req.user.userType;
|
|
||||||
|
|
||||||
// 삭제하려는 메뉴 조회
|
|
||||||
const currentMenu = await queryOne<any>(
|
|
||||||
`SELECT objid, company_code FROM menu_info WHERE objid = $1`,
|
|
||||||
[Number(menuId)]
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!currentMenu) {
|
for (const child of children) {
|
||||||
res.status(404).json({
|
allIds.push(child.objid);
|
||||||
success: false,
|
// 자식의 자식들도 재귀적으로 수집
|
||||||
message: `메뉴를 찾을 수 없습니다: ${menuId}`,
|
const grandChildren = await collectAllChildMenuIds(child.objid);
|
||||||
error: "Menu not found",
|
allIds.push(...grandChildren);
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 공통 메뉴(company_code = '*')는 최고 관리자만 삭제 가능
|
return allIds;
|
||||||
if (currentMenu.company_code === "*") {
|
|
||||||
if (userCompanyCode !== "*" || userType !== "SUPER_ADMIN") {
|
|
||||||
res.status(403).json({
|
|
||||||
success: false,
|
|
||||||
message: "공통 메뉴는 최고 관리자만 삭제할 수 있습니다.",
|
|
||||||
error: "Unauthorized to delete common menu",
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
} else if (userCompanyCode !== "*") {
|
|
||||||
// 회사 관리자는 자기 회사 메뉴만 삭제 가능
|
|
||||||
if (currentMenu.company_code !== userCompanyCode) {
|
|
||||||
res.status(403).json({
|
|
||||||
success: false,
|
|
||||||
message: "해당 회사의 메뉴를 삭제할 권한이 없습니다.",
|
|
||||||
error: "Unauthorized to delete menu for this company",
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 외래키 제약 조건이 있는 관련 테이블 데이터 먼저 정리
|
/**
|
||||||
const menuObjid = Number(menuId);
|
* 메뉴 및 관련 데이터 정리 헬퍼 함수
|
||||||
|
*/
|
||||||
|
async function cleanupMenuRelatedData(menuObjid: number): Promise<void> {
|
||||||
// 1. category_column_mapping에서 menu_objid를 NULL로 설정
|
// 1. category_column_mapping에서 menu_objid를 NULL로 설정
|
||||||
await query(
|
await query(
|
||||||
`UPDATE category_column_mapping SET menu_objid = NULL WHERE menu_objid = $1`,
|
`UPDATE category_column_mapping SET menu_objid = NULL WHERE menu_objid = $1`,
|
||||||
|
|
@ -1517,28 +1479,118 @@ export async function deleteMenu(
|
||||||
[menuObjid]
|
[menuObjid]
|
||||||
);
|
);
|
||||||
|
|
||||||
logger.info("메뉴 관련 데이터 정리 완료", { menuObjid });
|
// 7. screen_groups에서 menu_objid를 NULL로 설정
|
||||||
|
await query(
|
||||||
// Raw Query를 사용한 메뉴 삭제
|
`UPDATE screen_groups SET menu_objid = NULL WHERE menu_objid = $1`,
|
||||||
const [deletedMenu] = await query<any>(
|
|
||||||
`DELETE FROM menu_info WHERE objid = $1 RETURNING *`,
|
|
||||||
[menuObjid]
|
[menuObjid]
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
|
||||||
logger.info("메뉴 삭제 성공", { deletedMenu });
|
/**
|
||||||
|
* 메뉴 삭제
|
||||||
|
*/
|
||||||
|
export async function deleteMenu(
|
||||||
|
req: AuthenticatedRequest,
|
||||||
|
res: Response
|
||||||
|
): Promise<void> {
|
||||||
|
try {
|
||||||
|
const { menuId } = req.params;
|
||||||
|
logger.info(`메뉴 삭제 요청: menuId = ${menuId}`, { user: req.user });
|
||||||
|
|
||||||
|
// 사용자의 company_code 확인
|
||||||
|
if (!req.user?.companyCode) {
|
||||||
|
res.status(400).json({
|
||||||
|
success: false,
|
||||||
|
message: "사용자의 회사 코드를 찾을 수 없습니다.",
|
||||||
|
error: "Missing company_code",
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const userCompanyCode = req.user.companyCode;
|
||||||
|
const userType = req.user.userType;
|
||||||
|
|
||||||
|
// 삭제하려는 메뉴 조회
|
||||||
|
const currentMenu = await queryOne<any>(
|
||||||
|
`SELECT objid, company_code, menu_name_kor FROM menu_info WHERE objid = $1`,
|
||||||
|
[Number(menuId)]
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!currentMenu) {
|
||||||
|
res.status(404).json({
|
||||||
|
success: false,
|
||||||
|
message: `메뉴를 찾을 수 없습니다: ${menuId}`,
|
||||||
|
error: "Menu not found",
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 공통 메뉴(company_code = '*')는 최고 관리자만 삭제 가능
|
||||||
|
if (currentMenu.company_code === "*") {
|
||||||
|
if (userCompanyCode !== "*" || userType !== "SUPER_ADMIN") {
|
||||||
|
res.status(403).json({
|
||||||
|
success: false,
|
||||||
|
message: "공통 메뉴는 최고 관리자만 삭제할 수 있습니다.",
|
||||||
|
error: "Unauthorized to delete common menu",
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else if (userCompanyCode !== "*") {
|
||||||
|
// 회사 관리자는 자기 회사 메뉴만 삭제 가능
|
||||||
|
if (currentMenu.company_code !== userCompanyCode) {
|
||||||
|
res.status(403).json({
|
||||||
|
success: false,
|
||||||
|
message: "해당 회사의 메뉴를 삭제할 권한이 없습니다.",
|
||||||
|
error: "Unauthorized to delete menu for this company",
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const menuObjid = Number(menuId);
|
||||||
|
|
||||||
|
// 하위 메뉴들 재귀적으로 수집
|
||||||
|
const childMenuIds = await collectAllChildMenuIds(menuObjid);
|
||||||
|
const allMenuIdsToDelete = [menuObjid, ...childMenuIds];
|
||||||
|
|
||||||
|
logger.info(`메뉴 삭제 대상: 본인(${menuObjid}) + 하위 메뉴 ${childMenuIds.length}개`, {
|
||||||
|
menuName: currentMenu.menu_name_kor,
|
||||||
|
totalCount: allMenuIdsToDelete.length,
|
||||||
|
childMenuIds,
|
||||||
|
});
|
||||||
|
|
||||||
|
// 모든 삭제 대상 메뉴에 대해 관련 데이터 정리
|
||||||
|
for (const objid of allMenuIdsToDelete) {
|
||||||
|
await cleanupMenuRelatedData(objid);
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.info("메뉴 관련 데이터 정리 완료", {
|
||||||
|
menuObjid,
|
||||||
|
totalCleaned: allMenuIdsToDelete.length
|
||||||
|
});
|
||||||
|
|
||||||
|
// 하위 메뉴부터 역순으로 삭제 (외래키 제약 회피)
|
||||||
|
// 가장 깊은 하위부터 삭제해야 하므로 역순으로
|
||||||
|
const reversedIds = [...allMenuIdsToDelete].reverse();
|
||||||
|
|
||||||
|
for (const objid of reversedIds) {
|
||||||
|
await query(`DELETE FROM menu_info WHERE objid = $1`, [objid]);
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.info("메뉴 삭제 성공", {
|
||||||
|
deletedMenuObjid: menuObjid,
|
||||||
|
deletedMenuName: currentMenu.menu_name_kor,
|
||||||
|
totalDeleted: allMenuIdsToDelete.length,
|
||||||
|
});
|
||||||
|
|
||||||
const response: ApiResponse<any> = {
|
const response: ApiResponse<any> = {
|
||||||
success: true,
|
success: true,
|
||||||
message: "메뉴가 성공적으로 삭제되었습니다.",
|
message: `메뉴가 성공적으로 삭제되었습니다. (하위 메뉴 ${childMenuIds.length}개 포함)`,
|
||||||
data: {
|
data: {
|
||||||
objid: deletedMenu.objid.toString(),
|
objid: menuObjid.toString(),
|
||||||
menuNameKor: deletedMenu.menu_name_kor,
|
menuNameKor: currentMenu.menu_name_kor,
|
||||||
menuNameEng: deletedMenu.menu_name_eng,
|
deletedCount: allMenuIdsToDelete.length,
|
||||||
menuUrl: deletedMenu.menu_url,
|
deletedChildCount: childMenuIds.length,
|
||||||
menuDesc: deletedMenu.menu_desc,
|
|
||||||
status: deletedMenu.status,
|
|
||||||
writer: deletedMenu.writer,
|
|
||||||
regdate: new Date(deletedMenu.regdate).toISOString(),
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -1623,18 +1675,49 @@ export async function deleteMenusBatch(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 모든 삭제 대상 메뉴 ID 수집 (하위 메뉴 포함)
|
||||||
|
const allMenuIdsToDelete = new Set<number>();
|
||||||
|
|
||||||
|
for (const menuId of menuIds) {
|
||||||
|
const objid = Number(menuId);
|
||||||
|
allMenuIdsToDelete.add(objid);
|
||||||
|
|
||||||
|
// 하위 메뉴들 재귀적으로 수집
|
||||||
|
const childMenuIds = await collectAllChildMenuIds(objid);
|
||||||
|
childMenuIds.forEach(id => allMenuIdsToDelete.add(Number(id)));
|
||||||
|
}
|
||||||
|
|
||||||
|
const allIdsArray = Array.from(allMenuIdsToDelete);
|
||||||
|
|
||||||
|
logger.info(`메뉴 일괄 삭제 대상: 선택 ${menuIds.length}개 + 하위 메뉴 포함 총 ${allIdsArray.length}개`, {
|
||||||
|
selectedMenuIds: menuIds,
|
||||||
|
totalWithChildren: allIdsArray.length,
|
||||||
|
});
|
||||||
|
|
||||||
|
// 모든 삭제 대상 메뉴에 대해 관련 데이터 정리
|
||||||
|
for (const objid of allIdsArray) {
|
||||||
|
await cleanupMenuRelatedData(objid);
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.info("메뉴 관련 데이터 정리 완료", {
|
||||||
|
totalCleaned: allIdsArray.length
|
||||||
|
});
|
||||||
|
|
||||||
// Raw Query를 사용한 메뉴 일괄 삭제
|
// Raw Query를 사용한 메뉴 일괄 삭제
|
||||||
let deletedCount = 0;
|
let deletedCount = 0;
|
||||||
let failedCount = 0;
|
let failedCount = 0;
|
||||||
const deletedMenus: any[] = [];
|
const deletedMenus: any[] = [];
|
||||||
const failedMenuIds: string[] = [];
|
const failedMenuIds: string[] = [];
|
||||||
|
|
||||||
|
// 하위 메뉴부터 삭제하기 위해 역순으로 정렬
|
||||||
|
const reversedIds = [...allIdsArray].reverse();
|
||||||
|
|
||||||
// 각 메뉴 ID에 대해 삭제 시도
|
// 각 메뉴 ID에 대해 삭제 시도
|
||||||
for (const menuId of menuIds) {
|
for (const menuObjid of reversedIds) {
|
||||||
try {
|
try {
|
||||||
const result = await query<any>(
|
const result = await query<any>(
|
||||||
`DELETE FROM menu_info WHERE objid = $1 RETURNING *`,
|
`DELETE FROM menu_info WHERE objid = $1 RETURNING *`,
|
||||||
[Number(menuId)]
|
[menuObjid]
|
||||||
);
|
);
|
||||||
|
|
||||||
if (result.length > 0) {
|
if (result.length > 0) {
|
||||||
|
|
@ -1645,20 +1728,20 @@ export async function deleteMenusBatch(
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
failedCount++;
|
failedCount++;
|
||||||
failedMenuIds.push(menuId);
|
failedMenuIds.push(String(menuObjid));
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error(`메뉴 삭제 실패 (ID: ${menuId}):`, error);
|
logger.error(`메뉴 삭제 실패 (ID: ${menuObjid}):`, error);
|
||||||
failedCount++;
|
failedCount++;
|
||||||
failedMenuIds.push(menuId);
|
failedMenuIds.push(String(menuObjid));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info("메뉴 일괄 삭제 완료", {
|
logger.info("메뉴 일괄 삭제 완료", {
|
||||||
total: menuIds.length,
|
requested: menuIds.length,
|
||||||
|
totalWithChildren: allIdsArray.length,
|
||||||
deletedCount,
|
deletedCount,
|
||||||
failedCount,
|
failedCount,
|
||||||
deletedMenus,
|
|
||||||
failedMenuIds,
|
failedMenuIds,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,7 @@
|
||||||
import { Request, Response } from "express";
|
import { Request, Response } from "express";
|
||||||
import { getPool } from "../database/db";
|
import { getPool } from "../database/db";
|
||||||
import { logger } from "../utils/logger";
|
import { logger } from "../utils/logger";
|
||||||
|
import { AuthenticatedRequest } from "../types/auth";
|
||||||
import {
|
import {
|
||||||
syncScreenGroupsToMenu,
|
syncScreenGroupsToMenu,
|
||||||
syncMenuToScreenGroups,
|
syncMenuToScreenGroups,
|
||||||
|
|
@ -16,9 +17,9 @@ const pool = getPool();
|
||||||
// ============================================================
|
// ============================================================
|
||||||
|
|
||||||
// 화면 그룹 목록 조회
|
// 화면 그룹 목록 조회
|
||||||
export const getScreenGroups = async (req: Request, res: Response) => {
|
export const getScreenGroups = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const { page = 1, size = 20, searchTerm } = req.query;
|
const { page = 1, size = 20, searchTerm } = req.query;
|
||||||
const offset = (parseInt(page as string) - 1) * parseInt(size as string);
|
const offset = (parseInt(page as string) - 1) * parseInt(size as string);
|
||||||
|
|
||||||
|
|
@ -90,10 +91,10 @@ export const getScreenGroups = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면 그룹 상세 조회
|
// 화면 그룹 상세 조회
|
||||||
export const getScreenGroup = async (req: Request, res: Response) => {
|
export const getScreenGroup = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
|
|
||||||
let query = `
|
let query = `
|
||||||
SELECT sg.*,
|
SELECT sg.*,
|
||||||
|
|
@ -136,10 +137,10 @@ export const getScreenGroup = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면 그룹 생성
|
// 화면 그룹 생성
|
||||||
export const createScreenGroup = async (req: Request, res: Response) => {
|
export const createScreenGroup = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const userCompanyCode = (req.user as any).companyCode;
|
const userCompanyCode = req.user?.companyCode || "*";
|
||||||
const userId = (req.user as any).userId;
|
const userId = req.user?.userId || "";
|
||||||
const { group_name, group_code, main_table_name, description, icon, display_order, is_active, parent_group_id, target_company_code } = req.body;
|
const { group_name, group_code, main_table_name, description, icon, display_order, is_active, parent_group_id, target_company_code } = req.body;
|
||||||
|
|
||||||
if (!group_name || !group_code) {
|
if (!group_name || !group_code) {
|
||||||
|
|
@ -210,10 +211,10 @@ export const createScreenGroup = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면 그룹 수정
|
// 화면 그룹 수정
|
||||||
export const updateScreenGroup = async (req: Request, res: Response) => {
|
export const updateScreenGroup = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const userCompanyCode = (req.user as any).companyCode;
|
const userCompanyCode = req.user?.companyCode || "*";
|
||||||
const { group_name, group_code, main_table_name, description, icon, display_order, is_active, parent_group_id, target_company_code } = req.body;
|
const { group_name, group_code, main_table_name, description, icon, display_order, is_active, parent_group_id, target_company_code } = req.body;
|
||||||
|
|
||||||
// 회사 코드 결정: 최고 관리자가 특정 회사를 선택한 경우 해당 회사로, 아니면 현재 그룹의 회사 유지
|
// 회사 코드 결정: 최고 관리자가 특정 회사를 선택한 경우 해당 회사로, 아니면 현재 그룹의 회사 유지
|
||||||
|
|
@ -299,11 +300,11 @@ export const updateScreenGroup = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면 그룹 삭제
|
// 화면 그룹 삭제
|
||||||
export const deleteScreenGroup = async (req: Request, res: Response) => {
|
export const deleteScreenGroup = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
const client = await pool.connect();
|
const client = await pool.connect();
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
|
|
||||||
await client.query('BEGIN');
|
await client.query('BEGIN');
|
||||||
|
|
||||||
|
|
@ -366,10 +367,10 @@ export const deleteScreenGroup = async (req: Request, res: Response) => {
|
||||||
// ============================================================
|
// ============================================================
|
||||||
|
|
||||||
// 그룹에 화면 추가
|
// 그룹에 화면 추가
|
||||||
export const addScreenToGroup = async (req: Request, res: Response) => {
|
export const addScreenToGroup = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const userId = (req.user as any).userId;
|
const userId = req.user?.userId || "";
|
||||||
const { group_id, screen_id, screen_role, display_order, is_default } = req.body;
|
const { group_id, screen_id, screen_role, display_order, is_default } = req.body;
|
||||||
|
|
||||||
if (!group_id || !screen_id) {
|
if (!group_id || !screen_id) {
|
||||||
|
|
@ -406,10 +407,10 @@ export const addScreenToGroup = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 그룹에서 화면 제거
|
// 그룹에서 화면 제거
|
||||||
export const removeScreenFromGroup = async (req: Request, res: Response) => {
|
export const removeScreenFromGroup = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
|
|
||||||
let query = `DELETE FROM screen_group_screens WHERE id = $1`;
|
let query = `DELETE FROM screen_group_screens WHERE id = $1`;
|
||||||
const params: any[] = [id];
|
const params: any[] = [id];
|
||||||
|
|
@ -437,10 +438,10 @@ export const removeScreenFromGroup = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 그룹 내 화면 순서/역할 수정
|
// 그룹 내 화면 순서/역할 수정
|
||||||
export const updateScreenInGroup = async (req: Request, res: Response) => {
|
export const updateScreenInGroup = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const { screen_role, display_order, is_default } = req.body;
|
const { screen_role, display_order, is_default } = req.body;
|
||||||
|
|
||||||
let query = `
|
let query = `
|
||||||
|
|
@ -476,9 +477,9 @@ export const updateScreenInGroup = async (req: Request, res: Response) => {
|
||||||
// ============================================================
|
// ============================================================
|
||||||
|
|
||||||
// 화면 필드 조인 목록 조회
|
// 화면 필드 조인 목록 조회
|
||||||
export const getFieldJoins = async (req: Request, res: Response) => {
|
export const getFieldJoins = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const { screen_id } = req.query;
|
const { screen_id } = req.query;
|
||||||
|
|
||||||
let query = `
|
let query = `
|
||||||
|
|
@ -517,10 +518,10 @@ export const getFieldJoins = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면 필드 조인 생성
|
// 화면 필드 조인 생성
|
||||||
export const createFieldJoin = async (req: Request, res: Response) => {
|
export const createFieldJoin = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const userId = (req.user as any).userId;
|
const userId = req.user?.userId || "";
|
||||||
const {
|
const {
|
||||||
screen_id, layout_id, component_id, field_name,
|
screen_id, layout_id, component_id, field_name,
|
||||||
save_table, save_column, join_table, join_column, display_column,
|
save_table, save_column, join_table, join_column, display_column,
|
||||||
|
|
@ -558,10 +559,10 @@ export const createFieldJoin = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면 필드 조인 수정
|
// 화면 필드 조인 수정
|
||||||
export const updateFieldJoin = async (req: Request, res: Response) => {
|
export const updateFieldJoin = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const {
|
const {
|
||||||
layout_id, component_id, field_name,
|
layout_id, component_id, field_name,
|
||||||
save_table, save_column, join_table, join_column, display_column,
|
save_table, save_column, join_table, join_column, display_column,
|
||||||
|
|
@ -603,10 +604,10 @@ export const updateFieldJoin = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면 필드 조인 삭제
|
// 화면 필드 조인 삭제
|
||||||
export const deleteFieldJoin = async (req: Request, res: Response) => {
|
export const deleteFieldJoin = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
|
|
||||||
let query = `DELETE FROM screen_field_joins WHERE id = $1`;
|
let query = `DELETE FROM screen_field_joins WHERE id = $1`;
|
||||||
const params: any[] = [id];
|
const params: any[] = [id];
|
||||||
|
|
@ -637,9 +638,9 @@ export const deleteFieldJoin = async (req: Request, res: Response) => {
|
||||||
// ============================================================
|
// ============================================================
|
||||||
|
|
||||||
// 데이터 흐름 목록 조회
|
// 데이터 흐름 목록 조회
|
||||||
export const getDataFlows = async (req: Request, res: Response) => {
|
export const getDataFlows = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const { group_id, source_screen_id } = req.query;
|
const { group_id, source_screen_id } = req.query;
|
||||||
|
|
||||||
let query = `
|
let query = `
|
||||||
|
|
@ -687,10 +688,10 @@ export const getDataFlows = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 데이터 흐름 생성
|
// 데이터 흐름 생성
|
||||||
export const createDataFlow = async (req: Request, res: Response) => {
|
export const createDataFlow = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const userId = (req.user as any).userId;
|
const userId = req.user?.userId || "";
|
||||||
const {
|
const {
|
||||||
group_id, source_screen_id, source_action, target_screen_id, target_action,
|
group_id, source_screen_id, source_action, target_screen_id, target_action,
|
||||||
data_mapping, flow_type, flow_label, condition_expression, is_active
|
data_mapping, flow_type, flow_label, condition_expression, is_active
|
||||||
|
|
@ -726,10 +727,10 @@ export const createDataFlow = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 데이터 흐름 수정
|
// 데이터 흐름 수정
|
||||||
export const updateDataFlow = async (req: Request, res: Response) => {
|
export const updateDataFlow = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const {
|
const {
|
||||||
group_id, source_screen_id, source_action, target_screen_id, target_action,
|
group_id, source_screen_id, source_action, target_screen_id, target_action,
|
||||||
data_mapping, flow_type, flow_label, condition_expression, is_active
|
data_mapping, flow_type, flow_label, condition_expression, is_active
|
||||||
|
|
@ -769,10 +770,10 @@ export const updateDataFlow = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 데이터 흐름 삭제
|
// 데이터 흐름 삭제
|
||||||
export const deleteDataFlow = async (req: Request, res: Response) => {
|
export const deleteDataFlow = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
|
|
||||||
let query = `DELETE FROM screen_data_flows WHERE id = $1`;
|
let query = `DELETE FROM screen_data_flows WHERE id = $1`;
|
||||||
const params: any[] = [id];
|
const params: any[] = [id];
|
||||||
|
|
@ -803,9 +804,9 @@ export const deleteDataFlow = async (req: Request, res: Response) => {
|
||||||
// ============================================================
|
// ============================================================
|
||||||
|
|
||||||
// 화면-테이블 관계 목록 조회
|
// 화면-테이블 관계 목록 조회
|
||||||
export const getTableRelations = async (req: Request, res: Response) => {
|
export const getTableRelations = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const { screen_id, group_id } = req.query;
|
const { screen_id, group_id } = req.query;
|
||||||
|
|
||||||
let query = `
|
let query = `
|
||||||
|
|
@ -852,10 +853,10 @@ export const getTableRelations = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면-테이블 관계 생성
|
// 화면-테이블 관계 생성
|
||||||
export const createTableRelation = async (req: Request, res: Response) => {
|
export const createTableRelation = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const userId = (req.user as any).userId;
|
const userId = req.user?.userId || "";
|
||||||
const { group_id, screen_id, table_name, relation_type, crud_operations, description, is_active } = req.body;
|
const { group_id, screen_id, table_name, relation_type, crud_operations, description, is_active } = req.body;
|
||||||
|
|
||||||
if (!screen_id || !table_name) {
|
if (!screen_id || !table_name) {
|
||||||
|
|
@ -885,10 +886,10 @@ export const createTableRelation = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면-테이블 관계 수정
|
// 화면-테이블 관계 수정
|
||||||
export const updateTableRelation = async (req: Request, res: Response) => {
|
export const updateTableRelation = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
const { group_id, table_name, relation_type, crud_operations, description, is_active } = req.body;
|
const { group_id, table_name, relation_type, crud_operations, description, is_active } = req.body;
|
||||||
|
|
||||||
let query = `
|
let query = `
|
||||||
|
|
@ -920,10 +921,10 @@ export const updateTableRelation = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 화면-테이블 관계 삭제
|
// 화면-테이블 관계 삭제
|
||||||
export const deleteTableRelation = async (req: Request, res: Response) => {
|
export const deleteTableRelation = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { id } = req.params;
|
const { id } = req.params;
|
||||||
const companyCode = (req.user as any).companyCode;
|
const companyCode = req.user?.companyCode || "*";
|
||||||
|
|
||||||
let query = `DELETE FROM screen_table_relations WHERE id = $1`;
|
let query = `DELETE FROM screen_table_relations WHERE id = $1`;
|
||||||
const params: any[] = [id];
|
const params: any[] = [id];
|
||||||
|
|
@ -953,7 +954,7 @@ export const deleteTableRelation = async (req: Request, res: Response) => {
|
||||||
// ============================================================
|
// ============================================================
|
||||||
|
|
||||||
// 화면 레이아웃 요약 조회 (위젯 타입별 개수, 라벨 목록)
|
// 화면 레이아웃 요약 조회 (위젯 타입별 개수, 라벨 목록)
|
||||||
export const getScreenLayoutSummary = async (req: Request, res: Response) => {
|
export const getScreenLayoutSummary = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { screenId } = req.params;
|
const { screenId } = req.params;
|
||||||
|
|
||||||
|
|
@ -1021,7 +1022,7 @@ export const getScreenLayoutSummary = async (req: Request, res: Response) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// 여러 화면의 레이아웃 요약 일괄 조회 (미니어처 렌더링용 좌표 포함)
|
// 여러 화면의 레이아웃 요약 일괄 조회 (미니어처 렌더링용 좌표 포함)
|
||||||
export const getMultipleScreenLayoutSummary = async (req: Request, res: Response) => {
|
export const getMultipleScreenLayoutSummary = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { screenIds } = req.body;
|
const { screenIds } = req.body;
|
||||||
|
|
||||||
|
|
@ -1221,7 +1222,7 @@ export const getMultipleScreenLayoutSummary = async (req: Request, res: Response
|
||||||
// ============================================================
|
// ============================================================
|
||||||
|
|
||||||
// 여러 화면의 서브 테이블 정보 조회 (메인 테이블 → 서브 테이블 관계)
|
// 여러 화면의 서브 테이블 정보 조회 (메인 테이블 → 서브 테이블 관계)
|
||||||
export const getScreenSubTables = async (req: Request, res: Response) => {
|
export const getScreenSubTables = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { screenIds } = req.body;
|
const { screenIds } = req.body;
|
||||||
|
|
||||||
|
|
@ -2060,10 +2061,10 @@ export const getScreenSubTables = async (req: Request, res: Response) => {
|
||||||
* 화면관리 → 메뉴 동기화
|
* 화면관리 → 메뉴 동기화
|
||||||
* screen_groups를 menu_info로 동기화
|
* screen_groups를 menu_info로 동기화
|
||||||
*/
|
*/
|
||||||
export const syncScreenGroupsToMenuController = async (req: Request, res: Response) => {
|
export const syncScreenGroupsToMenuController = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const userCompanyCode = (req.user as any).companyCode;
|
const userCompanyCode = req.user?.companyCode || "*";
|
||||||
const userId = (req.user as any).userId;
|
const userId = req.user?.userId || "";
|
||||||
const { targetCompanyCode } = req.body;
|
const { targetCompanyCode } = req.body;
|
||||||
|
|
||||||
// 최고 관리자가 특정 회사를 지정한 경우 해당 회사로
|
// 최고 관리자가 특정 회사를 지정한 경우 해당 회사로
|
||||||
|
|
@ -2111,10 +2112,10 @@ export const syncScreenGroupsToMenuController = async (req: Request, res: Respon
|
||||||
* 메뉴 → 화면관리 동기화
|
* 메뉴 → 화면관리 동기화
|
||||||
* menu_info를 screen_groups로 동기화
|
* menu_info를 screen_groups로 동기화
|
||||||
*/
|
*/
|
||||||
export const syncMenuToScreenGroupsController = async (req: Request, res: Response) => {
|
export const syncMenuToScreenGroupsController = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const userCompanyCode = (req.user as any).companyCode;
|
const userCompanyCode = req.user?.companyCode || "*";
|
||||||
const userId = (req.user as any).userId;
|
const userId = req.user?.userId || "";
|
||||||
const { targetCompanyCode } = req.body;
|
const { targetCompanyCode } = req.body;
|
||||||
|
|
||||||
// 최고 관리자가 특정 회사를 지정한 경우 해당 회사로
|
// 최고 관리자가 특정 회사를 지정한 경우 해당 회사로
|
||||||
|
|
@ -2161,9 +2162,9 @@ export const syncMenuToScreenGroupsController = async (req: Request, res: Respon
|
||||||
/**
|
/**
|
||||||
* 동기화 상태 조회
|
* 동기화 상태 조회
|
||||||
*/
|
*/
|
||||||
export const getSyncStatusController = async (req: Request, res: Response) => {
|
export const getSyncStatusController = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const userCompanyCode = (req.user as any).companyCode;
|
const userCompanyCode = req.user?.companyCode || "*";
|
||||||
const { targetCompanyCode } = req.query;
|
const { targetCompanyCode } = req.query;
|
||||||
|
|
||||||
// 최고 관리자가 특정 회사를 지정한 경우 해당 회사로
|
// 최고 관리자가 특정 회사를 지정한 경우 해당 회사로
|
||||||
|
|
@ -2200,10 +2201,10 @@ export const getSyncStatusController = async (req: Request, res: Response) => {
|
||||||
* 전체 회사 동기화
|
* 전체 회사 동기화
|
||||||
* 모든 회사에 대해 양방향 동기화 수행 (최고 관리자만)
|
* 모든 회사에 대해 양방향 동기화 수행 (최고 관리자만)
|
||||||
*/
|
*/
|
||||||
export const syncAllCompaniesController = async (req: Request, res: Response) => {
|
export const syncAllCompaniesController = async (req: AuthenticatedRequest, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const userCompanyCode = (req.user as any).companyCode;
|
const userCompanyCode = req.user?.companyCode || "*";
|
||||||
const userId = (req.user as any).userId;
|
const userId = req.user?.userId || "";
|
||||||
|
|
||||||
// 최고 관리자만 전체 동기화 가능
|
// 최고 관리자만 전체 동기화 가능
|
||||||
if (userCompanyCode !== "*") {
|
if (userCompanyCode !== "*") {
|
||||||
|
|
|
||||||
|
|
@ -2090,7 +2090,7 @@ export class MenuCopyService {
|
||||||
menu.menu_url,
|
menu.menu_url,
|
||||||
menu.menu_desc,
|
menu.menu_desc,
|
||||||
userId,
|
userId,
|
||||||
menu.status,
|
'active', // 복제된 메뉴는 항상 활성화 상태
|
||||||
menu.system_name,
|
menu.system_name,
|
||||||
targetCompanyCode, // 새 회사 코드
|
targetCompanyCode, // 새 회사 코드
|
||||||
menu.lang_key,
|
menu.lang_key,
|
||||||
|
|
|
||||||
|
|
@ -142,7 +142,7 @@ export async function syncScreenGroupsToMenu(
|
||||||
const newObjid = Date.now();
|
const newObjid = Date.now();
|
||||||
const createRootQuery = `
|
const createRootQuery = `
|
||||||
INSERT INTO menu_info (objid, parent_obj_id, menu_name_kor, menu_name_eng, seq, menu_type, company_code, writer, regdate, status)
|
INSERT INTO menu_info (objid, parent_obj_id, menu_name_kor, menu_name_eng, seq, menu_type, company_code, writer, regdate, status)
|
||||||
VALUES ($1, 0, '사용자', 'User', 1, 1, $2, $3, NOW(), 'Y')
|
VALUES ($1, 0, '사용자', 'User', 1, 1, $2, $3, NOW(), 'active')
|
||||||
RETURNING objid
|
RETURNING objid
|
||||||
`;
|
`;
|
||||||
const createRootResult = await client.query(createRootQuery, [newObjid, companyCode, userId]);
|
const createRootResult = await client.query(createRootQuery, [newObjid, companyCode, userId]);
|
||||||
|
|
@ -159,12 +159,36 @@ export async function syncScreenGroupsToMenu(
|
||||||
groupIdToName.set(g.id, g.group_name?.trim().toLowerCase() || '');
|
groupIdToName.set(g.id, g.group_name?.trim().toLowerCase() || '');
|
||||||
});
|
});
|
||||||
|
|
||||||
// 5. 각 screen_group 처리
|
// 5. 최상위 회사 폴더 ID 찾기 (level 0, parent_group_id IS NULL)
|
||||||
|
// 이 폴더는 메뉴로 생성하지 않고, 하위 폴더들을 사용자 루트 바로 아래에 배치
|
||||||
|
const topLevelCompanyFolderIds = new Set<number>();
|
||||||
|
for (const group of screenGroupsResult.rows) {
|
||||||
|
if (group.group_level === 0 && group.parent_group_id === null) {
|
||||||
|
topLevelCompanyFolderIds.add(group.id);
|
||||||
|
// 최상위 폴더 → 사용자 루트에 매핑 (하위 폴더의 부모로 사용)
|
||||||
|
groupToMenuMap.set(group.id, userMenuRootObjid!);
|
||||||
|
logger.info("최상위 회사 폴더 스킵", { groupId: group.id, groupName: group.group_name });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 6. 각 screen_group 처리
|
||||||
for (const group of screenGroupsResult.rows) {
|
for (const group of screenGroupsResult.rows) {
|
||||||
const groupId = group.id;
|
const groupId = group.id;
|
||||||
const groupName = group.group_name?.trim();
|
const groupName = group.group_name?.trim();
|
||||||
const groupNameLower = groupName?.toLowerCase() || '';
|
const groupNameLower = groupName?.toLowerCase() || '';
|
||||||
|
|
||||||
|
// 최상위 회사 폴더는 메뉴로 생성하지 않고 스킵
|
||||||
|
if (topLevelCompanyFolderIds.has(groupId)) {
|
||||||
|
result.skipped++;
|
||||||
|
result.details.push({
|
||||||
|
action: 'skipped',
|
||||||
|
sourceName: groupName,
|
||||||
|
sourceId: groupId,
|
||||||
|
reason: '최상위 회사 폴더 (메뉴 생성 스킵)',
|
||||||
|
});
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// 이미 연결된 경우 - 실제로 메뉴가 존재하는지 확인
|
// 이미 연결된 경우 - 실제로 메뉴가 존재하는지 확인
|
||||||
if (group.menu_objid) {
|
if (group.menu_objid) {
|
||||||
const menuExists = existingMenuObjids.has(Number(group.menu_objid));
|
const menuExists = existingMenuObjids.has(Number(group.menu_objid));
|
||||||
|
|
@ -237,11 +261,17 @@ export async function syncScreenGroupsToMenu(
|
||||||
const newObjid = Date.now() + groupId; // 고유 ID 보장
|
const newObjid = Date.now() + groupId; // 고유 ID 보장
|
||||||
|
|
||||||
// 부모 메뉴 objid 결정
|
// 부모 메뉴 objid 결정
|
||||||
|
// 우선순위: groupToMenuMap > parent_menu_objid (존재 확인 필수)
|
||||||
let parentMenuObjid = userMenuRootObjid;
|
let parentMenuObjid = userMenuRootObjid;
|
||||||
if (group.parent_group_id && group.parent_menu_objid) {
|
if (group.parent_group_id && groupToMenuMap.has(group.parent_group_id)) {
|
||||||
parentMenuObjid = Number(group.parent_menu_objid);
|
// 현재 트랜잭션에서 생성된 부모 메뉴 사용
|
||||||
} else if (group.parent_group_id && groupToMenuMap.has(group.parent_group_id)) {
|
|
||||||
parentMenuObjid = groupToMenuMap.get(group.parent_group_id)!;
|
parentMenuObjid = groupToMenuMap.get(group.parent_group_id)!;
|
||||||
|
} else if (group.parent_group_id && group.parent_menu_objid) {
|
||||||
|
// 기존 parent_menu_objid가 실제로 존재하는지 확인
|
||||||
|
const parentMenuExists = existingMenuObjids.has(Number(group.parent_menu_objid));
|
||||||
|
if (parentMenuExists) {
|
||||||
|
parentMenuObjid = Number(group.parent_menu_objid);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 같은 부모 아래에서 가장 높은 seq 조회 후 +1
|
// 같은 부모 아래에서 가장 높은 seq 조회 후 +1
|
||||||
|
|
@ -261,7 +291,7 @@ export async function syncScreenGroupsToMenu(
|
||||||
INSERT INTO menu_info (
|
INSERT INTO menu_info (
|
||||||
objid, parent_obj_id, menu_name_kor, menu_name_eng,
|
objid, parent_obj_id, menu_name_kor, menu_name_eng,
|
||||||
seq, menu_type, company_code, writer, regdate, status, screen_group_id, menu_desc
|
seq, menu_type, company_code, writer, regdate, status, screen_group_id, menu_desc
|
||||||
) VALUES ($1, $2, $3, $4, $5, 1, $6, $7, NOW(), 'Y', $8, $9)
|
) VALUES ($1, $2, $3, $4, $5, 1, $6, $7, NOW(), 'active', $8, $9)
|
||||||
RETURNING objid
|
RETURNING objid
|
||||||
`;
|
`;
|
||||||
await client.query(insertMenuQuery, [
|
await client.query(insertMenuQuery, [
|
||||||
|
|
|
||||||
|
|
@ -1323,17 +1323,24 @@ export class TableManagementService {
|
||||||
// - "2," 로 시작
|
// - "2," 로 시작
|
||||||
// - ",2" 로 끝남
|
// - ",2" 로 끝남
|
||||||
// - ",2," 중간에 포함
|
// - ",2," 중간에 포함
|
||||||
const paramBase = paramIndex + (idx * 4);
|
const paramBase = paramIndex + idx * 4;
|
||||||
conditions.push(`(
|
conditions.push(`(
|
||||||
${columnName}::text = $${paramBase} OR
|
${columnName}::text = $${paramBase} OR
|
||||||
${columnName}::text LIKE $${paramBase + 1} OR
|
${columnName}::text LIKE $${paramBase + 1} OR
|
||||||
${columnName}::text LIKE $${paramBase + 2} OR
|
${columnName}::text LIKE $${paramBase + 2} OR
|
||||||
${columnName}::text LIKE $${paramBase + 3}
|
${columnName}::text LIKE $${paramBase + 3}
|
||||||
)`);
|
)`);
|
||||||
values.push(safeValue, `${safeValue},%`, `%,${safeValue}`, `%,${safeValue},%`);
|
values.push(
|
||||||
|
safeValue,
|
||||||
|
`${safeValue},%`,
|
||||||
|
`%,${safeValue}`,
|
||||||
|
`%,${safeValue},%`
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
logger.info(`🔍 다중 값 배열 검색: ${columnName} IN [${value.join(", ")}]`);
|
logger.info(
|
||||||
|
`🔍 다중 값 배열 검색: ${columnName} IN [${value.join(", ")}]`
|
||||||
|
);
|
||||||
return {
|
return {
|
||||||
whereClause: `(${conditions.join(" OR ")})`,
|
whereClause: `(${conditions.join(" OR ")})`,
|
||||||
values,
|
values,
|
||||||
|
|
@ -1775,18 +1782,26 @@ export class TableManagementService {
|
||||||
|
|
||||||
// displayColumn이 비어있거나 "none"이면 참조 테이블에서 자동 감지 (entityJoinService와 동일한 로직)
|
// displayColumn이 비어있거나 "none"이면 참조 테이블에서 자동 감지 (entityJoinService와 동일한 로직)
|
||||||
let displayColumn = entityTypeInfo.displayColumn;
|
let displayColumn = entityTypeInfo.displayColumn;
|
||||||
if (!displayColumn || displayColumn === "none" || displayColumn === "") {
|
if (
|
||||||
displayColumn = await this.findDisplayColumnForTable(referenceTable, referenceColumn);
|
!displayColumn ||
|
||||||
|
displayColumn === "none" ||
|
||||||
|
displayColumn === ""
|
||||||
|
) {
|
||||||
|
displayColumn = await this.findDisplayColumnForTable(
|
||||||
|
referenceTable,
|
||||||
|
referenceColumn
|
||||||
|
);
|
||||||
logger.info(
|
logger.info(
|
||||||
`🔍 [buildEntitySearchCondition] displayColumn 자동 감지: ${referenceTable} -> ${displayColumn}`
|
`🔍 [buildEntitySearchCondition] displayColumn 자동 감지: ${referenceTable} -> ${displayColumn}`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 참조 테이블의 표시 컬럼으로 검색
|
// 참조 테이블의 표시 컬럼으로 검색
|
||||||
|
// 🔧 main. 접두사 추가: EXISTS 서브쿼리에서 외부 테이블 참조 시 명시적으로 지정
|
||||||
return {
|
return {
|
||||||
whereClause: `EXISTS (
|
whereClause: `EXISTS (
|
||||||
SELECT 1 FROM ${referenceTable} ref
|
SELECT 1 FROM ${referenceTable} ref
|
||||||
WHERE ref.${referenceColumn} = ${columnName}
|
WHERE ref.${referenceColumn} = main.${columnName}
|
||||||
AND ref.${displayColumn} ILIKE $${paramIndex}
|
AND ref.${displayColumn} ILIKE $${paramIndex}
|
||||||
)`,
|
)`,
|
||||||
values: [`%${value}%`],
|
values: [`%${value}%`],
|
||||||
|
|
@ -2150,14 +2165,14 @@ export class TableManagementService {
|
||||||
// 안전한 테이블명 검증
|
// 안전한 테이블명 검증
|
||||||
const safeTableName = tableName.replace(/[^a-zA-Z0-9_]/g, "");
|
const safeTableName = tableName.replace(/[^a-zA-Z0-9_]/g, "");
|
||||||
|
|
||||||
// 전체 개수 조회
|
// 전체 개수 조회 (main 별칭 추가 - buildWhereClause가 main. 접두사를 사용하므로 필요)
|
||||||
const countQuery = `SELECT COUNT(*) as count FROM ${safeTableName} ${whereClause}`;
|
const countQuery = `SELECT COUNT(*) as count FROM ${safeTableName} main ${whereClause}`;
|
||||||
const countResult = await query<any>(countQuery, searchValues);
|
const countResult = await query<any>(countQuery, searchValues);
|
||||||
const total = parseInt(countResult[0].count);
|
const total = parseInt(countResult[0].count);
|
||||||
|
|
||||||
// 데이터 조회
|
// 데이터 조회 (main 별칭 추가)
|
||||||
const dataQuery = `
|
const dataQuery = `
|
||||||
SELECT * FROM ${safeTableName}
|
SELECT main.* FROM ${safeTableName} main
|
||||||
${whereClause}
|
${whereClause}
|
||||||
${orderClause}
|
${orderClause}
|
||||||
LIMIT $${paramIndex} OFFSET $${paramIndex + 1}
|
LIMIT $${paramIndex} OFFSET $${paramIndex + 1}
|
||||||
|
|
@ -2506,7 +2521,9 @@ export class TableManagementService {
|
||||||
});
|
});
|
||||||
|
|
||||||
if (skippedColumns.length > 0) {
|
if (skippedColumns.length > 0) {
|
||||||
logger.info(`⚠️ 테이블에 존재하지 않는 컬럼 스킵: ${skippedColumns.join(", ")}`);
|
logger.info(
|
||||||
|
`⚠️ 테이블에 존재하지 않는 컬럼 스킵: ${skippedColumns.join(", ")}`
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// WHERE 조건 생성 (PRIMARY KEY 우선, 없으면 모든 원본 데이터 사용)
|
// WHERE 조건 생성 (PRIMARY KEY 우선, 없으면 모든 원본 데이터 사용)
|
||||||
|
|
@ -2776,10 +2793,14 @@ export class TableManagementService {
|
||||||
// 실제 소스 컬럼이 partner_id인데 프론트엔드가 customer_id로 추론하는 경우 대응
|
// 실제 소스 컬럼이 partner_id인데 프론트엔드가 customer_id로 추론하는 경우 대응
|
||||||
if (!baseJoinConfig && (additionalColumn as any).referenceTable) {
|
if (!baseJoinConfig && (additionalColumn as any).referenceTable) {
|
||||||
baseJoinConfig = joinConfigs.find(
|
baseJoinConfig = joinConfigs.find(
|
||||||
(config) => config.referenceTable === (additionalColumn as any).referenceTable
|
(config) =>
|
||||||
|
config.referenceTable ===
|
||||||
|
(additionalColumn as any).referenceTable
|
||||||
);
|
);
|
||||||
if (baseJoinConfig) {
|
if (baseJoinConfig) {
|
||||||
logger.info(`🔄 referenceTable로 조인 설정 찾음: ${(additionalColumn as any).referenceTable} → ${baseJoinConfig.sourceColumn}`);
|
logger.info(
|
||||||
|
`🔄 referenceTable로 조인 설정 찾음: ${(additionalColumn as any).referenceTable} → ${baseJoinConfig.sourceColumn}`
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -2797,10 +2818,16 @@ export class TableManagementService {
|
||||||
const frontendSourceColumn = additionalColumn.sourceColumn; // 프론트엔드가 추론한 소스 컬럼 (customer_id)
|
const frontendSourceColumn = additionalColumn.sourceColumn; // 프론트엔드가 추론한 소스 컬럼 (customer_id)
|
||||||
if (originalJoinAlias.startsWith(`${frontendSourceColumn}_`)) {
|
if (originalJoinAlias.startsWith(`${frontendSourceColumn}_`)) {
|
||||||
// 프론트엔드가 추론한 소스 컬럼으로 시작하면 그 부분 제거
|
// 프론트엔드가 추론한 소스 컬럼으로 시작하면 그 부분 제거
|
||||||
actualColumnName = originalJoinAlias.replace(`${frontendSourceColumn}_`, "");
|
actualColumnName = originalJoinAlias.replace(
|
||||||
|
`${frontendSourceColumn}_`,
|
||||||
|
""
|
||||||
|
);
|
||||||
} else if (originalJoinAlias.startsWith(`${sourceColumn}_`)) {
|
} else if (originalJoinAlias.startsWith(`${sourceColumn}_`)) {
|
||||||
// 실제 소스 컬럼으로 시작하면 그 부분 제거
|
// 실제 소스 컬럼으로 시작하면 그 부분 제거
|
||||||
actualColumnName = originalJoinAlias.replace(`${sourceColumn}_`, "");
|
actualColumnName = originalJoinAlias.replace(
|
||||||
|
`${sourceColumn}_`,
|
||||||
|
""
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
// 어느 것도 아니면 원본 사용
|
// 어느 것도 아니면 원본 사용
|
||||||
actualColumnName = originalJoinAlias;
|
actualColumnName = originalJoinAlias;
|
||||||
|
|
@ -3199,8 +3226,10 @@ export class TableManagementService {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Entity 조인 컬럼 검색이 있는지 확인 (기본 조인 + 추가 조인 컬럼 모두 포함)
|
// Entity 조인 컬럼 검색이 있는지 확인 (기본 조인 + 추가 조인 컬럼 모두 포함)
|
||||||
|
// 🔧 sourceColumn도 포함: search={"order_no":"..."} 형태도 Entity 검색으로 인식
|
||||||
const allEntityColumns = [
|
const allEntityColumns = [
|
||||||
...joinConfigs.map((config) => config.aliasColumn),
|
...joinConfigs.map((config) => config.aliasColumn),
|
||||||
|
...joinConfigs.map((config) => config.sourceColumn), // 🔧 소스 컬럼도 포함
|
||||||
// 추가 조인 컬럼들도 포함 (writer_dept_code, company_code_status 등)
|
// 추가 조인 컬럼들도 포함 (writer_dept_code, company_code_status 등)
|
||||||
...joinConfigs.flatMap((config) => {
|
...joinConfigs.flatMap((config) => {
|
||||||
const additionalColumns = [];
|
const additionalColumns = [];
|
||||||
|
|
@ -3606,8 +3635,10 @@ export class TableManagementService {
|
||||||
});
|
});
|
||||||
|
|
||||||
// main. 접두사 추가 (조인 쿼리용)
|
// main. 접두사 추가 (조인 쿼리용)
|
||||||
|
// 🔧 이미 접두사(. 앞)가 있는 경우는 교체하지 않음 (ref.column, main.column 등)
|
||||||
|
// Negative lookbehind (?<!\.) 사용: 앞에 .이 없는 경우만 매칭
|
||||||
condition = condition.replace(
|
condition = condition.replace(
|
||||||
new RegExp(`\\b${columnName}\\b`, "g"),
|
new RegExp(`(?<!\\.)\\b${columnName}\\b`, "g"),
|
||||||
`main.${columnName}`
|
`main.${columnName}`
|
||||||
);
|
);
|
||||||
conditions.push(condition);
|
conditions.push(condition);
|
||||||
|
|
@ -3812,6 +3843,9 @@ export class TableManagementService {
|
||||||
"customer_mng",
|
"customer_mng",
|
||||||
"item_info",
|
"item_info",
|
||||||
"dept_info",
|
"dept_info",
|
||||||
|
"sales_order_mng", // 🔧 수주관리 테이블 추가
|
||||||
|
"sales_order_detail", // 🔧 수주상세 테이블 추가
|
||||||
|
"partner_info", // 🔧 거래처 테이블 추가
|
||||||
// 필요시 추가
|
// 필요시 추가
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|
@ -4730,15 +4764,19 @@ export class TableManagementService {
|
||||||
async detectTableEntityRelations(
|
async detectTableEntityRelations(
|
||||||
leftTable: string,
|
leftTable: string,
|
||||||
rightTable: string
|
rightTable: string
|
||||||
): Promise<Array<{
|
): Promise<
|
||||||
|
Array<{
|
||||||
leftColumn: string;
|
leftColumn: string;
|
||||||
rightColumn: string;
|
rightColumn: string;
|
||||||
direction: "left_to_right" | "right_to_left";
|
direction: "left_to_right" | "right_to_left";
|
||||||
inputType: string;
|
inputType: string;
|
||||||
displayColumn?: string;
|
displayColumn?: string;
|
||||||
}>> {
|
}>
|
||||||
|
> {
|
||||||
try {
|
try {
|
||||||
logger.info(`두 테이블 간 엔티티 관계 감지 시작: ${leftTable} <-> ${rightTable}`);
|
logger.info(
|
||||||
|
`두 테이블 간 엔티티 관계 감지 시작: ${leftTable} <-> ${rightTable}`
|
||||||
|
);
|
||||||
|
|
||||||
const relations: Array<{
|
const relations: Array<{
|
||||||
leftColumn: string;
|
leftColumn: string;
|
||||||
|
|
@ -4806,12 +4844,17 @@ export class TableManagementService {
|
||||||
|
|
||||||
logger.info(`엔티티 관계 감지 완료: ${relations.length}개 발견`);
|
logger.info(`엔티티 관계 감지 완료: ${relations.length}개 발견`);
|
||||||
relations.forEach((rel, idx) => {
|
relations.forEach((rel, idx) => {
|
||||||
logger.info(` ${idx + 1}. ${leftTable}.${rel.leftColumn} <-> ${rightTable}.${rel.rightColumn} (${rel.direction})`);
|
logger.info(
|
||||||
|
` ${idx + 1}. ${leftTable}.${rel.leftColumn} <-> ${rightTable}.${rel.rightColumn} (${rel.direction})`
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
return relations;
|
return relations;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error(`엔티티 관계 감지 실패: ${leftTable} <-> ${rightTable}`, error);
|
logger.error(
|
||||||
|
`엔티티 관계 감지 실패: ${leftTable} <-> ${rightTable}`,
|
||||||
|
error
|
||||||
|
);
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -172,6 +172,7 @@ export function ScreenGroupTreeView({
|
||||||
const [syncStatus, setSyncStatus] = useState<SyncStatus | null>(null);
|
const [syncStatus, setSyncStatus] = useState<SyncStatus | null>(null);
|
||||||
const [isSyncing, setIsSyncing] = useState(false);
|
const [isSyncing, setIsSyncing] = useState(false);
|
||||||
const [syncDirection, setSyncDirection] = useState<"screen-to-menu" | "menu-to-screen" | "all" | null>(null);
|
const [syncDirection, setSyncDirection] = useState<"screen-to-menu" | "menu-to-screen" | "all" | null>(null);
|
||||||
|
const [syncProgress, setSyncProgress] = useState<{ message: string; detail?: string } | null>(null);
|
||||||
|
|
||||||
// 회사 선택 (최고 관리자용)
|
// 회사 선택 (최고 관리자용)
|
||||||
const { user } = useAuth();
|
const { user } = useAuth();
|
||||||
|
|
@ -328,14 +329,31 @@ export function ScreenGroupTreeView({
|
||||||
|
|
||||||
setIsSyncing(true);
|
setIsSyncing(true);
|
||||||
setSyncDirection(direction);
|
setSyncDirection(direction);
|
||||||
|
setSyncProgress({
|
||||||
|
message: direction === "screen-to-menu"
|
||||||
|
? "화면관리 → 메뉴 동기화 중..."
|
||||||
|
: "메뉴 → 화면관리 동기화 중...",
|
||||||
|
detail: "데이터를 분석하고 있습니다..."
|
||||||
|
});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
setSyncProgress({
|
||||||
|
message: direction === "screen-to-menu"
|
||||||
|
? "화면관리 → 메뉴 동기화 중..."
|
||||||
|
: "메뉴 → 화면관리 동기화 중...",
|
||||||
|
detail: "동기화 작업을 수행하고 있습니다..."
|
||||||
|
});
|
||||||
|
|
||||||
const response = direction === "screen-to-menu"
|
const response = direction === "screen-to-menu"
|
||||||
? await syncScreenGroupsToMenu(targetCompanyCode)
|
? await syncScreenGroupsToMenu(targetCompanyCode)
|
||||||
: await syncMenuToScreenGroups(targetCompanyCode);
|
: await syncMenuToScreenGroups(targetCompanyCode);
|
||||||
|
|
||||||
if (response.success) {
|
if (response.success) {
|
||||||
const data = response.data;
|
const data = response.data;
|
||||||
|
setSyncProgress({
|
||||||
|
message: "동기화 완료!",
|
||||||
|
detail: `생성 ${data?.created || 0}개, 연결 ${data?.linked || 0}개, 스킵 ${data?.skipped || 0}개`
|
||||||
|
});
|
||||||
toast.success(
|
toast.success(
|
||||||
`동기화 완료: 생성 ${data?.created || 0}개, 연결 ${data?.linked || 0}개, 스킵 ${data?.skipped || 0}개`
|
`동기화 완료: 생성 ${data?.created || 0}개, 연결 ${data?.linked || 0}개, 스킵 ${data?.skipped || 0}개`
|
||||||
);
|
);
|
||||||
|
|
@ -347,13 +365,17 @@ export function ScreenGroupTreeView({
|
||||||
setSyncStatus(statusResponse.data);
|
setSyncStatus(statusResponse.data);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
setSyncProgress(null);
|
||||||
toast.error(`동기화 실패: ${response.error || "알 수 없는 오류"}`);
|
toast.error(`동기화 실패: ${response.error || "알 수 없는 오류"}`);
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
|
setSyncProgress(null);
|
||||||
toast.error(`동기화 실패: ${error.message}`);
|
toast.error(`동기화 실패: ${error.message}`);
|
||||||
} finally {
|
} finally {
|
||||||
setIsSyncing(false);
|
setIsSyncing(false);
|
||||||
setSyncDirection(null);
|
setSyncDirection(null);
|
||||||
|
// 3초 후 진행 메시지 초기화
|
||||||
|
setTimeout(() => setSyncProgress(null), 3000);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -366,27 +388,42 @@ export function ScreenGroupTreeView({
|
||||||
|
|
||||||
setIsSyncing(true);
|
setIsSyncing(true);
|
||||||
setSyncDirection("all");
|
setSyncDirection("all");
|
||||||
|
setSyncProgress({
|
||||||
|
message: "전체 회사 동기화 중...",
|
||||||
|
detail: "모든 회사의 데이터를 분석하고 있습니다..."
|
||||||
|
});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
setSyncProgress({
|
||||||
|
message: "전체 회사 동기화 중...",
|
||||||
|
detail: "양방향 동기화 작업을 수행하고 있습니다..."
|
||||||
|
});
|
||||||
|
|
||||||
const response = await syncAllCompanies();
|
const response = await syncAllCompanies();
|
||||||
|
|
||||||
if (response.success && response.data) {
|
if (response.success && response.data) {
|
||||||
const data = response.data;
|
const data = response.data;
|
||||||
|
setSyncProgress({
|
||||||
|
message: "전체 동기화 완료!",
|
||||||
|
detail: `${data.totalCompanies}개 회사, 생성 ${data.totalCreated}개, 연결 ${data.totalLinked}개`
|
||||||
|
});
|
||||||
toast.success(
|
toast.success(
|
||||||
`전체 동기화 완료: ${data.totalCompanies}개 회사, 생성 ${data.totalCreated}개, 연결 ${data.totalLinked}개`
|
`전체 동기화 완료: ${data.totalCompanies}개 회사, 생성 ${data.totalCreated}개, 연결 ${data.totalLinked}개`
|
||||||
);
|
);
|
||||||
// 그룹 데이터 새로고침
|
// 그룹 데이터 새로고침
|
||||||
await loadGroupsData();
|
await loadGroupsData();
|
||||||
// 동기화 다이얼로그 닫기
|
|
||||||
setIsSyncDialogOpen(false);
|
|
||||||
} else {
|
} else {
|
||||||
|
setSyncProgress(null);
|
||||||
toast.error(`전체 동기화 실패: ${response.error || "알 수 없는 오류"}`);
|
toast.error(`전체 동기화 실패: ${response.error || "알 수 없는 오류"}`);
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
|
setSyncProgress(null);
|
||||||
toast.error(`전체 동기화 실패: ${error.message}`);
|
toast.error(`전체 동기화 실패: ${error.message}`);
|
||||||
} finally {
|
} finally {
|
||||||
setIsSyncing(false);
|
setIsSyncing(false);
|
||||||
setSyncDirection(null);
|
setSyncDirection(null);
|
||||||
|
// 3초 후 진행 메시지 초기화
|
||||||
|
setTimeout(() => setSyncProgress(null), 3000);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -979,6 +1016,7 @@ export function ScreenGroupTreeView({
|
||||||
<Plus className="h-4 w-4" />
|
<Plus className="h-4 w-4" />
|
||||||
그룹 추가
|
그룹 추가
|
||||||
</Button>
|
</Button>
|
||||||
|
{isSuperAdmin && (
|
||||||
<Button
|
<Button
|
||||||
onClick={handleOpenSyncDialog}
|
onClick={handleOpenSyncDialog}
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
|
|
@ -988,6 +1026,7 @@ export function ScreenGroupTreeView({
|
||||||
<RefreshCw className="h-4 w-4" />
|
<RefreshCw className="h-4 w-4" />
|
||||||
메뉴 동기화
|
메뉴 동기화
|
||||||
</Button>
|
</Button>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* 트리 목록 */}
|
{/* 트리 목록 */}
|
||||||
|
|
@ -1816,7 +1855,23 @@ export function ScreenGroupTreeView({
|
||||||
|
|
||||||
{/* 메뉴-화면그룹 동기화 다이얼로그 */}
|
{/* 메뉴-화면그룹 동기화 다이얼로그 */}
|
||||||
<Dialog open={isSyncDialogOpen} onOpenChange={setIsSyncDialogOpen}>
|
<Dialog open={isSyncDialogOpen} onOpenChange={setIsSyncDialogOpen}>
|
||||||
<DialogContent className="max-w-[95vw] sm:max-w-[500px]">
|
<DialogContent className="max-w-[95vw] sm:max-w-[500px] overflow-hidden">
|
||||||
|
{/* 동기화 진행 중 오버레이 (삭제와 동일한 스타일) */}
|
||||||
|
{isSyncing && (
|
||||||
|
<div className="absolute inset-0 z-50 flex flex-col items-center justify-center rounded-lg bg-background/90 backdrop-blur-sm">
|
||||||
|
<Loader2 className="h-10 w-10 animate-spin text-primary" />
|
||||||
|
<p className="mt-4 text-sm font-medium">{syncProgress?.message || "동기화 중..."}</p>
|
||||||
|
{syncProgress?.detail && (
|
||||||
|
<p className="mt-1 text-xs text-muted-foreground">{syncProgress.detail}</p>
|
||||||
|
)}
|
||||||
|
<div className="mt-3 h-2 w-48 overflow-hidden rounded-full bg-secondary">
|
||||||
|
<div
|
||||||
|
className="h-full bg-primary animate-pulse"
|
||||||
|
style={{ width: "100%" }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
<DialogHeader>
|
<DialogHeader>
|
||||||
<DialogTitle className="text-base sm:text-lg">메뉴-화면 동기화</DialogTitle>
|
<DialogTitle className="text-base sm:text-lg">메뉴-화면 동기화</DialogTitle>
|
||||||
<DialogDescription className="text-xs sm:text-sm">
|
<DialogDescription className="text-xs sm:text-sm">
|
||||||
|
|
|
||||||
|
|
@ -296,24 +296,6 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
onFieldDrop,
|
onFieldDrop,
|
||||||
onExpandChange,
|
onExpandChange,
|
||||||
}) => {
|
}) => {
|
||||||
// 디버깅 로그
|
|
||||||
console.log("🔶 PivotGridComponent props:", {
|
|
||||||
title,
|
|
||||||
hasExternalData: !!externalData,
|
|
||||||
externalDataLength: externalData?.length,
|
|
||||||
initialFieldsLength: initialFields?.length,
|
|
||||||
});
|
|
||||||
|
|
||||||
// 🆕 데이터 샘플 확인
|
|
||||||
if (externalData && externalData.length > 0) {
|
|
||||||
console.log("🔶 첫 번째 데이터 샘플:", externalData[0]);
|
|
||||||
console.log("🔶 전체 데이터 개수:", externalData.length);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 🆕 필드 설정 확인
|
|
||||||
if (initialFields && initialFields.length > 0) {
|
|
||||||
console.log("🔶 필드 설정:", initialFields);
|
|
||||||
}
|
|
||||||
// ==================== 상태 ====================
|
// ==================== 상태 ====================
|
||||||
|
|
||||||
const [fields, setFields] = useState<PivotFieldConfig[]>(initialFields);
|
const [fields, setFields] = useState<PivotFieldConfig[]>(initialFields);
|
||||||
|
|
@ -384,20 +366,63 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
localStorage.setItem(stateStorageKey, JSON.stringify(stateToSave));
|
localStorage.setItem(stateStorageKey, JSON.stringify(stateToSave));
|
||||||
}, [fields, pivotState, sortConfig, columnWidths, stateStorageKey]);
|
}, [fields, pivotState, sortConfig, columnWidths, stateStorageKey]);
|
||||||
|
|
||||||
// 상태 복원 (localStorage)
|
// 상태 복원 (localStorage) - 프로덕션 안전성 강화
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (typeof window === "undefined") return;
|
if (typeof window === "undefined") return;
|
||||||
const savedState = localStorage.getItem(stateStorageKey);
|
|
||||||
if (savedState) {
|
|
||||||
try {
|
try {
|
||||||
|
const savedState = localStorage.getItem(stateStorageKey);
|
||||||
|
if (!savedState) return;
|
||||||
|
|
||||||
const parsed = JSON.parse(savedState);
|
const parsed = JSON.parse(savedState);
|
||||||
if (parsed.fields) setFields(parsed.fields);
|
|
||||||
if (parsed.pivotState) setPivotState(parsed.pivotState);
|
// 버전 체크 - 버전이 다르면 이전 상태 무시
|
||||||
if (parsed.sortConfig) setSortConfig(parsed.sortConfig);
|
if (parsed.version !== PIVOT_STATE_VERSION) {
|
||||||
if (parsed.columnWidths) setColumnWidths(parsed.columnWidths);
|
localStorage.removeItem(stateStorageKey);
|
||||||
} catch (e) {
|
return;
|
||||||
console.warn("피벗 상태 복원 실패:", e);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 필드 복원 시 유효성 검사 (중요!)
|
||||||
|
if (parsed.fields && Array.isArray(parsed.fields) && parsed.fields.length > 0) {
|
||||||
|
// 저장된 필드가 현재 데이터와 호환되는지 확인
|
||||||
|
const validFields = parsed.fields.filter((f: PivotFieldConfig) =>
|
||||||
|
f && typeof f.field === "string" && typeof f.area === "string"
|
||||||
|
);
|
||||||
|
|
||||||
|
if (validFields.length > 0) {
|
||||||
|
setFields(validFields);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// pivotState 복원 시 유효성 검사 (확장 경로 검증)
|
||||||
|
if (parsed.pivotState && typeof parsed.pivotState === "object") {
|
||||||
|
const restoredState: PivotGridState = {
|
||||||
|
// expandedRowPaths는 배열의 배열이어야 함
|
||||||
|
expandedRowPaths: Array.isArray(parsed.pivotState.expandedRowPaths)
|
||||||
|
? parsed.pivotState.expandedRowPaths.filter(
|
||||||
|
(p: unknown) => Array.isArray(p) && p.every(item => typeof item === "string")
|
||||||
|
)
|
||||||
|
: [],
|
||||||
|
// expandedColumnPaths도 동일하게 검증
|
||||||
|
expandedColumnPaths: Array.isArray(parsed.pivotState.expandedColumnPaths)
|
||||||
|
? parsed.pivotState.expandedColumnPaths.filter(
|
||||||
|
(p: unknown) => Array.isArray(p) && p.every(item => typeof item === "string")
|
||||||
|
)
|
||||||
|
: [],
|
||||||
|
sortConfig: parsed.pivotState.sortConfig || null,
|
||||||
|
filterConfig: parsed.pivotState.filterConfig || {},
|
||||||
|
};
|
||||||
|
setPivotState(restoredState);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parsed.sortConfig) setSortConfig(parsed.sortConfig);
|
||||||
|
if (parsed.columnWidths && typeof parsed.columnWidths === "object") {
|
||||||
|
setColumnWidths(parsed.columnWidths);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.warn("피벗 상태 복원 실패, localStorage 초기화:", e);
|
||||||
|
// 손상된 상태는 제거
|
||||||
|
localStorage.removeItem(stateStorageKey);
|
||||||
}
|
}
|
||||||
}, [stateStorageKey]);
|
}, [stateStorageKey]);
|
||||||
|
|
||||||
|
|
@ -432,10 +457,12 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
|
|
||||||
// 필터 영역 필드
|
// 필터 영역 필드
|
||||||
const filterFields = useMemo(
|
const filterFields = useMemo(
|
||||||
() =>
|
() => {
|
||||||
fields
|
const result = fields
|
||||||
.filter((f) => f.area === "filter" && f.visible !== false)
|
.filter((f) => f.area === "filter" && f.visible !== false)
|
||||||
.sort((a, b) => (a.areaIndex || 0) - (b.areaIndex || 0)),
|
.sort((a, b) => (a.areaIndex || 0) - (b.areaIndex || 0));
|
||||||
|
return result;
|
||||||
|
},
|
||||||
[fields]
|
[fields]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
@ -480,80 +507,86 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
|
|
||||||
if (activeFilters.length === 0) return data;
|
if (activeFilters.length === 0) return data;
|
||||||
|
|
||||||
return data.filter((row) => {
|
const result = data.filter((row) => {
|
||||||
return activeFilters.every((filter) => {
|
return activeFilters.every((filter) => {
|
||||||
const value = row[filter.field];
|
const rawValue = row[filter.field];
|
||||||
const filterValues = filter.filterValues || [];
|
const filterValues = filter.filterValues || [];
|
||||||
const filterType = filter.filterType || "include";
|
const filterType = filter.filterType || "include";
|
||||||
|
|
||||||
|
// 타입 안전한 비교: 값을 문자열로 변환하여 비교
|
||||||
|
const value = rawValue === null || rawValue === undefined
|
||||||
|
? "(빈 값)"
|
||||||
|
: String(rawValue);
|
||||||
|
|
||||||
if (filterType === "include") {
|
if (filterType === "include") {
|
||||||
return filterValues.includes(value);
|
return filterValues.some((fv) => String(fv) === value);
|
||||||
} else {
|
} else {
|
||||||
return !filterValues.includes(value);
|
return filterValues.every((fv) => String(fv) !== value);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 모든 데이터가 필터링되면 경고 (디버깅용)
|
||||||
|
if (result.length === 0 && data.length > 0) {
|
||||||
|
console.warn("⚠️ [PivotGrid] 필터로 인해 모든 데이터가 제거됨");
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
}, [data, fields]);
|
}, [data, fields]);
|
||||||
|
|
||||||
// ==================== 피벗 처리 ====================
|
// ==================== 피벗 처리 ====================
|
||||||
|
|
||||||
const pivotResult = useMemo<PivotResult | null>(() => {
|
const pivotResult = useMemo<PivotResult | null>(() => {
|
||||||
|
try {
|
||||||
if (!filteredData || filteredData.length === 0 || fields.length === 0) {
|
if (!filteredData || filteredData.length === 0 || fields.length === 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const visibleFields = fields.filter((f) => f.visible !== false);
|
// FieldChooser에서 이미 필드를 완전히 제거하므로 visible 필터링 불필요
|
||||||
// 행, 열, 데이터 영역에 필드가 하나도 없으면 null 반환 (필터는 제외)
|
// 행, 열, 데이터 영역에 필드가 하나도 없으면 null 반환 (필터는 제외)
|
||||||
if (visibleFields.filter((f) => ["row", "column", "data"].includes(f.area)).length === 0) {
|
if (fields.filter((f) => ["row", "column", "data"].includes(f.area)).length === 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const result = processPivotData(
|
const result = processPivotData(
|
||||||
filteredData,
|
filteredData,
|
||||||
visibleFields,
|
fields,
|
||||||
pivotState.expandedRowPaths,
|
pivotState.expandedRowPaths,
|
||||||
pivotState.expandedColumnPaths
|
pivotState.expandedColumnPaths
|
||||||
);
|
);
|
||||||
|
|
||||||
// 🆕 피벗 결과 확인
|
|
||||||
console.log("🔶 피벗 처리 결과:", {
|
|
||||||
hasResult: !!result,
|
|
||||||
flatRowsCount: result?.flatRows?.length,
|
|
||||||
flatColumnsCount: result?.flatColumns?.length,
|
|
||||||
dataMatrixSize: result?.dataMatrix?.size,
|
|
||||||
expandedRowPaths: pivotState.expandedRowPaths.length,
|
|
||||||
expandedColumnPaths: pivotState.expandedColumnPaths.length,
|
|
||||||
});
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("❌ [pivotResult] 피벗 처리 에러:", error);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}, [filteredData, fields, pivotState.expandedRowPaths, pivotState.expandedColumnPaths]);
|
}, [filteredData, fields, pivotState.expandedRowPaths, pivotState.expandedColumnPaths]);
|
||||||
|
|
||||||
// 🆕 초기 로드 시 첫 레벨 자동 확장
|
// 초기 로드 시 첫 레벨 자동 확장
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (pivotResult && pivotResult.flatRows.length > 0) {
|
try {
|
||||||
console.log("🔶 피벗 결과 생성됨:", {
|
if (pivotResult && pivotResult.flatRows && pivotResult.flatRows.length > 0 && !isInitialExpanded) {
|
||||||
flatRowsCount: pivotResult.flatRows.length,
|
|
||||||
expandedRowPaths: pivotState.expandedRowPaths.length,
|
|
||||||
isInitialExpanded,
|
|
||||||
});
|
|
||||||
|
|
||||||
// 첫 레벨 행들의 경로 수집 (level 0인 행들)
|
// 첫 레벨 행들의 경로 수집 (level 0인 행들)
|
||||||
const firstLevelRows = pivotResult.flatRows.filter(row => row.level === 0 && row.hasChildren);
|
const firstLevelRows = pivotResult.flatRows.filter((row) => row.level === 0 && row.hasChildren);
|
||||||
|
|
||||||
console.log("🔶 첫 레벨 행 (level 0, hasChildren):", firstLevelRows.map(r => ({ path: r.path, caption: r.caption })));
|
// 첫 레벨 행이 있으면 자동 확장
|
||||||
|
if (firstLevelRows.length > 0 && firstLevelRows.length < 100) {
|
||||||
// 초기 확장이 안 되어 있고, 첫 레벨 행이 있으면 자동 확장
|
const firstLevelPaths = firstLevelRows.map((row) => row.path);
|
||||||
if (!isInitialExpanded && firstLevelRows.length > 0) {
|
setPivotState((prev) => ({
|
||||||
const firstLevelPaths = firstLevelRows.map(row => row.path);
|
|
||||||
console.log("🔶 초기 자동 확장 실행:", firstLevelPaths);
|
|
||||||
setPivotState(prev => ({
|
|
||||||
...prev,
|
...prev,
|
||||||
expandedRowPaths: firstLevelPaths,
|
expandedRowPaths: firstLevelPaths,
|
||||||
}));
|
}));
|
||||||
setIsInitialExpanded(true);
|
setIsInitialExpanded(true);
|
||||||
|
} else {
|
||||||
|
// 행이 너무 많으면 자동 확장 건너뛰기
|
||||||
|
setIsInitialExpanded(true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, [pivotResult, isInitialExpanded, pivotState.expandedRowPaths.length]);
|
} catch (error) {
|
||||||
|
console.error("❌ [초기 확장] 에러:", error);
|
||||||
|
setIsInitialExpanded(true);
|
||||||
|
}
|
||||||
|
}, [pivotResult, isInitialExpanded]);
|
||||||
|
|
||||||
// 조건부 서식용 전체 값 수집
|
// 조건부 서식용 전체 값 수집
|
||||||
const allCellValues = useMemo(() => {
|
const allCellValues = useMemo(() => {
|
||||||
|
|
@ -718,8 +751,6 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
// 행 확장/축소
|
// 행 확장/축소
|
||||||
const handleToggleRowExpand = useCallback(
|
const handleToggleRowExpand = useCallback(
|
||||||
(path: string[]) => {
|
(path: string[]) => {
|
||||||
console.log("🔶 행 확장/축소 클릭:", path);
|
|
||||||
|
|
||||||
setPivotState((prev) => {
|
setPivotState((prev) => {
|
||||||
const pathKey = pathToKey(path);
|
const pathKey = pathToKey(path);
|
||||||
const existingIndex = prev.expandedRowPaths.findIndex(
|
const existingIndex = prev.expandedRowPaths.findIndex(
|
||||||
|
|
@ -728,16 +759,13 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
|
|
||||||
let newPaths: string[][];
|
let newPaths: string[][];
|
||||||
if (existingIndex >= 0) {
|
if (existingIndex >= 0) {
|
||||||
console.log("🔶 행 축소:", path);
|
|
||||||
newPaths = prev.expandedRowPaths.filter(
|
newPaths = prev.expandedRowPaths.filter(
|
||||||
(_, i) => i !== existingIndex
|
(_, i) => i !== existingIndex
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
console.log("🔶 행 확장:", path);
|
|
||||||
newPaths = [...prev.expandedRowPaths, path];
|
newPaths = [...prev.expandedRowPaths, path];
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log("🔶 새로운 확장 경로:", newPaths);
|
|
||||||
onExpandChange?.(newPaths);
|
onExpandChange?.(newPaths);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
@ -749,14 +777,40 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
[onExpandChange]
|
[onExpandChange]
|
||||||
);
|
);
|
||||||
|
|
||||||
// 전체 확장
|
// 전체 확장 (재귀적으로 모든 레벨 확장)
|
||||||
const handleExpandAll = useCallback(() => {
|
const handleExpandAll = useCallback(() => {
|
||||||
if (!pivotResult) return;
|
try {
|
||||||
|
if (!pivotResult) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 재귀적으로 모든 가능한 경로 생성
|
||||||
const allRowPaths: string[][] = [];
|
const allRowPaths: string[][] = [];
|
||||||
pivotResult.flatRows.forEach((row) => {
|
const rowFields = fields.filter((f) => f.area === "row" && f.visible !== false);
|
||||||
if (row.hasChildren) {
|
|
||||||
allRowPaths.push(row.path);
|
// 행 필드가 없으면 종료
|
||||||
|
if (rowFields.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 데이터에서 모든 고유한 경로 추출
|
||||||
|
const pathSet = new Set<string>();
|
||||||
|
filteredData.forEach((item) => {
|
||||||
|
// 마지막 레벨은 제외 (확장할 자식이 없으므로)
|
||||||
|
for (let depth = 1; depth < rowFields.length; depth++) {
|
||||||
|
const path = rowFields.slice(0, depth).map((f) => String(item[f.field] ?? ""));
|
||||||
|
const pathKey = JSON.stringify(path);
|
||||||
|
pathSet.add(pathKey);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Set을 배열로 변환 (최대 1000개로 제한하여 성능 보호)
|
||||||
|
const MAX_PATHS = 1000;
|
||||||
|
let count = 0;
|
||||||
|
pathSet.forEach((pathKey) => {
|
||||||
|
if (count < MAX_PATHS) {
|
||||||
|
allRowPaths.push(JSON.parse(pathKey));
|
||||||
|
count++;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
@ -765,7 +819,10 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
expandedRowPaths: allRowPaths,
|
expandedRowPaths: allRowPaths,
|
||||||
expandedColumnPaths: [],
|
expandedColumnPaths: [],
|
||||||
}));
|
}));
|
||||||
}, [pivotResult]);
|
} catch (error) {
|
||||||
|
console.error("❌ [handleExpandAll] 에러:", error);
|
||||||
|
}
|
||||||
|
}, [pivotResult, fields, filteredData]);
|
||||||
|
|
||||||
// 전체 축소
|
// 전체 축소
|
||||||
const handleCollapseAll = useCallback(() => {
|
const handleCollapseAll = useCallback(() => {
|
||||||
|
|
@ -888,6 +945,8 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
|
|
||||||
// 인쇄 기능 (PDF 내보내기보다 먼저 정의해야 함)
|
// 인쇄 기능 (PDF 내보내기보다 먼저 정의해야 함)
|
||||||
const handlePrint = useCallback(() => {
|
const handlePrint = useCallback(() => {
|
||||||
|
if (typeof window === "undefined") return;
|
||||||
|
|
||||||
const printContent = tableRef.current;
|
const printContent = tableRef.current;
|
||||||
if (!printContent) return;
|
if (!printContent) return;
|
||||||
|
|
||||||
|
|
@ -988,10 +1047,14 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
console.log("피벗 상태가 저장되었습니다.");
|
console.log("피벗 상태가 저장되었습니다.");
|
||||||
}, [saveStateToStorage]);
|
}, [saveStateToStorage]);
|
||||||
|
|
||||||
// 상태 초기화
|
// 상태 초기화 (확장/축소, 정렬, 필터만 초기화, 필드 설정은 유지)
|
||||||
const handleResetState = useCallback(() => {
|
const handleResetState = useCallback(() => {
|
||||||
|
// 로컬 스토리지에서 상태 제거 (SSR 보호)
|
||||||
|
if (typeof window !== "undefined") {
|
||||||
localStorage.removeItem(stateStorageKey);
|
localStorage.removeItem(stateStorageKey);
|
||||||
setFields(initialFields);
|
}
|
||||||
|
|
||||||
|
// 확장/축소, 정렬, 필터 상태만 초기화
|
||||||
setPivotState({
|
setPivotState({
|
||||||
expandedRowPaths: [],
|
expandedRowPaths: [],
|
||||||
expandedColumnPaths: [],
|
expandedColumnPaths: [],
|
||||||
|
|
@ -1002,7 +1065,7 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
setColumnWidths({});
|
setColumnWidths({});
|
||||||
setSelectedCell(null);
|
setSelectedCell(null);
|
||||||
setSelectionRange(null);
|
setSelectionRange(null);
|
||||||
}, [stateStorageKey, initialFields]);
|
}, [stateStorageKey]);
|
||||||
|
|
||||||
// 필드 숨기기/표시 상태
|
// 필드 숨기기/표시 상태
|
||||||
const [hiddenFields, setHiddenFields] = useState<Set<string>>(new Set());
|
const [hiddenFields, setHiddenFields] = useState<Set<string>>(new Set());
|
||||||
|
|
@ -1019,11 +1082,6 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
});
|
});
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
// 숨겨진 필드 제외한 활성 필드들
|
|
||||||
const visibleFields = useMemo(() => {
|
|
||||||
return fields.filter((f) => !hiddenFields.has(f.field));
|
|
||||||
}, [fields, hiddenFields]);
|
|
||||||
|
|
||||||
// 숨겨진 필드 목록
|
// 숨겨진 필드 목록
|
||||||
const hiddenFieldsList = useMemo(() => {
|
const hiddenFieldsList = useMemo(() => {
|
||||||
return fields.filter((f) => hiddenFields.has(f.field));
|
return fields.filter((f) => hiddenFields.has(f.field));
|
||||||
|
|
@ -1391,8 +1449,8 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
size="sm"
|
size="sm"
|
||||||
className="h-7 px-2"
|
className="h-7 px-2"
|
||||||
onClick={handleExpandAll}
|
onClick={handleCollapseAll}
|
||||||
title="전체 확장"
|
title="전체 축소"
|
||||||
>
|
>
|
||||||
<ChevronDown className="h-4 w-4" />
|
<ChevronDown className="h-4 w-4" />
|
||||||
</Button>
|
</Button>
|
||||||
|
|
@ -1401,8 +1459,8 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
size="sm"
|
size="sm"
|
||||||
className="h-7 px-2"
|
className="h-7 px-2"
|
||||||
onClick={handleCollapseAll}
|
onClick={handleExpandAll}
|
||||||
title="전체 축소"
|
title="전체 확장"
|
||||||
>
|
>
|
||||||
<ChevronRight className="h-4 w-4" />
|
<ChevronRight className="h-4 w-4" />
|
||||||
</Button>
|
</Button>
|
||||||
|
|
@ -1582,19 +1640,25 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
<button
|
<button
|
||||||
className={cn(
|
className={cn(
|
||||||
"flex items-center gap-1.5 px-2 py-1 rounded text-xs",
|
"flex items-center gap-1.5 px-2 py-1 rounded text-xs",
|
||||||
"border transition-colors",
|
"border transition-colors max-w-xs",
|
||||||
isFiltered
|
isFiltered
|
||||||
? "bg-orange-100 border-orange-300 text-orange-800 dark:bg-orange-900/30 dark:border-orange-700 dark:text-orange-200"
|
? "bg-orange-100 border-orange-300 text-orange-800 dark:bg-orange-900/30 dark:border-orange-700 dark:text-orange-200"
|
||||||
: "bg-background border-border hover:bg-accent"
|
: "bg-background border-border hover:bg-accent"
|
||||||
)}
|
)}
|
||||||
|
title={isFiltered ? `${filterField.caption}: ${selectedValues.join(", ")}` : filterField.caption}
|
||||||
>
|
>
|
||||||
<span>{filterField.caption}</span>
|
<span className="font-medium">{filterField.caption}:</span>
|
||||||
{isFiltered && (
|
{isFiltered ? (
|
||||||
<span className="bg-orange-500 text-white px-1 rounded text-[10px]">
|
<span className="truncate">
|
||||||
{selectedValues.length}
|
{selectedValues.length <= 2
|
||||||
|
? selectedValues.join(", ")
|
||||||
|
: `${selectedValues.slice(0, 2).join(", ")} 외 ${selectedValues.length - 2}개`
|
||||||
|
}
|
||||||
</span>
|
</span>
|
||||||
|
) : (
|
||||||
|
<span className="text-muted-foreground">전체</span>
|
||||||
)}
|
)}
|
||||||
<ChevronDown className="h-3 w-3" />
|
<ChevronDown className="h-3 w-3 shrink-0" />
|
||||||
</button>
|
</button>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|
@ -1608,7 +1672,14 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
<div
|
<div
|
||||||
ref={tableContainerRef}
|
ref={tableContainerRef}
|
||||||
className="flex-1 overflow-auto focus:outline-none"
|
className="flex-1 overflow-auto focus:outline-none"
|
||||||
style={{ maxHeight: enableVirtualScroll ? containerHeight : undefined }}
|
style={{
|
||||||
|
maxHeight: enableVirtualScroll && containerHeight > 0 ? containerHeight : undefined,
|
||||||
|
// 최소 200px 보장 + 데이터에 맞게 조정 (최대 400px)
|
||||||
|
minHeight: Math.max(
|
||||||
|
200, // 절대 최소값 - 블라인드 효과 방지
|
||||||
|
Math.min(400, (sortedFlatRows.length + 3) * ROW_HEIGHT + 50)
|
||||||
|
)
|
||||||
|
}}
|
||||||
tabIndex={0}
|
tabIndex={0}
|
||||||
onKeyDown={handleKeyDown}
|
onKeyDown={handleKeyDown}
|
||||||
>
|
>
|
||||||
|
|
@ -1883,12 +1954,15 @@ export const PivotGridComponent: React.FC<PivotGridProps> = ({
|
||||||
});
|
});
|
||||||
})()}
|
})()}
|
||||||
|
|
||||||
{/* 가상 스크롤 하단 여백 */}
|
{/* 가상 스크롤 하단 여백 - 음수 방지 */}
|
||||||
{enableVirtualScroll && (
|
{enableVirtualScroll && (() => {
|
||||||
<tr style={{ height: virtualScroll.totalHeight - virtualScroll.offsetTop - (visibleFlatRows.length * ROW_HEIGHT) }}>
|
const bottomPadding = Math.max(0, virtualScroll.totalHeight - virtualScroll.offsetTop - (visibleFlatRows.length * ROW_HEIGHT));
|
||||||
|
return bottomPadding > 0 ? (
|
||||||
|
<tr style={{ height: bottomPadding }}>
|
||||||
<td colSpan={rowFields.length + flatColumns.length + (totals?.showRowGrandTotals ? dataFields.length : 0)} />
|
<td colSpan={rowFields.length + flatColumns.length + (totals?.showRowGrandTotals ? dataFields.length : 0)} />
|
||||||
</tr>
|
</tr>
|
||||||
)}
|
) : null;
|
||||||
|
})()}
|
||||||
|
|
||||||
{/* 열 총계 행 (하단 위치 - 기본값) */}
|
{/* 열 총계 행 (하단 위치 - 기본값) */}
|
||||||
{totals?.showColumnGrandTotals && totals?.rowGrandTotalPosition !== "top" && (
|
{totals?.showColumnGrandTotals && totals?.rowGrandTotalPosition !== "top" && (
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState, Component, ErrorInfo, ReactNode } from "react";
|
||||||
import { AutoRegisteringComponentRenderer } from "../../AutoRegisteringComponentRenderer";
|
import { AutoRegisteringComponentRenderer } from "../../AutoRegisteringComponentRenderer";
|
||||||
import { createComponentDefinition } from "../../utils/createComponentDefinition";
|
import { createComponentDefinition } from "../../utils/createComponentDefinition";
|
||||||
import { ComponentCategory } from "@/types/component";
|
import { ComponentCategory } from "@/types/component";
|
||||||
|
|
@ -8,6 +8,66 @@ import { PivotGridComponent } from "./PivotGridComponent";
|
||||||
import { PivotGridConfigPanel } from "./PivotGridConfigPanel";
|
import { PivotGridConfigPanel } from "./PivotGridConfigPanel";
|
||||||
import { PivotFieldConfig } from "./types";
|
import { PivotFieldConfig } from "./types";
|
||||||
import { dataApi } from "@/lib/api/data";
|
import { dataApi } from "@/lib/api/data";
|
||||||
|
import { AlertCircle, RefreshCw } from "lucide-react";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
|
||||||
|
// ==================== 에러 경계 ====================
|
||||||
|
|
||||||
|
interface ErrorBoundaryState {
|
||||||
|
hasError: boolean;
|
||||||
|
error?: Error;
|
||||||
|
}
|
||||||
|
|
||||||
|
class PivotGridErrorBoundary extends Component<
|
||||||
|
{ children: ReactNode; onReset?: () => void },
|
||||||
|
ErrorBoundaryState
|
||||||
|
> {
|
||||||
|
constructor(props: { children: ReactNode; onReset?: () => void }) {
|
||||||
|
super(props);
|
||||||
|
this.state = { hasError: false };
|
||||||
|
}
|
||||||
|
|
||||||
|
static getDerivedStateFromError(error: Error): ErrorBoundaryState {
|
||||||
|
return { hasError: true, error };
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidCatch(error: Error, errorInfo: ErrorInfo) {
|
||||||
|
console.error("🔴 [PivotGrid] 렌더링 에러:", error);
|
||||||
|
console.error("🔴 [PivotGrid] 에러 정보:", errorInfo);
|
||||||
|
}
|
||||||
|
|
||||||
|
handleReset = () => {
|
||||||
|
this.setState({ hasError: false, error: undefined });
|
||||||
|
this.props.onReset?.();
|
||||||
|
};
|
||||||
|
|
||||||
|
render() {
|
||||||
|
if (this.state.hasError) {
|
||||||
|
return (
|
||||||
|
<div className="flex flex-col items-center justify-center p-8 text-center border border-destructive/50 rounded-lg bg-destructive/5">
|
||||||
|
<AlertCircle className="h-8 w-8 text-destructive mb-2" />
|
||||||
|
<h3 className="text-sm font-medium text-destructive mb-1">
|
||||||
|
피벗 그리드 오류
|
||||||
|
</h3>
|
||||||
|
<p className="text-xs text-muted-foreground mb-3 max-w-md">
|
||||||
|
{this.state.error?.message || "알 수 없는 오류가 발생했습니다."}
|
||||||
|
</p>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
onClick={this.handleReset}
|
||||||
|
className="gap-2"
|
||||||
|
>
|
||||||
|
<RefreshCw className="h-3.5 w-3.5" />
|
||||||
|
다시 시도
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.props.children;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// ==================== 샘플 데이터 (미리보기용) ====================
|
// ==================== 샘플 데이터 (미리보기용) ====================
|
||||||
|
|
||||||
|
|
@ -111,19 +171,14 @@ const PivotGridWrapper: React.FC<any> = (props) => {
|
||||||
|
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
try {
|
try {
|
||||||
console.log("🔷 [PivotGrid] 테이블 데이터 로딩 시작:", tableName);
|
|
||||||
|
|
||||||
const response = await dataApi.getTableData(tableName, {
|
const response = await dataApi.getTableData(tableName, {
|
||||||
page: 1,
|
page: 1,
|
||||||
size: 10000, // 피벗 분석용 대량 데이터 (pageSize → size)
|
size: 10000, // 피벗 분석용 대량 데이터
|
||||||
});
|
});
|
||||||
|
|
||||||
console.log("🔷 [PivotGrid] API 응답:", response);
|
|
||||||
|
|
||||||
// dataApi.getTableData는 { data, total, page, size, totalPages } 구조
|
// dataApi.getTableData는 { data, total, page, size, totalPages } 구조
|
||||||
if (response.data && Array.isArray(response.data)) {
|
if (response.data && Array.isArray(response.data)) {
|
||||||
setLoadedData(response.data);
|
setLoadedData(response.data);
|
||||||
console.log("✅ [PivotGrid] 데이터 로딩 완료:", response.data.length, "건");
|
|
||||||
} else {
|
} else {
|
||||||
console.error("❌ [PivotGrid] 데이터 로딩 실패: 응답에 data 배열이 없음");
|
console.error("❌ [PivotGrid] 데이터 로딩 실패: 응답에 data 배열이 없음");
|
||||||
setLoadedData([]);
|
setLoadedData([]);
|
||||||
|
|
@ -138,21 +193,6 @@ const PivotGridWrapper: React.FC<any> = (props) => {
|
||||||
loadTableData();
|
loadTableData();
|
||||||
}, [componentConfig.dataSource?.tableName, configData, props.isDesignMode]);
|
}, [componentConfig.dataSource?.tableName, configData, props.isDesignMode]);
|
||||||
|
|
||||||
// 디버깅 로그
|
|
||||||
console.log("🔷 PivotGridWrapper props:", {
|
|
||||||
isDesignMode: props.isDesignMode,
|
|
||||||
isInteractive: props.isInteractive,
|
|
||||||
hasComponentConfig: !!props.componentConfig,
|
|
||||||
hasConfig: !!props.config,
|
|
||||||
hasData: !!configData,
|
|
||||||
dataLength: configData?.length,
|
|
||||||
hasLoadedData: loadedData.length > 0,
|
|
||||||
loadedDataLength: loadedData.length,
|
|
||||||
hasFields: !!configFields,
|
|
||||||
fieldsLength: configFields?.length,
|
|
||||||
isLoading,
|
|
||||||
});
|
|
||||||
|
|
||||||
// 디자인 모드 판단:
|
// 디자인 모드 판단:
|
||||||
// 1. isDesignMode === true
|
// 1. isDesignMode === true
|
||||||
// 2. isInteractive === false (편집 모드)
|
// 2. isInteractive === false (편집 모드)
|
||||||
|
|
@ -173,13 +213,6 @@ const PivotGridWrapper: React.FC<any> = (props) => {
|
||||||
? (componentConfig.title || props.title || "피벗 그리드") + " (미리보기)"
|
? (componentConfig.title || props.title || "피벗 그리드") + " (미리보기)"
|
||||||
: (componentConfig.title || props.title);
|
: (componentConfig.title || props.title);
|
||||||
|
|
||||||
console.log("🔷 PivotGridWrapper final:", {
|
|
||||||
isDesignMode,
|
|
||||||
usePreviewData,
|
|
||||||
finalDataLength: finalData?.length,
|
|
||||||
finalFieldsLength: finalFields?.length,
|
|
||||||
});
|
|
||||||
|
|
||||||
// 총계 설정
|
// 총계 설정
|
||||||
const totalsConfig = componentConfig.totals || props.totals || {
|
const totalsConfig = componentConfig.totals || props.totals || {
|
||||||
showRowGrandTotals: true,
|
showRowGrandTotals: true,
|
||||||
|
|
@ -200,7 +233,9 @@ const PivotGridWrapper: React.FC<any> = (props) => {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 에러 경계로 감싸서 렌더링 에러 시 컴포넌트가 완전히 사라지지 않도록 함
|
||||||
return (
|
return (
|
||||||
|
<PivotGridErrorBoundary>
|
||||||
<PivotGridComponent
|
<PivotGridComponent
|
||||||
title={finalTitle}
|
title={finalTitle}
|
||||||
data={finalData}
|
data={finalData}
|
||||||
|
|
@ -218,6 +253,7 @@ const PivotGridWrapper: React.FC<any> = (props) => {
|
||||||
onFieldDrop={props.onFieldDrop}
|
onFieldDrop={props.onFieldDrop}
|
||||||
onExpandChange={props.onExpandChange}
|
onExpandChange={props.onExpandChange}
|
||||||
/>
|
/>
|
||||||
|
</PivotGridErrorBoundary>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -284,18 +320,6 @@ export class PivotGridRenderer extends AutoRegisteringComponentRenderer {
|
||||||
const configFields = componentConfig.fields || props.fields;
|
const configFields = componentConfig.fields || props.fields;
|
||||||
const configData = props.data;
|
const configData = props.data;
|
||||||
|
|
||||||
// 디버깅 로그
|
|
||||||
console.log("🔷 PivotGridRenderer props:", {
|
|
||||||
isDesignMode: props.isDesignMode,
|
|
||||||
isInteractive: props.isInteractive,
|
|
||||||
hasComponentConfig: !!props.componentConfig,
|
|
||||||
hasConfig: !!props.config,
|
|
||||||
hasData: !!configData,
|
|
||||||
dataLength: configData?.length,
|
|
||||||
hasFields: !!configFields,
|
|
||||||
fieldsLength: configFields?.length,
|
|
||||||
});
|
|
||||||
|
|
||||||
// 디자인 모드 판단:
|
// 디자인 모드 판단:
|
||||||
// 1. isDesignMode === true
|
// 1. isDesignMode === true
|
||||||
// 2. isInteractive === false (편집 모드)
|
// 2. isInteractive === false (편집 모드)
|
||||||
|
|
@ -314,13 +338,6 @@ export class PivotGridRenderer extends AutoRegisteringComponentRenderer {
|
||||||
? (componentConfig.title || props.title || "피벗 그리드") + " (미리보기)"
|
? (componentConfig.title || props.title || "피벗 그리드") + " (미리보기)"
|
||||||
: (componentConfig.title || props.title);
|
: (componentConfig.title || props.title);
|
||||||
|
|
||||||
console.log("🔷 PivotGridRenderer final:", {
|
|
||||||
isDesignMode,
|
|
||||||
usePreviewData,
|
|
||||||
finalDataLength: finalData?.length,
|
|
||||||
finalFieldsLength: finalFields?.length,
|
|
||||||
});
|
|
||||||
|
|
||||||
// 총계 설정
|
// 총계 설정
|
||||||
const totalsConfig = componentConfig.totals || props.totals || {
|
const totalsConfig = componentConfig.totals || props.totals || {
|
||||||
showRowGrandTotals: true,
|
showRowGrandTotals: true,
|
||||||
|
|
|
||||||
|
|
@ -267,11 +267,9 @@ export const FieldChooser: React.FC<FieldChooserProps> = ({
|
||||||
const existingConfig = selectedFields.find((f) => f.field === field.field);
|
const existingConfig = selectedFields.find((f) => f.field === field.field);
|
||||||
|
|
||||||
if (area === "none") {
|
if (area === "none") {
|
||||||
// 필드 제거 또는 숨기기
|
// 필드 완전 제거 (visible: false 대신 배열에서 제거)
|
||||||
if (existingConfig) {
|
if (existingConfig) {
|
||||||
const newFields = selectedFields.map((f) =>
|
const newFields = selectedFields.filter((f) => f.field !== field.field);
|
||||||
f.field === field.field ? { ...f, visible: false } : f
|
|
||||||
);
|
|
||||||
onFieldsChange(newFields);
|
onFieldsChange(newFields);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|
|
||||||
|
|
@ -25,6 +25,7 @@ import {
|
||||||
horizontalListSortingStrategy,
|
horizontalListSortingStrategy,
|
||||||
useSortable,
|
useSortable,
|
||||||
} from "@dnd-kit/sortable";
|
} from "@dnd-kit/sortable";
|
||||||
|
import { useDroppable } from "@dnd-kit/core";
|
||||||
import { CSS } from "@dnd-kit/utilities";
|
import { CSS } from "@dnd-kit/utilities";
|
||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
import { PivotFieldConfig, PivotAreaType } from "../types";
|
import { PivotFieldConfig, PivotAreaType } from "../types";
|
||||||
|
|
@ -244,22 +245,31 @@ const DroppableArea: React.FC<DroppableAreaProps> = ({
|
||||||
const areaFields = fields.filter((f) => f.area === area && f.visible !== false);
|
const areaFields = fields.filter((f) => f.area === area && f.visible !== false);
|
||||||
const fieldIds = areaFields.map((f) => `${area}-${f.field}`);
|
const fieldIds = areaFields.map((f) => `${area}-${f.field}`);
|
||||||
|
|
||||||
|
// 🆕 드롭 가능 영역 설정
|
||||||
|
const { setNodeRef, isOver: isOverDroppable } = useDroppable({
|
||||||
|
id: area, // "filter", "column", "row", "data"
|
||||||
|
});
|
||||||
|
|
||||||
|
const finalIsOver = isOver || isOverDroppable;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
|
ref={setNodeRef}
|
||||||
className={cn(
|
className={cn(
|
||||||
"flex-1 min-h-[44px] rounded border border-dashed p-1.5",
|
"flex-1 min-h-[60px] rounded border-2 border-dashed p-2",
|
||||||
"transition-colors duration-200",
|
"transition-all duration-200",
|
||||||
config.color,
|
config.color,
|
||||||
isOver && "border-primary bg-primary/5"
|
finalIsOver && "border-primary bg-primary/10 scale-[1.02]",
|
||||||
|
areaFields.length === 0 && "border-2" // 빈 영역일 때 테두리 강조
|
||||||
)}
|
)}
|
||||||
data-area={area}
|
data-area={area}
|
||||||
>
|
>
|
||||||
{/* 영역 헤더 */}
|
{/* 영역 헤더 */}
|
||||||
<div className="flex items-center gap-1 mb-1 text-[11px] font-medium text-muted-foreground">
|
<div className="flex items-center gap-1 mb-1.5 text-xs font-semibold text-muted-foreground">
|
||||||
{icon}
|
{icon}
|
||||||
<span>{title}</span>
|
<span>{title}</span>
|
||||||
{areaFields.length > 0 && (
|
{areaFields.length > 0 && (
|
||||||
<span className="text-[10px] bg-muted px-1 rounded">
|
<span className="text-[10px] bg-muted px-1.5 py-0.5 rounded">
|
||||||
{areaFields.length}
|
{areaFields.length}
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
|
|
@ -267,11 +277,16 @@ const DroppableArea: React.FC<DroppableAreaProps> = ({
|
||||||
|
|
||||||
{/* 필드 목록 */}
|
{/* 필드 목록 */}
|
||||||
<SortableContext items={fieldIds} strategy={horizontalListSortingStrategy}>
|
<SortableContext items={fieldIds} strategy={horizontalListSortingStrategy}>
|
||||||
<div className="flex flex-wrap gap-1 min-h-[22px]">
|
<div className="flex flex-wrap gap-1 min-h-[28px] relative">
|
||||||
{areaFields.length === 0 ? (
|
{areaFields.length === 0 ? (
|
||||||
<span className="text-[10px] text-muted-foreground/50 italic">
|
<div
|
||||||
필드를 여기로 드래그
|
className="flex items-center justify-center w-full py-1 pointer-events-none"
|
||||||
|
style={{ pointerEvents: 'none' }}
|
||||||
|
>
|
||||||
|
<span className="text-xs text-muted-foreground/70 italic font-medium">
|
||||||
|
← 필드를 여기로 드래그하세요
|
||||||
</span>
|
</span>
|
||||||
|
</div>
|
||||||
) : (
|
) : (
|
||||||
areaFields.map((field) => (
|
areaFields.map((field) => (
|
||||||
<SortableFieldChip
|
<SortableFieldChip
|
||||||
|
|
@ -339,8 +354,16 @@ export const FieldPanel: React.FC<FieldPanelProps> = ({
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 드롭 영역 감지
|
// 드롭 영역 감지 (영역 자체의 ID를 우선 확인)
|
||||||
const overId = over.id as string;
|
const overId = over.id as string;
|
||||||
|
|
||||||
|
// 1. overId가 영역 자체인 경우 (filter, column, row, data)
|
||||||
|
if (["filter", "column", "row", "data"].includes(overId)) {
|
||||||
|
setOverArea(overId as PivotAreaType);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. overId가 필드인 경우 (예: row-part_name)
|
||||||
const targetArea = overId.split("-")[0] as PivotAreaType;
|
const targetArea = overId.split("-")[0] as PivotAreaType;
|
||||||
if (["filter", "column", "row", "data"].includes(targetArea)) {
|
if (["filter", "column", "row", "data"].includes(targetArea)) {
|
||||||
setOverArea(targetArea);
|
setOverArea(targetArea);
|
||||||
|
|
@ -350,10 +373,13 @@ export const FieldPanel: React.FC<FieldPanelProps> = ({
|
||||||
// 드래그 종료
|
// 드래그 종료
|
||||||
const handleDragEnd = (event: DragEndEvent) => {
|
const handleDragEnd = (event: DragEndEvent) => {
|
||||||
const { active, over } = event;
|
const { active, over } = event;
|
||||||
|
const currentOverArea = overArea; // handleDragOver에서 감지한 영역 저장
|
||||||
setActiveId(null);
|
setActiveId(null);
|
||||||
setOverArea(null);
|
setOverArea(null);
|
||||||
|
|
||||||
if (!over) return;
|
if (!over) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const activeId = active.id as string;
|
const activeId = active.id as string;
|
||||||
const overId = over.id as string;
|
const overId = over.id as string;
|
||||||
|
|
@ -363,7 +389,16 @@ export const FieldPanel: React.FC<FieldPanelProps> = ({
|
||||||
PivotAreaType,
|
PivotAreaType,
|
||||||
string
|
string
|
||||||
];
|
];
|
||||||
const [targetArea] = overId.split("-") as [PivotAreaType, string];
|
|
||||||
|
// targetArea 결정: handleDragOver에서 감지한 영역 우선 사용
|
||||||
|
let targetArea: PivotAreaType;
|
||||||
|
if (currentOverArea) {
|
||||||
|
targetArea = currentOverArea;
|
||||||
|
} else if (["filter", "column", "row", "data"].includes(overId)) {
|
||||||
|
targetArea = overId as PivotAreaType;
|
||||||
|
} else {
|
||||||
|
targetArea = overId.split("-")[0] as PivotAreaType;
|
||||||
|
}
|
||||||
|
|
||||||
// 같은 영역 내 정렬
|
// 같은 영역 내 정렬
|
||||||
if (sourceArea === targetArea) {
|
if (sourceArea === targetArea) {
|
||||||
|
|
@ -406,6 +441,7 @@ export const FieldPanel: React.FC<FieldPanelProps> = ({
|
||||||
}
|
}
|
||||||
return f;
|
return f;
|
||||||
});
|
});
|
||||||
|
|
||||||
onFieldsChange(newFields);
|
onFieldsChange(newFields);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -51,14 +51,18 @@ export function useVirtualScroll(options: VirtualScrollOptions): VirtualScrollRe
|
||||||
// 보이는 아이템 수
|
// 보이는 아이템 수
|
||||||
const visibleCount = Math.ceil(containerHeight / itemHeight);
|
const visibleCount = Math.ceil(containerHeight / itemHeight);
|
||||||
|
|
||||||
// 시작/끝 인덱스 계산
|
// 시작/끝 인덱스 계산 (음수 방지)
|
||||||
const { startIndex, endIndex } = useMemo(() => {
|
const { startIndex, endIndex } = useMemo(() => {
|
||||||
|
// itemCount가 0이면 빈 배열
|
||||||
|
if (itemCount === 0) {
|
||||||
|
return { startIndex: 0, endIndex: -1 };
|
||||||
|
}
|
||||||
const start = Math.max(0, Math.floor(scrollTop / itemHeight) - overscan);
|
const start = Math.max(0, Math.floor(scrollTop / itemHeight) - overscan);
|
||||||
const end = Math.min(
|
const end = Math.min(
|
||||||
itemCount - 1,
|
itemCount - 1,
|
||||||
Math.ceil((scrollTop + containerHeight) / itemHeight) + overscan
|
Math.ceil((scrollTop + containerHeight) / itemHeight) + overscan
|
||||||
);
|
);
|
||||||
return { startIndex: start, endIndex: end };
|
return { startIndex: start, endIndex: Math.max(start, end) }; // end가 start보다 작지 않도록
|
||||||
}, [scrollTop, itemHeight, containerHeight, itemCount, overscan]);
|
}, [scrollTop, itemHeight, containerHeight, itemCount, overscan]);
|
||||||
|
|
||||||
// 전체 높이
|
// 전체 높이
|
||||||
|
|
|
||||||
|
|
@ -710,27 +710,19 @@ export function processPivotData(
|
||||||
.filter((f) => f.area === "data" && f.visible !== false)
|
.filter((f) => f.area === "data" && f.visible !== false)
|
||||||
.sort((a, b) => (a.areaIndex || 0) - (b.areaIndex || 0));
|
.sort((a, b) => (a.areaIndex || 0) - (b.areaIndex || 0));
|
||||||
|
|
||||||
const filterFields = fields.filter(
|
// 참고: 필터링은 PivotGridComponent에서 이미 처리됨
|
||||||
(f) => f.area === "filter" && f.visible !== false
|
// 여기서는 추가 필터링 없이 전달받은 데이터 사용
|
||||||
|
const filteredData = data;
|
||||||
|
|
||||||
|
// 확장 경로 Set 변환 (잘못된 형식 필터링)
|
||||||
|
const validRowPaths = (expandedRowPaths || []).filter(
|
||||||
|
(p): p is string[] => Array.isArray(p) && p.length > 0 && p.every(item => typeof item === "string")
|
||||||
);
|
);
|
||||||
|
const validColPaths = (expandedColumnPaths || []).filter(
|
||||||
// 필터 적용
|
(p): p is string[] => Array.isArray(p) && p.length > 0 && p.every(item => typeof item === "string")
|
||||||
let filteredData = data;
|
);
|
||||||
for (const filterField of filterFields) {
|
const expandedRowSet = new Set(validRowPaths.map(pathToKey));
|
||||||
if (filterField.filterValues && filterField.filterValues.length > 0) {
|
const expandedColSet = new Set(validColPaths.map(pathToKey));
|
||||||
filteredData = filteredData.filter((row) => {
|
|
||||||
const value = getFieldValue(row, filterField);
|
|
||||||
if (filterField.filterType === "exclude") {
|
|
||||||
return !filterField.filterValues!.includes(value);
|
|
||||||
}
|
|
||||||
return filterField.filterValues!.includes(value);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 확장 경로 Set 변환
|
|
||||||
const expandedRowSet = new Set(expandedRowPaths.map(pathToKey));
|
|
||||||
const expandedColSet = new Set(expandedColumnPaths.map(pathToKey));
|
|
||||||
|
|
||||||
// 기본 확장: 첫 번째 레벨 모두 확장
|
// 기본 확장: 첫 번째 레벨 모두 확장
|
||||||
if (expandedRowPaths.length === 0 && rowFields.length > 0) {
|
if (expandedRowPaths.length === 0 && rowFields.length > 0) {
|
||||||
|
|
|
||||||
|
|
@ -41,7 +41,7 @@ import {
|
||||||
Lock,
|
Lock,
|
||||||
} from "lucide-react";
|
} from "lucide-react";
|
||||||
import * as XLSX from "xlsx";
|
import * as XLSX from "xlsx";
|
||||||
import { FileText, ChevronRightIcon } from "lucide-react";
|
import { FileText, ChevronRightIcon, Search } from "lucide-react";
|
||||||
import { Checkbox } from "@/components/ui/checkbox";
|
import { Checkbox } from "@/components/ui/checkbox";
|
||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
|
|
@ -455,6 +455,7 @@ export const TableListComponent: React.FC<TableListComponentProps> = ({
|
||||||
|
|
||||||
// 🆕 컬럼 헤더 필터 상태 (상단에서 선언)
|
// 🆕 컬럼 헤더 필터 상태 (상단에서 선언)
|
||||||
const [headerFilters, setHeaderFilters] = useState<Record<string, Set<string>>>({});
|
const [headerFilters, setHeaderFilters] = useState<Record<string, Set<string>>>({});
|
||||||
|
const [headerLikeFilters, setHeaderLikeFilters] = useState<Record<string, string>>({}); // LIKE 검색용
|
||||||
const [openFilterColumn, setOpenFilterColumn] = useState<string | null>(null);
|
const [openFilterColumn, setOpenFilterColumn] = useState<string | null>(null);
|
||||||
|
|
||||||
// 🆕 Filter Builder (고급 필터) 관련 상태 - filteredData보다 먼저 정의해야 함
|
// 🆕 Filter Builder (고급 필터) 관련 상태 - filteredData보다 먼저 정의해야 함
|
||||||
|
|
@ -488,6 +489,22 @@ export const TableListComponent: React.FC<TableListComponentProps> = ({
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 2-1. 🆕 LIKE 검색 필터 적용
|
||||||
|
if (Object.keys(headerLikeFilters).length > 0) {
|
||||||
|
result = result.filter((row) => {
|
||||||
|
return Object.entries(headerLikeFilters).every(([columnName, searchText]) => {
|
||||||
|
if (!searchText || searchText.trim() === "") return true;
|
||||||
|
|
||||||
|
// 여러 가능한 컬럼명 시도
|
||||||
|
const cellValue = row[columnName] ?? row[columnName.toLowerCase()] ?? row[columnName.toUpperCase()];
|
||||||
|
const cellStr = cellValue !== null && cellValue !== undefined ? String(cellValue).toLowerCase() : "";
|
||||||
|
|
||||||
|
// LIKE 검색 (대소문자 무시)
|
||||||
|
return cellStr.includes(searchText.toLowerCase());
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
// 3. 🆕 Filter Builder 적용
|
// 3. 🆕 Filter Builder 적용
|
||||||
if (filterGroups.length > 0) {
|
if (filterGroups.length > 0) {
|
||||||
result = result.filter((row) => {
|
result = result.filter((row) => {
|
||||||
|
|
@ -541,7 +558,7 @@ export const TableListComponent: React.FC<TableListComponentProps> = ({
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}, [data, splitPanelPosition, splitPanelContext?.addedItemIds, headerFilters, filterGroups]);
|
}, [data, splitPanelPosition, splitPanelContext?.addedItemIds, headerFilters, headerLikeFilters, filterGroups]);
|
||||||
|
|
||||||
const [currentPage, setCurrentPage] = useState(1);
|
const [currentPage, setCurrentPage] = useState(1);
|
||||||
const [totalPages, setTotalPages] = useState(0);
|
const [totalPages, setTotalPages] = useState(0);
|
||||||
|
|
@ -2935,6 +2952,7 @@ export const TableListComponent: React.FC<TableListComponentProps> = ({
|
||||||
headerFilters: Object.fromEntries(
|
headerFilters: Object.fromEntries(
|
||||||
Object.entries(headerFilters).map(([key, set]) => [key, Array.from(set as Set<string>)]),
|
Object.entries(headerFilters).map(([key, set]) => [key, Array.from(set as Set<string>)]),
|
||||||
),
|
),
|
||||||
|
headerLikeFilters, // LIKE 검색 필터 저장
|
||||||
pageSize: localPageSize,
|
pageSize: localPageSize,
|
||||||
timestamp: Date.now(),
|
timestamp: Date.now(),
|
||||||
};
|
};
|
||||||
|
|
@ -2955,6 +2973,7 @@ export const TableListComponent: React.FC<TableListComponentProps> = ({
|
||||||
frozenColumnCount,
|
frozenColumnCount,
|
||||||
showGridLines,
|
showGridLines,
|
||||||
headerFilters,
|
headerFilters,
|
||||||
|
headerLikeFilters,
|
||||||
localPageSize,
|
localPageSize,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
|
@ -2991,6 +3010,9 @@ export const TableListComponent: React.FC<TableListComponentProps> = ({
|
||||||
});
|
});
|
||||||
setHeaderFilters(filters);
|
setHeaderFilters(filters);
|
||||||
}
|
}
|
||||||
|
if (state.headerLikeFilters) {
|
||||||
|
setHeaderLikeFilters(state.headerLikeFilters);
|
||||||
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("❌ 테이블 상태 복원 실패:", error);
|
console.error("❌ 테이블 상태 복원 실패:", error);
|
||||||
}
|
}
|
||||||
|
|
@ -5737,7 +5759,7 @@ export const TableListComponent: React.FC<TableListComponentProps> = ({
|
||||||
}}
|
}}
|
||||||
className={cn(
|
className={cn(
|
||||||
"hover:bg-primary/20 ml-1 rounded p-0.5 transition-colors",
|
"hover:bg-primary/20 ml-1 rounded p-0.5 transition-colors",
|
||||||
headerFilters[column.columnName]?.size > 0 && "text-primary bg-primary/10",
|
(headerFilters[column.columnName]?.size > 0 || headerLikeFilters[column.columnName]) && "text-primary bg-primary/10",
|
||||||
)}
|
)}
|
||||||
title="필터"
|
title="필터"
|
||||||
>
|
>
|
||||||
|
|
@ -5745,7 +5767,7 @@ export const TableListComponent: React.FC<TableListComponentProps> = ({
|
||||||
</button>
|
</button>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent
|
<PopoverContent
|
||||||
className="w-48 p-2"
|
className="w-56 p-2"
|
||||||
align="start"
|
align="start"
|
||||||
onClick={(e) => e.stopPropagation()}
|
onClick={(e) => e.stopPropagation()}
|
||||||
>
|
>
|
||||||
|
|
@ -5754,16 +5776,42 @@ export const TableListComponent: React.FC<TableListComponentProps> = ({
|
||||||
<span className="text-xs font-medium">
|
<span className="text-xs font-medium">
|
||||||
필터: {columnLabels[column.columnName] || column.displayName}
|
필터: {columnLabels[column.columnName] || column.displayName}
|
||||||
</span>
|
</span>
|
||||||
{headerFilters[column.columnName]?.size > 0 && (
|
{(headerFilters[column.columnName]?.size > 0 || headerLikeFilters[column.columnName]) && (
|
||||||
<button
|
<button
|
||||||
onClick={() => clearHeaderFilter(column.columnName)}
|
onClick={() => {
|
||||||
|
clearHeaderFilter(column.columnName);
|
||||||
|
setHeaderLikeFilters((prev) => {
|
||||||
|
const newFilters = { ...prev };
|
||||||
|
delete newFilters[column.columnName];
|
||||||
|
return newFilters;
|
||||||
|
});
|
||||||
|
}}
|
||||||
className="text-destructive text-xs hover:underline"
|
className="text-destructive text-xs hover:underline"
|
||||||
>
|
>
|
||||||
초기화
|
초기화
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
<div className="max-h-48 space-y-1 overflow-y-auto">
|
{/* LIKE 검색 입력 필드 */}
|
||||||
|
<div className="relative">
|
||||||
|
<Search className="text-muted-foreground absolute left-2 top-1/2 h-3 w-3 -translate-y-1/2" />
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
placeholder="검색어 입력 (포함)"
|
||||||
|
value={headerLikeFilters[column.columnName] || ""}
|
||||||
|
onChange={(e) => {
|
||||||
|
setHeaderLikeFilters((prev) => ({
|
||||||
|
...prev,
|
||||||
|
[column.columnName]: e.target.value,
|
||||||
|
}));
|
||||||
|
}}
|
||||||
|
className="border-input bg-background placeholder:text-muted-foreground h-7 w-full rounded-md border pl-7 pr-2 text-xs focus:outline-none focus:ring-1 focus:ring-primary"
|
||||||
|
onClick={(e) => e.stopPropagation()}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{/* 구분선 */}
|
||||||
|
<div className="text-muted-foreground border-t pt-2 text-[10px]">또는 값 선택:</div>
|
||||||
|
<div className="max-h-40 space-y-1 overflow-y-auto">
|
||||||
{columnUniqueValues[column.columnName]?.slice(0, 50).map((val) => {
|
{columnUniqueValues[column.columnName]?.slice(0, 50).map((val) => {
|
||||||
const isSelected = headerFilters[column.columnName]?.has(val);
|
const isSelected = headerFilters[column.columnName]?.has(val);
|
||||||
return (
|
return (
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue