gpt4 book ai didi

键入时更改 linux 终端中关键字的颜色

转载 作者:IT王子 更新时间:2023-10-29 01:25:31 25 4
gpt4 key购买 nike

我正在用 C 编写解释器,它在 ubuntu 控制台中运行。我想在键入时更改 intdoublefunction 等关键字的颜色。

示例:functio 应该是白色的,但是当我输入关键字 function 后,它的颜色应该立即变为蓝色。

可以回答我的问题的简单示例:当字母数为奇数时,我想将终端中的文本颜色设置为蓝色,而当字母数为偶数时,我想将其设置为红色,并且应该在键入时发生(而不是在按回车键之后)。

如何在终端中操作文本并在输入时更改其颜色?

最佳答案

在终端控制台上进行实时文本编辑是一种魔法,因为没有可靠的方法可以确保控制台支持您需要的所有转义序列。对于全屏编辑器,您可以使用 ncurses,但“内联”比看起来更难。举个例子:我的 Mac OS X 自己的终端可以优雅地处理单行输入,但当文本字符串包含不可打印的内容或跨度超出终端的宽度时,它会感到困惑。

以下不那么快(但仍然很脏)的 C 程序允许非常基本的输入。它需要跳过几个环节:在 OS X 上,getchar 将其输入回显到命令行;在输入 Enter 之前,终端本身 (!) 不会将其输入转发给正在运行的程序。其他控制台可能有其他怪癖。

单线编辑工作,唯一值得注意的是如何处理运行线以彩色显示部分。剩下的就是在显示字符串时遍历它,使用 ANSI 转义码来打开和关闭颜色。我的语法高亮方法非常基础;如果您有更多关键字,您可能希望以更快的方式查找它们。它也无法处理通配符,因此您无法突出显示“任何十进制数”之类的内容。

#include <stdio.h>
#include <string.h>
#include <ctype.h>

#include <unistd.h>
#include <termios.h>

#define PP_INVERSE "\x1b[7m"

#define PP_RED "\x1b[31m"
#define PP_BLUE "\x1b[34m"

#define PP_RESET "\x1b[39m\x1b[0m"

const char *keyword_list[] = {
"int",
"double",
"function"
};


/* ha ha */
#define SURELY_LARGE_ENOUGH 256

char line_buf[SURELY_LARGE_ENOUGH] = "";

void display_line (char *text, int cursor)
{
char *startptr, *endptr;
int i;

printf ("\r>");

startptr = text;

do
{
/* hop from one 'word' to another */

/* check for non-word first */
while (*startptr && !isalnum(*startptr))
{
putchar (*startptr);
startptr++;
}

if (*startptr)
{
endptr = startptr;

/* gather next word */
while (*endptr && isalnum(*endptr))
endptr++;

/* check if it's a known keyword */
for (i=0; i<sizeof(keyword_list)/sizeof(keyword_list[0]); i++)
{
if (keyword_list[i][0] == *startptr && strlen(keyword_list[i]) == endptr - startptr)
{
if (!strncmp (startptr, keyword_list[i], strlen(keyword_list[i])))
{
break;
}
}
}

if (i<sizeof(keyword_list)/sizeof(keyword_list[0]))
{
printf (PP_INVERSE "%s" PP_RESET, keyword_list[i]);
startptr = endptr;
} else
{
if ((endptr - startptr) & 1)
{
printf (PP_BLUE);
} else
{
printf (PP_RED);
}
while (startptr < endptr)
{
putchar (*startptr);
startptr++;
}
printf (PP_RESET);
}
}
} while (*startptr);

/* erase any following leftovers */
printf ("\x1B[K");

/* position cursor */
printf ("\r\x1B[%dC", cursor+1);
fflush (stdout);
}

/* turn off echo for getchar
see https://stackoverflow.com/questions/558009/ansi-c-no-echo-keyboard-input
*/
void getch_disable_echo (void)
{
struct termios t;

tcgetattr(STDIN_FILENO, &t);
t.c_lflag &= ~ECHO;
t.c_lflag &= ~ICANON;
tcsetattr(STDIN_FILENO, TCSANOW, &t);
}

void getch_enable_echo (void)
{
struct termios t;

tcgetattr(STDIN_FILENO, &t);
t.c_lflag |= ECHO;
t.c_lflag |= ICANON;
tcsetattr(STDIN_FILENO, TCSANOW, &t);
}

int get_input (void)
{
int cursor_pos = 0;
int key;

do
{
display_line (line_buf, cursor_pos);
key = getchar();
if (isprint(key))
{
if (!line_buf[cursor_pos])
line_buf[cursor_pos+1] = 0;
line_buf[cursor_pos] = key;
cursor_pos++;
}
/* poor man's backspace */
if (key == 127)
{
if (cursor_pos)
{
cursor_pos--;
line_buf[cursor_pos] = 0;
}
}
if (key == '\n')
{
printf ("\r%s \n", line_buf);
line_buf[0] = 0;
cursor_pos = 0;
}
} while (key != EOF);

return 0;
}

int main (void)
{
getch_disable_echo();

strcpy (line_buf, "this is a function test");
get_input ();

getch_enable_echo();
return 0;
}

如您所见,这已经是一个相当大的程序,而且它只允许进行基本的编辑。我省略了制表符完成、光标移动、撤消以及用户可能期望从全行编辑器获得的各种其他细节。与其重新发明轮子,您可以查看 GNU readline 的源代码并检查调整其重绘例程以允许自定义语法突出显示是否更容易。

除此之外,它工作得很好:

screen recording of live input

关于键入时更改 linux 终端中关键字的颜色,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47968822/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com