xwords/xwords4/linux/cursesmain.c
Eric House 598be04bef make curses app more like the rest
Lots of changes adding a games-list view to the app from which you
create new games, open and delete existing ones, etc. There's still
plenty that's unimplemented, but it's already more useful for testing
and development. Which is the point.
2020-01-24 09:05:16 -08:00

1234 lines
38 KiB
C

/* -*- compile-command: "make MEMDEBUG=TRUE -j3"; -*- */
/*
* Copyright 2000 - 2011 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.
*/
#ifdef PLATFORM_NCURSES
#include <ncurses.h>
#include <signal.h>
#include <assert.h>
#include <ctype.h>
#include <sys/time.h>
#include <time.h>
#include <netdb.h> /* gethostbyname */
#include <errno.h>
//#include <net/netinet.h>
#include <sys/poll.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>
#include <fcntl.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include "linuxmain.h"
#include "linuxutl.h"
#include "linuxdict.h"
#include "cursesmain.h"
#include "cursesask.h"
#include "cursesletterask.h"
#include "linuxbt.h"
#include "model.h"
#include "draw.h"
#include "board.h"
#include "engine.h"
/* #include "compipe.h" */
#include "xwproto.h"
#include "xwstream.h"
#include "xwstate.h"
#include "strutils.h"
#include "server.h"
#include "memstream.h"
#include "util.h"
#include "dbgutil.h"
#include "linuxsms.h"
#include "linuxudp.h"
#include "gamesdb.h"
#include "relaycon.h"
#include "smsproto.h"
#include "device.h"
#include "cursesmenu.h"
#include "cursesboard.h"
#include "curgamlistwin.h"
#include "gsrcwrap.h"
#ifndef CURSES_CELL_HT
# define CURSES_CELL_HT 1
#endif
#ifndef CURSES_CELL_WIDTH
# define CURSES_CELL_WIDTH 2
#endif
#define INFINITE_TIMEOUT -1
#define BOARD_SCORE_PADDING 3
struct CursesAppGlobals {
CommonAppGlobals cag;
CursesMenuState* menuState;
CursGameList* gameList;
CursesBoardState* cbState;
GSList* openGames;
WINDOW* mainWin;
int winWidth, winHeight;
XP_U16 nLinesMenu;
gchar* lastErr;
short statusLine;
struct sockaddr_in listenerSockAddr;
#ifdef USE_GLIBLOOP
GMainLoop* loop;
GList* sources;
int quitpipe[2];
#else
XP_Bool timeToExit;
short fdCount;
struct pollfd fdArray[FD_MAX]; /* one for stdio, one for listening socket */
int timepipe[2]; /* for reading/writing "user events" */
#endif
};
static bool handleOpenGame( void* closure, int key );
static bool handleNewGame( void* closure, int key );
static bool handleDeleteGame( void* closure, int key );
static bool handleSel( void* closure, int key );
const MenuList g_sharedMenuList[] = {
{ handleQuit, "Quit", "Q", 'Q' },
{ handleNewGame, "New Game", "N", 'N' },
{ handleOpenGame, "Open Sel.", "O", 'O' },
{ handleDeleteGame, "Delete Sel.", "D", 'D' },
{ handleSel, "Select up", "J", 'J' },
{ handleSel, "Select down", "K", 'K' },
/* { handleResend, "Resend", "R", 'R' }, */
/* { handleSpace, "Raise focus", "<spc>", ' ' }, */
/* { handleRet, "Click/tap", "<ret>", '\r' }, */
/* { handleHint, "Hint", "?", '?' }, */
/* #ifdef KEYBOARD_NAV */
/* { handleLeft, "Left", "H", 'H' }, */
/* { handleRight, "Right", "L", 'L' }, */
/* { handleUp, "Up", "J", 'J' }, */
/* { handleDown, "Down", "K", 'K' }, */
/* #endif */
/* { handleCommit, "Commit move", "C", 'C' }, */
/* { handleFlip, "Flip", "F", 'F' }, */
/* { handleToggleValues, "Show values", "V", 'V' }, */
/* { handleBackspace, "Remove from board", "<del>", 8 }, */
/* { handleUndo, "Undo prev", "U", 'U' }, */
/* { handleReplace, "uNdo cur", "N", 'N' }, */
{ NULL, NULL, NULL, '\0'}
};
#ifdef CURSES_SMALL_SCREEN
const MenuList g_rootMenuListShow[] = {
{ handleRootKeyShow, "Press . for menu", "", '.' },
{ NULL, NULL, NULL, '\0'}
};
const MenuList g_rootMenuListHide[] = {
{ handleRootKeyHide, "Clear menu", ".", '.' },
{ NULL, NULL, NULL, '\0'}
};
#endif
static CursesAppGlobals g_globals; /* must be global b/c of SIGWINCH_handler */
#ifdef KEYBOARD_NAV
/* static void changeMenuForFocus( CursesAppGlobals* globals, */
/* BoardObjectType obj ); */
/* static XP_Bool handleLeft( CursesAppGlobals* globals ); */
/* static XP_Bool handleRight( CursesAppGlobals* globals ); */
/* static XP_Bool handleUp( CursesAppGlobals* globals ); */
/* static XP_Bool handleDown( CursesAppGlobals* globals ); */
/* static XP_Bool handleFocusKey( CursesAppGlobals* globals, XP_Key key ); */
#else
# define handleFocusKey( g, k ) XP_FALSE
#endif
/* static void countMenuLines( const MenuList** menuLists, int maxX, int padding, */
/* int* nLinesP, int* nColsP ); */
/* static void drawMenuFromList( WINDOW* win, const MenuList** menuLists, */
/* int nLines, int padding ); */
/* static CursesMenuHandler getHandlerForKey( const MenuList* list, char ch ); */
#ifdef MEM_DEBUG
# define MEMPOOL cGlobals->util->mpool,
#else
# define MEMPOOL
#endif
/* extern int errno; */
static void
initCurses( CursesAppGlobals* aGlobals )
{
/* ncurses man page says most apps want this sequence */
aGlobals->mainWin = initscr();
cbreak();
noecho();
nonl();
intrflush(stdscr, FALSE);
keypad(stdscr, TRUE); /* effects wgetch only? */
getmaxyx( aGlobals->mainWin, aGlobals->winHeight, aGlobals->winWidth );
XP_LOGF( "%s: getmaxyx()->w:%d; h:%d", __func__, aGlobals->winWidth,
aGlobals->winHeight );
/* globals->statusLine = height - MENU_WINDOW_HEIGHT - 1; */
/* globals->menuWin = newwin( MENU_WINDOW_HEIGHT, width, */
/* height-MENU_WINDOW_HEIGHT, 0 ); */
/* nodelay(globals->menuWin, 1); /\* don't block on getch *\/ */
} /* initCurses */
#if 0
static void
showStatus( CursesAppGlobals* globals )
{
char* str;
switch ( globals->state ) {
case XW_SERVER_WAITING_CLIENT_SIGNON:
str = "Waiting for client[s] to connnect";
break;
case XW_SERVER_READY_TO_PLAY:
str = "It's somebody's move";
break;
default:
str = "unknown state";
}
standout();
mvaddstr( globals->statusLine, 0, str );
/* clrtoeol(); */
standend();
refresh();
} /* showStatus */
#endif
bool
handleQuit( void* closure, int XP_UNUSED(key) )
{
CursesAppGlobals* globals = (CursesAppGlobals*)closure;
g_main_loop_quit( globals->loop );
return XP_TRUE;
} /* handleQuit */
static void
figureDims( CursesAppGlobals* aGlobals, int* widthP, int* topP, int* heightP )
{
LaunchParams* params = aGlobals->cag.params;
*widthP = aGlobals->winWidth;
*topP = params->cursesListWinHt;
*heightP = aGlobals->winHeight - params->cursesListWinHt - MENU_WINDOW_HEIGHT;
}
static bool
handleOpenGame( void* closure, int XP_UNUSED(key) )
{
LOG_FUNC();
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
const GameInfo* gi = cgl_getSel( aGlobals->gameList );
if ( !!gi ) {
int width, top, height;
figureDims( aGlobals, &width, &top, &height );
cb_open( aGlobals->cbState, gi->rowid, width, top, height );
}
return XP_TRUE;
}
static bool
handleNewGame( void* closure, int XP_UNUSED(key) )
{
LOG_FUNC();
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
// aGlobals->cag.params->needsNewGame = XP_FALSE;
int width, top, height;
figureDims( aGlobals, &width, &top, &height );
cb_new( aGlobals->cbState, width, top, height );
return XP_TRUE;
}
static bool
handleDeleteGame( void* closure, int XP_UNUSED(key) )
{
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
const char* question = "Are you sure you want to delete the "
"selected game? This action cannot be undone";
const char* buttons[] = { "Cancel", "Ok", };
if ( 1 == cursesask( aGlobals->mainWin, question, /* ?? */
VSIZE(buttons), buttons ) ) {
const GameInfo* gib = cgl_getSel( aGlobals->gameList );
if ( !!gib ) {
deleteGame( aGlobals->cag.params->pDb, gib->rowid );
cgl_remove( aGlobals->gameList, gib->rowid );
}
}
return XP_TRUE;
}
static bool
handleSel( void* closure, int key )
{
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
XP_ASSERT( key == 'K' || key == 'J' );
bool down = key == 'J';
cgl_moveSel( aGlobals->gameList, down );
return true;
}
/* static XP_Bool */
/* handleResend( CursesAppGlobals* globals ) */
/* { */
/* if ( !!globals->cGlobals.game.comms ) { */
/* comms_resendAll( globals->cGlobals.game.comms, COMMS_CONN_NONE, */
/* XP_TRUE ); */
/* } */
/* return XP_TRUE; */
/* } */
/* #ifdef KEYBOARD_NAV */
/* static void */
/* checkAssignFocus( BoardCtxt* board ) */
/* { */
/* if ( OBJ_NONE == board_getFocusOwner(board) ) { */
/* board_focusChanged( board, OBJ_BOARD, XP_TRUE ); */
/* } */
/* } */
/* #else */
/* # define checkAssignFocus(b) */
/* #endif */
/* static XP_Bool */
/* handleSpace( CursesAppGlobals* globals ) */
/* { */
/* XP_Bool handled; */
/* checkAssignFocus( globals->cGlobals.game.board ); */
/* globals->doDraw = board_handleKey( globals->cGlobals.game.board, */
/* XP_RAISEFOCUS_KEY, &handled ); */
/* return XP_TRUE; */
/* } /\* handleSpace *\/ */
/* static XP_Bool */
/* handleRet( CursesAppGlobals* globals ) */
/* { */
/* XP_Bool handled; */
/* globals->doDraw = board_handleKey( globals->cGlobals.game.board, */
/* XP_RETURN_KEY, &handled ); */
/* return XP_TRUE; */
/* } /\* handleRet *\/ */
/* static XP_Bool */
/* handleHint( CursesAppGlobals* globals ) */
/* { */
/* XP_Bool redo; */
/* globals->doDraw = board_requestHint( globals->cGlobals.game.board, */
/* #ifdef XWFEATURE_SEARCHLIMIT */
/* XP_FALSE, */
/* #endif */
/* XP_FALSE, &redo ); */
/* return XP_TRUE; */
/* } /\* handleHint *\/ */
#ifdef CURSES_SMALL_SCREEN
static XP_Bool
handleRootKeyShow( CursesAppGlobals* globals )
{
WINDOW* win;
MenuList* lists[] = { g_sharedMenuList, globals->menuList,
g_rootMenuListHide, NULL };
int winMaxY, winMaxX;
wclear( globals->menuWin );
wrefresh( globals->menuWin );
getmaxyx( globals->boardWin, winMaxY, winMaxX );
int border = 2;
int width = winMaxX - (border * 2);
int padding = 1; /* for the box */
int nLines, nCols;
countMenuLines( lists, width, padding, &nLines, &nCols );
if ( width > nCols ) {
width = nCols;
}
win = newwin( nLines+(padding*2), width+(padding*2),
((winMaxY-nLines-padding-padding)/2), (winMaxX-width)/2 );
wclear( win );
box( win, '|', '-');
drawMenuFromList( win, lists, nLines, padding );
wrefresh( win );
CursesMenuHandler handler = NULL;
while ( !handler ) {
int ch = fgetc( stdin );
int i;
for ( i = 0; !!lists[i]; ++i ) {
handler = getHandlerForKey( lists[i], ch );
if ( !!handler ) {
break;
}
}
}
delwin( win );
touchwin( globals->boardWin );
wrefresh( globals->boardWin );
MenuList* ml[] = { g_rootMenuListShow, NULL };
drawMenuFromList( globals->menuWin, ml, 1, 0 );
wrefresh( globals->menuWin );
return handler != NULL && (*handler)(globals);
} /* handleRootKeyShow */
static XP_Bool
handleRootKeyHide( CursesAppGlobals* globals )
{
globals->doDraw = XP_TRUE;
return XP_TRUE;
}
#endif
/* static void */
/* fmtMenuItem( const MenuList* item, char* buf, int maxLen ) */
/* { */
/* snprintf( buf, maxLen, "%s %s", item->keyDesc, item->desc ); */
/* } */
/* static void */
/* countMenuLines( const MenuList** menuLists, int maxX, int padding, */
/* int* nLinesP, int* nColsP ) */
/* { */
/* int nCols = 0; */
/* /\* The menu space should be wider rather than taller, but line up by */
/* column. So we want to use as many columns as possible to minimize the */
/* number of lines. So start with one line and lay out. If that doesn't */
/* fit, try two. Given the number of lines, get the max width of each */
/* column. */
/* *\/ */
/* maxX -= padding * 2; /\* on left and right side *\/ */
/* int nLines; */
/* for ( nLines = 1; ; ++nLines ) { */
/* short line = 0; */
/* XP_Bool tooFewLines = XP_FALSE; */
/* int maxThisCol = 0; */
/* int i; */
/* nCols = 0; */
/* for ( i = 0; !tooFewLines && (NULL != menuLists[i]); ++i ) { */
/* const MenuList* entry; */
/* for ( entry = menuLists[i]; !tooFewLines && !!entry->handler; */
/* ++entry ) { */
/* int width; */
/* char buf[32]; */
/* /\* time to switch to new column? *\/ */
/* if ( line == nLines ) { */
/* nCols += maxThisCol; */
/* if ( nCols > maxX ) { */
/* tooFewLines = XP_TRUE; */
/* break; */
/* } */
/* maxThisCol = 0; */
/* line = 0; */
/* } */
/* fmtMenuItem( entry, buf, sizeof(buf) ); */
/* width = strlen(buf) + 2; /\* padding *\/ */
/* if ( maxThisCol < width ) { */
/* maxThisCol = width; */
/* } */
/* ++line; */
/* } */
/* } */
/* /\* If we get here without running out of space, we're done *\/ */
/* nCols += maxThisCol; */
/* if ( !tooFewLines && (nCols < maxX) ) { */
/* break; */
/* } */
/* } */
/* *nColsP = nCols; */
/* *nLinesP = nLines; */
/* } /\* countMenuLines *\/ */
/* static void */
/* drawMenuFromList( WINDOW* win, const MenuList** menuLists, */
/* int nLines, int padding ) */
/* { */
/* short line = 0, col, i; */
/* int winMaxY, winMaxX; */
/* getmaxyx( win, winMaxY, winMaxX ); */
/* XP_USE(winMaxY); */
/* int maxColWidth = 0; */
/* if ( 0 == nLines ) { */
/* int ignore; */
/* countMenuLines( menuLists, winMaxX, padding, &nLines, &ignore ); */
/* } */
/* col = 0; */
/* for ( i = 0; NULL != menuLists[i]; ++i ) { */
/* const MenuList* entry; */
/* for ( entry = menuLists[i]; !!entry->handler; ++entry ) { */
/* char buf[32]; */
/* fmtMenuItem( entry, buf, sizeof(buf) ); */
/* mvwaddstr( win, line+padding, col+padding, buf ); */
/* int width = strlen(buf) + 2; */
/* if ( width > maxColWidth ) { */
/* maxColWidth = width; */
/* } */
/* if ( ++line == nLines ) { */
/* line = 0; */
/* col += maxColWidth; */
/* maxColWidth = 0; */
/* } */
/* } */
/* } */
/* } /\* drawMenuFromList *\/ */
static void
SIGWINCH_handler( int signal )
{
int height, width;
assert( signal == SIGWINCH );
endwin();
/* (*globals.drawMenu)( &globals ); */
getmaxyx( stdscr, height, width );
XP_LOGF( "%s:, getmaxyx()->w:%d; h:%d", __func__, width, height );
wresize( g_globals.mainWin, height-MENU_WINDOW_HEIGHT, width );
// board_draw( g_globals.cGlobals.game.board );
} /* SIGWINCH_handler */
static void
SIGINTTERM_handler( int XP_UNUSED(signal) )
{
if ( 1 != write( g_globals.quitpipe[1], "!", 1 ) ) {
XP_ASSERT(0);
}
}
#ifdef USE_GLIBLOOP
static gboolean
handle_quitwrite( GIOChannel* XP_UNUSED(source), GIOCondition XP_UNUSED(condition), gpointer data )
{
CursesAppGlobals* globals = (CursesAppGlobals*)data;
handleQuit( globals, 0 );
return TRUE;
}
#endif
#ifndef USE_GLIBLOOP
#ifdef XWFEATURE_RELAY
static int
figureTimeout( CursesAppGlobals* globals )
{
int result = INFINITE_TIMEOUT;
XWTimerReason ii;
XP_U32 now = util_getCurSeconds( globals->cGlobals.params->util );
now *= 1000;
for ( ii = 0; ii < NUM_TIMERS_PLUS_ONE; ++ii ) {
TimerInfo* tip = &globals->cGlobals.timerInfo[ii];
if ( !!tip->proc ) {
XP_U32 then = tip->when * 1000;
if ( now >= then ) {
result = 0;
break; /* if one's immediate, we're done */
} else {
then -= now;
if ( result == -1 || then < result ) {
result = then;
}
}
}
}
return result;
} /* figureTimeout */
#else
# define figureTimeout(g) INFINITE_TIMEOUT
#endif
#ifdef XWFEATURE_RELAY
static void
fireCursesTimer( CursesAppGlobals* globals )
{
XWTimerReason ii;
TimerInfo* smallestTip = NULL;
for ( ii = 0; ii < NUM_TIMERS_PLUS_ONE; ++ii ) {
TimerInfo* tip = &globals->cGlobals.timerInfo[ii];
if ( !!tip->proc ) {
if ( !smallestTip ) {
smallestTip = tip;
} else if ( tip->when < smallestTip->when ) {
smallestTip = tip;
}
}
}
if ( !!smallestTip ) {
XP_U32 now = util_getCurSeconds( globals->cGlobals.params->util ) ;
if ( now >= smallestTip->when ) {
if ( linuxFireTimer( &globals->cGlobals,
smallestTip - globals->cGlobals.timerInfo ) ){
board_draw( globals->cGlobals.game.board );
}
} else {
XP_LOGF( "skipping timer: now (%ld) < when (%ld)",
now, smallestTip->when );
}
}
} /* fireCursesTimer */
#endif
#endif
/*
* Ok, so this doesn't block yet....
*/
#ifndef USE_GLIBLOOP
/* static XP_Bool */
/* blocking_gotEvent( CursesAppGlobals* globals, int* ch ) */
/* { */
/* XP_Bool result = XP_FALSE; */
/* int numEvents, ii; */
/* short fdIndex; */
/* XP_Bool redraw = XP_FALSE; */
/* int timeout = figureTimeout( globals ); */
/* numEvents = poll( globals->fdArray, globals->fdCount, timeout ); */
/* if ( timeout != INFINITE_TIMEOUT && numEvents == 0 ) { */
/* #ifdef XWFEATURE_RELAY */
/* fireCursesTimer( globals ); */
/* #endif */
/* } else if ( numEvents > 0 ) { */
/* /\* stdin first *\/ */
/* if ( (globals->fdArray[FD_STDIN].revents & POLLIN) != 0 ) { */
/* int evtCh = wgetch(globals->mainWin); */
/* XP_LOGF( "%s: got key: %x", __func__, evtCh ); */
/* *ch = evtCh; */
/* result = XP_TRUE; */
/* --numEvents; */
/* } */
/* if ( (globals->fdArray[FD_STDIN].revents & ~POLLIN ) ) { */
/* XP_LOGF( "some other events set on stdin" ); */
/* } */
/* if ( (globals->fdArray[FD_TIMEEVT].revents & POLLIN) != 0 ) { */
/* char ch; */
/* if ( 1 != read(globals->fdArray[FD_TIMEEVT].fd, &ch, 1 ) ) { */
/* XP_ASSERT(0); */
/* } */
/* } */
/* fdIndex = FD_FIRSTSOCKET; */
/* if ( numEvents > 0 ) { */
/* if ( (globals->fdArray[fdIndex].revents & ~POLLIN ) ) { */
/* XP_LOGF( "some other events set on socket %d", */
/* globals->fdArray[fdIndex].fd ); */
/* } */
/* if ( (globals->fdArray[fdIndex].revents & POLLIN) != 0 ) { */
/* --numEvents; */
/* if ( globals->fdArray[fdIndex].fd */
/* == globals->csInfo.server.serverSocket ) { */
/* /\* It's the listening socket: call platform's accept() */
/* wrapper *\/ */
/* (*globals->cGlobals.acceptor)( globals->fdArray[fdIndex].fd, */
/* globals ); */
/* } else { */
/* #ifndef XWFEATURE_STANDALONE_ONLY */
/* unsigned char buf[1024]; */
/* int nBytes; */
/* CommsAddrRec addrRec; */
/* CommsAddrRec* addrp = NULL; */
/* /\* It's a normal data socket *\/ */
/* switch ( globals->cGlobals.params->conType ) { */
/* #ifdef XWFEATURE_RELAY */
/* case COMMS_CONN_RELAY: */
/* nBytes = linux_relay_receive( &globals->cGlobals, buf, */
/* sizeof(buf) ); */
/* break; */
/* #endif */
/* #ifdef XWFEATURE_SMS */
/* case COMMS_CONN_SMS: */
/* addrp = &addrRec; */
/* nBytes = linux_sms_receive( &globals->cGlobals, */
/* globals->fdArray[fdIndex].fd, */
/* buf, sizeof(buf), addrp ); */
/* break; */
/* #endif */
/* #ifdef XWFEATURE_BLUETOOTH */
/* case COMMS_CONN_BT: */
/* nBytes = linux_bt_receive( globals->fdArray[fdIndex].fd, */
/* buf, sizeof(buf) ); */
/* break; */
/* #endif */
/* default: */
/* XP_ASSERT( 0 ); /\* fired *\/ */
/* } */
/* if ( nBytes != -1 ) { */
/* XWStreamCtxt* inboundS; */
/* redraw = XP_FALSE; */
/* inboundS = stream_from_msgbuf( &globals->cGlobals, */
/* buf, nBytes ); */
/* if ( !!inboundS ) { */
/* if ( comms_checkIncomingStream( */
/* globals->cGlobals.game.comms, */
/* inboundS, addrp ) ) { */
/* redraw = server_receiveMessage( */
/* globals->cGlobals.game.server, inboundS ); */
/* } */
/* stream_destroy( inboundS ); */
/* } */
/* /\* if there's something to draw resulting from the */
/* message, we need to give the main loop time to reflect */
/* that on the screen before giving the server another */
/* shot. So just call the idle proc. *\/ */
/* if ( redraw ) { */
/* curses_util_requestTime(globals->cGlobals.params->util); */
/* } */
/* } */
/* #else */
/* XP_ASSERT(0); /\* no socket activity in standalone game! *\/ */
/* #endif /\* #ifndef XWFEATURE_STANDALONE_ONLY *\/ */
/* } */
/* ++fdIndex; */
/* } */
/* } */
/* for ( ii = 0; ii < 5; ++ii ) { */
/* redraw = server_do( globals->cGlobals.game.server, NULL ) || redraw; */
/* } */
/* if ( redraw ) { */
/* /\* messages change a lot *\/ */
/* board_invalAll( globals->cGlobals.game.board ); */
/* board_draw( globals->cGlobals.game.board ); */
/* } */
/* saveGame( globals->cGlobals ); */
/* } */
/* return result; */
/* } /\* blocking_gotEvent *\/ */
#endif
/* static void */
/* remapKey( int* kp ) */
/* { */
/* /\* There's what the manual says I should get, and what I actually do from */
/* * a funky M$ keyboard.... */
/* *\/ */
/* int key = *kp; */
/* switch( key ) { */
/* case KEY_B2: /\* "center of keypad" *\/ */
/* key = '\r'; */
/* break; */
/* case KEY_DOWN: */
/* case 526: */
/* key = 'K'; */
/* break; */
/* case KEY_UP: */
/* case 523: */
/* key = 'J'; */
/* break; */
/* case KEY_LEFT: */
/* case 524: */
/* key = 'H'; */
/* break; */
/* case KEY_RIGHT: */
/* case 525: */
/* key = 'L'; */
/* break; */
/* default: */
/* if ( key > 0x7F ) { */
/* XP_LOGF( "%s(%d): no mapping", __func__, key ); */
/* } */
/* break; */
/* } */
/* *kp = key; */
/* } /\* remapKey *\/ */
typedef struct _MenuEntry {
MenuList* menuItem;
void* closure;
} MenuEntry;
/* void */
/* drawMenuLargeOrSmall( CursesAppGlobals* globals, const MenuList* menuList, */
/* void* closure ) */
/* { */
/* #ifdef CURSES_SMALL_SCREEN */
/* const MenuList* lists[] = { g_rootMenuListShow, NULL }; */
/* #else */
/* const MenuList* lists[] = { g_sharedMenuList, menuList, NULL }; */
/* #endif */
/* wclear( globals->menuWin ); */
/* drawMenuFromList( globals->menuWin, lists, 0, 0 ); */
/* wrefresh( globals->menuWin ); */
/* } */
#if 0
static void
initClientSocket( CursesAppGlobals* globals, char* serverName )
{
struct hostent* hostinfo;
hostinfo = gethostbyname( serverName );
if ( !hostinfo ) {
userError( globals, "unable to get host info for %s\n", serverName );
} else {
char* hostName = inet_ntoa( *(struct in_addr*)hostinfo->h_addr );
XP_LOGF( "gethostbyname returned %s", hostName );
globals->csInfo.client.serverAddr = inet_addr(hostName);
XP_LOGF( "inet_addr returned %lu",
globals->csInfo.client.serverAddr );
}
} /* initClientSocket */
#endif
/* static const MenuList* */
/* getHandlerForKey( const MenuList* list, char ch ) */
/* { */
/* MenuList* handler = NULL; */
/* while ( list->handler != NULL ) { */
/* if ( list->key == ch ) { */
/* handler = list->handler; */
/* break; */
/* } */
/* ++list; */
/* } */
/* return handler; */
/* } */
/* static XP_Bool */
/* handleKeyEvent( CursesAppGlobals* globals, const MenuList* list, char ch ) */
/* { */
/* const MenuList* entry = getHandlerForKey( list, ch ); */
/* XP_Bool result = XP_FALSE; */
/* if ( !!handler ) { */
/* result = (*entry->handler)(entry->closure); */
/* } */
/* return result; */
/* } /\* handleKeyEvent *\/ */
/* static XP_Bool */
/* passKeyToBoard( CursesAppGlobals* XP_UNUSED(globals), char XP_UNUSED(ch) ) */
/* { */
/* XP_ASSERT(0); */
/* /\* XP_Bool handled = ch >= 'a' && ch <= 'z'; *\/ */
/* /\* if ( handled ) { *\/ */
/* /\* ch += 'A' - 'a'; *\/ */
/* /\* globals->doDraw = board_handleKey( globals->cGlobals.game.board, *\/ */
/* /\* ch, NULL ); *\/ */
/* /\* } *\/ */
/* /\* return handled; *\/ */
/* } /\* passKeyToBoard *\/ */
#ifdef RELAY_VIA_HTTP
static void
onJoined( void* closure, const XP_UCHAR* connname, XWHostID hid )
{
LOG_FUNC();
CursesAppGlobals* globals = (CursesAppGlobals*)closure;
CommsCtxt* comms = globals->cGlobals.game.comms;
comms_gameJoined( comms, connname, hid );
}
/* static void */
/* relay_requestJoin_curses( void* closure, const XP_UCHAR* devID, const XP_UCHAR* room, */
/* XP_U16 nPlayersHere, XP_U16 nPlayersTotal, */
/* XP_U16 seed, XP_U16 lang ) */
/* { */
/* CursesAppGlobals* globals = (CursesAppGlobals*)closure; */
/* relaycon_join( globals->cGlobals.params, devID, room, nPlayersHere, nPlayersTotal, */
/* seed, lang, onJoined, globals ); */
/* } */
#endif
static void
cursesGotBuf( void* closure, const CommsAddrRec* addr,
const XP_U8* buf, XP_U16 len )
{
LOG_FUNC();
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
XP_U32 clientToken;
XP_ASSERT( sizeof(clientToken) < len );
XP_MEMCPY( &clientToken, &buf[0], sizeof(clientToken) );
buf += sizeof(clientToken);
len -= sizeof(clientToken);
sqlite3_int64 rowid;
XP_U16 gotSeed;
rowidFromToken( XP_NTOHL( clientToken ), &rowid, &gotSeed );
/* Figure out if the device is live, or we need to open the game */
XP_U16 seed = cb_feedBuffer( aGlobals->cbState, rowid, buf, len, addr );
XP_ASSERT( seed == 0 || gotSeed == seed );
XP_USE( seed );
/* if ( seed == comms_getChannelSeed( cGlobals->game.comms ) ) { */
/* gameGotBuf( cGlobals, XP_TRUE, buf, len, addr ); */
/* } else { */
/* XP_LOGF( "%s(): dropping packet; meant for a different device", */
/* __func__ ); */
/* } */
/* LOG_RETURN_VOID(); */
}
static void
cursesGotForRow( void* XP_UNUSED(closure), const CommsAddrRec* XP_UNUSED(from),
sqlite3_int64 XP_UNUSED(rowid), const XP_U8* XP_UNUSED(buf),
XP_U16 XP_UNUSED(len) )
{
// CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
LOG_FUNC();
/* gameGotBuf( &globals->cGlobals, XP_TRUE, buf, len, from ); */
XP_ASSERT( 0 );
LOG_RETURN_VOID();
}
static gint
curses_requestMsgs( gpointer data )
{
CursesAppGlobals* aGlobals = (CursesAppGlobals*)data;
XP_UCHAR devIDBuf[64] = {0};
db_fetch_safe( aGlobals->cag.params->pDb, KEY_RDEVID, devIDBuf,
sizeof(devIDBuf) );
if ( '\0' != devIDBuf[0] ) {
relaycon_requestMsgs( aGlobals->cag.params, devIDBuf );
} else {
XP_LOGF( "%s: not requesting messages as don't have relay id", __func__ );
}
return 0; /* don't run again */
}
static void
cursesNoticeRcvd( void* closure )
{
LOG_FUNC();
CursesAppGlobals* globals = (CursesAppGlobals*)closure;
#ifdef DEBUG
guint res =
#endif
ADD_ONETIME_IDLE( curses_requestMsgs, globals );
XP_ASSERT( res > 0 );
}
static gboolean
keepalive_timer( gpointer data )
{
LOG_FUNC();
curses_requestMsgs( data );
return TRUE;
}
static void
cursesDevIDReceived( void* closure, const XP_UCHAR* devID,
XP_U16 maxInterval )
{
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
// CommonGlobals* cGlobals = &globals->cGlobals;
sqlite3* pDb = aGlobals->cag.params->pDb;
if ( !!devID ) {
XP_LOGF( "%s(devID=%s)", __func__, devID );
/* If we already have one, make sure it's the same! Else store. */
gchar buf[64];
XP_Bool have = db_fetch_safe( pDb, KEY_RDEVID, buf, sizeof(buf) )
&& 0 == strcmp( buf, devID );
if ( !have ) {
db_store( pDb, KEY_RDEVID, devID );
}
(void)g_timeout_add_seconds( maxInterval, keepalive_timer, aGlobals );
} else {
XP_LOGF( "%s: bad relayid", __func__ );
db_remove( pDb, KEY_RDEVID );
DevIDType typ;
const XP_UCHAR* devID = linux_getDevID( aGlobals->cag.params, &typ );
relaycon_reg( aGlobals->cag.params, NULL, typ, devID );
}
}
static void
cursesErrorMsgRcvd( void* closure, const XP_UCHAR* msg )
{
CursesAppGlobals* globals = (CursesAppGlobals*)closure;
if ( !!globals->lastErr && 0 == strcmp( globals->lastErr, msg ) ) {
XP_LOGF( "skipping error message from relay" );
} else {
g_free( globals->lastErr );
globals->lastErr = g_strdup( msg );
const char* buttons[] = { "Ok" };
(void)cursesask( globals->mainWin, msg, VSIZE(buttons), buttons );
}
}
/* static gboolean */
/* chatsTimerFired( gpointer data ) */
/* { */
/* CursesAppGlobals* globals = (CursesAppGlobals*)data; */
/* XWGame* game = &globals->cGlobals.game; */
/* GameStateInfo gsi; */
/* game_getState( game, &gsi ); */
/* if ( gsi.canChat && 3 > globals->nChatsSent ) { */
/* XP_UCHAR msg[128]; */
/* struct tm* timp; */
/* struct timeval tv; */
/* struct timezone tz; */
/* gettimeofday( &tv, &tz ); */
/* timp = localtime( &tv.tv_sec ); */
/* snprintf( msg, sizeof(msg), "%x: Saying hi via chat at %.2d:%.2d:%.2d", */
/* comms_getChannelSeed( game->comms ), */
/* timp->tm_hour, timp->tm_min, timp->tm_sec ); */
/* XP_LOGF( "%s: sending \"%s\"", __func__, msg ); */
/* board_sendChat( game->board, msg ); */
/* ++globals->nChatsSent; */
/* } */
/* return TRUE; */
/* } */
/* static XP_U16 */
/* feedBufferCurses( CommonGlobals* cGlobals, sqlite3_int64 rowid, */
/* const XP_U8* buf, XP_U16 len, const CommsAddrRec* from ) */
/* { */
/* gameGotBuf( cGlobals, XP_TRUE, buf, len, from ); */
/* /\* GtkGameGlobals* globals = findOpenGame( apg, rowid ); *\/ */
/* /\* if ( !!globals ) { *\/ */
/* /\* gameGotBuf( &globals->cGlobals, XP_TRUE, buf, len, from ); *\/ */
/* /\* seed = comms_getChannelSeed( globals->cGlobals.game.comms ); *\/ */
/* /\* } else { *\/ */
/* /\* GtkGameGlobals tmpGlobals; *\/ */
/* /\* if ( loadGameNoDraw( &tmpGlobals, apg->params, rowid ) ) { *\/ */
/* /\* gameGotBuf( &tmpGlobals.cGlobals, XP_FALSE, buf, len, from ); *\/ */
/* /\* seed = comms_getChannelSeed( tmpGlobals.cGlobals.game.comms ); *\/ */
/* /\* saveGame( &tmpGlobals.cGlobals ); *\/ */
/* /\* } *\/ */
/* /\* freeGlobals( &tmpGlobals ); *\/ */
/* /\* } *\/ */
/* /\* return seed; *\/ */
/* } */
/* static void */
/* smsMsgReceivedCurses( void* closure, const CommsAddrRec* from, */
/* XP_U32 XP_UNUSED(gameID), */
/* const XP_U8* buf, XP_U16 len ) */
/* { */
/* LOG_FUNC(); */
/* CommonGlobals* cGlobals = (CommonGlobals*)closure; */
/* gameGotBuf( cGlobals, XP_TRUE, buf, len, from ); */
/* LOG_RETURN_VOID(); */
/* /\* LaunchParams* params = cGlobals->params; *\/ */
/* /\* sqlite3_int64 rowids[4]; *\/ */
/* /\* int nRowIDs = VSIZE(rowids); *\/ */
/* /\* getRowsForGameID( params->pDb, gameID, rowids, &nRowIDs ); *\/ */
/* /\* for ( int ii = 0; ii < nRowIDs; ++ii ) { *\/ */
/* /\* gameGotBuf( cGlobals, XP_TRUE, buf, len, from ); *\/ */
/* /\* // feedBufferCurses( cGlobals, rowids[ii], buf, len, from ); *\/ */
/* /\* } *\/ */
/* } */
static void
onGameSaved( CursesAppGlobals* aGlobals, sqlite3_int64 rowid, bool isNew )
{
cgl_refreshOne( aGlobals->gameList, rowid, isNew );
}
void
cursesmain( XP_Bool XP_UNUSED(isServer), LaunchParams* params )
{
memset( &g_globals, 0, sizeof(g_globals) );
g_globals.cag.params = params;
initCurses( &g_globals );
if ( !params->closeStdin ) {
g_globals.menuState = cmenu_init( g_globals.mainWin );
cmenu_push( g_globals.menuState, &g_globals, g_sharedMenuList, NULL );
}
g_globals.loop = g_main_loop_new( NULL, FALSE );
g_globals.cbState = cb_init( &g_globals, params, g_globals.menuState,
onGameSaved );
sqlite3* pDb = params->pDb;
g_globals.gameList = cgl_init( pDb, g_globals.winWidth, params->cursesListWinHt );
cgl_refresh( g_globals.gameList );
// g_globals.amServer = isServer;
/* g_globals.cGlobals.params = params; */
/* #ifdef XWFEATURE_RELAY */
/* g_globals.cGlobals.relaySocket = -1; */
/* #endif */
/* g_globals.cGlobals.socketAdded = curses_socket_added; */
/* g_globals.cGlobals.socketAddedClosure = &g_globals; */
/* g_globals.cGlobals.onSave = curses_onGameSaved; */
/* g_globals.cGlobals.onSaveClosure = &g_globals; */
/* g_globals.cGlobals.addAcceptor = curses_socket_acceptor; */
/* g_globals.cGlobals.cp.showBoardArrow = XP_TRUE; */
/* g_globals.cGlobals.cp.showRobotScores = params->showRobotScores; */
/* g_globals.cGlobals.cp.hideTileValues = params->hideValues; */
/* g_globals.cGlobals.cp.skipCommitConfirm = params->skipCommitConfirm; */
/* g_globals.cGlobals.cp.sortNewTiles = params->sortNewTiles; */
/* g_globals.cGlobals.cp.showColors = params->showColors; */
/* g_globals.cGlobals.cp.allowPeek = params->allowPeek; */
/* #ifdef XWFEATURE_SLOW_ROBOT */
/* g_globals.cGlobals.cp.robotThinkMin = params->robotThinkMin; */
/* g_globals.cGlobals.cp.robotThinkMax = params->robotThinkMax; */
/* g_globals.cGlobals.cp.robotTradePct = params->robotTradePct; */
/* #endif */
/* g_globals.cGlobals.gi = &params->pgi; */
/* setupUtil( &g_globals.cGlobals ); */
/* setupCursesUtilCallbacks( &g_globals, g_globals.cGlobals.util ); */
// initFromParams( &g_globals.cGlobals, params );
#ifdef XWFEATURE_RELAY
/* if ( addr_hasType( &params->addr, COMMS_CONN_RELAY ) ) { */
/* g_globals.cGlobals.defaultServerName */
/* = params->connInfo.relay.relayName; */
/* } */
#endif
/* if ( !params->closeStdin ) { */
/* cursesListenOnSocket( &g_globals, 0, handle_stdin ); */
/* } */
/* setOneSecondTimer( &g_globals.cGlobals ); */
# ifdef DEBUG
int piperesult =
# endif
pipe( g_globals.quitpipe );
XP_ASSERT( piperesult == 0 );
ADD_SOCKET( &g_globals, g_globals.quitpipe[0], handle_quitwrite );
struct sigaction act = { .sa_handler = SIGINTTERM_handler };
sigaction( SIGINT, &act, NULL );
sigaction( SIGTERM, &act, NULL );
struct sigaction act2 = { .sa_handler = SIGWINCH_handler };
sigaction( SIGWINCH, &act2, NULL );
if ( params->useUdp ) {
RelayConnProcs procs = {
.msgReceived = cursesGotBuf,
.msgForRow = cursesGotForRow,
.msgNoticeReceived = cursesNoticeRcvd,
.devIDReceived = cursesDevIDReceived,
.msgErrorMsg = cursesErrorMsgRcvd,
};
relaycon_init( params, &procs, &g_globals,
params->connInfo.relay.relayName,
params->connInfo.relay.defaultSendPort );
XP_Bool idIsNew = linux_setupDevidParams( params );
linux_doInitialReg( params, idIsNew );
}
if ( 0 == cgl_getNGames( g_globals.gameList ) ) {
handleNewGame( &g_globals, 0 );
}
g_main_loop_run( g_globals.loop );
cb_closeAll( g_globals.cbState );
#ifdef XWFEATURE_BLUETOOTH
// linux_bt_close( &g_globals.cGlobals );
#endif
#ifdef XWFEATURE_SMS
// linux_sms_close( &g_globals.cGlobals );
#endif
#ifdef XWFEATURE_IP_DIRECT
// linux_udp_close( &g_globals.cGlobals );
#endif
cgl_destroy( g_globals.gameList );
endwin();
device_store( params->dutil );
if ( params->useUdp ) {
relaycon_cleanup( params );
}
linux_sms_cleanup( params );
} /* cursesmain */
#endif /* PLATFORM_NCURSES */