Files
ev-charger-as/frontend/static/js/auth.js
byun 124ad0d165 기능 추가 — 옵저버 계정 및 현황 조회 포털
읽기 전용 옵저버 역할 추가. 신고 현황 확인만 가능하며 모든 쓰기 동작 차단.

- auth.py: require_viewer(admin+observer) 의존성 추가
- auth_router.py: register 엔드포인트에 role 파라미터 추가 (mechanic/observer)
- login.html: 회원가입 시 정비사/옵저버 역할 카드 선택 UI, 역할별 안내문구
- 로그인 후 observer → /pages/observer/dashboard.html 라우팅
- observer/dashboard.html: 통계 카드(상태별 건수) + 신고 현황 테이블(읽기전용)
- observer/reports.html: 상태·충전기ID·충전소명 필터 신고 목록
- accounts.html: 옵저버 필터·생성·승인 대기 역할 표시 추가

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-06-01 15:25:47 +09:00

96 lines
3.4 KiB
JavaScript

const Auth = (() => {
const KEY_TOKEN = 'ev_token';
const KEY_ROLE = 'ev_role';
const KEY_NAME = 'ev_name';
const KEY_ID = 'ev_uid';
function save(token, role, name, id) {
localStorage.setItem(KEY_TOKEN, token);
localStorage.setItem(KEY_ROLE, role);
localStorage.setItem(KEY_NAME, name);
localStorage.setItem(KEY_ID, id);
}
function token() { return localStorage.getItem(KEY_TOKEN); }
function role() { return localStorage.getItem(KEY_ROLE); }
function name() { return localStorage.getItem(KEY_NAME); }
function uid() { return localStorage.getItem(KEY_ID); }
function logout() {
[KEY_TOKEN, KEY_ROLE, KEY_NAME, KEY_ID].forEach(k => localStorage.removeItem(k));
location.href = '/pages/login.html';
}
function require(allowedRoles) {
if (!token()) { logout(); return false; }
if (allowedRoles && !allowedRoles.includes(role())) {
alert('접근 권한이 없습니다.');
history.back();
return false;
}
return true;
}
function renderNav(el) {
if (!el) return;
el.innerHTML = `
<button class="nav-hamburger" onclick="Auth.toggleMobileNav()" aria-label="메뉴">☰</button>
<span class="nav-user">
<span>${name()} <small style="color:var(--accent)">[${role()}]</small></span>
<a onclick="Auth.logout()">로그아웃</a>
</span>`;
// 모바일 오버레이 삽입 (중복 방지)
if (!document.getElementById('mobileNavOverlay')) {
const ov = document.createElement('div');
ov.id = 'mobileNavOverlay';
ov.className = 'mobile-nav-overlay';
ov.addEventListener('click', closeMobileNav);
document.body.appendChild(ov);
}
// 사이드바 링크 클릭 시 드로어 닫기
setTimeout(() => {
document.querySelectorAll('.sidebar a').forEach(a => {
a.addEventListener('click', closeMobileNav);
});
}, 0);
}
function toggleMobileNav() {
const sidebar = document.querySelector('.sidebar');
const overlay = document.getElementById('mobileNavOverlay');
if (!sidebar) return;
const opening = !sidebar.classList.contains('mobile-open');
sidebar.classList.toggle('mobile-open', opening);
overlay?.classList.toggle('show', opening);
}
function closeMobileNav() {
document.querySelector('.sidebar')?.classList.remove('mobile-open');
document.getElementById('mobileNavOverlay')?.classList.remove('show');
}
function statusBadge(status) {
const map = {
pending_approval: '승인대기', pending: '접수', in_progress: '처리중',
done: '완료', waiting: '부품대기', revisit: '재방문', closed: '상황종료',
registered: '등록', reviewing: '검토중', developing: '개발중',
deployed: '배포완료', observer: '옵저버',
};
return `<span class="badge s-${status}">${map[status] || status}</span>`;
}
function costStatusBadge(s) {
const map = { pending:'미처리', billed:'청구완료', waived:'면제', settled:'정산완료' };
return `<span class="badge s-cost-${s}">${map[s] || s}</span>`;
}
function fmtDt(dt) {
if (!dt) return '-';
return new Date(dt).toLocaleString('ko-KR', {year:'numeric',month:'2-digit',day:'2-digit',hour:'2-digit',minute:'2-digit'});
}
return { save, token, role, name, uid, logout, require, renderNav,
toggleMobileNav, closeMobileNav, statusBadge, costStatusBadge, fmtDt };
})();