ANSI Escape Sequence 下落的方块

ANSI Escape Sequence 下落的方块

1. ANSI Escape 的用途

无意中发现 B站有人讲解, 完全基于终端实现俄罗斯方块。 基本想法是借助于 ANSI Escape Sequence 实现方方块的绘制、 下落动态效果等。对于只了解 ansi escape sequence 用于 log 的颜色打印的人来说, 这无疑是拓宽了认识。

在这里插入图片描述

这一篇简单的列一下 ansi escape sequence 中的稀奇古怪的数字的含义, 并最终给出一个绿色方块下落的动态效果和对应的代码。 基于本篇给出的表格和代码, 可以把它拓展为自由落体的游戏效果, 也可以跟着 B 站视频更容易的写出终端里的俄罗斯方块。

同时也注意到, ansi escape sequence 有它的局限性, 无法绘制比较大的圆形, 使用 ansi escape sequence 会限制界面显示、 游戏开发的上限。

2. ANSI Escape Sequence 有什么用?

终端会把 ANSI Escape 序列的字符解释为命令,而不是原本的内容。这些命令在终端上控制如下内容:

  • 鼠标位置
  • 颜色
  • 字体样式
  • 其他选项

使用 ANSI Escape Sequence 做 log 打印的例子比较多, 但其实还可以那它用作绘图显示: 把终端当成是 256 色的图像, 在终端显示图像内容。

3. 形式

