xterm + vue3 + websocket 终端界面

 xterm.js    下载插件

// xterm
npm install --save xterm

// xterm-addon-fit    使终端适应包含元素
npm install --save xterm-addon-fit

// xterm-addon-attach   通过websocket附加到运行中的服务器进程
npm install --save xterm-addon-attach
<template>
  <div :class="props.type ? 'height305' : 'height160'">
    <el-row>
      <el-col :span="20">
        <div
          :class="['xterm', props.type ? 'heightA' : 'heightB']"
          ref="terminal"
          v-loading="loading"
          element-loading-text="拼命连接中"
        >
          <div class="terminal" id="terminal" ref="terminal"></div>
        </div>
        <div class="textarea">
          <textarea ref="textarea" v-model="quickCmd" />
          <div class="bottomOperate flexEnd">
            <el-button type="primary" @click="sendCmd" :disabled="!quickCmd"
              >发送命令</el-button
            >
          </div>
        </div>
      </el-col>
      <el-col :span="4">
        <div :class="['xtermR', props.type ? 'heightA' : 'heightBR']">
          <el-tabs
            v-model="tabActiveName"
            class="demo-tabs"
            @tab-click="handleClick"
          >
            <el-tab-pane label="常用命令" name="first">
              <div v-if="filteredGroups?.length > 0">
                <div class="marginBottom10">
                  <el-button
                    type="primary"
                    size="small"
                    @click="addCmdGroup('addGroup')"
                    >新增命令组</el-button
                  >
                  <el-button type="primary" size="small" @click="addCmd('add')"
                    >新增命令</el-button
                  >
                </div>
                <el-collapse
                  v-loading="loadingR"
                  :class="props.type ? 'listBoxA' : 'listBoxB'"
                >
                  <el-collapse-item
                    v-for="group in filteredGroups"
                    :name="group.name"
                    :key="group.name"
                    class="custom-collapse-item"
                  >
                    <template #title>
                      <div
                        class="flexSpaceBetween"
                        style="width: 100%"
                        @mouseenter="showActions(group.id, true)"
                        @mouseleave="showActions(group.id, false)"
                      >
                        <span class="collapse-title">{{ group.name }}</span>
                        <span v-show="actionStates[group.id]">
                          <el-button
                            link
                            type="primary"
                            @click="addCmdGroup('editGroup', group, $event)"
                            >编辑</el-button
                          >
                          <el-button
                            link
                            type="primary"
                            @click="del(group.id, 'group', $event)"
                            >删除</el-button
                          >
                        </span>
                      </div>
                    </template>
                    <template #default>
                      <div
                        v-for="item in group.device_command"
                        :key="item.id"
                        class="item flexSpaceBetween paddingRight20 marginBottom10"
                        @mouseenter="showActions1(item.id, true)"
                        @mouseleave="showActions1(item.id, false)"
                      >
                        <span
                          class="usualName"
                          @click="getName(item.name)"
                          :title="item.name"
                          >{{ item.name }}</span
                        >
                        <span v-show="actionStates1[item.id]" class="btns">
                          <el-button
                            link
                            type="primary"
                            @click="addCmd('edit', item, group.id)"
                            >编辑</el-button
                          >
                          <el-button link type="primary" @click="del(item.id)"
                            >删除</el-button
                          >
                        </span>
                      </div>
                    </template>
                  </el-collapse-item>
                </el-collapse>
              </div>
              <div class="flexCenter" v-else>暂无常用命令</div>
            </el-tab-pane>
            <el-tab-pane label="命令记录" name="second">
              <div
                :class="props.type ? 'listBoxA' : 'listBoxB'"
                v-if="globalStore.cmdRecordList?.length > 0"
              >
                <div
                  v-for="item in globalStore.cmdRecordList"
                  :key="item"
                  class="item flexSpaceBetween paddingRight20 marginBottom10"
                >
                  <span class="recordName" @click="getName(item)">{{
                    item
                  }}</span>
                </div>
              </div>
              <div class="flexCenter" v-else>暂无命令记录</div>
            </el-tab-pane>
          </el-tabs>
        </div>
      </el-col>
    </el-row>
  </div>
  <!-- 新增命令组 -->
  <AddTerminalGroup ref="addTerminalGroup" />
  <!-- 新增命令 -->
  <AddTerminal ref="addTerminal" />
