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将博客内容html导出为Markdown格式

《Python将博客内容html导出为Markdown格式》Python将博客内容html导出为Markdown格式,通过博客url地址抓取文章,分析并提取出文章标题和内容,将内容构建成html,再转... 目录一、为什么要搞?二、准备如何搞?三、说搞咱就搞!抓取文章提取内容构建html转存markdown

在React中引入Tailwind CSS的完整指南

《在React中引入TailwindCSS的完整指南》在现代前端开发中,使用UI库可以显著提高开发效率,TailwindCSS是一个功能类优先的CSS框架,本文将详细介绍如何在Reac... 目录前言一、Tailwind css 简介二、创建 React 项目使用 Create React App 创建项目

vue使用docxtemplater导出word

《vue使用docxtemplater导出word》docxtemplater是一种邮件合并工具,以编程方式使用并处理条件、循环,并且可以扩展以插入任何内容,下面我们来看看如何使用docxtempl... 目录docxtemplatervue使用docxtemplater导出word安装常用语法 封装导出方

SpringBoot中封装Cors自动配置方式

《SpringBoot中封装Cors自动配置方式》:本文主要介绍SpringBoot中封装Cors自动配置方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录SpringBoot封装Cors自动配置背景实现步骤1. 创建 GlobalCorsProperties

Vue中组件之间传值的六种方式(完整版)

《Vue中组件之间传值的六种方式(完整版)》组件是vue.js最强大的功能之一,而组件实例的作用域是相互独立的,这就意味着不同组件之间的数据无法相互引用,针对不同的使用场景,如何选择行之有效的通信方式... 目录前言方法一、props/$emit1.父组件向子组件传值2.子组件向父组件传值(通过事件形式)方

css中的 vertical-align与line-height作用详解

《css中的vertical-align与line-height作用详解》:本文主要介绍了CSS中的`vertical-align`和`line-height`属性,包括它们的作用、适用元素、属性值、常见使用场景、常见问题及解决方案,详细内容请阅读本文,希望能对你有所帮助... 目录vertical-ali

浅析CSS 中z - index属性的作用及在什么情况下会失效

《浅析CSS中z-index属性的作用及在什么情况下会失效》z-index属性用于控制元素的堆叠顺序,值越大,元素越显示在上层,它需要元素具有定位属性(如relative、absolute、fi... 目录1. z-index 属性的作用2. z-index 失效的情况2.1 元素没有定位属性2.2 元素处

Python实现html转png的完美方案介绍

《Python实现html转png的完美方案介绍》这篇文章主要为大家详细介绍了如何使用Python实现html转png功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 1.增强稳定性与错误处理建议使用三层异常捕获结构:try: with sync_playwright(

Vue 调用摄像头扫描条码功能实现代码

《Vue调用摄像头扫描条码功能实现代码》本文介绍了如何使用Vue.js和jsQR库来实现调用摄像头并扫描条码的功能,通过安装依赖、获取摄像头视频流、解析条码等步骤,实现了从开始扫描到停止扫描的完整流... 目录实现步骤:代码实现1. 安装依赖2. vue 页面代码功能说明注意事项以下是一个基于 Vue.js

CSS @media print 使用详解

《CSS@mediaprint使用详解》:本文主要介绍了CSS中的打印媒体查询@mediaprint包括基本语法、常见使用场景和代码示例,如隐藏非必要元素、调整字体和颜色、处理链接的URL显示、分页控制、调整边距和背景等,还提供了测试方法和关键注意事项,并分享了进阶技巧,详细内容请阅读本文,希望能对你有所帮助...