Resize terminal and scrolling problem with ncurses

2020-02-05 09:54发布

I'm programming in C using ncurses libraries (it's the first time) and I've two problems. I'm on ubuntu with the default terminal (gnome terminal).

1) I need to resize the terminal. I used resizeter() and resize_term(), but they fail.

2) I use scrollok() function and the problem is that I lose scrolled lines (when I get back with wscrl(), there are blank lines).

#include <ncurses.h>

int main() {

WINDOW *win, *win2;

int i;
char c;

initscr();
cbreak();
noecho();

win=newwin(8,20,1,1);
box(win,0,0);
win2=newwin(6,18,2,2);
scrollok(win2,1);
wrefresh(win);
wrefresh(win);

for(i=0;i<15;i++){
    c=wgetch(win2);
    if(c=='u'){
        wscrl(win2,-1);
        wrefresh(win2);
    }
    else{
        wprintw(win2,"%c\n",c);
        wrefresh(win2);
    }
}

delwin(win);
delwin(win2);
endwin();

return 0;
}

标签: c linux ncurses
3条回答
再贱就再见
2楼-- · 2020-02-05 10:15

You can't resize the terminal window from ncurses but you can resize the terminal which the resize system call.

#include <ncurses.h>
#include <stdlib.h>

int main(int argc, char *argv[]){
    WINDOW *ventana1;
    system("resize -s 30 80");
    initscr();
    start_color();
    ventana1 = newwin(15, 50, 0, 0);
    init_pair(1,COLOR_YELLOW,COLOR_BLUE);
    init_pair(2,COLOR_BLUE, COLOR_YELLOW);
    wbkgd(ventana1,COLOR_PAIR(1));
    wprintw(ventana1, "POLLO");
    wrefresh(ventana1);
    wgetch(ventana1);
    wgetch(ventana1);
    system("resize -s 20 60");
    wbkgd(ventana1,COLOR_PAIR(2));
    wprintw(ventana1, "POLLO");
    wrefresh(ventana1);
    wgetch(ventana1);
    wgetch(ventana1);
    system("resize -s 35 85");
    system("clear");
    wbkgd(ventana1,COLOR_PAIR(1));
    wprintw(ventana1, "POLLO");
    wrefresh(ventana1);
    wgetch(ventana1);
    wgetch(ventana1);
    delwin(ventana1);
    endwin();
    system("resize -s 25 75");
}
查看更多
该账号已被封号
3楼-- · 2020-02-05 10:32
  1. You can't resize the terminal window from ncurses. The functions you mention resize the part of the terminal window that is painted on by curses. The idea is you catch the SIGWINCH signal and call resizeterm in the handler when the user resizes the window from outside the application (using the mouse, probably).

  2. This is intended behavior, though poorly documented in ncurses and in the Unix standard/POSIX. NetBSD's curses docs state it explicitly:

    If n is positive then stdscr is scrolled up. n lines are lost from the top of stdscr and n blank lines are inserted at the bottom. If n is negative then stdscr is scrolled down. n blank lines are inserted at the top of stdscr and n lines are lost from the bottom.

    So you'll have to manually save input and reprint it when scrolling.

查看更多
放我归山
4楼-- · 2020-02-05 10:36

POSIX does not cover this case, because the curses document is not part of POSIX. The Open Group happens to maintain documentation for both:

As noted in the manual page for resizeterm, you should not call that function from within a signal handler, because it calls "unsafe" functions. The topic of "unsafe" functions is discussed in several places; that in gcc's documentation would do for a start.

Regarding documentation, @larsmans appears to be quoting from scroll(3), but not citing comparable links for ncurses and "POSIX". For what it's worth:

  • ncurses (seems to address the points implied to be unique to NetBSD)
  • X/Open (is necessarily more general, as it is intended to cover different implementations)

Back to OP's question:

  1. the sample program does not show OP's use of resizeterm nor of resize_term. It is not stated, but presumably OP resized the terminal window and the program did not respond. The manual page for resizeterm is clear enough that ncurses does not cause the terminal to resize. For that (on some terminals), one can use the -s option of resize (a utility program for xterm). If successful, that resizes the terminal, which in turn sends a SIGWINCH. ncurses has a predefined signal handler for that, but at the application level, handling KEY_RESIZE is the recommended way. There are several programs in ncurses-examples which do this.
  2. moving the lines up in a window necessarily moves some out of the window. That implies that lines are shifted into the window to replace those which have left. A "window" is just that: a limited-size view of data. For views with different size, the developer is advised to use "pads" (see manual page). The notes in the scroll manual page mention some issues with the color of blanks (the replacement lines). It is up to the application whether to leave the replacements blank or fill them with the application's data. Curses does not do that automatically (not even for pads).
查看更多
登录 后发表回答