Home | History | Annotate | Line # | Download | only in libcurses
addbytes.c revision 1.69.2.1
      1  1.69.2.1  perseant /*	$NetBSD: addbytes.c,v 1.69.2.1 2025/08/02 05:54:45 perseant 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.69.2.1  perseant __RCSID("$NetBSD: addbytes.c,v 1.69.2.1 2025/08/02 05:54:45 perseant 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.37   tnozaki #include <string.h>
     43      1.10       cgd #include "curses.h"
     44      1.15     blymn #include "curses_private.h"
     45      1.23     blymn #ifdef DEBUG
     46      1.23     blymn #include <assert.h>
     47      1.23     blymn #endif
     48       1.1       cgd 
     49      1.17     blymn #ifndef _CURSES_USE_MACROS
     50      1.17     blymn 
     51      1.17     blymn /*
     52      1.17     blymn  * addbytes --
     53      1.54    rillig  *      Add the characters to the current position in stdscr.
     54      1.17     blymn  */
     55      1.17     blymn int
     56      1.17     blymn addbytes(const char *bytes, int count)
     57      1.17     blymn {
     58      1.46       roy 
     59      1.41     blymn 	return _cursesi_waddbytes(stdscr, bytes, count, 0, 1);
     60      1.17     blymn }
     61      1.17     blymn 
     62      1.17     blymn /*
     63      1.17     blymn  * waddbytes --
     64      1.54    rillig  *      Add the characters to the current position in the given window.
     65      1.17     blymn  */
     66      1.17     blymn int
     67      1.17     blymn waddbytes(WINDOW *win, const char *bytes, int count)
     68      1.17     blymn {
     69      1.46       roy 
     70      1.41     blymn 	return _cursesi_waddbytes(win, bytes, count, 0, 1);
     71      1.17     blymn }
     72      1.17     blymn 
     73      1.17     blymn /*
     74      1.17     blymn  * mvaddbytes --
     75      1.17     blymn  *      Add the characters to stdscr at the location given.
     76      1.17     blymn  */
     77      1.17     blymn int
     78      1.17     blymn mvaddbytes(int y, int x, const char *bytes, int count)
     79      1.17     blymn {
     80      1.46       roy 
     81      1.17     blymn 	return mvwaddbytes(stdscr, y, x, bytes, count);
     82      1.17     blymn }
     83      1.17     blymn 
     84      1.17     blymn /*
     85      1.17     blymn  * mvwaddbytes --
     86      1.17     blymn  *      Add the characters to the given window at the location given.
     87      1.17     blymn  */
     88      1.17     blymn int
     89      1.17     blymn mvwaddbytes(WINDOW *win, int y, int x, const char *bytes, int count)
     90      1.17     blymn {
     91      1.46       roy 
     92      1.52     blymn 	if (wmove(win, y, x) == ERR)
     93      1.17     blymn 		return ERR;
     94      1.17     blymn 
     95      1.41     blymn 	return _cursesi_waddbytes(win, bytes, count, 0, 1);
     96      1.17     blymn }
     97      1.17     blymn 
     98      1.17     blymn #endif
     99      1.17     blymn 
    100      1.41     blymn int
    101      1.41     blymn __waddbytes(WINDOW *win, const char *bytes, int count, attr_t attr)
    102      1.41     blymn {
    103      1.46       roy 
    104      1.41     blymn 	return _cursesi_waddbytes(win, bytes, count, attr, 1);
    105      1.41     blymn }
    106      1.41     blymn 
    107       1.1       cgd /*
    108      1.41     blymn  * _cursesi_waddbytes --
    109      1.54    rillig  *	Add the characters to the current position in the given window.
    110      1.41     blymn  * if char_interp is non-zero then character interpretation is done on
    111      1.41     blymn  * the byte (i.e. \n to newline, \r to carriage return, \b to backspace
    112      1.41     blymn  * and so on).
    113       1.1       cgd  */
    114       1.5   mycroft int
    115      1.41     blymn _cursesi_waddbytes(WINDOW *win, const char *bytes, int count, attr_t attr,
    116      1.41     blymn 	    int char_interp)
    117       1.1       cgd {
    118      1.54    rillig 	int		*py = &win->cury, *px = &win->curx, err;
    119      1.15     blymn 	__LINE		*lp;
    120      1.32     blymn #ifdef HAVE_WCHAR
    121      1.68     blymn 	int		n;
    122      1.32     blymn 	cchar_t		cc;
    123      1.32     blymn 	wchar_t		wc;
    124      1.34   tnozaki 	mbstate_t	st;
    125      1.32     blymn #else
    126      1.32     blymn 	int		c;
    127      1.32     blymn #endif
    128      1.23     blymn #ifdef DEBUG
    129      1.23     blymn 	int             i;
    130       1.5   mycroft 
    131  1.69.2.1  perseant 	if (__predict_false(win == NULL))
    132  1.69.2.1  perseant 		return ERR;
    133  1.69.2.1  perseant 
    134      1.23     blymn 	for (i = 0; i < win->maxy; i++) {
    135      1.35       roy 		assert(win->alines[i]->sentinel == SENTINEL_VALUE);
    136      1.23     blymn 	}
    137      1.32     blymn 
    138      1.32     blymn 	__CTRACE(__CTRACE_INPUT, "ADDBYTES: add %d bytes\n", count);
    139      1.23     blymn #endif
    140      1.32     blymn 
    141      1.32     blymn 	err = OK;
    142      1.54    rillig 	lp = win->alines[*py];
    143       1.1       cgd 
    144      1.34   tnozaki #ifdef HAVE_WCHAR
    145      1.37   tnozaki 	(void)memset(&st, 0, sizeof(st));
    146      1.34   tnozaki #endif
    147      1.32     blymn 	while (count > 0) {
    148      1.32     blymn #ifndef HAVE_WCHAR
    149      1.11     mikel 		c = *bytes++;
    150      1.31       jdc 		__CTRACE(__CTRACE_INPUT, "ADDBYTES('%c', %x) at (%d, %d)\n",
    151      1.54    rillig 		    c, attr, *py, *px);
    152      1.54    rillig 		err = _cursesi_addbyte(win, &lp, py, px, c, attr, char_interp);
    153      1.32     blymn 		count--;
    154      1.32     blymn #else
    155      1.32     blymn 		/*
    156      1.54    rillig 		 * For wide-character support only, try to convert the
    157      1.32     blymn 		 * given string into a wide character - we do this because
    158      1.32     blymn 		 * this is how ncurses behaves (not that I think this is
    159      1.32     blymn 		 * actually the correct thing to do but if we don't do it
    160      1.32     blymn 		 * a lot of things that rely on this behaviour will break
    161      1.32     blymn 		 * and we will be blamed).  If the conversion succeeds
    162      1.32     blymn 		 * then we eat the n characters used to make the wide char
    163      1.32     blymn 		 * from the string.
    164      1.32     blymn 		 */
    165      1.34   tnozaki 		n = (int)mbrtowc(&wc, bytes, (size_t)count, &st);
    166      1.34   tnozaki 		if (n < 0) {
    167      1.34   tnozaki 			/* not a valid conversion just eat a char */
    168      1.32     blymn 			wc = *bytes;
    169      1.32     blymn 			n = 1;
    170      1.39     joerg 			(void)memset(&st, 0, sizeof(st));
    171      1.34   tnozaki 		} else if (wc == 0) {
    172      1.34   tnozaki 			break;
    173      1.32     blymn 		}
    174      1.55     blymn 
    175      1.54    rillig 		__CTRACE(__CTRACE_INPUT,
    176      1.68     blymn 		    "ADDBYTES WIDE(0x%04x [%s], %x) at (%d, %d), ate %d bytes\n",
    177      1.54    rillig 		    (unsigned)wc, unctrl((unsigned)wc), attr, *py, *px, n);
    178      1.32     blymn 		cc.vals[0] = wc;
    179      1.32     blymn 		cc.elements = 1;
    180      1.32     blymn 		cc.attributes = attr;
    181      1.54    rillig 		err = _cursesi_addwchar(win, &lp, py, px, &cc, char_interp);
    182      1.32     blymn 		bytes += n;
    183      1.68     blymn 		count -= n;
    184      1.32     blymn #endif
    185      1.32     blymn 	}
    186      1.32     blymn 
    187      1.32     blymn #ifdef DEBUG
    188      1.32     blymn 	for (i = 0; i < win->maxy; i++) {
    189      1.35       roy 		assert(win->alines[i]->sentinel == SENTINEL_VALUE);
    190      1.32     blymn 	}
    191      1.32     blymn #endif
    192      1.32     blymn 
    193      1.32     blymn 	return (err);
    194      1.32     blymn }
    195       1.5   mycroft 
    196      1.32     blymn /*
    197      1.32     blymn  * _cursesi_addbyte -
    198      1.32     blymn  *	Internal function to add a byte and update the row and column
    199      1.49       uwe  * positions as appropriate.  If char_interp is non-zero then
    200      1.49       uwe  * character interpretation is done on the byte.  This function is
    201      1.49       uwe  * only used in the narrow character version of curses.
    202      1.32     blymn  */
    203      1.32     blymn int
    204      1.32     blymn _cursesi_addbyte(WINDOW *win, __LINE **lp, int *y, int *x, int c,
    205      1.41     blymn 		 attr_t attr, int char_interp)
    206      1.32     blymn {
    207      1.41     blymn 	static char	 blank[] = " ";
    208      1.41     blymn 	int		 tabsize;
    209      1.61     blymn 	int		 newx, i, wcols;
    210      1.32     blymn 	attr_t		 attributes;
    211      1.32     blymn 
    212  1.69.2.1  perseant 	if (__predict_false(win == NULL))
    213  1.69.2.1  perseant 		return ERR;
    214  1.69.2.1  perseant 
    215      1.41     blymn 	if (char_interp) {
    216      1.41     blymn 		switch (c) {
    217      1.41     blymn 		case '\t':
    218      1.41     blymn 			tabsize = win->screen->TABSIZE;
    219      1.44  christos 			newx = tabsize - (*x % tabsize);
    220      1.55     blymn 			/* if at the bottom of the window and
    221      1.55     blymn 			   not allowed to scroll then just do
    222      1.55     blymn 			   what we can */
    223      1.55     blymn 			if ((*y == win->scr_b) &&
    224      1.55     blymn 			    !(win->flags & __SCROLLOK)) {
    225      1.55     blymn 				if ((*lp)->flags & __ISPASTEOL) {
    226      1.55     blymn 					return OK;
    227      1.55     blymn 				}
    228      1.55     blymn 
    229      1.55     blymn 				if (*x + newx > win->maxx - 1)
    230      1.55     blymn 					newx = win->maxx - *x - 1;
    231      1.55     blymn 			}
    232      1.55     blymn 
    233      1.44  christos 			for (i = 0; i < newx; i++) {
    234      1.41     blymn 				if (waddbytes(win, blank, 1) == ERR)
    235      1.47       roy 					return ERR;
    236      1.41     blymn 			}
    237      1.47       roy 			return OK;
    238      1.41     blymn 
    239      1.41     blymn 		case '\n':
    240      1.41     blymn 			wclrtoeol(win);
    241      1.41     blymn 			(*lp)->flags |= __ISPASTEOL;
    242      1.41     blymn 			break;
    243      1.41     blymn 
    244      1.41     blymn 		case '\r':
    245      1.41     blymn 			*x = 0;
    246      1.47       roy 			return OK;
    247      1.41     blymn 
    248      1.41     blymn 		case '\b':
    249      1.41     blymn 			if (--(*x) < 0)
    250      1.41     blymn 				*x = 0;
    251      1.47       roy 			return OK;
    252      1.41     blymn 		}
    253      1.41     blymn 	}
    254      1.32     blymn 
    255      1.41     blymn 	__CTRACE(__CTRACE_INPUT, "ADDBYTES(%p, %d, %d)\n", win, *y, *x);
    256      1.14    simonb 
    257      1.41     blymn 	if (char_interp && ((*lp)->flags & __ISPASTEOL)) {
    258      1.41     blymn 		*x = 0;
    259      1.41     blymn 		(*lp)->flags &= ~__ISPASTEOL;
    260      1.41     blymn 		if (*y == win->scr_b) {
    261      1.41     blymn 			__CTRACE(__CTRACE_INPUT,
    262      1.59       rin 			    "ADDBYTES - on bottom of scrolling region\n");
    263      1.41     blymn 			if (!(win->flags & __SCROLLOK))
    264      1.41     blymn 				return ERR;
    265      1.41     blymn 			scroll(win);
    266      1.41     blymn 		} else {
    267      1.41     blymn 			(*y)++;
    268      1.32     blymn 		}
    269      1.41     blymn 		*lp = win->alines[*y];
    270      1.41     blymn 		if (c == '\n')
    271      1.47       roy 			return OK;
    272      1.41     blymn 	}
    273      1.14    simonb 
    274      1.41     blymn 	__CTRACE(__CTRACE_INPUT,
    275      1.59       rin 	    "ADDBYTES: 1: y = %d, x = %d, firstch = %d, lastch = %d\n",
    276      1.59       rin 	    *y, *x, *win->alines[*y]->firstchp, *win->alines[*y]->lastchp);
    277      1.41     blymn 
    278      1.41     blymn 	attributes = (win->wattr | attr) & (__ATTRIBUTES & ~__COLOR);
    279      1.41     blymn 	if (attr & __COLOR)
    280      1.41     blymn 		attributes |= attr & __COLOR;
    281      1.41     blymn 	else if (win->wattr & __COLOR)
    282      1.41     blymn 		attributes |= win->wattr & __COLOR;
    283      1.41     blymn 
    284      1.61     blymn 
    285      1.61     blymn 	wcols = wcwidth(c);
    286      1.61     blymn 	if (wcols < 0)
    287      1.61     blymn 		wcols = 1;
    288      1.61     blymn 
    289      1.41     blymn 	/*
    290      1.41     blymn 	 * Always update the change pointers.  Otherwise,
    291      1.41     blymn 	 * we could end up not displaying 'blank' characters
    292      1.41     blymn 	 * when overlapping windows are displayed.
    293      1.41     blymn 	 */
    294      1.41     blymn 	newx = *x + win->ch_off;
    295      1.41     blymn 	(*lp)->flags |= __ISDIRTY;
    296      1.41     blymn 	/*
    297      1.41     blymn 	 * firstchp/lastchp are shared between
    298      1.41     blymn 	 * parent window and sub-window.
    299      1.41     blymn 	 */
    300      1.41     blymn 	if (newx < *(*lp)->firstchp)
    301      1.41     blymn 		*(*lp)->firstchp = newx;
    302      1.65     blymn 
    303      1.41     blymn 	if (newx > *(*lp)->lastchp)
    304      1.41     blymn 		*(*lp)->lastchp = newx;
    305      1.41     blymn 	__CTRACE(__CTRACE_INPUT, "ADDBYTES: change gives f/l: %d/%d [%d/%d]\n",
    306      1.59       rin 	    *(*lp)->firstchp, *(*lp)->lastchp,
    307      1.59       rin 	    *(*lp)->firstchp - win->ch_off,
    308      1.59       rin 	    *(*lp)->lastchp - win->ch_off);
    309      1.61     blymn 	if (win->bch != ' ' && c == ' ') {
    310      1.41     blymn 		(*lp)->line[*x].ch = win->bch;
    311      1.61     blymn #ifdef HAVE_CHAR
    312      1.61     blymn 		(*lp)->line[*x].wcols = win->wcols;
    313      1.61     blymn #endif
    314      1.61     blymn 	} else {
    315      1.41     blymn 		(*lp)->line[*x].ch = c;
    316      1.61     blymn #ifdef HAVE_CHAR
    317      1.61     blymn 		(*lp)->line[*x].wcols = wcols;
    318      1.61     blymn #endif
    319      1.61     blymn 	}
    320      1.41     blymn 
    321      1.65     blymn 	(*lp)->line[*x].cflags &= ~ (CA_BACKGROUND | CA_CONTINUATION);
    322      1.63     blymn 
    323      1.41     blymn 	if (attributes & __COLOR)
    324      1.41     blymn 		(*lp)->line[*x].attr =
    325      1.41     blymn 			attributes | (win->battr & ~__COLOR);
    326      1.41     blymn 	else
    327      1.41     blymn 		(*lp)->line[*x].attr = attributes | win->battr;
    328      1.41     blymn 
    329      1.41     blymn 	if (*x == win->maxx - 1)
    330      1.41     blymn 		(*lp)->flags |= __ISPASTEOL;
    331      1.41     blymn 	else
    332      1.41     blymn 		(*x)++;
    333      1.32     blymn 
    334      1.41     blymn 	__CTRACE(__CTRACE_INPUT,
    335      1.59       rin 	    "ADDBYTES: 2: y = %d, x = %d, firstch = %d, lastch = %d\n",
    336      1.59       rin 	    *y, *x, *win->alines[*y]->firstchp, *win->alines[*y]->lastchp);
    337      1.45       roy 	__sync(win);
    338      1.47       roy 	return OK;
    339      1.32     blymn }
    340      1.32     blymn 
    341      1.32     blymn /*
    342      1.32     blymn  * _cursesi_addwchar -
    343      1.32     blymn  *	Internal function to add a wide character and update the row
    344      1.32     blymn  * and column positions.
    345      1.32     blymn  */
    346      1.32     blymn int
    347      1.32     blymn _cursesi_addwchar(WINDOW *win, __LINE **lnp, int *y, int *x,
    348      1.41     blymn 		  const cchar_t *wch, int char_interp)
    349      1.32     blymn {
    350      1.32     blymn #ifndef HAVE_WCHAR
    351      1.47       roy 	return ERR;
    352      1.32     blymn #else
    353      1.40       roy 	int sx = 0, ex = 0, cw = 0, i = 0, newx = 0, tabsize;
    354      1.35       roy 	__LDATA *lp = &win->alines[*y]->line[*x], *tp = NULL;
    355      1.32     blymn 	nschar_t *np = NULL;
    356      1.32     blymn 	cchar_t cc;
    357      1.32     blymn 	attr_t attributes;
    358      1.32     blymn 
    359  1.69.2.1  perseant 	if (__predict_false(win == NULL))
    360  1.69.2.1  perseant 		return ERR;
    361  1.69.2.1  perseant 
    362      1.41     blymn 	if (char_interp) {
    363      1.41     blymn 		/* special characters handling */
    364      1.41     blymn 		switch (wch->vals[0]) {
    365      1.41     blymn 		case L'\b':
    366      1.41     blymn 			if (--*x < 0)
    367      1.41     blymn 				*x = 0;
    368      1.41     blymn 			return OK;
    369      1.41     blymn 		case L'\r':
    370      1.32     blymn 			*x = 0;
    371      1.41     blymn 			return OK;
    372      1.41     blymn 		case L'\n':
    373      1.41     blymn 			if (*y == win->scr_b) {
    374      1.41     blymn 				if (!(win->flags & __SCROLLOK))
    375      1.41     blymn 					return ERR;
    376      1.57     blymn 				wclrtoeol(win);
    377      1.41     blymn 				scroll(win);
    378      1.41     blymn 			} else {
    379      1.57     blymn 				wclrtoeol(win);
    380      1.41     blymn 				(*y)++;
    381      1.41     blymn 			}
    382      1.57     blymn 			*x = 0;
    383      1.57     blymn 			(*lnp)->flags &= ~__ISPASTEOL;
    384      1.41     blymn 			return OK;
    385      1.41     blymn 		case L'\t':
    386      1.41     blymn 			cc.vals[0] = L' ';
    387      1.41     blymn 			cc.elements = 1;
    388      1.41     blymn 			cc.attributes = win->wattr;
    389      1.41     blymn 			tabsize = win->screen->TABSIZE;
    390      1.44  christos 			newx = tabsize - (*x % tabsize);
    391      1.55     blymn 
    392      1.55     blymn 			/* if at the bottom of the window and
    393      1.55     blymn 			   not allowed to scroll then just do
    394      1.55     blymn 			   what we can */
    395      1.55     blymn 			if ((*y == win->scr_b) &&
    396      1.55     blymn 			    !(win->flags & __SCROLLOK)) {
    397      1.55     blymn 				if ((*lnp)->flags & __ISPASTEOL) {
    398      1.57     blymn 					return ERR;
    399      1.55     blymn 				}
    400      1.55     blymn 
    401      1.55     blymn 				if (*x + newx > win->maxx - 1)
    402      1.55     blymn 					newx = win->maxx - *x - 1;
    403      1.55     blymn 			}
    404      1.55     blymn 
    405      1.44  christos 			for (i = 0; i < newx; i++) {
    406      1.41     blymn 				if (wadd_wch(win, &cc) == ERR)
    407      1.41     blymn 					return ERR;
    408      1.41     blymn 			}
    409      1.41     blymn 			return OK;
    410      1.32     blymn 		}
    411      1.32     blymn 	}
    412      1.32     blymn 
    413      1.32     blymn 	/* check for non-spacing character */
    414      1.32     blymn 	if (!wcwidth(wch->vals[0])) {
    415      1.32     blymn 		__CTRACE(__CTRACE_INPUT,
    416      1.59       rin 		    "_cursesi_addwchar: char '%c' is non-spacing\n",
    417      1.59       rin 		    wch->vals[0]);
    418      1.61     blymn 		cw = (*lp).wcols;
    419      1.32     blymn 		if (cw < 0) {
    420      1.32     blymn 			lp += cw;
    421      1.32     blymn 			*x += cw;
    422      1.32     blymn 		}
    423      1.32     blymn 		for (i = 0; i < wch->elements; i++) {
    424      1.32     blymn 			if (!(np = (nschar_t *) malloc(sizeof(nschar_t))))
    425      1.60    rillig 				return ERR;
    426      1.32     blymn 			np->ch = wch->vals[i];
    427      1.32     blymn 			np->next = lp->nsp;
    428      1.32     blymn 			lp->nsp = np;
    429      1.32     blymn 		}
    430      1.32     blymn 		(*lnp)->flags |= __ISDIRTY;
    431      1.32     blymn 		newx = *x + win->ch_off;
    432      1.32     blymn 		if (newx < *(*lnp)->firstchp)
    433      1.32     blymn 			*(*lnp)->firstchp = newx;
    434      1.65     blymn 
    435      1.32     blymn 		if (newx > *(*lnp)->lastchp)
    436      1.32     blymn 			*(*lnp)->lastchp = newx;
    437      1.32     blymn 		__touchline(win, *y, *x, *x);
    438      1.32     blymn 		return OK;
    439      1.32     blymn 	}
    440      1.32     blymn 	/* check for new line first */
    441      1.41     blymn 	if (char_interp && ((*lnp)->flags & __ISPASTEOL)) {
    442      1.32     blymn 		*x = 0;
    443      1.32     blymn 		(*lnp)->flags &= ~__ISPASTEOL;
    444      1.32     blymn 		if (*y == win->scr_b) {
    445      1.32     blymn 			if (!(win->flags & __SCROLLOK))
    446      1.32     blymn 				return ERR;
    447      1.32     blymn 			scroll(win);
    448      1.32     blymn 		} else {
    449      1.32     blymn 			(*y)++;
    450      1.32     blymn 		}
    451      1.35       roy 		(*lnp) = win->alines[*y];
    452      1.35       roy 		lp = &win->alines[*y]->line[*x];
    453      1.32     blymn 	}
    454      1.32     blymn 	/* clear out the current character */
    455      1.61     blymn 	cw = (*lp).wcols;
    456      1.32     blymn 	if (cw >= 0) {
    457      1.32     blymn 		sx = *x;
    458      1.32     blymn 	} else {
    459      1.32     blymn 		for (sx = *x - 1; sx >= max(*x + cw, 0); sx--) {
    460      1.31       jdc 			__CTRACE(__CTRACE_INPUT,
    461      1.59       rin 			    "_cursesi_addwchar: clear current char (%d,%d)\n",
    462      1.59       rin 			    *y, sx);
    463      1.35       roy 			tp = &win->alines[*y]->line[sx];
    464      1.62     blymn 			tp->ch = win->bch;
    465      1.65     blymn 			tp->cflags = CA_BACKGROUND;
    466      1.32     blymn 			if (_cursesi_copy_nsp(win->bnsp, tp) == ERR)
    467      1.32     blymn 				return ERR;
    468      1.32     blymn 
    469      1.32     blymn 			tp->attr = win->battr;
    470      1.61     blymn 			tp->wcols = win->wcols;
    471      1.32     blymn 		}
    472      1.32     blymn 		sx = *x + cw;
    473      1.32     blymn 		(*lnp)->flags |= __ISDIRTY;
    474      1.32     blymn 		newx = sx + win->ch_off;
    475      1.32     blymn 		if (newx < *(*lnp)->firstchp)
    476      1.32     blymn 			*(*lnp)->firstchp = newx;
    477      1.32     blymn 	}
    478      1.32     blymn 
    479      1.32     blymn 	/* check for enough space before the end of line */
    480      1.32     blymn 	cw = wcwidth(wch->vals[0]);
    481      1.36  drochner 	if (cw < 0)
    482      1.36  drochner 		cw = 1;
    483      1.41     blymn 
    484      1.32     blymn 	if (cw > win->maxx - *x) {
    485      1.32     blymn 		__CTRACE(__CTRACE_INPUT,
    486      1.59       rin 		    "_cursesi_addwchar: clear EOL (%d,%d)\n", *y, *x);
    487      1.57     blymn 		if (*y == win->scr_b) {
    488      1.57     blymn 			if (!(win->flags & __SCROLLOK))
    489      1.57     blymn 				return ERR;
    490      1.57     blymn 			scroll(win);
    491      1.57     blymn 		}
    492      1.57     blymn 
    493      1.32     blymn 		(*lnp)->flags |= __ISDIRTY;
    494      1.32     blymn 		newx = *x + win->ch_off;
    495      1.32     blymn 		if (newx < *(*lnp)->firstchp)
    496      1.32     blymn 			*(*lnp)->firstchp = newx;
    497      1.65     blymn 
    498      1.32     blymn 		for (tp = lp; *x < win->maxx; tp++, (*x)++) {
    499      1.62     blymn 			tp->ch = win->bch;
    500      1.32     blymn 			if (_cursesi_copy_nsp(win->bnsp, tp) == ERR)
    501      1.32     blymn 				return ERR;
    502      1.32     blymn 			tp->attr = win->battr;
    503      1.61     blymn 			tp->wcols = win->wcols;
    504      1.65     blymn 			tp->cflags = CA_BACKGROUND;
    505      1.32     blymn 		}
    506      1.32     blymn 		newx = win->maxx - 1 + win->ch_off;
    507      1.32     blymn 		if (newx > *(*lnp)->lastchp)
    508      1.32     blymn 			*(*lnp)->lastchp = newx;
    509      1.32     blymn 		__touchline(win, *y, sx, (int) win->maxx - 1);
    510      1.32     blymn 		sx = *x = 0;
    511      1.57     blymn 		if (*y != win->scr_b) {
    512      1.32     blymn 			(*y)++;
    513      1.32     blymn 		}
    514      1.35       roy 		lp = &win->alines[*y]->line[0];
    515      1.35       roy 		(*lnp) = win->alines[*y];
    516      1.32     blymn 	}
    517      1.32     blymn 
    518      1.32     blymn 	/* add spacing character */
    519      1.32     blymn 	__CTRACE(__CTRACE_INPUT,
    520      1.59       rin 	    "_cursesi_addwchar: add character (%d,%d) 0x%x\n",
    521      1.59       rin 	    *y, *x, wch->vals[0]);
    522      1.32     blymn 	(*lnp)->flags |= __ISDIRTY;
    523      1.32     blymn 	newx = *x + win->ch_off;
    524      1.32     blymn 	if (newx < *(*lnp)->firstchp)
    525      1.32     blymn 		*(*lnp)->firstchp = newx;
    526      1.65     blymn 
    527      1.32     blymn 	if (lp->nsp) {
    528      1.32     blymn 		__cursesi_free_nsp(lp->nsp);
    529      1.32     blymn 		lp->nsp = NULL;
    530      1.32     blymn 	}
    531      1.32     blymn 
    532      1.32     blymn 	lp->ch = wch->vals[0];
    533      1.65     blymn 	lp->cflags &= ~ (CA_BACKGROUND | CA_CONTINUATION);
    534      1.32     blymn 
    535      1.32     blymn 	attributes = (win->wattr | wch->attributes)
    536      1.32     blymn 		& (WA_ATTRIBUTES & ~__COLOR);
    537      1.32     blymn 	if (wch->attributes & __COLOR)
    538      1.32     blymn 		attributes |= wch->attributes & __COLOR;
    539      1.32     blymn 	else if (win->wattr & __COLOR)
    540      1.32     blymn 		attributes |= win->wattr & __COLOR;
    541      1.32     blymn 	if (attributes & __COLOR)
    542      1.32     blymn 		lp->attr = attributes | (win->battr & ~__COLOR);
    543      1.32     blymn 	else
    544      1.32     blymn 		lp->attr = attributes | win->battr;
    545      1.32     blymn 
    546      1.61     blymn 	lp->wcols = cw;
    547      1.32     blymn 
    548      1.32     blymn 	__CTRACE(__CTRACE_INPUT,
    549      1.61     blymn 	    "_cursesi_addwchar: add spacing char 0x%x, attr 0x%x, width %d\n",
    550      1.61     blymn 	    lp->ch, lp->attr, lp->wcols);
    551      1.32     blymn 
    552      1.32     blymn 	if (wch->elements > 1) {
    553      1.32     blymn 		for (i = 1; i < wch->elements; i++) {
    554      1.43  christos 			np = malloc(sizeof(nschar_t));
    555      1.32     blymn 			if (!np)
    556      1.60    rillig 				return ERR;
    557      1.32     blymn 			np->ch = wch->vals[i];
    558      1.32     blymn 			np->next = lp->nsp;
    559      1.31       jdc 			__CTRACE(__CTRACE_INPUT,
    560      1.59       rin 			    "_cursesi_addwchar: add non-spacing char 0x%x\n",
    561      1.59       rin 			    np->ch);
    562      1.32     blymn 			lp->nsp = np;
    563      1.32     blymn 		}
    564      1.32     blymn 	}
    565      1.59       rin 	__CTRACE(__CTRACE_INPUT,
    566      1.59       rin 	    "_cursesi_addwchar: non-spacing list header: %p\n", lp->nsp);
    567      1.59       rin 	__CTRACE(__CTRACE_INPUT,
    568      1.59       rin 	    "_cursesi_addwchar: add rest columns (%d:%d)\n",
    569      1.59       rin 	    sx + 1, sx + cw - 1);
    570      1.59       rin 	__CTRACE(__CTRACE_INPUT, "_cursesi_addwchar: *x = %d, win->maxx = %d\n",
    571      1.59       rin 	    *x, win->maxx);
    572      1.65     blymn 	for (tp = lp + 1, *x = sx + 1, i = cw - 1; i > 0; tp++, (*x)++, i--) {
    573      1.65     blymn 		__CTRACE(__CTRACE_INPUT,
    574      1.65     blymn 		    "_cursesi_addwchar: setting continuation at x %d\n", *x);
    575      1.32     blymn 		if (tp->nsp) {
    576      1.32     blymn 			__cursesi_free_nsp(tp->nsp);
    577      1.32     blymn 			tp->nsp = NULL;
    578       1.5   mycroft 		}
    579      1.32     blymn 		tp->ch = wch->vals[0];
    580      1.32     blymn 		tp->attr = lp->attr & WA_ATTRIBUTES;
    581      1.32     blymn 		/* Mark as "continuation" cell */
    582      1.63     blymn 		tp->cflags |= CA_CONTINUATION;
    583      1.65     blymn 		tp->cflags &= ~ CA_BACKGROUND;
    584      1.65     blymn 		tp->wcols = i;
    585       1.5   mycroft 	}
    586      1.41     blymn 
    587      1.61     blymn 	if (*x >= win->maxx) {
    588      1.59       rin 		__CTRACE(__CTRACE_INPUT, "_cursesi_addwchar: do line wrap\n");
    589      1.57     blymn 		if (*y == win->scr_b) {
    590      1.59       rin 			__CTRACE(__CTRACE_INPUT,
    591      1.59       rin 			    "_cursesi_addwchar: at bottom of screen\n");
    592      1.57     blymn 			if (!(win->flags & __SCROLLOK))
    593      1.57     blymn 				return ERR;
    594      1.59       rin 			__CTRACE(__CTRACE_INPUT,
    595      1.59       rin 			    "_cursesi_addwchar: do a scroll\n");
    596      1.69     blymn 			if (!__NONL)
    597      1.69     blymn 				*x = 0;
    598      1.57     blymn 			scroll(win);
    599      1.57     blymn 		}
    600      1.32     blymn 		newx = win->maxx - 1 + win->ch_off;
    601      1.32     blymn 		if (newx > *(*lnp)->lastchp)
    602      1.32     blymn 			*(*lnp)->lastchp = newx;
    603      1.32     blymn 		__touchline(win, *y, sx, (int) win->maxx - 1);
    604      1.50     blymn 		*x = sx = 0;
    605      1.57     blymn 		if (*y != win->scr_b) {
    606      1.50     blymn 			(*y)++;
    607      1.50     blymn 		}
    608      1.50     blymn 		lp = &win->alines[*y]->line[0];
    609      1.50     blymn 		(*lnp) = win->alines[*y];
    610      1.64     blymn 		*(*lnp)->lastchp = win->ch_off + win->maxx - 1;
    611      1.32     blymn 	} else {
    612      1.53    rillig 		/* clear the remaining of the current character */
    613      1.32     blymn 		if (*x && *x < win->maxx) {
    614      1.32     blymn 			ex = sx + cw;
    615      1.35       roy 			tp = &win->alines[*y]->line[ex];
    616      1.61     blymn 			while (ex < win->maxx && tp->wcols < 0) {
    617      1.32     blymn 				__CTRACE(__CTRACE_INPUT,
    618      1.47       roy 				    "_cursesi_addwchar: clear "
    619      1.47       roy 				    "remaining of current char (%d,%d)nn",
    620      1.47       roy 				    *y, ex);
    621      1.62     blymn 				tp->ch = win->bch;
    622      1.65     blymn 				tp->cflags = CA_BACKGROUND;
    623      1.32     blymn 				if (_cursesi_copy_nsp(win->bnsp, tp) == ERR)
    624      1.32     blymn 					return ERR;
    625      1.32     blymn 				tp->attr = win->battr;
    626      1.61     blymn 				tp->wcols = win->wcols;
    627      1.32     blymn 				tp++, ex++;
    628      1.32     blymn 			}
    629      1.32     blymn 			newx = ex - 1 + win->ch_off;
    630      1.32     blymn 			if (newx > *(*lnp)->lastchp)
    631      1.32     blymn 				*(*lnp)->lastchp = newx;
    632      1.32     blymn 			__touchline(win, *y, sx, ex - 1);
    633      1.32     blymn 		}
    634      1.23     blymn 	}
    635      1.32     blymn 
    636      1.59       rin 	__CTRACE(__CTRACE_INPUT, "_cursesi_addwchar: %d : 0x%x\n",
    637      1.59       rin 	    lp->ch, lp->attr);
    638      1.59       rin 	__CTRACE(__CTRACE_INPUT,
    639      1.59       rin 	    "_cursesi_addwchar: *x = %d, *y = %d, win->maxx = %d\n",
    640      1.59       rin 	    *x, *y, win->maxx);
    641      1.45       roy 	__sync(win);
    642      1.32     blymn 	return OK;
    643      1.23     blymn #endif
    644       1.1       cgd }
    645