Ncurses存在调整大小的故障?

12

我正在编写一个使用 ncurses 的程序,并尝试使其正确响应终端大小调整。虽然我可以在程序中正确读取终端尺寸,但是 ncurses 似乎不能正确处理新的尺寸。这是一个(有点冗长的)示例程序:

#include <ncurses.h>
#include <string.h>
#include <signal.h>
#include <sys/ioctl.h>

void handle_winch(int sig){

    struct winsize w;
    ioctl(0, TIOCGWINSZ, &w);
    COLS = w.ws_col;
    LINES = w.ws_row;

    wresize(stdscr, LINES, COLS);
    clear();

    mvprintw(0, 0, "COLS = %d, LINES = %d", COLS, LINES);
    for (int i = 0; i < COLS; i++)
        mvaddch(1, i, '*');

    refresh();
}

int main(int argc, char *argv[]){

    initscr();

    struct sigaction sa;
    memset(&sa, 0, sizeof(struct sigaction));
    sa.sa_handler = handle_winch;
    sigaction(SIGWINCH, &sa, NULL);

    while(getch() != 27) {}

    endwin();
    return 0;
}

如果您运行它,您会看到终端尺寸被正确地检索。但是第二行应该在屏幕上绘制*字符的操作不起作用。尝试水平调整窗口大小,星号线将不会变长。

问题出在哪里?我知道可以暂时离开curses模式,但我更喜欢一个更简洁的解决方案。谢谢!

1个回答

26
不要设置COLSLINES,这些由ncurses管理。另外,在调整大小后重新初始化ncurses。这意味着不要调用wresize(),而是调用endwin()。在使用其他ncurses函数之前,请确保在endwin()调用后直接调用refresh()。您根本不需要ioctl(),因为ncurses会自动检测新的大小。所以你只需要一个endwin()调用即可。
void handle_winch(int sig)
{
    endwin();
    // Needs to be called after an endwin() so ncurses will initialize
    // itself with the new terminal dimensions.
    refresh();
    clear();

    mvprintw(0, 0, "COLS = %d, LINES = %d", COLS, LINES);
    for (int i = 0; i < COLS; i++)
        mvaddch(1, i, '*');
    refresh();
}

此外,一些ncurses版本被配置为提供自己的SIGWINCH处理程序。当发生调整大小时,这些版本将KEY_RESIZE作为键输入返回。如果您要利用它,您根本不需要信号处理程序。相反,您只需要:

#include <ncurses.h>
#include <string.h>

int main()
{

    initscr();

    int key;
    while ((key = getch()) != 27) {
        if (key == KEY_RESIZE) {
            clear();
            mvprintw(0, 0, "COLS = %d, LINES = %d", COLS, LINES);
            for (int i = 0; i < COLS; i++)
                mvaddch(1, i, '*');
            refresh();
        }
    }

    endwin();
    return 0;
}

不幸的是,你不能依赖于所有ncurses安装都配置了KEY_RESIZE,因此信号处理程序是最具可移植性的解决方案。


1
@ryyst 我也加了关于 KEY_RESIZE 的信息,以防你选择这个路线。 - Nikos C.
1
ncurses提供了resizeterm(lines,cols)函数,如果您不想使用KEY_RESIZE,可以使用它,但它也是ncurses的扩展功能。 - Craig
@NikosC。我在想,如果在使用use_window执行同步窗口更新的ncurses应用程序中,这个依赖信号的代码会表现如何?也就是说,有没有一种方法可以使其线程安全? - user67416
4
让我重申@NikosC所说的,当处理SIGWINCH时,您必须调用endwin(),然后是refresh()。只调用endwin()将使getmaxyx变量处于未定义状态。 - Joe
1
@kevr endwin() 不会对库进行去初始化。它只是禁用 curses 模式并将终端恢复到正常模式。下一个 refresh() 将再次启用 curses 模式。请参阅文档(man endwin)。 - Nikos C.
显示剩余2条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接