Home | History | Annotate | Line # | Download | only in libcurses
addbytes.c revision 1.32.4.1
      1  1.32.4.1     matt /*	$NetBSD: addbytes.c,v 1.32.4.1 2008/01/09 01:36:22 matt Exp $	*/
      2      1.11    mikel 
      3       1.1      cgd /*
      4      1.10      cgd  * Copyright (c) 1987, 1993, 1994
      5       1.7      cgd  *	The Regents of the University of California.  All rights reserved.
      6       1.1      cgd  *
      7       1.1      cgd  * Redistribution and use in source and binary forms, with or without
      8       1.1      cgd  * modification, are permitted provided that the following conditions
      9       1.1      cgd  * are met:
     10       1.1      cgd  * 1. Redistributions of source code must retain the above copyright
     11       1.1      cgd  *    notice, this list of conditions and the following disclaimer.
     12       1.1      cgd  * 2. Redistributions in binary form must reproduce the above copyright
     13       1.1      cgd  *    notice, this list of conditions and the following disclaimer in the
     14       1.1      cgd  *    documentation and/or other materials provided with the distribution.
     15      1.28      agc  * 3. Neither the name of the University nor the names of its contributors
     16       1.1      cgd  *    may be used to endorse or promote products derived from this software
     17       1.1      cgd  *    without specific prior written permission.
     18       1.1      cgd  *
     19       1.1      cgd  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
     20       1.1      cgd  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     21       1.1      cgd  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     22       1.1      cgd  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
     23       1.1      cgd  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     24       1.1      cgd  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     25       1.1      cgd  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     26       1.1      cgd  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     27       1.1      cgd  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     28       1.1      cgd  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     29       1.1      cgd  * SUCH DAMAGE.
     30       1.1      cgd  */
     31       1.1      cgd 
     32      1.11    mikel #include <sys/cdefs.h>
     33       1.1      cgd #ifndef lint
     34      1.11    mikel #if 0
     35      1.10      cgd static char sccsid[] = "@(#)addbytes.c	8.4 (Berkeley) 5/4/94";
     36      1.11    mikel #else
     37  1.32.4.1     matt __RCSID("$NetBSD: addbytes.c,v 1.32.4.1 2008/01/09 01:36:22 matt Exp $");
     38      1.11    mikel #endif
     39      1.13      mrg #endif				/* not lint */
     40       1.1      cgd 
     41      1.32    blymn #include <stdlib.h>
     42      1.10      cgd #include "curses.h"
     43      1.15    blymn #include "curses_private.h"
     44      1.23    blymn #ifdef DEBUG
     45      1.23    blymn #include <assert.h>
     46      1.23    blymn #endif
     47       1.1      cgd 
     48       1.7      cgd #define	SYNCH_IN	{y = win->cury; x = win->curx;}
     49       1.7      cgd #define	SYNCH_OUT	{win->cury = y; win->curx = x;}
     50      1.32    blymn #define	PSYNCH_IN	{*y = win->cury; *x = win->curx;}
     51      1.32    blymn #define	PSYNCH_OUT	{win->cury = *y; win->curx = *x;}
     52       1.3      alm 
     53      1.17    blymn #ifndef _CURSES_USE_MACROS
     54      1.17    blymn 
     55      1.17    blymn /*
     56      1.17    blymn  * addbytes --
     57      1.17    blymn  *      Add the character to the current position in stdscr.
     58      1.17    blymn  */
     59      1.17    blymn int
     60      1.17    blymn addbytes(const char *bytes, int count)
     61      1.17    blymn {
     62      1.17    blymn 	return __waddbytes(stdscr, bytes, count, 0);
     63      1.17    blymn }
     64      1.17    blymn 
     65      1.17    blymn /*
     66      1.17    blymn  * waddbytes --
     67      1.17    blymn  *      Add the character to the current position in the given window.
     68      1.17    blymn  */
     69      1.17    blymn int
     70      1.17    blymn waddbytes(WINDOW *win, const char *bytes, int count)
     71      1.17    blymn {
     72      1.17    blymn 	return __waddbytes(win, bytes, count, 0);
     73      1.17    blymn }
     74      1.17    blymn 
     75      1.17    blymn /*
     76      1.17    blymn  * mvaddbytes --
     77      1.17    blymn  *      Add the characters to stdscr at the location given.
     78      1.17    blymn  */
     79      1.17    blymn int
     80      1.17    blymn mvaddbytes(int y, int x, const char *bytes, int count)
     81      1.17    blymn {
     82      1.17    blymn 	return mvwaddbytes(stdscr, y, x, bytes, count);
     83      1.17    blymn }
     84      1.17    blymn 
     85      1.17    blymn /*
     86      1.17    blymn  * mvwaddbytes --
     87      1.17    blymn  *      Add the characters to the given window at the location given.
     88      1.17    blymn  */
     89      1.17    blymn int
     90      1.17    blymn mvwaddbytes(WINDOW *win, int y, int x, const char *bytes, int count)
     91      1.17    blymn {
     92      1.17    blymn 	if (wmove(win, y, x) == ERR)
     93      1.17    blymn 		return ERR;
     94      1.17    blymn 
     95      1.17    blymn 	return __waddbytes(win, bytes, count, 0);
     96      1.17    blymn }
     97      1.17    blymn 
     98      1.17    blymn #endif
     99      1.17    blymn 
    100       1.1      cgd /*
    101       1.5  mycroft  * waddbytes --
    102       1.5  mycroft  *	Add the character to the current position in the given window.
    103       1.1      cgd  */
    104       1.5  mycroft int
    105      1.17    blymn __waddbytes(WINDOW *win, const char *bytes, int count, attr_t attr)
    106       1.1      cgd {
    107      1.32    blymn 	int		 x, y, err;
    108      1.15    blymn 	__LINE		*lp;
    109      1.32    blymn #ifdef HAVE_WCHAR
    110      1.32    blymn 	int		n;
    111      1.32    blymn 	cchar_t		cc;
    112      1.32    blymn 	wchar_t		wc;
    113      1.32    blymn #else
    114      1.32    blymn 	int		c;
    115      1.32    blymn #endif
    116      1.23    blymn #ifdef DEBUG
    117      1.23    blymn 	int             i;
    118       1.5  mycroft 
    119      1.23    blymn 	for (i = 0; i < win->maxy; i++) {
    120      1.23    blymn 		assert(win->lines[i]->sentinel == SENTINEL_VALUE);
    121      1.23    blymn 	}
    122      1.32    blymn 
    123      1.32    blymn 	__CTRACE(__CTRACE_INPUT, "ADDBYTES: add %d bytes\n", count);
    124      1.23    blymn #endif
    125      1.32    blymn 
    126      1.32    blymn 	err = OK;
    127       1.5  mycroft 	SYNCH_IN;
    128      1.23    blymn 	lp = win->lines[y];
    129       1.1      cgd 
    130      1.32    blymn 	while (count > 0) {
    131      1.32    blymn #ifndef HAVE_WCHAR
    132      1.11    mikel 		c = *bytes++;
    133       1.5  mycroft #ifdef DEBUG
    134      1.31      jdc 		__CTRACE(__CTRACE_INPUT, "ADDBYTES('%c', %x) at (%d, %d)\n",
    135      1.31      jdc 		    c, attr, y, x);
    136       1.5  mycroft #endif
    137      1.32    blymn 		err = _cursesi_addbyte(win, &lp, &y, &x, c, attr);
    138      1.32    blymn 		count--;
    139      1.32    blymn #else
    140      1.32    blymn 		/*
    141      1.32    blymn 		 * For wide character support only, try and convert the
    142      1.32    blymn 		 * given string into a wide character - we do this because
    143      1.32    blymn 		 * this is how ncurses behaves (not that I think this is
    144      1.32    blymn 		 * actually the correct thing to do but if we don't do it
    145      1.32    blymn 		 * a lot of things that rely on this behaviour will break
    146      1.32    blymn 		 * and we will be blamed).  If the conversion succeeds
    147      1.32    blymn 		 * then we eat the n characters used to make the wide char
    148      1.32    blymn 		 * from the string.
    149      1.32    blymn 		 */
    150      1.32    blymn 		n = mbtowc(&wc, bytes, count);
    151      1.32    blymn 		if (n == 0)
    152       1.5  mycroft 			break;
    153      1.32    blymn 		else if (n < 0) { /* not a valid conversion just eat a char */
    154      1.32    blymn 			wc = *bytes;
    155      1.32    blymn 			n = 1;
    156      1.32    blymn 		}
    157      1.32    blymn 
    158      1.32    blymn 
    159      1.32    blymn #ifdef DEBUG
    160      1.32    blymn 	__CTRACE(__CTRACE_INPUT,
    161      1.32    blymn 		 "ADDBYTES WIDE(0x%x [%s], %x) at (%d, %d), ate %d bytes\n",
    162      1.32    blymn 		 (unsigned) wc, unctrl((unsigned) wc), attr, y, x, n);
    163      1.32    blymn #endif
    164      1.32    blymn 		cc.vals[0] = wc;
    165      1.32    blymn 		cc.elements = 1;
    166      1.32    blymn 		cc.attributes = attr;
    167      1.32    blymn 		err = _cursesi_addwchar(win, &lp, &y, &x, &cc);
    168      1.32    blymn 		bytes += n;
    169      1.32    blymn 		count -= n;
    170      1.32    blymn #endif
    171      1.32    blymn 	}
    172      1.32    blymn 
    173      1.32    blymn 	SYNCH_OUT;
    174      1.32    blymn 
    175      1.32    blymn #ifdef DEBUG
    176      1.32    blymn 	for (i = 0; i < win->maxy; i++) {
    177      1.32    blymn 		assert(win->lines[i]->sentinel == SENTINEL_VALUE);
    178      1.32    blymn 	}
    179      1.32    blymn #endif
    180      1.32    blymn 
    181      1.32    blymn 	return (err);
    182      1.32    blymn }
    183       1.5  mycroft 
    184      1.32    blymn /*
    185      1.32    blymn  * _cursesi_addbyte -
    186      1.32    blymn  *	Internal function to add a byte and update the row and column
    187      1.32    blymn  * positions as appropriate.  This function is only used in the narrow
    188      1.32    blymn  * character version of curses.
    189      1.32    blymn  */
    190      1.32    blymn int
    191      1.32    blymn _cursesi_addbyte(WINDOW *win, __LINE **lp, int *y, int *x, int c,
    192      1.32    blymn 		 attr_t attr)
    193      1.32    blymn {
    194      1.32    blymn 	static char	 blanks[] = "        ";
    195      1.32    blymn 	int		 newx;
    196      1.32    blymn 	attr_t		 attributes;
    197      1.32    blymn 
    198      1.32    blymn 	switch (c) {
    199      1.32    blymn 	case '\t':
    200      1.32    blymn 		PSYNCH_OUT;
    201      1.32    blymn 		if (waddbytes(win, blanks, 8 - (*x % 8)) == ERR)
    202      1.32    blymn 			return (ERR);
    203      1.32    blymn 		PSYNCH_IN;
    204      1.32    blymn 		break;
    205      1.32    blymn 
    206      1.32    blymn 	default:
    207       1.5  mycroft #ifdef DEBUG
    208      1.32    blymn 		__CTRACE(__CTRACE_INPUT, "ADDBYTES(%p, %d, %d)\n",
    209      1.32    blymn 			 win, *y, *x);
    210       1.5  mycroft #endif
    211      1.14   simonb 
    212      1.32    blymn 		if ((*lp)->flags & __ISPASTEOL) {
    213      1.32    blymn 		  newline:
    214      1.32    blymn 			*x = 0;
    215      1.32    blymn 			(*lp)->flags &= ~__ISPASTEOL;
    216      1.32    blymn 			if (*y == win->scr_b) {
    217      1.32    blymn #ifdef DEBUG
    218      1.32    blymn 				__CTRACE(__CTRACE_INPUT,
    219      1.32    blymn 					 "ADDBYTES - on bottom "
    220      1.32    blymn 					 "of scrolling region\n");
    221      1.32    blymn #endif
    222      1.32    blymn 				if (!(win->flags & __SCROLLOK))
    223      1.32    blymn 					return ERR;
    224      1.32    blymn 				PSYNCH_OUT;
    225      1.32    blymn 				scroll(win);
    226      1.32    blymn 				PSYNCH_IN;
    227      1.32    blymn 			} else {
    228      1.32    blymn 				(*y)++;
    229       1.7      cgd 			}
    230      1.32    blymn 			*lp = win->lines[*y];
    231      1.32    blymn 			if (c == '\n')
    232      1.32    blymn 				break;
    233      1.32    blymn 		}
    234      1.14   simonb 
    235      1.32    blymn 		attributes = (win->wattr | attr) &
    236      1.32    blymn 			(__ATTRIBUTES & ~__COLOR);
    237      1.32    blymn 		if (attr & __COLOR)
    238      1.32    blymn 			attributes |= attr & __COLOR;
    239      1.32    blymn 		else if (win->wattr & __COLOR)
    240      1.32    blymn 			attributes |= win->wattr & __COLOR;
    241      1.32    blymn #ifdef DEBUG
    242      1.32    blymn 		__CTRACE(__CTRACE_INPUT,
    243      1.32    blymn 			 "ADDBYTES: 1: y = %d, x = %d, firstch = %d, "
    244      1.32    blymn 			 "lastch = %d\n",
    245      1.32    blymn 			 *y, *x, *win->lines[*y]->firstchp,
    246      1.32    blymn 			 *win->lines[*y]->lastchp);
    247      1.32    blymn #endif
    248      1.32    blymn 		/*
    249      1.32    blymn 		 * Always update the change pointers.  Otherwise,
    250      1.32    blymn 		 * we could end up not displaying 'blank' characters
    251      1.32    blymn 		 * when overlapping windows are displayed.
    252      1.32    blymn 		 */
    253      1.32    blymn 		newx = *x + win->ch_off;
    254      1.32    blymn 		(*lp)->flags |= __ISDIRTY;
    255      1.32    blymn 		/*
    256      1.32    blymn 		 * firstchp/lastchp are shared between
    257      1.32    blymn 		 * parent window and sub-window.
    258      1.32    blymn 		 */
    259      1.32    blymn 		if (newx < *(*lp)->firstchp)
    260      1.32    blymn 			*(*lp)->firstchp = newx;
    261      1.32    blymn 		if (newx > *(*lp)->lastchp)
    262      1.32    blymn 			*(*lp)->lastchp = newx;
    263      1.32    blymn #ifdef DEBUG
    264      1.32    blymn 		__CTRACE(__CTRACE_INPUT,
    265      1.32    blymn 			 "ADDBYTES: change gives f/l: %d/%d [%d/%d]\n",
    266      1.32    blymn 			 *(*lp)->firstchp, *(*lp)->lastchp,
    267      1.32    blymn 			 *(*lp)->firstchp - win->ch_off,
    268      1.32    blymn 			 *(*lp)->lastchp - win->ch_off);
    269      1.32    blymn #endif
    270      1.32    blymn 		if (win->bch != ' ' && c == ' ')
    271      1.32    blymn 			(*lp)->line[*x].ch = win->bch;
    272      1.32    blymn 		else
    273      1.32    blymn 			(*lp)->line[*x].ch = c;
    274      1.32    blymn 
    275      1.32    blymn 		if (attributes & __COLOR)
    276      1.32    blymn 			(*lp)->line[*x].attr =
    277      1.32    blymn 				attributes | (win->battr & ~__COLOR);
    278      1.32    blymn 		else
    279      1.32    blymn 			(*lp)->line[*x].attr = attributes | win->battr;
    280      1.32    blymn 
    281      1.32    blymn 		if (*x == win->maxx - 1)
    282      1.32    blymn 			(*lp)->flags |= __ISPASTEOL;
    283      1.32    blymn 		else
    284      1.32    blymn 			(*x)++;
    285      1.32    blymn #ifdef DEBUG
    286      1.32    blymn 		__CTRACE(__CTRACE_INPUT,
    287      1.32    blymn 			 "ADDBYTES: 2: y = %d, x = %d, firstch = %d, "
    288      1.32    blymn 			 "lastch = %d\n",
    289      1.32    blymn 			 *y, *x, *win->lines[*y]->firstchp,
    290      1.32    blymn 			 *win->lines[*y]->lastchp);
    291      1.32    blymn #endif
    292      1.32    blymn 		break;
    293      1.32    blymn 	case '\n':
    294      1.32    blymn 		PSYNCH_OUT;
    295      1.32    blymn 		wclrtoeol(win);
    296      1.32    blymn 		PSYNCH_IN;
    297      1.32    blymn 		goto newline;
    298      1.32    blymn 	case '\r':
    299      1.32    blymn 		*x = 0;
    300      1.32    blymn 		break;
    301      1.32    blymn 	case '\b':
    302      1.32    blymn 		if (--(*x) < 0)
    303      1.32    blymn 			*x = 0;
    304      1.32    blymn 		break;
    305      1.32    blymn 	}
    306      1.32    blymn 
    307      1.32    blymn 	return (OK);
    308      1.32    blymn }
    309      1.32    blymn 
    310      1.32    blymn /*
    311      1.32    blymn  * _cursesi_addwchar -
    312      1.32    blymn  *	Internal function to add a wide character and update the row
    313      1.32    blymn  * and column positions.
    314      1.32    blymn  */
    315      1.32    blymn int
    316      1.32    blymn _cursesi_addwchar(WINDOW *win, __LINE **lnp, int *y, int *x,
    317      1.32    blymn 		  const cchar_t *wch)
    318      1.32    blymn {
    319      1.32    blymn #ifndef HAVE_WCHAR
    320      1.32    blymn 	return (ERR);
    321      1.32    blymn #else
    322      1.32    blymn 	int sx = 0, ex = 0, cw = 0, i = 0, newx = 0;
    323      1.32    blymn 	__LDATA *lp = &win->lines[*y]->line[*x], *tp = NULL;
    324      1.32    blymn 	nschar_t *np = NULL;
    325      1.32    blymn 	cchar_t cc;
    326      1.32    blymn 	attr_t attributes;
    327      1.32    blymn 
    328      1.32    blymn 	/* special characters handling */
    329      1.32    blymn 	switch (wch->vals[0]) {
    330      1.32    blymn 	case L'\b':
    331      1.32    blymn 		if (--*x < 0)
    332      1.32    blymn 			*x = 0;
    333      1.32    blymn 		win->curx = *x;
    334      1.32    blymn 		return OK;
    335      1.32    blymn 	case L'\r':
    336  1.32.4.1     matt 		*x = 0;
    337      1.32    blymn 		return OK;
    338      1.32    blymn 	case L'\n':
    339      1.32    blymn 		wclrtoeol(win);
    340      1.32    blymn 		PSYNCH_IN;
    341      1.32    blymn 		*x = 0;
    342      1.32    blymn 		(*lnp)->flags &= ~__ISPASTEOL;
    343      1.32    blymn 		if (*y == win->scr_b) {
    344      1.32    blymn 			if (!(win->flags & __SCROLLOK))
    345      1.32    blymn 				return ERR;
    346      1.32    blymn 			PSYNCH_OUT;
    347      1.32    blymn 			scroll(win);
    348      1.32    blymn 			PSYNCH_IN;
    349      1.32    blymn 		} else {
    350      1.32    blymn 			(*y)++;
    351      1.32    blymn 		}
    352      1.32    blymn 		PSYNCH_OUT;
    353      1.32    blymn 		return OK;
    354      1.32    blymn 	case L'\t':
    355      1.32    blymn 		cc.vals[0] = L' ';
    356      1.32    blymn 		cc.elements = 1;
    357      1.32    blymn 		cc.attributes = win->wattr;
    358      1.32    blymn 		for (i = 0; i < 8 - (*x % 8); i++) {
    359      1.32    blymn 			if (wadd_wch(win, &cc) == ERR)
    360      1.32    blymn 				return ERR;
    361      1.32    blymn 		}
    362      1.32    blymn 		return OK;
    363      1.32    blymn 	}
    364      1.32    blymn 
    365      1.32    blymn 	/* check for non-spacing character */
    366      1.32    blymn 	if (!wcwidth(wch->vals[0])) {
    367      1.32    blymn #ifdef DEBUG
    368      1.32    blymn 		__CTRACE(__CTRACE_INPUT,
    369      1.32    blymn 			 "_cursesi_addwchar: char '%c' is non-spacing\n",
    370      1.32    blymn 			 wch->vals[0]);
    371      1.32    blymn #endif /* DEBUG */
    372      1.32    blymn 		cw = WCOL(*lp);
    373      1.32    blymn 		if (cw < 0) {
    374      1.32    blymn 			lp += cw;
    375      1.32    blymn 			*x += cw;
    376      1.32    blymn 		}
    377      1.32    blymn 		for (i = 0; i < wch->elements; i++) {
    378      1.32    blymn 			if (!(np = (nschar_t *) malloc(sizeof(nschar_t))))
    379      1.32    blymn 				return ERR;;
    380      1.32    blymn 			np->ch = wch->vals[i];
    381      1.32    blymn 			np->next = lp->nsp;
    382      1.32    blymn 			lp->nsp = np;
    383      1.32    blymn 		}
    384      1.32    blymn 		(*lnp)->flags |= __ISDIRTY;
    385      1.32    blymn 		newx = *x + win->ch_off;
    386      1.32    blymn 		if (newx < *(*lnp)->firstchp)
    387      1.32    blymn 			*(*lnp)->firstchp = newx;
    388      1.32    blymn 		if (newx > *(*lnp)->lastchp)
    389      1.32    blymn 			*(*lnp)->lastchp = newx;
    390      1.32    blymn 		__touchline(win, *y, *x, *x);
    391      1.32    blymn 		return OK;
    392      1.32    blymn 	}
    393      1.32    blymn 	/* check for new line first */
    394      1.32    blymn 	if ((*lnp)->flags & __ISPASTEOL) {
    395      1.32    blymn 		*x = 0;
    396      1.32    blymn 		(*lnp)->flags &= ~__ISPASTEOL;
    397      1.32    blymn 		if (*y == win->scr_b) {
    398      1.32    blymn 			if (!(win->flags & __SCROLLOK))
    399      1.32    blymn 				return ERR;
    400      1.32    blymn 			PSYNCH_OUT;
    401      1.32    blymn 			scroll(win);
    402      1.32    blymn 			PSYNCH_IN;
    403      1.32    blymn 		} else {
    404      1.32    blymn 			(*y)++;
    405      1.32    blymn 		}
    406      1.32    blymn 		(*lnp) = win->lines[*y];
    407      1.32    blymn 		lp = &win->lines[*y]->line[*x];
    408      1.32    blymn 	}
    409      1.32    blymn 	/* clear out the current character */
    410      1.32    blymn 	cw = WCOL(*lp);
    411      1.32    blymn 	if (cw >= 0) {
    412      1.32    blymn 		sx = *x;
    413      1.32    blymn 	} else {
    414      1.32    blymn 		for (sx = *x - 1; sx >= max(*x + cw, 0); sx--) {
    415       1.5  mycroft #ifdef DEBUG
    416      1.31      jdc 			__CTRACE(__CTRACE_INPUT,
    417      1.32    blymn 				 "_cursesi_addwchar: clear current char (%d,%d)\n",
    418      1.32    blymn 				 *y, sx);
    419      1.32    blymn #endif /* DEBUG */
    420      1.32    blymn 			tp = &win->lines[*y]->line[sx];
    421      1.32    blymn 			tp->ch = (wchar_t) btowc((int) win->bch);
    422      1.32    blymn 			if (_cursesi_copy_nsp(win->bnsp, tp) == ERR)
    423      1.32    blymn 				return ERR;
    424      1.32    blymn 
    425      1.32    blymn 			tp->attr = win->battr;
    426      1.32    blymn 			SET_WCOL(*tp, 1);
    427      1.32    blymn 		}
    428      1.32    blymn 		sx = *x + cw;
    429      1.32    blymn 		(*lnp)->flags |= __ISDIRTY;
    430      1.32    blymn 		newx = sx + win->ch_off;
    431      1.32    blymn 		if (newx < *(*lnp)->firstchp)
    432      1.32    blymn 			*(*lnp)->firstchp = newx;
    433      1.32    blymn 	}
    434      1.32    blymn 
    435      1.32    blymn 	/* check for enough space before the end of line */
    436      1.32    blymn 	cw = wcwidth(wch->vals[0]);
    437      1.32    blymn 	if (cw > win->maxx - *x) {
    438      1.32    blymn #ifdef DEBUG
    439      1.32    blymn 		__CTRACE(__CTRACE_INPUT,
    440      1.32    blymn 			 "_cursesi_addwchar: clear EOL (%d,%d)\n",
    441      1.32    blymn 			 *y, *x);
    442      1.32    blymn #endif /* DEBUG */
    443      1.32    blymn 		(*lnp)->flags |= __ISDIRTY;
    444      1.32    blymn 		newx = *x + win->ch_off;
    445      1.32    blymn 		if (newx < *(*lnp)->firstchp)
    446      1.32    blymn 			*(*lnp)->firstchp = newx;
    447      1.32    blymn 		for (tp = lp; *x < win->maxx; tp++, (*x)++) {
    448      1.32    blymn 			tp->ch = (wchar_t) btowc((int) win->bch);
    449      1.32    blymn 			if (_cursesi_copy_nsp(win->bnsp, tp) == ERR)
    450      1.32    blymn 				return ERR;
    451      1.32    blymn 			tp->attr = win->battr;
    452      1.32    blymn 			SET_WCOL(*tp, 1);
    453      1.32    blymn 		}
    454      1.32    blymn 		newx = win->maxx - 1 + win->ch_off;
    455      1.32    blymn 		if (newx > *(*lnp)->lastchp)
    456      1.32    blymn 			*(*lnp)->lastchp = newx;
    457      1.32    blymn 		__touchline(win, *y, sx, (int) win->maxx - 1);
    458      1.32    blymn 		sx = *x = 0;
    459      1.32    blymn 		if (*y == win->scr_b) {
    460      1.32    blymn 			if (!(win->flags & __SCROLLOK))
    461      1.32    blymn 				return ERR;
    462      1.32    blymn 			PSYNCH_OUT;
    463      1.32    blymn 			scroll(win);
    464      1.32    blymn 			PSYNCH_IN;
    465      1.32    blymn 		} else {
    466      1.32    blymn 			(*y)++;
    467      1.32    blymn 		}
    468      1.32    blymn 		lp = &win->lines[*y]->line[0];
    469      1.32    blymn 		(*lnp) = win->lines[*y];
    470      1.32    blymn 	}
    471      1.32    blymn 	win->cury = *y;
    472      1.32    blymn 
    473      1.32    blymn 	/* add spacing character */
    474      1.32    blymn #ifdef DEBUG
    475      1.32    blymn 	__CTRACE(__CTRACE_INPUT,
    476      1.32    blymn 		 "_cursesi_addwchar: add character (%d,%d) 0x%x\n",
    477      1.32    blymn 		 *y, *x, wch->vals[0]);
    478      1.32    blymn #endif /* DEBUG */
    479      1.32    blymn 	(*lnp)->flags |= __ISDIRTY;
    480      1.32    blymn 	newx = *x + win->ch_off;
    481      1.32    blymn 	if (newx < *(*lnp)->firstchp)
    482      1.32    blymn 		*(*lnp)->firstchp = newx;
    483      1.32    blymn 	if (lp->nsp) {
    484      1.32    blymn 		__cursesi_free_nsp(lp->nsp);
    485      1.32    blymn 		lp->nsp = NULL;
    486      1.32    blymn 	}
    487      1.32    blymn 
    488      1.32    blymn 	lp->ch = wch->vals[0];
    489      1.32    blymn 
    490      1.32    blymn 	attributes = (win->wattr | wch->attributes)
    491      1.32    blymn 		& (WA_ATTRIBUTES & ~__COLOR);
    492      1.32    blymn 	if (wch->attributes & __COLOR)
    493      1.32    blymn 		attributes |= wch->attributes & __COLOR;
    494      1.32    blymn 	else if (win->wattr & __COLOR)
    495      1.32    blymn 		attributes |= win->wattr & __COLOR;
    496      1.32    blymn 	if (attributes & __COLOR)
    497      1.32    blymn 		lp->attr = attributes | (win->battr & ~__COLOR);
    498      1.32    blymn 	else
    499      1.32    blymn 		lp->attr = attributes | win->battr;
    500      1.32    blymn 
    501      1.32    blymn 	SET_WCOL(*lp, cw);
    502      1.32    blymn 
    503      1.32    blymn #ifdef DEBUG
    504      1.32    blymn 	__CTRACE(__CTRACE_INPUT,
    505      1.32    blymn 		 "_cursesi_addwchar: add spacing char 0x%x, attr 0x%x\n",
    506      1.32    blymn 		 lp->ch, lp->attr);
    507      1.32    blymn #endif /* DEBUG */
    508      1.32    blymn 
    509      1.32    blymn 	if (wch->elements > 1) {
    510      1.32    blymn 		for (i = 1; i < wch->elements; i++) {
    511      1.32    blymn 			np = (nschar_t *)malloc(sizeof(nschar_t));
    512      1.32    blymn 			if (!np)
    513      1.32    blymn 				return ERR;;
    514      1.32    blymn 			np->ch = wch->vals[i];
    515      1.32    blymn 			np->next = lp->nsp;
    516      1.24      jdc #ifdef DEBUG
    517      1.31      jdc 			__CTRACE(__CTRACE_INPUT,
    518      1.32    blymn 			    "_cursesi_addwchar: add non-spacing char 0x%x\n", np->ch);
    519      1.32    blymn #endif /* DEBUG */
    520      1.32    blymn 			lp->nsp = np;
    521      1.32    blymn 		}
    522      1.32    blymn 	}
    523       1.5  mycroft #ifdef DEBUG
    524      1.32    blymn 	__CTRACE(__CTRACE_INPUT, "_cursesi_addwchar: non-spacing list header: %p\n",
    525      1.32    blymn 	    lp->nsp);
    526      1.32    blymn 	__CTRACE(__CTRACE_INPUT, "_cursesi_addwchar: add rest columns (%d:%d)\n",
    527      1.32    blymn 		sx + 1, sx + cw - 1);
    528      1.32    blymn #endif /* DEBUG */
    529      1.32    blymn 	for (tp = lp + 1, *x = sx + 1; *x - sx <= cw - 1; tp++, (*x)++) {
    530      1.32    blymn 		if (tp->nsp) {
    531      1.32    blymn 			__cursesi_free_nsp(tp->nsp);
    532      1.32    blymn 			tp->nsp = NULL;
    533       1.5  mycroft 		}
    534      1.32    blymn 		tp->ch = wch->vals[0];
    535      1.32    blymn 		tp->attr = lp->attr & WA_ATTRIBUTES;
    536      1.32    blymn 		/* Mark as "continuation" cell */
    537      1.32    blymn 		tp->attr |= __WCWIDTH;
    538       1.5  mycroft 	}
    539      1.32    blymn 	if (*x == win->maxx) {
    540      1.32    blymn 		(*lnp)->flags |= __ISPASTEOL;
    541      1.32    blymn 		newx = win->maxx - 1 + win->ch_off;
    542      1.32    blymn 		if (newx > *(*lnp)->lastchp)
    543      1.32    blymn 			*(*lnp)->lastchp = newx;
    544      1.32    blymn 		__touchline(win, *y, sx, (int) win->maxx - 1);
    545      1.32    blymn 		win->curx = sx;
    546      1.32    blymn 	} else {
    547      1.32    blymn 		win->curx = *x;
    548      1.32    blymn 
    549      1.32    blymn 		/* clear the remining of the current characer */
    550      1.32    blymn 		if (*x && *x < win->maxx) {
    551      1.32    blymn 			ex = sx + cw;
    552      1.32    blymn 			tp = &win->lines[*y]->line[ex];
    553      1.32    blymn 			while (ex < win->maxx && WCOL(*tp) < 0) {
    554      1.23    blymn #ifdef DEBUG
    555      1.32    blymn 				__CTRACE(__CTRACE_INPUT,
    556      1.32    blymn 					 "_cursesi_addwchar: clear "
    557      1.32    blymn 					 "remaining of current char (%d,%d)nn",
    558      1.32    blymn 					 *y, ex);
    559      1.32    blymn #endif /* DEBUG */
    560      1.32    blymn 				tp->ch = (wchar_t) btowc((int) win->bch);
    561      1.32    blymn 				if (_cursesi_copy_nsp(win->bnsp, tp) == ERR)
    562      1.32    blymn 					return ERR;
    563      1.32    blymn 				tp->attr = win->battr;
    564      1.32    blymn 				SET_WCOL(*tp, 1);
    565      1.32    blymn 				tp++, ex++;
    566      1.32    blymn 			}
    567      1.32    blymn 			newx = ex - 1 + win->ch_off;
    568      1.32    blymn 			if (newx > *(*lnp)->lastchp)
    569      1.32    blymn 				*(*lnp)->lastchp = newx;
    570      1.32    blymn 			__touchline(win, *y, sx, ex - 1);
    571      1.32    blymn 		}
    572      1.23    blymn 	}
    573      1.32    blymn 
    574      1.32    blymn #ifdef DEBUG
    575      1.32    blymn 	__CTRACE(__CTRACE_INPUT, "add_wch: %d : 0x%x\n", lp->ch, lp->attr);
    576      1.32    blymn #endif /* DEBUG */
    577      1.32    blymn 	return OK;
    578      1.23    blymn #endif
    579       1.1      cgd }
    580