Files
AUTO-MAS-test/frontend/src/views/scheduler/SchedulerLogPanel.vue
2025-09-19 21:36:09 +08:00

268 lines
5.6 KiB
Vue

<template>
<div class="log-panel">
<a-card class="section-card" :bordered="false">
<template #title>
<div class="section-header">
<h3>日志</h3>
<div class="log-controls">
<a-space size="small">
<a-button @click="clearLogs" :disabled="logs.length === 0" size="small">
清空日志
</a-button>
<a-button @click="scrollToBottom" :disabled="logs.length === 0" size="small">
滚动到底部
</a-button>
</a-space>
</div>
</div>
</template>
<div class="log-content" :ref="setLogRef" @scroll="onScroll">
<div v-if="logs.length === 0" class="empty-state-mini">
<a-empty description="暂无日志信息" />
</div>
<div
v-for="(log, index) in logs"
:key="`${tabKey}-${index}-${log.timestamp}`"
:class="['log-line', `log-${log.type}`]"
>
<span class="log-time">{{ log.time }}</span>
<span class="log-message">{{ log.message }}</span>
</div>
</div>
</a-card>
</div>
</template>
<script setup lang="ts">
import { nextTick } from 'vue'
import type { LogEntry } from './schedulerConstants'
interface Props {
logs: LogEntry[]
tabKey: string
isLogAtBottom: boolean
}
interface Emits {
(e: 'scroll', isAtBottom: boolean): void
(e: 'setRef', el: HTMLElement | null, key: string): void
(e: 'clearLogs'): void
}
const props = defineProps<Props>()
const emit = defineEmits<Emits>()
const setLogRef = (el: HTMLElement | null) => {
emit('setRef', el, props.tabKey)
}
const onScroll = (event: Event) => {
const el = event.target as HTMLElement
if (!el) return
const threshold = 5
const isAtBottom = el.scrollHeight - el.scrollTop - el.clientHeight <= threshold
emit('scroll', isAtBottom)
}
const scrollToBottom = () => {
nextTick(() => {
const el = document.querySelector(
`[data-tab-key="${props.tabKey}"] .log-content`
) as HTMLElement
if (el) {
el.scrollTo({
top: el.scrollHeight,
behavior: 'smooth',
})
}
})
}
const clearLogs = () => {
emit('clearLogs')
}
</script>
<style scoped>
.log-panel {
height: 100%;
display: flex;
flex-direction: column;
}
.section-card {
border-radius: 12px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.06);
border: 1px solid var(--ant-color-border-secondary);
height: 100%;
}
.section-card :deep(.ant-card-head) {
border-bottom: 1px solid var(--ant-color-border-secondary);
padding: 0 16px;
border-radius: 12px 12px 0 0;
}
.section-card :deep(.ant-card-body) {
padding: 0;
height: calc(100% - 52px);
}
.section-header {
display: flex;
justify-content: space-between;
align-items: center;
width: 100%;
}
.section-header h3 {
margin: 0;
font-size: 18px;
font-weight: 600;
color: var(--ant-color-text-heading);
}
.log-controls {
display: flex;
gap: 8px;
}
.log-content {
height: 100%;
padding: 16px;
background: var(--ant-color-bg-container);
overflow-y: auto;
font-family: 'Courier New', monospace;
font-size: 13px;
line-height: 1.5;
}
.empty-state-mini {
display: flex;
align-items: center;
justify-content: center;
height: 100%;
min-height: 300px;
}
.log-line {
margin-bottom: 4px;
padding: 4px 8px;
border-radius: 4px;
word-wrap: break-word;
}
.log-time {
color: var(--ant-color-text-secondary);
margin-right: 12px;
font-weight: 500;
}
.log-message {
color: var(--ant-color-text);
}
.log-info {
background-color: transparent;
}
.log-error {
background-color: var(--ant-color-error-bg);
border-left: 4px solid var(--ant-color-error);
}
.log-error .log-message {
color: var(--ant-color-error-text);
}
.log-warning {
background-color: var(--ant-color-warning-bg);
border-left: 4px solid var(--ant-color-warning);
}
.log-warning .log-message {
color: var(--ant-color-warning-text);
}
.log-success {
background-color: var(--ant-color-success-bg);
border-left: 4px solid var(--ant-color-success);
}
.log-success .log-message {
color: var(--ant-color-success-text);
}
/* 暗色模式适配 */
@media (prefers-color-scheme: dark) {
.section-card {
background: var(--ant-color-bg-container, #1f1f1f);
border: 1px solid var(--ant-color-border, #424242);
}
.section-card :deep(.ant-card-head) {
background: var(--ant-color-bg-layout, #141414);
border-bottom: 1px solid var(--ant-color-border, #424242);
}
.section-card :deep(.ant-card-body) {
background: var(--ant-color-bg-container, #1f1f1f);
}
.section-header h3 {
color: var(--ant-color-text-heading, #ffffff);
}
.log-content {
background: var(--ant-color-bg-container, #1f1f1f);
}
.log-time {
color: var(--ant-color-text-secondary, #bfbfbf);
}
.log-message {
color: var(--ant-color-text, #ffffff);
}
.log-error {
background-color: rgba(255, 77, 79, 0.1);
border-left: 4px solid var(--ant-color-error, #ff4d4f);
}
.log-error .log-message {
color: var(--ant-color-error, #ff7875);
}
.log-warning {
background-color: rgba(250, 173, 20, 0.1);
border-left: 4px solid var(--ant-color-warning, #faad14);
}
.log-warning .log-message {
color: var(--ant-color-warning, #ffc53d);
}
.log-success {
background-color: rgba(82, 196, 26, 0.1);
border-left: 4px solid var(--ant-color-success, #52c41a);
}
.log-success .log-message {
color: var(--ant-color-success, #73d13d);
}
}
@media (max-width: 768px) {
.log-content {
padding: 12px;
}
.section-card :deep(.ant-card-head) {
padding: 0 16px;
}
}
</style>