wresize.c revision 262685
1/****************************************************************************
2 * Copyright (c) 1998-2010,2011 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 1996-on                                        *
31 *     and: Juergen Pfeifer                                                 *
32 ****************************************************************************/
33
34#include <curses.priv.h>
35
36MODULE_ID("$Id: wresize.c,v 1.35 2011/05/21 18:55:07 tom Exp $")
37
38static int
39cleanup_lines(struct ldat *data, int length)
40{
41    while (--length >= 0)
42	free(data[length].text);
43    free(data);
44    return ERR;
45}
46
47/*
48 * If we have reallocated the ldat structs, we will have to repair pointers
49 * used in subwindows.
50 */
51static void
52repair_subwindows(WINDOW *cmp)
53{
54    WINDOWLIST *wp;
55    struct ldat *pline = cmp->_line;
56    int row;
57#ifdef USE_SP_WINDOWLIST
58    SCREEN *sp = _nc_screen_of(cmp);
59#endif
60
61    _nc_lock_global(curses);
62
63    for (each_window(SP_PARM, wp)) {
64	WINDOW *tst = &(wp->win);
65
66	if (tst->_parent == cmp) {
67
68	    if (tst->_pary > cmp->_maxy)
69		tst->_pary = cmp->_maxy;
70	    if (tst->_parx > cmp->_maxx)
71		tst->_parx = cmp->_maxx;
72
73	    if (tst->_maxy + tst->_pary > cmp->_maxy)
74		tst->_maxy = (NCURSES_SIZE_T) (cmp->_maxy - tst->_pary);
75	    if (tst->_maxx + tst->_parx > cmp->_maxx)
76		tst->_maxx = (NCURSES_SIZE_T) (cmp->_maxx - tst->_parx);
77
78	    for (row = 0; row <= tst->_maxy; ++row) {
79		tst->_line[row].text = &pline[tst->_pary + row].text[tst->_parx];
80	    }
81	    repair_subwindows(tst);
82	}
83    }
84    _nc_unlock_global(curses);
85}
86
87/*
88 * Reallocate a curses WINDOW struct to either shrink or grow to the specified
89 * new lines/columns.  If it grows, the new character cells are filled with
90 * blanks.  The application is responsible for repainting the blank area.
91 */
92NCURSES_EXPORT(int)
93wresize(WINDOW *win, int ToLines, int ToCols)
94{
95    int col, row, size_x, size_y;
96    struct ldat *pline;
97    struct ldat *new_lines = 0;
98
99#ifdef TRACE
100    T((T_CALLED("wresize(%p,%d,%d)"), (void *) win, ToLines, ToCols));
101    if (win) {
102	TR(TRACE_UPDATE, ("...beg (%ld, %ld), max(%ld,%ld), reg(%ld,%ld)",
103			  (long) win->_begy, (long) win->_begx,
104			  (long) win->_maxy, (long) win->_maxx,
105			  (long) win->_regtop, (long) win->_regbottom));
106	if (USE_TRACEF(TRACE_UPDATE)) {
107	    _tracedump("...before", win);
108	    _nc_unlock_global(tracef);
109	}
110    }
111#endif
112
113    if (!win || --ToLines < 0 || --ToCols < 0)
114	returnCode(ERR);
115
116    size_x = win->_maxx;
117    size_y = win->_maxy;
118
119    if (ToLines == size_y
120	&& ToCols == size_x)
121	returnCode(OK);
122
123    if ((win->_flags & _SUBWIN)) {
124	/*
125	 * Check if the new limits will fit into the parent window's size.  If
126	 * not, do not resize.  We could adjust the location of the subwindow,
127	 * but the application may not like that.
128	 */
129	if (win->_pary + ToLines > win->_parent->_maxy
130	    || win->_parx + ToCols > win->_parent->_maxx) {
131	    returnCode(ERR);
132	}
133	pline = win->_parent->_line;
134    } else {
135	pline = 0;
136    }
137
138    /*
139     * Allocate new memory as needed.  Do the allocations without modifying
140     * the original window, in case an allocation fails.  Always allocate
141     * (at least temporarily) the array pointing to the individual lines.
142     */
143    new_lines = typeCalloc(struct ldat, (unsigned) (ToLines + 1));
144    if (new_lines == 0)
145	returnCode(ERR);
146
147    /*
148     * For each line in the target, allocate or adjust pointers for the
149     * corresponding text, depending on whether this is a window or a
150     * subwindow.
151     */
152    for (row = 0; row <= ToLines; ++row) {
153	int begin = (row > size_y) ? 0 : (size_x + 1);
154	int end = ToCols;
155	NCURSES_CH_T *s;
156
157	if (!(win->_flags & _SUBWIN)) {
158	    if (row <= size_y) {
159		if (ToCols != size_x) {
160		    s = typeMalloc(NCURSES_CH_T, (unsigned) ToCols + 1);
161		    if (s == 0)
162			returnCode(cleanup_lines(new_lines, row));
163		    for (col = 0; col <= ToCols; ++col) {
164			s[col] = (col <= size_x
165				  ? win->_line[row].text[col]
166				  : win->_nc_bkgd);
167		    }
168		} else {
169		    s = win->_line[row].text;
170		}
171	    } else {
172		s = typeMalloc(NCURSES_CH_T, (unsigned) ToCols + 1);
173		if (s == 0)
174		    returnCode(cleanup_lines(new_lines, row));
175		for (col = 0; col <= ToCols; ++col)
176		    s[col] = win->_nc_bkgd;
177	    }
178	} else if (pline != 0 && pline[win->_pary + row].text != 0) {
179	    s = &pline[win->_pary + row].text[win->_parx];
180	} else {
181	    s = 0;
182	}
183
184	if_USE_SCROLL_HINTS(new_lines[row].oldindex = row);
185	if (row <= size_y) {
186	    new_lines[row].firstchar = win->_line[row].firstchar;
187	    new_lines[row].lastchar = win->_line[row].lastchar;
188	}
189	if ((ToCols != size_x) || (row > size_y)) {
190	    if (end >= begin) {	/* growing */
191		if (new_lines[row].firstchar < begin)
192		    new_lines[row].firstchar = (NCURSES_SIZE_T) begin;
193	    } else {		/* shrinking */
194		new_lines[row].firstchar = 0;
195	    }
196	    new_lines[row].lastchar = (NCURSES_SIZE_T) ToCols;
197	}
198	new_lines[row].text = s;
199    }
200
201    /*
202     * Dispose of unwanted memory.
203     */
204    if (!(win->_flags & _SUBWIN)) {
205	if (ToCols == size_x) {
206	    for (row = ToLines + 1; row <= size_y; row++) {
207		free(win->_line[row].text);
208	    }
209	} else {
210	    for (row = 0; row <= size_y; row++) {
211		free(win->_line[row].text);
212	    }
213	}
214    }
215
216    free(win->_line);
217    win->_line = new_lines;
218
219    /*
220     * Finally, adjust the parameters showing screen size and cursor
221     * position:
222     */
223    win->_maxx = (NCURSES_SIZE_T) ToCols;
224    win->_maxy = (NCURSES_SIZE_T) ToLines;
225
226    if (win->_regtop > win->_maxy)
227	win->_regtop = win->_maxy;
228    if (win->_regbottom > win->_maxy
229	|| win->_regbottom == size_y)
230	win->_regbottom = win->_maxy;
231
232    if (win->_curx > win->_maxx)
233	win->_curx = win->_maxx;
234    if (win->_cury > win->_maxy)
235	win->_cury = win->_maxy;
236
237    /*
238     * Check for subwindows of this one, and readjust pointers to our text,
239     * if needed.
240     */
241    repair_subwindows(win);
242
243#ifdef TRACE
244    TR(TRACE_UPDATE, ("...beg (%ld, %ld), max(%ld,%ld), reg(%ld,%ld)",
245		      (long) win->_begy, (long) win->_begx,
246		      (long) win->_maxy, (long) win->_maxx,
247		      (long) win->_regtop, (long) win->_regbottom));
248    if (USE_TRACEF(TRACE_UPDATE)) {
249	_tracedump("...after:", win);
250	_nc_unlock_global(tracef);
251    }
252#endif
253    returnCode(OK);
254}
255