</template>
<script setup>
import "xterm/css/xterm.css";
import { Terminal } from "xterm";
import { FitAddon } from "xterm-addon-fit";
import { debounce } from "lodash";
import { ElMessage, ElMessageBox } from "element-plus";
import {
  ref,
  reactive,
  onMounted,
  onBeforeUnmount,
  computed,
  nextTick,
  getCurrentInstance,
} from "vue";
import { useGlobalStore } from "@/stores/modules/global.js";
import AddTerminalGroup from "./AddTerminalGroup.vue";
import AddTerminal from "./AddTerminal.vue";
import {
  commandGroupIndex,
  commandGroupDel,
  commandDel,
} from "@/api/equipment";
import { WebSocketUrl } from "@/api/request";

const props = defineProps({
  type: {
    type: String,
    default: () => {
      return "";
    },
  },
  currentPathRes: {
    type: String,
    default: () => {
      return "/";
    },
  },
});
const globalStore = useGlobalStore();
const { proxy } = getCurrentInstance();
const searchTerm = ref("");
const tabActiveName = ref("first");
const cmdRecordList = ref(globalStore.cmdRecordList); // 命令历史记录
const loadingR = ref(false);
const groups = ref([]);
const quickCmd = ref("");
const actionStates = ref({});
const actionStates1 = ref({});

const filteredGroups = computed(() => {
  if (!searchTerm.value) {
    return groups.value;
  }
  return groups.value
    .map((group) => {
      const filteredItems = group.device_command.filter((item) =>
        item.includes(searchTerm.value)
      );
      return {
        ...group,
        device_command: filteredItems,
      };
    })
    .filter((group) => group.device_command.length > 0);
});

const showActions = (id, show) => {
  actionStates.value[id] = show;
};

const showActions1 = (id, show) => {
  actionStates1.value[id] = show;
};

const addCmdGroup = (type, row, event) => {
  if (event) event.stopPropagation();
  nextTick(() => {
    proxy.$refs["addTerminalGroup"].showDialog({
      type,
      row,
    });
  });
};

const addCmd = (type, row, group_id) => {
  nextTick(() => {
    proxy.$refs["addTerminal"].showDialog({
      type,
      groupList: groups.value,
      row,
      group_id,
    });
  });
};

const getName = (val) => {
  quickCmd.value = val;
};

// 发送命令
const sendCmd = () => {
  if (isWsOpen()) {
    terminalSocket.value.send(quickCmd.value);
    // 处理命令历史记录
    handleCmdRecordList(quickCmd.value);
  }
};

const handleCmdRecordList = (newCmd) => {
  if (newCmd) {
    // 对新命令进行trim处理
    const trimmedCmd = newCmd.trim();
    // 检查是否有重复值并删除
    const index = cmdRecordList.value.indexOf(trimmedCmd);
    if (index !== -1) {
      cmdRecordList.value.splice(index, 1);
    }

    // 将新命令添加到数组最前面
    cmdRecordList.value.unshift(trimmedCmd);

    globalStore.setCmdRecordList(cmdRecordList.value);
  }
};

const del = (id, group, event) => {
  if (event) event.stopPropagation();

  ElMessageBox.confirm("确认删除吗?", "删除", {
    confirmButtonText: "确定",
    cancelButtonText: "取消",
    type: "warning",
  })
    .then(() => {
      if (group) {
        commandGroupDel({ id }).then((res) => {
          if (res.status === 200) {
            ElMessage.success("删除成功");
            getTableData();
          }
        });
      } else {
        commandDel({ id }).then((res) => {
          if (res.status === 200) {
            ElMessage.success("删除成功");
            getTableData();
          }
        });
      }
    })
    .catch(() => {});
};

