mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-22 07:28:16 +01:00
b81bd46645
I don't know why, but in my tests the relay seems to be delivering messages to the wrong device. The linux device detects this. It used to assert, but now just drops the message. If this is happening on Android it might be why I'm seeing crashes...
1546 lines
50 KiB
C
1546 lines
50 KiB
C
/* -*- compile-command: "make MEMDEBUG=TRUE -j5"; -*- */
|
|
/*
|
|
* Copyright 2000 - 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.
|
|
*/
|
|
#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/ioctl.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];
|
|
int winchpipe[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 */
|
|
if ( !aGlobals->cag.params->closeStdin ) {
|
|
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, cb_dims* dims )
|
|
{
|
|
LaunchParams* params = aGlobals->cag.params;
|
|
dims->width = aGlobals->winWidth;
|
|
dims->top = params->cursesListWinHt;
|
|
dims->height = 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 );
|
|
XP_ASSERT( !!gi );
|
|
cb_dims dims;
|
|
figureDims( aGlobals, &dims );
|
|
cb_open( aGlobals->cbState, gi->rowid, &dims );
|
|
return XP_TRUE;
|
|
}
|
|
|
|
static bool
|
|
handleNewGame( void* closure, int XP_UNUSED(key) )
|
|
{
|
|
LOG_FUNC();
|
|
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
|
|
|
|
// aGlobals->cag.params->needsNewGame = XP_FALSE;
|
|
|
|
cb_dims dims;
|
|
figureDims( aGlobals, &dims );
|
|
if ( !cb_new( aGlobals->cbState, &dims ) ) {
|
|
/* This erases the whole screen. Fix later. PENDING */
|
|
/* const char* buttons[] = { "Ok", }; */
|
|
/* (void)cursesask( aGlobals->mainWin, "Unable to create game (check params?)", */
|
|
/* VSIZE(buttons), buttons ); */
|
|
}
|
|
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
|
|
writeToPipe( int pipe )
|
|
{
|
|
if ( 1 != write( pipe, "!", 1 ) ) {
|
|
XP_ASSERT(0);
|
|
}
|
|
}
|
|
|
|
static void
|
|
readFromPipe( GIOChannel* source )
|
|
{
|
|
int pipe = g_io_channel_unix_get_fd( source );
|
|
char ch;
|
|
#ifdef DEBUG
|
|
ssize_t nRead =
|
|
#endif
|
|
read( pipe, &ch, sizeof(ch) );
|
|
XP_ASSERT( nRead == sizeof(ch) && ch == '!' );
|
|
}
|
|
|
|
static void
|
|
SIGWINCH_handler( int signal )
|
|
{
|
|
assert( signal == SIGWINCH );
|
|
|
|
/* Write to pipe to force update */
|
|
writeToPipe( g_globals.winchpipe[1] );
|
|
} /* SIGWINCH_handler */
|
|
|
|
static void
|
|
SIGINTTERM_handler( int XP_UNUSED(signal) )
|
|
{
|
|
writeToPipe( g_globals.quitpipe[1] );
|
|
}
|
|
|
|
#ifdef USE_GLIBLOOP
|
|
static gboolean
|
|
handle_quitwrite( GIOChannel* source, GIOCondition XP_UNUSED(condition), gpointer data )
|
|
{
|
|
LOG_FUNC();
|
|
readFromPipe( source );
|
|
CursesAppGlobals* aGlobals = (CursesAppGlobals*)data;
|
|
cb_closeAll( aGlobals->cbState );
|
|
handleQuit( aGlobals, 0 );
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
handle_winchwrite( GIOChannel* source, GIOCondition XP_UNUSED_DBG(condition), gpointer data )
|
|
{
|
|
XP_LOGF( "%s(condition=%x)", __func__, condition );
|
|
CursesAppGlobals* aGlobals = (CursesAppGlobals*)data;
|
|
|
|
/* Read from the pipe so it won't call again */
|
|
readFromPipe( source );
|
|
|
|
struct winsize ws;
|
|
ioctl( STDIN_FILENO, TIOCGWINSZ, &ws );
|
|
XP_LOGF( "%s(): lines %d, columns %d", __func__, ws.ws_row, ws.ws_col );
|
|
aGlobals->winHeight = ws.ws_row;
|
|
aGlobals->winWidth = ws.ws_col;
|
|
|
|
resize_term( ws.ws_row, ws.ws_col );
|
|
|
|
cgl_resized( aGlobals->gameList, g_globals.winWidth,
|
|
g_globals.cag.params->cursesListWinHt );
|
|
if ( !!aGlobals->menuState ) {
|
|
cmenu_resized( aGlobals->menuState );
|
|
}
|
|
|
|
cb_dims dims;
|
|
figureDims( aGlobals, &dims );
|
|
cb_resized( aGlobals->cbState, &dims );
|
|
|
|
LOG_RETURN_VOID();
|
|
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
|
|
|
|
/* <<<<<<< HEAD */
|
|
/* static void */
|
|
/* curses_util_informNeedPassword( XW_UtilCtxt* XP_UNUSED(uc), */
|
|
/* XP_U16 XP_UNUSED_DBG(playerNum), */
|
|
/* const XP_UCHAR* XP_UNUSED_DBG(name) ) */
|
|
/* { */
|
|
/* XP_WARNF( "curses_util_informNeedPassword(num=%d, name=%s", playerNum, name ); */
|
|
/* } /\* curses_util_askPassword *\/ */
|
|
|
|
/* static void */
|
|
/* curses_util_yOffsetChange( XW_UtilCtxt* XP_UNUSED(uc), */
|
|
/* XP_U16 XP_UNUSED(maxOffset), */
|
|
/* XP_U16 XP_UNUSED(oldOffset), XP_U16 XP_UNUSED(newOffset) ) */
|
|
/* { */
|
|
/* /\* if ( oldOffset != newOffset ) { *\/ */
|
|
/* /\* XP_WARNF( "curses_util_yOffsetChange(%d,%d,%d) not implemented", *\/ */
|
|
/* /\* maxOffset, oldOffset, newOffset ); *\/ */
|
|
/* /\* } *\/ */
|
|
/* } /\* curses_util_yOffsetChange *\/ */
|
|
|
|
/* #ifdef XWFEATURE_TURNCHANGENOTIFY */
|
|
/* static void */
|
|
/* curses_util_turnChanged( XW_UtilCtxt* XP_UNUSED(uc), XP_S16 XP_UNUSED_DBG(newTurn) ) */
|
|
/* { */
|
|
/* XP_LOGF( "%s(turn=%d)", __func__, newTurn ); */
|
|
/* } */
|
|
/* #endif */
|
|
|
|
/* static void */
|
|
/* curses_util_notifyDupStatus( XW_UtilCtxt* XP_UNUSED(uc), */
|
|
/* XP_Bool amHost, */
|
|
/* const XP_UCHAR* msg ) */
|
|
/* { */
|
|
/* XP_LOGF( "%s(amHost=%d, msg=%s)", __func__, amHost, msg ); */
|
|
/* } */
|
|
|
|
/* static void */
|
|
/* curses_util_notifyIllegalWords( XW_UtilCtxt* XP_UNUSED(uc), */
|
|
/* BadWordInfo* XP_UNUSED(bwi), */
|
|
/* XP_U16 XP_UNUSED(player), */
|
|
/* XP_Bool XP_UNUSED(turnLost) ) */
|
|
/* { */
|
|
/* XP_WARNF( "curses_util_notifyIllegalWords not implemented" ); */
|
|
/* } /\* curses_util_notifyIllegalWord *\/ */
|
|
|
|
/* static void */
|
|
/* curses_util_remSelected( XW_UtilCtxt* uc ) */
|
|
/* { */
|
|
/* CursesAppGlobals* globals = (CursesAppGlobals*)uc->closure; */
|
|
/* XWStreamCtxt* stream; */
|
|
/* XP_UCHAR* text; */
|
|
|
|
/* stream = mem_stream_make_raw( MPPARM(globals->cGlobals.util->mpool) */
|
|
/* globals->cGlobals.params->vtMgr ); */
|
|
/* board_formatRemainingTiles( globals->cGlobals.game.board, stream ); */
|
|
|
|
/* text = strFromStream( stream ); */
|
|
|
|
/* const char* buttons[] = { "Ok" }; */
|
|
/* (void)cursesask( globals, text, VSIZE(buttons), buttons ); */
|
|
|
|
/* free( text ); */
|
|
/* } */
|
|
|
|
/* #ifndef XWFEATURE_STANDALONE_ONLY */
|
|
/* static XWStreamCtxt* */
|
|
/* curses_util_makeStreamFromAddr(XW_UtilCtxt* uc, XP_PlayerAddr channelNo ) */
|
|
/* { */
|
|
/* CursesAppGlobals* globals = (CursesAppGlobals*)uc->closure; */
|
|
/* LaunchParams* params = globals->cGlobals.params; */
|
|
|
|
/* XWStreamCtxt* stream = mem_stream_make( MPPARM(uc->mpool) params->vtMgr, */
|
|
/* &globals->cGlobals, channelNo, */
|
|
/* sendOnClose ); */
|
|
/* return stream; */
|
|
/* } /\* curses_util_makeStreamFromAddr *\/ */
|
|
/* #endif */
|
|
|
|
/* #ifdef XWFEATURE_CHAT */
|
|
/* static void */
|
|
/* curses_util_showChat( XW_UtilCtxt* uc, */
|
|
/* const XP_UCHAR* const XP_UNUSED_DBG(msg), */
|
|
/* XP_S16 XP_UNUSED_DBG(from), XP_U32 XP_UNUSED(timestamp) ) */
|
|
/* { */
|
|
/* CursesAppGlobals* globals = (CursesAppGlobals*)uc->closure; */
|
|
/* globals->nChatsSent = 0; */
|
|
/* # ifdef DEBUG */
|
|
/* const XP_UCHAR* name = "<unknown>"; */
|
|
/* if ( 0 <= from ) { */
|
|
/* CommonGlobals* cGlobals = &globals->cGlobals; */
|
|
/* name = cGlobals->gi->players[from].name; */
|
|
/* } */
|
|
/* XP_LOGF( "%s: got \"%s\" from %s", __func__, msg, name ); */
|
|
/* # endif */
|
|
/* } */
|
|
/* #endif */
|
|
|
|
/* static void */
|
|
/* setupCursesUtilCallbacks( CursesAppGlobals* globals, XW_UtilCtxt* util ) */
|
|
/* { */
|
|
/* util->vtable->m_util_userError = curses_util_userError; */
|
|
|
|
/* util->vtable->m_util_informNeedPassword = curses_util_informNeedPassword; */
|
|
/* util->vtable->m_util_yOffsetChange = curses_util_yOffsetChange; */
|
|
/* #ifdef XWFEATURE_TURNCHANGENOTIFY */
|
|
/* util->vtable->m_util_turnChanged = curses_util_turnChanged; */
|
|
/* #endif */
|
|
/* util->vtable->m_util_notifyDupStatus = curses_util_notifyDupStatus; */
|
|
/* util->vtable->m_util_notifyIllegalWords = curses_util_notifyIllegalWords; */
|
|
/* util->vtable->m_util_remSelected = curses_util_remSelected; */
|
|
/* #ifndef XWFEATURE_STANDALONE_ONLY */
|
|
/* util->vtable->m_util_makeStreamFromAddr = curses_util_makeStreamFromAddr; */
|
|
/* #endif */
|
|
/* #ifdef XWFEATURE_CHAT */
|
|
/* util->vtable->m_util_showChat = curses_util_showChat; */
|
|
/* #endif */
|
|
|
|
/* util->vtable->m_util_notifyMove = curses_util_notifyMove; */
|
|
/* util->vtable->m_util_notifyTrade = curses_util_notifyTrade; */
|
|
/* util->vtable->m_util_notifyPickTileBlank = curses_util_notifyPickTileBlank; */
|
|
/* util->vtable->m_util_informNeedPickTiles = curses_util_informNeedPickTiles; */
|
|
/* util->vtable->m_util_trayHiddenChange = curses_util_trayHiddenChange; */
|
|
/* util->vtable->m_util_informMove = curses_util_informMove; */
|
|
/* util->vtable->m_util_informUndo = curses_util_informUndo; */
|
|
/* util->vtable->m_util_notifyGameOver = curses_util_notifyGameOver; */
|
|
/* util->vtable->m_util_informNetDict = curses_util_informNetDict; */
|
|
/* util->vtable->m_util_setIsServer = curses_util_setIsServer; */
|
|
|
|
/* #ifdef XWFEATURE_HILITECELL */
|
|
/* util->vtable->m_util_hiliteCell = curses_util_hiliteCell; */
|
|
/* #endif */
|
|
/* util->vtable->m_util_engineProgressCallback = */
|
|
/* curses_util_engineProgressCallback; */
|
|
|
|
/* util->vtable->m_util_setTimer = curses_util_setTimer; */
|
|
/* util->vtable->m_util_clearTimer = curses_util_clearTimer; */
|
|
/* util->vtable->m_util_requestTime = curses_util_requestTime; */
|
|
|
|
/* util->closure = globals; */
|
|
/* } /\* setupCursesUtilCallbacks *\/ */
|
|
|
|
/* static CursesMenuHandler */
|
|
/* getHandlerForKey( const MenuList* list, char ch ) */
|
|
/* { */
|
|
/* CursesMenuHandler 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 ) */
|
|
/* { */
|
|
/* CursesMenuHandler handler = getHandlerForKey( list, ch ); */
|
|
/* XP_Bool result = XP_FALSE; */
|
|
/* if ( !!handler ) { */
|
|
/* result = (*handler)(globals); */
|
|
/* } */
|
|
/* return result; */
|
|
/* } /\* handleKeyEvent *\/ */
|
|
|
|
/* static XP_Bool */
|
|
/* passKeyToBoard( CursesAppGlobals* globals, char ch ) */
|
|
/* { */
|
|
/* 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 *\/ */
|
|
|
|
/* static void */
|
|
/* positionSizeStuff( CursesAppGlobals* globals, int width, int height ) */
|
|
/* { */
|
|
/* CommonGlobals* cGlobals = &globals->cGlobals; */
|
|
/* BoardCtxt* board = cGlobals->game.board; */
|
|
/* #ifdef COMMON_LAYOUT */
|
|
|
|
/* BoardDims dims; */
|
|
/* board_figureLayout( board, cGlobals->gi, */
|
|
/* 0, 0, width, height, 100, */
|
|
/* 150, 200, /\* percents *\/ */
|
|
/* width*75/100, 2, 1, */
|
|
/* XP_FALSE, &dims ); */
|
|
/* board_applyLayout( board, &dims ); */
|
|
/* ======= */
|
|
/* >>>>>>> android_branch */
|
|
|
|
/* 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
|
|
inviteReceivedCurses( CursesAppGlobals* aGlobals, const NetLaunchInfo* invite,
|
|
const CommsAddrRec* returnAddr )
|
|
{
|
|
sqlite3_int64 rowids[1];
|
|
int nRowIDs = VSIZE(rowids);
|
|
getRowsForGameID( aGlobals->cag.params->pDb, invite->gameID, rowids, &nRowIDs );
|
|
bool doIt = 0 == nRowIDs;
|
|
if ( ! doIt && !!aGlobals->mainWin ) {
|
|
const gchar* question = "Duplicate invitation received. Accept anyway?";
|
|
const char* buttons[] = { "Yes", "No" };
|
|
doIt = 0 == cursesask( aGlobals->mainWin, question, VSIZE(buttons), buttons );
|
|
}
|
|
if ( doIt ) {
|
|
cb_dims dims;
|
|
figureDims( aGlobals, &dims );
|
|
cb_newFor( aGlobals->cbState, invite, returnAddr, &dims );
|
|
} else {
|
|
XP_LOGFF( "Not accepting duplicate invitation (nRowIDs(gameID=%d) was %d",
|
|
invite->gameID, nRowIDs );
|
|
}
|
|
}
|
|
|
|
static void
|
|
relayInviteReceivedCurses( void* closure, NetLaunchInfo* invite )
|
|
{
|
|
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
|
|
CommsAddrRec addr = {0};
|
|
nli_makeAddrRec( invite, &addr );
|
|
inviteReceivedCurses( aGlobals, invite, &addr );
|
|
}
|
|
|
|
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 */
|
|
cb_feedRow( aGlobals->cbState, rowid, gotSeed, buf, len, addr );
|
|
|
|
/* 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
|
|
smsInviteReceivedCurses( void* closure, const NetLaunchInfo* nli,
|
|
const CommsAddrRec* returnAddr )
|
|
{
|
|
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
|
|
inviteReceivedCurses( aGlobals, nli, returnAddr );
|
|
}
|
|
|
|
static void
|
|
smsMsgReceivedCurses( void* closure, const CommsAddrRec* from, XP_U32 gameID,
|
|
const XP_U8* buf, XP_U16 len )
|
|
{
|
|
CursesAppGlobals* aGlobals = (CursesAppGlobals*)closure;
|
|
cb_feedGame( aGlobals->cbState, gameID, buf, len, from );
|
|
}
|
|
|
|
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;
|
|
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 );
|
|
XP_LOGFF( "storing new devid: %s", devID );
|
|
cgl_draw( aGlobals->gameList );
|
|
}
|
|
(void)g_timeout_add_seconds( maxInterval, keepalive_timer, aGlobals );
|
|
} else {
|
|
XP_LOGFF( "bad relayid" );
|
|
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 );
|
|
|
|
g_globals.gameList = cgl_init( params, 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 = ¶ms->pgi; */
|
|
/* setupUtil( &g_globals.cGlobals ); */
|
|
/* setupCursesUtilCallbacks( &g_globals, g_globals.cGlobals.util ); */
|
|
|
|
// initFromParams( &g_globals.cGlobals, params );
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
/* if ( addr_hasType( ¶ms->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 );
|
|
|
|
pipe( g_globals.winchpipe );
|
|
ADD_SOCKET( &g_globals, g_globals.winchpipe[0], handle_winchwrite );
|
|
|
|
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 = {
|
|
.inviteReceived = relayInviteReceivedCurses,
|
|
.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 );
|
|
}
|
|
|
|
#ifdef XWFEATURE_SMS
|
|
gchar* myPhone = NULL;
|
|
XP_U16 myPort = 0;
|
|
if ( parseSMSParams( params, &myPhone, &myPort ) ) {
|
|
SMSProcs smsProcs = {
|
|
.inviteReceived = smsInviteReceivedCurses,
|
|
.msgReceived = smsMsgReceivedCurses,
|
|
};
|
|
linux_sms_init( params, myPhone, myPort, &smsProcs, &g_globals );
|
|
}
|
|
#endif
|
|
|
|
if ( 0 == cgl_getNGames( g_globals.gameList ) ) {
|
|
if ( params->forceNewGame ) {
|
|
handleNewGame( &g_globals, 0 );
|
|
}
|
|
} else {
|
|
/* Always open a game. Without that it won't attempt to connect and
|
|
stalls are likely in the test script case at least. If that's
|
|
annoying when running manually add a launch flag */
|
|
handleOpenGame( &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 */
|