vue插件--xterm封装

2024-01-01 14:20

本文主要是介绍vue插件--xterm封装,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

安装
npm install xterm xterm-addon-fit -D
两种模式
  • log:日志输出
  • shell:终端命令
<template><div :id="id" class="xterm"></div>
</template>
<script>
import { defineComponent, onMounted, onBeforeUnmount, watch, nextTick, ref } from "vue";
import { initWebSocket, closeWebsocket, sendWebsocket } from "../utils/websocket";
import { startWith } from "../utils/string";
//xterm
import { Terminal } from "xterm";
import { FitAddon } from "xterm-addon-fit";
import "xterm/css/xterm.css";
import "xterm/lib/xterm.js";
export default defineComponent({name: "XTerm",props: {type: String,width: Number,data: String,url: String,},setup(props) {let xterm = null;let type = "";let width = ref(0);let id = ref("logxterm");const fitAddon = new FitAddon();type = type ? type : props.type;if (type == "log") {id.value = "logxterm";} else {id.value = "shellterm";}function initTerm() {width.value = width.value ? width.value : props.width;let rows = width.value / 12;xterm = new Terminal({rows: parseInt(rows),cols: 40,cursorStyle: "underline", //光标样式cursorBlink: true, // 光标闪烁convertEol: true, //启用时,光标将设置为下一行的开头disableStdin: true, //是否应禁用输入。theme: {foreground: "white", //字体background: "#334963", //背景色cursor: "help", //设置光标},});xterm.loadAddon(fitAddon);xterm.open(document.getElementById(id.value));fitAddon.fit();if (!xterm._initialized) {xterm._initialized = true;}if (type == "log") {xterm.write("logging--------------------logging\n");if (props.data) xterm.write(props.data);} else if (type == "shell") {xterm.focus();//限制和后端交互,只有输入回车键才显示结果xterm.prompt = function () {xterm.write("\r\n~$ ");};xterm.prompt();xterm.writeln("~ Welcome to the command execution window");let url = props.url.replace("http", "ws");initWebSocket(url);let termdata = "";let oldtermdata = "";let invalidkey = [27,33,34,35,36,37,39,38,40,45,144,9,12,16,17,18,20,112,113,114,115,116,117,118,119,120,121,122,123,175,174,179,173,172,180,170,];xterm.onKey(function (e) {const printable =!e.domEvent.altKey &&!e.domEvent.altGraphKey &&!e.domEvent.ctrlKey &&!e.domEvent.metaKey;if (e.domEvent.keyCode === 13) {//回车let data = 0 + termdata + "\n";sendWebsocket(data, function () {let data = e.data;data = data.substr(1); //去掉第一位,值为1if (startWith(data, oldtermdata + "\r\n")) {xterm.write(data.replace(oldtermdata, ""));} else {xterm.write(data);}});oldtermdata = termdata;termdata = "";} else if (e.domEvent.keyCode === 8) {//删除if (xterm._core.buffer.x > 2) {if (termdata.length > 0) {xterm.write("\b \b");termdata = termdata.substring(0, termdata.length - 1);}}} else if (printable) {if (invalidkey.indexOf(e.domEvent.keyCode) < 0) {//不是特殊字符的时候可以执行oldtermdata = "";termdata = termdata + e.key;xterm.write(e.key);}}});}window.addEventListener("resize", function () {fitAddon.fit();});}function fit() {fitAddon.fit();}function dispose() {xterm.dispose();}watch(props, (newProps) => {type = newProps.type;width.value = newProps.type.width;if (newProps.type == "log") {xterm.write(newProps.data);fit();} else if (newProps.type == "shell") {if (newProps.url) {xterm.dispose();initTerm();}}});onMounted(() => {nextTick(() => {if (!xterm) initTerm();});});onBeforeUnmount(() => {dispose();if (type == "shell") closeWebsocket();});return {initTerm,fit,dispose,id,};},
});
</script>
页面使用:dialog里用加v-if
<!--日志-->
<template><div class="xterm-container"><XTerm:data="logtermdata"type="log"ref="logxterm"class="term":width="width"></XTerm></div><el-dialog v-model="visible" title="终端" width="800px"><div class="xterm-container"><XTermv-if="termVisible"type="shell"ref="shellterm"class="term":width="width":url="websocketUrl"></XTerm></div></el-dialog>
</template>
<script>
import {vue,nexTick,ref} from "vue"
import XTerm from "../components/xterm";
export default {setup() {const width = ref(0)const visible = ref(false)const logxterm = ref(null)function open(){visible.value = true;nextTick(() => {//计算宽度width.value = document.getElementsByClassName("term")[0].offsetWidth;//初始化logxterm.value.initTerm();});}return{width,visible,logxterm,open}},components: {XTerm,}
}
</script>

这篇关于vue插件--xterm封装的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python用Flask封装API及调用详解

《Python用Flask封装API及调用详解》本文介绍Flask的优势(轻量、灵活、易扩展),对比GET/POST表单/JSON请求方式,涵盖错误处理、开发建议及生产环境部署注意事项... 目录一、Flask的优势一、基础设置二、GET请求方式服务端代码客户端调用三、POST表单方式服务端代码客户端调用四

基于Python Playwright进行前端性能测试的脚本实现

《基于PythonPlaywright进行前端性能测试的脚本实现》在当今Web应用开发中,性能优化是提升用户体验的关键因素之一,本文将介绍如何使用Playwright构建一个自动化性能测试工具,希望... 目录引言工具概述整体架构核心实现解析1. 浏览器初始化2. 性能数据收集3. 资源分析4. 关键性能指

从入门到精通详解LangChain加载HTML内容的全攻略

《从入门到精通详解LangChain加载HTML内容的全攻略》这篇文章主要为大家详细介绍了如何用LangChain优雅地处理HTML内容,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录引言:当大语言模型遇见html一、HTML加载器为什么需要专门的HTML加载器核心加载器对比表二

Spring Boot Maven 插件如何构建可执行 JAR 的核心配置

《SpringBootMaven插件如何构建可执行JAR的核心配置》SpringBoot核心Maven插件,用于生成可执行JAR/WAR,内置服务器简化部署,支持热部署、多环境配置及依赖管理... 目录前言一、插件的核心功能与目标1.1 插件的定位1.2 插件的 Goals(目标)1.3 插件定位1.4 核

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件

浏览器插件cursor实现自动注册、续杯的详细过程

《浏览器插件cursor实现自动注册、续杯的详细过程》Cursor简易注册助手脚本通过自动化邮箱填写和验证码获取流程,大大简化了Cursor的注册过程,它不仅提高了注册效率,还通过友好的用户界面和详细... 目录前言功能概述使用方法安装脚本使用流程邮箱输入页面验证码页面实战演示技术实现核心功能实现1. 随机

Golang如何对cron进行二次封装实现指定时间执行定时任务

《Golang如何对cron进行二次封装实现指定时间执行定时任务》:本文主要介绍Golang如何对cron进行二次封装实现指定时间执行定时任务问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录背景cron库下载代码示例【1】结构体定义【2】定时任务开启【3】使用示例【4】控制台输出总结背景

前端如何通过nginx访问本地端口

《前端如何通过nginx访问本地端口》:本文主要介绍前端如何通过nginx访问本地端口的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、nginx安装1、下载(1)下载地址(2)系统选择(3)版本选择2、安装部署(1)解压(2)配置文件修改(3)启动(4)

HTML中meta标签的常见使用案例(示例详解)

《HTML中meta标签的常见使用案例(示例详解)》HTMLmeta标签用于提供文档元数据,涵盖字符编码、SEO优化、社交媒体集成、移动设备适配、浏览器控制及安全隐私设置,优化页面显示与搜索引擎索引... 目录html中meta标签的常见使用案例一、基础功能二、搜索引擎优化(seo)三、社交媒体集成四、移动

HTML input 标签示例详解

《HTMLinput标签示例详解》input标签主要用于接收用户的输入,随type属性值的不同,变换其具体功能,本文通过实例图文并茂的形式给大家介绍HTMLinput标签,感兴趣的朋友一... 目录通用属性输入框单行文本输入框 text密码输入框 password数字输入框 number电子邮件输入编程框