MIG*_*-23 5 c terminal termcap
我想知道如何在我的程序中获取光标位置 (x, y),而不在屏幕上写任何东西,也不会一直跟踪它。
我找到了一种使用此函数获取其位置的方法(我在这里不检查读取、写入等的返回来编写有关此主题的较小代码,但我在我的程序中执行此操作):
void get_cursor_position(int *col, int *rows)
{
int a = 0;
int i = 0;
char buf[4];
write(1, "\033[6n", 4); // string asking for the cursor position
read(1, buf, 4);
while (buf[i])
{
if (buf[i] >= 48 && buf[i] <= 57)
{
if (a == 0)
*rows = atoi(&buf[i]) - 1;
else
*col = atoi(&buf[i]) - 1;
a++;
}
i++;
}
}
Run Code Online (Sandbox Code Playgroud)
这个函数给了我准确的光标位置(*rows = y,*col = x),但它写在屏幕上。
如何在不在屏幕上写任何东西的情况下获得光标位置?
(如果光标位于打印的字符之一上,它将覆盖它。)
应该在发送转义序列之前和之后切换 echo 吗?
这是一个学校项目,所以我只能使用termcap,我不能使用ncurses函数,唯一允许的函数是tputs、tgoto、tgetstr、tgetnum、tgetflag。
有几个问题:
\n\n规范模式是缓冲的 (see below)
这是在标准输出read的文件描述符上完成的 (that may happen to work — sometimes — but don\'t count on it)
这read does not read enough characters to get a typical response
响应将有两个十进制整数,用分号分隔;
响应将具有最终特征(如果read实际要求足够的字符,这将成为一个问题......)
进一步阅读:
\n\n\n\n\n在规范模式输入处理中,终端输入以行为单位进行处理。行由换行符 (
\nNL)、文件结束符 (EOF) 或行结束符 (EOL) 分隔。有关EOF和的更多信息,请参阅特殊字符EOL。这意味着在键入整行或收到信号之前,read请求不会返回。另外,无论 read() 调用请求多少字节,最多都会返回一行。然而,没有必要一次读完整行;可以在 read() 中请求任意数量的字节,甚至是一个字节,而不会丢失信息。
\n CSI Ps n 设备状态报告 (DSR)。\n Ps = 5 -> 状态报告。\n 结果(“OK”)为 CSI 0 n\n Ps = 6 -> 报告光标位置 (CPR) [row; \n 结果是 CSI r ;转R\n\n\n
也就是说,您的程序应该准备好读取Escape[后跟两个十进制整数(其长度没有固定限制)以及另外两个字符;和R。
顺便说一句,termcap本身对您的解决方案没有什么帮助。虽然 ncurses 在终端数据库中定义了一些相关功能:
\n\n# u9 terminal enquire string (equiv. to ANSI/ECMA-48 DA)\n# u8 terminal answerback description\n# u7 cursor position request (equiv. to VT100/ANSI/ECMA-48 DSR 6)\n# u6 cursor position report (equiv. to ANSI/ECMA-48 CPR)\nRun Code Online (Sandbox Code Playgroud)\n\n很少有程序使用这些,并且在任何情况下您都会发现很难在 termcap 应用程序中使用光标位置报告。
\n