//获取表格数据
const getTableData = () => {
  loadingR.value = true;
  commandGroupIndex()
    .then((res) => {
      loadingR.value = false;
      if (res.status === 200) {
        groups.value = res.data?.list;
      }
    })
    .catch((error) => {
      loadingR.value = false;
    });
};
// 命令列表
getTableData();
//终端信息
const loading = ref(false);
const terminal = ref(null);
const fitAddon = new FitAddon();
let first = ref(true);
let terminalSocket = ref(null);
let term = ref(null);

// 初始化WS
const initWS = () => {
  if (!terminalSocket.value) {
    createWS();
  }

  if (terminalSocket.value && terminalSocket.value.readyState > 1) {
    terminalSocket.value.close();
    createWS();
  }
};

// 创建WS
const createWS = () => {
  loading.value = true;
  terminalSocket.value = new WebSocket(
    WebSocketUrl + globalStore.wsUrl
  );
  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;
  let sendData = JSON.stringify({
    t: "conn",
  });
  terminalSocket.value.send(sendData);
};
//WebSocket收到服务器消息
const onWSReceive = (event) => {
  // 首次接收消息,发送给后端,进行同步适配尺寸
  if (first.value === true) {
    first.value = false;
    resizeRemoteTerminal();
    if (props.type === "termDia") {
      autoWriteInfo();
    }
  }
  const blob = new Blob([event.data.toString()], {
    type: "text/plain",
  });
  //将Blob 对象转换成字符串
  const reader = new FileReader();
  reader.readAsText(blob, "utf-8");
  reader.onload = (e) => {
    // 可以根据返回值判断使用何种颜色或者字体,不过返回值自带了一些字体颜色
    writeOfColor(reader.result);
  };
};

//WebSocket 连接出错
const errorRealTerminal = (ex) => {
  let message = ex.message;
  if (!message) message = "disconnected";
  term.value.write(`\x1b[31m${message}\x1b[m\r\n`);
  loading.value = false;
};
//WebSocket 连接已关闭
const closeRealTerminal = () => {
  loading.value = false;
};

// 初始化Terminal
const initTerm = () => {
  term.value = new Terminal({
    rendererType: "canvas", //渲染类型
    // rows: 50, //行数,影响最小高度
    // cols: 100, // 列数,影响最小宽度
    convertEol: true, //启用时,光标将设置为下一行的开头
    // scrollback: 50, //终端中的滚动条回滚量
    disableStdin: false, //是否应禁用输入。
    cursorStyle: "underline", //光标样式
    cursorBlink: true, //光标闪烁
    theme: {
      foreground: "#F8F8F8",
      background: "#2D2E2C",
      cursor: "help", //设置光标
      lineHeight: 16,
    },
    fontFamily: '"Cascadia Code", Menlo, monospace',
  });
  // writeDefaultInfo();
  // 弹框自动输入
  term.value.open(terminal.value); //挂载dom窗口
  term.value.loadAddon(fitAddon); //自适应尺寸
  term.value.focus();
  termData(); //Terminal 事件挂载
};

const autoWriteInfo = () => {
  let sendData = "\n" + "cd " + props.currentPathRes + "\n";
  // term.value.write(`\x1b[37m${sendData}\x1b[m`);
  // term.value.write("\r\n");
  if (isWsOpen()) {
    terminalSocket.value.send(sendData);
  }
};

const writeDefaultInfo = () => {
  let defaultInfo = [
    "┌\x1b[1m terminals \x1b[0m─────────────────────────────────────────────────────────────────┐ ",
    "│                                                                            │ ",
    "│  \x1b[1;34m 欢迎使用XS  SSH   \x1b[0m                                                       │ ",
    "│                                                                            │ ",
    "└────────────────────────────────────────────────────────────────────────────┘ ",
  ];
  term.value.write(defaultInfo.join("\n\r"));
  term.value.write("\r\n");
  // writeOfColor('我是加粗斜体红色的字呀', '1;3;', '31m')
};

