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

2023-11-25 02:15

本文主要是介绍vue3 终端实现 (vue3+xterm+websocket),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

目录

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

这篇关于vue3 终端实现 (vue3+xterm+websocket)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/422958

相关文章

SpringBoot集成XXL-JOB实现任务管理全流程

《SpringBoot集成XXL-JOB实现任务管理全流程》XXL-JOB是一款轻量级分布式任务调度平台,功能丰富、界面简洁、易于扩展,本文介绍如何通过SpringBoot项目,使用RestTempl... 目录一、前言二、项目结构简述三、Maven 依赖四、Controller 代码详解五、Service

Python 基于http.server模块实现简单http服务的代码举例

《Python基于http.server模块实现简单http服务的代码举例》Pythonhttp.server模块通过继承BaseHTTPRequestHandler处理HTTP请求,使用Threa... 目录测试环境代码实现相关介绍模块简介类及相关函数简介参考链接测试环境win11专业版python

GO语言短变量声明的实现示例

《GO语言短变量声明的实现示例》在Go语言中,短变量声明是一种简洁的变量声明方式,使用:=运算符,可以自动推断变量类型,下面就来具体介绍一下如何使用,感兴趣的可以了解一下... 目录基本语法功能特点与var的区别适用场景注意事项基本语法variableName := value功能特点1、自动类型推

基于Python实现自动化邮件发送系统的完整指南

《基于Python实现自动化邮件发送系统的完整指南》在现代软件开发和自动化流程中,邮件通知是一个常见且实用的功能,无论是用于发送报告、告警信息还是用户提醒,通过Python实现自动化的邮件发送功能都能... 目录一、前言:二、项目概述三、配置文件 `.env` 解析四、代码结构解析1. 导入模块2. 加载环

使用shardingsphere实现mysql数据库分片方式

《使用shardingsphere实现mysql数据库分片方式》本文介绍如何使用ShardingSphere-JDBC在SpringBoot中实现MySQL水平分库,涵盖分片策略、路由算法及零侵入配置... 目录一、ShardingSphere 简介1.1 对比1.2 核心概念1.3 Sharding-Sp

Java+AI驱动实现PDF文件数据提取与解析

《Java+AI驱动实现PDF文件数据提取与解析》本文将和大家分享一套基于AI的体检报告智能评估方案,详细介绍从PDF上传、内容提取到AI分析、数据存储的全流程自动化实现方法,感兴趣的可以了解下... 目录一、核心流程:从上传到评估的完整链路二、第一步:解析 PDF,提取体检报告内容1. 引入依赖2. 封装

Java实现复杂查询优化的7个技巧小结

《Java实现复杂查询优化的7个技巧小结》在Java项目中,复杂查询是开发者面临的“硬骨头”,本文将通过7个实战技巧,结合代码示例和性能对比,手把手教你如何让复杂查询变得优雅,大家可以根据需求进行选择... 目录一、复杂查询的痛点:为何你的代码“又臭又长”1.1冗余变量与中间状态1.2重复查询与性能陷阱1.

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

Redis实现分布式锁全过程

《Redis实现分布式锁全过程》文章介绍Redis实现分布式锁的方法,包括使用SETNX和EXPIRE命令确保互斥性与防死锁,Redisson客户端提供的便捷接口,以及Redlock算法通过多节点共识... 目录Redis实现分布式锁1. 分布式锁的基本原理2. 使用 Redis 实现分布式锁2.1 获取锁

Linux实现查看某一端口是否开放

《Linux实现查看某一端口是否开放》文章介绍了三种检查端口6379是否开放的方法:通过lsof查看进程占用,用netstat区分TCP/UDP监听状态,以及用telnet测试远程连接可达性... 目录1、使用lsof 命令来查看端口是否开放2、使用netstat 命令来查看端口是否开放3、使用telnet