xref: /netbsd-src/lib/libcurses/resize.c (revision da5f4674a3fc214be3572d358b66af40ab9401e7)
1 /*	$NetBSD: resize.c,v 1.9 2003/07/30 11:11:55 dsl Exp $	*/
2 
3 /*
4  * Copyright (c) 2001
5  *	Brett Lymn.
6  *
7  * This code has been donated to The NetBSD Foundation by the Author.
8  *
9  * Redistribution and use in source and binary forms, with or without
10  * modification, are permitted provided that the following conditions
11  * are met:
12  * 1. Redistributions of source code must retain the above copyright
13  *    notice, this list of conditions and the following disclaimer.
14  * 2. Redistributions in binary form must reproduce the above copyright
15  *    notice, this list of conditions and the following disclaimer in the
16  *    documentation and/or other materials provided with the distribution.
17  * 3. All advertising materials mentioning features or use of this software
18  *    must display the following acknowledgement:
19  *	This product includes software developed by the University of
20  *	California, Berkeley and its contributors.
21  * 4. Neither the name of the University nor the names of its contributors
22  *    may be used to endorse or promote products derived from this software
23  *    without specific prior written permission.
24  *
25  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
26  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
27  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
28  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
29  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
30  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
31  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
32  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
33  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
34  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
35  * SUCH DAMAGE.
36  */
37 
38 #include <sys/cdefs.h>
39 #ifndef lint
40 #if 0
41 static char sccsid[] = "@(#)resize.c   blymn 2001/08/26";
42 #else
43 __RCSID("$NetBSD: resize.c,v 1.9 2003/07/30 11:11:55 dsl Exp $");
44 #endif
45 #endif				/* not lint */
46 
47 #include <stdlib.h>
48 
49 #include "curses.h"
50 #include "curses_private.h"
51 
52 static int __resizewin(WINDOW *win, int nlines, int ncols);
53 
54 /*
55  * wresize --
56  *	Resize the given window to the new size.
57  */
58 int
59 wresize(WINDOW *win, int req_nlines, int req_ncols)
60 {
61 	int	nlines = req_nlines;
62 	int	ncols = req_ncols;
63 
64 	if (win == NULL)
65 		return ERR;
66 
67 	nlines = req_nlines;
68 	ncols = req_ncols;
69 	if (win->orig == NULL) {
70 		/* bound window to screen */
71 		if (win->begy + nlines > LINES)
72 			nlines = 0;
73 		if (nlines <= 0)
74 			nlines += LINES - win->begy;
75 		if (win->begx + ncols > COLS)
76 			ncols = 0;
77 		if (ncols <= 0)
78 			ncols += COLS - win->begx;
79 	} else {
80 		/* subwins must fit inside the parent - check this */
81 		if (win->begy + nlines > win->orig->begy + win->orig->maxy)
82 			nlines = 0;
83 		if (nlines <= 0)
84 			nlines += win->orig->begy + win->orig->maxy - win->begy;
85 		if (win->begx + ncols > win->orig->begx + win->orig->maxx)
86 			ncols = 0;
87 		if (ncols <= 0)
88 			ncols += win->orig->begx + win->orig->maxx - win->begx;
89 	}
90 
91 	if ((__resizewin(win, nlines, ncols)) == ERR)
92 		return ERR;
93 
94 	win->reqy = req_nlines;
95 	win->reqx = req_ncols;
96 
97 	return OK;
98 }
99 
100 /*
101  * resizeterm --
102  *      Resize the terminal window, resizing the dependent windows.
103  */
104 int
105 resizeterm(int nlines, int ncols)
106 {
107 	WINDOW *win;
108 	struct __winlist *list;
109 	int newlines, newcols;
110 
111 	  /* don't worry if things have not changed... we would like to
112 	     do this but some bastard programs update LINES and COLS before
113 	     calling resizeterm thus negating it's effect.
114 	if ((nlines == LINES) && (ncols == COLS))
115 	return OK;*/
116 
117 #ifdef	DEBUG
118 	__CTRACE("resizeterm: (%d, %d)\n", nlines, ncols);
119 #endif
120 
121 
122 	for (list = _cursesi_screen->winlistp; list != NULL; list = list->nextp) {
123 		win = list->winp;
124 
125 		newlines = win->reqy;
126 		if (win->begy + newlines >= nlines)
127 			newlines = 0;
128 		if (newlines == 0)
129 			newlines = nlines - win->begy;
130 
131 		newcols = win->reqx;
132 		if (win->begx + newcols >= ncols)
133 			newcols = 0;
134 		if (newcols == 0)
135 			newcols = ncols - win->begx;
136 
137 		if (__resizewin(win, newlines, newcols) != OK)
138 			return ERR;
139 	}
140 
141 	LINES = nlines;
142 	COLS = ncols;
143 
144 	  /* tweak the flags now that we have updated the LINES and COLS */
145 	for (list = _cursesi_screen->winlistp; list != NULL; list = list->nextp) {
146 		if (!(win->flags & __ISPAD))
147 			__swflags(list->winp);
148 	}
149 
150 	wrefresh(curscr);
151 	return OK;
152 }
153 
154 /*
155  * __resizewin --
156  *	Resize the given window.
157  */
158 static int
159 __resizewin(WINDOW *win, int nlines, int ncols)
160 {
161 	__LINE			*lp, *olp, **newlines, *newlspace;
162 	__LDATA			*sp;
163 	__LDATA                 *newwspace;
164 	int			 i, j;
165 	int			 y, x;
166 	WINDOW			*swin;
167 
168 #ifdef	DEBUG
169 	__CTRACE("resize: (%p, %d, %d)\n", win, nlines, ncols);
170 	__CTRACE("resize: win->wattr = %08x\n", win->wattr);
171 	__CTRACE("resize: win->flags = %#.4x\n", win->flags);
172 	__CTRACE("resize: win->maxy = %d\n", win->maxy);
173 	__CTRACE("resize: win->maxx = %d\n", win->maxx);
174 	__CTRACE("resize: win->begy = %d\n", win->begy);
175 	__CTRACE("resize: win->begx = %d\n", win->begx);
176 	__CTRACE("resize: win->scr_t = %d\n", win->scr_t);
177 	__CTRACE("resize: win->scr_b = %d\n", win->scr_b);
178 #endif
179 
180 	if (nlines <= 0 || ncols <= 0)
181 		nlines = ncols = 0;
182 	else {
183 		/* Reallocate line pointer array and line space. */
184 		newlines = realloc(win->lines, nlines * sizeof(__LINE *));
185 		if (newlines == NULL)
186 			return ERR;
187 		win->lines = newlines;
188 
189 		newlspace = realloc(win->lspace, nlines * sizeof(__LINE));
190 		if (newlspace == NULL)
191 			return ERR;
192 		win->lspace = newlspace;
193 	}
194 
195 	/* Don't allocate window and line space if it's a subwindow */
196 	if (win->orig == NULL) {
197 		/*
198 		 * Allocate window space in one chunk.
199 		 */
200 		if (ncols != 0) {
201 			newwspace = realloc(win->wspace,
202 					    ncols * nlines * sizeof(__LDATA));
203 			if (newwspace == NULL)
204 				return ERR;
205 			win->wspace = newwspace;
206 		}
207 
208 		/*
209 		 * Point line pointers to line space, and lines themselves into
210 		 * window space.
211 		 */
212 		for (lp = win->lspace, i = 0; i < nlines; i++, lp++) {
213 			win->lines[i] = lp;
214 			lp->line = &win->wspace[i * ncols];
215 #ifdef DEBUG
216 			lp->sentinel = SENTINEL_VALUE;
217 #endif
218 			lp->firstchp = &lp->firstch;
219 			lp->lastchp = &lp->lastch;
220 			lp->firstch = 0;
221 			lp->lastch = ncols - 1;
222 			lp->flags = __ISDIRTY;
223 		}
224 	} else {
225 
226 		win->ch_off = win->begx - win->orig->begx;
227 		  /* Point line pointers to line space. */
228 		for (lp = win->lspace, i = 0; i < nlines; i++, lp++) {
229 			win->lines[i] = lp;
230 			olp = win->orig->lines[i + win->begy
231 					      - win->orig->begy];
232 			lp->line = &olp->line[win->ch_off];
233 #ifdef DEBUG
234 			lp->sentinel = SENTINEL_VALUE;
235 #endif
236 			lp->firstchp = &olp->firstch;
237 			lp->lastchp = &olp->lastch;
238 			lp->flags = __ISDIRTY;
239 		}
240 	}
241 
242 
243 	win->cury = win->curx = 0;
244 	win->maxy = nlines;
245 	win->maxx = ncols;
246 	win->scr_b = win->maxy - 1;
247 	__swflags(win);
248 
249 	  /*
250 	   * we must zot the window contents otherwise lines may pick
251 	   * up attributes from the previous line when the window is
252 	   * made smaller.  The client will redraw the window anyway
253 	   * so this is no big deal.
254 	   */
255 	for (i = 0; i < win->maxy; i++) {
256 		lp = win->lines[i];
257 		for (sp = lp->line, j = 0; j < win->maxx; j++, sp++) {
258 			sp->ch = ' ';
259 			sp->bch = ' ';
260 			sp->attr = 0;
261 			sp->battr = 0;
262 		}
263 		lp->hash = __hash((char *)(void *)lp->line,
264 				  (size_t) (ncols * __LDATASIZE));
265 	}
266 
267 #ifdef DEBUG
268 	__CTRACE("resize: win->wattr = %08x\n", win->wattr);
269 	__CTRACE("resize: win->flags = %#.4x\n", win->flags);
270 	__CTRACE("resize: win->maxy = %d\n", win->maxy);
271 	__CTRACE("resize: win->maxx = %d\n", win->maxx);
272 	__CTRACE("resize: win->begy = %d\n", win->begy);
273 	__CTRACE("resize: win->begx = %d\n", win->begx);
274 	__CTRACE("resize: win->scr_t = %d\n", win->scr_t);
275 	__CTRACE("resize: win->scr_b = %d\n", win->scr_b);
276 #endif
277 
278 	if (win->orig == NULL) {
279 		/* bound subwindows to new size and fixup their pointers */
280 		for (swin = win->nextp; swin != win; swin = swin->nextp) {
281 			y = swin->reqy;
282 			if (swin->begy + y > win->begy + win->maxy)
283 				y = 0;
284 			if (y <= 0)
285 				y += win->begy + win->maxy - swin->begy;
286 			x = swin->reqx;
287 			if (swin->begx + x > win->begx + win->maxx)
288 				x = 0;
289 			if (x <= 0)
290 				x += win->begy + win->maxx - swin->begx;
291 			__resizewin(swin, y, x);
292 		}
293 	}
294 
295 	return OK;
296 }
297 
298