Home | History | Annotate | Line # | Download | only in libcurses
touchwin.c revision 1.31
      1  1.31     blymn /*	$NetBSD: touchwin.c,v 1.31 2019/04/24 07:09:44 blymn 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.31     blymn __RCSID("$NetBSD: touchwin.c,v 1.31 2019/04/24 07:09:44 blymn 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 #ifdef DEBUG
     71  1.31     blymn 	__CTRACE(__CTRACE_LINE, "is_linetouched: (%p, line %d, dirty %d)\n",
     72  1.31     blymn 	    win, line, (win->alines[line]->flags & __ISDIRTY));
     73  1.31     blymn #endif
     74  1.30       roy 	return (win->alines[line]->flags & __ISDIRTY) != 0;
     75  1.13     blymn }
     76  1.16     blymn 
     77  1.13     blymn /*
     78  1.20       jdc  * touchline --
     79  1.20       jdc  *	Touch count lines starting at start.  This is the SUS v2 compliant
     80  1.20       jdc  *	version.
     81   1.4       cgd  */
     82   1.4       cgd int
     83  1.11     blymn touchline(WINDOW *win, int start, int count)
     84   1.4       cgd {
     85  1.22       jdc #ifdef DEBUG
     86  1.23       jdc 	__CTRACE(__CTRACE_LINE, "touchline: (%p, %d, %d)\n", win, start, count);
     87  1.22       jdc #endif
     88  1.13     blymn 	return wtouchln(win, start, count, 1);
     89  1.13     blymn }
     90  1.13     blymn 
     91  1.13     blymn /*
     92  1.20       jdc  * wredrawln --
     93  1.20       jdc  *	Mark count lines starting at start as corrupted.  Implemented using
     94  1.20       jdc  *	wtouchln().
     95  1.20       jdc  */
     96  1.20       jdc int wredrawln(WINDOW *win, int start, int count)
     97  1.20       jdc {
     98  1.20       jdc #ifdef DEBUG
     99  1.23       jdc 	__CTRACE(__CTRACE_LINE, "wredrawln: (%p, %d, %d)\n", win, start, count);
    100  1.20       jdc #endif
    101  1.20       jdc 	return wtouchln(win, start, count, 1);
    102  1.20       jdc }
    103  1.20       jdc 
    104  1.20       jdc /*
    105  1.13     blymn  * is_wintouched --
    106  1.24     blymn  *	Check if the window has been touched.
    107  1.13     blymn  */
    108  1.13     blymn bool
    109  1.13     blymn is_wintouched(WINDOW *win)
    110  1.13     blymn {
    111  1.13     blymn 	int y, maxy;
    112  1.10     blymn 
    113  1.31     blymn #ifdef DEBUG
    114  1.31     blymn 	__CTRACE(__CTRACE_LINE, "is_wintouched: (%p, maxy %d)\n", win,
    115  1.31     blymn 	    win->maxy);
    116  1.31     blymn #endif
    117  1.13     blymn 	maxy = win->maxy;
    118  1.13     blymn 	for (y = 0; y < maxy; y++) {
    119  1.13     blymn 		if (is_linetouched(win, y) == TRUE)
    120  1.13     blymn 			return TRUE;
    121  1.10     blymn 	}
    122  1.10     blymn 
    123  1.13     blymn 	return FALSE;
    124   1.4       cgd }
    125   1.4       cgd 
    126   1.1       cgd /*
    127   1.3   mycroft  * touchwin --
    128   1.3   mycroft  *	Make it look like the whole window has been changed.
    129   1.1       cgd  */
    130   1.3   mycroft int
    131  1.11     blymn touchwin(WINDOW *win)
    132   1.1       cgd {
    133   1.3   mycroft #ifdef DEBUG
    134  1.23       jdc 	__CTRACE(__CTRACE_LINE, "touchwin: (%p)\n", win);
    135  1.20       jdc #endif
    136  1.20       jdc 	return wtouchln(win, 0, win->maxy, 1);
    137  1.20       jdc }
    138  1.20       jdc 
    139  1.20       jdc /*
    140  1.20       jdc  * redrawwin --
    141  1.20       jdc  *	Mark entire window as corrupted.  Implemented using wtouchln().
    142  1.20       jdc  */
    143  1.20       jdc int
    144  1.20       jdc redrawwin(WINDOW *win)
    145  1.20       jdc {
    146  1.20       jdc #ifdef DEBUG
    147  1.23       jdc 	__CTRACE(__CTRACE_LINE, "redrawwin: (%p)\n", win);
    148   1.4       cgd #endif
    149  1.13     blymn 	return wtouchln(win, 0, win->maxy, 1);
    150  1.13     blymn }
    151  1.13     blymn 
    152  1.13     blymn /*
    153  1.13     blymn  * untouchwin --
    154  1.24     blymn  *	 Make it look like the window has not been changed.
    155  1.13     blymn  */
    156  1.13     blymn int
    157  1.13     blymn untouchwin(WINDOW *win)
    158  1.13     blymn {
    159  1.22       jdc #ifdef DEBUG
    160  1.23       jdc 	__CTRACE(__CTRACE_LINE, "untouchwin: (%p)\n", win);
    161  1.22       jdc #endif
    162  1.13     blymn 	return wtouchln(win, 0, win->maxy, 0);
    163   1.4       cgd }
    164   1.4       cgd 
    165  1.13     blymn /*
    166  1.13     blymn  * wtouchln --
    167  1.24     blymn  *	If changed is 1 then touch n lines starting at line.  If changed
    168  1.24     blymn  *	is 0 then mark the lines as unchanged.
    169  1.13     blymn  */
    170  1.13     blymn int
    171  1.13     blymn wtouchln(WINDOW *win, int line, int n, int changed)
    172  1.13     blymn {
    173  1.14   mycroft 	int	y;
    174  1.14   mycroft 	__LINE	*wlp;
    175  1.13     blymn 
    176  1.17       jdc #ifdef DEBUG
    177  1.23       jdc 	__CTRACE(__CTRACE_LINE, "wtouchln: (%p) %d, %d, %d\n",
    178  1.23       jdc 	    win, line, n, changed);
    179  1.17       jdc #endif
    180  1.21       jdc 	if (line + n > win->maxy)
    181  1.21       jdc 		line = win->maxy - n;
    182  1.13     blymn 	for (y = line; y < line + n; y++) {
    183  1.13     blymn 		if (changed == 1)
    184  1.27     blymn 			_cursesi_touchline_force(win, y, 0,
    185  1.27     blymn 			    (int) win->maxx - 1, 1);
    186  1.13     blymn 		else {
    187  1.25       roy 			wlp = win->alines[y];
    188  1.14   mycroft 			if (*wlp->firstchp >= win->ch_off &&
    189  1.14   mycroft 			    *wlp->firstchp < win->maxx + win->ch_off)
    190  1.14   mycroft 				*wlp->firstchp = win->maxx + win->ch_off;
    191  1.14   mycroft 			if (*wlp->lastchp >= win->ch_off &&
    192  1.14   mycroft 			    *wlp->lastchp < win->maxx + win->ch_off)
    193  1.14   mycroft 				*wlp->lastchp = win->ch_off;
    194  1.27     blymn 			wlp->flags &= ~(__ISDIRTY | __ISFORCED);
    195  1.13     blymn 		}
    196  1.13     blymn 	}
    197  1.13     blymn 
    198  1.13     blymn 	return OK;
    199  1.13     blymn }
    200  1.27     blymn 
    201   1.4       cgd int
    202  1.11     blymn __touchwin(WINDOW *win)
    203   1.4       cgd {
    204  1.24     blymn 	int	 y, maxy;
    205   1.4       cgd 
    206   1.4       cgd #ifdef DEBUG
    207  1.23       jdc 	__CTRACE(__CTRACE_LINE, "__touchwin: (%p)\n", win);
    208   1.3   mycroft #endif
    209   1.4       cgd 	maxy = win->maxy;
    210   1.1       cgd 	for (y = 0; y < maxy; y++)
    211  1.15   mycroft 		__touchline(win, y, 0, (int) win->maxx - 1);
    212  1.30       roy 	return OK;
    213   1.1       cgd }
    214   1.1       cgd 
    215   1.3   mycroft int
    216  1.15   mycroft __touchline(WINDOW *win, int y, int sx, int ex)
    217   1.1       cgd {
    218  1.30       roy 
    219  1.30       roy 	return _cursesi_touchline_force(win, y, sx, ex, 0);
    220  1.27     blymn }
    221  1.27     blymn 
    222  1.27     blymn /*
    223  1.27     blymn  * Touch line y on window win starting from column sx and ending at
    224  1.27     blymn  * column ex.  If force is 1 then we mark this line as a forced update
    225  1.27     blymn  * which will bypass screen optimisation in the refresh code to rewrite
    226  1.27     blymn  * this line unconditionally (even if refresh thinks the screen matches
    227  1.27     blymn  * what is in the virtscr)
    228  1.27     blymn  */
    229  1.27     blymn static int
    230  1.27     blymn _cursesi_touchline_force(WINDOW *win, int y, int sx, int ex, int force)
    231  1.27     blymn {
    232  1.30       roy 
    233   1.3   mycroft #ifdef DEBUG
    234  1.28       jdc 	__CTRACE(__CTRACE_LINE, "__touchline: (%p, %d, %d, %d, %d)\n",
    235  1.28       jdc 	    win, y, sx, ex, force);
    236  1.23       jdc 	__CTRACE(__CTRACE_LINE, "__touchline: first = %d, last = %d\n",
    237  1.25       roy 	    *win->alines[y]->firstchp, *win->alines[y]->lastchp);
    238   1.3   mycroft #endif
    239   1.4       cgd 	sx += win->ch_off;
    240   1.4       cgd 	ex += win->ch_off;
    241  1.27     blymn 	win->alines[y]->flags |= __ISDIRTY;
    242  1.27     blymn 	if (force == 1)
    243  1.27     blymn 		win->alines[y]->flags |= __ISFORCED;
    244  1.12       jdc 	/* firstchp/lastchp are shared between parent window and sub-window. */
    245  1.25       roy 	if (*win->alines[y]->firstchp > sx)
    246  1.25       roy 		*win->alines[y]->firstchp = sx;
    247  1.25       roy 	if (*win->alines[y]->lastchp < ex)
    248  1.25       roy 		*win->alines[y]->lastchp = ex;
    249   1.3   mycroft #ifdef DEBUG
    250  1.23       jdc 	__CTRACE(__CTRACE_LINE, "__touchline: first = %d, last = %d\n",
    251  1.25       roy 	    *win->alines[y]->firstchp, *win->alines[y]->lastchp);
    252   1.3   mycroft #endif
    253  1.30       roy 	return OK;
    254   1.1       cgd }
    255  1.26  drochner 
    256  1.26  drochner void
    257  1.26  drochner wsyncup(WINDOW *win)
    258  1.26  drochner {
    259  1.26  drochner 
    260  1.26  drochner 	do {
    261  1.27     blymn 		__touchwin(win);
    262  1.26  drochner 		win = win->orig;
    263  1.26  drochner 	} while (win);
    264  1.26  drochner }
    265  1.26  drochner 
    266  1.26  drochner void
    267  1.26  drochner wsyncdown(WINDOW *win)
    268  1.26  drochner {
    269  1.26  drochner 	WINDOW *w = win->orig;
    270  1.26  drochner 
    271  1.26  drochner 	while (w) {
    272  1.26  drochner 		if (is_wintouched(w)) {
    273  1.27     blymn 			__touchwin(win);
    274  1.26  drochner 			break;
    275  1.26  drochner 		}
    276  1.26  drochner 		w = w->orig;
    277  1.26  drochner 	}
    278  1.26  drochner }
    279