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