const writeOfColor = (txt, fontCss = "", bgColor = "") => {
  // 在Linux脚本中以 \x1B[ 开始,中间前部分是样式+内容,以 \x1B[0m 结尾
  // 示例 \x1B[1;3;31m 内容 \x1B[0m
  // fontCss
  // 0;-4;字体样式(0;正常 1;加粗 2;变细 3;斜体 4;下划线)
  // bgColor
  // 30m-37m字体颜色(30m:黑色 31m:红色 32m:绿色 33m:棕色字 34m:蓝色 35m:洋红色/紫色 36m:蓝绿色/浅蓝色 37m:白色)
  // 40m-47m背景颜色(40m:黑色 41m:红色 42m:绿色 43m:棕色字 44m:蓝色 45m:洋红色/紫色 46m:蓝绿色/浅蓝色 47m:白色)
  // console.log("writeOfColor", term)
  term.value.write(`\x1b[37m${fontCss}${bgColor}${txt}\x1b[m`);
  // term.value.write(`\x1B[${fontCss}${bgColor}${txt}\x1B[0m`);
};

// 终端输入触发事件
const termData = () => {
  fitAddon.fit();
  // 输入与粘贴的情况,onData不能重复绑定,不然会发送多次
  term.value.onData((data) => {
    // console.log(data, "传入服务器");
    if (isWsOpen()) {
      terminalSocket.value.send(data);
    }
  });
  // 终端尺寸变化触发
  term.value.onResize(() => {
    resizeRemoteTerminal();
  });
};

//尺寸同步 发送给后端,调整后端终端大小,和前端保持一致,不然前端只是范围变大了,命令还是会换行
const resizeRemoteTerminal = () => {
  const { cols, rows } = term.value;
  if (isWsOpen()) {
    terminalSocket.value.send(
      JSON.stringify({
        t: "resize",
        width: rows,
        height: cols,
      })
    );
  }
};

// 是否连接中0 1 2 3 状态
const isWsOpen = () => {
  // console.log(terminalSocket.value, "terminalSocket.value");
  const readyState = terminalSocket.value && terminalSocket.value.readyState;
  return 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();
  let sendData = JSON.stringify({
    t: "close",
  });
  if (isWsOpen()) {
    terminalSocket.value.send(sendData);
    terminalSocket.value && terminalSocket.value.close();
  }
});

// 暴露方法
defineExpose({ getTableData });
</script>
<style lang="scss" scoped>
.xterm {
  position: relative;
  width: 100%;
  background: rgb(45, 46, 44);
}

.xtermR {
  position: relative;
  width: 100%;
  background: #fff;
  padding: 10px;
  position: relative;
  // overflow: hidden;
  .listBoxA {
    overflow-y: auto;
    height: calc(100vh - 450px);
  }
  .listBoxB {
    overflow-y: auto;
    height: calc(100vh - 300px);
  }
}

.heightA {
  height: calc(100vh - 400px);
}
.heightB {
  height: calc(100vh - 235px);
}
.heightBR {
  height: calc(100vh - 155px);
}

.usualName {
  width: calc(100% - 80px);
  display: inline-block;
  cursor: pointer;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}

.btns {
  width: 80px;
}

.textarea {
  overflow: hidden;
  position: relative;
  height: 80px;
  background: #ffffff;

  textarea {
    width: 100%;
    height: 90px;
    border: 0 none;
    outline: none;
    resize: none;
    font-size: 15px;
    overflow-y: auto;
    padding: 5px;
    background: #ffffff;
  }
  .bottomOperate {
    position: absolute;
    right: 10px;
    bottom: 10px;
  }
}
.recordName {
  font-size: 13px;
  color: #303133;
  cursor: pointer;
  margin-bottom: 10px;
  width: 100%;
}
.flexCenter {
  font-size: 14px;
  padding-top: 150px;
}
</style>

 此页面兼容了弹框和非弹框页面,做了两种样式处理判断

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/946192.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

