Files
nexus-terminal/packages/frontend/src/components/StatusMonitor.vue
T
yinjianm 2233e3fa4f feat(ui): 重设计文件管理器书签与传输面板
新增书签作用域与连接关联,后端为 favorite_paths
补充 scope 和 connection_id 字段及查询写入支持

前端重构书签弹窗与编辑表单,支持本地/云端筛选、
作用域选择与多语言文案更新

文件管理器工具栏改为紧凑图标样式,上传入口合并为
下拉菜单,并新增底部传输面板统一展示上传任务

同时优化 SSH 终端运行态为显式状态机,并为短命令
补充最短可见时间,避免运行中标记闪烁难以感知
2026-05-01 22:54:29 +08:00

1048 lines
32 KiB
Vue

<template>
<div class="status-monitor" :class="{ 'status-monitor--inactive': !activeSessionId }">
<div v-if="!activeSessionId" class="status-state">
<i class="fas fa-plug status-state__icon"></i>
<span class="status-state__title">{{ t('layout.noActiveSession.title') }}</span>
</div>
<div v-else-if="currentStatusError" class="status-state status-state--error">
<i class="fas fa-exclamation-triangle status-state__icon"></i>
<span class="status-state__title">{{ t('statusMonitor.errorPrefix') }} {{ currentStatusError }}</span>
</div>
<div v-else-if="!currentServerStatus" class="status-state">
<i class="fas fa-spinner fa-spin status-state__icon"></i>
<span class="status-state__title">{{ t('statusMonitor.loading') }}</span>
</div>
<section v-else class="sm-shell">
<!-- System Header -->
<header class="sm-header">
<div class="sm-header__row">
<div class="sm-header__left">
<i class="fas fa-desktop sm-header__icon"></i>
<span class="sm-header__label">{{ t('statusMonitor.title') }}</span>
</div>
<div class="sm-header__right">
<button
v-if="statusMonitorShowIpBoolean && activeSessionId && sessionIpAddress"
class="sm-chip sm-chip--interactive"
type="button"
:title="sessionIpAddress"
@click="copyIpToClipboard(sessionIpAddress)"
>
{{ sessionIpAddress }}
</button>
<span class="sm-live-dot"></span>
</div>
</div>
<div class="sm-header__tags">
<span class="sm-tag">{{ displayOsName }}</span>
</div>
<div class="sm-header__meta">
<span class="sm-meta">{{ t('statusMonitor.timezoneLabel') }} {{ timezoneDisplay }}</span>
<span class="sm-meta">{{ t('statusMonitor.uptimeLabel') }} {{ uptimeDisplay }}</span>
</div>
</header>
<!-- CPU Module -->
<section class="sm-section">
<div class="sm-section__head">
<div class="sm-section__title-row">
<i class="fas fa-microchip sm-section__icon"></i>
<span class="sm-section__title">CPU</span>
</div>
<StatusMonitorCpuHistoryChart :cpu-history="currentCpuHistory" :compact="true" class="sm-mini-chart" />
</div>
<div class="sm-cpu-cores">
<div
v-for="item in cpuCoreItems"
:key="item.key"
class="sm-cpu-core"
>
<span class="sm-cpu-core__index">{{ item.index }}</span>
<div class="sm-cpu-core__bar">
<div
class="sm-cpu-core__fill"
:class="getCpuFillClass(item.percent)"
:style="{ width: `${item.percent}%` }"
></div>
</div>
<span class="sm-cpu-core__val">{{ item.value }}</span>
</div>
</div>
<button
type="button"
class="sm-link-btn"
@click="isCpuCoreModalVisible = true"
>
{{ t('statusMonitor.cpuViewAllCores') }}
</button>
</section>
<!-- Memory Module -->
<section class="sm-section">
<div class="sm-section__head">
<div class="sm-section__title-row">
<i class="fas fa-memory sm-section__icon"></i>
<span class="sm-section__title">{{ t('statusMonitor.memoryCardTitle') }}</span>
</div>
<span class="sm-badge">{{ memoryTotalDisplay }}</span>
</div>
<div class="sm-memory-row">
<div class="sm-memory-ring" :style="memoryRingStyle">
<div class="sm-memory-ring__center">{{ memoryPercentDisplay }}</div>
</div>
<div class="sm-memory-stats">
<div v-for="item in memoryStatItems" :key="item.key" class="sm-memory-stat">
<span class="sm-dot" :class="`sm-dot--${item.key}`"></span>
<span class="sm-memory-stat__label">{{ item.label }}</span>
<span class="sm-memory-stat__value">{{ item.value }}</span>
</div>
</div>
</div>
</section>
<!-- Network Module -->
<section class="sm-section">
<div class="sm-section__head">
<div class="sm-section__title-row">
<i class="fas fa-network-wired sm-section__icon"></i>
<span class="sm-section__title">{{ t('statusMonitor.networkLabel') }}</span>
</div>
<StatusMonitorNetworkHistoryChart
:download-history="currentNetRxHistory"
:upload-history="currentNetTxHistory"
:compact="true"
class="sm-mini-chart"
/>
</div>
<div class="sm-net-table">
<div class="sm-net-table__head">
<span></span>
<span>{{ t('statusMonitor.networkSpeedTitleUnit', { unit: '' }).replace(/[()]/g, '').trim() }}</span>
<span>{{ t('statusMonitor.totalTrafficLabel') }}</span>
</div>
<div
v-for="item in networkFlowItems"
:key="item.key"
class="sm-net-row"
>
<span class="sm-net-row__label" :class="`sm-net-row__label--${item.tone}`">
<i :class="['fas', item.icon]"></i>
{{ item.label }}
</span>
<span class="sm-net-row__val">{{ item.value }}</span>
<span class="sm-net-row__total">{{ item.totalValue }}</span>
</div>
</div>
</section>
<!-- Process Module -->
<section class="sm-section">
<div class="sm-section__head">
<div class="sm-section__title-row">
<i class="fas fa-list-ul sm-section__icon"></i>
<span class="sm-section__title">{{ t('statusMonitor.processManager.title') }}</span>
</div>
<button class="sm-link-btn sm-link-btn--icon" type="button" @click="isProcessManagerVisible = true">
<i class="fas fa-th-list"></i>
{{ t('statusMonitor.processManager.viewAll') }}
</button>
</div>
<div class="sm-proc-table" v-if="processPreviewItems.length > 0">
<div class="sm-proc-head">
<span>CPU</span>
<span>MEM</span>
<span>CMD</span>
</div>
<div
v-for="item in processPreviewItems"
:key="item.pid"
class="sm-proc-row"
>
<span class="sm-proc-row__cpu" :class="{ 'sm-proc-row__cpu--hot': item.cpu > 50 }">{{ item.cpu.toFixed(1) }}%</span>
<span class="sm-proc-row__mem">{{ item.memPercent !== undefined ? item.memPercent.toFixed(1) + '%' : formatProcessMemory(item.memMb) }}</span>
<span class="sm-proc-row__cmd">
<span class="sm-proc-row__cmd-text" :title="item.command">{{ truncateCommand(item.command) }}</span>
<span class="sm-proc-row__pid">PID {{ item.pid }}</span>
</span>
</div>
</div>
<div v-else class="sm-empty">
{{ t('statusMonitor.processManager.empty') }}
</div>
</section>
<!-- Disk Module -->
<section class="sm-section">
<div class="sm-section__head">
<div class="sm-section__title-row">
<i class="fas fa-hdd sm-section__icon"></i>
<span class="sm-section__title">{{ t('statusMonitor.diskCardTitle') }}</span>
</div>
<span class="sm-badge">{{ diskUsageDisplay }}</span>
</div>
<div class="sm-disk-device">
<span class="sm-disk-device__mount">{{ diskMountPointDisplay }}</span>
<span class="sm-disk-device__type">{{ t('statusMonitor.diskTypeLabel') }} {{ diskFsTypeDisplay }}</span>
</div>
<div class="sm-disk-io">
<div class="sm-disk-io__item">
<div class="sm-disk-io__icon"></div>
<div class="sm-disk-io__col">
<span class="sm-disk-io__label">{{ t('statusMonitor.diskReadRateLabel') }}</span>
<span class="sm-disk-io__val">{{ diskReadRateDisplay }}</span>
</div>
</div>
<div class="sm-disk-io__item">
<div class="sm-disk-io__icon"></div>
<div class="sm-disk-io__col">
<span class="sm-disk-io__label">{{ t('statusMonitor.diskWriteRateLabel') }}</span>
<span class="sm-disk-io__val">{{ diskWriteRateDisplay }}</span>
</div>
</div>
</div>
<div class="sm-disk-summary">
<div class="sm-disk-summary__head">
<span>{{ t('statusMonitor.diskMountLabel') }}</span>
<span>{{ t('statusMonitor.diskSizeLabel') }}</span>
<span>{{ t('statusMonitor.diskAvailableLabel') }}</span>
<span>{{ t('statusMonitor.diskUsedPercentLabel') }}</span>
</div>
<div class="sm-disk-summary__row">
<span class="sm-disk-summary__mount">{{ diskMountPointDisplay }}</span>
<span>{{ diskSizeDisplay }}</span>
<span>{{ diskAvailableDisplay }}</span>
<span class="sm-disk-summary__pct">{{ diskPercentDisplay }}</span>
</div>
</div>
</section>
</section>
<ProcessManagerModal
:is-visible="isProcessManagerVisible"
:session-id="activeSessionId"
@close="isProcessManagerVisible = false"
/>
<StatusMonitorCpuCoreModal
:is-visible="isCpuCoreModalVisible"
:cpu-core-items="cpuCoreItems"
:total-cpu-percent="displayCpuPercent"
@close="isCpuCoreModalVisible = false"
/>
</div>
</template>
<script setup lang="ts">
import { computed, ref, watch, type CSSProperties, type PropType } from 'vue';
import { useI18n } from 'vue-i18n';
import { storeToRefs } from 'pinia';
import ProcessManagerModal from './ProcessManagerModal.vue';
import StatusMonitorCpuCoreModal from './StatusMonitorCpuCoreModal.vue';
import StatusMonitorCpuHistoryChart from './StatusMonitorCpuHistoryChart.vue';
import StatusMonitorNetworkHistoryChart from './StatusMonitorNetworkHistoryChart.vue';
import { useSessionStore } from '../stores/session.store';
import { useSettingsStore } from '../stores/settings.store';
import { useConnectionsStore } from '../stores/connections.store';
import { useUiNotificationsStore } from '../stores/uiNotifications.store';
import type { ProcessListItem, ServerStatus } from '../types/server.types';
const { t } = useI18n();
const sessionStore = useSessionStore();
const settingsStore = useSettingsStore();
const connectionsStore = useConnectionsStore();
const uiNotificationsStore = useUiNotificationsStore();
const { sessions } = storeToRefs(sessionStore);
const { statusMonitorShowIpBoolean } = storeToRefs(settingsStore);
const isCpuCoreModalVisible = ref(false);
const isProcessManagerVisible = ref(false);
const props = defineProps({
activeSessionId: {
type: String as PropType<string | null>,
required: false,
default: null,
},
});
const clampPercent = (value?: number): number => {
if (typeof value !== 'number' || !Number.isFinite(value)) return 0;
return Math.max(0, Math.min(100, value));
};
const currentSessionState = computed(() => (props.activeSessionId ? sessions.value.get(props.activeSessionId) : null));
const currentServerStatus = computed<ServerStatus | null>(() => currentSessionState.value?.statusMonitorManager?.serverStatus?.value ?? null);
const currentCpuHistory = computed<readonly (number | null)[]>(() => currentSessionState.value?.statusMonitorManager?.cpuHistory?.value ?? Array(24).fill(null));
const currentNetRxHistory = computed<readonly (number | null)[]>(() => currentSessionState.value?.statusMonitorManager?.netRxHistory?.value ?? Array(24).fill(null));
const currentNetTxHistory = computed<readonly (number | null)[]>(() => currentSessionState.value?.statusMonitorManager?.netTxHistory?.value ?? Array(24).fill(null));
const displayCpuPercent = computed(() => clampPercent(currentServerStatus.value?.cpuPercent));
const displayMemoryPercent = computed(() => clampPercent(currentServerStatus.value?.memPercent));
const displayDiskPercent = computed(() => clampPercent(currentServerStatus.value?.diskPercent));
const currentStatusError = computed<string | null>(() => currentSessionState.value?.statusMonitorManager?.statusError?.value ?? null);
const timezoneDisplay = computed(() => currentServerStatus.value?.timezone || t('statusMonitor.notAvailable'));
const processTotalDisplay = computed(() => currentServerStatus.value?.processTotal ?? 0);
const processRunningDisplay = computed(() => currentServerStatus.value?.processRunning ?? 0);
const processSleepingDisplay = computed(() => currentServerStatus.value?.processSleeping ?? 0);
const cachedCpuModel = ref<string | null>(null);
const cachedCpuCores = ref<number | null>(null);
const cachedOsName = ref<string | null>(null);
watch(currentServerStatus, newData => {
if (!newData) return;
if (newData.cpuModel) cachedCpuModel.value = newData.cpuModel;
if (typeof newData.cpuCores === 'number' && Number.isFinite(newData.cpuCores)) cachedCpuCores.value = newData.cpuCores;
if (newData.osName) cachedOsName.value = newData.osName;
}, { immediate: true });
const displayOsName = computed(() => (currentServerStatus.value?.osName ?? cachedOsName.value) || t('statusMonitor.notAvailable'));
const formatBytesPerSecond = (bytes?: number): string => {
if (bytes === undefined || bytes === null || isNaN(bytes)) return t('statusMonitor.notAvailable');
if (bytes < 1024) return `${bytes} ${t('statusMonitor.bytesPerSecond')}`;
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} ${t('statusMonitor.kiloBytesPerSecond')}`;
if (bytes < 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024)).toFixed(1)} ${t('statusMonitor.megaBytesPerSecond')}`;
return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)} ${t('statusMonitor.gigaBytesPerSecond')}`;
};
const formatBytes = (bytes?: number): string => {
if (bytes === undefined || bytes === null || isNaN(bytes)) return t('statusMonitor.notAvailable');
if (bytes < 1024) return `${bytes} B`;
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;
if (bytes < 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024)).toFixed(1)} ${t('statusMonitor.megaBytes')}`;
if (bytes < 1024 * 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)} ${t('statusMonitor.gigaBytes')}`;
return `${(bytes / (1024 * 1024 * 1024 * 1024)).toFixed(1)} TB`;
};
const formatCompactBytes = (bytes?: number): string => {
if (bytes === undefined || bytes === null || isNaN(bytes)) return t('statusMonitor.notAvailable');
if (bytes < 1024) return `${bytes.toFixed(1)} B`;
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;
if (bytes < 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;
return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)} GB`;
};
const formatStorageSizeFromKb = (kb?: number, compact = false): string => {
if (kb === undefined || kb === null || isNaN(kb)) return t('statusMonitor.notAvailable');
const units = compact ? ['KB', 'M', 'G', 'T'] : ['KB', t('statusMonitor.megaBytes'), t('statusMonitor.gigaBytes'), 'TB'];
let value = kb;
let unitIndex = 0;
while (value >= 1024 && unitIndex < units.length - 1) {
value /= 1024;
unitIndex += 1;
}
return `${value.toFixed(1)} ${units[unitIndex]}`;
};
const formatMemorySize = (mb?: number): string => {
if (mb === undefined || mb === null || isNaN(mb)) return t('statusMonitor.notAvailable');
if (mb < 1024) return `${mb.toFixed(1)} ${t('statusMonitor.megaBytes')}`;
return `${(mb / 1024).toFixed(1)} ${t('statusMonitor.gigaBytes')}`;
};
const formatUptime = (seconds?: number): string => {
if (seconds === undefined || seconds === null || !Number.isFinite(seconds) || seconds < 0) return t('statusMonitor.notAvailable');
const days = Math.floor(seconds / 86400);
const hours = Math.floor((seconds % 86400) / 3600);
const minutes = Math.floor((seconds % 3600) / 60);
if (days > 0) return `${days}${t('statusMonitor.uptimeDaySuffix')} ${hours}${t('statusMonitor.uptimeHourSuffix')}`;
if (hours > 0) return `${hours}${t('statusMonitor.uptimeHourSuffix')} ${minutes}${t('statusMonitor.uptimeMinuteSuffix')}`;
return `${minutes}${t('statusMonitor.uptimeMinuteSuffix')}`;
};
const formatProcessMemory = (mb?: number): string => {
if (mb === undefined || mb === null || !Number.isFinite(mb)) return t('statusMonitor.notAvailable');
if (mb < 1024) return `${mb.toFixed(1)} M`;
return `${(mb / 1024).toFixed(1)} G`;
};
const truncateCommand = (cmd: string): string => {
if (!cmd) return '';
const parts = cmd.split('/');
const basename = parts[parts.length - 1] || cmd;
return basename.length > 24 ? basename.slice(0, 22) + '...' : basename;
};
const getCpuFillClass = (percent: number): string => {
if (percent >= 90) return 'sm-cpu-core__fill--critical';
if (percent >= 60) return 'sm-cpu-core__fill--warn';
return 'sm-cpu-core__fill--normal';
};
const memoryTotalValue = computed(() => currentServerStatus.value?.memTotal ?? 0);
const memoryUsedValue = computed(() => currentServerStatus.value?.memUsed ?? 0);
const memoryCachedValue = computed(() => currentServerStatus.value?.memCached ?? 0);
const memoryFreeValue = computed(() => {
const data = currentServerStatus.value;
if (data?.memFree !== undefined) return data.memFree;
if (data?.memTotal !== undefined && data?.memUsed !== undefined) return Math.max(data.memTotal - data.memUsed - (data.memCached ?? 0), 0);
return 0;
});
const memoryTotalDisplay = computed(() => formatMemorySize(currentServerStatus.value?.memTotal));
const memoryPercentDisplay = computed(() => `${Math.round(displayMemoryPercent.value)}%`);
const memoryRingStyle = computed<CSSProperties>(() => {
const total = memoryTotalValue.value;
if (total <= 0) return { background: 'conic-gradient(#334155 0% 100%)' };
const usedPercent = Math.min(100, (memoryUsedValue.value / total) * 100);
const cachedPercent = Math.min(100 - usedPercent, (memoryCachedValue.value / total) * 100);
const usedEnd = usedPercent;
const cacheEnd = usedPercent + cachedPercent;
return { background: `conic-gradient(#ef5350 0 ${usedEnd}%, #8f96a3 ${usedEnd}% ${cacheEnd}%, #4ade80 ${cacheEnd}% 100%)` };
});
const memoryStatItems = computed(() => [
{ key: 'used', label: t('statusMonitor.memoryUsedStat'), value: formatMemorySize(memoryUsedValue.value) },
{ key: 'cached', label: t('statusMonitor.memoryCachedStat'), value: formatMemorySize(memoryCachedValue.value) },
{ key: 'free', label: t('statusMonitor.memoryFreeStat'), value: formatMemorySize(memoryFreeValue.value) },
]);
const diskUsageDisplay = computed(() => {
const data = currentServerStatus.value;
if (!data || data.diskUsed === undefined || data.diskTotal === undefined) return t('statusMonitor.notAvailable');
return `${formatStorageSizeFromKb(data.diskUsed, true)} / ${formatStorageSizeFromKb(data.diskTotal, true)}`;
});
const diskFsTypeDisplay = computed(() => currentServerStatus.value?.diskFsType || t('statusMonitor.notAvailable'));
const diskReadRateDisplay = computed(() => formatCompactBytes(currentServerStatus.value?.diskReadRate));
const diskWriteRateDisplay = computed(() => formatCompactBytes(currentServerStatus.value?.diskWriteRate));
const diskMountPointDisplay = computed(() => currentServerStatus.value?.diskMountPoint || t('statusMonitor.notAvailable'));
const diskSizeDisplay = computed(() => formatStorageSizeFromKb(currentServerStatus.value?.diskTotal, true));
const diskAvailableDisplay = computed(() => formatStorageSizeFromKb(currentServerStatus.value?.diskAvailable, true));
const diskPercentDisplay = computed(() => `${Math.round(displayDiskPercent.value)}%`);
const sessionIpAddress = computed(() => {
const sessionState = currentSessionState.value;
if (!sessionState?.connectionId) return null;
const connectionIdAsNumber = parseInt(sessionState.connectionId, 10);
if (isNaN(connectionIdAsNumber)) return null;
const connectionInfo = connectionsStore.connections.find(conn => conn.id === connectionIdAsNumber);
return connectionInfo?.host || null;
});
const uptimeDisplay = computed(() => formatUptime(currentServerStatus.value?.uptimeSeconds));
const topProcessPreview = computed<readonly ProcessListItem[]>(() => currentServerStatus.value?.topProcesses ?? []);
const cpuCoreItems = computed(() => {
const rawPercents = currentServerStatus.value?.cpuCorePercents;
const fallbackCoreCount = (() => {
const currentCores = currentServerStatus.value?.cpuCores;
if (typeof currentCores !== 'number' || !Number.isFinite(currentCores) || currentCores <= 0) return 0;
return Math.round(currentCores);
})();
const normalizedPercents = Array.isArray(rawPercents) && rawPercents.length > 0
? rawPercents
: Array.from({ length: fallbackCoreCount }, () => 0);
return normalizedPercents.map((percent, idx) => {
const clampedPercent = clampPercent(percent);
return {
key: `cpu-core-${idx}`,
index: idx,
label: t('statusMonitor.cpuCoreLabel', { index: idx + 1 }),
value: `${Math.round(clampedPercent)}%`,
percent: clampedPercent,
};
});
});
const networkFlowItems = computed(() => [
{
key: 'upload',
label: t('statusMonitor.uploadLabel'),
value: formatBytesPerSecond(currentServerStatus.value?.netTxRate),
totalValue: formatBytes(currentServerStatus.value?.netTxTotalBytes),
tone: 'up',
icon: 'fa-arrow-up',
},
{
key: 'download',
label: t('statusMonitor.downloadLabel'),
value: formatBytesPerSecond(currentServerStatus.value?.netRxRate),
totalValue: formatBytes(currentServerStatus.value?.netRxTotalBytes),
tone: 'down',
icon: 'fa-arrow-down',
},
]);
const networkRateUnitLabel = computed(() => {
const maxRate = Math.max(currentServerStatus.value?.netRxRate ?? 0, currentServerStatus.value?.netTxRate ?? 0);
return maxRate >= 1024 * 1024 ? 'MB/s' : 'KB/s';
});
const processPreviewItems = computed<readonly ProcessListItem[]>(() => topProcessPreview.value.slice(0, 4));
const copyIpToClipboard = async (ipAddress: string | null) => {
if (!ipAddress) return;
try {
await navigator.clipboard.writeText(ipAddress);
uiNotificationsStore.showSuccess(t('common.copied', '已复制'));
} catch (err) {
console.error('Failed to copy IP address: ', err);
uiNotificationsStore.showError(t('statusMonitor.copyIpError', '复制 IP 失败'));
}
};
</script>
<style scoped>
.status-monitor {
height: 100%;
overflow-y: auto;
padding: 10px;
background: linear-gradient(180deg, #0f1419 0%, #0b1015 100%);
color: #e2e8f0;
font-family: -apple-system, BlinkMacSystemFont, 'SF Pro Text', 'Segoe UI', system-ui, sans-serif;
}
.status-monitor--inactive {
background: linear-gradient(180deg, #111827, #0f172a);
}
.sm-shell {
display: flex;
flex-direction: column;
gap: 2px;
}
.status-state {
display: flex;
min-height: 100%;
flex-direction: column;
align-items: center;
justify-content: center;
gap: 10px;
text-align: center;
color: #9ca3af;
}
.status-state--error { color: #f87171; }
.status-state__icon { font-size: 28px; }
.status-state__title { font-size: 14px; font-weight: 600; }
/* ── Header ── */
.sm-header {
padding: 10px 12px;
border-bottom: 1px solid rgba(148, 163, 184, 0.08);
}
.sm-header__row {
display: flex;
align-items: center;
justify-content: space-between;
gap: 8px;
}
.sm-header__left {
display: flex;
align-items: center;
gap: 6px;
}
.sm-header__icon {
color: #64748b;
font-size: 12px;
}
.sm-header__label {
color: #94a3b8;
font-size: 11px;
font-weight: 600;
letter-spacing: 0.05em;
}
.sm-header__right {
display: flex;
align-items: center;
gap: 8px;
}
.sm-chip {
padding: 2px 8px;
border-radius: 4px;
background: rgba(148, 163, 184, 0.1);
color: #94a3b8;
font-size: 11px;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
border: none;
}
.sm-chip--interactive {
cursor: pointer;
transition: color 0.15s;
}
.sm-chip--interactive:hover { color: #e2e8f0; }
.sm-live-dot {
width: 7px;
height: 7px;
border-radius: 50%;
background: #4ade80;
box-shadow: 0 0 6px rgba(74, 222, 128, 0.6);
flex-shrink: 0;
}
.sm-header__tags {
display: flex;
flex-wrap: wrap;
gap: 6px;
margin-top: 8px;
}
.sm-tag {
display: inline-block;
padding: 3px 8px;
border-radius: 4px;
border: 1px solid rgba(148, 163, 184, 0.12);
background: rgba(148, 163, 184, 0.06);
color: #cbd5e1;
font-size: 12px;
font-weight: 500;
}
.sm-header__meta {
display: flex;
flex-wrap: wrap;
gap: 14px;
margin-top: 8px;
color: #64748b;
font-size: 11px;
}
.sm-meta {
white-space: nowrap;
}
/* ── Section (shared) ── */
.sm-section {
padding: 10px 12px;
border-bottom: 1px solid rgba(148, 163, 184, 0.08);
}
.sm-section:last-child { border-bottom: none; }
.sm-section__head {
display: flex;
align-items: center;
justify-content: space-between;
gap: 8px;
margin-bottom: 8px;
}
.sm-section__title-row {
display: flex;
align-items: center;
gap: 6px;
flex-shrink: 0;
}
.sm-section__icon {
color: #64748b;
font-size: 11px;
}
.sm-section__title {
color: #e2e8f0;
font-size: 13px;
font-weight: 600;
}
.sm-badge {
padding: 2px 8px;
border-radius: 4px;
background: rgba(148, 163, 184, 0.1);
color: #94a3b8;
font-size: 11px;
font-weight: 600;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
white-space: nowrap;
}
.sm-mini-chart {
flex: 1;
min-width: 0;
max-width: 160px;
height: 28px;
overflow: hidden;
}
.sm-link-btn {
display: inline-flex;
align-items: center;
gap: 5px;
padding: 0;
margin-top: 6px;
border: none;
background: none;
color: #64748b;
font-size: 11px;
cursor: pointer;
transition: color 0.15s;
}
.sm-link-btn:hover { color: #94a3b8; }
.sm-link-btn--icon i { font-size: 10px; }
/* ── CPU ── */
.sm-cpu-cores {
display: flex;
flex-direction: column;
gap: 3px;
}
.sm-cpu-core {
display: flex;
align-items: center;
gap: 6px;
}
.sm-cpu-core__index {
width: 12px;
color: #64748b;
font-size: 10px;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
text-align: right;
flex-shrink: 0;
}
.sm-cpu-core__bar {
flex: 1;
height: 10px;
border-radius: 2px;
background: rgba(148, 163, 184, 0.08);
overflow: hidden;
}
.sm-cpu-core__fill {
height: 100%;
border-radius: 2px;
transition: width 0.4s ease;
min-width: 1px;
}
.sm-cpu-core__fill--normal { background: #22c55e; }
.sm-cpu-core__fill--warn { background: #f59e0b; }
.sm-cpu-core__fill--critical { background: #ef4444; }
.sm-cpu-core__val {
width: 36px;
color: #cbd5e1;
font-size: 11px;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
text-align: right;
flex-shrink: 0;
}
/* ── Memory ── */
.sm-memory-row {
display: flex;
align-items: center;
gap: 12px;
}
.sm-memory-ring {
position: relative;
width: 52px;
height: 52px;
border-radius: 50%;
flex-shrink: 0;
}
.sm-memory-ring::after {
content: '';
position: absolute;
inset: 9px;
border-radius: 50%;
background: #0f1419;
}
.sm-memory-ring__center {
position: absolute;
inset: 0;
z-index: 1;
display: flex;
align-items: center;
justify-content: center;
color: #f8fafc;
font-size: 11px;
font-weight: 700;
}
.sm-memory-stats {
display: flex;
gap: 12px;
flex: 1;
min-width: 0;
}
.sm-memory-stat {
display: flex;
align-items: center;
gap: 5px;
white-space: nowrap;
}
.sm-dot {
width: 6px;
height: 6px;
border-radius: 50%;
flex-shrink: 0;
}
.sm-dot--used { background: #ef5350; }
.sm-dot--cached { background: #9ca3af; }
.sm-dot--free { background: #4ade80; }
.sm-memory-stat__label {
color: #94a3b8;
font-size: 10px;
}
.sm-memory-stat__value {
color: #e2e8f0;
font-size: 11px;
font-weight: 700;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
}
/* ── Network ── */
.sm-net-table {
display: flex;
flex-direction: column;
gap: 2px;
}
.sm-net-table__head {
display: grid;
grid-template-columns: minmax(0, 1fr) minmax(0, 1fr) minmax(0, 1fr);
gap: 6px;
padding: 0 0 4px;
border-bottom: 1px solid rgba(148, 163, 184, 0.06);
color: #64748b;
font-size: 10px;
}
.sm-net-table__head span:not(:first-child) { text-align: right; }
.sm-net-row {
display: grid;
grid-template-columns: minmax(0, 1fr) minmax(0, 1fr) minmax(0, 1fr);
gap: 6px;
padding: 4px 0;
align-items: center;
}
.sm-net-row__label {
display: flex;
align-items: center;
gap: 5px;
font-size: 11px;
font-weight: 600;
color: #cbd5e1;
}
.sm-net-row__label i { width: 10px; text-align: center; font-size: 10px; }
.sm-net-row__label--up i { color: #34d399; }
.sm-net-row__label--down i { color: #60a5fa; }
.sm-net-row__val,
.sm-net-row__total {
text-align: right;
font-size: 11px;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
color: #e2e8f0;
font-weight: 600;
}
.sm-net-row__total { color: #94a3b8; font-weight: 400; }
/* ── Process ── */
.sm-proc-table { display: flex; flex-direction: column; gap: 0; }
.sm-proc-head {
display: grid;
grid-template-columns: 60px 52px minmax(0, 1fr);
gap: 6px;
padding: 0 0 4px;
border-bottom: 1px solid rgba(148, 163, 184, 0.06);
color: #64748b;
font-size: 10px;
font-weight: 600;
}
.sm-proc-row {
display: grid;
grid-template-columns: 60px 52px minmax(0, 1fr);
gap: 6px;
padding: 5px 0;
border-bottom: 1px solid rgba(148, 163, 184, 0.04);
align-items: center;
font-size: 11px;
}
.sm-proc-row:last-child { border-bottom: none; }
.sm-proc-row__cpu {
color: #e2e8f0;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
font-weight: 700;
}
.sm-proc-row__cpu--hot { color: #f87171; }
.sm-proc-row__mem {
color: #94a3b8;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
}
.sm-proc-row__cmd {
display: flex;
flex-direction: column;
min-width: 0;
}
.sm-proc-row__cmd-text {
color: #cbd5e1;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
font-size: 11px;
}
.sm-proc-row__pid {
color: #475569;
font-size: 10px;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
}
.sm-empty {
padding: 10px;
color: #64748b;
font-size: 12px;
text-align: center;
}
/* ── Disk ── */
.sm-disk-device {
display: flex;
align-items: center;
gap: 8px;
margin-bottom: 8px;
}
.sm-disk-device__mount {
color: #e2e8f0;
font-size: 14px;
font-weight: 700;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
}
.sm-disk-device__type {
padding: 1px 6px;
border-radius: 3px;
background: rgba(148, 163, 184, 0.1);
color: #94a3b8;
font-size: 10px;
font-weight: 600;
}
.sm-disk-io {
display: flex;
gap: 16px;
margin-bottom: 10px;
}
.sm-disk-io__item {
display: flex;
align-items: center;
gap: 8px;
}
.sm-disk-io__icon {
width: 16px;
height: 24px;
border-radius: 2px;
background: rgba(148, 163, 184, 0.12);
flex-shrink: 0;
}
.sm-disk-io__col { display: flex; flex-direction: column; }
.sm-disk-io__label {
color: #64748b;
font-size: 10px;
}
.sm-disk-io__val {
color: #e2e8f0;
font-size: 13px;
font-weight: 700;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
}
.sm-disk-summary {
border-top: 1px solid rgba(148, 163, 184, 0.06);
padding-top: 8px;
}
.sm-disk-summary__head {
display: grid;
grid-template-columns: minmax(0, 0.8fr) repeat(3, minmax(0, 1fr));
gap: 6px;
padding-bottom: 4px;
color: #64748b;
font-size: 10px;
font-weight: 600;
}
.sm-disk-summary__row {
display: grid;
grid-template-columns: minmax(0, 0.8fr) repeat(3, minmax(0, 1fr));
gap: 6px;
color: #e2e8f0;
font-size: 11px;
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
font-weight: 600;
}
.sm-disk-summary__mount { color: #86efac; }
.sm-disk-summary__pct {
display: inline-flex;
align-items: center;
justify-content: center;
padding: 1px 6px;
border-radius: 3px;
background: rgba(74, 222, 128, 0.12);
color: #86efac;
font-size: 10px;
width: fit-content;
}
/* ── Responsive ── */
@container (max-width: 220px) {
.sm-memory-row { flex-direction: column; align-items: flex-start; gap: 8px; }
.sm-memory-stats { flex-wrap: wrap; }
.sm-disk-io { flex-direction: column; gap: 8px; }
}
@media (max-width: 640px) {
.status-monitor { padding: 8px; }
}
</style>