xterm使用

news2024/11/29 2:41:13

xterm使用

  • 前言
  • 1. xterm介绍
  • 2. xterm使用
    • 2.1 xterm简单示例
    • 2.2 xterm监听输入并在终端中实时显示
      • 方式1:onKey监听
      • 方式2:onData监听
      • onData和onKey什么区别
    • 2.3 xterm与vue整合
    • 2.3 xterm + vue + websocket
  • 附录
    • 配置说明

前言

vue与xterm整合记录

1. xterm介绍

xterm 是一个基于 Web 技术的终端模拟器,它可以在浏览器中创建交互式终端界面。xterm 提供了一种强大的方式来将终端功能集成到 Web 应用程序中,使用户能够在浏览器中与远程服务器或本地命令行进行交互

2. xterm使用

2.1 xterm简单示例

<!DOCTYPE html>
<html>
<head>
    <title>xterm-01</title>
    <!-- 引入 xterm 的 CSS 文件 -->
    <link rel="stylesheet" href="https://unpkg.com/xterm@4.18.0/css/xterm.css">
</head>
<body>
    <h1>xterm 简单示例</h1>
    <!-- 创建一个容器用于放置 xterm 组件 -->
    <div id="terminal"></div>

    <!-- 引入 xterm 的 JavaScript 文件 -->
    <script src="https://unpkg.com/xterm@4.18.0/lib/xterm.js"></script>
    <script>
        // 创建一个 xterm 组件并将其连接到容器
        const xterm = new Terminal();
        xterm.open(document.getElementById('terminal'));

        // 向终端写入消息
        xterm.writeln('欢迎使用 xterm!');
    </script>
</body>
</html>

效果:
在这里插入图片描述
默认宽度不会自适应,可以通过xterm-addon-fit.js实现宽度适配

https://unpkg.com/xterm-addon-fit@0.6.0/lib/xterm-addon-fit.js

<!DOCTYPE html>
<html>
<head>
    <title>xterm-01</title>
    <!-- 引入 xterm 的 CSS 文件 -->
    <link rel="stylesheet" href="https://unpkg.com/xterm@4.18.0/css/xterm.css">
</head>
<body>
<h1>xterm 宽度自适应示例</h1>
<!-- 创建一个容器用于放置 xterm 组件 -->
<div id="terminal"></div>

<!-- 引入 xterm 的 JavaScript 文件 -->
<script src="https://unpkg.com/xterm@4.18.0/lib/xterm.js"></script>
<!--xterm 横向自适应-->
<script  src="https://unpkg.com/xterm-addon-fit@0.6.0/lib/xterm-addon-fit.js"></script>
<script>
    // 创建一个 xterm 组件并将其连接到容器
    const xterm = new Terminal();
    const fitAddon = new FitAddon.FitAddon();
    xterm.loadAddon(fitAddon)

    xterm.open(document.getElementById('terminal'));
    //黑窗口适应实际div宽高
    fitAddon.fit()

    // 向终端写入消息
    xterm.writeln('欢迎使用 xterm!');
</script>
</body>
</html>

效果:
在这里插入图片描述

2.2 xterm监听输入并在终端中实时显示

方式1:onKey监听

<!DOCTYPE html>
<html>
<head>
    <title>xterm-02</title>
    <!-- 引入 xterm 的 CSS 文件 -->
    <link rel="stylesheet" href="https://unpkg.com/xterm@4.18.0/css/xterm.css">
</head>
<body>
<h1>xterm 监听输入并在终端中实时显示</h1>
<!-- 创建一个容器用于放置 xterm 组件 -->
<div id="terminal"></div>

<!-- 引入 xterm 的 JavaScript 文件 -->
<script src="https://unpkg.com/xterm@4.18.0/lib/xterm.js"></script>
<script>
    // 创建一个 xterm 组件并将其连接到容器
    const xterm = new Terminal();
    xterm.open(document.getElementById('terminal'));

    // 向终端写入消息
    xterm.writeln('欢迎使用 xterm!');

    // 监听输入并在终端中实时显示
    xterm.onKey((e) => {
        const printable = !e.domEvent.altKey && !e.domEvent.ctrlKey && !e.domEvent.metaKey;
        if (e.domEvent.keyCode === 13) {
            // 处理回车键,添加换行
            xterm.writeln('');
        } else if (e.domEvent.keyCode === 8) {
            // 处理退格键,删除最后一个字符
            xterm.write('\b \b');
        } else if (printable) {
            // 处理可打印字符
            xterm.write(e.key);
        }
    });
</script>
</body>
</html>

方式2:onData监听

