시스템 모니터

This commit is contained in:
2025-12-28 12:03:48 +09:00
parent dbae6649bc
commit a871ec8008
73 changed files with 21354 additions and 1 deletions

View File

@@ -0,0 +1,213 @@
<template>
<div class="control-panel">
<div class="control-row">
<span class="control-label">조회 간격:</span>
<div class="interval-buttons">
<button
v-for="min in [1, 2, 3, 4, 5]"
:key="min"
:class="['interval-btn', { active: autoRefresh && interval === min }]"
@click="selectInterval(min)"
>
{{ min }}
</button>
</div>
<div class="control-divider"></div>
<button
:class="['auto-refresh-btn', { active: autoRefresh }]"
@click="toggleAutoRefresh"
>
{{ autoRefresh ? '⏸ 자동갱신 ON' : '▶ 자동갱신 OFF' }}
</button>
<div class="last-fetch-info">
마지막 조회: <span class="last-fetch-time">{{ relativeTime }}</span>
</div>
</div>
<div class="control-row">
<span class="control-label">특정시간:</span>
<input
type="datetime-local"
v-model="selectedDatetime"
class="datetime-input"
:disabled="autoRefresh || fetchState !== 'idle'"
:class="{ disabled: autoRefresh || fetchState !== 'idle' }"
/>
<button
class="refresh-btn"
:disabled="autoRefresh || fetchState !== 'idle'"
:class="{
disabled: autoRefresh,
loading: fetchState === 'loading',
success: fetchState === 'success'
}"
@click="doRefresh"
>
<span v-if="fetchState === 'loading'" class="loading-spinner"></span>
{{ buttonText }}
</button>
<span v-if="autoRefresh" class="hint-text">자동갱신 OFF 특정시간 조회 가능</span>
</div>
</div>
</template>
<script setup lang="ts">
const props = defineProps<{
interval: number
autoRefresh: boolean
lastFetchTime: Date | null
fetchState: 'idle' | 'loading' | 'success'
}>()
const emit = defineEmits<{
(e: 'update:interval', value: number): void
(e: 'update:autoRefresh', value: boolean): void
(e: 'fetchAt', datetime: string): void
(e: 'refresh'): void
}>()
// props destructure for template
const { interval, autoRefresh, fetchState } = toRefs(props)
const selectedDatetime = ref('')
const relativeTime = ref('-')
const buttonText = computed(() => {
switch (props.fetchState) {
case 'loading': return '조회 중...'
case 'success': return '✓ 완료'
default: return '조회'
}
})
function getCurrentDatetimeLocal(): string {
const now = new Date()
const y = now.getFullYear()
const m = String(now.getMonth() + 1).padStart(2, '0')
const d = String(now.getDate()).padStart(2, '0')
const h = String(now.getHours()).padStart(2, '0')
const min = String(now.getMinutes()).padStart(2, '0')
return `${y}-${m}-${d}T${h}:${min}`
}
function getRelativeTime(date: Date | null): string {
if (!date) return '-'
const now = new Date()
const diff = Math.floor((now.getTime() - date.getTime()) / 1000)
if (diff < 5) return '방금 전'
if (diff < 60) return `${diff}초 전`
if (diff < 3600) return `${Math.floor(diff / 60)}분 전`
if (diff < 86400) return `${Math.floor(diff / 3600)}시간 전`
return `${Math.floor(diff / 86400)}일 전`
}
function updateRelativeTime() {
relativeTime.value = getRelativeTime(props.lastFetchTime)
}
let timeTimer: ReturnType<typeof window.setInterval> | null = null
onMounted(() => {
updateRelativeTime()
timeTimer = window.setInterval(updateRelativeTime, 1000)
})
onUnmounted(() => {
if (timeTimer) {
window.clearInterval(timeTimer)
}
})
watch(() => props.lastFetchTime, () => {
updateRelativeTime()
})
// 조회간격 선택 → 자동갱신 ON + 특정시간 초기화
function selectInterval(min: number) {
selectedDatetime.value = ''
emit('update:interval', min)
if (!props.autoRefresh) {
emit('update:autoRefresh', true)
}
emit('refresh')
}
function toggleAutoRefresh() {
const newValue = !props.autoRefresh
emit('update:autoRefresh', newValue)
if (newValue) {
// 자동갱신 ON 시 특정시간 초기화 + 즉시 조회
selectedDatetime.value = ''
emit('refresh')
} else {
// 자동갱신 OFF 시 현재시간을 기본값으로 설정
selectedDatetime.value = getCurrentDatetimeLocal()
}
}
function doRefresh() {
// 자동갱신 OFF일 때만 동작
if (props.autoRefresh || props.fetchState !== 'idle') return
// 특정시간 입력된 경우 → 해당 시간 조회
if (selectedDatetime.value) {
emit('fetchAt', selectedDatetime.value.replace('T', ' '))
} else {
// 특정시간 미입력 → 현재 시점 조회
emit('refresh')
}
}
</script>
<style scoped>
.datetime-input.disabled {
opacity: 0.5;
cursor: not-allowed;
}
.refresh-btn.disabled {
opacity: 0.5;
cursor: not-allowed;
}
.refresh-btn.loading {
cursor: wait;
background: var(--btn-primary-bg);
}
.refresh-btn.success {
background: var(--success-border);
border-color: var(--success-border);
}
.loading-spinner {
display: inline-block;
width: 12px;
height: 12px;
border: 2px solid #fff;
border-top-color: transparent;
border-radius: 50%;
animation: spin 0.8s linear infinite;
margin-right: 6px;
vertical-align: middle;
}
@keyframes spin {
to {
transform: rotate(360deg);
}
}
.hint-text {
font-size: 12px;
color: var(--text-dim);
font-style: italic;
margin-left: 8px;
}
</style>