[2025] 如何在 Windows 计算机上轻松越狱 IOS 设备

笔记 1. 首次启动越狱工具时&#xff0c;会提示您安装驱动程序。单击“是”确认安装&#xff0c;然后再次运行越狱工具。 2. 对于Apple 6s-7P和iPad系列&#xff08;iOS14.4及以上&#xff09;&#xff0c;您应该点击“Optinos”并勾选“允许未经测试的iOS/iPadOS/tvOS版本”&…

网页排名:PageRank 算法的前世今生

PageRank算法全解析&#xff1a;从理论到实践 引言 PageRank 是由拉里佩奇&#xff08;Larry Page&#xff09;和谢尔盖布林&#xff08;Sergey Brin&#xff09;在1996年发明的一种链接分析算法&#xff0c;最初用于Google搜索引擎来评估网页的重要性。该算法通过模拟随机浏览…

嵌入式开发之使用 FileZilla 在 Windows 和 Ubuntu 之间传文件

01-FileZilla简介 FileZilla 是一个常用的文件传输工具&#xff0c;它支持多种文件传输协议&#xff0c;包括以下主要协议&#xff1a; FTP (File Transfer Protocol) 这是 FileZilla 最基本支持的协议。FTP 是一种明文传输协议&#xff0c;不加密数据&#xff08;包括用户名和…

Jmeter的安装与使用

1.下载压缩包&#xff0c;并解压到本地 2.在bin目录下找到jmeter.bat双击打开图形化界面 3.在测试计划上点击右键添加一个线程组 4.可以自定义线程数&#xff0c;Ramp_Up表示在该时间内将一组线程将运行完毕&#xff0c;循环次数可自定义 5.在线程组点击右键添加配置元件…

pycharm pytorch tensor张量可视化,view as array

Evaluate Expression 调试过程中&#xff0c;需要查看比如attn_weight 张量tensor的值。 方法一&#xff1a;attn_weight.detach().numpy(),view as array 方法二&#xff1a;attn_weight.cpu().numpy(),view as array

XIAO ESP32 S3网络摄像头——2视频获取

本文主要是使用XIAO Esp32 S3制作网络摄像头的第2步,获取摄像头图像。 1、效果如下: 2、所需硬件 3、代码实现 3.1硬件代码: #include "WiFi.h" #include "WiFiClient.h" #include "esp_camera.h" #include "camera_pins.h"// 设…

数据仓库中的指标体系模型介绍

数据仓库中的指标体系介绍 文章目录 数据仓库中的指标体系介绍前言什么是指标体系指标体系设计有哪些模型?1. 指标分层模型2. 维度模型3. 指标树模型4. KPI&#xff08;关键绩效指标&#xff09;模型5. 主题域模型6.平衡计分卡&#xff08;BSC&#xff09;模型7.数据指标框架模…

K3知识点

提示&#xff1a;文章 文章目录 前言一、顺序队列和链式队列题目 顺序队列和链式队列的定义和特性实际应用场景顺序表题目 链式队列 二、AVL树三、红黑树四、二叉排序树五、树的概念题目1左子树右子树前序遍历、中序遍历&#xff0c;后序遍历先根遍历、中根遍历左孩子右孩子题目…

jQuery学习笔记1

// jQuery的入口函数 // 1.等着DOM结构渲染完毕即可执行内部代码&#xff0c;不必等到所以外部资源加载完毕&#xff0c;jQuery帮我们完成了封装 // 相当于原生js中的DOMContentLoaded <script src"./jquery.min.js"></script> <style>div {width…

HTML——41有序列表

<!DOCTYPE html> <html><head><meta charset"UTF-8"><title>有序列表</title></head><body><!--有序列表&#xff1a;--><!--1.列表中各个元素在逻辑上有先后顺序&#xff0c;但不存在一定的级别关系-->…

