0

在 C 中使用带有 Readline 库的自动完成时,在键入tab两次键时会重新打印提示:

(prompt) view NAME_OF_F (user presses tab twice)
NAME_OF_FILE1   NAME_OF_FILE2   (suggestions by Readline)
(prompt) view NAME_OF_F 

我想通过在第一行打印下面的建议来抑制在第 3 行重新打印提示,如下所示:

(prompt) view NAME_OF_F (user presses tab twice)
NAME_OF_FILE1   NAME_OF_FILE2   (suggestions by Readline)

我希望光标回到有提示的第一行的末尾。

编译gcc -Wall -O0 -ggdb -fno-builtin rline.c -o rline -lreadline -ltermcap

这是一个代码示例:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <readline/readline.h>

int execute_line(char *line);
void initialize_readline();
static char **fileman_completion(char *text, int start, int end);
static char *command_generator(char *text, int state);

char *command[] = { "view", "quit", (char *)NULL };

int done; /* When non-zero, this global means the user is done using this program. */

int main(int argc, char **argv)
{
    char *line;
    initialize_readline();  /* Bind our completer. */
    for ( ; done == 0; ) {
        line = readline("> ");

        if (!line) 
            break;

        if (*line) 
            execute_line(line);
        free(line);
    }
    return 0;
}

/* String to pass to system().  This is for the VIEW command. */
static char syscom[1024];

int execute_line(char *line)
{
    int i = 0;
    char *word;
    /* Isolate the command word. */
    while (line[i] && whitespace(line[i]))
        i++;
    word = line + i;

    while (line[i] && !whitespace(line[i])) i++;

    if (line[i]) line[i++] = '\0';

    if (strcmp(word, "quit") == 0) {
        done = 1;
        return 0;
    } else if (strcmp(word, "view")) {
        fprintf(stderr, "%s: Choose only \"view FILE\" or \"quit\" as your command.\n", word);
        return -1;
    }

    /* Get argument to command, if any. */
    while (whitespace(line[i])) i++;

    word = line + i;
    if(!word || !*word) {
        fprintf(stderr, "view: Argument required.\n");
        return -1;
    }
    sprintf(syscom, "more %s", word);
    return system(syscom);
}

void initialize_readline()
{
    rl_readline_name = "rline";
    rl_attempted_completion_function = (rl_completion_func_t *)fileman_completion;
}

static char **fileman_completion(char *text, int start, int end)
{
    if (start == 0)
        return rl_completion_matches(text, (rl_compentry_func_t *)*command_generator);
    return NULL;
}

static char *command_generator(char *text, int state)
{
    static int list_index, len;
    char *name;
    if (!state) {
        list_index = 0;
        len = strlen(text);
    }
    while ((name = command[list_index++]))
        if (strncmp(name, text, len) == 0)
            return strdup(name);
    return NULL;
}

该程序只接受view FILE_NAME查看文件内容和quit退出程序的命令。该示例是此处找到的示例程序的简化版本。

4

2 回答 2

0

感谢@rici 的启发。我通过一些修改让它与他的功能一起使用。为了使其正常工作,您需要下载 readline 库。在 readline 的文件中,rlprivate.h我删除了 lineschar **lines;#include "realdine.h". display.c然后在你自己的 .c 中你必须有一个#include </PATH/TO/display.c>. 在该 display.c 中,#include 指向修改后的 rlprivate.h。所有这一切让我可以访问_rl_move_vert(1).

static void display_matches(char** matches, int len, int max)
{
    int saved_point = rl_point;
    char *saved_line = rl_copy_text(0, rl_end);
    rl_save_prompt();
    rl_replace_line("", 0); // Clear the previous text
    putp(cursor_up);
    _rl_move_vert(1);
    rl_display_match_list(matches, len, max);
    putp(cursor_up);
    rl_restore_prompt();
    rl_replace_line(saved_line, 0);
    rl_point = saved_point;
    rl_redisplay();
    putp(cursor_down);
    free(saved_line);
}

于 2020-10-10T23:45:09.903 回答
0

我不认为 readline 内置了类似的东西,但是如果您想尝试自己编写逻辑,它确实提供了很多自定义的可能性。

您可以尝试编写自定义rl_completion_display_matches_hook来显示完成列表。但我并不完全清楚之后如何恢复光标位置。我认为 readline 没有用于查找或重置光标位置的公共界面。(当然,完成列表可能太大以至于原始命令滚动到屏幕之外。)

作为替代方案,我可以使用钩子在当前行的顶部打印完成列表,然后在完成列表之后重新显示提示(尽管我通过假设当前输入总是只有一行来欺骗)。这不是您所要求的,但它可能对演示有用。我使用了以下自定义匹配打印机:

static void display_matches(char** matches, int len, int max) {
    putp(carriage_return);
    putp(clr_eol);
    putp(cursor_up);
    rl_display_match_list(matches, len, max);
    rl_forced_update_display();
}

我还在初始化函数中添加了以下内容:

    rl_completion_display_matches_hook = display_matches;
    setupterm(NULL, 1, (int*)0);
于 2020-10-05T04:55:19.480 回答