xref: /freebsd/contrib/ncurses/panel/panel.c (revision 0e3d540892016a47f6a68ec9ba2879d35ce5f7c2)
10e3d5408SPeter Wemm /****************************************************************************
20e3d5408SPeter Wemm  * Copyright (c) 1998 Free Software Foundation, Inc.                        *
30e3d5408SPeter Wemm  *                                                                          *
40e3d5408SPeter Wemm  * Permission is hereby granted, free of charge, to any person obtaining a  *
50e3d5408SPeter Wemm  * copy of this software and associated documentation files (the            *
60e3d5408SPeter Wemm  * "Software"), to deal in the Software without restriction, including      *
70e3d5408SPeter Wemm  * without limitation the rights to use, copy, modify, merge, publish,      *
80e3d5408SPeter Wemm  * distribute, distribute with modifications, sublicense, and/or sell       *
90e3d5408SPeter Wemm  * copies of the Software, and to permit persons to whom the Software is    *
100e3d5408SPeter Wemm  * furnished to do so, subject to the following conditions:                 *
110e3d5408SPeter Wemm  *                                                                          *
120e3d5408SPeter Wemm  * The above copyright notice and this permission notice shall be included  *
130e3d5408SPeter Wemm  * in all copies or substantial portions of the Software.                   *
140e3d5408SPeter Wemm  *                                                                          *
150e3d5408SPeter Wemm  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS  *
160e3d5408SPeter Wemm  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF               *
170e3d5408SPeter Wemm  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.   *
180e3d5408SPeter Wemm  * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,   *
190e3d5408SPeter Wemm  * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR    *
200e3d5408SPeter Wemm  * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR    *
210e3d5408SPeter Wemm  * THE USE OR OTHER DEALINGS IN THE SOFTWARE.                               *
220e3d5408SPeter Wemm  *                                                                          *
230e3d5408SPeter Wemm  * Except as contained in this notice, the name(s) of the above copyright   *
240e3d5408SPeter Wemm  * holders shall not be used in advertising or otherwise to promote the     *
250e3d5408SPeter Wemm  * sale, use or other dealings in this Software without prior written       *
260e3d5408SPeter Wemm  * authorization.                                                           *
270e3d5408SPeter Wemm  ****************************************************************************/
280e3d5408SPeter Wemm 
290e3d5408SPeter Wemm /****************************************************************************
300e3d5408SPeter Wemm  *  Author: Zeyd M. Ben-Halim <zmbenhal@netcom.com> 1995                    *
310e3d5408SPeter Wemm  *     and: Eric S. Raymond <esr@snark.thyrsus.com>                         *
320e3d5408SPeter Wemm  ****************************************************************************/
330e3d5408SPeter Wemm 
340e3d5408SPeter Wemm /* panel.c -- implementation of panels library, some core routines */
350e3d5408SPeter Wemm #include "panel.priv.h"
360e3d5408SPeter Wemm 
370e3d5408SPeter Wemm MODULE_ID("$Id: panel.c,v 1.16 1998/09/19 21:26:31 Todd.Miller Exp $")
380e3d5408SPeter Wemm 
390e3d5408SPeter Wemm #ifdef TRACE
400e3d5408SPeter Wemm #ifndef TRACE_TXT
410e3d5408SPeter Wemm const char *_nc_my_visbuf(const void *ptr)
420e3d5408SPeter Wemm {
430e3d5408SPeter Wemm 	char temp[32];
440e3d5408SPeter Wemm 	if (ptr != 0)
450e3d5408SPeter Wemm 		sprintf(temp, "ptr:%p", ptr);
460e3d5408SPeter Wemm 	else
470e3d5408SPeter Wemm 		strcpy(temp, "<null>");
480e3d5408SPeter Wemm 	return _nc_visbuf(temp);
490e3d5408SPeter Wemm }
500e3d5408SPeter Wemm #endif
510e3d5408SPeter Wemm #endif
520e3d5408SPeter Wemm 
530e3d5408SPeter Wemm 
540e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
550e3d5408SPeter Wemm 	dPanel(text,pan)
560e3d5408SPeter Wemm --------------------------------------------------------------------------*/
570e3d5408SPeter Wemm #ifdef TRACE
580e3d5408SPeter Wemm void
590e3d5408SPeter Wemm _nc_dPanel(const char *text, const PANEL *pan)
600e3d5408SPeter Wemm {
610e3d5408SPeter Wemm 	_tracef("%s id=%s b=%s a=%s y=%d x=%d",
620e3d5408SPeter Wemm 		text, USER_PTR(pan->user),
630e3d5408SPeter Wemm 		(pan->below) ?  USER_PTR(pan->below->user) : "--",
640e3d5408SPeter Wemm 		(pan->above) ?  USER_PTR(pan->above->user) : "--",
650e3d5408SPeter Wemm 		pan->wstarty, pan->wstartx);
660e3d5408SPeter Wemm }
670e3d5408SPeter Wemm #endif
680e3d5408SPeter Wemm 
690e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
700e3d5408SPeter Wemm 	dStack(fmt,num,pan)
710e3d5408SPeter Wemm --------------------------------------------------------------------------*/
720e3d5408SPeter Wemm #ifdef TRACE
730e3d5408SPeter Wemm void
740e3d5408SPeter Wemm _nc_dStack(const char *fmt, int num, const PANEL *pan)
750e3d5408SPeter Wemm {
760e3d5408SPeter Wemm   char s80[80];
770e3d5408SPeter Wemm 
780e3d5408SPeter Wemm   sprintf(s80,fmt,num,pan);
790e3d5408SPeter Wemm   _tracef("%s b=%s t=%s",s80,
800e3d5408SPeter Wemm 	  (_nc_bottom_panel) ?  USER_PTR(_nc_bottom_panel->user) : "--",
810e3d5408SPeter Wemm 	  (_nc_top_panel)    ?  USER_PTR(_nc_top_panel->user)    : "--");
820e3d5408SPeter Wemm   if(pan)
830e3d5408SPeter Wemm     _tracef("pan id=%s", USER_PTR(pan->user));
840e3d5408SPeter Wemm   pan = _nc_bottom_panel;
850e3d5408SPeter Wemm   while(pan)
860e3d5408SPeter Wemm     {
870e3d5408SPeter Wemm       dPanel("stk",pan);
880e3d5408SPeter Wemm       pan = pan->above;
890e3d5408SPeter Wemm     }
900e3d5408SPeter Wemm }
910e3d5408SPeter Wemm #endif
920e3d5408SPeter Wemm 
930e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
940e3d5408SPeter Wemm 	Wnoutrefresh(pan) - debugging hook for wnoutrefresh
950e3d5408SPeter Wemm --------------------------------------------------------------------------*/
960e3d5408SPeter Wemm #ifdef TRACE
970e3d5408SPeter Wemm void
980e3d5408SPeter Wemm _nc_Wnoutrefresh(const PANEL *pan)
990e3d5408SPeter Wemm {
1000e3d5408SPeter Wemm   dPanel("wnoutrefresh",pan);
1010e3d5408SPeter Wemm   wnoutrefresh(pan->win);
1020e3d5408SPeter Wemm }
1030e3d5408SPeter Wemm #endif
1040e3d5408SPeter Wemm 
1050e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
1060e3d5408SPeter Wemm 	Touchpan(pan)
1070e3d5408SPeter Wemm --------------------------------------------------------------------------*/
1080e3d5408SPeter Wemm #ifdef TRACE
1090e3d5408SPeter Wemm void
1100e3d5408SPeter Wemm _nc_Touchpan(const PANEL *pan)
1110e3d5408SPeter Wemm {
1120e3d5408SPeter Wemm   dPanel("Touchpan",pan);
1130e3d5408SPeter Wemm   touchwin(pan->win);
1140e3d5408SPeter Wemm }
1150e3d5408SPeter Wemm #endif
1160e3d5408SPeter Wemm 
1170e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
1180e3d5408SPeter Wemm 	Touchline(pan,start,count)
1190e3d5408SPeter Wemm --------------------------------------------------------------------------*/
1200e3d5408SPeter Wemm #ifdef TRACE
1210e3d5408SPeter Wemm void
1220e3d5408SPeter Wemm _nc_Touchline(const PANEL *pan, int start, int count)
1230e3d5408SPeter Wemm {
1240e3d5408SPeter Wemm   char s80[80];
1250e3d5408SPeter Wemm   sprintf(s80,"Touchline s=%d c=%d",start,count);
1260e3d5408SPeter Wemm   dPanel(s80,pan);
1270e3d5408SPeter Wemm   touchline(pan->win,start,count);
1280e3d5408SPeter Wemm }
1290e3d5408SPeter Wemm #endif
1300e3d5408SPeter Wemm 
1310e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
1320e3d5408SPeter Wemm 	__panels_overlapped(pan1,pan2) - check panel overlapped
1330e3d5408SPeter Wemm --------------------------------------------------------------------------*/
1340e3d5408SPeter Wemm static INLINE bool
1350e3d5408SPeter Wemm __panels_overlapped(register const PANEL *pan1, register const PANEL *pan2)
1360e3d5408SPeter Wemm {
1370e3d5408SPeter Wemm   if(!pan1 || !pan2)
1380e3d5408SPeter Wemm     return(FALSE);
1390e3d5408SPeter Wemm 
1400e3d5408SPeter Wemm   dBug(("__panels_overlapped %s %s", USER_PTR(pan1->user), USER_PTR(pan2->user)));
1410e3d5408SPeter Wemm   /* pan1 intersects with pan2 ? */
1420e3d5408SPeter Wemm   if( (((pan1->wstarty >= pan2->wstarty) && (pan1->wstarty < pan2->wendy)) ||
1430e3d5408SPeter Wemm        ((pan2->wstarty >= pan1->wstarty) && (pan2->wstarty < pan1->wendy))) &&
1440e3d5408SPeter Wemm       (((pan1->wstartx >= pan2->wstartx) && (pan1->wstartx < pan2->wendx)) ||
1450e3d5408SPeter Wemm        ((pan2->wstartx >= pan1->wstartx) && (pan2->wstartx < pan1->wendx)))
1460e3d5408SPeter Wemm       ) return(TRUE);
1470e3d5408SPeter Wemm   else {
1480e3d5408SPeter Wemm     dBug(("  no"));
1490e3d5408SPeter Wemm     return(FALSE);
1500e3d5408SPeter Wemm   }
1510e3d5408SPeter Wemm }
1520e3d5408SPeter Wemm 
1530e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
1540e3d5408SPeter Wemm 	_nc_free_obscure(pan)
1550e3d5408SPeter Wemm --------------------------------------------------------------------------*/
1560e3d5408SPeter Wemm void
1570e3d5408SPeter Wemm _nc_free_obscure(PANEL *pan)
1580e3d5408SPeter Wemm {
1590e3d5408SPeter Wemm   PANELCONS *tobs = pan->obscure;			/* "this" one */
1600e3d5408SPeter Wemm   PANELCONS *nobs;					/* "next" one */
1610e3d5408SPeter Wemm 
1620e3d5408SPeter Wemm   while(tobs)
1630e3d5408SPeter Wemm     {
1640e3d5408SPeter Wemm       nobs = tobs->above;
1650e3d5408SPeter Wemm       free((char *)tobs);
1660e3d5408SPeter Wemm       tobs = nobs;
1670e3d5408SPeter Wemm     }
1680e3d5408SPeter Wemm   pan->obscure = (PANELCONS *)0;
1690e3d5408SPeter Wemm }
1700e3d5408SPeter Wemm 
1710e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
1720e3d5408SPeter Wemm 	__override(pan,show)
1730e3d5408SPeter Wemm --------------------------------------------------------------------------*/
1740e3d5408SPeter Wemm void
1750e3d5408SPeter Wemm _nc_override(const PANEL *pan, int show)
1760e3d5408SPeter Wemm {
1770e3d5408SPeter Wemm   int y;
1780e3d5408SPeter Wemm   PANEL *pan2;
1790e3d5408SPeter Wemm   PANELCONS *tobs = pan->obscure;			   /* "this" one */
1800e3d5408SPeter Wemm 
1810e3d5408SPeter Wemm   dBug(("_nc_override %s,%d", USER_PTR(pan->user),show));
1820e3d5408SPeter Wemm 
1830e3d5408SPeter Wemm   switch (show)
1840e3d5408SPeter Wemm     {
1850e3d5408SPeter Wemm     case P_TOUCH:
1860e3d5408SPeter Wemm       Touchpan(pan);
1870e3d5408SPeter Wemm       /* The following while loop will now mark all panel window lines
1880e3d5408SPeter Wemm        * obscured by use or obscuring us as touched, so they will be
1890e3d5408SPeter Wemm        * updated.
1900e3d5408SPeter Wemm        */
1910e3d5408SPeter Wemm       break;
1920e3d5408SPeter Wemm     case P_UPDATE:
1930e3d5408SPeter Wemm       while(tobs && (tobs->pan != pan))
1940e3d5408SPeter Wemm 	tobs = tobs->above;
1950e3d5408SPeter Wemm       /* The next loop will now only go through the panels obscuring pan;
1960e3d5408SPeter Wemm        * it updates all the lines in the obscuring panels in sync. with
1970e3d5408SPeter Wemm        * the lines touched in pan itself. This is called in update_panels()
1980e3d5408SPeter Wemm        * in a loop from the bottom_panel to the top_panel, resulting in
1990e3d5408SPeter Wemm        * the desired update effect.
2000e3d5408SPeter Wemm        */
2010e3d5408SPeter Wemm       break;
2020e3d5408SPeter Wemm     default:
2030e3d5408SPeter Wemm       return;
2040e3d5408SPeter Wemm     }
2050e3d5408SPeter Wemm 
2060e3d5408SPeter Wemm   while(tobs)
2070e3d5408SPeter Wemm     {
2080e3d5408SPeter Wemm       if((pan2 = tobs->pan) != pan) {
2090e3d5408SPeter Wemm 	dBug(("test obs pan=%s pan2=%s", USER_PTR(pan->user), USER_PTR(pan2->user)));
2100e3d5408SPeter Wemm 	for(y = pan->wstarty; y < pan->wendy; y++) {
2110e3d5408SPeter Wemm 	  if( (y >= pan2->wstarty) && (y < pan2->wendy) &&
2120e3d5408SPeter Wemm 	      ((is_linetouched(pan->win,y - pan->wstarty) == TRUE)) )
2130e3d5408SPeter Wemm 	    Touchline(pan2,y - pan2->wstarty,1);
2140e3d5408SPeter Wemm 	}
2150e3d5408SPeter Wemm       }
2160e3d5408SPeter Wemm       tobs = tobs->above;
2170e3d5408SPeter Wemm     }
2180e3d5408SPeter Wemm }
2190e3d5408SPeter Wemm 
2200e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
2210e3d5408SPeter Wemm 	__calculate_obscure()
2220e3d5408SPeter Wemm --------------------------------------------------------------------------*/
2230e3d5408SPeter Wemm void
2240e3d5408SPeter Wemm _nc_calculate_obscure(void)
2250e3d5408SPeter Wemm {
2260e3d5408SPeter Wemm   PANEL *pan;
2270e3d5408SPeter Wemm   PANEL *pan2;
2280e3d5408SPeter Wemm   PANELCONS *tobs;			/* "this" one */
2290e3d5408SPeter Wemm   PANELCONS *lobs = (PANELCONS *)0;	/* last one */
2300e3d5408SPeter Wemm 
2310e3d5408SPeter Wemm   pan = _nc_bottom_panel;
2320e3d5408SPeter Wemm   while(pan)
2330e3d5408SPeter Wemm     {
2340e3d5408SPeter Wemm       if(pan->obscure)
2350e3d5408SPeter Wemm 	_nc_free_obscure(pan);
2360e3d5408SPeter Wemm       dBug(("--> __calculate_obscure %s", USER_PTR(pan->user)));
2370e3d5408SPeter Wemm       lobs = (PANELCONS *)0;		/* last one */
2380e3d5408SPeter Wemm       pan2 = _nc_bottom_panel;
2390e3d5408SPeter Wemm       /* This loop builds a list of panels obsured by pan or obscuring
2400e3d5408SPeter Wemm 	 pan; pan itself is in the list; all panels before pan are
2410e3d5408SPeter Wemm 	 obscured by pan, all panels after pan are obscuring pan. */
2420e3d5408SPeter Wemm       while(pan2)
2430e3d5408SPeter Wemm 	{
2440e3d5408SPeter Wemm 	  if(__panels_overlapped(pan,pan2))
2450e3d5408SPeter Wemm 	    {
2460e3d5408SPeter Wemm 	      if(!(tobs = (PANELCONS *)malloc(sizeof(PANELCONS))))
2470e3d5408SPeter Wemm 		return;
2480e3d5408SPeter Wemm 	      tobs->pan = pan2;
2490e3d5408SPeter Wemm 	      dPanel("obscured",pan2);
2500e3d5408SPeter Wemm 	      tobs->above = (PANELCONS *)0;
2510e3d5408SPeter Wemm 	      if(lobs)
2520e3d5408SPeter Wemm 		lobs->above = tobs;
2530e3d5408SPeter Wemm 	      else
2540e3d5408SPeter Wemm 		pan->obscure = tobs;
2550e3d5408SPeter Wemm 	      lobs  = tobs;
2560e3d5408SPeter Wemm 	    }
2570e3d5408SPeter Wemm 	  pan2 = pan2->above;
2580e3d5408SPeter Wemm 	}
2590e3d5408SPeter Wemm       _nc_override(pan,P_TOUCH);
2600e3d5408SPeter Wemm       pan = pan->above;
2610e3d5408SPeter Wemm     }
2620e3d5408SPeter Wemm }
2630e3d5408SPeter Wemm 
2640e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
2650e3d5408SPeter Wemm 	_nc_panel_is_linked(pan) - check to see if panel is in the stack
2660e3d5408SPeter Wemm --------------------------------------------------------------------------*/
2670e3d5408SPeter Wemm bool
2680e3d5408SPeter Wemm _nc_panel_is_linked(const PANEL *pan)
2690e3d5408SPeter Wemm {
2700e3d5408SPeter Wemm   /* This works! The only case where it would fail is, when the list has
2710e3d5408SPeter Wemm      only one element. But this could only be the pseudo panel at the bottom */
2720e3d5408SPeter Wemm   return ( ((pan->above!=(PANEL *)0) ||
2730e3d5408SPeter Wemm 	    (pan->below!=(PANEL *)0) ||
2740e3d5408SPeter Wemm 	    (pan==_nc_bottom_panel)) ? TRUE : FALSE );
2750e3d5408SPeter Wemm }
2760e3d5408SPeter Wemm 
2770e3d5408SPeter Wemm 
2780e3d5408SPeter Wemm /*+-------------------------------------------------------------------------
2790e3d5408SPeter Wemm 	__panel_link_bottom(pan) - link panel into stack at bottom
2800e3d5408SPeter Wemm --------------------------------------------------------------------------*/
2810e3d5408SPeter Wemm void
2820e3d5408SPeter Wemm _nc_panel_link_bottom(PANEL *pan)
2830e3d5408SPeter Wemm {
2840e3d5408SPeter Wemm #ifdef TRACE
2850e3d5408SPeter Wemm   dStack("<lb%d>",1,pan);
2860e3d5408SPeter Wemm   if(_nc_panel_is_linked(pan))
2870e3d5408SPeter Wemm     return;
2880e3d5408SPeter Wemm #endif
2890e3d5408SPeter Wemm 
2900e3d5408SPeter Wemm   pan->above = (PANEL *)0;
2910e3d5408SPeter Wemm   pan->below = (PANEL *)0;
2920e3d5408SPeter Wemm   if(_nc_bottom_panel)
2930e3d5408SPeter Wemm     { /* the stdscr pseudo panel always stays real bottom;
2940e3d5408SPeter Wemm          so we insert after bottom panel*/
2950e3d5408SPeter Wemm       pan->below = _nc_bottom_panel;
2960e3d5408SPeter Wemm       pan->above = _nc_bottom_panel->above;
2970e3d5408SPeter Wemm       if (pan->above)
2980e3d5408SPeter Wemm 	pan->above->below = pan;
2990e3d5408SPeter Wemm       _nc_bottom_panel->above = pan;
3000e3d5408SPeter Wemm     }
3010e3d5408SPeter Wemm   else
3020e3d5408SPeter Wemm     _nc_bottom_panel = pan;
3030e3d5408SPeter Wemm   if(!_nc_top_panel)
3040e3d5408SPeter Wemm     _nc_top_panel = pan;
3050e3d5408SPeter Wemm   assert(_nc_bottom_panel == _nc_stdscr_pseudo_panel);
3060e3d5408SPeter Wemm   _nc_calculate_obscure();
3070e3d5408SPeter Wemm   dStack("<lb%d>",9,pan);
3080e3d5408SPeter Wemm }
309