3.1 格式概况

  • 绝大多数有用的序列: ESC[ 开头
  • 序列的重置: ESC[0m, 意思是各种设置的属性都撤销掉,恢复为没有设置时的状态

一些“黑话”:

  • CSI (Control Sequence Introducer): ESC[ 的别名, ASCII escape 数值是27, 实际使用时 ESC 换成 \x1b(16进制), \033(8进制) 或 \e
  • SGR (Select Graphic Rendition): CSI n m 的别名,用于设定字符的颜色和风格。其中:
    • CSI 要换成 \x1b[, \033[\e[
    • n 要换成具体的数字,在 0~107 之间

根据 wikipedia 得到的解释:

  • 起始: ESC[ (这个组合又叫做 CSI, Control Sequence Introducer)
  • parameter bytes: 任意数量的 0x30-0x3F 范围的字符, 也就是 0-9:;<=>?
  • intermediate bytes: 任意数量的 0x20-0x2F 范围的字符,也就是 !"#$%&'()*+,-./
  • final bytes: 任意数量的 0x40-0x7E 范围的字符,也就是 “@A–Z[]^_`a–z{|}~”
  • private bytes: 包含 <=>? 或 0x70-0x7E 范围(p-z{|}~) 的字符, 各厂商自行定义和使用的
  • 设定多个属性: ; 分隔的单个属性
  • 重置: ESC[0m

而网上其他资料, 以及实际验证, 发现维基百科有遗漏内容, ESC[ (CSI) 之后可以紧跟着 0~0x2F 范围的数字, 例如 n=1 对应到 “字体加粗” 的属性。

3.2 格式的具体情况

n名字含义、作用
0Reset or normal重置所有属性
1Bold or increased intensity字体加粗
2Faint, decresed intensity, or dim字体变暗
3Italic斜体。据说没有被广泛使用
4Underline下划线. 算是扩展, 在 Kitty, VTE, mintty, iTERM2, Konsole 里有效
5Slow blink设定光标闪烁时间在每分钟内小于150次(暂时不会用)
6Rapid blink光标闪烁加速,每分钟内超过150次; 没有被广泛支持
7Reverse video or invert对调背景和前景的颜色
8Conceal or hide没有被广泛的支持,iTerm2 上没有效果
9Crossed-out, or strike让字符带有删除线
10Primary(default) font默认字体
11~19Alternative font选择编号为 n-10 的字体
20Fraktur(Gothic)很少使用。iTerm2 上没有效果
21Doubly underlined; or: not bold双下划线、或者不要加粗
22Normal intensity既不加粗、也不变暗
23Neither italic, nor blackletter既不斜体, 也不黑色字母
24Not underlined不要有单个下划线, 也不要有双下划线
25Not blinking不要闪烁光标
26Proportional spacing终端上没有在使用
27Not reservediTerm2 上没有效果
28Reveal不要"隐瞒"
29Not crossed out去掉“删除线"
30-37Set foreground color设置前景颜色
30Black 黑色前景
31Red 红色前景
32Green 绿色前景
33Yellow 黄色前景
34Blue 蓝色前景
35Magenta 紫色前景
36Cyan 靛蓝色前景
37White 白色前景
38Set foreground color设置前景颜色, 接下来的参数是 5;n2;r;g;b
39Default foreground color默认前景颜色
40-47设置背景颜色
40Black 黑色背景
41Red 红色背景
42Green 绿色背景
43Yellow 黄色背景
44Blue 蓝色背景
45Magenta 紫色背景
46Cyan 靛蓝色背景
47White 白色背景
48Set background color设置前景颜色, 接下来的参数是 5;n2;r;g;b
49默认背景颜色
50Disable proportional spacing禁用等比例空格
51Framedmintty 中被实现为 emoji 选择器(?)
52Encircled同上
53Overlinked没效果
54Neither framed nor encircled
55Not overlined
58Set underline color设置下划线颜色。不是标准规定的。Kitty, VTE, iTerm2里有实现;下一个参数需要是 5;n2;r;g;b 形式
59Default underline color默认下划线颜色. 非标准。在 Kitty, VTE, iTerm2里有实现
60~65通常没有实现
73-74Superscript, Subscript上标和下标。只在 mintty 里有实现
75-76Neither superscript nor subscript取消上标和下标
90-97Set bright foreground color设置前景颜色亮度。非标准. iTerm2里有效
90Bright Black亮黑色前景色
91Bright Red亮红色前景色
92Bright Green亮绿色前景色
93Bright Yellow亮黄色前景色
94Bright Blue亮蓝色前景色
95Bright Magenta亮紫色前景色
96Bright Cyan亮靛蓝色前景色
97Bright White亮白色前景色
100-107Set bright background color背景颜色亮度. iTerm2里有效
100Bright Black亮黑色背景色
101Bright Red亮红色背景色
102Bright Green亮绿色背景色
103Bright Yellow亮黄色背景色
104Bright Blue亮蓝色背景色
105Bright Magenta亮紫色背景色
106Bright Cyan亮靛蓝色背景色
107Bright White亮白色背景色

其中 n 为 38 是设置前景颜色 ESC[38;5;{ID}m, n 为 48 是设置背景颜色 ESC[48;5;{ID}m, ID 是具体的颜色, 见下图:

在这里插入图片描述

3.3 常见私有模式 (Common Private Modes)

https://gist.github.com/fnky/458719343aabd01cfb17a3a4f7296797

ESC 代码序列描述
ESC[?25l]隐藏光标
ESC[?25h]显示光标
ESC[?47l]恢复屏幕
ESC[?47h]保存屏幕
ESC[?1049h]启用可选buffer
ESC[?1049l]禁用可选bufer

3.4 控制光标

https://gist.github.com/fnky/458719343aabd01cfb17a3a4f7296797#cursor-controls

ESC 代码序列描述
ESC[H光标移动到 (0, 0) 位置
ESC[#A光标向上移动 #
ESC[#B光标向下移动 #

3.5 擦除功能 (Erase Functions)

https://gist.github.com/fnky/458719343aabd01cfb17a3a4f7296797#erase-functions

ESC CodeDescription
ESC[J清除光标位置到屏幕结束位置
ESC[0JESC[J
ESC[1J清除光标位置到屏幕开始
ESC[2J清除整个屏幕
ESC[3J清除保存的行
ESC[K清除当前光标位置到当前行末尾
ESC[0KESC[K
ESC[1K删除当前光标位置到当前行首
ESC[2K删除整行

4. 简单例子

在这里插入图片描述

#include <stdio.h>
#include <iostream>
#include <string>
#include <vector>
#include <numeric>int main()
{std::vector<int> codes = {1, 2, 3, 4, 7, 8, 9};std::generate_n(std::back_inserter(codes), 8, [n = 30]() mutable { return n++; });std::generate_n(std::back_inserter(codes), 8, [n = 40]() mutable { return n++; });std::generate_n(std::back_inserter(codes), 8, [n = 90]() mutable { return n++; });std::generate_n(std::back_inserter(codes), 8, [n = 100]() mutable { return n++; });for (const auto code : codes){printf("\e[%dmHello\e[mworld  (n=%d)\n", code, code);}printf("\e[1;34mHello\e[0mworld  (n=1;34)\n");printf("\e[38;5;2mHello\e[0mworld  (n=38;5;2)\n");printf("\e[48;5;2mHello\e[0mworld  (n=48;5;2)\n");return 0;
}

5. 复杂例子 - 方块下落

绘制最小的绿色矩形: 打印“空格” 字符, 并且让空格字符的前景颜色红色的:

printf("\e[42m \e[0m\n"); 

绘制较大的红色矩形: 每一行打印多个空格, 连续打印多行; 每一行打印时使用转义字符。

printf("\e[42m    \e[0m\n"); 

绘制会下落的红色矩形框: 先绘制一个,长度持续增加的。

void draw_box()
{for (int i = 0; i < 10; i++){printf("\e[42m    \e[0m\n");std::this_thread::sleep_for(std::chrono::milliseconds(100));}
}

通过 ANSI Escape Sequence, 修改光标位置, 然后再绘制矩形:

void draw_box2()
{printf("\e[H");     // 光标移动到 (0,0) 位置printf("\e[42m    \e[0m"); // 绘制绿色背景的空格printf("\e[1B");  // 光标往下一行。 注意此时 column 方向上, 光标不是在0位置printf("\e[43m    \e[0m"); // 绘制黄色背景的空格
}

让每一行的绘制, 都从第 6 列开始绘制, 并且每次绘制后, 等待 100 毫秒:

void draw_box6()
{printf("\e[?25l"); // 隐藏光标, 避免光标导致的白色小方框for (int i = 0; i < 10; i++){printf("\e[2J");    // 清空整个屏幕printf("\x1b[%d;%dH\e[0m", i, 6); // 光标一定到第i 行,第 6 列printf("\e[42m    \e[0m"); // 绘制绿色矩形: 也就是绘制绿色背景的空格fflush(stdout); // 确保绘制到控制台std::this_thread::sleep_for(std::chrono::milliseconds(500)); // 暂停 500 毫秒,营造下落的视觉效果}printf("\e[?25h"); // 恢复光标的可见性
}

在这里插入图片描述

6. ANSI Escape 的局限

无法绘制圆形。 因为终端绘制的最小单位, 是单个字符,每个字符通常是竖条而不是正方形, 并且竖条比较大, 大于通常看到的图像像素。 这就导致, 稍微复杂的图形无法绘制, 需要选择其他的方案:

  • 使用 opencv 的 Mat 绘制, 用 imshow 显示
  • 使用 SFML / SDL / Dear imgui / Qt 绘制和显示

7. References

  • 手把手教你写俄罗斯方块:2-如何在终端上绘图
  • ANSI Escape Codes
  • C语言实现 log 库

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

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

相关文章

webgis后端安卓系统部署攻略

目录 前言 一、将后端项目编译ARM64 二、安卓手机安装termux 1.更换为国内源 2.安装ssh远程访问 3.安装文件远程访问 三、安装postgis数据库 1.安装数据库 2.数据库配置 3.数据导入 四、后端项目部署 五、自启动设置 总结 前言 因为之前一直做的H5APP开发&#xf…

春节特辑 | 催婚大作战与奇妙相亲记

点击文末“阅读原文”即可参与节目互动 剪辑、音频 / 卷圈 运营 / SandLiu 卷圈 监制 / 姝琦 封面 / 姝琦Midjourney 产品统筹 / bobo 场地支持 / 声湃轩天津录音间 催催催&#xff0c;一到春节&#xff0c;七大姑八大姨都来纷纷关心“怎么样了&#xff1f;还单着呢&…

C++类和对象(7)

目录 3. 友元 3.1 友元函数 3.2 友元类 4. 内部类 5.匿名对象 6.拷贝对象时的一些编译器优化 7. 再次理解类和对象 3. 友元 友元提供了一种突破封装的方式&#xff0c;有时提供了便利。但是友元会增加耦合度&#xff0c;破坏了封装&#xff0c;所以 友元不宜多用。 友元…

SAP-PS-001-006问题预算占用与订单实际金额不一致

前言 PS模块最复杂的业务场景主要就是ETO&#xff08;Engineering-To-Order&#xff09;&#xff0c;也就是边设计边生产边采购的三边业务。 意味着从前端设计开始的成本就已经要进行收集&#xff0c;其次对于大型非标设备的生产发货只是一个环节&#xff0c;发货后还会涉及到现…

解决hive表新增的字段查询为空null问题

Hive分区表新增字段&#xff0c;查询时数据为NULL的解决方案 由于业务拓展&#xff0c;需要往hive分区表新增新的字段&#xff0c;hive版本为2点多。 于是利用 alter table table_name add columns (col_name string )新增字段&#xff0c;然后向已存在分区中插入数据&#x…

单片机基础入门:简单介绍51单片机的工作原理

在电子技术领域&#xff0c;单片机是实现智能化控制不可或缺的关键元件。它们集成了许多功能于一身&#xff0c;成为了各种电子系统的心脏。为了更好地理解单片机如何工作&#xff0c;本文将重点介绍51单片机的基本组成和工作原理。 51单片机是一种广泛使用的微控制器&#xf…

Leetcode2786. 访问数组中的位置使分数最大

Every day a Leetcode 题目来源&#xff1a;2786. 访问数组中的位置使分数最大 解法1&#xff1a;动态规划 状态数组&#xff1a; dp[i][0]: 访问下标范围 [0, i] 中的元素且最后访问的元素是偶数时的最大得分&#xff1b;dp[i][1]: 访问下标范围 [0, i] 中的元素且最后访问…

SpringBoot框架入门指南

文章目录 SpringBoot的特点Spring&#xff0c;SpringBoot的区别SpringBoot常用注解标签SpringBoot概述SpringBoot简单Demo搭建读取配置文件的内容 SpringBoot自动配置Condition自定义beanSpringBoot常用注解原理EnableAutoConfiguration SpringBoot监听机制SpringBoot启动流程分…

【05】C++ 内存管理

文章目录 &#x1f308; Ⅰ C 内存分布&#x1f308; Ⅱ C 内存管理方式1. new 和 delete 操作内置类型2. new 和 delete 操作自定义类型 &#x1f308; Ⅲ operator new 和 operator delete&#x1f308; Ⅳ new 和 delete 的实现原理1. 内置数据类型2. 自定义数据类型 &#…

vscode wsl远程连接 权限问题

问题描述&#xff1a;执行命令时遇到Operation not permitted 和 Permission denied问题&#xff0c;是有关ip地址和创建文件的权限问题&#xff0c;参考网络上更改wsl.conf文件等方法均无法解决&#xff0c;只能加sudo来解决

【北邮鲁鹏老师计算机视觉课程笔记】01 introduction

1 生活中的计算机视觉 生活中的各种计算机视觉识别系统已经广泛地应用起来了。 2 计算机视觉与其他学科的关系 认知科学和神经科学是研究人类视觉系统的&#xff0c;如果能把人类视觉系统学习得更好&#xff0c;可以迁移到计算机视觉。是计算机视觉的理论基础。 算法、系统、框…

【web前端开发】HTML及CSS简单页面布局练习

案例一 网页课程 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthdevice-wi…

2 月 9 日算法练习- 数据结构 - 除夕快乐♪٩(´ω`)و♪

翻转括号序列 暴力过20%数据 思路&#xff1a;括号合法序列问题可以利用前缀和&#xff0c;将"(“看成 1&#xff0c;”)"看成 0&#xff0c;规律是到某个位置为止的前缀和>0并且到最后前缀和0。 #include<bits/stdc.h> using namespace std; const int N…

代码随想录 Leetcode55. 跳跃游戏

题目&#xff1a; 代码(首刷自解 2024年2月9日&#xff09;&#xff1a; class Solution { public:bool canJump(vector<int>& nums) {int noz 0;for (int i nums.size() - 2; i > 0; --i) {if (nums[i] 0) {noz;continue;} else {if (nums[i] > noz) noz …

跟着cherno手搓游戏引擎【22】CameraController、Resize

前置&#xff1a; YOTO.h: #pragma once//用于YOTO APP#include "YOTO/Application.h" #include"YOTO/Layer.h" #include "YOTO/Log.h"#include"YOTO/Core/Timestep.h"#include"YOTO/Input.h" #include"YOTO/KeyCod…

随机MM引流源码PHP开源版

引流源码最新随机MM开源版PHP源码&#xff0c;非常简洁好看的单页全解代码没任何加密 直接上传即可用无需数据库支持主机空间

【动态规划】【C++算法】LeetCoce996正方形数组的数目

作者推荐 【动态规划】【前缀和】【C算法】LCP 57. 打地鼠 本文涉及知识点 动态规划汇总 LeetCoce996正方形数组的数目 给定一个非负整数数组 A&#xff0c;如果该数组每对相邻元素之和是一个完全平方数&#xff0c;则称这一数组为正方形数组。 返回 A 的正方形排列的数目…

PbootCMS采集插件使用教程

这篇Pboot采集教程教你使用PbootCMS采集插件&#xff0c;自动批量采集网页文章数据&#xff0c;并发布到PbootCMS系统&#xff0c;快速丰富网站的内容。 1. 下载并安装PbootCMS采集插件 1-1&#xff09;PbootCMS采集插件免费下载&#xff1a;Pboot采集插件-PbootCMS发布模块下…

物品冷启动01_优化目标评价(包括基尼系数)

文章目录 物品冷启动冷启动的类型“新”按常规推送链路的角度按产品生态角度 物品冷启动的目标和评价指标作者侧用户侧 冷启动的衡量 物品冷启动 冷启动的类型 冷启动的内容种类包括很多方面&#xff0c;本文只介绍UGC的冷启动。 所谓UGC&#xff0c;就是User Generate Conte…

ARP欺骗攻击利用之抓取https协议的用户名与密码

1.首先安装sslstrip 命令执行&#xff1a;apt-get install sslstrip 2.启动arp欺骗 arpspoof -i ech0 -t 192.168.159.148 192.168.159.2 arpspoof -i ech0(网卡) -t 目标机ip 本地局域网关 3.命令行输入: vim /etc/ettercap/etter.conf进入配置文件 找到下红框的内容&a…