mirror of
https://gitee.com/dromara/mayfly-go
synced 2025-12-08 08:50:25 +08:00
refactor: 终端重构、系统参数配置调整
This commit is contained in:
286
mayfly_go_web/src/components/terminal/TerminalBody.vue
Normal file
286
mayfly_go_web/src/components/terminal/TerminalBody.vue
Normal file
@@ -0,0 +1,286 @@
|
||||
<template>
|
||||
<div id="terminal-body" :style="{ height, background: themeConfig.terminalBackground }">
|
||||
<div ref="terminalRef" class="terminal" />
|
||||
|
||||
<TerminalSearch ref="terminalSearchRef" :search-addon="state.addon.search" @close="focus" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script lang="ts" setup>
|
||||
import 'xterm/css/xterm.css';
|
||||
import { Terminal } from 'xterm';
|
||||
import { FitAddon } from 'xterm-addon-fit';
|
||||
import { SearchAddon } from 'xterm-addon-search';
|
||||
import { WebLinksAddon } from 'xterm-addon-web-links';
|
||||
|
||||
import { storeToRefs } from 'pinia';
|
||||
import { useThemeConfig } from '@/store/themeConfig';
|
||||
import { ref, nextTick, reactive, onMounted, onBeforeUnmount, watch } from 'vue';
|
||||
import TerminalSearch from './TerminalSearch.vue';
|
||||
import { debounce } from 'lodash';
|
||||
import { TerminalStatus } from './common';
|
||||
|
||||
const props = defineProps({
|
||||
/**
|
||||
* 初始化执行命令
|
||||
*/
|
||||
cmd: { type: String },
|
||||
/**
|
||||
* 连接url
|
||||
*/
|
||||
socketUrl: {
|
||||
type: String,
|
||||
},
|
||||
/**
|
||||
* 高度
|
||||
*/
|
||||
height: {
|
||||
type: [String, Number],
|
||||
default: '100%',
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['statusChange']);
|
||||
|
||||
const terminalRef: any = ref(null);
|
||||
const terminalSearchRef: any = ref(null);
|
||||
|
||||
const { themeConfig } = storeToRefs(useThemeConfig());
|
||||
|
||||
// 终端实例
|
||||
let term: Terminal;
|
||||
let socket: WebSocket;
|
||||
let pingInterval: any;
|
||||
|
||||
const state = reactive({
|
||||
// 插件
|
||||
addon: {
|
||||
fit: null as any,
|
||||
search: null as any,
|
||||
weblinks: null as any,
|
||||
},
|
||||
status: TerminalStatus.NoConnected,
|
||||
});
|
||||
|
||||
onMounted(() => {
|
||||
nextTick(() => {
|
||||
init();
|
||||
});
|
||||
});
|
||||
|
||||
watch(
|
||||
() => state.status,
|
||||
() => {
|
||||
emit('statusChange', state.status);
|
||||
}
|
||||
);
|
||||
|
||||
onBeforeUnmount(() => {
|
||||
close();
|
||||
});
|
||||
|
||||
function init() {
|
||||
if (term) {
|
||||
console.log('重新连接...');
|
||||
close();
|
||||
}
|
||||
term = new Terminal({
|
||||
fontSize: themeConfig.value.terminalFontSize || 15,
|
||||
fontWeight: themeConfig.value.terminalFontWeight || 'normal',
|
||||
fontFamily: 'JetBrainsMono, monaco, Consolas, Lucida Console, monospace',
|
||||
cursorBlink: true,
|
||||
disableStdin: false,
|
||||
allowProposedApi: true,
|
||||
theme: {
|
||||
foreground: themeConfig.value.terminalForeground || '#7e9192', //字体
|
||||
background: themeConfig.value.terminalBackground || '#002833', //背景色
|
||||
cursor: themeConfig.value.terminalCursor || '#268F81', //设置光标
|
||||
// cursorAccent: "red", // 光标停止颜色
|
||||
} as any,
|
||||
});
|
||||
term.open(terminalRef.value);
|
||||
|
||||
// 注册自适应组件
|
||||
const fitAddon = new FitAddon();
|
||||
state.addon.fit = fitAddon;
|
||||
term.loadAddon(fitAddon);
|
||||
fitTerminal();
|
||||
|
||||
// 注册搜索组件
|
||||
const searchAddon = new SearchAddon();
|
||||
state.addon.search = searchAddon;
|
||||
term.loadAddon(searchAddon);
|
||||
|
||||
// 注册 url link组件
|
||||
const weblinks = new WebLinksAddon();
|
||||
state.addon.weblinks = weblinks;
|
||||
term.loadAddon(weblinks);
|
||||
|
||||
// 初始化websocket
|
||||
initSocket();
|
||||
}
|
||||
|
||||
/**
|
||||
* 连接成功
|
||||
*/
|
||||
const onConnected = () => {
|
||||
// 注册心跳
|
||||
pingInterval = setInterval(sendPing, 15000);
|
||||
|
||||
// 注册 terminal 事件
|
||||
term.onResize((event) => sendResize(event.cols, event.rows));
|
||||
term.onData((event) => sendCmd(event));
|
||||
|
||||
// 注册自定义快捷键
|
||||
term.attachCustomKeyEventHandler((event: KeyboardEvent) => {
|
||||
// 注册搜索键 ctrl + f
|
||||
if (event.key === 'f' && (event.ctrlKey || event.metaKey) && event.type === 'keydown') {
|
||||
event.preventDefault();
|
||||
terminalSearchRef.value.open();
|
||||
}
|
||||
|
||||
return true;
|
||||
});
|
||||
|
||||
// resize
|
||||
sendResize(term.cols, term.rows);
|
||||
// 注册窗口大小监听器
|
||||
window.addEventListener('resize', debounce(fitTerminal, 400));
|
||||
|
||||
focus();
|
||||
|
||||
// 如果有初始要执行的命令,则发送执行命令
|
||||
if (props.cmd) {
|
||||
sendCmd(props.cmd + ' \r');
|
||||
}
|
||||
|
||||
state.status = TerminalStatus.Connected;
|
||||
};
|
||||
|
||||
// 自适应终端
|
||||
const fitTerminal = () => {
|
||||
const dimensions = state.addon.fit && state.addon.fit.proposeDimensions();
|
||||
if (!dimensions) {
|
||||
return;
|
||||
}
|
||||
if (dimensions?.cols && dimensions?.rows) {
|
||||
term.resize(dimensions.cols, dimensions.rows);
|
||||
}
|
||||
};
|
||||
|
||||
const focus = () => {
|
||||
setTimeout(() => term.focus(), 400);
|
||||
};
|
||||
|
||||
const clear = () => {
|
||||
term.clear();
|
||||
term.clearSelection();
|
||||
term.focus();
|
||||
};
|
||||
|
||||
function initSocket() {
|
||||
if (props.socketUrl) {
|
||||
socket = new WebSocket(props.socketUrl);
|
||||
}
|
||||
|
||||
// 监听socket连接
|
||||
socket.onopen = () => {
|
||||
onConnected();
|
||||
};
|
||||
|
||||
// 监听socket错误信息
|
||||
socket.onerror = (e: Event) => {
|
||||
term.writeln('\r\n\x1b[31m提示: 连接错误...');
|
||||
state.status = TerminalStatus.Error;
|
||||
console.log('连接错误', e);
|
||||
};
|
||||
|
||||
socket.onclose = (e: CloseEvent) => {
|
||||
console.log('terminal socket close...', e.reason);
|
||||
// 关闭窗口大小监听器
|
||||
window.removeEventListener('resize', debounce(fitTerminal, 100));
|
||||
// 清除 ping
|
||||
pingInterval && clearInterval(pingInterval);
|
||||
state.status = TerminalStatus.Disconnected;
|
||||
};
|
||||
|
||||
// 监听socket消息
|
||||
socket.onmessage = getMessage;
|
||||
}
|
||||
|
||||
function getMessage(msg: any) {
|
||||
// msg.data是真正后端返回的数据
|
||||
term.write(msg.data);
|
||||
}
|
||||
|
||||
enum MsgType {
|
||||
Resize = 1,
|
||||
Data = 2,
|
||||
Ping = 3,
|
||||
}
|
||||
|
||||
const send = (msg: any) => {
|
||||
socket.send(JSON.stringify(msg));
|
||||
};
|
||||
|
||||
const sendResize = (cols: number, rows: number) => {
|
||||
send({
|
||||
type: MsgType.Resize,
|
||||
Cols: cols,
|
||||
Rows: rows,
|
||||
});
|
||||
};
|
||||
|
||||
const sendPing = () => {
|
||||
send({
|
||||
type: MsgType.Ping,
|
||||
msg: 'ping',
|
||||
});
|
||||
};
|
||||
|
||||
function sendCmd(key: any) {
|
||||
send({
|
||||
type: MsgType.Data,
|
||||
msg: key,
|
||||
});
|
||||
}
|
||||
|
||||
function closeSocket() {
|
||||
// 关闭 websocket
|
||||
socket && socket.readyState === 1 && socket.close();
|
||||
// 清除 ping
|
||||
pingInterval && clearInterval(pingInterval);
|
||||
}
|
||||
|
||||
function close() {
|
||||
console.log('in terminal body close');
|
||||
closeSocket();
|
||||
if (term) {
|
||||
state.addon.search.dispose();
|
||||
state.addon.fit.dispose();
|
||||
state.addon.weblinks.dispose();
|
||||
term.dispose();
|
||||
}
|
||||
}
|
||||
|
||||
const getStatus = (): TerminalStatus => {
|
||||
return state.status;
|
||||
};
|
||||
|
||||
defineExpose({ init, fitTerminal, focus, clear, close, getStatus });
|
||||
</script>
|
||||
<style lang="scss">
|
||||
#terminal-body {
|
||||
background: #212529;
|
||||
width: 100%;
|
||||
|
||||
.terminal {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
|
||||
.xterm .xterm-viewport {
|
||||
overflow-y: hidden;
|
||||
}
|
||||
}
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user