<!DOCTYPE html>
<html>
<head>
    <title>xterm-02</title>
    <!-- 引入 xterm 的 CSS 文件 -->
    <link rel="stylesheet" href="https://unpkg.com/xterm@4.18.0/css/xterm.css">
</head>
<body>
<h1>xterm 监听输入并在终端中实时显示</h1>
<!-- 创建一个容器用于放置 xterm 组件 -->
<div id="terminal"></div>

<!-- 引入 xterm 的 JavaScript 文件 -->
<script src="https://unpkg.com/xterm@4.18.0/lib/xterm.js"></script>
<script>
    // 创建一个 xterm 组件并将其连接到容器
    const xterm = new Terminal();
    xterm.open(document.getElementById('terminal'));

    // 向终端写入消息
    xterm.writeln('欢迎使用 xterm!');

    // 监听输入并在终端中实时显示
    xterm.onData((data) => {
        const printable = data.match(/[\x20-\x7E]/); // 匹配可打印字符的正则表达式
        if (data === '\r' || data === '\x0D') {
            // 处理回车键,添加换行
            xterm.writeln('');
        } else if (data === '\x08' || data === '\x7F') {
            // 处理退格键,删除最后一个字符
            xterm.write('\b \b');
        } else if (printable) {
            // 处理可打印字符
            xterm.write(data);
        }
    });
</script>
</body>
</html>

效果:
在这里插入图片描述

onData和onKey什么区别

xterm.onData: 用于捕获用户在终端中输入的所有数据,包括不可见字符(例如回车、换行等)。它不区分特殊键和可打印字符,而是将用户输入的所有字符都视为数据

xterm.onData((data) => {
    // 处理用户输入的数据,包括特殊字符
    console.log(data);
});

xterm.onKey: 用于捕获用户在终端中按下的键盘事件,但它通常用于处理可打印字符(即可见字符,例如字母、数字、符号等)。它区分特殊键和可打印字符,以便你可以根据需要对它们进行处理。

xterm.onKey((e) => {
    if (e.domEvent.key === 'Enter') {
        // 处理回车键事件
    } else if (e.domEvent.key === 'Backspace') {
        // 处理退格键事件
    } else {
        // 处理可打印字符事件
        console.log(e.key);
    }
});

场景推荐

  • xterm.onData 获取所有数据(onKey就没法复制后粘贴数据)适合和后端websocket持续交互传输
  • xterm.onKey 可以做成快捷键,和xterm.onData 结合使用

在这里插入图片描述

<!DOCTYPE html>
<html>
<head>
    <title>xterm-02</title>
    <!-- 引入 xterm 的 CSS 文件 -->
    <link rel="stylesheet" href="https://unpkg.com/xterm@4.18.0/css/xterm.css">
</head>
<body>
<h1>xterm 监听输入并在终端中实时显示</h1>
<!-- 创建一个容器用于放置 xterm 组件 -->
<div id="terminal"></div>
<div>
    <h2>快捷键命令列表</h2>
    <ul id="shortcut-list"></ul>
    <form id="shortcut-form">
        <label for="shortcut">快捷键:</label>
        <input type="text" id="shortcut" name="shortcut">
        <label for="command">命令:</label>
        <input type="text" id="command" name="command">
        <button type="button" onclick="addShortcut()">添加</button>
    </form>
</div>

<!-- 引入 xterm 的 JavaScript 文件 -->
<script src="https://unpkg.com/xterm@4.18.0/lib/xterm.js"></script>
<script>
    // 创建一个 xterm 组件并将其连接到容器
    const xterm = new Terminal();
    xterm.open(document.getElementById('terminal'));

    // 向终端写入消息
    xterm.writeln('欢迎使用 xterm!');

    // 监听输入并在终端中实时显示
    xterm.onData((data) => {
        const printable = data.match(/[\x20-\x7E]/); // 匹配可打印字符的正则表达式
        if (data === '\r' || data === '\x0D') {
            // 处理回车键,添加换行
            xterm.writeln('');
        } else if (data === '\x08' || data === '\x7F') {
            // 处理退格键,删除最后一个字符
            xterm.write('\b \b');
        } else if (printable) {
            // 处理可打印字符
            xterm.write(data);
        }
    });

    // 快捷键与命令的映射
    const shortcutMap = {};

    // 在快捷键命令列表中展示
    const shortcutList = document.getElementById('shortcut-list');
    function updateShortcutList() {
        shortcutList.innerHTML = '';
        for (const shortcut in shortcutMap) {
            const command = shortcutMap[shortcut];
            const listItem = document.createElement('li');
            listItem.textContent = `${shortcut}: ${command}`;
            shortcutList.appendChild(listItem);
        }
    }

    // 添加快捷键与命令的关系
    function addShortcut() {
        const shortcutInput = document.getElementById('shortcut');
        const commandInput = document.getElementById('command');
        const shortcut = shortcutInput.value;
        const command = commandInput.value;
        if (shortcut && command) {
            shortcutMap[shortcut] = command;
            shortcutInput.value = '';
            commandInput.value = '';
            updateShortcutList();
        }
    }

    // 在 xterm 组件上监听键盘事件,在终端中输入相应的命令
    xterm.onKey((e) => {
        const shortcut = getShortcutKey(e.domEvent);
        if (shortcut && shortcutMap[shortcut]) {
            const command = shortcutMap[shortcut];
            xterm.write(command);
        }
    });

    // 获取组合键的字符串表示(Ctrl+A)
    function getShortcutKey(event) {
        const isMac = navigator.platform.toUpperCase().indexOf('MAC') >= 0;
        const ctrlKey = isMac ? event.metaKey : event.ctrlKey;
        if (ctrlKey) {
            return `Ctrl+${String.fromCharCode(event.keyCode)}`;
        }
        return null;
    }
