Implement global table sorting, dashboard UI enhancements, and secret cloud access
This commit is contained in:
46
src/core/tableHandler.ts
Normal file
46
src/core/tableHandler.ts
Normal file
@@ -0,0 +1,46 @@
|
||||
/**
|
||||
* 공통 테이블 핸들러
|
||||
*/
|
||||
|
||||
export type SortDirection = 'asc' | 'desc';
|
||||
|
||||
export interface SortState {
|
||||
key: string;
|
||||
direction: SortDirection;
|
||||
}
|
||||
|
||||
/**
|
||||
* 테이블 헤더에 정렬 이벤트를 바인딩합니다.
|
||||
* @param table 대상 테이블 요소
|
||||
* @param currentState 현재 정렬 상태
|
||||
* @param onSort 정렬 변경 시 호출될 콜백
|
||||
*/
|
||||
export function setupTableSorting(
|
||||
table: HTMLTableElement,
|
||||
currentState: SortState,
|
||||
onSort: (key: string, direction: SortDirection) => void
|
||||
) {
|
||||
const headers = table.querySelectorAll('th[data-sort]');
|
||||
|
||||
headers.forEach(th => {
|
||||
const key = th.getAttribute('data-sort')!;
|
||||
th.classList.add('sortable');
|
||||
|
||||
// 현재 정렬 상태 표시
|
||||
if (currentState.key === key) {
|
||||
th.classList.add(currentState.direction);
|
||||
} else {
|
||||
th.classList.remove('asc', 'desc');
|
||||
}
|
||||
|
||||
th.onclick = () => {
|
||||
let nextDirection: SortDirection = 'asc';
|
||||
|
||||
if (currentState.key === key) {
|
||||
nextDirection = currentState.direction === 'asc' ? 'desc' : 'asc';
|
||||
}
|
||||
|
||||
onSort(key, nextDirection);
|
||||
};
|
||||
});
|
||||
}
|
||||
@@ -71,22 +71,55 @@ export function getAssetChanges(oldAsset: any, newAsset: any, fields: {key: stri
|
||||
}
|
||||
|
||||
/**
|
||||
* 자산 목록 정렬 (방안 C: 구매법인별 -> 자산번호 순)
|
||||
* 자산 목록 정렬 (기본: 법인별 -> 자산번호 순)
|
||||
*/
|
||||
export function sortAssets<T>(list: T[]): T[] {
|
||||
return [...list].sort((a: any, b: any) => {
|
||||
// 1순위: 구매법인 (한글 가나다순)
|
||||
const corpA = String(a.법인 || '').trim();
|
||||
const corpB = String(b.법인 || '').trim();
|
||||
// 1순위: 법인 (가나다순)
|
||||
const corpA = String(a.법인 || a.corp || '').trim();
|
||||
const corpB = String(b.법인 || b.corp || '').trim();
|
||||
if (corpA < corpB) return -1;
|
||||
if (corpA > corpB) return 1;
|
||||
|
||||
// 2순위: 자산번호 (영문/숫자순)
|
||||
const codeA = String(a.자산코드 || a.자산번호 || '').trim();
|
||||
const codeB = String(b.자산코드 || b.자산번호 || '').trim();
|
||||
// 2순위: 자산번호/코드 (영문/숫자순)
|
||||
const codeA = String(a.자산코드 || a.자산번호 || a.id || '').trim();
|
||||
const codeB = String(b.자산코드 || b.자산번호 || b.id || '').trim();
|
||||
if (codeA < codeB) return -1;
|
||||
if (codeA > codeB) return 1;
|
||||
|
||||
return 0;
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 동적 정렬 함수
|
||||
* @param list 정렬할 목록
|
||||
* @param key 정렬 기준 필드
|
||||
* @param direction 정렬 방향 ('asc' | 'desc')
|
||||
*/
|
||||
export function dynamicSort<T>(list: T[], key: string, direction: 'asc' | 'desc'): T[] {
|
||||
return [...list].sort((a: any, b: any) => {
|
||||
let valA = a[key];
|
||||
let valB = b[key];
|
||||
|
||||
// 숫자인 경우 처리
|
||||
if (typeof valA === 'number' && typeof valB === 'number') {
|
||||
return direction === 'asc' ? valA - valB : valB - valA;
|
||||
}
|
||||
|
||||
// 금액 필드 (숫자형 문자열 포함) 처리
|
||||
if (key === '금액' || key === 'price' || key === '수량' || key === 'qty') {
|
||||
const numA = typeof valA === 'number' ? valA : parseInt(String(valA || '0').replace(/[^0-9-]/g, ''), 10);
|
||||
const numB = typeof valB === 'number' ? valB : parseInt(String(valB || '0').replace(/[^0-9-]/g, ''), 10);
|
||||
return direction === 'asc' ? numA - numB : numB - numA;
|
||||
}
|
||||
|
||||
// 문자열 정렬 (기본)
|
||||
valA = String(valA || '').toLowerCase();
|
||||
valB = String(valB || '').toLowerCase();
|
||||
|
||||
if (valA < valB) return direction === 'asc' ? -1 : 1;
|
||||
if (valA > valB) return direction === 'asc' ? 1 : -1;
|
||||
return 0;
|
||||
});
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user