mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-10 05:26:10 +01:00
ecacb0587a
My version of curses doesn't seem to have its own SIGWINCH handler that does the right thing, so use a pipe to let my handler signal to have work done outside the interrupt context (when window resizing etc. that might call malloc() isn't safe.)
248 lines
6.5 KiB
C
248 lines
6.5 KiB
C
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
|
/*
|
|
* Copyright 1997-2020 by Eric House (xwords@eehouse.org). All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
#include <glib.h>
|
|
#include <ncurses.h>
|
|
|
|
#include "cursesmenu.h"
|
|
#include "xptypes.h"
|
|
#include "comtypes.h"
|
|
#include "cursesmain.h"
|
|
#include "linuxmain.h"
|
|
#include "gsrcwrap.h"
|
|
|
|
struct CursesMenuState {
|
|
WINDOW* menuWin;
|
|
WINDOW* mainWin;
|
|
GSList* menuLists;
|
|
bool altPressed;
|
|
};
|
|
|
|
static bool handleKeyEvent( CursesMenuState* state, char ch, bool altPressed );
|
|
|
|
static gboolean
|
|
handle_stdin( GIOChannel* XP_UNUSED_DBG(source), GIOCondition condition,
|
|
gpointer data )
|
|
{
|
|
if ( 0 != (G_IO_IN & condition) ) {
|
|
#ifdef DEBUG
|
|
gint fd = g_io_channel_unix_get_fd( source );
|
|
XP_ASSERT( 0 == fd );
|
|
#endif
|
|
CursesMenuState* state = (CursesMenuState*)data;
|
|
int ch = wgetch( state->menuWin );
|
|
// Alt (at least pressed with <ret>) comes in as a separate keypress
|
|
// immediately before. So don't distribute it, but instead OR a
|
|
// special bit into the key sent out.
|
|
if ( ch == 0x1b ) {
|
|
state->altPressed = true;
|
|
} else {
|
|
handleKeyEvent( state, ch, state->altPressed );
|
|
state->altPressed = false;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
sizeWindow( CursesMenuState* state )
|
|
{
|
|
int width, height;
|
|
getmaxyx( state->mainWin, height, width );
|
|
if ( !!state->menuWin ) {
|
|
werase( state->menuWin );
|
|
wrefresh( state->menuWin );
|
|
wresize( state->menuWin, MENU_WINDOW_HEIGHT, width );
|
|
mvwin( state->menuWin, height-MENU_WINDOW_HEIGHT, 0 );
|
|
} else {
|
|
state->menuWin = newwin( MENU_WINDOW_HEIGHT, width,
|
|
height-MENU_WINDOW_HEIGHT, 0 );
|
|
}
|
|
}
|
|
|
|
CursesMenuState*
|
|
cmenu_init( WINDOW* mainWindow )
|
|
{
|
|
CursesMenuState* result = g_malloc0( sizeof(*result) );
|
|
result->mainWin = mainWindow;
|
|
|
|
sizeWindow( result );
|
|
nodelay( result->menuWin, 1 ); /* don't block on getch */
|
|
|
|
ADD_SOCKET( result, 0, handle_stdin );
|
|
|
|
return result;
|
|
}
|
|
|
|
void
|
|
cmenu_dispose( CursesMenuState* state )
|
|
{
|
|
XP_ASSERT( !!state );
|
|
g_free( state );
|
|
}
|
|
|
|
void
|
|
cmenu_resized( CursesMenuState* state )
|
|
{
|
|
XP_LOGF( "%s(%p)", __func__, state );
|
|
sizeWindow( state );
|
|
cmenu_draw( state );
|
|
}
|
|
|
|
typedef struct _MenuListElem {
|
|
MenuList* list;
|
|
void* closure;
|
|
} MenuListElem;
|
|
#define PUSH_TOKEN ((MenuListElem*)-1)
|
|
|
|
void
|
|
cmenu_pop( CursesMenuState* state )
|
|
{
|
|
/* pop off the front of the list until we've popped a PUSH_TOKEN */
|
|
while ( !!state->menuLists ) {
|
|
MenuListElem* elem = state->menuLists->data;
|
|
state->menuLists = g_slist_remove_link( state->menuLists,
|
|
state->menuLists );
|
|
if ( PUSH_TOKEN == elem ) {
|
|
break;
|
|
} else {
|
|
g_free( elem );
|
|
}
|
|
}
|
|
cmenu_draw( state );
|
|
}
|
|
|
|
static void
|
|
addMenus( CursesMenuState* state, void* closure, va_list ap )
|
|
{
|
|
for ( ; ; ) {
|
|
MenuList* param = va_arg(ap, MenuList*);
|
|
if ( !param ) {
|
|
break;
|
|
}
|
|
MenuListElem* elem = g_malloc0( sizeof( *elem ) );
|
|
elem->closure = closure;
|
|
elem->list = param;
|
|
state->menuLists = g_slist_prepend( state->menuLists, elem );
|
|
}
|
|
cmenu_draw( state );
|
|
}
|
|
|
|
void
|
|
cmenu_addMenus( CursesMenuState* state, void* closure, ... )
|
|
{
|
|
va_list ap;
|
|
va_start( ap, closure );
|
|
addMenus( state, closure, ap );
|
|
va_end(ap);
|
|
}
|
|
|
|
void cmenu_push( CursesMenuState* state, void* closure, ... )
|
|
{
|
|
if ( !!state ) {
|
|
state->menuLists = g_slist_prepend( state->menuLists, PUSH_TOKEN );
|
|
|
|
va_list ap;
|
|
va_start( ap, closure );
|
|
addMenus( state, closure, ap );
|
|
va_end(ap);
|
|
}
|
|
}
|
|
|
|
void
|
|
cmenu_removeMenus( CursesMenuState* state, ... )
|
|
{
|
|
va_list ap;
|
|
va_start( ap, state );
|
|
for ( ; ; ) {
|
|
MenuList* param = va_arg(ap, MenuList*);
|
|
if ( !param ) {
|
|
break;
|
|
}
|
|
for ( GSList* iter = state->menuLists; !!iter; iter = iter->next ) {
|
|
MenuListElem* elem = iter->data;
|
|
if ( elem->list == param ) {
|
|
state->menuLists = g_slist_remove( state->menuLists, elem );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
va_end( ap );
|
|
|
|
cmenu_draw( state );
|
|
}
|
|
|
|
static bool
|
|
handleKeyEvent( CursesMenuState* state, char ch, bool altPressed )
|
|
{
|
|
bool result = false;
|
|
for ( GSList* iter = state->menuLists; !!iter; iter = iter->next ) {
|
|
const MenuListElem* elem = iter->data;
|
|
if ( PUSH_TOKEN == elem ) {
|
|
break;
|
|
}
|
|
|
|
int altBit = altPressed ? ALT_BIT : 0;
|
|
for ( const MenuList* list = elem->list; !!list->handler; ++list ) {
|
|
if ( list->key == ch ) {
|
|
result = (*list->handler)(elem->closure, ch | altBit);
|
|
goto done;
|
|
}
|
|
}
|
|
}
|
|
done:
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
fmtMenuItem( const MenuList* item, char* buf, int maxLen )
|
|
{
|
|
snprintf( buf, maxLen, "%s %s", item->keyDesc, item->desc );
|
|
}
|
|
|
|
void
|
|
cmenu_draw( const CursesMenuState* state )
|
|
{
|
|
WINDOW* win = state->menuWin;
|
|
wclear( win );
|
|
int line = 0;
|
|
int col = 0;
|
|
int maxLen = 0;
|
|
for ( GSList* iter = state->menuLists; !!iter; iter = iter->next ) {
|
|
const MenuListElem* elem = iter->data;
|
|
if ( PUSH_TOKEN == elem ) {
|
|
break;
|
|
}
|
|
for ( MenuList* list = elem->list; !!list->handler; ++list ) {
|
|
char buf[32];
|
|
fmtMenuItem( list, buf, sizeof(buf) );
|
|
int len = strlen(buf);
|
|
if ( maxLen < len ) {
|
|
maxLen = len;
|
|
}
|
|
|
|
mvwaddstr( win, line, col, buf );
|
|
if ( ++line >= MENU_WINDOW_HEIGHT ) {
|
|
line = 0;
|
|
col += maxLen + 1;
|
|
maxLen = 0;
|
|
}
|
|
}
|
|
}
|
|
wrefresh( win );
|
|
}
|