</script>
</body>
</html>

2.3 xterm与vue整合

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>iView Tabs with xterm.js</title>

    <!-- 引入 Vue.js -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2.6.14/dist/vue.js"></script>

    <!-- 引入 iView 样式和脚本 -->
    <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/iview@2.9.1/dist/styles/iview.css">
    <script src="https://cdn.jsdelivr.net/npm/iview@2.9.1/dist/iview.min.js"></script>

    <!-- 引入样式 -->
    <link rel="stylesheet" href="https://unpkg.com/xterm@4.18.0/css/xterm.css">
    <script src="https://unpkg.com/xterm@4.18.0/lib/xterm.js"></script>
</head>
<body>
<div id="app">
    <tabs v-model="activeTabName" @on-click="handleTabClick">
        <tab-pane v-for="tab in tabs" :name="tab.name" :label="tab.label">
            <!-- 在这里渲染 xterm 组件 -->
            <div :ref="'xterm_' + tab.name"></div>
        </tab-pane>
    </tabs>
</div>

<script>
    new Vue({
        el: '#app',
        data() {
            return {
                activeTabName: '', // 当前激活的标签页名称
                tabs: [ // 标签页数据,可以根据需要进行动态添加
                    { name: 'tab1', label: 'Tab 1' },
                    { name: 'tab2', label: 'Tab 2' },
                    // 添加更多标签页...
                ],
                xtermInstances: {},
            };
        },
        methods: {
            handleTabClick(tabName) {
                // 处理标签页点击事件
                this.activeTabName = tabName;
                if (!this.xtermInstances[tabName]) {
                    this.initializeXterm(this.activeTabName);
                }

            },
            initializeXterm(tabName) {
                // 初始化 xterm 终端
                const xterm = new Terminal({
                    convertEol: true,//控制终端是否自动将 \n 转换为 \r\n。
                    cursorBlink: true,//指定光标是否闪烁
                    screenKeys: true
                });
                xterm.open(this.$refs['xterm_' + tabName][0]);

                // 写入数据
                xterm.write(tabName + " xterm created \r\n")

                // 监听键盘数据
                xterm.onData((data) => {
                    // console.log(tabName + ':'+ data)
                    // if (data =='\r') {
                    //     data = '\r\n'
                    // }

                    xterm.write(data)
                })

                this.xtermInstances[tabName] = xterm;
            }
        },
        mounted() {
            // 初始化第一个标签页的 xterm 终端
            this.activeTabName = this.tabs[0].name;
            this.initializeXterm(this.activeTabName);
        }
    });
</script>
</body>
</html>

效果
在这里插入图片描述

2.3 xterm + vue + websocket

附录

配置说明

配置选项作用默认值
colsrows指定终端的列数和行数,控制终端的大小。cols: 80rows: 24
cursorBlink指定光标是否闪烁。cursorBlink: false
cursorStyle指定光标的样式,可以是 "block""underline""bar"cursorStyle: "block"
bellStyle指定响铃(Bell)的样式,可以是 "none""sound""visual"bellStyle: "none"
theme定义终端的主题,可以通过设置不同的颜色和样式来自定义终端的外观。theme: { foreground: "#ffffff", background: "#000000" }
fontFamily指定终端的字体族(font family)。fontFamily: "Menlo, Monaco, Consolas, "Courier New", monospace"
fontSize指定终端的字体大小。fontSize: 12
fontWeight指定终端字体的粗细。fontWeight: "normal"
lineHeight指定行高。lineHeight: 1.0
fontSmoothing字体平滑设置,可以是 "antialiased""subpixel-antialiased""none"fontSmoothing: "antialiased"
convertEol控制终端是否自动将 \n 转换为 \r\nconvertEol: true
scrollback指定滚动缓冲区的行数,用于在终端中查看历史输出。scrollback: 1000
screenReaderMode启用屏幕阅读器模式,以支持可访问性。screenReaderMode: false
tabStopWidth指定制表符的宽度,以控制文本缩进。tabStopWidth: 8
rendererType指定绘图渲染器的类型,可以是 "canvas""dom"rendererType: "canvas"
rightClickSelectsWord控制右键单击是否选择整个单词。rightClickSelectsWord: false

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

