8121ff154dd58085c1d567821d9d12e104043700
[ncurses.git] / ncurses / base / wresize.c
1 /****************************************************************************
2  * Copyright (c) 1998,1999,2000 Free Software Foundation, Inc.              *
3  *                                                                          *
4  * Permission is hereby granted, free of charge, to any person obtaining a  *
5  * copy of this software and associated documentation files (the            *
6  * "Software"), to deal in the Software without restriction, including      *
7  * without limitation the rights to use, copy, modify, merge, publish,      *
8  * distribute, distribute with modifications, sublicense, and/or sell       *
9  * copies of the Software, and to permit persons to whom the Software is    *
10  * furnished to do so, subject to the following conditions:                 *
11  *                                                                          *
12  * The above copyright notice and this permission notice shall be included  *
13  * in all copies or substantial portions of the Software.                   *
14  *                                                                          *
15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS  *
16  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF               *
17  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.   *
18  * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,   *
19  * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR    *
20  * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR    *
21  * THE USE OR OTHER DEALINGS IN THE SOFTWARE.                               *
22  *                                                                          *
23  * Except as contained in this notice, the name(s) of the above copyright   *
24  * holders shall not be used in advertising or otherwise to promote the     *
25  * sale, use or other dealings in this Software without prior written       *
26  * authorization.                                                           *
27  ****************************************************************************/
28
29 /****************************************************************************
30  *  Author: Thomas E. Dickey <dickey@clark.net> 1996,1997                   *
31  ****************************************************************************/
32
33 #include <curses.priv.h>
34
35 MODULE_ID("$Id: wresize.c,v 1.16 2000/03/05 00:14:35 tom Exp $")
36
37 /*
38  * Reallocate a curses WINDOW struct to either shrink or grow to the specified
39  * new lines/columns.  If it grows, the new character cells are filled with
40  * blanks.  The application is responsible for repainting the blank area.
41  */
42
43 #define DOALLOC(p,t,n)  typeRealloc(t, n, p)
44 #define ld_ALLOC(p,n)   DOALLOC(p,struct ldat,n)
45 #define c_ALLOC(p,n)    DOALLOC(p,chtype,n)
46
47 int
48 wresize(WINDOW *win, int ToLines, int ToCols)
49 {
50     register int row;
51     int size_x, size_y;
52     struct ldat *pline;
53     chtype blank;
54
55 #ifdef TRACE
56     T((T_CALLED("wresize(%p,%d,%d)"), win, ToLines, ToCols));
57     if (win) {
58         TR(TRACE_UPDATE, ("...beg (%d, %d), max(%d,%d), reg(%d,%d)",
59                 win->_begy, win->_begx,
60                 win->_maxy, win->_maxx,
61                 win->_regtop, win->_regbottom));
62         if (_nc_tracing & TRACE_UPDATE)
63             _tracedump("...before", win);
64     }
65 #endif
66
67     if (!win || --ToLines < 0 || --ToCols < 0)
68         returnCode(ERR);
69
70     size_x = win->_maxx;
71     size_y = win->_maxy;
72
73     if (ToLines == size_y
74         && ToCols == size_x)
75         returnCode(OK);
76
77     if ((win->_flags & _SUBWIN)) {
78         /*
79          * Check if the new limits will fit into the parent window's size.  If
80          * not, do not resize.  We could adjust the location of the subwindow,
81          * but the application may not like that.
82          */
83         if (win->_pary + ToLines > win->_parent->_maxy
84             || win->_parx + ToCols > win->_parent->_maxx) {
85             returnCode(ERR);
86         }
87         pline = win->_parent->_line;
88     } else {
89         pline = 0;
90     }
91
92     /*
93      * If the number of lines has changed, adjust the size of the overall
94      * vector:
95      */
96     if (ToLines != size_y) {
97         if (!(win->_flags & _SUBWIN)) {
98             for (row = ToLines + 1; row <= size_y; row++)
99                 free((char *) (win->_line[row].text));
100         }
101
102         win->_line = ld_ALLOC(win->_line, ToLines + 1);
103         if (win->_line == 0)
104             returnCode(ERR);
105
106         for (row = size_y + 1; row <= ToLines; row++) {
107             win->_line[row].text = 0;
108             win->_line[row].firstchar = 0;
109             win->_line[row].lastchar = ToCols;
110             if ((win->_flags & _SUBWIN)) {
111                 win->_line[row].text =
112                     &pline[win->_pary + row].text[win->_parx];
113             }
114         }
115     }
116
117     /*
118      * Adjust the width of the columns:
119      */
120     blank = _nc_background(win);
121     for (row = 0; row <= ToLines; row++) {
122         chtype *s = win->_line[row].text;
123         int begin = (s == 0) ? 0 : size_x + 1;
124         int end = ToCols;
125
126         if_USE_SCROLL_HINTS(win->_line[row].oldindex = row);
127
128         if (ToCols != size_x || s == 0) {
129             if (!(win->_flags & _SUBWIN)) {
130                 win->_line[row].text = s = c_ALLOC(s, ToCols + 1);
131                 if (win->_line[row].text == 0)
132                     returnCode(ERR);
133             } else if (s == 0) {
134                 win->_line[row].text = s =
135                     &pline[win->_pary + row].text[win->_parx];
136             }
137
138             if (end >= begin) { /* growing */
139                 if (win->_line[row].firstchar < begin)
140                     win->_line[row].firstchar = begin;
141                 win->_line[row].lastchar = ToCols;
142                 do {
143                     s[end] = blank;
144                 } while (--end >= begin);
145             } else {            /* shrinking */
146                 win->_line[row].firstchar = 0;
147                 win->_line[row].lastchar = ToCols;
148             }
149         }
150     }
151
152     /*
153      * Finally, adjust the parameters showing screen size and cursor
154      * position:
155      */
156     win->_maxx = ToCols;
157     win->_maxy = ToLines;
158
159     if (win->_regtop > win->_maxy)
160         win->_regtop = win->_maxy;
161     if (win->_regbottom > win->_maxy
162         || win->_regbottom == size_y)
163         win->_regbottom = win->_maxy;
164
165     if (win->_curx > win->_maxx)
166         win->_curx = win->_maxx;
167     if (win->_cury > win->_maxy)
168         win->_cury = win->_maxy;
169
170 #ifdef TRACE
171     TR(TRACE_UPDATE, ("...beg (%d, %d), max(%d,%d), reg(%d,%d)",
172             win->_begy, win->_begx,
173             win->_maxy, win->_maxx,
174             win->_regtop, win->_regbottom));
175     if (_nc_tracing & TRACE_UPDATE)
176         _tracedump("...after:", win);
177 #endif
178     returnCode(OK);
179 }