NapCatQQ/packages/napcat-webui-frontend/src/components/log_com/realtime.tsx
手瓜一十雪 4360775eff
refactor: 整体重构 (#1381)
* feat: pnpm new

* Refactor build and release workflows, update dependencies

Switch build scripts and workflows from npm to pnpm, update build and artifact paths, and simplify release workflow by removing version detection and changelog steps. Add new dependencies (silk-wasm, express, ws, node-pty-prebuilt-multiarch), update exports in package.json files, and add vite config for napcat-framework. Also, rename manifest.json for framework package and fix static asset copying in shell build config.
2025-11-13 15:39:42 +08:00

115 lines
3.0 KiB
TypeScript

import { Button } from '@heroui/button';
import type { Selection } from '@react-types/shared';
import { useEffect, useRef, useState } from 'react';
import toast from 'react-hot-toast';
import { IoDownloadOutline } from 'react-icons/io5';
import { colorizeLogLevelWithTag } from '@/utils/terminal';
import WebUIManager, { Log } from '@/controllers/webui_manager';
import type { XTermRef } from '../xterm';
import XTerm from '../xterm';
import LogLevelSelect from './log_level_select';
const RealTimeLogs = () => {
const Xterm = useRef<XTermRef>(null);
const [logLevel, setLogLevel] = useState<Selection>(
new Set(['info', 'warn', 'error'])
);
const [dataArr, setDataArr] = useState<Log[]>([]);
const onDownloadLog = () => {
const logContent = dataArr
.filter((log) => {
if (logLevel === 'all') {
return true;
}
return logLevel.has(log.level);
})
.map((log) => colorizeLogLevelWithTag(log.message, log.level))
.join('\r\n');
const blob = new Blob([logContent], { type: 'text/plain' });
const url = URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
a.download = 'napcat.log';
a.click();
URL.revokeObjectURL(url);
};
const writeStream = () => {
try {
const _data = dataArr
.filter((log) => {
if (logLevel === 'all') {
return true;
}
return logLevel.has(log.level);
})
.map((log) => colorizeLogLevelWithTag(log.message, log.level))
.join('\r\n');
Xterm.current?.clear();
Xterm.current?.write(_data);
} catch (error) {
console.error(error);
toast.error('获取实时日志失败');
}
};
useEffect(() => {
writeStream();
}, [logLevel, dataArr]);
useEffect(() => {
const subscribeLogs = () => {
try {
const source = WebUIManager.getRealTimeLogs((data) => {
setDataArr((prev) => {
const newData = [...prev, ...data];
if (newData.length > 1000) {
newData.splice(0, newData.length - 1000);
}
return newData;
});
});
return () => {
source.close();
};
} catch (_error) {
toast.error('获取实时日志失败');
}
};
const close = subscribeLogs();
return () => {
console.log('close');
close?.();
};
}, []);
return (
<>
<title> - NapCat WebUI</title>
<div className='flex items-center gap-2'>
<LogLevelSelect
selectedKeys={logLevel}
onSelectionChange={setLogLevel}
/>
<Button
className='flex-shrink-0'
onPress={onDownloadLog}
startContent={<IoDownloadOutline className='text-lg' />}
>
</Button>
</div>
<div className='flex-1 h-full overflow-hidden'>
<XTerm ref={Xterm} />
</div>
</>
);
};
export default RealTimeLogs;