vue3 终端实现 (vue3+xterm+websocket)

news/2025/1/1 16:18:25/

 

目录

一、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>


http://www.ppmy.cn/news/1237952.html

相关文章

post请求参数全大写后台接不到参数

post请求参数全大写后台接不到参数 开发过程中&#xff0c;我们一般都习惯用驼峰命名法&#xff0c;但是特殊情况要求请求参数全大写&#xff08;或者首字母大写&#xff09;&#xff0c;测试验证的时候发现&#xff0c;接收不到请求参数。 前端请求传递&#xff1a; 服务端接…

【MyBatis <if> <where>标签介绍】

文章目录 <if>标签<where>标签<foreach>标签 <if>标签 <if>标签允许我们在SQL语句中添加条件判断。 <if test"condition"><!-- 当条件满足时执行的SQL语句 --> </if>其中&#xff0c;test属性是一个表达式&#x…

浏览器没收到返回,后端也没报错,php的json_encode问题bug

今天网站遇到个问题&#xff0c;后端返回异常&#xff0c;但是浏览器状态码200&#xff0c;但是看不到结果。经过排查发现&#xff0c;我们在返回结果的时候使用了json_encode返回给前端&#xff0c;结果里面的字符编码异常&#xff0c;导致json_encode异常&#xff0c;但是php…

App Inventor 2 什么情况下需要使用字典?

介绍 字典在其他语言中称为映射、关联数组或列表&#xff0c;是一种将一个值&#xff08;通常称为键&#xff09;与另一个值关联的数据结构。 Q&#xff1a;App Inventor 2 什么情况下需要使用字典&#xff1f; A&#xff1a;列表能完成字典的绝大部分功能&#xff0c;不过字…

告别传统Office,办公软件将如何选择

各家奶茶店的商战正如火如荼地进行着&#xff0c;各种办公软件之间的竞争亦是弥漫着无形的硝烟。WPS虽然凭借其操作便利、简单易上手的优势获得不少打工人的青睐&#xff0c;即使是在手机端&#xff0c;也可进行简单的数据处理。但是&#xff0c;正所谓“术业有专攻”&#xff…

竞赛选题 题目:基于LSTM的预测算法 - 股票预测 天气预测 房价预测

文章目录 0 简介1 基于 Keras 用 LSTM 网络做时间序列预测2 长短记忆网络3 LSTM 网络结构和原理3.1 LSTM核心思想3.2 遗忘门3.3 输入门3.4 输出门 4 基于LSTM的天气预测4.1 数据集4.2 预测示例 5 基于LSTM的股票价格预测5.1 数据集5.2 实现代码 6 lstm 预测航空旅客数目数据集预…

Vue3+vite 处理静态资源,解决服务器不显示动态循环img问题

注意&#xff1a; vue2webpack中&#xff0c;通常使用require来动态渲染静态资源。但在vue3vite中&#xff0c;不支持require语法&#xff0c;因此使用require会报undefined&#xff0c;所以官方推荐使用import来动态渲染静态资源。 实现方式动态渲染静态资源 vue2webpack 使…

Mac如何搭建Vue项目

目录 一、安装node 二、安装NPM 1、本地安装和全局安装 2、通过Node.js官方安装程序安装 3、通过Homebrew安装 三、NPM常用命令 1、查看模块的版本号 2、安装指定版本 3、卸载模块 4、更新模块 5、查看模块信息 6、查看模块地址 7、更新命令 8、卸载NPM 四、安装…