View File

@@ -0,0 +1,84 @@
<template>
<div :class="['network-card', statusClass]" @click="goToList">
<div class="card-icon">{{ icon }}</div>
<div class="card-title">{{ title }}</div>
<div class="card-status" v-if="status && status.last_checked_at">
<span class="status-icon">{{ status.is_healthy ? '✅' : '❌' }}</span>
<span class="status-text">{{ status.is_healthy ? '정상' : '오류' }}</span>
</div>
<div class="card-status" v-else>
<span class="status-icon"></span>
<span class="status-text">대기</span>
</div>
<div class="card-time" v-if="status && status.last_checked_at">
{{ formatTimeAgo(status.last_checked_at) }}
</div>
</div>
</template>
<script setup lang="ts">
const props = defineProps<{
type: 'pubnet' | 'privnet'
title: string
icon: string
status: {
is_healthy: number
last_checked_at: string | null
last_target_name: string | null
} | null
}>()
const router = useRouter()
const statusClass = computed(() => {
if (!props.status || !props.status.last_checked_at) return 'pending'
return props.status.is_healthy ? 'healthy' : 'unhealthy'
})
function goToList() {
router.push(`/network/${props.type}`)
}
function formatTimeAgo(datetime: string | null): string {
if (!datetime) return '-'
const now = new Date()
const then = new Date(datetime.replace(' ', 'T') + '+09:00')
const diff = Math.floor((now.getTime() - then.getTime()) / 1000)
if (diff < 60) return `${diff}초 전`
if (diff < 3600) return `${Math.floor(diff / 60)}분 전`
if (diff < 86400) return `${Math.floor(diff / 3600)}시간 전`
return `${Math.floor(diff / 86400)}일 전`
}
</script>
<style scoped>
.network-card {
display: flex;
flex-direction: column;
align-items: center;
padding: 16px 12px;
background: var(--bg-secondary);
border: 1px solid var(--border-color);
border-radius: 12px;
cursor: pointer;
transition: all 0.2s;
text-align: center;
}
.network-card:hover {
background: var(--bg-tertiary, #f8fafc);
transform: translateY(-2px);
box-shadow: 0 4px 12px rgba(0,0,0,0.08);
}
.network-card.healthy { border-top: 4px solid #22c55e; }
.network-card.unhealthy { border-top: 4px solid #ef4444; }
.network-card.pending { border-top: 4px solid #9ca3af; }
.card-icon { font-size: 28px; margin-bottom: 8px; }
.card-title { font-size: 14px; font-weight: 600; color: var(--text-primary); margin-bottom: 10px; }
.card-status { display: flex; align-items: center; justify-content: center; gap: 6px; margin-bottom: 4px; }
.status-icon { font-size: 16px; }
.status-text { font-size: 15px; font-weight: 500; color: var(--text-secondary); }
.card-time { font-size: 12px; color: var(--text-muted); }
</style>

View File

@@ -0,0 +1,319 @@
<template>
<div class="server-portlet">
<div class="portlet-header">
<h2 class="portlet-title">🖥 서버 현황</h2>
<div class="summary-badges">
<span class="badge">
서버 {{ summary.servers.total }}
<span class="level-counts">
<span class="lv">🟢{{ summary.servers.normal }}</span>
<span class="lv">🟡{{ summary.servers.warning }}</span>
<span class="lv">🟠{{ summary.servers.critical }}</span>
<span class="lv">🔴{{ summary.servers.danger }}</span>
<span class="lv">{{ summary.servers.offline }}</span>
</span>
</span>
<span class="divider">|</span>
<span class="badge">
컨테이너 {{ summary.containers.total }}
<span class="level-counts">
<span class="lv">🟢{{ summary.containers.normal }}</span>
<span class="lv">🟡{{ summary.containers.warning }}</span>
<span class="lv">🟠{{ summary.containers.critical }}</span>
<span class="lv">🔴{{ (summary.containers.danger || 0) + summary.containers.stopped }}</span>
</span>
</span>
</div>
</div>
<div class="server-grid">
<!-- 서버 유닛들 -->
<div
v-for="server in servers"
:key="server.target_id"
:class="['server-unit', server.level]"
>
<!-- 서버 정보 (왼쪽) -->
<div class="server-info" @dblclick="goToServerStatus(server.target_id)">
<div class="server-name">
<span class="level-icon">{{ levelIcon(server.level) }}</span>
<span class="name">{{ server.server_name }}</span>
<span class="container-count" v-if="server.level !== 'offline'">📦{{ server.container_summary.total }}</span>
</div>
<template v-if="server.level !== 'offline'">
<div class="metric-row">
<span class="metric-label">CPU</span>
<div class="progress-bar">
<div :class="['progress-fill', server.cpu_level]" :style="{ width: (server.cpu_percent || 0) + '%' }"></div>
</div>
<span :class="['metric-value', server.cpu_level]">{{ server.cpu_percent?.toFixed(0) || '-' }}</span>
</div>
<div class="metric-row">
<span class="metric-label">MEM</span>
<div class="progress-bar">
<div :class="['progress-fill', server.memory_level]" :style="{ width: (server.memory_percent || 0) + '%' }"></div>
</div>
<span :class="['metric-value', server.memory_level]">{{ server.memory_percent?.toFixed(0) || '-' }}</span>
</div>
<div class="metric-row">
<span class="metric-label">DISK</span>
<div class="progress-bar">
<div :class="['progress-fill', server.disk_level]" :style="{ width: (server.disk_percent || 0) + '%' }"></div>
</div>
<span :class="['metric-value', server.disk_level]">{{ server.disk_percent?.toFixed(0) || '-' }}</span>
</div>
</template>
<template v-else>
<div class="offline-info">
<div class="offline-text">오프라인</div>
<div class="offline-time">{{ formatTimeAgo(server.last_collected) }}</div>
</div>
</template>
</div>
<!-- 컨테이너 영역 (오른쪽) -->
<div class="container-area" v-if="server.level !== 'offline'">
<div
v-for="container in sortContainers(server.containers)"
:key="server.target_id + '-' + container.name"
:class="['container-card', container.level]"
@click="goToServerStatus(server.target_id, container.name)"
>
<div class="card-header">
<div class="card-name">
<span class="card-level">{{ levelIcon(container.level) }}</span>
<span class="name">{{ container.name }}</span>
</div>
<span class="card-uptime" v-if="container.status === 'running'">{{ container.uptime || '-' }}</span>
</div>
<template v-if="container.status === 'running'">
<div class="card-metrics">
<div class="card-metric">
<span class="label">CPU</span>
<div class="mini-bar">
<div :class="['mini-fill', getContainerCpuLevel(container)]" :style="{ width: (container.cpu_percent || 0) + '%' }"></div>
</div>
<span class="value">{{ container.cpu_percent?.toFixed(0) || '-' }}%</span>
</div>
<div class="card-metric">
<span class="label">MEM</span>
<div class="mini-bar">
<div :class="['mini-fill', getContainerMemLevel(container)]" :style="{ width: getMemPercent(container) + '%' }"></div>
</div>
<span class="value">{{ formatMemoryShort(container.memory_usage) }}</span>
</div>
<div class="card-metric">
<span class="label">RX</span>
<span class="value net">{{ formatNetworkShort(container.network_rx) }}</span>
</div>
<div class="card-metric">
<span class="label">TX</span>
<span class="value net">{{ formatNetworkShort(container.network_tx) }}</span>
</div>
</div>
</template>
<template v-else>
<div class="card-stopped">{{ container.status }}</div>
</template>
</div>
<!-- 컨테이너 없음 -->
<div class="no-container" v-if="server.containers.length === 0">
<span>컨테이너 없음</span>
</div>
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
interface ContainerStatus {
name: string
status: string
level: string
cpu_percent: number | null
memory_usage: number | null
memory_limit: number | null
uptime: string | null
network_rx: number | null
network_tx: number | null
}
interface ServerStatus {
target_id: number
server_name: string
level: string
cpu_percent: number | null
cpu_level: string
memory_percent: number | null
memory_level: string
disk_percent: number | null
disk_level: string
last_collected: string | null
containers: ContainerStatus[]
container_summary: { total: number; normal: number; warning: number; critical: number; stopped: number }
}
interface Summary {
servers: { total: number; normal: number; warning: number; critical: number; danger: number; offline: number }
containers: { total: number; normal: number; warning: number; critical: number; danger: number; stopped: number }
}
const props = defineProps<{
servers: ServerStatus[]
summary: Summary
}>()
const emit = defineEmits<{
(e: 'navigate', path: string): void
}>()
const levelPriority: Record<string, number> = { stopped: 3, critical: 2, danger: 2, warning: 1, normal: 0 }
function sortContainers(containers: ContainerStatus[]) {
return [...containers].sort((a, b) => a.name.localeCompare(b.name))
}
function goToServerStatus(targetId: number, containerName?: string) {
let path = `/server/history?target=${targetId}`
if (containerName) path += `&container=${containerName}`
emit('navigate', path)
}
function levelIcon(level: string): string {
const icons: Record<string, string> = { normal: '🟢', warning: '🟡', critical: '🟠', danger: '🔴', offline: '⚫', stopped: '🔴' }
return icons[level] || '⚪'
}
function getContainerCpuLevel(c: ContainerStatus): string {
if (c.cpu_percent === null) return 'normal'
if (c.cpu_percent >= 95) return 'danger'
if (c.cpu_percent >= 90) return 'critical'
if (c.cpu_percent >= 80) return 'warning'
return 'normal'
}
function getContainerMemLevel(c: ContainerStatus): string {
const pct = getMemPercent(c)
if (pct >= 95) return 'danger'
if (pct >= 90) return 'critical'
if (pct >= 80) return 'warning'
return 'normal'
}
function getMemPercent(c: ContainerStatus): number {
if (!c.memory_limit || !c.memory_usage) return 0
return (c.memory_usage / c.memory_limit) * 100
}
function formatMemoryShort(bytes: number | null): string {
if (bytes === null || bytes === undefined) return '-'
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(0)}K`
if (bytes < 1024 * 1024 * 1024) return `${(bytes / 1024 / 1024).toFixed(0)}M`
return `${(bytes / 1024 / 1024 / 1024).toFixed(1)}G`
}
function formatNetworkShort(bytes: number | null): string {
if (bytes === null || bytes === undefined) return '-'
if (bytes < 1024) return `${bytes}B`
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(0)}K`
if (bytes < 1024 * 1024 * 1024) return `${(bytes / 1024 / 1024).toFixed(1)}M`
return `${(bytes / 1024 / 1024 / 1024).toFixed(2)}G`
}
function formatTimeAgo(datetime: string | null): string {
if (!datetime) return '-'
const now = new Date()
const then = new Date(datetime.replace(' ', 'T') + '+09:00')
const diff = Math.floor((now.getTime() - then.getTime()) / 1000)
if (diff < 60) return `${diff}초 전`
if (diff < 3600) return `${Math.floor(diff / 60)}분 전`
if (diff < 86400) return `${Math.floor(diff / 3600)}시간 전`
return `${Math.floor(diff / 86400)}일 전`
}
</script>
<style scoped>
.server-portlet { background: var(--bg-secondary); border: 1px solid var(--border-color); border-radius: 12px; overflow: hidden; }
.portlet-header { display: flex; justify-content: space-between; align-items: center; padding: 14px 18px; border-bottom: 1px solid var(--border-color); background: var(--bg-tertiary, #f8fafc); }
.portlet-title { margin: 0; font-size: 18px; font-weight: 600; color: var(--text-primary); }
.summary-badges { display: flex; align-items: center; gap: 12px; font-size: 14px; color: var(--text-secondary); }
.divider { color: var(--border-color); }
.level-counts { margin-left: 8px; }
.level-counts .lv { margin-right: 6px; }
/* 서버 그리드 - flex-wrap */
.server-grid { display: flex; flex-wrap: wrap; gap: 12px; padding: 16px; align-items: flex-start; }
/* 서버 유닛 */
.server-unit { display: flex; background: var(--bg-primary); border: 1px solid var(--border-color); border-radius: 10px; overflow: hidden; }
.server-unit.warning { border-left: 3px solid #eab308; }
.server-unit.critical { border-left: 3px solid #f97316; }
.server-unit.danger { border-left: 3px solid #ef4444; }
.server-unit.offline { border-left: 3px solid #6b7280; opacity: 0.7; }
/* 서버 정보 (왼쪽) */
.server-info { width: 150px; min-width: 150px; padding: 12px; border-right: 1px solid var(--border-color); cursor: pointer; }
.server-info:hover { background: var(--bg-tertiary, #f8fafc); }
.server-name { display: flex; align-items: center; gap: 6px; margin-bottom: 10px; }
.server-name .level-icon { font-size: 12px; flex-shrink: 0; }
.server-name .name { font-size: 15px; font-weight: 600; color: var(--text-primary); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; flex: 1; }
.server-name .container-count { font-size: 12px; color: var(--text-muted); flex-shrink: 0; }
/* 서버 메트릭 */
.metric-row { display: flex; align-items: center; gap: 6px; margin-bottom: 6px; }
.metric-label { font-size: 11px; font-weight: 500; color: var(--text-muted); width: 28px; }
.progress-bar { flex: 1; height: 6px; background: var(--bg-tertiary, #e5e7eb); border-radius: 3px; overflow: hidden; }
.progress-fill { height: 100%; border-radius: 3px; transition: width 0.3s; }
.progress-fill.normal { background: #22c55e; }
.progress-fill.warning { background: #eab308; }
.progress-fill.critical { background: #f97316; }
.progress-fill.danger { background: #ef4444; }
.metric-value { font-size: 12px; font-weight: 600; width: 24px; text-align: right; }
.metric-value.normal { color: #16a34a; }
.metric-value.warning { color: #ca8a04; }
.metric-value.critical { color: #ea580c; }
.metric-value.danger { color: #dc2626; }
.offline-info { text-align: center; padding: 16px 0; color: var(--text-muted); }
.offline-text { font-size: 14px; margin-bottom: 4px; }
.offline-time { font-size: 12px; opacity: 0.7; }
/* 컨테이너 영역 */
.container-area { display: flex; flex-wrap: wrap; gap: 8px; padding: 10px; align-content: flex-start; min-width: 100px; }
/* 컨테이너 카드 */
.container-card { width: 200px; padding: 10px; border-radius: 8px; border: 1px solid var(--border-color); background: var(--bg-secondary); cursor: pointer; transition: all 0.15s; overflow: hidden; }
.container-card:hover { transform: translateY(-1px); box-shadow: 0 2px 8px rgba(0,0,0,0.08); }
.container-card.normal { background: #f0fdf4; border-color: #86efac; }
.container-card.warning { background: #fefce8; border-color: #fde047; }
.container-card.critical { background: #fff7ed; border-color: #fdba74; }
.container-card.danger { background: #fef2f2; border-color: #fca5a5; }
.container-card.stopped { background: #fef2f2; border-color: #fca5a5; }
.card-header { display: flex; align-items: center; justify-content: space-between; margin-bottom: 8px; overflow: hidden; }
.card-name { display: flex; align-items: center; gap: 4px; flex: 1; min-width: 0; overflow: hidden; }
.card-name .card-level { font-size: 10px; flex-shrink: 0; }
.card-name .name { font-size: 13px; font-weight: 600; color: var(--text-primary); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.card-uptime { font-size: 11px; color: var(--text-muted); flex-shrink: 0; margin-left: 8px; white-space: nowrap; }
.card-metrics { display: flex; flex-wrap: wrap; gap: 4px 8px; }
.card-metric { display: flex; align-items: center; gap: 4px; width: calc(50% - 4px); overflow: hidden; }
.card-metric .label { font-size: 10px; color: var(--text-muted); width: 22px; flex-shrink: 0; }
.mini-bar { flex: 1; height: 5px; background: rgba(0,0,0,0.1); border-radius: 2px; overflow: hidden; min-width: 20px; }
.mini-fill { height: 100%; border-radius: 2px; }
.mini-fill.normal { background: #22c55e; }
.mini-fill.warning { background: #eab308; }
.mini-fill.critical { background: #f97316; }
.mini-fill.danger { background: #ef4444; }
.card-metric .value { font-size: 11px; font-weight: 500; color: var(--text-secondary); width: 36px; text-align: right; flex-shrink: 0; overflow: hidden; text-overflow: ellipsis; }
.card-stopped { font-size: 12px; color: var(--text-muted); font-style: italic; text-align: center; padding: 6px 0; }
.no-container { font-size: 13px; color: var(--text-muted); padding: 12px; display: flex; align-items: center; justify-content: center; }
</style>

View File

@@ -0,0 +1,74 @@
<template>
<aside class="sidebar">
<div class="sidebar-header">
<div class="sidebar-logo">
<span class="logo-icon">📡</span>
<span>OSOLIT Monitor</span>
</div>
</div>
<nav class="sidebar-nav">
<NuxtLink to="/" class="nav-item" :class="{ active: route.path === '/' }">
<span class="icon">📊</span>
<span>대시보드</span>
</NuxtLink>
<div class="nav-group-title">네트워크</div>
<NuxtLink to="/network/pubnet" class="nav-item nav-sub-item" :class="{ active: route.path === '/network/pubnet' }">
<span class="icon">🌐</span>
<span>Public Network</span>
</NuxtLink>
<NuxtLink to="/network/privnet" class="nav-item nav-sub-item" :class="{ active: route.path === '/network/privnet' }">
<span class="icon">🔒</span>
<span>Private Network</span>
</NuxtLink>
<div class="nav-group-title">서버</div>
<NuxtLink to="/server/list" class="nav-item nav-sub-item" :class="{ active: route.path === '/server/list' }">
<span class="icon">🖥</span>
<span>Server Targets</span>
</NuxtLink>
<NuxtLink to="/server/history" class="nav-item nav-sub-item" :class="{ active: route.path === '/server/history' }">
<span class="icon">📈</span>
<span>Server Status</span>
</NuxtLink>
<div class="nav-group-title">이상감지</div>
<NuxtLink to="/anomaly/short-term" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/short-term' }">
<span class="icon"></span>
<span>단기 변화율</span>
</NuxtLink>
<NuxtLink to="/anomaly/zscore" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/zscore' }">
<span class="icon">📊</span>
<span>Z-Score 분석</span>
</NuxtLink>
<NuxtLink to="/anomaly/baseline" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/baseline' }">
<span class="icon">🕐</span>
<span>시간대별 베이스라인</span>
</NuxtLink>
<NuxtLink to="/anomaly/trend" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/trend' }">
<span class="icon">📉</span>
<span>추세 분석</span>
</NuxtLink>
<div class="nav-group-title">설정</div>
<NuxtLink to="/settings/thresholds" class="nav-item nav-sub-item" :class="{ active: route.path === '/settings/thresholds' }">
<span class="icon"></span>
<span>임계값 설정</span>
</NuxtLink>
</nav>
</aside>
</template>
<script setup lang="ts">
const route = useRoute()
</script>

View File

@@ -0,0 +1,112 @@
<template>
<div
class="theme-switch"
:class="{ dark: theme === 'dark' }"
@click="toggleTheme"
:title="theme === 'dark' ? '라이트 모드로 전환' : '다크 모드로 전환'"
>
<div class="switch-track">
<span class="switch-icon sun"></span>
<span class="switch-icon moon">🌙</span>
</div>
<div class="switch-thumb"></div>
</div>
</template>
<script setup lang="ts">
const { theme, toggleTheme, initTheme } = useTheme()
onMounted(() => {
initTheme()
})
</script>
<style scoped>
.theme-switch {
position: relative;
width: 56px;
height: 28px;
background: #e0e0e0;
border: 2px solid #ccc;
border-radius: 14px;
cursor: pointer;
transition: background 0.3s ease, border-color 0.3s ease;
box-shadow: inset 0 1px 3px rgba(0, 0, 0, 0.1);
}
.theme-switch.dark {
background: #3a3a3a;
border-color: #666;
}
.switch-track {
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
display: flex;
justify-content: space-between;
align-items: center;
padding: 0 6px;
}
.switch-icon {
font-size: 14px;
transition: opacity 0.3s ease;
}
.switch-icon.sun {
opacity: 0.4;
}
.switch-icon.moon {
opacity: 0.4;
}
.theme-switch.dark .switch-icon.sun {
opacity: 0.4;
}
.theme-switch.dark .switch-icon.moon {
opacity: 0.4;
}
.switch-thumb {
position: absolute;
top: 2px;
left: 2px;
width: 20px;
height: 20px;
background: #1a1a1a;
border-radius: 50%;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);
transition: transform 0.3s cubic-bezier(0.4, 0, 0.2, 1), background 0.3s ease;
}
.theme-switch.dark .switch-thumb {
transform: translateX(28px);
background: #fff;
}
/* 호버 효과 */
.theme-switch:hover {
border-color: #999;
}
.theme-switch.dark:hover {
border-color: #888;
}
.theme-switch:hover .switch-thumb {
box-shadow: 0 2px 6px rgba(0, 0, 0, 0.3);
}
.theme-switch:active .switch-thumb {
width: 24px;
}
.theme-switch.dark:active .switch-thumb {
transform: translateX(24px);
}
</style>