You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
583 lines
15 KiB
583 lines
15 KiB
3 years ago
|
/* PDCurses */
|
||
|
|
||
|
#include <curspriv.h>
|
||
|
|
||
|
/*man-start**************************************************************
|
||
|
|
||
|
window
|
||
|
------
|
||
|
|
||
|
### Synopsis
|
||
|
|
||
|
WINDOW *newwin(int nlines, int ncols, int begy, int begx);
|
||
|
WINDOW *derwin(WINDOW* orig, int nlines, int ncols,
|
||
|
int begy, int begx);
|
||
|
WINDOW *subwin(WINDOW* orig, int nlines, int ncols,
|
||
|
int begy, int begx);
|
||
|
WINDOW *dupwin(WINDOW *win);
|
||
|
int delwin(WINDOW *win);
|
||
|
int mvwin(WINDOW *win, int y, int x);
|
||
|
int mvderwin(WINDOW *win, int pary, int parx);
|
||
|
int syncok(WINDOW *win, bool bf);
|
||
|
void wsyncup(WINDOW *win);
|
||
|
void wcursyncup(WINDOW *win);
|
||
|
void wsyncdown(WINDOW *win);
|
||
|
|
||
|
WINDOW *resize_window(WINDOW *win, int nlines, int ncols);
|
||
|
int wresize(WINDOW *win, int nlines, int ncols);
|
||
|
WINDOW *PDC_makelines(WINDOW *win);
|
||
|
WINDOW *PDC_makenew(int nlines, int ncols, int begy, int begx);
|
||
|
void PDC_sync(WINDOW *win);
|
||
|
|
||
|
### Description
|
||
|
|
||
|
newwin() creates a new window with the given number of lines, nlines
|
||
|
and columns, ncols. The upper left corner of the window is at line
|
||
|
begy, column begx. If nlines is zero, it defaults to LINES - begy;
|
||
|
ncols to COLS - begx. Create a new full-screen window by calling
|
||
|
newwin(0, 0, 0, 0).
|
||
|
|
||
|
delwin() deletes the named window, freeing all associated memory. In
|
||
|
the case of overlapping windows, subwindows should be deleted before
|
||
|
the main window.
|
||
|
|
||
|
mvwin() moves the window so that the upper left-hand corner is at
|
||
|
position (y,x). If the move would cause the window to be off the
|
||
|
screen, it is an error and the window is not moved. Moving subwindows
|
||
|
is allowed.
|
||
|
|
||
|
subwin() creates a new subwindow within a window. The dimensions of
|
||
|
the subwindow are nlines lines and ncols columns. The subwindow is at
|
||
|
position (begy, begx) on the screen. This position is relative to the
|
||
|
screen, and not to the window orig. Changes made to either window
|
||
|
will affect both. When using this routine, you will often need to
|
||
|
call touchwin() before calling wrefresh().
|
||
|
|
||
|
derwin() is the same as subwin(), except that begy and begx are
|
||
|
relative to the origin of the window orig rather than the screen.
|
||
|
There is no difference between subwindows and derived windows.
|
||
|
|
||
|
mvderwin() moves a derived window (or subwindow) inside its parent
|
||
|
window. The screen-relative parameters of the window are not changed.
|
||
|
This routine is used to display different parts of the parent window
|
||
|
at the same physical position on the screen.
|
||
|
|
||
|
dupwin() creates an exact duplicate of the window win.
|
||
|
|
||
|
wsyncup() causes a touchwin() of all of the window's parents.
|
||
|
|
||
|
If wsyncok() is called with a second argument of TRUE, this causes a
|
||
|
wsyncup() to be called every time the window is changed.
|
||
|
|
||
|
wcursyncup() causes the current cursor position of all of a window's
|
||
|
ancestors to reflect the current cursor position of the current
|
||
|
window.
|
||
|
|
||
|
wsyncdown() causes a touchwin() of the current window if any of its
|
||
|
parent's windows have been touched.
|
||
|
|
||
|
resize_window() allows the user to resize an existing window. It
|
||
|
returns the pointer to the new window, or NULL on failure.
|
||
|
|
||
|
wresize() is an ncurses-compatible wrapper for resize_window(). Note
|
||
|
that, unlike ncurses, it will NOT process any subwindows of the
|
||
|
window. (However, you still can call it _on_ subwindows.) It returns
|
||
|
OK or ERR.
|
||
|
|
||
|
PDC_makenew() allocates all data for a new WINDOW * except the actual
|
||
|
lines themselves. If it's unable to allocate memory for the window
|
||
|
structure, it will free all allocated memory and return a NULL
|
||
|
pointer.
|
||
|
|
||
|
PDC_makelines() allocates the memory for the lines.
|
||
|
|
||
|
PDC_sync() handles wrefresh() and wsyncup() calls when a window is
|
||
|
changed.
|
||
|
|
||
|
### Return Value
|
||
|
|
||
|
newwin(), subwin(), derwin() and dupwin() return a pointer to the new
|
||
|
window, or NULL on failure. delwin(), mvwin(), mvderwin() and
|
||
|
syncok() return OK or ERR. wsyncup(), wcursyncup() and wsyncdown()
|
||
|
return nothing.
|
||
|
|
||
|
### Errors
|
||
|
|
||
|
It is an error to call resize_window() before calling initscr().
|
||
|
Also, an error will be generated if we fail to create a newly sized
|
||
|
replacement window for curscr, or stdscr. This could happen when
|
||
|
increasing the window size. NOTE: If this happens, the previously
|
||
|
successfully allocated windows are left alone; i.e., the resize is
|
||
|
NOT cancelled for those windows.
|
||
|
|
||
|
### Portability
|
||
|
X/Open ncurses NetBSD
|
||
|
newwin Y Y Y
|
||
|
delwin Y Y Y
|
||
|
mvwin Y Y Y
|
||
|
subwin Y Y Y
|
||
|
derwin Y Y Y
|
||
|
mvderwin Y Y Y
|
||
|
dupwin Y Y Y
|
||
|
wsyncup Y Y Y
|
||
|
syncok Y Y Y
|
||
|
wcursyncup Y Y Y
|
||
|
wsyncdown Y Y Y
|
||
|
wresize - Y Y
|
||
|
resize_window - - -
|
||
|
PDC_makelines - - -
|
||
|
PDC_makenew - - -
|
||
|
PDC_sync - - -
|
||
|
|
||
|
**man-end****************************************************************/
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
WINDOW *PDC_makenew(int nlines, int ncols, int begy, int begx)
|
||
|
{
|
||
|
WINDOW *win;
|
||
|
|
||
|
PDC_LOG(("PDC_makenew() - called: lines %d cols %d begy %d begx %d\n",
|
||
|
nlines, ncols, begy, begx));
|
||
|
|
||
|
/* allocate the window structure itself */
|
||
|
|
||
|
win = calloc(1, sizeof(WINDOW));
|
||
|
if (!win)
|
||
|
return win;
|
||
|
|
||
|
/* allocate the line pointer array */
|
||
|
|
||
|
win->_y = malloc(nlines * sizeof(chtype *));
|
||
|
if (!win->_y)
|
||
|
{
|
||
|
free(win);
|
||
|
return (WINDOW *)NULL;
|
||
|
}
|
||
|
|
||
|
/* allocate the minchng and maxchng arrays */
|
||
|
|
||
|
win->_firstch = malloc(nlines * sizeof(int));
|
||
|
if (!win->_firstch)
|
||
|
{
|
||
|
free(win->_y);
|
||
|
free(win);
|
||
|
return (WINDOW *)NULL;
|
||
|
}
|
||
|
|
||
|
win->_lastch = malloc(nlines * sizeof(int));
|
||
|
if (!win->_lastch)
|
||
|
{
|
||
|
free(win->_firstch);
|
||
|
free(win->_y);
|
||
|
free(win);
|
||
|
return (WINDOW *)NULL;
|
||
|
}
|
||
|
|
||
|
/* initialize window variables */
|
||
|
|
||
|
win->_maxy = nlines; /* real max screen size */
|
||
|
win->_maxx = ncols; /* real max screen size */
|
||
|
win->_begy = begy;
|
||
|
win->_begx = begx;
|
||
|
win->_bkgd = ' '; /* wrs 4/10/93 -- initialize background to blank */
|
||
|
win->_clear = (bool) ((nlines == LINES) && (ncols == COLS));
|
||
|
win->_bmarg = nlines - 1;
|
||
|
win->_parx = win->_pary = -1;
|
||
|
|
||
|
/* init to say window all changed */
|
||
|
|
||
|
touchwin(win);
|
||
|
|
||
|
return win;
|
||
|
}
|
||
|
|
||
|
WINDOW *PDC_makelines(WINDOW *win)
|
||
|
{
|
||
|
int i, j, nlines, ncols;
|
||
|
|
||
|
PDC_LOG(("PDC_makelines() - called\n"));
|
||
|
|
||
|
if (!win)
|
||
|
return (WINDOW *)NULL;
|
||
|
|
||
|
nlines = win->_maxy;
|
||
|
ncols = win->_maxx;
|
||
|
|
||
|
for (i = 0; i < nlines; i++)
|
||
|
{
|
||
|
win->_y[i] = malloc(ncols * sizeof(chtype));
|
||
|
if (!win->_y[i])
|
||
|
{
|
||
|
/* if error, free all the data */
|
||
|
|
||
|
for (j = 0; j < i; j++)
|
||
|
free(win->_y[j]);
|
||
|
|
||
|
free(win->_firstch);
|
||
|
free(win->_lastch);
|
||
|
free(win->_y);
|
||
|
free(win);
|
||
|
|
||
|
return (WINDOW *)NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return win;
|
||
|
}
|
||
|
|
||
|
void PDC_sync(WINDOW *win)
|
||
|
{
|
||
|
PDC_LOG(("PDC_sync() - called:\n"));
|
||
|
|
||
|
if (win->_immed)
|
||
|
wrefresh(win);
|
||
|
if (win->_sync)
|
||
|
wsyncup(win);
|
||
|
}
|
||
|
|
||
|
WINDOW *newwin(int nlines, int ncols, int begy, int begx)
|
||
|
{
|
||
|
WINDOW *win;
|
||
|
|
||
|
PDC_LOG(("newwin() - called:lines=%d cols=%d begy=%d begx=%d\n",
|
||
|
nlines, ncols, begy, begx));
|
||
|
|
||
|
if (!nlines)
|
||
|
nlines = LINES - begy;
|
||
|
if (!ncols)
|
||
|
ncols = COLS - begx;
|
||
|
|
||
|
if (!SP || begy + nlines > SP->lines || begx + ncols > SP->cols)
|
||
|
return (WINDOW *)NULL;
|
||
|
|
||
|
win = PDC_makenew(nlines, ncols, begy, begx);
|
||
|
if (win)
|
||
|
win = PDC_makelines(win);
|
||
|
|
||
|
if (win)
|
||
|
werase(win);
|
||
|
|
||
|
return win;
|
||
|
}
|
||
|
|
||
|
int delwin(WINDOW *win)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
PDC_LOG(("delwin() - called\n"));
|
||
|
|
||
|
if (!win)
|
||
|
return ERR;
|
||
|
|
||
|
/* subwindows use parents' lines */
|
||
|
|
||
|
if (!(win->_flags & (_SUBWIN|_SUBPAD)))
|
||
|
for (i = 0; i < win->_maxy && win->_y[i]; i++)
|
||
|
if (win->_y[i])
|
||
|
free(win->_y[i]);
|
||
|
|
||
|
free(win->_firstch);
|
||
|
free(win->_lastch);
|
||
|
free(win->_y);
|
||
|
free(win);
|
||
|
|
||
|
return OK;
|
||
|
}
|
||
|
|
||
|
int mvwin(WINDOW *win, int y, int x)
|
||
|
{
|
||
|
PDC_LOG(("mvwin() - called\n"));
|
||
|
|
||
|
if (!win || (y + win->_maxy > LINES || y < 0)
|
||
|
|| (x + win->_maxx > COLS || x < 0))
|
||
|
return ERR;
|
||
|
|
||
|
win->_begy = y;
|
||
|
win->_begx = x;
|
||
|
touchwin(win);
|
||
|
|
||
|
return OK;
|
||
|
}
|
||
|
|
||
|
WINDOW *subwin(WINDOW *orig, int nlines, int ncols, int begy, int begx)
|
||
|
{
|
||
|
WINDOW *win;
|
||
|
int i, j, k;
|
||
|
|
||
|
PDC_LOG(("subwin() - called: lines %d cols %d begy %d begx %d\n",
|
||
|
nlines, ncols, begy, begx));
|
||
|
|
||
|
/* make sure window fits inside the original one */
|
||
|
|
||
|
if (!orig || (begy < orig->_begy) || (begx < orig->_begx) ||
|
||
|
(begy + nlines) > (orig->_begy + orig->_maxy) ||
|
||
|
(begx + ncols) > (orig->_begx + orig->_maxx))
|
||
|
return (WINDOW *)NULL;
|
||
|
|
||
|
j = begy - orig->_begy;
|
||
|
k = begx - orig->_begx;
|
||
|
|
||
|
if (!nlines)
|
||
|
nlines = orig->_maxy - 1 - j;
|
||
|
if (!ncols)
|
||
|
ncols = orig->_maxx - 1 - k;
|
||
|
|
||
|
win = PDC_makenew(nlines, ncols, begy, begx);
|
||
|
if (!win)
|
||
|
return (WINDOW *)NULL;
|
||
|
|
||
|
/* initialize window variables */
|
||
|
|
||
|
win->_attrs = orig->_attrs;
|
||
|
win->_bkgd = orig->_bkgd;
|
||
|
win->_leaveit = orig->_leaveit;
|
||
|
win->_scroll = orig->_scroll;
|
||
|
win->_nodelay = orig->_nodelay;
|
||
|
win->_delayms = orig->_delayms;
|
||
|
win->_use_keypad = orig->_use_keypad;
|
||
|
win->_immed = orig->_immed;
|
||
|
win->_sync = orig->_sync;
|
||
|
win->_pary = j;
|
||
|
win->_parx = k;
|
||
|
win->_parent = orig;
|
||
|
|
||
|
for (i = 0; i < nlines; i++, j++)
|
||
|
win->_y[i] = orig->_y[j] + k;
|
||
|
|
||
|
win->_flags |= _SUBWIN;
|
||
|
|
||
|
return win;
|
||
|
}
|
||
|
|
||
|
WINDOW *derwin(WINDOW *orig, int nlines, int ncols, int begy, int begx)
|
||
|
{
|
||
|
return subwin(orig, nlines, ncols, begy + orig->_begy, begx + orig->_begx);
|
||
|
}
|
||
|
|
||
|
int mvderwin(WINDOW *win, int pary, int parx)
|
||
|
{
|
||
|
int i, j;
|
||
|
WINDOW *mypar;
|
||
|
|
||
|
if (!win || !(win->_parent))
|
||
|
return ERR;
|
||
|
|
||
|
mypar = win->_parent;
|
||
|
|
||
|
if (pary < 0 || parx < 0 || (pary + win->_maxy) > mypar->_maxy ||
|
||
|
(parx + win->_maxx) > mypar->_maxx)
|
||
|
return ERR;
|
||
|
|
||
|
j = pary;
|
||
|
|
||
|
for (i = 0; i < win->_maxy; i++)
|
||
|
win->_y[i] = (mypar->_y[j++]) + parx;
|
||
|
|
||
|
win->_pary = pary;
|
||
|
win->_parx = parx;
|
||
|
|
||
|
return OK;
|
||
|
}
|
||
|
|
||
|
WINDOW *dupwin(WINDOW *win)
|
||
|
{
|
||
|
WINDOW *new;
|
||
|
chtype *ptr, *ptr1;
|
||
|
int nlines, ncols, begy, begx, i;
|
||
|
|
||
|
if (!win)
|
||
|
return (WINDOW *)NULL;
|
||
|
|
||
|
nlines = win->_maxy;
|
||
|
ncols = win->_maxx;
|
||
|
begy = win->_begy;
|
||
|
begx = win->_begx;
|
||
|
|
||
|
new = PDC_makenew(nlines, ncols, begy, begx);
|
||
|
if (new)
|
||
|
new = PDC_makelines(new);
|
||
|
|
||
|
if (!new)
|
||
|
return (WINDOW *)NULL;
|
||
|
|
||
|
/* copy the contents of win into new */
|
||
|
|
||
|
for (i = 0; i < nlines; i++)
|
||
|
{
|
||
|
for (ptr = new->_y[i], ptr1 = win->_y[i];
|
||
|
ptr < new->_y[i] + ncols; ptr++, ptr1++)
|
||
|
*ptr = *ptr1;
|
||
|
|
||
|
new->_firstch[i] = 0;
|
||
|
new->_lastch[i] = ncols - 1;
|
||
|
}
|
||
|
|
||
|
new->_curx = win->_curx;
|
||
|
new->_cury = win->_cury;
|
||
|
new->_maxy = win->_maxy;
|
||
|
new->_maxx = win->_maxx;
|
||
|
new->_begy = win->_begy;
|
||
|
new->_begx = win->_begx;
|
||
|
new->_flags = win->_flags;
|
||
|
new->_attrs = win->_attrs;
|
||
|
new->_clear = win->_clear;
|
||
|
new->_leaveit = win->_leaveit;
|
||
|
new->_scroll = win->_scroll;
|
||
|
new->_nodelay = win->_nodelay;
|
||
|
new->_delayms = win->_delayms;
|
||
|
new->_use_keypad = win->_use_keypad;
|
||
|
new->_tmarg = win->_tmarg;
|
||
|
new->_bmarg = win->_bmarg;
|
||
|
new->_parx = win->_parx;
|
||
|
new->_pary = win->_pary;
|
||
|
new->_parent = win->_parent;
|
||
|
new->_bkgd = win->_bkgd;
|
||
|
new->_flags = win->_flags;
|
||
|
|
||
|
return new;
|
||
|
}
|
||
|
|
||
|
WINDOW *resize_window(WINDOW *win, int nlines, int ncols)
|
||
|
{
|
||
|
WINDOW *new;
|
||
|
int i, save_cury, save_curx, new_begy, new_begx;
|
||
|
|
||
|
PDC_LOG(("resize_window() - called: nlines %d ncols %d\n",
|
||
|
nlines, ncols));
|
||
|
|
||
|
if (!win || !SP)
|
||
|
return (WINDOW *)NULL;
|
||
|
|
||
|
if (win->_flags & _SUBPAD)
|
||
|
{
|
||
|
new = subpad(win->_parent, nlines, ncols, win->_begy, win->_begx);
|
||
|
if (!new)
|
||
|
return (WINDOW *)NULL;
|
||
|
}
|
||
|
else if (win->_flags & _SUBWIN)
|
||
|
{
|
||
|
new = subwin(win->_parent, nlines, ncols, win->_begy, win->_begx);
|
||
|
if (!new)
|
||
|
return (WINDOW *)NULL;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (win == SP->slk_winptr)
|
||
|
{
|
||
|
new_begy = SP->lines - SP->slklines;
|
||
|
new_begx = 0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
new_begy = win->_begy;
|
||
|
new_begx = win->_begx;
|
||
|
}
|
||
|
|
||
|
new = PDC_makenew(nlines, ncols, new_begy, new_begx);
|
||
|
if (!new)
|
||
|
return (WINDOW *)NULL;
|
||
|
}
|
||
|
|
||
|
save_curx = min(win->_curx, (new->_maxx - 1));
|
||
|
save_cury = min(win->_cury, (new->_maxy - 1));
|
||
|
|
||
|
if (!(win->_flags & (_SUBPAD|_SUBWIN)))
|
||
|
{
|
||
|
new = PDC_makelines(new);
|
||
|
if (!new)
|
||
|
return (WINDOW *)NULL;
|
||
|
|
||
|
new->_bkgd = win->_bkgd;
|
||
|
werase(new);
|
||
|
|
||
|
copywin(win, new, 0, 0, 0, 0, min(win->_maxy, new->_maxy) - 1,
|
||
|
min(win->_maxx, new->_maxx) - 1, FALSE);
|
||
|
|
||
|
for (i = 0; i < win->_maxy && win->_y[i]; i++)
|
||
|
if (win->_y[i])
|
||
|
free(win->_y[i]);
|
||
|
}
|
||
|
|
||
|
new->_flags = win->_flags;
|
||
|
new->_attrs = win->_attrs;
|
||
|
new->_clear = win->_clear;
|
||
|
new->_leaveit = win->_leaveit;
|
||
|
new->_scroll = win->_scroll;
|
||
|
new->_nodelay = win->_nodelay;
|
||
|
new->_delayms = win->_delayms;
|
||
|
new->_use_keypad = win->_use_keypad;
|
||
|
new->_tmarg = (win->_tmarg > new->_maxy - 1) ? 0 : win->_tmarg;
|
||
|
new->_bmarg = (win->_bmarg == win->_maxy - 1) ?
|
||
|
new->_maxy - 1 : min(win->_bmarg, (new->_maxy - 1));
|
||
|
new->_parent = win->_parent;
|
||
|
new->_immed = win->_immed;
|
||
|
new->_sync = win->_sync;
|
||
|
new->_bkgd = win->_bkgd;
|
||
|
|
||
|
new->_curx = save_curx;
|
||
|
new->_cury = save_cury;
|
||
|
|
||
|
free(win->_firstch);
|
||
|
free(win->_lastch);
|
||
|
free(win->_y);
|
||
|
|
||
|
*win = *new;
|
||
|
free(new);
|
||
|
|
||
|
return win;
|
||
|
}
|
||
|
|
||
|
int wresize(WINDOW *win, int nlines, int ncols)
|
||
|
{
|
||
|
return (resize_window(win, nlines, ncols) ? OK : ERR);
|
||
|
}
|
||
|
|
||
|
void wsyncup(WINDOW *win)
|
||
|
{
|
||
|
WINDOW *tmp;
|
||
|
|
||
|
PDC_LOG(("wsyncup() - called\n"));
|
||
|
|
||
|
for (tmp = win; tmp; tmp = tmp->_parent)
|
||
|
touchwin(tmp);
|
||
|
}
|
||
|
|
||
|
int syncok(WINDOW *win, bool bf)
|
||
|
{
|
||
|
PDC_LOG(("syncok() - called\n"));
|
||
|
|
||
|
if (!win)
|
||
|
return ERR;
|
||
|
|
||
|
win->_sync = bf;
|
||
|
|
||
|
return OK;
|
||
|
}
|
||
|
|
||
|
void wcursyncup(WINDOW *win)
|
||
|
{
|
||
|
WINDOW *tmp;
|
||
|
|
||
|
PDC_LOG(("wcursyncup() - called\n"));
|
||
|
|
||
|
for (tmp = win; tmp && tmp->_parent; tmp = tmp->_parent)
|
||
|
wmove(tmp->_parent, tmp->_pary + tmp->_cury, tmp->_parx + tmp->_curx);
|
||
|
}
|
||
|
|
||
|
void wsyncdown(WINDOW *win)
|
||
|
{
|
||
|
WINDOW *tmp;
|
||
|
|
||
|
PDC_LOG(("wsyncdown() - called\n"));
|
||
|
|
||
|
for (tmp = win; tmp; tmp = tmp->_parent)
|
||
|
{
|
||
|
if (is_wintouched(tmp))
|
||
|
{
|
||
|
touchwin(win);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|