feat(tkuser): 탭 카테고리 그룹핑 + 설비 관리 탭 추가 + tkfb admin 페이지 통합
- tkuser 탭을 5개 카테고리로 그룹핑 (인력/현장/업무/거래/시스템) - 설비 관리 탭 신규 추가 (CRUD, 필터, 상세 보기) - tkfb 사이드바 admin 메뉴 6개를 tkuser 외부 링크로 교체 - tkfb admin HTML 6개를 tkuser 리다이렉트로 변경 - gateway 알림 벨 링크를 tkuser로 변경 - _tkuserBase 헬퍼로 개발/운영 환경 자동 분기 Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -6,6 +6,8 @@ body { font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; back
|
||||
.tab-btn.active { background: #334155; color: white; }
|
||||
.tab-btn:not(.active) { color: #64748b; }
|
||||
.tab-btn:not(.active):hover { background: #e2e8f0; }
|
||||
.tab-group-label { font-size: 0.65rem; font-weight: 600; color: #94a3b8; text-transform: uppercase; letter-spacing: 0.05em; padding: 0 0.25rem; white-space: nowrap; flex-shrink: 0; }
|
||||
.tab-divider { width: 1px; height: 1.25rem; background: #e2e8f0; margin: 0 0.25rem; flex-shrink: 0; }
|
||||
.system-section { border-left: 4px solid; }
|
||||
.system-section.system1 { border-color: #3b82f6; }
|
||||
.system-section.system3 { border-color: #8b5cf6; }
|
||||
@@ -27,6 +29,8 @@ body { font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; back
|
||||
}
|
||||
#tabNav::-webkit-scrollbar { display: none; }
|
||||
#tabNav button { white-space: nowrap; flex-shrink: 0; font-size: 0.8rem; padding: 0.375rem 0.75rem; }
|
||||
.tab-group-label { font-size: 0.6rem; padding: 0 0.125rem; }
|
||||
.tab-divider { margin: 0 0.125rem; }
|
||||
.tab-scroll-hint {
|
||||
position: absolute; right: 0; top: 0; bottom: 0; width: 2rem;
|
||||
background: linear-gradient(to right, transparent, #fff);
|
||||
|
||||
@@ -194,7 +194,7 @@ async function init() {
|
||||
// URL ?tab= 파라미터로 탭 자동 전환 (화이트리스트 + URL 정리)
|
||||
const ALLOWED_TABS = ['users','projects','workplaces','workers','departments',
|
||||
'permissions','issueTypes','tasks','vacations','partners','vendors',
|
||||
'consumables','notificationRecipients'];
|
||||
'consumables','notificationRecipients','equipments'];
|
||||
const urlTab = new URLSearchParams(location.search).get('tab');
|
||||
if (urlTab && ALLOWED_TABS.includes(urlTab)) {
|
||||
const tabBtn = document.querySelector(`.tab-btn[data-tab="${urlTab}"]`);
|
||||
|
||||
253
user-management/web/static/js/tkuser-equipments.js
Normal file
253
user-management/web/static/js/tkuser-equipments.js
Normal file
@@ -0,0 +1,253 @@
|
||||
/* ===== tkuser 설비(Equipment) CRUD ===== */
|
||||
let equipmentsLoaded = false;
|
||||
let equipmentsList = [];
|
||||
let selectedEquipmentIdTkuser = null;
|
||||
|
||||
const EQUIPMENT_STATUS_MAP = {
|
||||
active: { label: '활성', cls: 'bg-green-100 text-green-700' },
|
||||
maintenance: { label: '정비중', cls: 'bg-yellow-100 text-yellow-700' },
|
||||
inactive: { label: '비활성', cls: 'bg-gray-100 text-gray-400' }
|
||||
};
|
||||
|
||||
async function loadEquipmentsTab() {
|
||||
if (equipmentsLoaded) return;
|
||||
equipmentsLoaded = true;
|
||||
if (currentUser && ['admin', 'system'].includes(currentUser.role)) {
|
||||
document.getElementById('btnAddEquipment')?.classList.remove('hidden');
|
||||
}
|
||||
await Promise.all([loadEquipmentFilters(), loadEquipmentsList()]);
|
||||
}
|
||||
|
||||
async function loadEquipmentFilters() {
|
||||
try {
|
||||
// 작업장 필터
|
||||
const wRes = await api('/workplaces');
|
||||
const workplaces = wRes.data || [];
|
||||
const wSel = document.getElementById('equipmentFilterWorkplace');
|
||||
if (wSel) {
|
||||
const current = wSel.value;
|
||||
wSel.innerHTML = '<option value="">전체 작업장</option>' +
|
||||
workplaces.map(w => `<option value="${w.workplace_id}">${escHtml(w.workplace_name)}</option>`).join('');
|
||||
wSel.value = current;
|
||||
}
|
||||
// 유형 필터
|
||||
const tRes = await api('/equipments/types');
|
||||
const types = tRes.data || [];
|
||||
const tSel = document.getElementById('equipmentFilterType');
|
||||
if (tSel) {
|
||||
const current = tSel.value;
|
||||
tSel.innerHTML = '<option value="">전체 유형</option>' +
|
||||
types.map(t => `<option value="${escHtml(t)}">${escHtml(t)}</option>`).join('');
|
||||
tSel.value = current;
|
||||
}
|
||||
// 모달 select 채우기 (작업장)
|
||||
['newEquipmentWorkplaceTkuser', 'editEquipmentWorkplaceTkuser'].forEach(id => {
|
||||
const el = document.getElementById(id);
|
||||
if (el) el.innerHTML = '<option value="">선택</option>' +
|
||||
workplaces.map(w => `<option value="${w.workplace_id}">${escHtml(w.workplace_name)}</option>`).join('');
|
||||
});
|
||||
} catch (e) { /* 필터 로드 실패는 무시 */ }
|
||||
}
|
||||
|
||||
async function loadEquipmentsList() {
|
||||
try {
|
||||
const workplaceId = document.getElementById('equipmentFilterWorkplace')?.value || '';
|
||||
const eqType = document.getElementById('equipmentFilterType')?.value || '';
|
||||
const status = document.getElementById('equipmentFilterStatus')?.value || '';
|
||||
const search = document.getElementById('equipmentSearchTkuser')?.value?.trim() || '';
|
||||
const params = new URLSearchParams();
|
||||
if (workplaceId) params.set('workplace_id', workplaceId);
|
||||
if (eqType) params.set('equipment_type', eqType);
|
||||
if (status) params.set('status', status);
|
||||
if (search) params.set('search', search);
|
||||
const r = await api('/equipments?' + params.toString());
|
||||
equipmentsList = r.data || [];
|
||||
renderEquipmentsListTkuser();
|
||||
} catch (e) {
|
||||
document.getElementById('equipmentsListTkuser').innerHTML = `<div class="text-red-500 text-center py-6"><i class="fas fa-exclamation-triangle text-xl"></i><p class="text-sm mt-2">${e.message}</p></div>`;
|
||||
}
|
||||
}
|
||||
|
||||
function renderEquipmentsListTkuser() {
|
||||
const c = document.getElementById('equipmentsListTkuser');
|
||||
if (!equipmentsList.length) {
|
||||
c.innerHTML = '<p class="text-gray-400 text-center py-4 text-sm">등록된 설비가 없습니다.</p>';
|
||||
return;
|
||||
}
|
||||
const isAdmin = currentUser && ['admin', 'system'].includes(currentUser.role);
|
||||
c.innerHTML = equipmentsList.map(eq => {
|
||||
const st = EQUIPMENT_STATUS_MAP[eq.status] || EQUIPMENT_STATUS_MAP.active;
|
||||
return `<div class="flex items-center justify-between p-2.5 bg-gray-50 rounded-lg hover:bg-gray-100 transition-colors cursor-pointer ${selectedEquipmentIdTkuser === eq.equipment_id ? 'ring-2 ring-indigo-400' : ''}" onclick="selectEquipmentTkuser(${eq.equipment_id})">
|
||||
<div class="flex-1 min-w-0">
|
||||
<div class="text-sm font-medium text-gray-800 truncate">
|
||||
<i class="fas fa-cog mr-1.5 text-gray-400 text-xs"></i>${escHtml(eq.equipment_name)}
|
||||
<span class="px-1.5 py-0.5 rounded text-xs ${st.cls} ml-1">${st.label}</span>
|
||||
</div>
|
||||
<div class="text-xs text-gray-500 flex items-center gap-1.5 mt-0.5">
|
||||
<span>${escHtml(eq.equipment_code)}</span>
|
||||
${eq.workplace_name ? `<span>· ${escHtml(eq.workplace_name)}</span>` : ''}
|
||||
${eq.equipment_type ? `<span>· ${escHtml(eq.equipment_type)}</span>` : ''}
|
||||
</div>
|
||||
</div>
|
||||
${isAdmin ? `<div class="flex gap-1 ml-2 flex-shrink-0">
|
||||
<button onclick="event.stopPropagation(); openEditEquipmentTkuser(${eq.equipment_id})" class="p-1.5 text-slate-500 hover:text-slate-700 hover:bg-slate-200 rounded" title="수정"><i class="fas fa-pen text-xs"></i></button>
|
||||
<button onclick="event.stopPropagation(); deleteEquipmentTkuser(${eq.equipment_id}, '${escHtml(eq.equipment_name).replace(/'/g, "\\'")}')" class="p-1.5 text-red-400 hover:text-red-600 hover:bg-red-100 rounded" title="삭제"><i class="fas fa-trash text-xs"></i></button>
|
||||
</div>` : ''}
|
||||
</div>`;
|
||||
}).join('');
|
||||
}
|
||||
|
||||
async function selectEquipmentTkuser(id) {
|
||||
selectedEquipmentIdTkuser = id;
|
||||
renderEquipmentsListTkuser();
|
||||
try {
|
||||
const r = await api(`/equipments/${id}`);
|
||||
const eq = r.data;
|
||||
renderEquipmentDetailTkuser(eq);
|
||||
document.getElementById('equipmentDetailTkuser').classList.remove('hidden');
|
||||
document.getElementById('equipmentEmptyTkuser').classList.add('hidden');
|
||||
} catch (e) {
|
||||
showToast('상세 조회 실패: ' + e.message, 'error');
|
||||
}
|
||||
}
|
||||
|
||||
function renderEquipmentDetailTkuser(eq) {
|
||||
const st = EQUIPMENT_STATUS_MAP[eq.status] || EQUIPMENT_STATUS_MAP.active;
|
||||
const installDate = eq.installation_date ? eq.installation_date.substring(0, 10) : '-';
|
||||
document.getElementById('equipmentDetailTkuser').innerHTML = `
|
||||
<div class="bg-white rounded-xl shadow-sm p-5">
|
||||
<div class="flex items-center justify-between mb-3">
|
||||
<h3 class="text-lg font-semibold text-gray-800">${escHtml(eq.equipment_name)}</h3>
|
||||
<span class="px-2 py-1 rounded text-xs font-medium ${st.cls}">${st.label}</span>
|
||||
</div>
|
||||
<div class="grid grid-cols-2 gap-3 text-sm">
|
||||
<div><span class="text-gray-500">관리번호:</span> <span class="font-medium">${escHtml(eq.equipment_code)}</span></div>
|
||||
<div><span class="text-gray-500">설비유형:</span> <span class="font-medium">${escHtml(eq.equipment_type) || '-'}</span></div>
|
||||
<div><span class="text-gray-500">작업장:</span> <span class="font-medium">${escHtml(eq.workplace_name) || '-'}</span></div>
|
||||
<div><span class="text-gray-500">제조사:</span> <span class="font-medium">${escHtml(eq.manufacturer) || '-'}</span></div>
|
||||
<div><span class="text-gray-500">모델명:</span> <span class="font-medium">${escHtml(eq.model_name) || '-'}</span></div>
|
||||
<div><span class="text-gray-500">시리얼번호:</span> <span class="font-medium">${escHtml(eq.serial_number) || '-'}</span></div>
|
||||
<div><span class="text-gray-500">설치일:</span> <span class="font-medium">${installDate}</span></div>
|
||||
<div><span class="text-gray-500">공급업체:</span> <span class="font-medium">${escHtml(eq.supplier) || '-'}</span></div>
|
||||
${eq.purchase_price ? `<div><span class="text-gray-500">구매가격:</span> <span class="font-medium">${Number(eq.purchase_price).toLocaleString()}원</span></div>` : ''}
|
||||
${eq.specifications ? `<div class="col-span-2"><span class="text-gray-500">사양:</span> <span class="font-medium">${escHtml(eq.specifications)}</span></div>` : ''}
|
||||
${eq.notes ? `<div class="col-span-2"><span class="text-gray-500">비고:</span> <span class="font-medium">${escHtml(eq.notes)}</span></div>` : ''}
|
||||
</div>
|
||||
</div>`;
|
||||
}
|
||||
|
||||
/* ===== 설비 등록 ===== */
|
||||
function openAddEquipmentTkuser() {
|
||||
// 다음 코드 자동 생성
|
||||
api('/equipments/next-code').then(r => {
|
||||
document.getElementById('newEquipmentCodeTkuser').value = r.data || '';
|
||||
}).catch(() => {});
|
||||
document.getElementById('addEquipmentModalTkuser').classList.remove('hidden');
|
||||
}
|
||||
function closeAddEquipmentTkuser() { document.getElementById('addEquipmentModalTkuser').classList.add('hidden'); document.getElementById('addEquipmentFormTkuser').reset(); }
|
||||
|
||||
async function submitAddEquipmentTkuser(e) {
|
||||
e.preventDefault();
|
||||
const data = {
|
||||
equipment_code: document.getElementById('newEquipmentCodeTkuser').value.trim(),
|
||||
equipment_name: document.getElementById('newEquipmentNameTkuser').value.trim(),
|
||||
equipment_type: document.getElementById('newEquipmentTypeTkuser').value.trim() || null,
|
||||
workplace_id: document.getElementById('newEquipmentWorkplaceTkuser').value || null,
|
||||
manufacturer: document.getElementById('newEquipmentManufacturerTkuser').value.trim() || null,
|
||||
model_name: document.getElementById('newEquipmentModelTkuser').value.trim() || null,
|
||||
serial_number: document.getElementById('newEquipmentSerialTkuser').value.trim() || null,
|
||||
installation_date: document.getElementById('newEquipmentInstallDateTkuser').value || null,
|
||||
supplier: document.getElementById('newEquipmentSupplierTkuser').value.trim() || null,
|
||||
purchase_price: document.getElementById('newEquipmentPriceTkuser').value || null,
|
||||
status: document.getElementById('newEquipmentStatusTkuser').value || 'active',
|
||||
specifications: document.getElementById('newEquipmentSpecsTkuser').value.trim() || null,
|
||||
notes: document.getElementById('newEquipmentNotesTkuser').value.trim() || null,
|
||||
};
|
||||
if (!data.equipment_code) { showToast('관리번호는 필수입니다', 'error'); return; }
|
||||
if (!data.equipment_name) { showToast('설비명은 필수입니다', 'error'); return; }
|
||||
try {
|
||||
await api('/equipments', { method: 'POST', body: JSON.stringify(data) });
|
||||
showToast('설비가 등록되었습니다');
|
||||
closeAddEquipmentTkuser();
|
||||
await loadEquipmentFilters();
|
||||
await loadEquipmentsList();
|
||||
} catch (e) { showToast(e.message, 'error'); }
|
||||
}
|
||||
|
||||
/* ===== 설비 수정 ===== */
|
||||
function openEditEquipmentTkuser(id) {
|
||||
const eq = equipmentsList.find(x => x.equipment_id === id);
|
||||
if (!eq) return;
|
||||
document.getElementById('editEquipmentIdTkuser').value = eq.equipment_id;
|
||||
document.getElementById('editEquipmentCodeTkuser').value = eq.equipment_code;
|
||||
document.getElementById('editEquipmentNameTkuser').value = eq.equipment_name;
|
||||
document.getElementById('editEquipmentTypeTkuser').value = eq.equipment_type || '';
|
||||
document.getElementById('editEquipmentWorkplaceTkuser').value = eq.workplace_id || '';
|
||||
document.getElementById('editEquipmentManufacturerTkuser').value = eq.manufacturer || '';
|
||||
document.getElementById('editEquipmentModelTkuser').value = eq.model_name || '';
|
||||
document.getElementById('editEquipmentSerialTkuser').value = eq.serial_number || '';
|
||||
document.getElementById('editEquipmentInstallDateTkuser').value = eq.installation_date ? eq.installation_date.substring(0, 10) : '';
|
||||
document.getElementById('editEquipmentSupplierTkuser').value = eq.supplier || '';
|
||||
document.getElementById('editEquipmentPriceTkuser').value = eq.purchase_price || '';
|
||||
document.getElementById('editEquipmentStatusTkuser').value = eq.status || 'active';
|
||||
document.getElementById('editEquipmentSpecsTkuser').value = eq.specifications || '';
|
||||
document.getElementById('editEquipmentNotesTkuser').value = eq.notes || '';
|
||||
document.getElementById('editEquipmentModalTkuser').classList.remove('hidden');
|
||||
}
|
||||
function closeEditEquipmentTkuser() { document.getElementById('editEquipmentModalTkuser').classList.add('hidden'); }
|
||||
|
||||
async function submitEditEquipmentTkuser(e) {
|
||||
e.preventDefault();
|
||||
const id = document.getElementById('editEquipmentIdTkuser').value;
|
||||
const data = {
|
||||
equipment_code: document.getElementById('editEquipmentCodeTkuser').value.trim(),
|
||||
equipment_name: document.getElementById('editEquipmentNameTkuser').value.trim(),
|
||||
equipment_type: document.getElementById('editEquipmentTypeTkuser').value.trim() || null,
|
||||
workplace_id: document.getElementById('editEquipmentWorkplaceTkuser').value || null,
|
||||
manufacturer: document.getElementById('editEquipmentManufacturerTkuser').value.trim() || null,
|
||||
model_name: document.getElementById('editEquipmentModelTkuser').value.trim() || null,
|
||||
serial_number: document.getElementById('editEquipmentSerialTkuser').value.trim() || null,
|
||||
installation_date: document.getElementById('editEquipmentInstallDateTkuser').value || null,
|
||||
supplier: document.getElementById('editEquipmentSupplierTkuser').value.trim() || null,
|
||||
purchase_price: document.getElementById('editEquipmentPriceTkuser').value || null,
|
||||
status: document.getElementById('editEquipmentStatusTkuser').value || 'active',
|
||||
specifications: document.getElementById('editEquipmentSpecsTkuser').value.trim() || null,
|
||||
notes: document.getElementById('editEquipmentNotesTkuser').value.trim() || null,
|
||||
};
|
||||
try {
|
||||
await api(`/equipments/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
||||
showToast('수정되었습니다');
|
||||
closeEditEquipmentTkuser();
|
||||
await loadEquipmentFilters();
|
||||
await loadEquipmentsList();
|
||||
if (selectedEquipmentIdTkuser == id) selectEquipmentTkuser(id);
|
||||
} catch (e) { showToast(e.message, 'error'); }
|
||||
}
|
||||
|
||||
/* ===== 설비 삭제 ===== */
|
||||
async function deleteEquipmentTkuser(id, name) {
|
||||
if (!confirm(`"${name}" 설비를 삭제하시겠습니까?`)) return;
|
||||
try {
|
||||
await api(`/equipments/${id}`, { method: 'DELETE' });
|
||||
showToast('삭제 완료');
|
||||
await loadEquipmentsList();
|
||||
if (selectedEquipmentIdTkuser === id) {
|
||||
document.getElementById('equipmentDetailTkuser').classList.add('hidden');
|
||||
document.getElementById('equipmentEmptyTkuser').classList.remove('hidden');
|
||||
selectedEquipmentIdTkuser = null;
|
||||
}
|
||||
} catch (e) { showToast(e.message, 'error'); }
|
||||
}
|
||||
|
||||
/* ===== 필터 ===== */
|
||||
let equipmentSearchTimeout;
|
||||
function filterEquipmentsTkuser() {
|
||||
clearTimeout(equipmentSearchTimeout);
|
||||
equipmentSearchTimeout = setTimeout(loadEquipmentsList, 300);
|
||||
}
|
||||
|
||||
// 검색/필터 이벤트 + 모달 폼 이벤트
|
||||
document.addEventListener('DOMContentLoaded', () => {
|
||||
document.getElementById('addEquipmentFormTkuser')?.addEventListener('submit', submitAddEquipmentTkuser);
|
||||
document.getElementById('editEquipmentFormTkuser')?.addEventListener('submit', submitEditEquipmentTkuser);
|
||||
});
|
||||
@@ -20,7 +20,7 @@ function switchTab(name, event) {
|
||||
const headerInner = document.getElementById('headerInner');
|
||||
const wideClass = 'max-w-[1600px]';
|
||||
const defaultClass = 'max-w-7xl';
|
||||
if (name === 'workplaces' || name === 'tasks' || name === 'vacations') {
|
||||
if (name === 'workplaces' || name === 'tasks' || name === 'vacations' || name === 'equipments') {
|
||||
[mainEl, navInner, headerInner].forEach(el => { el.classList.remove(defaultClass); el.classList.add(wideClass); });
|
||||
} else {
|
||||
[mainEl, navInner, headerInner].forEach(el => { el.classList.remove(wideClass); el.classList.add(defaultClass); });
|
||||
@@ -37,4 +37,5 @@ function switchTab(name, event) {
|
||||
if (name === 'vendors' && !vendorsLoaded) loadVendorsTab();
|
||||
if (name === 'consumables' && !consumablesLoaded) loadConsumablesTab();
|
||||
if (name === 'notificationRecipients' && !nrLoaded) loadNotificationRecipientsTab();
|
||||
if (name === 'equipments' && !equipmentsLoaded) loadEquipmentsTab();
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user