jvm深入研究文档--java中的堆--详解!--jvm底层探索(1)

阿丹&#xff1a; JVM的内存分区包括以下几个部分&#xff1a; 堆区&#xff08;Heap&#xff09; - 这是JVM的主要部分&#xff0c;用于存储实例对象和大多数Java对象&#xff0c;如数组和用户定义的类。方法区&#xff08;Method Area&#xff09; - 这是线程私有的&#x…

内网穿透的应用-NAS私有云存储 - 搭建Nextcloud私有云盘并公网远程访问

文章目录 摘要1. 环境搭建2. 测试局域网访问3. 内网穿透3.1 ubuntu本地安装cpolar3.2 创建隧道3.3 测试公网访问 4 配置固定http公网地址4.1 保留一个二级子域名4.1 配置固定二级子域名4.3 测试访问公网固定二级子域名 摘要 Nextcloud,它是ownCloud的一个分支,是一个文件共享服…

利用fiddler正向代理前端请求到本地后端

前景&#xff1a;在实际开发测试环境中&#xff0c;&#xff08;前后端均已上线到测试服务器或前端以上线而后端还在开发中)。在测试过程中&#xff08;前端页面点击&#xff0c;功能测试&#xff09;发现了bug或异常点。 正常排查问题可能是先利用浏览器检查工具查看接口的返回…

Unity Bolt UGUI事件注册方式总结

Bolt插件提供了丰富的事件注册方式&#xff0c;开发者几乎不用编写任何代码就可以完成事件的注册&#xff0c;进行交互。下面是我使用UI事件注册的相关总结。 1、通过UI控件自身拖拽实现事件的注册。 Button的事件注册&#xff1a; 新建一个UnityEvent事件&#xff0c; Butt…

PASCAL VOC2012数据集详细介绍

PASCAL VOC2012数据集详细介绍 0、数据集介绍2、Pascal VOC数据集目标类别3、 数据集下载与目录结构4、目标检测任务5、语义分割任务6、实例分割任务7、类别索引与名称对应关系 0、数据集介绍 2、Pascal VOC数据集目标类别 在Pascal VOC数据集中主要包含20个目标类别&#xff…

uniapp开发h5 调用微信sdk 全网最全指南!!!! 血泪史!!!

目录 场景&#xff1a; 技术栈&#xff1a; 遇到的问题先抛出来&#xff1a; 1.通过后端同学获取调用微信sdk所需的签名过程中&#xff0c;遇到的跨域问题 2.使用微信sdk前提必须是微信容器&#xff0c;换句话说就是微信浏览器打开&#xff0c;才能使用微信sdk 3.如何在开…

方案:浅析利用AI智能识别与视频监控技术打造智慧水产养殖监管系统

一、方案背景 针对目前水产养殖集约、高产、高效、生态、安全的发展需求&#xff0c;基于智能传感、智慧物联网、人工智能、视频监控等技术打造智慧水产系统&#xff0c;成为当前行业的发展趋势。传统的人工观察水产养殖方式较为单一&#xff0c;难以及时发现人员非法入侵、偷…

Windows系统如何部署Wing FTP Server与公网远程访问【内网穿透】

Wing FTP Server安装配置结合内网穿透实现公网访问本地站点 文章目录 Wing FTP Server安装配置结合内网穿透实现公网访问本地站点前言1.Wing FTP Server下载安装2.Wing FTP Server配置部署3.安装cpolar内网穿透3.1 注册账号3.2 下载cpolar客户端3.3 登录cpolar web ui管理界面3…

ChatGLM 实现一个BERT

前言 本文包含大量源码和讲解,通过段落和横线分割了各个模块,同时网站配备了侧边栏,帮助大家在各个小节中快速跳转,希望大家阅读完能对BERT有深刻的了解。同时建议通过pycharm、vscode等工具对bert源码进行单步调试,调试到对应的模块再对比看本章节的讲解。 涉及到的jupyt…