典型常见的基于知识蒸馏的目标检测方法总结二

来源&#xff1a;https://github.com/LutingWang/awesome-knowledge-distillation-for-object-detection收录的方法 NeurIPS 2017&#xff1a;Learning Efficient Object Detection Models with Knowledge Distillation CVPR 2017&#xff1a;Mimicking Very Efficient Networ…

计算机网络-L2TP VPN基础实验配置

一、概述 上次大概了解了L2TP的基本原理和使用场景&#xff0c;今天来模拟一个小实验&#xff0c;使用Ensp的网卡桥接到本地电脑试下L2TP拨号&#xff0c;今天主要使用标准的L2TP&#xff0c;其实在这个基础上可以加上IPSec进行加密&#xff0c;提高安全性。 网络拓扑 拓扑说明…

基于BiTCN双向时间卷积网络实现电力负荷多元时序预测(PyTorch版)

Bidirectional Temporal Convolutional Network \begin{aligned} &\text{\Large \color{#CDA59E}Bidirectional Temporal Convolutional Network}\\ \end{aligned} ​Bidirectional Temporal Convolutional Network​ Bidirectional Temporal Convolutional Network (BiTC…

Linux C/C++编程-网络程序架构与套接字类型

【图书推荐】《Linux C与C一线开发实践&#xff08;第2版&#xff09;》_linux c与c一线开发实践pdf-CSDN博客《Linux C与C一线开发实践&#xff08;第2版&#xff09;&#xff08;Linux技术丛书&#xff09;》(朱文伟&#xff0c;李建英)【摘要 书评 试读】- 京东图书 (jd.com…

北京某新能源汽车生产及办公网络综合监控项目

北京某新能源汽车是某世界500强汽车集团旗下的新能源公司&#xff0c;也是国内首个获得新能源汽车生产资质、首家进行混合所有制改造、首批践行国有控股企业员工持股的新能源汽车企业&#xff0c;其主营业务包括纯电动乘用车研发设计、生产制造与销售服务。 项目现状 在企业全…

【LeetCode】2506、统计相似字符串对的数目

【LeetCode】2506、统计相似字符串对的数目 文章目录 一、哈希表位运算1.1 哈希表位运算 二、多语言解法 一、哈希表位运算 1.1 哈希表位运算 每个字符串, 可用一个 int 表示. (每个字符 是 int 的一个位) 哈希表记录各 字符组合 出现的次数 步骤: 遇到一个字符串, 得到 ma…

gitlab 还原合并请求

事情是这样的&#xff1a; 菜鸡从 test 分支切了个名为 pref-art 的分支出来&#xff0c;发布后一机灵&#xff0c;发现错了&#xff0c;于是在本地用 git branch -d pref-art 将该分支删掉了。之后切到了 prod 分支&#xff0c;再切出了一个相同名称的 pref-art 分支出来&…

Uncaught ReferenceError: __VUE_HMR_RUNTIME__ is not defined

Syntax Error: Error: vitejs/plugin-vue requires vue (>3.2.13) or vue/compiler-sfc to be present in the dependency tree. 第一步 npm install vue/compiler-sfc npm run dev 运行成功&#xff0c;本地打开页面是空白&#xff0c;控制台报错 重新下载了vue-loa…

LeetCode--排序算法(堆排序、归并排序、快速排序)

排序算法 归并排序算法思路代码时间复杂度 堆排序什么是堆&#xff1f;如何维护堆&#xff1f;如何建堆&#xff1f;堆排序时间复杂度 快速排序算法思想代码时间复杂度 归并排序 算法思路 归并排序算法有两个基本的操作&#xff0c;一个是分&#xff0c;也就是把原数组划分成…

vim里搜索关键字

vim是linux文本编辑器的命令&#xff0c;再vi的基础上做了功能增强 使用方法如下 1. / 关键字, 回车即可, 按n键查找关键字下一个位置 2.? 关键字, 回车即可, 按n键查找关键字下一个位置 3.示例