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