【多目标跟踪】 TrackFormer 耗时三天 单句翻译!!!

【多目标跟踪】 TrackFormer 耗时三天 单句翻译&#xff01;&#xff01;&#xff01; TrackFormer: Multi-Object Tracking with Transformers Abstract The challenging task of multi-object tracking (MOT) re-quires simultaneous reasoning about track initiali…

纽禄美卡Neuromeka亮相美国FABTECH,展示用于焊接的3D视觉协作机器人

原创 | 文 BFT机器人 纽禄美卡Neuromeka公司在由美国精密成型协会、美国焊接协会、化工涂料协会等5大协会举办的美国金属加工及焊接展览会FABTECH上精彩亮相。这家总部位于韩国首尔的公司成立于2013年&#xff0c;是机器人解决方案领域的领先供应商&#xff0c;致力于提高各种…

通过 chatgpt 协助完成网站数据破解

Chatgpt 的出现极大地提升了程序员的工作效率&#xff0c;常见的使用场景包括代码自动生成、代码静态检查等&#xff0c;那么 chatgpt 能否用于某些网站的数据破解工作呢&#xff1f; 问题 某天线上服务开始报警&#xff0c;原来是某个视频网站无法获取到其 cdn 地址导致的下…

【AD】【规则设置】设置四层板

设置四层板 一般 4层板&#xff0c;都会把 地 和 VCC放在内层。1、使用快捷键D-K 进入层叠管理器&#xff0c;添加负片层添加完后&#xff0c;修改层名&#xff0c;方便辨识修改格式&#xff1a;属性层号 2、进入相应layer 设置网络设置GND层设置VCC层特点&#xff1a;在层内可…

【HackTheBox Topology】打靶记录

一、信息收集 1、nmap 扫描发现22 80 端口 2、访问80端口 找到两个域名 topology.htb latex.topology.htb 3、子域扫描发现如下两个域名 dev.topology.htb stats.topology.htb C:\root> gobuster vhost -u http://topology.htb --append-domain -w /usr/share/seclists…

【Android Framework系列】第15章 Fragment+ViewPager与Viewpager2相关原理

1 前言 上一章节【Android Framework系列】第14章 Fragment核心原理(AndroidX版本&#xff09;我们学习了Fragment的核心原理&#xff0c;本章节学习常用的FragmentViewPager以及FragmentViewPager2的相关使用和一些基本的源码分析。 2 FragmentViewPager 我们常用的两个Page…

【基本数据结构 四】线性数据结构:队列

学习了栈后,再来看看第四种线性表结构,也就是队列,队列和栈一样也是一种受限的线性表结构,和栈后进先出的操作方式不同的是,队列是FIFO的结构,也就是先进先出的操作方式。 队列的定义 队列这个概念非常好理解。可以把它想象成排队买票,先来的先买,后来的人只能站末尾…

iTOP-RK3568开发板Linux 修改kernel logo

本文档配套资料在网盘资料“iTOP-3568 开发板\02_【iTOP-RK3568 开发板】开发资料\10_Linux 系统开发配套资料\05_Linux 修改内核 logo 配套资料”路径下。 5.3.1 准备 logo 系统默认内核 logo&#xff0c;如下图所示&#xff1a; 如 果 想 要 替 换 这 个 logo, 首 先 要 制…

SAP服务器文件管理

SAP服务器文件管理 文件说明&#xff1a;对于SAP服务器的文件管理&#xff0c;系统给出3个事物码&#xff0c;分别是显示目录的AL11&#xff0c;下载文件的 CG3Y和上传文件的CG3Z。 AL11显示目录:以查找系统参数文件为例&#xff0c;在前台执行事物码AL11进入&#xff0c;如图…

【面试经典150 | 数组】多数元素

文章目录 写在前面Tag题目来源题目解读解题思路方法一&#xff1a;哈希表方法二&#xff1a;排序方法三&#xff1a;摩尔投票法 写在最后 写在前面 本专栏专注于分析与讲解【面试经典150】算法&#xff0c;两到三天更新一篇文章&#xff0c;欢迎催更…… 专栏内容以分析题目为主…

HelpLook全新升级!定制AI问答机器人,企业内容中心焕新

一直以来&#xff0c;企业都在努力解决内外部“企业知识管理”问题&#xff1a;从纸质手册发放&#xff0c;转线上电子文档传阅(pdf/ppt/word等)&#xff0c;再到整理客户常见问题(FAQ)和内部知识库(wiki)&#xff0c;但始终没有找到一套完整方案将“企业知识”很好地集中管理及…