xref: /minix3/lib/libcurses/add_wchstr.c (revision 51ffecc181005cb45a40108612ee28d1daaeeb86)
1*51ffecc1SBen Gras /*   $NetBSD: add_wchstr.c,v 1.4 2010/02/23 19:48:26 drochner Exp $ */
2*51ffecc1SBen Gras 
3*51ffecc1SBen Gras /*
4*51ffecc1SBen Gras  * Copyright (c) 2005 The NetBSD Foundation Inc.
5*51ffecc1SBen Gras  * All rights reserved.
6*51ffecc1SBen Gras  *
7*51ffecc1SBen Gras  * This code is derived from code donated to the NetBSD Foundation
8*51ffecc1SBen Gras  * by Ruibiao Qiu <ruibiao@arl.wustl.edu,ruibiao@gmail.com>.
9*51ffecc1SBen Gras  *
10*51ffecc1SBen Gras  *
11*51ffecc1SBen Gras  * Redistribution and use in source and binary forms, with or without
12*51ffecc1SBen Gras  * modification, are permitted provided that the following conditions
13*51ffecc1SBen Gras  * are met:
14*51ffecc1SBen Gras  * 1. Redistributions of source code must retain the above copyright
15*51ffecc1SBen Gras  *    notice, this list of conditions and the following disclaimer.
16*51ffecc1SBen Gras  * 2. Redistributions in binary form must reproduce the above copyright
17*51ffecc1SBen Gras  *    notice, this list of conditions and the following disclaimer in the
18*51ffecc1SBen Gras  *    documentation and/or other materials provided with the distribution.
19*51ffecc1SBen Gras  * 3. Neither the name of the NetBSD Foundation nor the names of its
20*51ffecc1SBen Gras  *    contributors may be used to endorse or promote products derived
21*51ffecc1SBen Gras  *    from this software without specific prior written permission.
22*51ffecc1SBen Gras  *
23*51ffecc1SBen Gras  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND
24*51ffecc1SBen Gras  * CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
25*51ffecc1SBen Gras  * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
26*51ffecc1SBen Gras  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
27*51ffecc1SBen Gras  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
28*51ffecc1SBen Gras  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
29*51ffecc1SBen Gras  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
30*51ffecc1SBen Gras  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
31*51ffecc1SBen Gras  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
32*51ffecc1SBen Gras  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
33*51ffecc1SBen Gras  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
34*51ffecc1SBen Gras  * SUCH DAMAGE.
35*51ffecc1SBen Gras  */
36*51ffecc1SBen Gras 
37*51ffecc1SBen Gras #include <sys/cdefs.h>
38*51ffecc1SBen Gras #ifndef lint
39*51ffecc1SBen Gras __RCSID("$NetBSD: add_wchstr.c,v 1.4 2010/02/23 19:48:26 drochner Exp $");
40*51ffecc1SBen Gras #endif				/* not lint */
41*51ffecc1SBen Gras 
42*51ffecc1SBen Gras #include <stdlib.h>
43*51ffecc1SBen Gras 
44*51ffecc1SBen Gras #include "curses.h"
45*51ffecc1SBen Gras #include "curses_private.h"
46*51ffecc1SBen Gras 
47*51ffecc1SBen Gras /*
48*51ffecc1SBen Gras  * add_wchstr --
49*51ffecc1SBen Gras  *  Add a wide string to stdscr starting at (_cury, _curx).
50*51ffecc1SBen Gras  */
51*51ffecc1SBen Gras int
add_wchstr(const cchar_t * wchstr)52*51ffecc1SBen Gras add_wchstr(const cchar_t *wchstr)
53*51ffecc1SBen Gras {
54*51ffecc1SBen Gras #ifndef HAVE_WCHAR
55*51ffecc1SBen Gras 	return ERR;
56*51ffecc1SBen Gras #else
57*51ffecc1SBen Gras 	return wadd_wchnstr(stdscr, wchstr, -1);
58*51ffecc1SBen Gras #endif
59*51ffecc1SBen Gras }
60*51ffecc1SBen Gras 
61*51ffecc1SBen Gras 
62*51ffecc1SBen Gras /*
63*51ffecc1SBen Gras  * wadd_wchstr --
64*51ffecc1SBen Gras  *      Add a string to the given window starting at (_cury, _curx).
65*51ffecc1SBen Gras  */
66*51ffecc1SBen Gras int
wadd_wchstr(WINDOW * win,const cchar_t * wchstr)67*51ffecc1SBen Gras wadd_wchstr(WINDOW *win, const cchar_t *wchstr)
68*51ffecc1SBen Gras {
69*51ffecc1SBen Gras #ifndef HAVE_WCHAR
70*51ffecc1SBen Gras 	return ERR;
71*51ffecc1SBen Gras #else
72*51ffecc1SBen Gras 	return wadd_wchnstr(win, wchstr, -1);
73*51ffecc1SBen Gras #endif
74*51ffecc1SBen Gras }
75*51ffecc1SBen Gras 
76*51ffecc1SBen Gras 
77*51ffecc1SBen Gras /*
78*51ffecc1SBen Gras  * add_wchnstr --
79*51ffecc1SBen Gras  *      Add a string (at most n characters) to stdscr starting
80*51ffecc1SBen Gras  *	at (_cury, _curx).  If n is negative, add the entire string.
81*51ffecc1SBen Gras  */
82*51ffecc1SBen Gras int
add_wchnstr(const cchar_t * wchstr,int n)83*51ffecc1SBen Gras add_wchnstr(const cchar_t *wchstr, int n)
84*51ffecc1SBen Gras {
85*51ffecc1SBen Gras #ifndef HAVE_WCHAR
86*51ffecc1SBen Gras 	return ERR;
87*51ffecc1SBen Gras #else
88*51ffecc1SBen Gras 	return wadd_wchnstr(stdscr, wchstr, n);
89*51ffecc1SBen Gras #endif
90*51ffecc1SBen Gras }
91*51ffecc1SBen Gras 
92*51ffecc1SBen Gras 
93*51ffecc1SBen Gras /*
94*51ffecc1SBen Gras  * mvadd_wchstr --
95*51ffecc1SBen Gras  *      Add a string to stdscr starting at (y, x)
96*51ffecc1SBen Gras  */
97*51ffecc1SBen Gras int
mvadd_wchstr(int y,int x,const cchar_t * wchstr)98*51ffecc1SBen Gras mvadd_wchstr(int y, int x, const cchar_t *wchstr)
99*51ffecc1SBen Gras {
100*51ffecc1SBen Gras #ifndef HAVE_WCHAR
101*51ffecc1SBen Gras 	return ERR;
102*51ffecc1SBen Gras #else
103*51ffecc1SBen Gras 	return mvwadd_wchnstr(stdscr, y, x, wchstr, -1);
104*51ffecc1SBen Gras #endif
105*51ffecc1SBen Gras }
106*51ffecc1SBen Gras 
107*51ffecc1SBen Gras 
108*51ffecc1SBen Gras /*
109*51ffecc1SBen Gras  * mvwadd_wchstr --
110*51ffecc1SBen Gras  *      Add a string to the given window starting at (y, x)
111*51ffecc1SBen Gras  */
112*51ffecc1SBen Gras int
mvwadd_wchstr(WINDOW * win,int y,int x,const cchar_t * wchstr)113*51ffecc1SBen Gras mvwadd_wchstr(WINDOW *win, int y, int x, const cchar_t *wchstr)
114*51ffecc1SBen Gras {
115*51ffecc1SBen Gras #ifndef HAVE_WCHAR
116*51ffecc1SBen Gras 	return ERR;
117*51ffecc1SBen Gras #else
118*51ffecc1SBen Gras 	return mvwadd_wchnstr(win, y, x, wchstr, -1);
119*51ffecc1SBen Gras #endif
120*51ffecc1SBen Gras }
121*51ffecc1SBen Gras 
122*51ffecc1SBen Gras 
123*51ffecc1SBen Gras /*
124*51ffecc1SBen Gras  * mvadd_wchnstr --
125*51ffecc1SBen Gras  *      Add a string of at most n characters to stdscr
126*51ffecc1SBen Gras  *      starting at (y, x).
127*51ffecc1SBen Gras  */
128*51ffecc1SBen Gras int
mvadd_wchnstr(int y,int x,const cchar_t * wchstr,int n)129*51ffecc1SBen Gras mvadd_wchnstr(int y, int x, const cchar_t *wchstr, int n)
130*51ffecc1SBen Gras {
131*51ffecc1SBen Gras #ifndef HAVE_WCHAR
132*51ffecc1SBen Gras 	return ERR;
133*51ffecc1SBen Gras #else
134*51ffecc1SBen Gras 	return mvwadd_wchnstr(stdscr, y, x, wchstr, n);
135*51ffecc1SBen Gras #endif
136*51ffecc1SBen Gras }
137*51ffecc1SBen Gras 
138*51ffecc1SBen Gras 
139*51ffecc1SBen Gras /*
140*51ffecc1SBen Gras  * mvwadd_wchnstr --
141*51ffecc1SBen Gras  *      Add a string of at most n characters to the given window
142*51ffecc1SBen Gras  *      starting at (y, x).
143*51ffecc1SBen Gras  */
144*51ffecc1SBen Gras int
mvwadd_wchnstr(WINDOW * win,int y,int x,const cchar_t * wchstr,int n)145*51ffecc1SBen Gras mvwadd_wchnstr(WINDOW *win, int y, int x, const cchar_t *wchstr, int n)
146*51ffecc1SBen Gras {
147*51ffecc1SBen Gras #ifndef HAVE_WCHAR
148*51ffecc1SBen Gras 	return ERR;
149*51ffecc1SBen Gras #else
150*51ffecc1SBen Gras 	if (wmove(win, y, x) == ERR)
151*51ffecc1SBen Gras 		return ERR;
152*51ffecc1SBen Gras 
153*51ffecc1SBen Gras 	return wadd_wchnstr(win, wchstr, n);
154*51ffecc1SBen Gras #endif
155*51ffecc1SBen Gras }
156*51ffecc1SBen Gras 
157*51ffecc1SBen Gras 
158*51ffecc1SBen Gras /*
159*51ffecc1SBen Gras  * wadd_wchnstr --
160*51ffecc1SBen Gras  *	Add a string (at most n wide characters) to the given window
161*51ffecc1SBen Gras  *	starting at (_cury, _curx).  If n is -1, add the entire string.
162*51ffecc1SBen Gras  */
163*51ffecc1SBen Gras int
wadd_wchnstr(WINDOW * win,const cchar_t * wchstr,int n)164*51ffecc1SBen Gras wadd_wchnstr(WINDOW *win, const cchar_t *wchstr, int n)
165*51ffecc1SBen Gras {
166*51ffecc1SBen Gras #ifndef HAVE_WCHAR
167*51ffecc1SBen Gras 	return ERR;
168*51ffecc1SBen Gras #else
169*51ffecc1SBen Gras 	const cchar_t *chp;
170*51ffecc1SBen Gras 	wchar_t wc;
171*51ffecc1SBen Gras 	int cw, x, y, sx, ex, newx, i, cnt;
172*51ffecc1SBen Gras 	__LDATA *lp, *tp;
173*51ffecc1SBen Gras 	nschar_t *np, *tnp;
174*51ffecc1SBen Gras 	__LINE *lnp;
175*51ffecc1SBen Gras 
176*51ffecc1SBen Gras #ifdef DEBUG
177*51ffecc1SBen Gras 	__CTRACE(__CTRACE_INPUT,
178*51ffecc1SBen Gras 	    "wadd_wchnstr: win = %p, wchstr = %p, n = %d\n", win, wchstr, n);
179*51ffecc1SBen Gras #endif
180*51ffecc1SBen Gras 
181*51ffecc1SBen Gras 	if (!wchstr)
182*51ffecc1SBen Gras 		return OK;
183*51ffecc1SBen Gras 
184*51ffecc1SBen Gras 	/* compute length of the cchar string */
185*51ffecc1SBen Gras 	if (n < -1)
186*51ffecc1SBen Gras 		return ERR;
187*51ffecc1SBen Gras 	if (n >= 0)
188*51ffecc1SBen Gras 		for (chp = wchstr, cnt = 0; n && chp->vals[0];
189*51ffecc1SBen Gras 			n--, chp++, ++cnt);
190*51ffecc1SBen Gras 	else
191*51ffecc1SBen Gras 		for (chp = wchstr, cnt = 0; chp->vals[0]; chp++, ++cnt);
192*51ffecc1SBen Gras #ifdef DEBUG
193*51ffecc1SBen Gras 	__CTRACE(__CTRACE_INPUT, "wadd_wchnstr: len=%d\n", cnt);
194*51ffecc1SBen Gras #endif /* DEBUG */
195*51ffecc1SBen Gras 	chp = wchstr;
196*51ffecc1SBen Gras 	x = win->curx;
197*51ffecc1SBen Gras 	y = win->cury;
198*51ffecc1SBen Gras 	lp = &win->alines[y]->line[x];
199*51ffecc1SBen Gras 	lnp = win->alines[y];
200*51ffecc1SBen Gras 
201*51ffecc1SBen Gras 	cw = WCOL(*lp);
202*51ffecc1SBen Gras 	if (cw >= 0) {
203*51ffecc1SBen Gras 		sx = x;
204*51ffecc1SBen Gras 	} else {
205*51ffecc1SBen Gras 		if (wcwidth(chp->vals[0])) {
206*51ffecc1SBen Gras 			/* clear the partial character before cursor */
207*51ffecc1SBen Gras 			for (tp = lp + cw; tp < lp; tp++) {
208*51ffecc1SBen Gras 				tp->ch = (wchar_t) btowc((int) win->bch);
209*51ffecc1SBen Gras 				if (_cursesi_copy_nsp(win->bnsp, tp) == ERR)
210*51ffecc1SBen Gras 					return ERR;
211*51ffecc1SBen Gras 				tp->attr = win->battr;
212*51ffecc1SBen Gras 				SET_WCOL(*tp, 1);
213*51ffecc1SBen Gras 				np = tp->nsp;
214*51ffecc1SBen Gras 			}
215*51ffecc1SBen Gras 		} else {
216*51ffecc1SBen Gras 			/* move to the start of current char */
217*51ffecc1SBen Gras 			lp += cw;
218*51ffecc1SBen Gras 			x += cw;
219*51ffecc1SBen Gras 		}
220*51ffecc1SBen Gras 		sx = x + cw;
221*51ffecc1SBen Gras 	}
222*51ffecc1SBen Gras 	lnp->flags |= __ISDIRTY;
223*51ffecc1SBen Gras 	newx = sx + win->ch_off;
224*51ffecc1SBen Gras 	if (newx < *lnp->firstchp)
225*51ffecc1SBen Gras 		*lnp->firstchp = newx;
226*51ffecc1SBen Gras 
227*51ffecc1SBen Gras 	/* add characters in the string */
228*51ffecc1SBen Gras 	ex = x;
229*51ffecc1SBen Gras 	while (cnt) {
230*51ffecc1SBen Gras 		x = ex;
231*51ffecc1SBen Gras 		wc = chp->vals[0];
232*51ffecc1SBen Gras #ifdef DEBUG
233*51ffecc1SBen Gras 		__CTRACE(__CTRACE_INPUT, "wadd_wchnstr: adding %x", wc);
234*51ffecc1SBen Gras #endif /* DEBUG */
235*51ffecc1SBen Gras 		cw = wcwidth(wc);
236*51ffecc1SBen Gras 		if (cw < 0)
237*51ffecc1SBen Gras 			cw = 1;
238*51ffecc1SBen Gras 		if (cw) {
239*51ffecc1SBen Gras 			/* spacing character */
240*51ffecc1SBen Gras #ifdef DEBUG
241*51ffecc1SBen Gras 			__CTRACE(__CTRACE_INPUT,
242*51ffecc1SBen Gras 			    " as a spacing char(width=%d)\n", cw);
243*51ffecc1SBen Gras #endif /* DEBUG */
244*51ffecc1SBen Gras 			if (cw > win->maxx - ex) {
245*51ffecc1SBen Gras 				/* clear to EOL */
246*51ffecc1SBen Gras 				while (ex < win->maxx) {
247*51ffecc1SBen Gras 					lp->ch = (wchar_t)
248*51ffecc1SBen Gras 						btowc((int) win->bch);
249*51ffecc1SBen Gras 					if (_cursesi_copy_nsp(win->bnsp, lp)
250*51ffecc1SBen Gras 					    == ERR)
251*51ffecc1SBen Gras 						return ERR;
252*51ffecc1SBen Gras 					lp->attr = win->battr;
253*51ffecc1SBen Gras 					SET_WCOL(*lp, 1);
254*51ffecc1SBen Gras 					lp++, ex++;
255*51ffecc1SBen Gras 				}
256*51ffecc1SBen Gras 				ex = win->maxx - 1;
257*51ffecc1SBen Gras 				break;
258*51ffecc1SBen Gras 			}
259*51ffecc1SBen Gras 			/* this could combine with the insertion of
260*51ffecc1SBen Gras 			 * non-spacing char */
261*51ffecc1SBen Gras 			np = lp->nsp;
262*51ffecc1SBen Gras 			if (np) {
263*51ffecc1SBen Gras 				while (np) {
264*51ffecc1SBen Gras 					tnp = np->next;
265*51ffecc1SBen Gras 					free(np);
266*51ffecc1SBen Gras 					np = tnp;
267*51ffecc1SBen Gras 				}
268*51ffecc1SBen Gras 				lp->nsp = NULL;
269*51ffecc1SBen Gras 			}
270*51ffecc1SBen Gras 			lp->ch = chp->vals[0];
271*51ffecc1SBen Gras 			lp->attr = chp->attributes & WA_ATTRIBUTES;
272*51ffecc1SBen Gras 			SET_WCOL(*lp, cw);
273*51ffecc1SBen Gras 			if (chp->elements > 1) {
274*51ffecc1SBen Gras 				for (i = 1; i < chp->elements; i++) {
275*51ffecc1SBen Gras 					np = (nschar_t *)
276*51ffecc1SBen Gras 						malloc(sizeof(nschar_t));
277*51ffecc1SBen Gras 					if (!np)
278*51ffecc1SBen Gras 						return ERR;
279*51ffecc1SBen Gras 					np->ch = chp->vals[i];
280*51ffecc1SBen Gras 					np->next = lp->nsp;
281*51ffecc1SBen Gras 					lp->nsp = np;
282*51ffecc1SBen Gras 				}
283*51ffecc1SBen Gras 			}
284*51ffecc1SBen Gras 			lp++, ex++;
285*51ffecc1SBen Gras #ifdef DEBUG
286*51ffecc1SBen Gras 			__CTRACE(__CTRACE_INPUT,
287*51ffecc1SBen Gras 				"wadd_wchnstr: ex = %d, x = %d, cw = %d\n",
288*51ffecc1SBen Gras 				 ex, x, cw);
289*51ffecc1SBen Gras #endif /* DEBUG */
290*51ffecc1SBen Gras 			while (ex - x <= cw - 1) {
291*51ffecc1SBen Gras 				np = lp->nsp;
292*51ffecc1SBen Gras 				if (np) {
293*51ffecc1SBen Gras 					while (np) {
294*51ffecc1SBen Gras 						tnp = np->next;
295*51ffecc1SBen Gras 						free(np);
296*51ffecc1SBen Gras 						np = tnp;
297*51ffecc1SBen Gras 					}
298*51ffecc1SBen Gras 					lp->nsp = NULL;
299*51ffecc1SBen Gras 				}
300*51ffecc1SBen Gras 				lp->ch = chp->vals[0];
301*51ffecc1SBen Gras 				lp->attr = chp->attributes & WA_ATTRIBUTES;
302*51ffecc1SBen Gras 				SET_WCOL(*lp, x - ex);
303*51ffecc1SBen Gras 				lp++, ex++;
304*51ffecc1SBen Gras 			}
305*51ffecc1SBen Gras 		} else {
306*51ffecc1SBen Gras 			/* non-spacing character */
307*51ffecc1SBen Gras #ifdef DEBUG
308*51ffecc1SBen Gras 			__CTRACE(__CTRACE_INPUT,
309*51ffecc1SBen Gras 				"wadd_wchnstr: as non-spacing char");
310*51ffecc1SBen Gras #endif /* DEBUG */
311*51ffecc1SBen Gras 			for (i = 0; i < chp->elements; i++) {
312*51ffecc1SBen Gras 				np = (nschar_t *)malloc(sizeof(nschar_t));
313*51ffecc1SBen Gras 				if (!np)
314*51ffecc1SBen Gras 					return ERR;
315*51ffecc1SBen Gras 				np->ch = chp->vals[i];
316*51ffecc1SBen Gras 				np->next = lp->nsp;
317*51ffecc1SBen Gras 				lp->nsp = np;
318*51ffecc1SBen Gras 			}
319*51ffecc1SBen Gras 		}
320*51ffecc1SBen Gras 		cnt--, chp++;
321*51ffecc1SBen Gras 	}
322*51ffecc1SBen Gras #ifdef DEBUG
323*51ffecc1SBen Gras 	for (i = sx; i < ex; i++) {
324*51ffecc1SBen Gras 		__CTRACE(__CTRACE_INPUT, "wadd_wchnstr: (%d,%d)=(%x,%x,%p)\n",
325*51ffecc1SBen Gras 		    win->cury, i, win->alines[win->cury]->line[i].ch,
326*51ffecc1SBen Gras 		    win->alines[win->cury]->line[i].attr,
327*51ffecc1SBen Gras 		    win->alines[win->cury]->line[i].nsp);
328*51ffecc1SBen Gras 	}
329*51ffecc1SBen Gras #endif /* DEBUG */
330*51ffecc1SBen Gras 	lnp->flags |= __ISDIRTY;
331*51ffecc1SBen Gras 	newx = ex + win->ch_off;
332*51ffecc1SBen Gras 	if (newx > *lnp->lastchp)
333*51ffecc1SBen Gras 		*lnp->lastchp = newx;
334*51ffecc1SBen Gras 	__touchline(win, y, sx, ex);
335*51ffecc1SBen Gras 
336*51ffecc1SBen Gras 	return OK;
337*51ffecc1SBen Gras #endif /* HAVE_WCHAR */
338*51ffecc1SBen Gras }
339