소스 수정
This commit is contained in:
@@ -1,67 +1,72 @@
|
||||
<template>
|
||||
<aside class="sidebar">
|
||||
<aside :class="['sidebar', { collapsed: isCollapsed }]">
|
||||
<div class="sidebar-header">
|
||||
<div class="sidebar-logo">
|
||||
<span class="logo-icon">📡</span>
|
||||
<button class="toggle-btn" @click="toggle" :title="isCollapsed ? '메뉴 열기' : '메뉴 닫기'">
|
||||
<span class="hamburger">☰</span>
|
||||
</button>
|
||||
<div class="sidebar-logo" v-show="!isCollapsed">
|
||||
<span>OSOLIT Monitor</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<nav class="sidebar-nav">
|
||||
<NuxtLink to="/" class="nav-item" :class="{ active: route.path === '/' }">
|
||||
<NuxtLink to="/" class="nav-item" :class="{ active: route.path === '/' }" :title="isCollapsed ? '대시보드' : ''">
|
||||
<span class="icon">📊</span>
|
||||
<span>대시보드</span>
|
||||
<span class="label">대시보드</span>
|
||||
</NuxtLink>
|
||||
|
||||
<div class="nav-group-title">네트워크</div>
|
||||
<div class="nav-group-title" v-show="!isCollapsed">네트워크</div>
|
||||
<div class="nav-divider" v-show="isCollapsed"></div>
|
||||
|
||||
<NuxtLink to="/network/pubnet" class="nav-item nav-sub-item" :class="{ active: route.path === '/network/pubnet' }">
|
||||
<NuxtLink to="/network/pubnet" class="nav-item nav-sub-item" :class="{ active: route.path === '/network/pubnet' }" :title="isCollapsed ? 'Public Network' : ''">
|
||||
<span class="icon">🌐</span>
|
||||
<span>Public Network</span>
|
||||
<span class="label">Public Network</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/network/privnet" class="nav-item nav-sub-item" :class="{ active: route.path === '/network/privnet' }">
|
||||
<NuxtLink to="/network/privnet" class="nav-item nav-sub-item" :class="{ active: route.path === '/network/privnet' }" :title="isCollapsed ? 'Private Network' : ''">
|
||||
<span class="icon">🔒</span>
|
||||
<span>Private Network</span>
|
||||
<span class="label">Private Network</span>
|
||||
</NuxtLink>
|
||||
|
||||
<div class="nav-group-title">서버</div>
|
||||
<div class="nav-group-title" v-show="!isCollapsed">서버</div>
|
||||
<div class="nav-divider" v-show="isCollapsed"></div>
|
||||
|
||||
<NuxtLink to="/server/list" class="nav-item nav-sub-item" :class="{ active: route.path === '/server/list' }">
|
||||
<NuxtLink to="/server/list" class="nav-item nav-sub-item" :class="{ active: route.path === '/server/list' }" :title="isCollapsed ? 'Server Targets' : ''">
|
||||
<span class="icon">🖥️</span>
|
||||
<span>Server Targets</span>
|
||||
<span class="label">Server Targets</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/server/history" class="nav-item nav-sub-item" :class="{ active: route.path === '/server/history' }">
|
||||
<NuxtLink to="/server/history" class="nav-item nav-sub-item" :class="{ active: route.path === '/server/history' }" :title="isCollapsed ? 'Server Status' : ''">
|
||||
<span class="icon">📈</span>
|
||||
<span>Server Status</span>
|
||||
<span class="label">Server Status</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/settings/thresholds" class="nav-item nav-sub-item" :class="{ active: route.path === '/settings/thresholds' }">
|
||||
<NuxtLink to="/settings/thresholds" class="nav-item nav-sub-item" :class="{ active: route.path === '/settings/thresholds' }" :title="isCollapsed ? 'Thresholds' : ''">
|
||||
<span class="icon">⚙️</span>
|
||||
<span>Thresholds</span>
|
||||
<span class="label">Thresholds</span>
|
||||
</NuxtLink>
|
||||
|
||||
<div class="nav-group-title">이상감지</div>
|
||||
<div class="nav-group-title" v-show="!isCollapsed">이상감지</div>
|
||||
<div class="nav-divider" v-show="isCollapsed"></div>
|
||||
|
||||
<NuxtLink to="/anomaly/short-term" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/short-term' }">
|
||||
<NuxtLink to="/anomaly/short-term" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/short-term' }" :title="isCollapsed ? '단기 변화율' : ''">
|
||||
<span class="icon">⚡</span>
|
||||
<span>단기 변화율</span>
|
||||
<span class="label">단기 변화율</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/anomaly/zscore" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/zscore' }">
|
||||
<NuxtLink to="/anomaly/zscore" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/zscore' }" :title="isCollapsed ? 'Z-Score 분석' : ''">
|
||||
<span class="icon">📊</span>
|
||||
<span>Z-Score 분석</span>
|
||||
<span class="label">Z-Score 분석</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/anomaly/baseline" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/baseline' }">
|
||||
<NuxtLink to="/anomaly/baseline" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/baseline' }" :title="isCollapsed ? '시간대별 베이스라인' : ''">
|
||||
<span class="icon">🕐</span>
|
||||
<span>시간대별 베이스라인</span>
|
||||
<span class="label">시간대별 베이스라인</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/anomaly/trend" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/trend' }">
|
||||
<NuxtLink to="/anomaly/trend" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/trend' }" :title="isCollapsed ? '추세 분석' : ''">
|
||||
<span class="icon">📉</span>
|
||||
<span>추세 분석</span>
|
||||
<span class="label">추세 분석</span>
|
||||
</NuxtLink>
|
||||
</nav>
|
||||
</aside>
|
||||
@@ -69,4 +74,129 @@
|
||||
|
||||
<script setup lang="ts">
|
||||
const route = useRoute()
|
||||
const { isCollapsed, toggle } = useSidebar()
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.sidebar {
|
||||
width: 200px;
|
||||
min-width: 200px;
|
||||
background: var(--bg-secondary);
|
||||
border-right: 1px solid var(--border-color);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
transition: all 0.2s ease;
|
||||
}
|
||||
|
||||
.sidebar.collapsed {
|
||||
width: 50px;
|
||||
min-width: 50px;
|
||||
}
|
||||
|
||||
.sidebar-header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding: 12px;
|
||||
border-bottom: 1px solid var(--border-color);
|
||||
}
|
||||
|
||||
.toggle-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
cursor: pointer;
|
||||
padding: 6px 8px;
|
||||
border-radius: 6px;
|
||||
transition: background 0.2s;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.toggle-btn:hover {
|
||||
background: var(--bg-tertiary, #f1f5f9);
|
||||
}
|
||||
|
||||
.hamburger {
|
||||
font-size: 18px;
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.sidebar-logo {
|
||||
font-size: 14px;
|
||||
font-weight: 600;
|
||||
color: var(--text-primary);
|
||||
white-space: nowrap;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.sidebar-nav {
|
||||
flex: 1;
|
||||
padding: 8px;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
.nav-item {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
padding: 10px 12px;
|
||||
border-radius: 8px;
|
||||
color: var(--text-secondary);
|
||||
text-decoration: none;
|
||||
transition: all 0.15s;
|
||||
margin-bottom: 2px;
|
||||
}
|
||||
|
||||
.collapsed .nav-item {
|
||||
justify-content: center;
|
||||
padding: 10px 8px;
|
||||
}
|
||||
|
||||
.nav-item:hover {
|
||||
background: var(--bg-tertiary, #f1f5f9);
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.nav-item.active {
|
||||
background: var(--accent-bg, #eff6ff);
|
||||
color: var(--accent-color, #3b82f6);
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.nav-item .icon {
|
||||
font-size: 16px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.nav-item .label {
|
||||
font-size: 13px;
|
||||
white-space: nowrap;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.collapsed .nav-item .label {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.nav-group-title {
|
||||
font-size: 11px;
|
||||
font-weight: 600;
|
||||
color: var(--text-muted);
|
||||
padding: 12px 12px 6px;
|
||||
text-transform: uppercase;
|
||||
letter-spacing: 0.5px;
|
||||
}
|
||||
|
||||
.nav-divider {
|
||||
height: 1px;
|
||||
background: var(--border-color);
|
||||
margin: 8px 6px;
|
||||
}
|
||||
|
||||
.nav-sub-item {
|
||||
padding-left: 16px;
|
||||
}
|
||||
|
||||
.collapsed .nav-sub-item {
|
||||
padding-left: 8px;
|
||||
}
|
||||
</style>
|
||||
|
||||
Reference in New Issue
Block a user