目录
一、xterm介绍
二、效果展示
三、vue文件实现代码
一、xterm介绍
xterm是一个使用 TypeScript 编写的前端终端组件,可以直接在浏览器中实现一个命令行终端应用,通常与websocket
一起使用。
二、效果展示
三、vue文件实现代码
<template><div class="bg-main"><divref="terminal"v-loading="loading"class="terminal"element-loading-text="拼命连接中"></div></div>
</template>
<script setup>import { ref, onMounted, onBeforeUnmount } from 'vue'import { debounce } from 'lodash'import { Terminal } from 'xterm'import { FitAddon } from 'xterm-addon-fit'import 'xterm/css/xterm.css'const terminal = ref(null)const fitAddon = new FitAddon()let first = ref(true)let loading = ref(true)let terminalSocket = ref(null)let term = ref(null)// 初始化WSconst initWS = () => {if (!terminalSocket.value) {createWS()}if (terminalSocket.value && terminalSocket.value.readyState > 1) {terminalSocket.value.close()createWS()}}// 创建WSconst createWS = () => {// const url = `/access/Api/ws/ssh/b172df81-2485-453d-a6ff-120c03821536?userName=test&passwd=1`terminalSocket.value = new WebSocket(`wss://XXXX`)terminalSocket.value.onopen = runRealTerminal //WebSocket 连接已建立terminalSocket.value.onmessage = onWSReceive //收到服务器消息terminalSocket.value.onclose = closeRealTerminal //WebSocket 连接已关闭terminalSocket.value.onerror = errorRealTerminal //WebSocket 连接出错}//WebSocket 连接已建立const runRealTerminal = () => {loading.value = false}//WebSocket收到服务器消息const onWSReceive = (message) => {// 首次接收消息,发送给后端,进行同步适配尺寸if (first.value === true) {first.value = falseresizeRemoteTerminal()}const data = message.data// base64解密const reader = new FileReader()reader.onload = function (e) {const base64Content = e.target.resultconsole.log(base64Content, 1)term.value.write(base64Content)}reader.readAsText(data) // 以text文本显示readAsTextterm.value.element && term.value.focus()}//WebSocket 连接出错const errorRealTerminal = (ex) => {let message = ex.messageif (!message) message = 'disconnected'term.value.write(`\x1b[31m${message}\x1b[m\r\n`)console.log('err')}//WebSocket 连接已关闭const closeRealTerminal = () => {console.log('close')}// 初始化Terminalconst initTerm = () => {term.value = new Terminal({// lineHeight: 1.2,fontSize: 14,fontFamily: "Monaco, Menlo, Consolas, 'Courier New', monospace",theme: {background: '#181d28',},// 光标闪烁cursorBlink: true,cursorStyle: 'underline',// scrollback: 100,// tabStopWidth: 4,})term.value.open(terminal.value) //挂载dom窗口term.value.loadAddon(fitAddon) //自适应尺寸// 不能初始化的时候fit,需要等terminal准备就绪,可以设置延时操作setTimeout(() => {fitAddon.fit()}, 5)termData() //Terminal 事件挂载}// 终端输入触发事件const termData = () => {// 输入与粘贴的情况,onData不能重复绑定,不然会发送多次term.value.onData((data) => {console.log(data, '传入服务器')if (isWsOpen()) {terminalSocket.value.send(JSON.stringify({type: 'terminal',data: {base64: btoa(data),},}))}})// 终端尺寸变化触发term.value.onResize(() => {resizeRemoteTerminal()})}//尺寸同步 发送给后端,调整后端终端大小,和前端保持一致,不然前端只是范围变大了,命令还是会换行const resizeRemoteTerminal = () => {const { cols, rows } = term.valueif (isWsOpen()) {terminalSocket.value.send(JSON.stringify({type: 'resize',data: {rows: rows,cols: cols,},}))}}// 是否连接中0 1 2 3 状态const isWsOpen = () => {const readyState = terminalSocket.value && terminalSocket.value.readyStatereturn readyState === 1}// 适应浏览器尺寸变化const fitTerm = () => {fitAddon.fit()}const onResize = debounce(() => fitTerm(), 500)const onTerminalResize = () => {window.addEventListener('resize', onResize)}const removeResizeListener = () => {window.removeEventListener('resize', onResize)}onMounted(() => {initWS()initTerm()onTerminalResize()})onBeforeUnmount(() => {removeResizeListener()terminalSocket.value && terminalSocket.value.close()})
</script>
<style lang="scss" scoped>.terminal {width: 100%;height: calc(100% - 62px);}
</style>