Home | History | Annotate | Line # | Download | only in libcurses
touchwin.c revision 1.33
      1  1.33       rin /*	$NetBSD: touchwin.c,v 1.33 2021/09/06 07:03:50 rin Exp $	*/
      2   1.7     mikel 
      3   1.1       cgd /*
      4   1.6       cgd  * Copyright (c) 1981, 1993, 1994
      5   1.4       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.19       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.7     mikel #include <sys/cdefs.h>
     33   1.1       cgd #ifndef lint
     34   1.7     mikel #if 0
     35   1.6       cgd static char sccsid[] = "@(#)touchwin.c	8.2 (Berkeley) 5/4/94";
     36   1.7     mikel #else
     37  1.33       rin __RCSID("$NetBSD: touchwin.c,v 1.33 2021/09/06 07:03:50 rin Exp $");
     38   1.7     mikel #endif
     39   1.9       mrg #endif				/* not lint */
     40   1.1       cgd 
     41   1.6       cgd #include "curses.h"
     42  1.10     blymn #include "curses_private.h"
     43   1.4       cgd 
     44  1.27     blymn static int _cursesi_touchline_force(WINDOW *, int, int, int, int);
     45  1.27     blymn 
     46   1.4       cgd /*
     47  1.29       roy  * __sync --
     48  1.29       roy  *	To be called after each window change.
     49  1.29       roy  */
     50  1.29       roy void
     51  1.29       roy __sync(WINDOW *win)
     52  1.29       roy {
     53  1.29       roy 
     54  1.29       roy 	if (win->flags & __IMMEDOK)
     55  1.29       roy 		wrefresh(win);
     56  1.29       roy 	if (win->flags & __SYNCOK)
     57  1.29       roy 		wsyncup(win);
     58  1.29       roy }
     59  1.29       roy 
     60  1.29       roy /*
     61  1.13     blymn  * is_linetouched --
     62  1.24     blymn  *	Indicate if line has been touched or not.
     63  1.13     blymn  */
     64  1.13     blymn bool
     65  1.13     blymn is_linetouched(WINDOW *win, int line)
     66  1.13     blymn {
     67  1.13     blymn 	if (line > win->maxy)
     68  1.13     blymn 		return FALSE;
     69  1.16     blymn 
     70  1.31     blymn 	__CTRACE(__CTRACE_LINE, "is_linetouched: (%p, line %d, dirty %d)\n",
     71  1.31     blymn 	    win, line, (win->alines[line]->flags & __ISDIRTY));
     72  1.30       roy 	return (win->alines[line]->flags & __ISDIRTY) != 0;
     73  1.13     blymn }
     74  1.16     blymn 
     75  1.13     blymn /*
     76  1.20       jdc  * touchline --
     77  1.20       jdc  *	Touch count lines starting at start.  This is the SUS v2 compliant
     78  1.20       jdc  *	version.
     79   1.4       cgd  */
     80   1.4       cgd int
     81  1.11     blymn touchline(WINDOW *win, int start, int count)
     82   1.4       cgd {
     83  1.23       jdc 	__CTRACE(__CTRACE_LINE, "touchline: (%p, %d, %d)\n", win, start, count);
     84  1.13     blymn 	return wtouchln(win, start, count, 1);
     85  1.13     blymn }
     86  1.13     blymn 
     87  1.13     blymn /*
     88  1.20       jdc  * wredrawln --
     89  1.20       jdc  *	Mark count lines starting at start as corrupted.  Implemented using
     90  1.20       jdc  *	wtouchln().
     91  1.20       jdc  */
     92  1.20       jdc int wredrawln(WINDOW *win, int start, int count)
     93  1.20       jdc {
     94  1.23       jdc 	__CTRACE(__CTRACE_LINE, "wredrawln: (%p, %d, %d)\n", win, start, count);
     95  1.20       jdc 	return wtouchln(win, start, count, 1);
     96  1.20       jdc }
     97  1.20       jdc 
     98  1.20       jdc /*
     99  1.13     blymn  * is_wintouched --
    100  1.24     blymn  *	Check if the window has been touched.
    101  1.13     blymn  */
    102  1.13     blymn bool
    103  1.13     blymn is_wintouched(WINDOW *win)
    104  1.13     blymn {
    105  1.13     blymn 	int y, maxy;
    106  1.10     blymn 
    107  1.31     blymn 	__CTRACE(__CTRACE_LINE, "is_wintouched: (%p, maxy %d)\n", win,
    108  1.31     blymn 	    win->maxy);
    109  1.13     blymn 	maxy = win->maxy;
    110  1.13     blymn 	for (y = 0; y < maxy; y++) {
    111  1.13     blymn 		if (is_linetouched(win, y) == TRUE)
    112  1.13     blymn 			return TRUE;
    113  1.10     blymn 	}
    114  1.10     blymn 
    115  1.13     blymn 	return FALSE;
    116   1.4       cgd }
    117   1.4       cgd 
    118   1.1       cgd /*
    119   1.3   mycroft  * touchwin --
    120   1.3   mycroft  *	Make it look like the whole window has been changed.
    121   1.1       cgd  */
    122   1.3   mycroft int
    123  1.11     blymn touchwin(WINDOW *win)
    124   1.1       cgd {
    125  1.23       jdc 	__CTRACE(__CTRACE_LINE, "touchwin: (%p)\n", win);
    126  1.20       jdc 	return wtouchln(win, 0, win->maxy, 1);
    127  1.20       jdc }
    128  1.20       jdc 
    129  1.20       jdc /*
    130  1.20       jdc  * redrawwin --
    131  1.20       jdc  *	Mark entire window as corrupted.  Implemented using wtouchln().
    132  1.20       jdc  */
    133  1.20       jdc int
    134  1.20       jdc redrawwin(WINDOW *win)
    135  1.20       jdc {
    136  1.23       jdc 	__CTRACE(__CTRACE_LINE, "redrawwin: (%p)\n", win);
    137  1.13     blymn 	return wtouchln(win, 0, win->maxy, 1);
    138  1.13     blymn }
    139  1.13     blymn 
    140  1.13     blymn /*
    141  1.13     blymn  * untouchwin --
    142  1.24     blymn  *	 Make it look like the window has not been changed.
    143  1.13     blymn  */
    144  1.13     blymn int
    145  1.13     blymn untouchwin(WINDOW *win)
    146  1.13     blymn {
    147  1.23       jdc 	__CTRACE(__CTRACE_LINE, "untouchwin: (%p)\n", win);
    148  1.13     blymn 	return wtouchln(win, 0, win->maxy, 0);
    149   1.4       cgd }
    150   1.4       cgd 
    151  1.13     blymn /*
    152  1.13     blymn  * wtouchln --
    153  1.24     blymn  *	If changed is 1 then touch n lines starting at line.  If changed
    154  1.24     blymn  *	is 0 then mark the lines as unchanged.
    155  1.13     blymn  */
    156  1.13     blymn int
    157  1.13     blymn wtouchln(WINDOW *win, int line, int n, int changed)
    158  1.13     blymn {
    159  1.14   mycroft 	int	y;
    160  1.14   mycroft 	__LINE	*wlp;
    161  1.13     blymn 
    162  1.23       jdc 	__CTRACE(__CTRACE_LINE, "wtouchln: (%p) %d, %d, %d\n",
    163  1.23       jdc 	    win, line, n, changed);
    164  1.32       uwe 	if (line < 0 || win->maxy <= line)
    165  1.32       uwe 		return ERR;
    166  1.32       uwe 	if (n < 0)
    167  1.32       uwe 		return ERR;
    168  1.32       uwe 	if (n > win->maxy - line)
    169  1.32       uwe 		n = win->maxy - line;
    170  1.32       uwe 
    171  1.13     blymn 	for (y = line; y < line + n; y++) {
    172  1.13     blymn 		if (changed == 1)
    173  1.27     blymn 			_cursesi_touchline_force(win, y, 0,
    174  1.27     blymn 			    (int) win->maxx - 1, 1);
    175  1.13     blymn 		else {
    176  1.25       roy 			wlp = win->alines[y];
    177  1.14   mycroft 			if (*wlp->firstchp >= win->ch_off &&
    178  1.14   mycroft 			    *wlp->firstchp < win->maxx + win->ch_off)
    179  1.14   mycroft 				*wlp->firstchp = win->maxx + win->ch_off;
    180  1.14   mycroft 			if (*wlp->lastchp >= win->ch_off &&
    181  1.14   mycroft 			    *wlp->lastchp < win->maxx + win->ch_off)
    182  1.14   mycroft 				*wlp->lastchp = win->ch_off;
    183  1.27     blymn 			wlp->flags &= ~(__ISDIRTY | __ISFORCED);
    184  1.13     blymn 		}
    185  1.13     blymn 	}
    186  1.13     blymn 
    187  1.13     blymn 	return OK;
    188  1.13     blymn }
    189  1.27     blymn 
    190   1.4       cgd int
    191  1.11     blymn __touchwin(WINDOW *win)
    192   1.4       cgd {
    193  1.24     blymn 	int	 y, maxy;
    194   1.4       cgd 
    195  1.23       jdc 	__CTRACE(__CTRACE_LINE, "__touchwin: (%p)\n", win);
    196   1.4       cgd 	maxy = win->maxy;
    197   1.1       cgd 	for (y = 0; y < maxy; y++)
    198  1.15   mycroft 		__touchline(win, y, 0, (int) win->maxx - 1);
    199  1.30       roy 	return OK;
    200   1.1       cgd }
    201   1.1       cgd 
    202   1.3   mycroft int
    203  1.15   mycroft __touchline(WINDOW *win, int y, int sx, int ex)
    204   1.1       cgd {
    205  1.30       roy 
    206  1.30       roy 	return _cursesi_touchline_force(win, y, sx, ex, 0);
    207  1.27     blymn }
    208  1.27     blymn 
    209  1.27     blymn /*
    210  1.27     blymn  * Touch line y on window win starting from column sx and ending at
    211  1.27     blymn  * column ex.  If force is 1 then we mark this line as a forced update
    212  1.27     blymn  * which will bypass screen optimisation in the refresh code to rewrite
    213  1.27     blymn  * this line unconditionally (even if refresh thinks the screen matches
    214  1.27     blymn  * what is in the virtscr)
    215  1.27     blymn  */
    216  1.27     blymn static int
    217  1.27     blymn _cursesi_touchline_force(WINDOW *win, int y, int sx, int ex, int force)
    218  1.27     blymn {
    219  1.30       roy 
    220  1.28       jdc 	__CTRACE(__CTRACE_LINE, "__touchline: (%p, %d, %d, %d, %d)\n",
    221  1.28       jdc 	    win, y, sx, ex, force);
    222  1.23       jdc 	__CTRACE(__CTRACE_LINE, "__touchline: first = %d, last = %d\n",
    223  1.25       roy 	    *win->alines[y]->firstchp, *win->alines[y]->lastchp);
    224   1.4       cgd 	sx += win->ch_off;
    225   1.4       cgd 	ex += win->ch_off;
    226  1.27     blymn 	win->alines[y]->flags |= __ISDIRTY;
    227  1.27     blymn 	if (force == 1)
    228  1.27     blymn 		win->alines[y]->flags |= __ISFORCED;
    229  1.12       jdc 	/* firstchp/lastchp are shared between parent window and sub-window. */
    230  1.25       roy 	if (*win->alines[y]->firstchp > sx)
    231  1.25       roy 		*win->alines[y]->firstchp = sx;
    232  1.25       roy 	if (*win->alines[y]->lastchp < ex)
    233  1.25       roy 		*win->alines[y]->lastchp = ex;
    234  1.23       jdc 	__CTRACE(__CTRACE_LINE, "__touchline: first = %d, last = %d\n",
    235  1.25       roy 	    *win->alines[y]->firstchp, *win->alines[y]->lastchp);
    236  1.30       roy 	return OK;
    237   1.1       cgd }
    238  1.26  drochner 
    239  1.26  drochner void
    240  1.26  drochner wsyncup(WINDOW *win)
    241  1.26  drochner {
    242  1.26  drochner 
    243  1.26  drochner 	do {
    244  1.27     blymn 		__touchwin(win);
    245  1.26  drochner 		win = win->orig;
    246  1.26  drochner 	} while (win);
    247  1.26  drochner }
    248  1.26  drochner 
    249  1.26  drochner void
    250  1.26  drochner wsyncdown(WINDOW *win)
    251  1.26  drochner {
    252  1.26  drochner 	WINDOW *w = win->orig;
    253  1.26  drochner 
    254  1.26  drochner 	while (w) {
    255  1.26  drochner 		if (is_wintouched(w)) {
    256  1.27     blymn 			__touchwin(win);
    257  1.26  drochner 			break;
    258  1.26  drochner 		}
    259  1.26  drochner 		w = w->orig;
    260  1.26  drochner 	}
    261  1.26  drochner }
    262